ua_services_nodemanagement.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. #include "ua_server_internal.h"
  2. #include "ua_services.h"
  3. #include "ua_namespace_0.h"
  4. #include "ua_statuscodes.h"
  5. #include "ua_nodestore.h"
  6. #include "ua_services_internal.h"
  7. #include "ua_session.h"
  8. #include "ua_util.h"
  9. #define COPY_STANDARDATTRIBUTES do { \
  10. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_DISPLAYNAME) { \
  11. vnode->displayName = attr.displayName; \
  12. UA_LocalizedText_init(&attr.displayName); \
  13. } \
  14. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_DESCRIPTION) { \
  15. vnode->description = attr.description; \
  16. UA_LocalizedText_init(&attr.description); \
  17. } \
  18. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_WRITEMASK) \
  19. vnode->writeMask = attr.writeMask; \
  20. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_USERWRITEMASK) \
  21. vnode->userWriteMask = attr.userWriteMask; \
  22. } while(0)
  23. static UA_StatusCode parseVariableNode(UA_ExtensionObject *attributes, UA_Node **new_node,
  24. const UA_TypeVTable **vt) {
  25. if(attributes->typeId.identifier.numeric !=
  26. UA_NODEIDS[UA_VARIABLEATTRIBUTES].identifier.numeric + UA_ENCODINGOFFSET_BINARY)
  27. return UA_STATUSCODE_BADNODEATTRIBUTESINVALID;
  28. UA_VariableAttributes attr;
  29. UA_UInt32 pos = 0;
  30. // todo return more informative error codes from decodeBinary
  31. if(UA_VariableAttributes_decodeBinary(&attributes->body, &pos, &attr) != UA_STATUSCODE_GOOD)
  32. return UA_STATUSCODE_BADNODEATTRIBUTESINVALID;
  33. UA_VariableNode *vnode = UA_VariableNode_new();
  34. if(!vnode) {
  35. UA_VariableAttributes_deleteMembers(&attr);
  36. return UA_STATUSCODE_BADOUTOFMEMORY;
  37. }
  38. // now copy all the attributes. This potentially removes them from the decoded attributes.
  39. COPY_STANDARDATTRIBUTES;
  40. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_ACCESSLEVEL)
  41. vnode->accessLevel = attr.accessLevel;
  42. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_USERACCESSLEVEL)
  43. vnode->userAccessLevel = attr.userAccessLevel;
  44. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_HISTORIZING)
  45. vnode->historizing = attr.historizing;
  46. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_MINIMUMSAMPLINGINTERVAL)
  47. vnode->minimumSamplingInterval = attr.minimumSamplingInterval;
  48. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_VALUERANK)
  49. vnode->valueRank = attr.valueRank;
  50. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_ARRAYDIMENSIONS) {
  51. vnode->arrayDimensionsSize = attr.arrayDimensionsSize;
  52. vnode->arrayDimensions = attr.arrayDimensions;
  53. attr.arrayDimensionsSize = -1;
  54. attr.arrayDimensions = UA_NULL;
  55. }
  56. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_DATATYPE ||
  57. attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_OBJECTTYPEORDATATYPE) {
  58. vnode->dataType = attr.dataType;
  59. UA_NodeId_init(&attr.dataType);
  60. }
  61. if(attr.specifiedAttributes & UA_NODEATTRIBUTESMASK_VALUE) {
  62. vnode->value = attr.value;
  63. UA_Variant_init(&attr.value);
  64. }
  65. UA_VariableAttributes_deleteMembers(&attr);
  66. *new_node = (UA_Node*)vnode;
  67. *vt = &UA_TYPES[UA_VARIABLENODE];
  68. return UA_STATUSCODE_GOOD;
  69. }
  70. static void addNodeFromAttributes(UA_Server *server, UA_Session *session, UA_AddNodesItem *item,
  71. UA_AddNodesResult *result) {
  72. // adding nodes to ns0 is not allowed over the wire
  73. if(item->requestedNewNodeId.nodeId.namespaceIndex == 0) {
  74. result->statusCode = UA_STATUSCODE_BADNODEIDREJECTED;
  75. return;
  76. }
  77. // parse the node
  78. UA_Node *node;
  79. const UA_TypeVTable *nodeVT = UA_NULL;
  80. if(item->nodeClass == UA_NODECLASS_VARIABLE)
  81. result->statusCode = parseVariableNode(&item->nodeAttributes, &node, &nodeVT);
  82. else // add more node types here..
  83. result->statusCode = UA_STATUSCODE_BADNOTIMPLEMENTED;
  84. if(result->statusCode != UA_STATUSCODE_GOOD)
  85. return;
  86. // add the node
  87. *result = UA_Server_addNodeWithSession(server, session, (const UA_Node **)&node,
  88. &item->parentNodeId, &item->referenceTypeId);
  89. if(result->statusCode != UA_STATUSCODE_GOOD)
  90. nodeVT->delete(node);
  91. }
  92. void Service_AddNodes(UA_Server *server, UA_Session *session, const UA_AddNodesRequest *request,
  93. UA_AddNodesResponse *response) {
  94. if(request->nodesToAddSize <= 0) {
  95. response->responseHeader.serviceResult = UA_STATUSCODE_BADNOTHINGTODO;
  96. return;
  97. }
  98. UA_StatusCode retval = UA_Array_new((void**)&response->results, request->nodesToAddSize,
  99. &UA_TYPES[UA_ADDNODESRESULT]);
  100. if(retval) {
  101. response->responseHeader.serviceResult = retval;
  102. return;
  103. }
  104. /* ### Begin External Namespaces */
  105. UA_Boolean *isExternal = UA_alloca(sizeof(UA_Boolean) * request->nodesToAddSize);
  106. memset(isExternal, UA_FALSE, sizeof(UA_Boolean)*request->nodesToAddSize);
  107. UA_UInt32 *indices = UA_alloca(sizeof(UA_UInt32) * request->nodesToAddSize);
  108. for(UA_Int32 j = 0;j<server->externalNamespacesSize;j++) {
  109. UA_UInt32 indexSize = 0;
  110. for(UA_Int32 i = 0;i < request->nodesToAddSize;i++) {
  111. if(request->nodesToAdd[i].requestedNewNodeId.nodeId.namespaceIndex != server->externalNamespaces[j].index)
  112. continue;
  113. isExternal[i] = UA_TRUE;
  114. indices[indexSize] = i;
  115. indexSize++;
  116. }
  117. if(indexSize == 0)
  118. continue;
  119. UA_ExternalNodeStore *ens = &server->externalNamespaces[j].externalNodeStore;
  120. ens->addNodes(ens->ensHandle, &request->requestHeader, request->nodesToAdd,
  121. indices, indexSize, response->results, response->diagnosticInfos);
  122. }
  123. /* ### End External Namespaces */
  124. response->resultsSize = request->nodesToAddSize;
  125. for(int i = 0;i < request->nodesToAddSize;i++) {
  126. if(!isExternal[i])
  127. addNodeFromAttributes(server, session, &request->nodesToAdd[i], &response->results[i]);
  128. }
  129. }
  130. void Service_AddReferences(UA_Server *server, UA_Session *session, const UA_AddReferencesRequest *request,
  131. UA_AddReferencesResponse *response) {
  132. if(request->referencesToAddSize <= 0) {
  133. response->responseHeader.serviceResult = UA_STATUSCODE_BADNOTHINGTODO;
  134. return;
  135. }
  136. response->results = UA_alloc(sizeof(UA_StatusCode)*request->referencesToAddSize);
  137. if(!response->results) {
  138. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  139. return;
  140. }
  141. response->resultsSize = request->referencesToAddSize;
  142. for(UA_Int32 i = 0;i < response->resultsSize;i++)
  143. response->results[i] = UA_Server_addReferenceWithSession(server, session,
  144. &request->referencesToAdd[i]);
  145. }
  146. void Service_DeleteNodes(UA_Server *server, UA_Session *session, const UA_DeleteNodesRequest *request,
  147. UA_DeleteNodesResponse *response) {
  148. }
  149. void Service_DeleteReferences(UA_Server *server, UA_Session *session,
  150. const UA_DeleteReferencesRequest *request,
  151. UA_DeleteReferencesResponse *response) {
  152. }