ua_securechannel_manager.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. #include "ua_securechannel_manager.h"
  2. #include "ua_session.h"
  3. #include "ua_statuscodes.h"
  4. UA_StatusCode UA_SecureChannelManager_init(UA_SecureChannelManager *cm, UA_UInt32 maxChannelCount,
  5. UA_UInt32 tokenLifetime, UA_UInt32 startChannelId,
  6. UA_UInt32 startTokenId) {
  7. LIST_INIT(&cm->channels);
  8. cm->lastChannelId = startChannelId;
  9. cm->lastTokenId = startTokenId;
  10. cm->maxChannelLifetime = tokenLifetime;
  11. cm->maxChannelCount = maxChannelCount;
  12. return UA_STATUSCODE_GOOD;
  13. }
  14. void UA_SecureChannelManager_deleteMembers(UA_SecureChannelManager *cm) {
  15. channel_list_entry *current = LIST_FIRST(&cm->channels);
  16. while(current) {
  17. LIST_REMOVE(current, pointers);
  18. UA_SecureChannel_deleteMembersCleanup(&current->channel);
  19. UA_free(current);
  20. current = LIST_FIRST(&cm->channels);
  21. }
  22. }
  23. void UA_SecureChannelManager_cleanupTimedOut(UA_SecureChannelManager *cm, UA_DateTime now) {
  24. channel_list_entry *entry = LIST_FIRST(&cm->channels);
  25. /* remove channels that were not renewed or who have no connection attached */
  26. while(entry) {
  27. if(entry->channel.securityToken.createdAt +
  28. (10000 * entry->channel.securityToken.revisedLifetime) > now &&
  29. entry->channel.connection) {
  30. entry = LIST_NEXT(entry, pointers);
  31. }
  32. else {
  33. channel_list_entry *next = LIST_NEXT(entry, pointers);
  34. LIST_REMOVE(entry, pointers);
  35. UA_SecureChannel_deleteMembersCleanup(&entry->channel);
  36. UA_free(entry);
  37. entry = next;
  38. }
  39. }
  40. }
  41. UA_StatusCode UA_SecureChannelManager_open(UA_SecureChannelManager *cm, UA_Connection *conn,
  42. const UA_OpenSecureChannelRequest *request,
  43. UA_OpenSecureChannelResponse *response) {
  44. switch(request->securityMode) {
  45. case UA_MESSAGESECURITYMODE_NONE:
  46. break;
  47. case UA_MESSAGESECURITYMODE_INVALID:
  48. case UA_MESSAGESECURITYMODE_SIGN:
  49. case UA_MESSAGESECURITYMODE_SIGNANDENCRYPT:
  50. default:
  51. response->responseHeader.serviceResult = UA_STATUSCODE_BADSECURITYMODEREJECTED;
  52. return response->responseHeader.serviceResult;
  53. }
  54. channel_list_entry *entry = UA_malloc(sizeof(channel_list_entry));
  55. if(!entry)
  56. return UA_STATUSCODE_BADOUTOFMEMORY;
  57. UA_SecureChannel_init(&entry->channel);
  58. response->responseHeader.stringTableSize = 0;
  59. response->responseHeader.timestamp = UA_DateTime_now();
  60. response->serverProtocolVersion = 0;
  61. entry->channel.securityToken.channelId = cm->lastChannelId++;
  62. entry->channel.securityToken.tokenId = cm->lastTokenId++;
  63. entry->channel.securityToken.createdAt = UA_DateTime_now();
  64. entry->channel.securityToken.revisedLifetime = (request->requestedLifetime > cm->maxChannelLifetime) ?
  65. cm->maxChannelLifetime : request->requestedLifetime;
  66. //FIXME: pragmatic workaround to get clients requesting lifetime of 0 working
  67. if(entry->channel.securityToken.revisedLifetime == 0) {
  68. entry->channel.securityToken.revisedLifetime = cm->maxChannelLifetime;
  69. //FIXME: I'd log it, but there is no pointer to the logger
  70. // printf("Warning: client requests token lifetime of 0 in OpenSecureChannelRequest setting it to %llu\n", cm->maxChannelLifetime);
  71. }
  72. UA_ByteString_copy(&request->clientNonce, &entry->channel.clientNonce);
  73. entry->channel.serverAsymAlgSettings.securityPolicyUri =
  74. UA_STRING_ALLOC("http://opcfoundation.org/UA/SecurityPolicy#None");
  75. UA_SecureChannel_generateNonce(&entry->channel.serverNonce);
  76. UA_ByteString_copy(&entry->channel.serverNonce, &response->serverNonce);
  77. UA_ChannelSecurityToken_copy(&entry->channel.securityToken, &response->securityToken);
  78. UA_Connection_attachSecureChannel(conn, &entry->channel);
  79. LIST_INSERT_HEAD(&cm->channels, entry, pointers);
  80. return UA_STATUSCODE_GOOD;
  81. }
  82. UA_StatusCode UA_SecureChannelManager_renew(UA_SecureChannelManager *cm, UA_Connection *conn,
  83. const UA_OpenSecureChannelRequest *request,
  84. UA_OpenSecureChannelResponse *response)
  85. {
  86. UA_SecureChannel *channel = conn->channel;
  87. if(channel == UA_NULL)
  88. return UA_STATUSCODE_BADINTERNALERROR;
  89. channel->securityToken.tokenId = cm->lastTokenId++;
  90. channel->securityToken.createdAt = UA_DateTime_now(); // todo: is wanted?
  91. channel->securityToken.revisedLifetime = (request->requestedLifetime > cm->maxChannelLifetime) ?
  92. cm->maxChannelLifetime : request->requestedLifetime;
  93. //FIXME: pragmatic workaround to get clients requesting lifetime of 0 working
  94. if(channel->securityToken.revisedLifetime == 0){
  95. channel->securityToken.revisedLifetime = cm->maxChannelLifetime;
  96. //FIXME: I'd log it, but there is no pointer to the logger
  97. // printf("Warning: client requests token lifetime of 0 in renewing SecureChannel setting it to %llu\n", cm->maxChannelLifetime);
  98. }
  99. if(channel->serverNonce.data != UA_NULL)
  100. UA_ByteString_deleteMembers(&channel->serverNonce);
  101. UA_SecureChannel_generateNonce(&channel->serverNonce);
  102. UA_ByteString_copy(&channel->serverNonce, &response->serverNonce);
  103. UA_ChannelSecurityToken_copy(&channel->securityToken, &response->securityToken);
  104. return UA_STATUSCODE_GOOD;
  105. }
  106. UA_SecureChannel * UA_SecureChannelManager_get(UA_SecureChannelManager *cm, UA_UInt32 channelId) {
  107. channel_list_entry *entry;
  108. LIST_FOREACH(entry, &cm->channels, pointers) {
  109. if(entry->channel.securityToken.channelId == channelId)
  110. return &entry->channel;
  111. }
  112. return UA_NULL;
  113. }
  114. UA_StatusCode UA_SecureChannelManager_close(UA_SecureChannelManager *cm, UA_UInt32 channelId) {
  115. channel_list_entry *entry;
  116. LIST_FOREACH(entry, &cm->channels, pointers) {
  117. if(entry->channel.securityToken.channelId == channelId)
  118. break;
  119. }
  120. if(!entry)
  121. return UA_STATUSCODE_BADINTERNALERROR;
  122. LIST_REMOVE(entry, pointers);
  123. UA_SecureChannel_deleteMembersCleanup(&entry->channel);
  124. UA_free(entry);
  125. return UA_STATUSCODE_GOOD;
  126. }