ua_server_discovery.c 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. /* This Source Code Form is subject to the terms of the Mozilla Public
  2. * License, v. 2.0. If a copy of the MPL was not distributed with this
  3. * file, You can obtain one at http://mozilla.org/MPL/2.0/.
  4. *
  5. * Copyright 2017 (c) Fraunhofer IOSB (Author: Julius Pfrommer)
  6. * Copyright 2017 (c) Stefan Profanter, fortiss GmbH
  7. */
  8. #include "ua_server_internal.h"
  9. #include "ua_client.h"
  10. #ifdef UA_ENABLE_DISCOVERY
  11. static UA_StatusCode
  12. register_server_with_discovery_server(UA_Server *server,
  13. const char* discoveryServerUrl,
  14. const UA_Boolean isUnregister,
  15. const char* semaphoreFilePath) {
  16. if(!discoveryServerUrl) {
  17. UA_LOG_ERROR(server->config.logger, UA_LOGCATEGORY_SERVER,
  18. "No discovery server url provided");
  19. return UA_STATUSCODE_BADINTERNALERROR;
  20. }
  21. /* Create the client */
  22. UA_ClientConfig clientConfig = UA_Server_getClientConfig();
  23. UA_Client *client = UA_Client_new(clientConfig);
  24. if(!client)
  25. return UA_STATUSCODE_BADOUTOFMEMORY;
  26. /* Connect the client */
  27. UA_StatusCode retval = UA_Client_connect(client, discoveryServerUrl);
  28. if(retval != UA_STATUSCODE_GOOD) {
  29. UA_LOG_ERROR(server->config.logger, UA_LOGCATEGORY_CLIENT,
  30. "Connecting to the discovery server failed with statuscode %s",
  31. UA_StatusCode_name(retval));
  32. UA_Client_disconnect(client);
  33. UA_Client_delete(client);
  34. return retval;
  35. }
  36. /* Prepare the request. Do not cleanup the request after the service call,
  37. * as the members are stack-allocated or point into the server config. */
  38. UA_RegisterServer2Request request;
  39. UA_RegisterServer2Request_init(&request);
  40. request.requestHeader.timestamp = UA_DateTime_now();
  41. request.requestHeader.timeoutHint = 10000;
  42. request.server.isOnline = !isUnregister;
  43. request.server.serverUri = server->config.applicationDescription.applicationUri;
  44. request.server.productUri = server->config.applicationDescription.productUri;
  45. request.server.serverType = server->config.applicationDescription.applicationType;
  46. request.server.gatewayServerUri = server->config.applicationDescription.gatewayServerUri;
  47. if(semaphoreFilePath) {
  48. #ifdef UA_ENABLE_DISCOVERY_SEMAPHORE
  49. request.server.semaphoreFilePath =
  50. UA_STRING((char*)(uintptr_t)semaphoreFilePath); /* dirty cast */
  51. #else
  52. UA_LOG_WARNING(server->config.logger, UA_LOGCATEGORY_CLIENT,
  53. "Ignoring semaphore file path. open62541 not compiled "
  54. "with UA_ENABLE_DISCOVERY_SEMAPHORE=ON");
  55. #endif
  56. }
  57. request.server.serverNames = &server->config.applicationDescription.applicationName;
  58. request.server.serverNamesSize = 1;
  59. /* Copy the discovery urls from the server config and the network layers*/
  60. size_t config_discurls = server->config.applicationDescription.discoveryUrlsSize;
  61. size_t nl_discurls = server->config.networkLayersSize;
  62. size_t total_discurls = config_discurls + nl_discurls;
  63. UA_STACKARRAY(UA_String, urlsBuf, total_discurls);
  64. request.server.discoveryUrls = urlsBuf;
  65. request.server.discoveryUrlsSize = total_discurls;
  66. for(size_t i = 0; i < config_discurls; ++i)
  67. request.server.discoveryUrls[i] = server->config.applicationDescription.discoveryUrls[i];
  68. /* TODO: Add nl only if discoveryUrl not already present */
  69. for(size_t i = 0; i < nl_discurls; ++i) {
  70. UA_ServerNetworkLayer *nl = &server->config.networkLayers[i];
  71. request.server.discoveryUrls[config_discurls + i] = nl->discoveryUrl;
  72. }
  73. UA_MdnsDiscoveryConfiguration mdnsConfig;
  74. UA_MdnsDiscoveryConfiguration_init(&mdnsConfig);
  75. request.discoveryConfigurationSize = 1;
  76. request.discoveryConfiguration = UA_ExtensionObject_new();
  77. UA_ExtensionObject_init(&request.discoveryConfiguration[0]);
  78. request.discoveryConfiguration[0].encoding = UA_EXTENSIONOBJECT_DECODED_NODELETE;
  79. request.discoveryConfiguration[0].content.decoded.type = &UA_TYPES[UA_TYPES_MDNSDISCOVERYCONFIGURATION];
  80. request.discoveryConfiguration[0].content.decoded.data = &mdnsConfig;
  81. mdnsConfig.mdnsServerName = server->config.mdnsServerName;
  82. mdnsConfig.serverCapabilities = server->config.serverCapabilities;
  83. mdnsConfig.serverCapabilitiesSize = server->config.serverCapabilitiesSize;
  84. // First try with RegisterServer2, if that isn't implemented, use RegisterServer
  85. UA_RegisterServer2Response response;
  86. __UA_Client_Service(client, &request, &UA_TYPES[UA_TYPES_REGISTERSERVER2REQUEST],
  87. &response, &UA_TYPES[UA_TYPES_REGISTERSERVER2RESPONSE]);
  88. UA_StatusCode serviceResult = response.responseHeader.serviceResult;
  89. UA_RegisterServer2Response_deleteMembers(&response);
  90. UA_ExtensionObject_delete(request.discoveryConfiguration);
  91. if(serviceResult == UA_STATUSCODE_BADNOTIMPLEMENTED ||
  92. serviceResult == UA_STATUSCODE_BADSERVICEUNSUPPORTED) {
  93. /* Try RegisterServer */
  94. UA_RegisterServerRequest request_fallback;
  95. UA_RegisterServerRequest_init(&request_fallback);
  96. /* Copy from RegisterServer2 request */
  97. request_fallback.requestHeader = request.requestHeader;
  98. request_fallback.server = request.server;
  99. UA_RegisterServerResponse response_fallback;
  100. __UA_Client_Service(client, &request_fallback,
  101. &UA_TYPES[UA_TYPES_REGISTERSERVERREQUEST],
  102. &response_fallback,
  103. &UA_TYPES[UA_TYPES_REGISTERSERVERRESPONSE]);
  104. serviceResult = response_fallback.responseHeader.serviceResult;
  105. UA_RegisterServerResponse_deleteMembers(&response_fallback);
  106. }
  107. if(serviceResult != UA_STATUSCODE_GOOD) {
  108. UA_LOG_ERROR(server->config.logger, UA_LOGCATEGORY_CLIENT,
  109. "RegisterServer/RegisterServer2 failed with statuscode %s",
  110. UA_StatusCode_name(serviceResult));
  111. }
  112. UA_Client_disconnect(client);
  113. UA_Client_delete(client);
  114. return serviceResult;
  115. }
  116. UA_StatusCode
  117. UA_Server_register_discovery(UA_Server *server, const char* discoveryServerUrl,
  118. const char* semaphoreFilePath) {
  119. return register_server_with_discovery_server(server, discoveryServerUrl,
  120. UA_FALSE, semaphoreFilePath);
  121. }
  122. UA_StatusCode
  123. UA_Server_unregister_discovery(UA_Server *server, const char* discoveryServerUrl) {
  124. return register_server_with_discovery_server(server, discoveryServerUrl,
  125. UA_TRUE, NULL);
  126. }
  127. #endif /* UA_ENABLE_DISCOVERY */