// Copyright (c) Microsoft. All rights reserved. // Licensed under the MIT license. See LICENSE file in the project root for full license information. #include #include "azure_c_shared_utility/gballoc.h" #include "azure_prov_client/prov_transport_amqp_client.h" #include "azure_prov_client/internal/prov_transport_amqp_common.h" #include "azure_c_shared_utility/xio.h" #include "azure_c_shared_utility/tlsio.h" #include "azure_c_shared_utility/http_proxy_io.h" #include "azure_c_shared_utility/platform.h" #include "azure_c_shared_utility/xlogging.h" #include "azure_c_shared_utility/shared_util_options.h" #include "azure_uamqp_c/saslclientio.h" #define PROV_AMQP_PORT_NUM 5671 static PROV_TRANSPORT_IO_INFO* amqp_transport_io(const char* fqdn, SASL_MECHANISM_HANDLE* sasl_mechanism, const HTTP_PROXY_OPTIONS* proxy_info) { PROV_TRANSPORT_IO_INFO* result; HTTP_PROXY_IO_CONFIG proxy_config; TLSIO_CONFIG tls_io_config; memset(&tls_io_config, 0, sizeof(TLSIO_CONFIG)); tls_io_config.hostname = fqdn; tls_io_config.port = PROV_AMQP_PORT_NUM; if (proxy_info != NULL) { proxy_config.hostname = tls_io_config.hostname; proxy_config.port = PROV_AMQP_PORT_NUM; proxy_config.proxy_hostname = proxy_info->host_address; proxy_config.proxy_port = proxy_info->port; proxy_config.username = proxy_info->username; proxy_config.password = proxy_info->password; tls_io_config.underlying_io_interface = http_proxy_io_get_interface_description(); tls_io_config.underlying_io_parameters = &proxy_config; } const IO_INTERFACE_DESCRIPTION* tlsio_interface = platform_get_default_tlsio(); if (tlsio_interface == NULL) { LogError("platform_get_default_tlsio return NULL IO Interface"); result = NULL; } else if ((result = (PROV_TRANSPORT_IO_INFO*)malloc(sizeof(PROV_TRANSPORT_IO_INFO))) == NULL) { LogError("failure allocating prov_transport info"); result = NULL; } else { memset(result, 0, sizeof(PROV_TRANSPORT_IO_INFO)); result->transport_handle = xio_create(tlsio_interface, &tls_io_config); if (result->transport_handle == NULL) { LogError("failed calling xio_create on underlying io"); free(result); result = NULL; } else { if (sasl_mechanism != NULL) { const IO_INTERFACE_DESCRIPTION* saslio_interface; SASLCLIENTIO_CONFIG sasl_io_config; sasl_io_config.underlying_io = result->transport_handle; sasl_io_config.sasl_mechanism = *sasl_mechanism; saslio_interface = saslclientio_get_interface_description(); if (saslio_interface == NULL) { LogError("failed calling xio_create on underlying io"); xio_destroy(result->transport_handle); free(result); result = NULL; } else { result->sasl_handle = xio_create(saslio_interface, &sasl_io_config); if (result->sasl_handle == NULL) { LogError("failed calling xio_create on sasl client interface"); xio_destroy(result->transport_handle); free(result); result = NULL; } } } } } return result; } PROV_DEVICE_TRANSPORT_HANDLE prov_transport_amqp_create(const char* uri, TRANSPORT_HSM_TYPE type, const char* scope_id, const char* api_version, PROV_TRANSPORT_ERROR_CALLBACK error_cb, void* error_ctx) { return prov_transport_common_amqp_create(uri, type, scope_id, api_version, amqp_transport_io, error_cb, error_ctx); } void prov_transport_amqp_destroy(PROV_DEVICE_TRANSPORT_HANDLE handle) { prov_transport_common_amqp_destroy(handle); } int prov_transport_amqp_open(PROV_DEVICE_TRANSPORT_HANDLE handle, const char* registration_id, BUFFER_HANDLE ek, BUFFER_HANDLE srk, PROV_DEVICE_TRANSPORT_REGISTER_CALLBACK data_callback, void* user_ctx, PROV_DEVICE_TRANSPORT_STATUS_CALLBACK status_cb, void* status_ctx, PROV_TRANSPORT_CHALLENGE_CALLBACK reg_challenge_cb, void* challenge_ctx) { return prov_transport_common_amqp_open(handle, registration_id, ek, srk, data_callback, user_ctx, status_cb, status_ctx, reg_challenge_cb, challenge_ctx); } int prov_transport_amqp_close(PROV_DEVICE_TRANSPORT_HANDLE handle) { return prov_transport_common_amqp_close(handle); } int prov_transport_amqp_register_device(PROV_DEVICE_TRANSPORT_HANDLE handle, PROV_TRANSPORT_JSON_PARSE json_parse_cb, PROV_TRANSPORT_CREATE_JSON_PAYLOAD json_create_cb, void* json_ctx) { return prov_transport_common_amqp_register_device(handle, json_parse_cb, json_create_cb, json_ctx); } int prov_transport_amqp_get_operation_status(PROV_DEVICE_TRANSPORT_HANDLE handle) { return prov_transport_common_amqp_get_operation_status(handle); } void prov_transport_amqp_dowork(PROV_DEVICE_TRANSPORT_HANDLE handle) { prov_transport_common_amqp_dowork(handle); } int prov_transport_amqp_set_trace(PROV_DEVICE_TRANSPORT_HANDLE handle, bool trace_on) { return prov_transport_common_amqp_set_trace(handle, trace_on); } static int prov_transport_amqp_x509_cert(PROV_DEVICE_TRANSPORT_HANDLE handle, const char* certificate, const char* private_key) { return prov_transport_common_amqp_x509_cert(handle, certificate, private_key); } static int prov_transport_amqp_set_trusted_cert(PROV_DEVICE_TRANSPORT_HANDLE handle, const char* certificate) { return prov_transport_common_amqp_set_trusted_cert(handle, certificate); } static int prov_transport_amqp_set_proxy(PROV_DEVICE_TRANSPORT_HANDLE handle, const HTTP_PROXY_OPTIONS* proxy_options) { return prov_transport_common_amqp_set_proxy(handle, proxy_options); } static int prov_transport_amqp_set_option(PROV_DEVICE_TRANSPORT_HANDLE handle, const char* option, const void* value) { // needs to invoke correct behavior return prov_transport_common_amqp_set_option(handle, option, value); } static PROV_DEVICE_TRANSPORT_PROVIDER amqp_transport_func = { prov_transport_amqp_create, prov_transport_amqp_destroy, prov_transport_amqp_open, prov_transport_amqp_close, prov_transport_amqp_register_device, prov_transport_amqp_get_operation_status, prov_transport_amqp_dowork, prov_transport_amqp_set_trace, prov_transport_amqp_x509_cert, prov_transport_amqp_set_trusted_cert, prov_transport_amqp_set_proxy, prov_transport_amqp_set_option }; const PROV_DEVICE_TRANSPORT_PROVIDER* Prov_Device_AMQP_Protocol(void) { return &amqp_transport_func; }