ua_services_discovery.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. #include "ua_server_internal.h"
  2. #include "ua_services.h"
  3. #include "ua_util.h"
  4. void Service_FindServers(UA_Server *server, UA_Session *session,
  5. const UA_FindServersRequest *request, UA_FindServersResponse *response) {
  6. UA_LOG_DEBUG_SESSION(server->config.logger, session, "Processing FindServersRequest");
  7. /* copy ApplicationDescription from the config */
  8. UA_ApplicationDescription *descr = UA_malloc(sizeof(UA_ApplicationDescription));
  9. if(!descr) {
  10. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  11. return;
  12. }
  13. response->responseHeader.serviceResult =
  14. UA_ApplicationDescription_copy(&server->config.applicationDescription, descr);
  15. if(response->responseHeader.serviceResult != UA_STATUSCODE_GOOD) {
  16. UA_free(descr);
  17. return;
  18. }
  19. /* add the discoveryUrls from the networklayers */
  20. UA_String *disc = UA_realloc(descr->discoveryUrls, sizeof(UA_String) *
  21. (descr->discoveryUrlsSize + server->config.networkLayersSize));
  22. if(!disc) {
  23. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  24. UA_ApplicationDescription_delete(descr);
  25. return;
  26. }
  27. size_t existing = descr->discoveryUrlsSize;
  28. descr->discoveryUrls = disc;
  29. descr->discoveryUrlsSize += server->config.networkLayersSize;
  30. // TODO: Add nl only if discoveryUrl not already present
  31. for(size_t i = 0; i < server->config.networkLayersSize; i++) {
  32. UA_ServerNetworkLayer *nl = &server->config.networkLayers[i];
  33. UA_String_copy(&nl->discoveryUrl, &descr->discoveryUrls[existing + i]);
  34. }
  35. response->servers = descr;
  36. response->serversSize = 1;
  37. }
  38. void Service_GetEndpoints(UA_Server *server, UA_Session *session, const UA_GetEndpointsRequest *request,
  39. UA_GetEndpointsResponse *response) {
  40. /* If the client expects to see a specific endpointurl, mirror it back. If
  41. not, clone the endpoints with the discovery url of all networklayers. */
  42. const UA_String *endpointUrl = &request->endpointUrl;
  43. if(endpointUrl->length > 0) {
  44. UA_LOG_DEBUG_SESSION(server->config.logger, session, "Processing GetEndpointsRequest with endpointUrl " \
  45. UA_PRINTF_STRING_FORMAT, UA_PRINTF_STRING_DATA(*endpointUrl));
  46. } else {
  47. UA_LOG_DEBUG_SESSION(server->config.logger, session, "Processing GetEndpointsRequest with an empty endpointUrl");
  48. }
  49. /* test if the supported binary profile shall be returned */
  50. #ifdef NO_ALLOCA
  51. UA_Boolean relevant_endpoints[server->endpointDescriptionsSize];
  52. #else
  53. UA_Boolean *relevant_endpoints = UA_alloca(sizeof(UA_Boolean) * server->endpointDescriptionsSize);
  54. #endif
  55. memset(relevant_endpoints, 0, sizeof(UA_Boolean) * server->endpointDescriptionsSize);
  56. size_t relevant_count = 0;
  57. if(request->profileUrisSize == 0) {
  58. for(size_t j = 0; j < server->endpointDescriptionsSize; j++)
  59. relevant_endpoints[j] = true;
  60. relevant_count = server->endpointDescriptionsSize;
  61. } else {
  62. for(size_t j = 0; j < server->endpointDescriptionsSize; j++) {
  63. for(size_t i = 0; i < request->profileUrisSize; i++) {
  64. if(!UA_String_equal(&request->profileUris[i], &server->endpointDescriptions[j].transportProfileUri))
  65. continue;
  66. relevant_endpoints[j] = true;
  67. relevant_count++;
  68. break;
  69. }
  70. }
  71. }
  72. if(relevant_count == 0) {
  73. response->endpointsSize = 0;
  74. return;
  75. }
  76. /* Clone the endpoint for each networklayer? */
  77. size_t clone_times = 1;
  78. UA_Boolean nl_endpointurl = false;
  79. if(endpointUrl->length == 0) {
  80. clone_times = server->config.networkLayersSize;
  81. nl_endpointurl = true;
  82. }
  83. response->endpoints = UA_Array_new(relevant_count * clone_times, &UA_TYPES[UA_TYPES_ENDPOINTDESCRIPTION]);
  84. if(!response->endpoints) {
  85. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  86. return;
  87. }
  88. response->endpointsSize = relevant_count * clone_times;
  89. size_t k = 0;
  90. UA_StatusCode retval = UA_STATUSCODE_GOOD;
  91. for(size_t i = 0; i < clone_times; i++) {
  92. if(nl_endpointurl)
  93. endpointUrl = &server->config.networkLayers[i].discoveryUrl;
  94. for(size_t j = 0; j < server->endpointDescriptionsSize; j++) {
  95. if(!relevant_endpoints[j])
  96. continue;
  97. retval |= UA_EndpointDescription_copy(&server->endpointDescriptions[j], &response->endpoints[k]);
  98. retval |= UA_String_copy(endpointUrl, &response->endpoints[k].endpointUrl);
  99. k++;
  100. }
  101. }
  102. if(retval != UA_STATUSCODE_GOOD) {
  103. response->responseHeader.serviceResult = retval;
  104. UA_Array_delete(response->endpoints, response->endpointsSize, &UA_TYPES[UA_TYPES_ENDPOINTDESCRIPTION]);
  105. response->endpoints = NULL;
  106. response->endpointsSize = 0;
  107. return;
  108. }
  109. }