2012-11-28 07:03:05 +04:00
|
|
|
/**
|
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
|
|
* RPC over HTTP (ncacn_http)
|
|
|
|
*
|
|
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "ncacn_http.h"
|
|
|
|
|
|
|
|
#include <winpr/crt.h>
|
|
|
|
#include <winpr/tchar.h>
|
2012-12-14 05:23:37 +04:00
|
|
|
#include <winpr/stream.h>
|
2012-11-28 07:03:05 +04:00
|
|
|
#include <winpr/dsparse.h>
|
2013-12-13 19:11:36 +04:00
|
|
|
#include <winpr/winhttp.h>
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2015-02-03 02:50:26 +03:00
|
|
|
#define TAG FREERDP_TAG("core.gateway.ntlm")
|
2015-01-13 21:50:46 +03:00
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
static wStream* rpc_ntlm_http_request(HttpContext* http, const char* method,
|
2018-01-15 14:43:37 +03:00
|
|
|
int contentLength, SecBuffer* ntlmToken)
|
2012-11-28 07:03:05 +04:00
|
|
|
{
|
2018-09-27 16:04:41 +03:00
|
|
|
wStream* s = NULL;
|
2018-09-27 16:19:41 +03:00
|
|
|
HttpRequest* request = NULL;
|
2015-02-11 23:26:22 +03:00
|
|
|
char* base64NtlmToken = NULL;
|
2018-09-27 16:04:41 +03:00
|
|
|
const char* uri;
|
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
if (!http || !method || !ntlmToken)
|
2018-09-27 16:04:41 +03:00
|
|
|
goto fail;
|
|
|
|
|
2015-02-11 23:26:22 +03:00
|
|
|
request = http_request_new();
|
2015-01-14 00:35:34 +03:00
|
|
|
|
2015-02-11 23:26:22 +03:00
|
|
|
if (ntlmToken)
|
|
|
|
base64NtlmToken = crypto_base64_encode(ntlmToken->pvBuffer, ntlmToken->cbBuffer);
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2018-09-27 16:04:41 +03:00
|
|
|
uri = http_context_get_uri(http);
|
|
|
|
|
|
|
|
if (!http_request_set_method(request, method) ||
|
|
|
|
!http_request_set_content_length(request, contentLength) ||
|
|
|
|
!http_request_set_uri(request, uri))
|
|
|
|
return NULL;
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2015-02-11 23:26:22 +03:00
|
|
|
if (base64NtlmToken)
|
2015-01-14 00:35:34 +03:00
|
|
|
{
|
2018-09-27 16:04:41 +03:00
|
|
|
if (!http_request_set_auth_scheme(request, "NTLM") ||
|
|
|
|
!http_request_set_auth_param(request, base64NtlmToken))
|
|
|
|
goto fail;
|
2015-01-14 00:35:34 +03:00
|
|
|
}
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2015-02-11 23:26:22 +03:00
|
|
|
s = http_request_write(http, request);
|
2018-09-27 16:04:41 +03:00
|
|
|
fail:
|
2015-02-11 23:26:22 +03:00
|
|
|
http_request_free(request);
|
|
|
|
free(base64NtlmToken);
|
2012-11-28 07:03:05 +04:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2018-09-27 17:05:14 +03:00
|
|
|
BOOL rpc_ncacn_http_send_in_channel_request(RpcChannel* inChannel)
|
2012-11-28 07:03:05 +04:00
|
|
|
{
|
2012-12-14 05:23:37 +04:00
|
|
|
wStream* s;
|
2015-02-02 04:47:43 +03:00
|
|
|
int status;
|
|
|
|
int contentLength;
|
|
|
|
BOOL continueNeeded;
|
2018-09-27 16:19:41 +03:00
|
|
|
rdpNtlm* ntlm;
|
|
|
|
HttpContext* http;
|
|
|
|
|
|
|
|
if (!inChannel || !inChannel->ntlm || !inChannel->http)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
ntlm = inChannel->ntlm;
|
|
|
|
http = inChannel->http;
|
2015-02-02 04:47:43 +03:00
|
|
|
continueNeeded = ntlm_authenticate(ntlm);
|
|
|
|
contentLength = (continueNeeded) ? 0 : 0x40000000;
|
2018-09-27 16:19:41 +03:00
|
|
|
s = rpc_ntlm_http_request(http, "RPC_IN_DATA", contentLength, &ntlm->outputBuffer[0]);
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2015-02-02 04:47:43 +03:00
|
|
|
if (!s)
|
|
|
|
return -1;
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2018-09-27 17:05:14 +03:00
|
|
|
status = rpc_channel_write(inChannel, Stream_Buffer(s), Stream_Length(s));
|
2012-12-14 05:23:37 +04:00
|
|
|
Stream_Free(s, TRUE);
|
2015-02-02 04:47:43 +03:00
|
|
|
return (status > 0) ? 1 : -1;
|
2012-11-28 07:03:05 +04:00
|
|
|
}
|
|
|
|
|
2018-09-27 17:05:14 +03:00
|
|
|
BOOL rpc_ncacn_http_recv_in_channel_response(RpcChannel* inChannel,
|
2018-01-15 14:43:37 +03:00
|
|
|
HttpResponse* response)
|
2012-11-28 07:03:05 +04:00
|
|
|
{
|
2018-09-27 17:05:14 +03:00
|
|
|
const char* token64 = NULL;
|
2015-02-03 22:44:31 +03:00
|
|
|
int ntlmTokenLength = 0;
|
|
|
|
BYTE* ntlmTokenData = NULL;
|
2018-09-27 16:19:41 +03:00
|
|
|
rdpNtlm* ntlm;
|
|
|
|
|
|
|
|
if (!inChannel || !response || !inChannel->ntlm)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
ntlm = inChannel->ntlm;
|
2018-09-27 16:04:41 +03:00
|
|
|
token64 = http_response_get_auth_token(response, "NTLM");
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2018-09-27 16:04:41 +03:00
|
|
|
if (token64)
|
2015-02-03 22:44:31 +03:00
|
|
|
crypto_base64_decode(token64, strlen(token64), &ntlmTokenData, &ntlmTokenLength);
|
|
|
|
|
|
|
|
if (ntlmTokenData && ntlmTokenLength)
|
|
|
|
{
|
|
|
|
ntlm->inputBuffer[0].pvBuffer = ntlmTokenData;
|
|
|
|
ntlm->inputBuffer[0].cbBuffer = ntlmTokenLength;
|
2013-12-13 19:11:36 +04:00
|
|
|
}
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
return TRUE;
|
2012-11-28 07:03:05 +04:00
|
|
|
}
|
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
BOOL rpc_ncacn_http_ntlm_init(rdpContext* context, RpcChannel* channel)
|
2012-11-28 07:03:05 +04:00
|
|
|
{
|
2018-09-27 16:19:41 +03:00
|
|
|
rdpTls* tls;
|
|
|
|
rdpNtlm* ntlm;
|
|
|
|
rdpSettings* settings;
|
|
|
|
freerdp* instance;
|
|
|
|
|
|
|
|
if (!context || !channel)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
tls = channel->tls;
|
|
|
|
ntlm = channel->ntlm;
|
|
|
|
settings = context->settings;
|
|
|
|
instance = context->instance;
|
|
|
|
|
|
|
|
if (!tls || !ntlm || !instance || !settings)
|
|
|
|
return FALSE;
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2014-05-21 19:32:14 +04:00
|
|
|
if (!settings->GatewayPassword || !settings->GatewayUsername ||
|
2018-01-15 14:43:37 +03:00
|
|
|
!strlen(settings->GatewayPassword) || !strlen(settings->GatewayUsername))
|
2013-12-07 07:15:45 +04:00
|
|
|
{
|
|
|
|
if (instance->GatewayAuthenticate)
|
|
|
|
{
|
2014-05-21 19:32:14 +04:00
|
|
|
BOOL proceed = instance->GatewayAuthenticate(instance, &settings->GatewayUsername,
|
2018-01-15 14:43:37 +03:00
|
|
|
&settings->GatewayPassword, &settings->GatewayDomain);
|
2013-12-07 07:15:45 +04:00
|
|
|
|
|
|
|
if (!proceed)
|
2013-12-13 19:11:36 +04:00
|
|
|
{
|
2015-02-11 22:27:29 +03:00
|
|
|
freerdp_set_last_error(context, FREERDP_ERROR_CONNECT_CANCELLED);
|
2018-09-27 16:19:41 +03:00
|
|
|
return TRUE;
|
2013-12-13 19:11:36 +04:00
|
|
|
}
|
2013-12-07 07:15:45 +04:00
|
|
|
|
|
|
|
if (settings->GatewayUseSameCredentials)
|
|
|
|
{
|
2015-06-17 23:08:02 +03:00
|
|
|
if (settings->GatewayUsername)
|
|
|
|
{
|
|
|
|
free(settings->Username);
|
2018-01-15 14:43:37 +03:00
|
|
|
|
2015-06-17 23:08:02 +03:00
|
|
|
if (!(settings->Username = _strdup(settings->GatewayUsername)))
|
2018-09-27 16:19:41 +03:00
|
|
|
return FALSE;
|
2015-06-17 23:08:02 +03:00
|
|
|
}
|
2018-01-15 14:43:37 +03:00
|
|
|
|
2015-06-17 23:08:02 +03:00
|
|
|
if (settings->GatewayDomain)
|
|
|
|
{
|
|
|
|
free(settings->Domain);
|
2018-01-15 14:43:37 +03:00
|
|
|
|
2015-06-17 23:08:02 +03:00
|
|
|
if (!(settings->Domain = _strdup(settings->GatewayDomain)))
|
2018-09-27 16:19:41 +03:00
|
|
|
return FALSE;
|
2015-06-17 23:08:02 +03:00
|
|
|
}
|
2018-01-15 14:43:37 +03:00
|
|
|
|
2015-06-17 23:08:02 +03:00
|
|
|
if (settings->GatewayPassword)
|
|
|
|
{
|
|
|
|
free(settings->Password);
|
2018-01-15 14:43:37 +03:00
|
|
|
|
2015-06-17 23:08:02 +03:00
|
|
|
if (!(settings->Password = _strdup(settings->GatewayPassword)))
|
2018-09-27 16:19:41 +03:00
|
|
|
return FALSE;
|
2015-06-17 23:08:02 +03:00
|
|
|
}
|
2013-12-07 07:15:45 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-13 19:11:36 +04:00
|
|
|
if (!ntlm_client_init(ntlm, TRUE, settings->GatewayUsername,
|
2018-01-15 14:43:37 +03:00
|
|
|
settings->GatewayDomain, settings->GatewayPassword, tls->Bindings))
|
2013-12-13 19:11:36 +04:00
|
|
|
{
|
2018-09-27 16:19:41 +03:00
|
|
|
return TRUE;
|
2013-12-13 19:11:36 +04:00
|
|
|
}
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2013-12-13 19:11:36 +04:00
|
|
|
if (!ntlm_client_make_spn(ntlm, _T("HTTP"), settings->GatewayHostname))
|
|
|
|
{
|
2018-09-27 16:19:41 +03:00
|
|
|
return TRUE;
|
2013-12-13 19:11:36 +04:00
|
|
|
}
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
return TRUE;
|
2012-11-28 07:03:05 +04:00
|
|
|
}
|
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
void rpc_ncacn_http_ntlm_uninit(RpcChannel* channel)
|
2015-02-03 22:44:31 +03:00
|
|
|
{
|
2018-09-27 16:19:41 +03:00
|
|
|
if (!channel)
|
|
|
|
return;
|
|
|
|
|
2015-02-12 20:03:15 +03:00
|
|
|
ntlm_free(channel->ntlm);
|
|
|
|
channel->ntlm = NULL;
|
2015-02-03 22:44:31 +03:00
|
|
|
}
|
|
|
|
|
2018-09-27 17:05:14 +03:00
|
|
|
BOOL rpc_ncacn_http_send_out_channel_request(RpcChannel* outChannel,
|
2018-01-15 14:43:37 +03:00
|
|
|
BOOL replacement)
|
2012-11-28 07:03:05 +04:00
|
|
|
{
|
2018-09-27 16:19:41 +03:00
|
|
|
BOOL rc = TRUE;
|
2012-12-14 05:23:37 +04:00
|
|
|
wStream* s;
|
2015-02-03 22:44:31 +03:00
|
|
|
int contentLength;
|
|
|
|
BOOL continueNeeded;
|
2018-09-27 16:19:41 +03:00
|
|
|
rdpNtlm* ntlm;
|
|
|
|
HttpContext* http;
|
|
|
|
|
|
|
|
if (!outChannel || !outChannel->ntlm || !outChannel->http)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
ntlm = outChannel->ntlm;
|
|
|
|
http = outChannel->http;
|
2015-02-03 22:44:31 +03:00
|
|
|
continueNeeded = ntlm_authenticate(ntlm);
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2015-02-12 20:03:15 +03:00
|
|
|
if (!replacement)
|
|
|
|
contentLength = (continueNeeded) ? 0 : 76;
|
|
|
|
else
|
|
|
|
contentLength = (continueNeeded) ? 0 : 120;
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
s = rpc_ntlm_http_request(http, "RPC_OUT_DATA", contentLength, &ntlm->outputBuffer[0]);
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2015-02-02 04:47:43 +03:00
|
|
|
if (!s)
|
|
|
|
return -1;
|
|
|
|
|
2018-09-27 17:05:14 +03:00
|
|
|
if (rpc_channel_write(outChannel, Stream_Buffer(s), Stream_Length(s)) < 0)
|
2018-09-27 16:19:41 +03:00
|
|
|
rc = FALSE;
|
|
|
|
|
2012-12-14 05:23:37 +04:00
|
|
|
Stream_Free(s, TRUE);
|
2018-09-27 16:19:41 +03:00
|
|
|
return rc;
|
2012-11-28 07:03:05 +04:00
|
|
|
}
|
|
|
|
|
2018-09-27 17:05:14 +03:00
|
|
|
BOOL rpc_ncacn_http_recv_out_channel_response(RpcChannel* outChannel,
|
2018-01-15 14:43:37 +03:00
|
|
|
HttpResponse* response)
|
2012-11-28 07:03:05 +04:00
|
|
|
{
|
2018-09-27 17:05:14 +03:00
|
|
|
const char* token64 = NULL;
|
2015-02-03 22:44:31 +03:00
|
|
|
int ntlmTokenLength = 0;
|
|
|
|
BYTE* ntlmTokenData = NULL;
|
2018-09-27 16:19:41 +03:00
|
|
|
rdpNtlm* ntlm;
|
|
|
|
|
|
|
|
if (!outChannel || !response || !outChannel->ntlm)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
ntlm = outChannel->ntlm;
|
2018-09-27 16:04:41 +03:00
|
|
|
token64 = http_response_get_auth_token(response, "NTLM");
|
2012-11-28 07:03:05 +04:00
|
|
|
|
2018-09-27 16:04:41 +03:00
|
|
|
if (token64)
|
2015-02-03 22:44:31 +03:00
|
|
|
crypto_base64_decode(token64, strlen(token64), &ntlmTokenData, &ntlmTokenLength);
|
|
|
|
|
|
|
|
if (ntlmTokenData && ntlmTokenLength)
|
|
|
|
{
|
|
|
|
ntlm->inputBuffer[0].pvBuffer = ntlmTokenData;
|
|
|
|
ntlm->inputBuffer[0].cbBuffer = ntlmTokenLength;
|
2013-12-07 07:15:45 +04:00
|
|
|
}
|
2014-12-11 19:25:34 +03:00
|
|
|
|
2018-09-27 16:19:41 +03:00
|
|
|
return TRUE;
|
2012-11-28 07:03:05 +04:00
|
|
|
}
|