VirtualBox

source: vbox/trunk/include/iprt/cpp/restclient.h@ 83146

Last change on this file since 83146 was 82968, checked in by vboxsync, 5 years ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.0 KB
Line 
1/** @file
2 * IPRT - C++ Representational State Transfer (REST) Client Classes.
3 */
4
5/*
6 * Copyright (C) 2008-2020 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef IPRT_INCLUDED_cpp_restclient_h
27#define IPRT_INCLUDED_cpp_restclient_h
28#ifndef RT_WITHOUT_PRAGMA_ONCE
29# pragma once
30#endif
31
32#include <iprt/http.h>
33#include <iprt/cpp/restbase.h>
34#include <iprt/cpp/reststringmap.h>
35
36
37/** @defgroup grp_rt_cpp_restclient C++ Representational State Transfer (REST) Client Classes.
38 * @ingroup grp_rt_cpp
39 * @{
40 */
41
42/**
43 * Specialization of RTCRestBinary for use with body parameters in a client.
44 *
45 * This enables registering data callbacks for provinding data to upload.
46 */
47class RT_DECL_CLASS RTCRestBinaryParameter : public RTCRestBinary
48{
49public:
50 /** Default constructor. */
51 RTCRestBinaryParameter() RT_NOEXCEPT;
52
53 /** Safe copy assignment method. */
54 virtual int assignCopy(RTCRestBinaryParameter const &a_rThat) RT_NOEXCEPT;
55 /** Safe copy assignment method.
56 * @note Resets callbacks and ASSUMES that @a a_cbData is the content length. */
57 virtual int assignCopy(RTCRestBinary const &a_rThat) RT_NOEXCEPT RT_OVERRIDE;
58 /** Safe copy assignment method.
59 * @note Resets callbacks and ASSUMES that @a a_cbData is the content length. */
60 virtual int assignCopy(void const *a_pvData, size_t a_cbData) RT_NOEXCEPT RT_OVERRIDE;
61
62 /**
63 * Use the specified data buffer directly.
64 * @note Resets callbacks and ASSUMES that @a a_cbData is the content length. */
65 virtual int assignReadOnly(void const *a_pvData, size_t a_cbData) RT_NOEXCEPT RT_OVERRIDE;
66 /**
67 * Use the specified data buffer directly.
68 * @note This will assert and work like assignReadOnly. */
69 virtual int assignWriteable(void *a_pvBuf, size_t a_cbBuf) RT_NOEXCEPT RT_OVERRIDE;
70
71 /** Make a clone of this object. */
72 inline RTCRestBinaryParameter *clone() const RT_NOEXCEPT { return (RTCRestBinaryParameter *)baseClone(); }
73
74 /* Overridden methods: */
75 virtual RTCRestObjectBase *baseClone() const RT_NOEXCEPT RT_OVERRIDE;
76 virtual int resetToDefault() RT_NOEXCEPT RT_OVERRIDE;
77 virtual const char *typeName(void) const RT_NOEXCEPT RT_OVERRIDE;
78
79 /** Factory method. */
80 static DECLCALLBACK(RTCRestObjectBase *) createInstance(void) RT_NOEXCEPT;
81
82 /**
83 * Retrieves the callback data.
84 */
85 inline void *getCallbackData() const RT_NOEXCEPT { return m_pvCallbackData; }
86
87 /**
88 * Sets the content-type for an upload.
89 *
90 * @returns VINF_SUCCESS or VERR_NO_STR_MEMORY.
91 * @param a_pszContentType The content type to set.
92 * If NULL, no content type is set.
93 */
94 int setContentType(const char *a_pszContentType) RT_NOEXCEPT;
95
96 /**
97 * Gets the content type that was set.
98 */
99 inline RTCString const &getContentType() const RT_NOEXCEPT { return m_strContentType; }
100
101 /**
102 * Gets the content-length value (UINT64_MAX if not available).
103 */
104 inline uint64_t getContentLength() const RT_NOEXCEPT { return m_cbContentLength; }
105
106 /**
107 * Callback for producing bytes to upload.
108 *
109 * @returns IPRT status code.
110 * @param a_pThis The related string object.
111 * @param a_pvDst Where to put the bytes.
112 * @param a_cbDst Max number of bytes to produce.
113 * @param a_offContent The byte offset corresponding to the start of @a a_pvDst.
114 * @param a_pcbActual Where to return the number of bytes actually produced.
115 *
116 * @remarks Use getCallbackData to get the user data.
117 *
118 * @note The @a a_offContent parameter does not imply random access or anthing
119 * like that, it is just a convenience provided by the caller. The value
120 * is the sum of the previously returned @a *pcbActual values.
121 */
122 typedef DECLCALLBACK(int) FNPRODUCER(RTCRestBinaryParameter *a_pThis, void *a_pvDst, size_t a_cbDst,
123 uint64_t a_offContent, size_t *a_pcbActual) /*RT_NOEXCEPT*/;
124 /** Pointer to a byte producer callback. */
125 typedef FNPRODUCER *PFNPRODUCER;
126
127 /**
128 * Sets the producer callback.
129 *
130 * @param a_pfnProducer The callback function for producing data.
131 * @param a_pvCallbackData Data the can be retrieved from the callback
132 * using getCallbackData().
133 * @param a_cbContentLength The amount of data that will be uploaded and
134 * to be set as the value of the content-length
135 * header field. Pass UINT64_MAX if not known.
136 *
137 * @note This will drop any buffer previously registered using setUploadData().
138 */
139 void setProducerCallback(PFNPRODUCER a_pfnProducer, void *a_pvCallbackData = NULL, uint64_t a_cbContentLength = UINT64_MAX) RT_NOEXCEPT;
140
141 /**
142 * Preprares transmission via the @a a_hHttp client instance.
143 *
144 * @returns IPRT status code.
145 * @param a_hHttp The HTTP client instance.
146 * @internal
147 */
148 virtual int xmitPrepare(RTHTTP a_hHttp) const RT_NOEXCEPT;
149
150 /**
151 * For completing and/or undoing setup from xmitPrepare.
152 *
153 * @param a_hHttp The HTTP client instance.
154 * @internal
155 */
156 virtual void xmitComplete(RTHTTP a_hHttp) const RT_NOEXCEPT;
157
158protected:
159 /** Number of bytes corresponding to content-length.
160 * UINT64_MAX if not known. Used both for unploads and downloads. */
161 uint64_t m_cbContentLength;
162 /** The content type if set (upload only). */
163 RTCString m_strContentType;
164 /** Pointer to user-registered producer callback function (upload only). */
165 PFNPRODUCER m_pfnProducer;
166 /** User argument for both callbacks (both). */
167 void *m_pvCallbackData;
168
169 /** @copydoc FNRTHTTPUPLOADCALLBACK */
170 static DECLCALLBACK(int) xmitHttpCallback(RTHTTP hHttp, void *pvBuf, size_t cbBuf, uint64_t offContent,
171 size_t *pcbActual, void *pvUser) RT_NOEXCEPT;
172
173private:
174 /* No copy constructor or copy assignment: */
175 RTCRestBinaryParameter(RTCRestBinaryParameter const &a_rThat);
176 RTCRestBinaryParameter &operator=(RTCRestBinaryParameter const &a_rThat);
177};
178
179
180/**
181 * Specialization of RTCRestBinary for use with responses in a client.
182 *
183 * This enables registering data callbacks for consuming downloaded data.
184 */
185class RT_DECL_CLASS RTCRestBinaryResponse : public RTCRestBinary
186{
187public:
188 /** Default constructor. */
189 RTCRestBinaryResponse() RT_NOEXCEPT;
190
191 /** Safe copy assignment method. */
192 virtual int assignCopy(RTCRestBinaryResponse const &a_rThat) RT_NOEXCEPT;
193 /** Safe copy assignment method. */
194 virtual int assignCopy(RTCRestBinary const &a_rThat) RT_NOEXCEPT RT_OVERRIDE;
195 /** Safe copy assignment method.
196 * @note This will assert and fail as it makes no sense for a download. */
197 virtual int assignCopy(void const *a_pvData, size_t a_cbData) RT_NOEXCEPT RT_OVERRIDE;
198
199 /**
200 * Use the specified data buffer directly.
201 * @note This will assert and fail as it makes no sense for a download.
202 */
203 virtual int assignReadOnly(void const *a_pvData, size_t a_cbData) RT_NOEXCEPT RT_OVERRIDE;
204 /**
205 * Use the specified data buffer directly.
206 * @note This will drop any previously registered producer callback and user data.
207 */
208 virtual int assignWriteable(void *a_pvBuf, size_t a_cbBuf) RT_NOEXCEPT RT_OVERRIDE;
209
210 /** Make a clone of this object. */
211 inline RTCRestBinaryResponse *clone() const RT_NOEXCEPT { return (RTCRestBinaryResponse *)baseClone(); }
212
213 /* Overridden methods: */
214 virtual RTCRestObjectBase *baseClone() const RT_NOEXCEPT RT_OVERRIDE;
215 virtual int resetToDefault() RT_NOEXCEPT RT_OVERRIDE;
216 virtual const char *typeName(void) const RT_NOEXCEPT RT_OVERRIDE;
217
218 /** Factory method. */
219 static DECLCALLBACK(RTCRestObjectBase *) createInstance(void) RT_NOEXCEPT;
220
221 /**
222 * Retrieves the callback data.
223 */
224 inline void *getCallbackData() const RT_NOEXCEPT { return m_pvCallbackData; }
225
226 /**
227 * Sets the max size to download to memory.
228 *
229 * This also indicates the intention to download to a memory buffer, so it
230 * will drop any previously registered consumer callback and its user data.
231 *
232 * @param a_cbMaxDownload Maximum number of bytes to download to memory.
233 * If 0, a default is selected (currently 32MiB for
234 * 32-bit hosts and 128MiB for 64-bit).
235 */
236 void setMaxDownloadSize(size_t a_cbMaxDownload) RT_NOEXCEPT;
237
238 /**
239 * Gets the content-length value (UINT64_MAX if not available).
240 */
241 inline uint64_t getContentLength() const RT_NOEXCEPT { return m_cbContentLength; }
242
243 /**
244 * Callback for consuming downloaded bytes.
245 *
246 * @returns IPRT status code.
247 * @param a_pThis The related string object.
248 * @param a_pvSrc Buffer containing the bytes.
249 * @param a_cbSrc The number of bytes in the buffer.
250 * @param a_uHttpStatus The HTTP status code.
251 * @param a_offContent The byte offset corresponding to the start of @a a_pvSrc.
252 * @param a_cbContent The content length field value, UINT64_MAX if not available.
253 *
254 * @remarks Use getCallbackData to get the user data.
255 *
256 * @note The @a a_offContent parameter does not imply random access or anthing
257 * like that, it is just a convenience provided by the caller. The value
258 * is the sum of the previous @a a_cbSrc values.
259 */
260 typedef DECLCALLBACK(int) FNCONSUMER(RTCRestBinaryResponse *a_pThis, const void *a_pvSrc, size_t a_cbSrc,
261 uint32_t a_uHttpStatus, uint64_t a_offContent, uint64_t a_cbContent) /*RT_NOEXCEPT*/;
262 /** Pointer to a byte consumer callback. */
263 typedef FNCONSUMER *PFNCONSUMER;
264
265 /**
266 * Sets the consumer callback.
267 *
268 * @param a_pfnConsumer The callback function for consuming downloaded data.
269 * NULL if data should be stored in m_pbData (the default).
270 * @param a_pvCallbackData Data the can be retrieved from the callback
271 * using getCallbackData().
272 */
273 void setConsumerCallback(PFNCONSUMER a_pfnConsumer, void *a_pvCallbackData = NULL) RT_NOEXCEPT;
274
275 /**
276 * Preprares for receiving via the @a a_hHttp client instance.
277 *
278 * @returns IPRT status code.
279 * @param a_hHttp The HTTP client instance.
280 * @param a_fCallbackFlags The HTTP callback flags (status code spec).
281 * @internal
282 */
283 virtual int receivePrepare(RTHTTP a_hHttp, uint32_t a_fCallbackFlags) RT_NOEXCEPT;
284
285 /**
286 * For completing and/or undoing setup from receivePrepare.
287 *
288 * @param a_hHttp The HTTP client instance.
289 * @internal
290 */
291 virtual void receiveComplete(RTHTTP a_hHttp) RT_NOEXCEPT;
292
293protected:
294 /** Number of bytes corresponding to content-length.
295 * UINT64_MAX if not known. Used both for unploads and downloads. */
296 uint64_t m_cbContentLength;
297 /** Number of bytes downloaded thus far. */
298 uint64_t m_cbDownloaded;
299 /** Pointer to user-registered consumer callback function (download only). */
300 PFNCONSUMER m_pfnConsumer;
301 /** User argument for both callbacks (both). */
302 void *m_pvCallbackData;
303 /** Maximum data to download to memory (download only). */
304 size_t m_cbMaxDownload;
305
306 /** @copydoc FNRTHTTPDOWNLOADCALLBACK */
307 static DECLCALLBACK(int) receiveHttpCallback(RTHTTP hHttp, void const *pvBuf, size_t cbBuf, uint32_t uHttpStatus,
308 uint64_t offContent, uint64_t cbContent, void *pvUser) RT_NOEXCEPT;
309
310private:
311 /* No copy constructor or copy assignment: */
312 RTCRestBinaryResponse(RTCRestBinaryResponse const &a_rThat);
313 RTCRestBinaryResponse &operator=(RTCRestBinaryResponse const &a_rThat);
314};
315
316
317/**
318 * Base class for REST client requests.
319 *
320 * This encapsulates parameters and helps transform them into a HTTP request.
321 *
322 * Parameters can be transfered in a number of places:
323 * - Path part of the URL.
324 * - Query part of the URL.
325 * - HTTP header fields.
326 * - FORM body.
327 * - JSON body.
328 * - XML body.
329 * - ...
330 *
331 * They can be require or optional. The latter may have default values. In
332 * swagger 3 they can also be nullable, which means the null-indicator cannot
333 * be used for tracking optional parameters.
334 */
335class RT_DECL_CLASS RTCRestClientRequestBase
336{
337public:
338 RTCRestClientRequestBase() RT_NOEXCEPT;
339 virtual ~RTCRestClientRequestBase();
340 RTCRestClientRequestBase(RTCRestClientRequestBase const &a_rThat) RT_NOEXCEPT;
341 RTCRestClientRequestBase &operator=(RTCRestClientRequestBase const &a_rThat) RT_NOEXCEPT;
342
343 /**
344 * Reset all members to default values.
345 * @returns IPRT status code.
346 */
347 virtual int resetToDefault() RT_NOEXCEPT = 0;
348
349 /**
350 * Prepares the HTTP handle for transmitting this request.
351 *
352 * @returns IPRT status code.
353 * @param a_pStrPath Where to set path parameters. Will be appended to the base path.
354 * @param a_pStrQuery Where to set query parameters.
355 * @param a_hHttp Where to set header parameters and such.
356 * @param a_pStrBody Where to set body parameters.
357 */
358 virtual int xmitPrepare(RTCString *a_pStrPath, RTCString *a_pStrQuery, RTHTTP a_hHttp, RTCString *a_pStrBody) const RT_NOEXCEPT = 0;
359
360 /**
361 * Always called after the request has been transmitted.
362 *
363 * @param a_rcStatus Negative numbers are IPRT errors, positive are HTTP status codes.
364 * @param a_hHttp The HTTP handle the request was performed on.
365 */
366 virtual void xmitComplete(int a_rcStatus, RTHTTP a_hHttp) const RT_NOEXCEPT = 0;
367
368 /**
369 * Checks if there are were any assignment errors.
370 */
371 inline bool hasAssignmentErrors() const RT_NOEXCEPT { return m_fErrorSet != 0; }
372
373protected:
374 /** Set of fields that have been explicitly assigned a value. */
375 uint64_t m_fIsSet;
376 /** Set of fields where value assigning failed. */
377 uint64_t m_fErrorSet;
378
379 /** Path parameter descriptor. */
380 typedef struct
381 {
382 const char *pszName; /**< The name string to replace (including {}). */
383 size_t cchName; /**< Length of pszName. */
384 uint32_t fFlags; /**< The toString flags. */
385 uint8_t iBitNo; /**< The parameter bit number. */
386 } PATHPARAMDESC;
387
388 /** Path parameter state. */
389 typedef struct
390 {
391 RTCRestObjectBase const *pObj; /**< Pointer to the parameter object. */
392 size_t offName; /**< Maintained by worker. */
393 } PATHPARAMSTATE;
394
395 /**
396 * Do path parameters.
397 *
398 * @returns IPRT status code
399 * @param a_pStrPath The destination path.
400 * @param a_pszPathTemplate The path template string.
401 * @param a_cchPathTemplate The length of the path template string.
402 * @param a_paPathParams The path parameter descriptors (static).
403 * @param a_paPathParamStates The path parameter objects and states.
404 * @param a_cPathParams Number of path parameters.
405 */
406 int doPathParameters(RTCString *a_pStrPath, const char *a_pszPathTemplate, size_t a_cchPathTemplate,
407 PATHPARAMDESC const *a_paPathParams, PATHPARAMSTATE *a_paPathParamStates, size_t a_cPathParams) const RT_NOEXCEPT;
408
409 /** Query parameter descriptor. */
410 typedef struct
411 {
412 const char *pszName; /**< The parameter name. */
413 uint32_t fFlags; /**< The toString flags. */
414 bool fRequired; /**< Required or not. */
415 uint8_t iBitNo; /**< The parameter bit number. */
416 } QUERYPARAMDESC;
417
418 /**
419 * Do query parameters.
420 *
421 * @returns IPRT status code
422 * @param a_pStrQuery The destination string.
423 * @param a_paQueryParams The query parameter descriptors.
424 * @param a_papQueryParamObjs The query parameter objects, parallel to @a a_paQueryParams.
425 * @param a_cQueryParams Number of query parameters.
426 */
427 int doQueryParameters(RTCString *a_pStrQuery, QUERYPARAMDESC const *a_paQueryParams,
428 RTCRestObjectBase const **a_papQueryParamObjs, size_t a_cQueryParams) const RT_NOEXCEPT;
429
430 /** Header parameter descriptor. */
431 typedef struct
432 {
433 const char *pszName; /**< The parameter name. */
434 uint32_t fFlags; /**< The toString flags. */
435 bool fRequired; /**< Required or not. */
436 uint8_t iBitNo; /**< The parameter bit number. */
437 bool fMapCollection; /**< Collect headers starting with pszName into a map. */
438 } HEADERPARAMDESC;
439
440 /**
441 * Do header parameters.
442 *
443 * @returns IPRT status code
444 * @param a_hHttp Where to set header parameters.
445 * @param a_paHeaderParams The header parameter descriptors.
446 * @param a_papHeaderParamObjs The header parameter objects, parallel to @a a_paHeaderParams.
447 * @param a_cHeaderParams Number of header parameters.
448 */
449 int doHeaderParameters(RTHTTP a_hHttp, HEADERPARAMDESC const *a_paHeaderParams,
450 RTCRestObjectBase const **a_papHeaderParamObjs, size_t a_cHeaderParams) const RT_NOEXCEPT;
451};
452
453
454/**
455 * Base class for REST client responses.
456 */
457class RT_DECL_CLASS RTCRestClientResponseBase
458{
459public:
460 /** Default constructor. */
461 RTCRestClientResponseBase() RT_NOEXCEPT;
462 /** Destructor. */
463 virtual ~RTCRestClientResponseBase();
464 /** Copy constructor. */
465 RTCRestClientResponseBase(RTCRestClientResponseBase const &a_rThat);
466 /** Copy assignment operator. */
467 RTCRestClientResponseBase &operator=(RTCRestClientResponseBase const &a_rThat);
468
469 /**
470 * Resets the object state.
471 */
472 virtual void reset(void) RT_NOEXCEPT;
473
474 /**
475 * Prepares the HTTP handle for receiving the response.
476 *
477 * This may install callbacks and such like.
478 *
479 * When overridden, the parent class must always be called.
480 *
481 * @returns IPRT status code.
482 * @param a_hHttp The HTTP handle to prepare for receiving.
483 */
484 virtual int receivePrepare(RTHTTP a_hHttp) RT_NOEXCEPT;
485
486 /**
487 * Called when the HTTP request has been completely received.
488 *
489 * @param a_rcStatus Negative numbers are IPRT errors, positive are HTTP status codes.
490 * @param a_hHttp The HTTP handle the request was performed on.
491 * This can be NIL_RTHTTP should something fail early, in
492 * which case it is possible receivePrepare() wasn't called.
493 *
494 * @note Called before consumeBody() but after consumeHeader().
495 */
496 virtual void receiveComplete(int a_rcStatus, RTHTTP a_hHttp) RT_NOEXCEPT;
497
498 /**
499 * Callback that consumes HTTP body data from the server.
500 *
501 * @param a_pchData Body data.
502 * @param a_cbData Amount of body data.
503 *
504 * @note Called after consumeHeader().
505 */
506 virtual void consumeBody(const char *a_pchData, size_t a_cbData) RT_NOEXCEPT;
507
508 /**
509 * Called after status, headers and body all have been presented.
510 *
511 * @returns IPRT status code.
512 */
513 virtual void receiveFinal() RT_NOEXCEPT;
514
515 /**
516 * Getter for m_rcStatus.
517 * @returns Negative numbers are IPRT errors, positive are HTTP status codes.
518 */
519 inline int getStatus() const RT_NOEXCEPT { return m_rcStatus; }
520
521 /**
522 * Getter for m_rcHttp.
523 * @returns HTTP status code or VERR_NOT_AVAILABLE.
524 */
525 inline int getHttpStatus() const RT_NOEXCEPT { return m_rcHttp; }
526
527 /**
528 * Getter for m_pErrInfo.
529 */
530 inline PCRTERRINFO getErrInfo(void) const RT_NOEXCEPT { return m_pErrInfo; }
531
532 /**
533 * Getter for m_strContentType.
534 */
535 inline RTCString const &getContentType(void) const RT_NOEXCEPT { return m_strContentType; }
536
537
538protected:
539 /** Negative numbers are IPRT errors, positive are HTTP status codes. */
540 int m_rcStatus;
541 /** The HTTP status code, VERR_NOT_AVAILABLE if not set. */
542 int m_rcHttp;
543 /** Error information. */
544 PRTERRINFO m_pErrInfo;
545 /** The value of the Content-Type header field. */
546 RTCString m_strContentType;
547
548 PRTERRINFO getErrInfoInternal(void) RT_NOEXCEPT;
549 void deleteErrInfo(void) RT_NOEXCEPT;
550 void copyErrInfo(PCRTERRINFO pErrInfo) RT_NOEXCEPT;
551
552 /**
553 * Reports an error (or warning if a_rc non-negative).
554 *
555 * @returns a_rc
556 * @param a_rc The status code to report and return. The first
557 * error status is assigned to m_rcStatus, subsequent
558 * ones as well as informational statuses are not
559 * recorded by m_rcStatus.
560 * @param a_pszFormat The message format string.
561 * @param ... Message arguments.
562 */
563 int addError(int a_rc, const char *a_pszFormat, ...) RT_NOEXCEPT;
564
565 /**
566 * Deserializes a header field value.
567 *
568 * @returns IPRT status code.
569 * @param a_pObj The object to deserialize into.
570 * @param a_pchValue Pointer to the value (not zero terminated).
571 * Not necessarily valid UTF-8!
572 * @param a_cchValue The value length.
573 * @param a_fFlags Flags to pass to fromString().
574 * @param a_pszErrorTag The error tag (field name).
575 */
576 int deserializeHeader(RTCRestObjectBase *a_pObj, const char *a_pchValue, size_t a_cchValue,
577 uint32_t a_fFlags, const char *a_pszErrorTag) RT_NOEXCEPT;
578
579 /**
580 * Deserializes a header field value.
581 *
582 * @returns IPRT status code.
583 * @param a_pMap The string map object to deserialize into.
584 * @param a_pchField Pointer to the map field name. (Caller dropped the prefix.)
585 * Not necessarily valid UTF-8!
586 * @param a_cchField Length of field name.
587 * @param a_pchValue Pointer to the value (not zero terminated).
588 * Not necessarily valid UTF-8!
589 * @param a_cchValue The value length.
590 * @param a_fFlags Flags to pass to fromString().
591 * @param a_pszErrorTag The error tag (field name).
592 */
593 int deserializeHeaderIntoMap(RTCRestStringMapBase *a_pMap, const char *a_pchField, size_t a_cchField,
594 const char *a_pchValue, size_t a_cchValue, uint32_t a_fFlags, const char *a_pszErrorTag) RT_NOEXCEPT;
595
596 /**
597 * Helper that does the deserializing of the response body
598 * via deserializeBodyFromJsonCursor().
599 *
600 * @param a_pchData The body blob.
601 * @param a_cbData The size of the body blob.
602 * @param a_pszBodyName The name of the body parameter.
603 */
604 void deserializeBody(const char *a_pchData, size_t a_cbData, const char *a_pszBodyName) RT_NOEXCEPT;
605
606 /**
607 * Called by deserializeBody to do the actual body deserialization.
608 *
609 * @param a_rCursor The JSON cursor.
610 */
611 virtual void deserializeBodyFromJsonCursor(RTCRestJsonCursor const &a_rCursor) RT_NOEXCEPT;
612
613 /**
614 * Primary json cursor for parsing bodies.
615 */
616 class PrimaryJsonCursorForBody : public RTCRestJsonPrimaryCursor
617 {
618 public:
619 RTCRestClientResponseBase *m_pThat; /**< Pointer to response object. */
620 PrimaryJsonCursorForBody(RTJSONVAL hValue, const char *pszName, RTCRestClientResponseBase *a_pThat) RT_NOEXCEPT;
621 virtual int addError(RTCRestJsonCursor const &a_rCursor, int a_rc, const char *a_pszFormat, ...) RT_NOEXCEPT RT_OVERRIDE;
622 virtual int unknownField(RTCRestJsonCursor const &a_rCursor) RT_NOEXCEPT RT_OVERRIDE;
623 };
624
625
626 /**
627 * Consumes a header.
628 *
629 * Child classes can override this to pick up their header fields, but must
630 * always call the parent class.
631 *
632 * @returns IPRT status code.
633 * @param a_uMatchWord Match word constructed by RTHTTP_MAKE_HDR_MATCH_WORD
634 * @param a_pchField The field name (not zero terminated).
635 * Not necessarily valid UTF-8!
636 * @param a_cchField The length of the field.
637 * @param a_pchValue The field value (not zero terminated).
638 * @param a_cchValue The length of the value.
639 */
640 virtual int consumeHeader(uint32_t a_uMatchWord, const char *a_pchField, size_t a_cchField,
641 const char *a_pchValue, size_t a_cchValue) RT_NOEXCEPT;
642
643private:
644 /** Callback for use with RTHttpSetHeaderCallback. */
645 static DECLCALLBACK(int) receiveHttpHeaderCallback(RTHTTP hHttp, uint32_t uMatchWord, const char *pchField, size_t cchField,
646 const char *pchValue, size_t cchValue, void *pvUser) RT_NOEXCEPT;
647};
648
649
650/**
651 * Base class for REST client responses.
652 */
653class RT_DECL_CLASS RTCRestClientApiBase
654{
655public:
656 RTCRestClientApiBase() RT_NOEXCEPT;
657 virtual ~RTCRestClientApiBase();
658
659 /** @name Host and Base path (URL) handling.
660 * @{ */
661 /**
662 * Gets the server URL.
663 */
664 const char *getServerUrl(void) const RT_NOEXCEPT;
665
666 /**
667 * Sets the whole server URL.
668 * @returns IPRT status code.
669 * @param a_pszUrl The new server URL. NULL/empty to reset to default.
670 */
671 int setServerUrl(const char *a_pszUrl) RT_NOEXCEPT;
672
673 /**
674 * Sets the scheme part of the the server URL.
675 * @returns IPRT status code.
676 * @param a_pszScheme The new scheme. Does not accept NULL or empty string.
677 */
678 int setServerScheme(const char *a_pszScheme) RT_NOEXCEPT;
679
680 /**
681 * Sets the authority (hostname + port) part of the the server URL.
682 * @returns IPRT status code.
683 * @param a_pszAuthority The new authority. Does not accept NULL or empty string.
684 */
685 int setServerAuthority(const char *a_pszAuthority) RT_NOEXCEPT;
686
687 /**
688 * Sets the base path part of the the server URL.
689 * @returns IPRT status code.
690 * @param a_pszBasePath The new base path. Does not accept NULL or empty string.
691 */
692 int setServerBasePath(const char *a_pszBasePath) RT_NOEXCEPT;
693
694 /**
695 * Gets the default server URL as specified in the specs.
696 * @returns Server URL.
697 */
698 virtual const char *getDefaultServerUrl() const RT_NOEXCEPT = 0;
699
700 /**
701 * Gets the default server base path as specified in the specs.
702 * @returns Host string (start of URL).
703 */
704 virtual const char *getDefaultServerBasePath() const RT_NOEXCEPT = 0;
705 /** @} */
706
707 /** Flags to doCall. */
708 enum
709 {
710 kDoCall_OciReqSignExcludeBody = 1, /**< Exclude the body when doing OCI request signing. */
711 kDoCall_RequireBody = 2 /**< The body is required. */
712 };
713
714protected:
715 /** Handle to the HTTP connection object. */
716 RTHTTP m_hHttp;
717 /** The server URL to use. If empty use the default. */
718 RTCString m_strServerUrl;
719
720 /* Make non-copyable (RTCNonCopyable causes warnings): */
721 RTCRestClientApiBase(RTCRestClientApiBase const &);
722 RTCRestClientApiBase *operator=(RTCRestClientApiBase const &);
723
724 /**
725 * Re-initializes the HTTP instance.
726 *
727 * @returns IPRT status code.
728 */
729 virtual int reinitHttpInstance() RT_NOEXCEPT;
730
731 /**
732 * Hook that's called when doCall has fully assembled the request.
733 *
734 * Can be used for request signing and similar final steps.
735 *
736 * @returns IPRT status code.
737 * @param a_hHttp The HTTP client instance.
738 * @param a_rStrFullUrl The full URL.
739 * @param a_enmHttpMethod The HTTP request method.
740 * @param a_rStrXmitBody The body text.
741 * @param a_fFlags kDoCall_XXX.
742 */
743 virtual int xmitReady(RTHTTP a_hHttp, RTCString const &a_rStrFullUrl, RTHTTPMETHOD a_enmHttpMethod,
744 RTCString const &a_rStrXmitBody, uint32_t a_fFlags) RT_NOEXCEPT;
745
746 /**
747 * Implements stuff for making an API call.
748 *
749 * @returns a_pResponse->getStatus()
750 * @param a_rRequest Reference to the request object.
751 * @param a_enmHttpMethod The HTTP request method.
752 * @param a_pResponse Pointer to the response object.
753 * @param a_pszMethod The method name, for logging purposes.
754 * @param a_fFlags kDoCall_XXX.
755 */
756 virtual int doCall(RTCRestClientRequestBase const &a_rRequest, RTHTTPMETHOD a_enmHttpMethod,
757 RTCRestClientResponseBase *a_pResponse, const char *a_pszMethod, uint32_t a_fFlags) RT_NOEXCEPT;
758
759 /**
760 * Implements OCI style request signing.
761 *
762 * @returns IPRT status code.
763 * @param a_hHttp The HTTP client instance.
764 * @param a_rStrFullUrl The full URL.
765 * @param a_enmHttpMethod The HTTP request method.
766 * @param a_rStrXmitBody The body text.
767 * @param a_fFlags kDoCall_XXX.
768 * @param a_hKey The key to use for signing.
769 * @param a_rStrKeyId The key ID.
770 *
771 * @remarks The signing scheme is covered by a series of drafts RFC, the latest being:
772 * https://tools.ietf.org/html/draft-cavage-http-signatures-10
773 */
774 int ociSignRequest(RTHTTP a_hHttp, RTCString const &a_rStrFullUrl, RTHTTPMETHOD a_enmHttpMethod,
775 RTCString const &a_rStrXmitBody, uint32_t a_fFlags, RTCRKEY a_hKey, RTCString const &a_rStrKeyId) RT_NOEXCEPT;
776
777 /**
778 * Worker for the server URL modifiers.
779 *
780 * @returns IPRT status code.
781 * @param a_pszServerUrl The current server URL (for comparing).
782 * @param a_offDst The offset of the component in the current server URL.
783 * @param a_cchDst The current component length.
784 * @param a_pszSrc The new URL component value.
785 * @param a_cchSrc The length of the new component.
786 */
787 int setServerUrlPart(const char *a_pszServerUrl, size_t a_offDst, size_t a_cchDst, const char *a_pszSrc, size_t a_cchSrc) RT_NOEXCEPT;
788};
789
790/** @} */
791
792#endif /* !IPRT_INCLUDED_cpp_restclient_h */
793
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette