Blame libfreerdp/core/gateway/http.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Hypertext Transfer Protocol (HTTP)
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <errno.h>
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
#include <winpr/print.h>
Packit 1fb8d4
#include <winpr/stream.h>
Packit 1fb8d4
#include <winpr/string.h>
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_VALGRIND_MEMCHECK_H
Packit 1fb8d4
#include <valgrind/memcheck.h>
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include "http.h"
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("core.gateway.http")
Packit 1fb8d4
Packit 1fb8d4
#define RESPONSE_SIZE_LIMIT 64 * 1024 * 1024
Packit 1fb8d4
Packit 1fb8d4
struct _http_context
Packit 1fb8d4
{
Packit 1fb8d4
	char* Method;
Packit 1fb8d4
	char* URI;
Packit 1fb8d4
	char* UserAgent;
Packit 1fb8d4
	char* Host;
Packit 1fb8d4
	char* Accept;
Packit 1fb8d4
	char* CacheControl;
Packit 1fb8d4
	char* Connection;
Packit 1fb8d4
	char* Pragma;
Packit 1fb8d4
	char* RdgConnectionId;
Packit 1fb8d4
	char* RdgAuthScheme;
Packit 1fb8d4
};
Packit 1fb8d4
Packit 1fb8d4
struct _http_request
Packit 1fb8d4
{
Packit 1fb8d4
	char* Method;
Packit 1fb8d4
	char* URI;
Packit 1fb8d4
	char* AuthScheme;
Packit 1fb8d4
	char* AuthParam;
Packit 1fb8d4
	char* Authorization;
Packit 1fb8d4
	size_t ContentLength;
Packit 1fb8d4
	char* Content;
Packit 1fb8d4
	char* TransferEncoding;
Packit 1fb8d4
};
Packit 1fb8d4
Packit 1fb8d4
struct _http_response
Packit 1fb8d4
{
Packit 1fb8d4
	size_t count;
Packit 1fb8d4
	char** lines;
Packit 1fb8d4
Packit 1fb8d4
	long StatusCode;
Packit 1fb8d4
	const char* ReasonPhrase;
Packit 1fb8d4
Packit 1fb8d4
	size_t ContentLength;
Packit 1fb8d4
	const char* ContentType;
Packit 1fb8d4
Packit 1fb8d4
	size_t BodyLength;
Packit 1fb8d4
	BYTE* BodyContent;
Packit 1fb8d4
Packit 1fb8d4
	wListDictionary* Authenticates;
Packit 1fb8d4
	wStream* data;
Packit 1fb8d4
};
Packit 1fb8d4
Packit 1fb8d4
static char* string_strnstr(char* str1, const char* str2, size_t slen)
Packit 1fb8d4
{
Packit 1fb8d4
	char c, sc;
Packit 1fb8d4
	size_t len;
Packit 1fb8d4
Packit 1fb8d4
	if ((c = *str2++) != '\0')
Packit 1fb8d4
	{
Packit Service 5a9772
		len = strnlen(str2, slen + 1);
Packit 1fb8d4
Packit 1fb8d4
		do
Packit 1fb8d4
		{
Packit 1fb8d4
			do
Packit 1fb8d4
			{
Packit 1fb8d4
				if (slen-- < 1 || (sc = *str1++) == '\0')
Packit 1fb8d4
					return NULL;
Packit Service 5a9772
			} while (sc != c);
Packit 1fb8d4
Packit 1fb8d4
			if (len > slen)
Packit 1fb8d4
				return NULL;
Packit Service 5a9772
		} while (strncmp(str1, str2, len) != 0);
Packit 1fb8d4
Packit 1fb8d4
		str1--;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return str1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL strings_equals_nocase(const void* obj1, const void* obj2)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!obj1 || !obj2)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return _stricmp(obj1, obj2) == 0;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
HttpContext* http_context_new(void)
Packit 1fb8d4
{
Packit Service 5a9772
	return (HttpContext*)calloc(1, sizeof(HttpContext));
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_method(HttpContext* context, const char* Method)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !Method)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->Method);
Packit 1fb8d4
	context->Method = _strdup(Method);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->Method)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* http_context_get_uri(HttpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	return context->URI;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_uri(HttpContext* context, const char* URI)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !URI)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->URI);
Packit 1fb8d4
	context->URI = _strdup(URI);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->URI)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_user_agent(HttpContext* context, const char* UserAgent)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !UserAgent)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->UserAgent);
Packit 1fb8d4
	context->UserAgent = _strdup(UserAgent);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->UserAgent)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_host(HttpContext* context, const char* Host)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !Host)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->Host);
Packit 1fb8d4
	context->Host = _strdup(Host);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->Host)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_accept(HttpContext* context, const char* Accept)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !Accept)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->Accept);
Packit 1fb8d4
	context->Accept = _strdup(Accept);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->Accept)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_cache_control(HttpContext* context, const char* CacheControl)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !CacheControl)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->CacheControl);
Packit 1fb8d4
	context->CacheControl = _strdup(CacheControl);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->CacheControl)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_connection(HttpContext* context, const char* Connection)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !Connection)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->Connection);
Packit 1fb8d4
	context->Connection = _strdup(Connection);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->Connection)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_pragma(HttpContext* context, const char* Pragma)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !Pragma)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->Pragma);
Packit 1fb8d4
	context->Pragma = _strdup(Pragma);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->Pragma)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_rdg_connection_id(HttpContext* context, const char* RdgConnectionId)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !RdgConnectionId)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->RdgConnectionId);
Packit 1fb8d4
	context->RdgConnectionId = _strdup(RdgConnectionId);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->RdgConnectionId)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_context_set_rdg_auth_scheme(HttpContext* context, const char* RdgAuthScheme)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !RdgAuthScheme)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(context->RdgAuthScheme);
Packit 1fb8d4
	context->RdgAuthScheme = _strdup(RdgAuthScheme);
Packit 1fb8d4
	return context->RdgAuthScheme != NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void http_context_free(HttpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	if (context)
Packit 1fb8d4
	{
Packit 1fb8d4
		free(context->UserAgent);
Packit 1fb8d4
		free(context->Host);
Packit 1fb8d4
		free(context->URI);
Packit 1fb8d4
		free(context->Accept);
Packit 1fb8d4
		free(context->Method);
Packit 1fb8d4
		free(context->CacheControl);
Packit 1fb8d4
		free(context->Connection);
Packit 1fb8d4
		free(context->Pragma);
Packit 1fb8d4
		free(context->RdgConnectionId);
Packit 1fb8d4
		free(context->RdgAuthScheme);
Packit 1fb8d4
		free(context);
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_request_set_method(HttpRequest* request, const char* Method)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request || !Method)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(request->Method);
Packit 1fb8d4
	request->Method = _strdup(Method);
Packit 1fb8d4
Packit 1fb8d4
	if (!request->Method)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_request_set_uri(HttpRequest* request, const char* URI)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request || !URI)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(request->URI);
Packit 1fb8d4
	request->URI = _strdup(URI);
Packit 1fb8d4
Packit 1fb8d4
	if (!request->URI)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_request_set_auth_scheme(HttpRequest* request, const char* AuthScheme)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request || !AuthScheme)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(request->AuthScheme);
Packit 1fb8d4
	request->AuthScheme = _strdup(AuthScheme);
Packit 1fb8d4
Packit 1fb8d4
	if (!request->AuthScheme)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_request_set_auth_param(HttpRequest* request, const char* AuthParam)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request || !AuthParam)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(request->AuthParam);
Packit 1fb8d4
	request->AuthParam = _strdup(AuthParam);
Packit 1fb8d4
Packit 1fb8d4
	if (!request->AuthParam)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_request_set_transfer_encoding(HttpRequest* request, const char* TransferEncoding)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request || !TransferEncoding)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	free(request->TransferEncoding);
Packit 1fb8d4
	request->TransferEncoding = _strdup(TransferEncoding);
Packit 1fb8d4
Packit 1fb8d4
	if (!request->TransferEncoding)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_encode_print(wStream* s, const char* fmt, ...)
Packit 1fb8d4
{
Packit 1fb8d4
	char* str;
Packit 1fb8d4
	va_list ap;
Packit 1fb8d4
	int length, used;
Packit 1fb8d4
Packit 1fb8d4
	if (!s || !fmt)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	va_start(ap, fmt);
Packit 1fb8d4
	length = vsnprintf(NULL, 0, fmt, ap) + 1;
Packit 1fb8d4
	va_end(ap);
Packit 1fb8d4
Packit 1fb8d4
	if (!Stream_EnsureRemainingCapacity(s, (size_t)length))
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	str = (char*)Stream_Pointer(s);
Packit 1fb8d4
	va_start(ap, fmt);
Packit 1fb8d4
	used = vsnprintf(str, (size_t)length, fmt, ap);
Packit 1fb8d4
	va_end(ap);
Packit 1fb8d4
Packit 1fb8d4
	/* Strip the trailing '\0' from the string. */
Packit 1fb8d4
	if ((used + 1) != length)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	Stream_Seek(s, (size_t)used);
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_encode_body_line(wStream* s, const char* param, const char* value)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!s || !param || !value)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return http_encode_print(s, "%s: %s\r\n", param, value);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_encode_content_length_line(wStream* s, size_t ContentLength)
Packit 1fb8d4
{
Packit Service 5a9772
	return http_encode_print(s, "Content-Length: %" PRIdz "\r\n", ContentLength);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_encode_header_line(wStream* s, const char* Method, const char* URI)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!s || !Method || !URI)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return http_encode_print(s, "%s %s HTTP/1.1\r\n", Method, URI);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_encode_authorization_line(wStream* s, const char* AuthScheme,
Packit Service 5a9772
                                           const char* AuthParam)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!s || !AuthScheme || !AuthParam)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return http_encode_print(s, "Authorization: %s %s\r\n", AuthScheme, AuthParam);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
wStream* http_request_write(HttpContext* context, HttpRequest* request)
Packit 1fb8d4
{
Packit 1fb8d4
	wStream* s;
Packit 1fb8d4
Packit 1fb8d4
	if (!context || !request)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	s = Stream_New(NULL, 1024);
Packit 1fb8d4
Packit 1fb8d4
	if (!s)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	if (!http_encode_header_line(s, request->Method, request->URI) ||
Packit 1fb8d4
	    !http_encode_body_line(s, "Cache-Control", context->CacheControl) ||
Packit 1fb8d4
	    !http_encode_body_line(s, "Connection", context->Connection) ||
Packit 1fb8d4
	    !http_encode_body_line(s, "Pragma", context->Pragma) ||
Packit 1fb8d4
	    !http_encode_body_line(s, "Accept", context->Accept) ||
Packit 1fb8d4
	    !http_encode_body_line(s, "User-Agent", context->UserAgent) ||
Packit 1fb8d4
	    !http_encode_body_line(s, "Host", context->Host))
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	if (context->RdgConnectionId)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!http_encode_body_line(s, "RDG-Connection-Id", context->RdgConnectionId))
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (context->RdgAuthScheme)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!http_encode_body_line(s, "RDG-Auth-Scheme", context->RdgAuthScheme))
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (request->TransferEncoding)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!http_encode_body_line(s, "Transfer-Encoding", request->TransferEncoding))
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!http_encode_content_length_line(s, request->ContentLength))
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (request->Authorization)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!http_encode_body_line(s, "Authorization", request->Authorization))
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (request->AuthScheme && request->AuthParam)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!http_encode_authorization_line(s, request->AuthScheme, request->AuthParam))
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	Stream_Write(s, "\r\n", 2);
Packit 1fb8d4
	Stream_SealLength(s);
Packit 1fb8d4
	return s;
Packit 1fb8d4
fail:
Packit 1fb8d4
	Stream_Free(s, TRUE);
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
HttpRequest* http_request_new(void)
Packit 1fb8d4
{
Packit Service 5a9772
	return (HttpRequest*)calloc(1, sizeof(HttpRequest));
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void http_request_free(HttpRequest* request)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	free(request->AuthParam);
Packit 1fb8d4
	free(request->AuthScheme);
Packit 1fb8d4
	free(request->Authorization);
Packit 1fb8d4
	free(request->Content);
Packit 1fb8d4
	free(request->Method);
Packit 1fb8d4
	free(request->URI);
Packit 1fb8d4
	free(request->TransferEncoding);
Packit 1fb8d4
	free(request);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static BOOL http_response_parse_header_status_line(HttpResponse* response, const char* status_line)
Packit 1fb8d4
{
Packit Service 5a9772
	BOOL rc = FALSE;
Packit 1fb8d4
	char* separator = NULL;
Packit 1fb8d4
	char* status_code;
Packit 1fb8d4
	char* reason_phrase;
Packit 1fb8d4
Packit 1fb8d4
	if (!response)
Packit Service 5a9772
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	if (status_line)
Packit 1fb8d4
		separator = strchr(status_line, ' ');
Packit 1fb8d4
Packit 1fb8d4
	if (!separator)
Packit Service 5a9772
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	status_code = separator + 1;
Packit 1fb8d4
	separator = strchr(status_code, ' ');
Packit 1fb8d4
Packit 1fb8d4
	if (!separator)
Packit Service 5a9772
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	reason_phrase = separator + 1;
Packit 1fb8d4
	*separator = '\0';
Packit 1fb8d4
	errno = 0;
Packit 1fb8d4
	{
Packit 1fb8d4
		long val = strtol(status_code, NULL, 0);
Packit 1fb8d4
Packit 1fb8d4
		if ((errno != 0) || (val < 0) || (val > INT16_MAX))
Packit Service 5a9772
			goto fail;
Packit 1fb8d4
Packit 1fb8d4
		response->StatusCode = strtol(status_code, NULL, 0);
Packit 1fb8d4
	}
Packit 1fb8d4
	response->ReasonPhrase = reason_phrase;
Packit 1fb8d4
Packit 1fb8d4
	if (!response->ReasonPhrase)
Packit Service 5a9772
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	*separator = ' ';
Packit Service 5a9772
	rc = TRUE;
Packit Service 5a9772
fail:
Packit Service 5a9772
Packit Service 5a9772
	if (!rc)
Packit Service 5a9772
		WLog_ERR(TAG, "http_response_parse_header_status_line failed [%s]", status_line);
Packit Service 5a9772
Packit Service 5a9772
	return rc;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_response_parse_header_field(HttpResponse* response, const char* name,
Packit Service 5a9772
                                             const char* value)
Packit 1fb8d4
{
Packit 1fb8d4
	BOOL status = TRUE;
Packit 1fb8d4
Packit 1fb8d4
	if (!response || !name)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (_stricmp(name, "Content-Length") == 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		unsigned long long val;
Packit 1fb8d4
		errno = 0;
Packit 1fb8d4
		val = _strtoui64(value, NULL, 0);
Packit 1fb8d4
Packit 1fb8d4
		if ((errno != 0) || (val > INT32_MAX))
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		response->ContentLength = val;
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (_stricmp(name, "Content-Type") == 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		response->ContentType = value;
Packit 1fb8d4
Packit 1fb8d4
		if (!response->ContentType)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (_stricmp(name, "WWW-Authenticate") == 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		char* separator = NULL;
Packit 1fb8d4
		const char* authScheme = NULL;
Packit 1fb8d4
		char* authValue = NULL;
Packit 1fb8d4
		separator = strchr(value, ' ');
Packit 1fb8d4
Packit 1fb8d4
		if (separator)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* WWW-Authenticate: Basic realm=""
Packit 1fb8d4
			 * WWW-Authenticate: NTLM base64token
Packit 1fb8d4
			 * WWW-Authenticate: Digest realm="testrealm@host.com", qop="auth, auth-int",
Packit 1fb8d4
			 * 					nonce="dcd98b7102dd2f0e8b11d0f600bfb0c093",
Packit 1fb8d4
			 * 					opaque="5ccc069c403ebaf9f0171e9517f40e41"
Packit 1fb8d4
			 */
Packit 1fb8d4
			*separator = '\0';
Packit 1fb8d4
			authScheme = value;
Packit 1fb8d4
			authValue = separator + 1;
Packit 1fb8d4
Packit 1fb8d4
			if (!authScheme || !authValue)
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			authScheme = value;
Packit 1fb8d4
Packit 1fb8d4
			if (!authScheme)
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
Packit 1fb8d4
			authValue = NULL;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		status = ListDictionary_Add(response->Authenticates, authScheme, authValue);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_response_parse_header(HttpResponse* response)
Packit 1fb8d4
{
Packit Service 5a9772
	BOOL rc = FALSE;
Packit 1fb8d4
	char c;
Packit 1fb8d4
	size_t count;
Packit 1fb8d4
	char* line;
Packit 1fb8d4
	char* name;
Packit 1fb8d4
	char* value;
Packit 1fb8d4
	char* colon_pos;
Packit 1fb8d4
	char* end_of_header;
Packit 1fb8d4
	char end_of_header_char;
Packit 1fb8d4
Packit 1fb8d4
	if (!response)
Packit Service 5a9772
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	if (!response->lines)
Packit Service 5a9772
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	if (!http_response_parse_header_status_line(response, response->lines[0]))
Packit Service 5a9772
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	for (count = 1; count < response->count; count++)
Packit 1fb8d4
	{
Packit 1fb8d4
		line = response->lines[count];
Packit 1fb8d4
Packit 1fb8d4
		/**
Packit 1fb8d4
		 * name         end_of_header
Packit 1fb8d4
		 * |            |
Packit 1fb8d4
		 * v            v
Packit 1fb8d4
		 * <header name>   :     <header value>
Packit 1fb8d4
		 *                 ^     ^
Packit 1fb8d4
		 *                 |     |
Packit 1fb8d4
		 *         colon_pos     value
Packit 1fb8d4
		 */
Packit 1fb8d4
		if (line)
Packit 1fb8d4
			colon_pos = strchr(line, ':');
Packit 1fb8d4
		else
Packit 1fb8d4
			colon_pos = NULL;
Packit 1fb8d4
Packit 1fb8d4
		if ((colon_pos == NULL) || (colon_pos == line))
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		/* retrieve the position just after header name */
Packit 1fb8d4
		for (end_of_header = colon_pos; end_of_header != line; end_of_header--)
Packit 1fb8d4
		{
Packit 1fb8d4
			c = end_of_header[-1];
Packit 1fb8d4
Packit 1fb8d4
			if (c != ' ' && c != '\t' && c != ':')
Packit 1fb8d4
				break;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (end_of_header == line)
Packit Service 5a9772
			goto fail;
Packit 1fb8d4
Packit 1fb8d4
		end_of_header_char = *end_of_header;
Packit 1fb8d4
		*end_of_header = '\0';
Packit 1fb8d4
		name = line;
Packit 1fb8d4
Packit 1fb8d4
		/* eat space and tabs before header value */
Packit 1fb8d4
		for (value = colon_pos + 1; *value; value++)
Packit 1fb8d4
		{
Packit 1fb8d4
			if ((*value != ' ') && (*value != '\t'))
Packit 1fb8d4
				break;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (!http_response_parse_header_field(response, name, value))
Packit Service 5a9772
			goto fail;
Packit 1fb8d4
Packit 1fb8d4
		*end_of_header = end_of_header_char;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	rc = TRUE;
Packit Service 5a9772
fail:
Packit Service 5a9772
Packit Service 5a9772
	if (!rc)
Packit Service 5a9772
		WLog_ERR(TAG, "%s: parsing failed", __FUNCTION__);
Packit Service 5a9772
Packit Service 5a9772
	return rc;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_response_print(HttpResponse* response)
Packit 1fb8d4
{
Packit 1fb8d4
	size_t i;
Packit 1fb8d4
Packit 1fb8d4
	if (!response)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < response->count; i++)
Packit 1fb8d4
		WLog_ERR(TAG, "%s", response->lines[i]);
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL http_use_content_length(const char* cur)
Packit 1fb8d4
{
Packit 1fb8d4
	size_t pos = 0;
Packit 1fb8d4
Packit 1fb8d4
	if (!cur)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (_strnicmp(cur, "application/rpc", 15) == 0)
Packit 1fb8d4
		pos = 15;
Packit 1fb8d4
	else if (_strnicmp(cur, "text/plain", 10) == 0)
Packit 1fb8d4
		pos = 10;
Packit 1fb8d4
	else if (_strnicmp(cur, "text/html", 9) == 0)
Packit 1fb8d4
		pos = 9;
Packit 1fb8d4
Packit 1fb8d4
	if (pos > 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		char end = cur[pos];
Packit 1fb8d4
Packit 1fb8d4
		switch (end)
Packit 1fb8d4
		{
Packit 1fb8d4
			case ' ':
Packit 1fb8d4
			case ';':
Packit 1fb8d4
			case '\0':
Packit 1fb8d4
			case '\r':
Packit 1fb8d4
			case '\n':
Packit 1fb8d4
				return TRUE;
Packit 1fb8d4
Packit 1fb8d4
			default:
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static int print_bio_error(const char* str, size_t len, void* bp)
Packit Service 5a9772
{
Packit Service 5a9772
	WINPR_UNUSED(len);
Packit Service 5a9772
	WINPR_UNUSED(bp);
Packit Service 5a9772
	WLog_ERR(TAG, "%s", str);
Packit Service 5a9772
	return len;
Packit Service 5a9772
}
Packit Service 5a9772
Packit 1fb8d4
HttpResponse* http_response_recv(rdpTls* tls, BOOL readContentLength)
Packit 1fb8d4
{
Packit 1fb8d4
	size_t size;
Packit 1fb8d4
	size_t position;
Packit 1fb8d4
	size_t bodyLength = 0;
Packit 1fb8d4
	size_t payloadOffset;
Packit 1fb8d4
	HttpResponse* response;
Packit 1fb8d4
	size = 2048;
Packit 1fb8d4
	payloadOffset = 0;
Packit 1fb8d4
	response = http_response_new();
Packit 1fb8d4
Packit 1fb8d4
	if (!response)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	response->ContentLength = 0;
Packit 1fb8d4
Packit 1fb8d4
	while (payloadOffset == 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		size_t s;
Packit 1fb8d4
		char* end;
Packit 1fb8d4
		/* Read until we encounter \r\n\r\n */
Packit 1fb8d4
		int status = BIO_read(tls->bio, Stream_Pointer(response->data), 1);
Packit 1fb8d4
Packit 1fb8d4
		if (status <= 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			if (!BIO_should_retry(tls->bio))
Packit Service 5a9772
			{
Packit Service 5a9772
				WLog_ERR(TAG, "%s: Retries exceeded", __FUNCTION__);
Packit Service 5a9772
				ERR_print_errors_cb(print_bio_error, NULL);
Packit 1fb8d4
				goto out_error;
Packit Service 5a9772
			}
Packit 1fb8d4
Packit 1fb8d4
			USleep(100);
Packit 1fb8d4
			continue;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_VALGRIND_MEMCHECK_H
Packit 1fb8d4
		VALGRIND_MAKE_MEM_DEFINED(Stream_Pointer(response->data), status);
Packit 1fb8d4
#endif
Packit 1fb8d4
		Stream_Seek(response->data, (size_t)status);
Packit 1fb8d4
Packit 1fb8d4
		if (!Stream_EnsureRemainingCapacity(response->data, 1024))
Packit 1fb8d4
			goto out_error;
Packit 1fb8d4
Packit 1fb8d4
		position = Stream_GetPosition(response->data);
Packit 1fb8d4
Packit 1fb8d4
		if (position < 4)
Packit 1fb8d4
			continue;
Packit 1fb8d4
		else if (position > RESPONSE_SIZE_LIMIT)
Packit 1fb8d4
		{
Packit Service 5a9772
			WLog_ERR(TAG, "Request header too large! (%" PRIdz " bytes) Aborting!", bodyLength);
Packit 1fb8d4
			goto out_error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* Always check at most the lase 8 bytes for occurance of the desired
Packit 1fb8d4
		 * sequence of \r\n\r\n */
Packit 1fb8d4
		s = (position > 8) ? 8 : position;
Packit 1fb8d4
		end = (char*)Stream_Pointer(response->data) - s;
Packit 1fb8d4
Packit 1fb8d4
		if (string_strnstr(end, "\r\n\r\n", s) != NULL)
Packit 1fb8d4
			payloadOffset = Stream_GetPosition(response->data);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (payloadOffset)
Packit 1fb8d4
	{
Packit 1fb8d4
		size_t count = 0;
Packit 1fb8d4
		char* buffer = (char*)Stream_Buffer(response->data);
Packit Service 5a9772
		char* line = (char*)Stream_Buffer(response->data);
Packit Service 5a9772
		char* context = NULL;
Packit 1fb8d4
Packit 1fb8d4
		while ((line = string_strnstr(line, "\r\n", payloadOffset - (line - buffer) - 2UL)))
Packit 1fb8d4
		{
Packit 1fb8d4
			line += 2;
Packit 1fb8d4
			count++;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		response->count = count;
Packit 1fb8d4
Packit 1fb8d4
		if (count)
Packit 1fb8d4
		{
Packit Service 5a9772
			response->lines = (char**)calloc(response->count, sizeof(char*));
Packit 1fb8d4
Packit 1fb8d4
			if (!response->lines)
Packit 1fb8d4
				goto out_error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		buffer[payloadOffset - 1] = '\0';
Packit 1fb8d4
		buffer[payloadOffset - 2] = '\0';
Packit 1fb8d4
		count = 0;
Packit Service 5a9772
		line = strtok_s(buffer, "\r\n", &context);
Packit 1fb8d4
Packit 1fb8d4
		while (line && (response->count > count))
Packit 1fb8d4
		{
Packit 1fb8d4
			response->lines[count] = line;
Packit Service 5a9772
			line = strtok_s(NULL, "\r\n", &context);
Packit 1fb8d4
			count++;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (!http_response_parse_header(response))
Packit 1fb8d4
			goto out_error;
Packit 1fb8d4
Packit 1fb8d4
		response->BodyLength = Stream_GetPosition(response->data) - payloadOffset;
Packit 1fb8d4
		bodyLength = response->BodyLength; /* expected body length */
Packit 1fb8d4
Packit 1fb8d4
		if (readContentLength)
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* cur = response->ContentType;
Packit 1fb8d4
Packit 1fb8d4
			while (cur != NULL)
Packit 1fb8d4
			{
Packit 1fb8d4
				if (http_use_content_length(cur))
Packit 1fb8d4
				{
Packit 1fb8d4
					if (response->ContentLength < RESPONSE_SIZE_LIMIT)
Packit 1fb8d4
						bodyLength = response->ContentLength;
Packit 1fb8d4
Packit 1fb8d4
					break;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				cur = strchr(cur, ';');
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (bodyLength > RESPONSE_SIZE_LIMIT)
Packit 1fb8d4
		{
Packit Service 5a9772
			WLog_ERR(TAG, "Expected request body too large! (%" PRIdz " bytes) Aborting!",
Packit Service 5a9772
			         bodyLength);
Packit 1fb8d4
			goto out_error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* Fetch remaining body! */
Packit 1fb8d4
		while (response->BodyLength < bodyLength)
Packit 1fb8d4
		{
Packit 1fb8d4
			int status;
Packit 1fb8d4
Packit 1fb8d4
			if (!Stream_EnsureRemainingCapacity(response->data, bodyLength - response->BodyLength))
Packit 1fb8d4
				goto out_error;
Packit 1fb8d4
Packit Service 5a9772
			status = BIO_read(tls->bio, Stream_Pointer(response->data),
Packit Service 5a9772
			                  bodyLength - response->BodyLength);
Packit 1fb8d4
Packit 1fb8d4
			if (status <= 0)
Packit 1fb8d4
			{
Packit 1fb8d4
				if (!BIO_should_retry(tls->bio))
Packit Service 5a9772
				{
Packit Service 5a9772
					WLog_ERR(TAG, "%s: Retries exceeded", __FUNCTION__);
Packit Service 5a9772
					ERR_print_errors_cb(print_bio_error, NULL);
Packit 1fb8d4
					goto out_error;
Packit Service 5a9772
				}
Packit 1fb8d4
Packit 1fb8d4
				USleep(100);
Packit 1fb8d4
				continue;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			Stream_Seek(response->data, (size_t)status);
Packit 1fb8d4
			response->BodyLength += (unsigned long)status;
Packit 1fb8d4
Packit 1fb8d4
			if (response->BodyLength > RESPONSE_SIZE_LIMIT)
Packit 1fb8d4
			{
Packit Service 5a9772
				WLog_ERR(TAG, "Request body too large! (%" PRIdz " bytes) Aborting!",
Packit Service 5a9772
				         response->BodyLength);
Packit 1fb8d4
				goto out_error;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (response->BodyLength > 0)
Packit 1fb8d4
			response->BodyContent = &(Stream_Buffer(response->data))[payloadOffset];
Packit 1fb8d4
Packit 1fb8d4
		if (bodyLength != response->BodyLength)
Packit 1fb8d4
		{
Packit Service 5a9772
			WLog_WARN(TAG, "%s: %s unexpected body length: actual: %d, expected: %d", __FUNCTION__,
Packit 1fb8d4
			          response->ContentType, response->BodyLength, bodyLength);
Packit 1fb8d4
Packit 1fb8d4
			if (bodyLength > 0)
Packit 1fb8d4
				response->BodyLength = MIN(bodyLength, response->BodyLength);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return response;
Packit 1fb8d4
out_error:
Packit 1fb8d4
	http_response_free(response);
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
HttpResponse* http_response_new(void)
Packit 1fb8d4
{
Packit Service 5a9772
	HttpResponse* response = (HttpResponse*)calloc(1, sizeof(HttpResponse));
Packit 1fb8d4
Packit 1fb8d4
	if (!response)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	response->Authenticates = ListDictionary_New(FALSE);
Packit 1fb8d4
Packit 1fb8d4
	if (!response->Authenticates)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	response->data = Stream_New(NULL, 2048);
Packit 1fb8d4
Packit 1fb8d4
	if (!response->data)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	ListDictionary_KeyObject(response->Authenticates)->fnObjectEquals = strings_equals_nocase;
Packit 1fb8d4
	ListDictionary_ValueObject(response->Authenticates)->fnObjectEquals = strings_equals_nocase;
Packit 1fb8d4
	return response;
Packit 1fb8d4
fail:
Packit 1fb8d4
	http_response_free(response);
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void http_response_free(HttpResponse* response)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!response)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	free(response->lines);
Packit 1fb8d4
	ListDictionary_Free(response->Authenticates);
Packit 1fb8d4
	Stream_Free(response->data, TRUE);
Packit 1fb8d4
	free(response);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* http_request_get_uri(HttpRequest* request)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	return request->URI;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
SSIZE_T http_request_get_content_length(HttpRequest* request)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	return (SSIZE_T)request->ContentLength;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL http_request_set_content_length(HttpRequest* request, size_t length)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!request)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	request->ContentLength = length;
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
long http_response_get_status_code(HttpResponse* response)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!response)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	return response->StatusCode;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
SSIZE_T http_response_get_body_length(HttpResponse* response)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!response)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	return (SSIZE_T)response->BodyLength;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* http_response_get_auth_token(HttpResponse* respone, const char* method)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!respone || !method)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	if (!ListDictionary_Contains(respone->Authenticates, method))
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	return ListDictionary_GetItemValue(respone->Authenticates, method);
Packit 1fb8d4
}