ua_session.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  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 2018 (c) Fraunhofer IOSB (Author: Julius Pfrommer)
  6. * Copyright 2018 (c) Thomas Stalder, Blue Time Concept SA
  7. */
  8. #include "ua_session.h"
  9. #ifdef UA_ENABLE_SUBSCRIPTIONS
  10. #include "ua_server_internal.h"
  11. #include "ua_subscription.h"
  12. #endif
  13. #define UA_SESSION_NONCELENTH 32
  14. void UA_Session_init(UA_Session *session) {
  15. memset(session, 0, sizeof(UA_Session));
  16. session->availableContinuationPoints = UA_MAXCONTINUATIONPOINTS;
  17. #ifdef UA_ENABLE_SUBSCRIPTIONS
  18. SIMPLEQ_INIT(&session->responseQueue);
  19. #endif
  20. }
  21. void UA_Session_deleteMembersCleanup(UA_Session *session, UA_Server* server) {
  22. UA_Session_detachFromSecureChannel(session);
  23. UA_ApplicationDescription_deleteMembers(&session->clientDescription);
  24. UA_NodeId_deleteMembers(&session->header.authenticationToken);
  25. UA_NodeId_deleteMembers(&session->sessionId);
  26. UA_String_deleteMembers(&session->sessionName);
  27. UA_ByteString_deleteMembers(&session->serverNonce);
  28. struct ContinuationPointEntry *cp, *temp;
  29. LIST_FOREACH_SAFE(cp, &session->continuationPoints, pointers, temp) {
  30. LIST_REMOVE(cp, pointers);
  31. UA_ByteString_deleteMembers(&cp->identifier);
  32. UA_BrowseDescription_deleteMembers(&cp->browseDescription);
  33. UA_free(cp);
  34. }
  35. #ifdef UA_ENABLE_SUBSCRIPTIONS
  36. UA_Subscription *sub, *tempsub;
  37. LIST_FOREACH_SAFE(sub, &session->serverSubscriptions, listEntry, tempsub) {
  38. UA_Session_deleteSubscription(server, session, sub->subscriptionId);
  39. }
  40. UA_PublishResponseEntry *entry;
  41. while((entry = UA_Session_dequeuePublishReq(session))) {
  42. UA_PublishResponse_deleteMembers(&entry->response);
  43. UA_free(entry);
  44. }
  45. #endif
  46. }
  47. void UA_Session_attachToSecureChannel(UA_Session *session, UA_SecureChannel *channel) {
  48. LIST_INSERT_HEAD(&channel->sessions, &session->header, pointers);
  49. session->header.channel = channel;
  50. }
  51. void UA_Session_detachFromSecureChannel(UA_Session *session) {
  52. if(!session->header.channel)
  53. return;
  54. session->header.channel = NULL;
  55. LIST_REMOVE(&session->header, pointers);
  56. }
  57. UA_StatusCode
  58. UA_Session_generateNonce(UA_Session *session) {
  59. UA_SecureChannel *channel = session->header.channel;
  60. if(!channel || !channel->securityPolicy)
  61. return UA_STATUSCODE_BADINTERNALERROR;
  62. /* Is the length of the previous nonce correct? */
  63. if(session->serverNonce.length != UA_SESSION_NONCELENTH) {
  64. UA_ByteString_deleteMembers(&session->serverNonce);
  65. UA_StatusCode retval =
  66. UA_ByteString_allocBuffer(&session->serverNonce, UA_SESSION_NONCELENTH);
  67. if(retval != UA_STATUSCODE_GOOD)
  68. return retval;
  69. }
  70. return channel->securityPolicy->symmetricModule.
  71. generateNonce(channel->securityPolicy, &session->serverNonce);
  72. }
  73. void UA_Session_updateLifetime(UA_Session *session) {
  74. session->validTill = UA_DateTime_nowMonotonic() +
  75. (UA_DateTime)(session->timeout * UA_DATETIME_MSEC);
  76. }
  77. #ifdef UA_ENABLE_SUBSCRIPTIONS
  78. void UA_Session_addSubscription(UA_Session *session, UA_Subscription *newSubscription) {
  79. newSubscription->subscriptionId = ++session->lastSubscriptionId;
  80. LIST_INSERT_HEAD(&session->serverSubscriptions, newSubscription, listEntry);
  81. session->numSubscriptions++;
  82. }
  83. UA_StatusCode
  84. UA_Session_deleteSubscription(UA_Server *server, UA_Session *session,
  85. UA_UInt32 subscriptionId) {
  86. UA_Subscription *sub = UA_Session_getSubscriptionById(session, subscriptionId);
  87. if(!sub)
  88. return UA_STATUSCODE_BADSUBSCRIPTIONIDINVALID;
  89. UA_Subscription_deleteMembers(server, sub);
  90. /* Add a delayed callback to remove the subscription when the currently
  91. * scheduled jobs have completed. There is no actual delayed callback. Just
  92. * free the structure. */
  93. sub->delayedFreePointers.callback = NULL;
  94. UA_WorkQueue_enqueueDelayed(&server->workQueue, &sub->delayedFreePointers);
  95. /* Remove from the session */
  96. LIST_REMOVE(sub, listEntry);
  97. UA_assert(session->numSubscriptions > 0);
  98. session->numSubscriptions--;
  99. return UA_STATUSCODE_GOOD;
  100. }
  101. UA_Subscription *
  102. UA_Session_getSubscriptionById(UA_Session *session, UA_UInt32 subscriptionId) {
  103. UA_Subscription *sub;
  104. LIST_FOREACH(sub, &session->serverSubscriptions, listEntry) {
  105. if(sub->subscriptionId == subscriptionId)
  106. break;
  107. }
  108. return sub;
  109. }
  110. UA_PublishResponseEntry*
  111. UA_Session_dequeuePublishReq(UA_Session *session) {
  112. UA_PublishResponseEntry* entry = SIMPLEQ_FIRST(&session->responseQueue);
  113. if(entry) {
  114. SIMPLEQ_REMOVE_HEAD(&session->responseQueue, listEntry);
  115. session->numPublishReq--;
  116. }
  117. return entry;
  118. }
  119. void
  120. UA_Session_queuePublishReq(UA_Session *session, UA_PublishResponseEntry* entry, UA_Boolean head) {
  121. if(!head)
  122. SIMPLEQ_INSERT_TAIL(&session->responseQueue, entry, listEntry);
  123. else
  124. SIMPLEQ_INSERT_HEAD(&session->responseQueue, entry, listEntry);
  125. session->numPublishReq++;
  126. }
  127. #endif