Stroika Library 3.0d22
 
Loading...
Searching...
No Matches
WebService/Sources/WebServer.cpp
1/*
2 * Copyright(c) Sophist Solutions, Inc. 1990-2026. All rights reserved
3 */
4#include "Stroika/Frameworks/StroikaPreComp.h"
5
6#include "Stroika/Foundation/Characters/FloatConversion.h"
7#include "Stroika/Foundation/Characters/String2Int.h"
9#include "Stroika/Foundation/Common/Property.h"
11#include "Stroika/Foundation/IO/Network/HTTP/Exception.h"
12#include "Stroika/Foundation/IO/Network/HTTP/Headers.h"
13#include "Stroika/Foundation/IO/Network/HTTP/Methods.h"
15
16#include "Stroika/Frameworks/WebServer/ConnectionManager.h"
17#include "Stroika/Frameworks/WebServer/Router.h"
18#include "Stroika/Frameworks/WebService/Server/Basic.h"
19#include "Stroika/Frameworks/WebService/Server/ObjectRequestHandler.h"
20#include "Stroika/Frameworks/WebService/Server/VariantValue.h"
21
22#include "WebServer.h"
23
24#include "AppVersion.h"
25
26using namespace std;
27
28using namespace Stroika::Foundation;
31using namespace Stroika::Foundation::Execution;
33using namespace Stroika::Foundation::Traversal;
34
35using namespace Stroika::Frameworks::WebServer;
36using namespace Stroika::Frameworks::WebService;
37using namespace Stroika::Frameworks::WebService::Server;
38using namespace Stroika::Frameworks::WebService::Server::VariantValue;
39
40using Memory::BLOB;
41using Memory::MakeSharedPtr;
43using Stroika::Frameworks::WebServer::Response;
45
46using namespace StroikaSample::WebServices;
47
48namespace {
49 const Headers kDefaultResponseHeaders_{[] () {
50 Headers h;
51 h.server = "Stroika-Sample-WebServices/"_k + AppVersion::kVersion.AsMajorMinorString ();
52 return h;
53 }()};
54}
55
56namespace {
57 const ObjectRequestHandler::Options kBinaryOpObjRequestOptions_ = {.fObjectMapper = Model::kMapper,
58 .fExtractVariantValueFromRequest = ExtractArgumentsAsVariantValue::FromRequest,
59 .fTreatBodyAsListOfArguments = Iterable<String>{"arg1"sv, "arg2"sv}};
60}
61
62/*
63 * It's often helpful to structure together, routes, special interceptors, with your connection manager, to package up
64 * all the logic /options for HTTP interface.
65 *
66 * This particular organization also makes it easy to save instance variables with the webserver (like a pointer to a handler)
67 * and access them from the Route handler functions.
68 */
69class WebServer::Rep_ {
70public:
71 const Sequence<Route> kRoutes_; // rules saying how to map urls to code
72 shared_ptr<IWSAPI> fWSImpl_; // application logic actually handling webservices
73 ConnectionManager fConnectionMgr_; // manage http connection objects, thread pool, etc
74
75 static const WebServiceMethodDescription kVariables_;
76
77 static const WebServiceMethodDescription kPlus_;
78 static const WebServiceMethodDescription kMinus;
79 static const WebServiceMethodDescription kTimes;
80 static const WebServiceMethodDescription kDivide;
81
82 Rep_ (uint16_t portNumber, const shared_ptr<IWSAPI>& wsImpl)
83 : kRoutes_{Route{""_RegEx, DefaultPage_},
84
85 Route{HTTP::MethodsRegEx::kPost, "SetAppState"_RegEx, SetAppState_},
86
87 Route{"FRED"_RegEx,
88 [] (Request&, Response& response) {
89 response.write ("FRED");
90 response.contentType = InternetMediaTypes::kText_PLAIN;
91 }},
92
93 /*
94 * the 'variable' API demonstrates a typical REST style CRUD usage - where the 'arguments' mainly come from
95 * the URL itself.
96 */
97 Route{"variables(/?)"_RegEx,
98 [this] (Message& m) {
99 WriteResponse (m.rwResponse (), kVariables_, kMapper.FromObject (fWSImpl_->Variables_GET ()));
100 }},
101 Route{"variables/(.+)"_RegEx,
102 [this] (Message& m, const String& varName) {
103 WriteResponse (m.rwResponse (), kVariables_, kMapper.FromObject (fWSImpl_->Variables_GET (varName)));
104 }},
105 Route{HTTP::MethodsRegEx::kPostOrPut, "variables/(.+)"_RegEx,
106 [this] (Message& m, const String& varName) {
107 optional<Number> number;
108 // demo getting argument from the body
109 if (not number) {
110 // read if content-type is text (not json)
111 if (m.request ().contentType () and
112 InternetMediaTypeRegistry::sThe->IsA (InternetMediaTypes::kText_PLAIN, *m.request ().contentType ())) {
113 String argsAsString = Streams::BinaryToText::Reader::New (m.rwRequest ().GetBody ()).ReadAll ();
114 number = kMapper.ToObject<Number> (DataExchange::VariantValue{argsAsString});
115 }
116 }
117 // demo getting argument from the query argument
118 if (not number) {
119 static const String kValueParamName_ = "value"sv;
120 Mapping<String, DataExchange::VariantValue> args = PickoutParamValuesFromURL (m.request ());
121 number = Model::kMapper.ToObject<Number> (args.LookupValue (kValueParamName_));
122 }
123 // demo getting either query arg, or url encoded arg
124 if (not number) {
125 static const String kValueParamName_ = "value"sv;
126 // NOTE - PickoutParamValues combines PickoutParamValuesFromURL, and PickoutParamValuesFromBody. You can use
127 // Either one of those instead. PickoutParamValuesFromURL assumes you know the name of the parameter, and its
128 // encoded in the query string. PickoutParamValuesFromBody assumes you have something equivalent you can parse ouf
129 // of the body, either json encoded or form-encoded (as of 2.1d23, only json encoded supported)
130 Mapping<String, DataExchange::VariantValue> args = PickoutParamValues (m.rwRequest ());
131 number = Model::kMapper.ToObject<Number> (args.LookupValue (kValueParamName_));
132 }
133 if (not number) {
134 Throw (HTTP::ClientErrorException{"Expected argument to PUT/POST variable"sv});
135 }
136 fWSImpl_->Variables_SET (varName, *number);
137 WriteResponse (m.rwResponse (), kVariables_);
138 }},
139 Route{HTTP::MethodsRegEx::kDelete, "variables/(.+)"_RegEx,
140 /*
141 * varName - the first string argument to the request handler - comes from the first (and only) capture in the regexp.
142 */
143 [this] (Message& m, const String& varName) {
144 fWSImpl_->Variables_DELETE (varName);
145 WriteResponse (m.rwResponse (), kVariables_);
146 }},
147
148 /*
149 * plus, minus, times, and divide, test-void-return all all demonstrate passing in variables through either the POST body, or query-arguments.
150 */
151 Route{HTTP::MethodsRegEx::kPost, "plus"_RegEx,
152 ObjectRequestHandler::Factory{kBinaryOpObjRequestOptions_,
153 [this] (Number arg1, Number arg2) { return fWSImpl_->plus (arg1, arg2); }}},
154 Route{HTTP::MethodsRegEx::kPost, "minus"_RegEx,
155 ObjectRequestHandler::Factory{kBinaryOpObjRequestOptions_,
156 [this] (Number arg1, Number arg2) { return fWSImpl_->minus (arg1, arg2); }}},
157 Route{HTTP::MethodsRegEx::kPost, "times"_RegEx,
158 ObjectRequestHandler::Factory{kBinaryOpObjRequestOptions_,
159 [this] (Number arg1, Number arg2) { return fWSImpl_->times (arg1, arg2); }}},
160 Route{HTTP::MethodsRegEx::kPost, "divide"_RegEx,
161 ObjectRequestHandler::Factory{kBinaryOpObjRequestOptions_,
162 [this] (Number arg1, Number arg2) { return fWSImpl_->divide (arg1, arg2); }}},
163 Route{HTTP::MethodsRegEx::kPost, "test-void-return"_RegEx,
166 .fTreatBodyAsListOfArguments = Iterable<String>{"err-if-more-than-10"sv}},
167 [] (double check) {
168 if (check > 10) {
169 Throw (Exception{"more than 10"sv});
170 }
171 }}}}
172 , fWSImpl_{wsImpl}
173 , fConnectionMgr_{SocketAddresses (InternetAddresses_Any (), portNumber), kRoutes_,
174 ConnectionManager::Options{.fDefaultResponseHeaders = kDefaultResponseHeaders_}}
175 {
176 }
177 // Can declare arguments as Request&,Response&
178 static void DefaultPage_ (Request&, Response& response)
179 {
180 WriteDocsPage (response,
182 kVariables_,
183 kPlus_,
184 kMinus,
185 kTimes,
186 kDivide,
187 },
188 DocsOptions{"Stroika Sample WebService - Web Methods"_k, "Note - curl lines all in bash quoting syntax"_k});
189 }
190 static void SetAppState_ (Message& message)
191 {
192 String argsAsString = Streams::BinaryToText::Reader::New (message.rwRequest ().GetBody ()).ReadAll ();
193 message.rwResponse ().writeln ("<html><body><p>Hi SetAppState ("sv + argsAsString + ")</p></body></html>"sv);
194 message.rwResponse ().contentType = InternetMediaTypes::kHTML;
195 }
196};
197
198/*
199 * Documentation on WSAPIs
200 */
201const WebServiceMethodDescription WebServer::Rep_::kVariables_{
202 "variables"_k,
203 Set<String>{HTTP::Methods::kGet, HTTP::Methods::kPost, HTTP::Methods::kDelete},
204 InternetMediaTypes::kJSON,
205 {},
206 Sequence<String>{"curl http://localhost:8080/variables -v --output -", "curl http://localhost:8080/variables/x -v --output -",
207 "curl -X POST http://localhost:8080/variables/x -v --output -",
208 "curl -H \"Content-Type: application/json\" -X POST -d '{\"value\": 3}' http://localhost:8080/variables/x --output -",
209 "curl -H \"Content-Type: text/plain\" -X POST -d 3 http://localhost:8080/variables/x --output -"},
210 Sequence<String>{"@todo - this is a rough draft (but functional). It could use alot of cleanup and review to see WHICH way I recommend "
211 "using, and just provide the recommended ways in samples"},
212};
213
214const WebServiceMethodDescription WebServer::Rep_::kPlus_{
215 "plus"_k,
216 Set<String>{HTTP::Methods::kPost},
217 InternetMediaTypes::kJSON,
218 {},
220 "curl -H \"Content-Type: application/json\" -X POST -d '{\"arg1\": 3, \"arg2\": 5 }' http://localhost:8080/plus --output -",
221 "curl -X POST 'http://localhost:8080/plus?arg1=3&arg2=5' --output -",
222 },
223 Sequence<String>{"add the two argument numbers"},
224};
225const WebServiceMethodDescription WebServer::Rep_::kMinus{
226 "minus"_k,
227 Set<String>{HTTP::Methods::kPost},
228 InternetMediaTypes::kJSON,
229 {},
231 "curl -H \"Content-Type: application/json\" -X POST -d '{\"arg1\": 4.5, \"arg2\": -3.23 }' http://localhost:8080/minus --output -",
232 },
233 Sequence<String>{"subtract the two argument numbers"},
234};
235const WebServiceMethodDescription WebServer::Rep_::kTimes{
236 "times"_k,
237 Set<String>{HTTP::Methods::kPost},
238 InternetMediaTypes::kJSON,
239 {},
241 "curl -H \"Content-Type: application/json\" -X POST -d '{\"arg1\":\"2 + 4i\", \"arg2\": 3.2 }' http://localhost:8080/times "
242 "--output -",
243 "curl -H \"Content-Type: application/json\" -X POST -d '{\"arg1\":\"2 + i\", \"arg2\": \"2 - i\" }' http://localhost:8080/times "
244 "--output -",
245 },
246 Sequence<String>{"multiply the two argument numbers"},
247};
248const WebServiceMethodDescription WebServer::Rep_::kDivide{
249 "divide"_k,
250 Set<String>{HTTP::Methods::kPost},
251 InternetMediaTypes::kJSON,
252 {},
254 "curl -H \"Content-Type: application/json\" -X POST -d '{\"arg1\":\"2 + i\", \"arg2\": 0 }' http://localhost:8080/divide --output "
255 "-",
256 },
257 Sequence<String>{"divide the two argument numbers"},
258};
259
260WebServer::WebServer (uint16_t portNumber, const shared_ptr<IWSAPI>& wsImpl)
261 : fRep_{MakeSharedPtr<Rep_> (portNumber, wsImpl)}
262{
263}
auto MakeSharedPtr(ARGS_TYPE &&... args) -> shared_ptr< T >
same as make_shared, but if type T has block allocation, then use block allocation for the 'shared pa...
String is like std::u32string, except it is much easier to use, often much more space efficient,...
Definition String.h:201
nonvirtual mapped_type LookupValue(ArgByValueType< key_type > key, ArgByValueType< mapped_type > defaultValue=mapped_type{}) const
Definition Mapping.inl:166
A generalization of a vector: a container whose elements are keyed by the natural numbers.
Set<T> is a container of T, where once an item is added, additionally adds () do nothing.
static Execution::Synchronized< InternetMediaTypeRegistry > sThe
nonvirtual VariantValue FromObject(const T &from) const
nonvirtual T ToObject(const VariantValue &v) const
Simple variant-value (case variant union) object, with (variant) basic types analogous to a value in ...
Exception<> is a replacement (subclass) for any std c++ exception class (e.g. the default 'std::excep...
Definition Exceptions.h:157
ClientErrorException is to capture exceptions caused by a bad (e.g ill-formed) request.
roughly equivalent to Association<String,String>, except that the class is smart about certain keys a...
Definition Headers.h:129
Common::Property< optional< String > > server
Definition Headers.h:404
nonvirtual String ReadAll(size_t upTo=numeric_limits< size_t >::max()) const
Iterable<T> is a base class for containers which easily produce an Iterator<T> to traverse them.
Definition Iterable.h:237
Common::ReadOnlyProperty< Response & > rwResponse
Definition Message.h:93
Common::ReadOnlyProperty< Request & > rwRequest
Definition Message.h:80
this represents a HTTP request object for the WebServer module
ObjectRequestHandler::Factory is a way to construct a WebServer::RequestHandler from an ObjectVariant...
void Throw(T &&e2Throw)
identical to builtin C++ 'throw' except that it does helpful, type dependent DbgTrace() messages firs...
Definition Throw.inl:43
Traversal::Iterable< SocketAddress > SocketAddresses(const Traversal::Iterable< InternetAddress > &internetAddresses, PortType portNumber)
Traversal::Iterable< InternetAddress > InternetAddresses_Any(InternetProtocol::IP::IPVersionSupport ipSupport=InternetProtocol::IP::IPVersionSupport::eDEFAULT)
Ptr New(const InputStream::Ptr< byte > &src, optional< AutomaticCodeCvtFlags > codeCvtFlags={}, optional< SeekableFlag > seekable={}, ReadAhead readAhead=eReadAheadAllowed)
Create an InputStream::Ptr<Character> from the arguments (usually binary source) - which can be used ...
STL namespace.
ConnectionManager::Options specify things like default headers, caching policies, binding flags (not ...
Options for ObjectRequestHandler - mostly the ObjectVariantMapper, but also a few others depending on...