azure-core
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
win_http_request.hpp
Go to the documentation of this file.
1// Copyright (c) Microsoft Corporation.
2// Licensed under the MIT License.
3
4// cspell:words PCCERT HCERTSTORE
5
11#pragma once
12
14#include "azure/core/url.hpp"
15
16#if !defined(WIN32_LEAN_AND_MEAN)
17#define WIN32_LEAN_AND_MEAN
18#endif
19#if !defined(NOMINMAX)
20#define NOMINMAX
21#endif
22
23#include <windows.h>
24
25#include <memory>
26#include <mutex>
27#pragma warning(push)
28#pragma warning(disable : 6553)
29#pragma warning(disable : 6387) // An argument in result_macros.h may be '0', for the function
30 // 'GetProcAddress'.
31#include <wincrypt.h>
32
33#include <wil\resource.h>
34#pragma warning(pop)
35#include <winhttp.h>
36
37namespace Azure { namespace Core { namespace Http { namespace _detail {
38
39 class WinHttpRequest;
48 class WinHttpAction final {
49
50 // Containing HTTP request, used during the status operation callback.
51 WinHttpRequest* const m_httpRequest{};
52 wil::unique_event m_actionCompleteEvent;
53 // Mutex protecting all mutable members of the class.
54 std::mutex m_actionCompleteMutex;
55 DWORD m_expectedStatus{};
56 DWORD m_stowedError{};
57 DWORD_PTR m_stowedErrorInformation{};
58 DWORD m_bytesAvailable{};
59
60 /*
61 * Callback from WinHTTP called after the TLS certificates are received when the caller sets
62 * expected TLS root certificates.
63 */
64 static void CALLBACK StatusCallback(
65 HINTERNET hInternet,
66 DWORD_PTR dwContext,
67 DWORD dwInternetStatus,
68 LPVOID lpvStatusInformation,
69 DWORD dwStatusInformationLength);
70
71 /*
72 * Callback from WinHTTP called after the TLS certificates are received when the caller sets
73 * expected TLS root certificates.
74 */
75 void OnHttpStatusOperation(
76 HINTERNET hInternet,
77 DWORD internetStatus,
78 LPVOID statusInformation,
79 DWORD statusInformationLength);
80
81 public:
87 WinHttpAction(WinHttpRequest* request)
88 // Create a non-inheritable anonymous manual reset event initialized as unset.
89 : m_httpRequest(request), m_actionCompleteEvent(CreateEvent(nullptr, TRUE, FALSE, nullptr))
90 {
91 if (!m_actionCompleteEvent)
92 {
93 throw std::runtime_error("Error creating Action Complete Event.");
94 }
95 }
96
103 bool RegisterWinHttpStatusCallback(
104 Azure::Core::_internal::UniqueHandle<HINTERNET> const& internetHandle);
105
126 bool WaitForAction(
127 std::function<void()> initiateAction,
128 DWORD expectedCallbackStatus,
129 Azure::Core::Context const& context,
130 Azure::DateTime::duration const& pollDuration = std::chrono::milliseconds(800));
131
135 void CompleteAction();
136
141 void CompleteActionWithData(DWORD bytesAvailable);
142
147 void CompleteActionWithError(DWORD_PTR stowedErrorInformation, DWORD stowedError);
148 DWORD GetStowedError();
149 DWORD_PTR GetStowedErrorInformation();
150 DWORD GetBytesAvailable();
151 };
152
156 class WinHttpRequest final {
157 bool m_requestHandleClosed{false};
158 Azure::Core::_internal::UniqueHandle<HINTERNET> m_requestHandle;
159 std::unique_ptr<WinHttpAction> m_httpAction;
160 std::vector<std::string> m_expectedTlsRootCertificates;
161 wil::unique_cert_context m_tlsClientCertificate;
162
163 /*
164 * Adds the specified trusted certificates to the specified certificate store.
165 */
166 bool AddCertificatesToStore(
167 std::vector<std::string> const& trustedCertificates,
168 HCERTSTORE const hCertStore) const;
169 /*
170 * Verifies that the certificate context is in the trustedCertificates set of certificates.
171 */
172 bool VerifyCertificatesInChain(
173 std::vector<std::string> const& trustedCertificates,
174 PCCERT_CONTEXT serverCertificate) const;
175
176 public:
177 WinHttpRequest(
178 Azure::Core::_internal::UniqueHandle<HINTERNET> const& connectionHandle,
179 Azure::Core::Url const& url,
180 Azure::Core::Http::HttpMethod const& method,
181 PCCERT_CONTEXT tlsClientCertificate,
182 WinHttpTransportOptions const& options);
183
184 ~WinHttpRequest();
185 void MarkRequestHandleClosed() { m_requestHandleClosed = true; };
186 void Upload(Azure::Core::Http::Request& request, Azure::Core::Context const& context);
187 void SendRequest(Azure::Core::Http::Request& request, Azure::Core::Context const& context);
188 void ReceiveResponse(Azure::Core::Context const& context);
189 int64_t GetContentLength(HttpMethod requestMethod, HttpStatusCode responseStatusCode);
190 std::unique_ptr<RawResponse> SendRequestAndGetResponse(HttpMethod requestMethod);
191 size_t ReadData(uint8_t* buffer, size_t bufferSize, Azure::Core::Context const& context);
192 void EnableWebSocketsSupport();
193 void HandleExpectedTlsRootCertificates(HINTERNET hInternet);
194 };
195
196 class WinHttpStream final : public Azure::Core::IO::BodyStream {
197 private:
198 std::unique_ptr<_detail::WinHttpRequest> m_requestHandle;
199 bool m_isEOF;
200
212 int64_t m_contentLength;
213
214 int64_t m_streamTotalRead;
215
225 size_t OnRead(uint8_t* buffer, size_t count, Azure::Core::Context const& context) override;
226
227 public:
228 WinHttpStream(std::unique_ptr<_detail::WinHttpRequest>& requestHandle, int64_t contentLength)
229 : m_requestHandle(std::move(requestHandle)), m_contentLength(contentLength), m_isEOF(false),
230 m_streamTotalRead(0)
231 {
232 }
233
239 int64_t Length() const override { return this->m_contentLength; }
240 };
241
242}}}} // namespace Azure::Core::Http::_detail
A context is a node within a unidirectional tree that represents deadlines and key/value pairs.
Definition context.hpp:72
The method to be performed on the resource identified by the Request.
Definition http.hpp:95
A request message from a client to a server.
Definition http.hpp:183
Used to read data to/from a service.
Definition body_stream.hpp:32
virtual int64_t Length() const =0
Get the length of the data.
Represents the location where a request will be performed.
Definition url.hpp:46
Compute the hash value for the input binary data, using SHA256, SHA384 and SHA512.
Definition azure_assert.hpp:57
Uniform Resource Locator (URL).
Azure::Core::Http::HttpTransport implementation via WinHTTP.
const struct _CERT_CONTEXT * PCCERT_CONTEXT
Declaration of a Windows PCCERT_CONTEXT structure from the Windows SDK.
Definition win_http_transport.hpp:28