ua_client_highlevel_subscriptions.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. #include "ua_client_highlevel.h"
  2. #include "ua_client_internal.h"
  3. #include "ua_util.h"
  4. #include "ua_types_generated_encoding_binary.h"
  5. const UA_SubscriptionSettings UA_SubscriptionSettings_standard = {
  6. .requestedPublishingInterval = 0.0,
  7. .requestedLifetimeCount = 100,
  8. .requestedMaxKeepAliveCount = 10,
  9. .maxNotificationsPerPublish = 10,
  10. .publishingEnabled = UA_TRUE,
  11. .priority = 0
  12. };
  13. UA_StatusCode UA_Client_Subscriptions_new(UA_Client *client, UA_SubscriptionSettings settings,
  14. UA_UInt32 *newSubscriptionId) {
  15. UA_CreateSubscriptionRequest request;
  16. UA_CreateSubscriptionRequest_init(&request);
  17. request.requestedPublishingInterval = settings.requestedPublishingInterval;
  18. request.requestedLifetimeCount = settings.requestedLifetimeCount;
  19. request.requestedMaxKeepAliveCount = settings.requestedMaxKeepAliveCount;
  20. request.maxNotificationsPerPublish = settings.maxNotificationsPerPublish;
  21. request.publishingEnabled = settings.publishingEnabled;
  22. request.priority = settings.priority;
  23. UA_CreateSubscriptionResponse response = UA_Client_Service_createSubscription(client, request);
  24. UA_StatusCode retval = response.responseHeader.serviceResult;
  25. if(retval == UA_STATUSCODE_GOOD) {
  26. UA_Client_Subscription *newSub = UA_malloc(sizeof(UA_Client_Subscription));
  27. LIST_INIT(&newSub->MonitoredItems);
  28. newSub->LifeTime = response.revisedLifetimeCount;
  29. newSub->KeepAliveCount = response.revisedMaxKeepAliveCount;
  30. newSub->PublishingInterval = response.revisedPublishingInterval;
  31. newSub->SubscriptionID = response.subscriptionId;
  32. newSub->NotificationsPerPublish = request.maxNotificationsPerPublish;
  33. newSub->Priority = request.priority;
  34. if(newSubscriptionId)
  35. *newSubscriptionId = newSub->SubscriptionID;
  36. LIST_INSERT_HEAD(&client->subscriptions, newSub, listEntry);
  37. }
  38. UA_CreateSubscriptionResponse_deleteMembers(&response);
  39. return retval;
  40. }
  41. UA_StatusCode UA_Client_Subscriptions_remove(UA_Client *client, UA_UInt32 subscriptionId) {
  42. UA_Client_Subscription *sub;
  43. UA_StatusCode retval = UA_STATUSCODE_GOOD;
  44. LIST_FOREACH(sub, &client->subscriptions, listEntry) {
  45. if(sub->SubscriptionID == subscriptionId)
  46. break;
  47. }
  48. // Problem? We do not have this subscription registeres. Maybe the server should
  49. // be consulted at this point?
  50. if(!sub)
  51. return UA_STATUSCODE_BADSUBSCRIPTIONIDINVALID;
  52. UA_DeleteSubscriptionsRequest request;
  53. UA_DeleteSubscriptionsRequest_init(&request);
  54. request.subscriptionIdsSize = 1;
  55. request.subscriptionIds = (UA_UInt32 *) UA_malloc(sizeof(UA_UInt32));
  56. *request.subscriptionIds = sub->SubscriptionID;
  57. UA_Client_MonitoredItem *mon, *tmpmon;
  58. LIST_FOREACH_SAFE(mon, &sub->MonitoredItems, listEntry, tmpmon) {
  59. retval |= UA_Client_Subscriptions_removeMonitoredItem(client, sub->SubscriptionID,
  60. mon->MonitoredItemId);
  61. }
  62. if(retval != UA_STATUSCODE_GOOD){
  63. UA_DeleteSubscriptionsRequest_deleteMembers(&request);
  64. return retval;
  65. }
  66. UA_DeleteSubscriptionsResponse response = UA_Client_Service_deleteSubscriptions(client, request);
  67. if(response.resultsSize > 0)
  68. retval = response.results[0];
  69. else
  70. retval = response.responseHeader.serviceResult;
  71. if(retval == UA_STATUSCODE_GOOD) {
  72. LIST_REMOVE(sub, listEntry);
  73. UA_free(sub);
  74. }
  75. UA_DeleteSubscriptionsRequest_deleteMembers(&request);
  76. UA_DeleteSubscriptionsResponse_deleteMembers(&response);
  77. return retval;
  78. }
  79. UA_StatusCode
  80. UA_Client_Subscriptions_addMonitoredItem(UA_Client *client, UA_UInt32 subscriptionId,
  81. UA_NodeId nodeId, UA_UInt32 attributeID,
  82. void *handlingFunction, UA_UInt32 *newMonitoredItemId) {
  83. UA_Client_Subscription *sub;
  84. LIST_FOREACH(sub, &client->subscriptions, listEntry) {
  85. if(sub->SubscriptionID == subscriptionId)
  86. break;
  87. }
  88. if(!sub)
  89. return UA_STATUSCODE_BADSUBSCRIPTIONIDINVALID;
  90. UA_CreateMonitoredItemsRequest request;
  91. UA_CreateMonitoredItemsRequest_init(&request);
  92. request.subscriptionId = subscriptionId;
  93. UA_MonitoredItemCreateRequest item;
  94. UA_MonitoredItemCreateRequest_init(&item);
  95. item.itemToMonitor.nodeId = nodeId;
  96. item.itemToMonitor.attributeId = attributeID;
  97. item.monitoringMode = UA_MONITORINGMODE_REPORTING;
  98. item.requestedParameters.clientHandle = ++(client->monitoredItemHandles);
  99. item.requestedParameters.samplingInterval = sub->PublishingInterval;
  100. item.requestedParameters.discardOldest = UA_TRUE;
  101. item.requestedParameters.queueSize = 1;
  102. request.itemsToCreate = &item;
  103. request.itemsToCreateSize = 1;
  104. // Filter can be left void for now, only changes are supported (UA_Expert does the same with changeItems)
  105. UA_CreateMonitoredItemsResponse response = UA_Client_Service_createMonitoredItems(client, request);
  106. UA_StatusCode retval;
  107. // slight misuse of retval here to check if the deletion was successfull.
  108. if(response.resultsSize == 0)
  109. retval = response.responseHeader.serviceResult;
  110. else
  111. retval = response.results[0].statusCode;
  112. if(retval == UA_STATUSCODE_GOOD) {
  113. UA_Client_MonitoredItem *newMon = UA_malloc(sizeof(UA_Client_MonitoredItem));
  114. newMon->MonitoringMode = UA_MONITORINGMODE_REPORTING;
  115. UA_NodeId_copy(&nodeId, &newMon->monitoredNodeId);
  116. newMon->AttributeID = attributeID;
  117. newMon->ClientHandle = client->monitoredItemHandles;
  118. newMon->SamplingInterval = sub->PublishingInterval;
  119. newMon->QueueSize = 1;
  120. newMon->DiscardOldest = UA_TRUE;
  121. newMon->handler = handlingFunction;
  122. newMon->MonitoredItemId = response.results[0].monitoredItemId;
  123. LIST_INSERT_HEAD(&sub->MonitoredItems, newMon, listEntry);
  124. *newMonitoredItemId = newMon->MonitoredItemId;
  125. }
  126. UA_CreateMonitoredItemsResponse_deleteMembers(&response);
  127. return retval;
  128. }
  129. UA_StatusCode
  130. UA_Client_Subscriptions_removeMonitoredItem(UA_Client *client, UA_UInt32 subscriptionId,
  131. UA_UInt32 monitoredItemId) {
  132. UA_Client_Subscription *sub;
  133. LIST_FOREACH(sub, &client->subscriptions, listEntry) {
  134. if(sub->SubscriptionID == subscriptionId)
  135. break;
  136. }
  137. if(!sub)
  138. return UA_STATUSCODE_BADSUBSCRIPTIONIDINVALID;
  139. UA_Client_MonitoredItem *mon;
  140. LIST_FOREACH(mon, &sub->MonitoredItems, listEntry) {
  141. if(mon->MonitoredItemId == monitoredItemId)
  142. break;
  143. }
  144. if(!mon)
  145. return UA_STATUSCODE_BADMONITOREDITEMIDINVALID;
  146. UA_DeleteMonitoredItemsRequest request;
  147. UA_DeleteMonitoredItemsRequest_init(&request);
  148. request.subscriptionId = sub->SubscriptionID;
  149. request.monitoredItemIdsSize = 1;
  150. request.monitoredItemIds = (UA_UInt32 *) UA_malloc(sizeof(UA_UInt32));
  151. request.monitoredItemIds[0] = mon->MonitoredItemId;
  152. UA_DeleteMonitoredItemsResponse response = UA_Client_Service_deleteMonitoredItems(client, request);
  153. UA_StatusCode retval = UA_STATUSCODE_GOOD;
  154. if(response.resultsSize > 1)
  155. retval = response.results[0];
  156. else
  157. retval = response.responseHeader.serviceResult;
  158. if(retval == UA_STATUSCODE_GOOD) {
  159. LIST_REMOVE(mon, listEntry);
  160. UA_NodeId_deleteMembers(&mon->monitoredNodeId);
  161. UA_free(mon);
  162. }
  163. UA_DeleteMonitoredItemsRequest_deleteMembers(&request);
  164. UA_DeleteMonitoredItemsResponse_deleteMembers(&response);
  165. return retval;
  166. }
  167. static UA_Boolean
  168. UA_Client_processPublishRx(UA_Client *client, UA_PublishResponse response) {
  169. if(response.responseHeader.serviceResult != UA_STATUSCODE_GOOD)
  170. return UA_FALSE;
  171. // Check if the server has acknowledged any of our ACKS
  172. // Note that a list of serverside status codes may be send without valid publish data, i.e.
  173. // during keepalives or no data availability
  174. UA_Client_NotificationsAckNumber *ack, *tmpAck;
  175. size_t i = 0;
  176. LIST_FOREACH_SAFE(ack, &client->pendingNotificationsAcks, listEntry, tmpAck) {
  177. if(response.results[i] == UA_STATUSCODE_GOOD ||
  178. response.results[i] == UA_STATUSCODE_BADSEQUENCENUMBERINVALID) {
  179. LIST_REMOVE(ack, listEntry);
  180. UA_free(ack);
  181. }
  182. i++;
  183. }
  184. if(response.subscriptionId == 0)
  185. return UA_FALSE;
  186. UA_Client_Subscription *sub;
  187. LIST_FOREACH(sub, &client->subscriptions, listEntry) {
  188. if(sub->SubscriptionID == response.subscriptionId)
  189. break;
  190. }
  191. if(!sub)
  192. return UA_FALSE;
  193. UA_NotificationMessage msg = response.notificationMessage;
  194. UA_Client_MonitoredItem *mon;
  195. for(size_t k = 0; k < msg.notificationDataSize; k++) {
  196. if(msg.notificationData[k].encoding != UA_EXTENSIONOBJECT_DECODED)
  197. continue;
  198. if(msg.notificationData[k].content.decoded.type == &UA_TYPES[UA_TYPES_DATACHANGENOTIFICATION]) {
  199. // This is a dataChangeNotification
  200. UA_DataChangeNotification *dataChangeNotification = msg.notificationData[k].content.decoded.data;
  201. for(size_t i = 0; i < dataChangeNotification->monitoredItemsSize; i++) {
  202. UA_MonitoredItemNotification *mitemNot = &dataChangeNotification->monitoredItems[i];
  203. // find this client handle
  204. LIST_FOREACH(mon, &sub->MonitoredItems, listEntry) {
  205. if(mon->ClientHandle == mitemNot->clientHandle) {
  206. mon->handler(mitemNot->clientHandle, &mitemNot->value);
  207. break;
  208. }
  209. }
  210. }
  211. continue;
  212. }
  213. /* if(msg.notificationData[k].typeId.namespaceIndex == 0 && */
  214. /* msg.notificationData[k].typeId.identifier.numeric == 820 ) { */
  215. /* //FIXME: This is a statusChangeNotification (not supported yet) */
  216. /* continue; */
  217. /* } */
  218. /* if(msg.notificationData[k].typeId.namespaceIndex == 0 && */
  219. /* msg.notificationData[k].typeId.identifier.numeric == 916 ) { */
  220. /* //FIXME: This is an EventNotification */
  221. /* continue; */
  222. /* } */
  223. }
  224. /* We processed this message, add it to the list of pending acks (but make
  225. sure it's not in the list first) */
  226. LIST_FOREACH(tmpAck, &client->pendingNotificationsAcks, listEntry) {
  227. if(tmpAck->subAck.sequenceNumber == msg.sequenceNumber &&
  228. tmpAck->subAck.subscriptionId == response.subscriptionId)
  229. break;
  230. }
  231. if(!tmpAck) {
  232. tmpAck = UA_malloc(sizeof(UA_Client_NotificationsAckNumber));
  233. tmpAck->subAck.sequenceNumber = msg.sequenceNumber;
  234. tmpAck->subAck.subscriptionId = sub->SubscriptionID;
  235. LIST_INSERT_HEAD(&client->pendingNotificationsAcks, tmpAck, listEntry);
  236. }
  237. return response.moreNotifications;
  238. }
  239. void UA_Client_Subscriptions_manuallySendPublishRequest(UA_Client *client) {
  240. UA_Boolean moreNotifications = UA_TRUE;
  241. do {
  242. UA_PublishRequest request;
  243. UA_PublishRequest_init(&request);
  244. request.subscriptionAcknowledgementsSize = 0;
  245. UA_Client_NotificationsAckNumber *ack;
  246. LIST_FOREACH(ack, &client->pendingNotificationsAcks, listEntry)
  247. request.subscriptionAcknowledgementsSize++;
  248. request.subscriptionAcknowledgements = UA_malloc(sizeof(UA_SubscriptionAcknowledgement) *
  249. request.subscriptionAcknowledgementsSize);
  250. int index = 0 ;
  251. LIST_FOREACH(ack, &client->pendingNotificationsAcks, listEntry) {
  252. request.subscriptionAcknowledgements[index].sequenceNumber = ack->subAck.sequenceNumber;
  253. request.subscriptionAcknowledgements[index].subscriptionId = ack->subAck.subscriptionId;
  254. index++;
  255. }
  256. UA_PublishResponse response = UA_Client_Service_publish(client, request);
  257. if(response.responseHeader.serviceResult == UA_STATUSCODE_GOOD)
  258. moreNotifications = UA_Client_processPublishRx(client, response);
  259. else
  260. moreNotifications = UA_FALSE;
  261. UA_PublishResponse_deleteMembers(&response);
  262. UA_PublishRequest_deleteMembers(&request);
  263. } while(moreNotifications == UA_TRUE);
  264. }
2025/03/19 13:29:41 [TRACE] Session ID: 4d9cbd54ea4bdb7a 2025/03/19 13:29:41 [TRACE] CSRF Token: wN964sb8QWBRSPZ2F1zwcGGKV7Q6MTc0MjM4NzM3NDAyMzAwNjYyNg== 2025/03/19 13:29:41 [TRACE] Template: repo/home