check_services_attributes.c 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <time.h>
  4. #include "check.h"
  5. #include "server/ua_nodestore.h"
  6. #include "server/ua_services.h"
  7. #include "ua_client.h"
  8. #include "ua_nodeids.h"
  9. #include "ua_statuscodes.h"
  10. #include "ua_types.h"
  11. #include "ua_util.h"
  12. #include "server/ua_server_internal.h"
  13. //#include "server/ua_services_attribute.c"
  14. #ifdef UA_MULTITHREADING
  15. #include <pthread.h>
  16. #include <urcu.h>
  17. #endif
  18. static void copyNames(UA_Node *node, char *name) {
  19. node->browseName = UA_QUALIFIEDNAME_ALLOC(0, name);
  20. node->displayName = UA_LOCALIZEDTEXT_ALLOC("", name);
  21. node->description = UA_LOCALIZEDTEXT_ALLOC("", name);
  22. }
  23. static UA_Server* makeTestSequence(void) {
  24. UA_Server *server = UA_Server_new(UA_ServerConfig_standard);
  25. /* VariableNode */
  26. UA_Variant *myIntegerVariant = UA_Variant_new();
  27. UA_Int32 myInteger = 42;
  28. UA_Variant_setScalarCopy(myIntegerVariant, &myInteger,
  29. &UA_TYPES[UA_TYPES_INT32]);
  30. const UA_QualifiedName myIntegerName = UA_QUALIFIEDNAME(1, "the answer");
  31. const UA_NodeId myIntegerNodeId = UA_NODEID_STRING(1, "the.answer");
  32. UA_NodeId parentNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER);
  33. UA_NodeId parentReferenceNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES);
  34. UA_Server_addVariableNode(server, myIntegerNodeId, myIntegerName, UA_LOCALIZEDTEXT("en_US","the answer"), UA_LOCALIZEDTEXT("en_US","the answer"), 0, 0,
  35. parentNodeId, parentReferenceNodeId, myIntegerVariant, NULL);
  36. /* ObjectNode */
  37. UA_Server_addObjectNode(server, UA_NODEID_NUMERIC(1, 50), UA_QUALIFIEDNAME(1, "Demo"), UA_LOCALIZEDTEXT("en_US","Demo"),
  38. UA_LOCALIZEDTEXT("en_US","Demo"), 0, 0, UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER),
  39. UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES), UA_EXPANDEDNODEID_NUMERIC(0, UA_NS0ID_FOLDERTYPE), NULL);
  40. /* ReferenceTypeNode */
  41. UA_ReferenceTypeNode *organizes = UA_ReferenceTypeNode_new();
  42. copyNames((UA_Node*)organizes, "Organizes");
  43. organizes->inverseName = UA_LOCALIZEDTEXT_ALLOC("", "OrganizedBy");
  44. organizes->nodeId.identifier.numeric = UA_NS0ID_ORGANIZES;
  45. organizes->isAbstract = UA_FALSE;
  46. organizes->symmetric = UA_FALSE;
  47. UA_Server_addNode(server, (UA_Node*)organizes, UA_EXPANDEDNODEID_NUMERIC(0, UA_NS0ID_HIERARCHICALREFERENCES), UA_NODEID_NUMERIC(0, UA_NS0ID_HASSUBTYPE));
  48. /* ViewNode */
  49. UA_ViewNode *viewtest = UA_ViewNode_new();
  50. copyNames((UA_Node*)viewtest, "Viewtest");
  51. viewtest->nodeId.identifier.numeric = UA_NS0ID_VIEWNODE;
  52. UA_Server_addNode(server, (UA_Node*)viewtest,
  53. UA_EXPANDEDNODEID_NUMERIC(0, UA_NS0ID_VIEWSFOLDER),
  54. UA_NODEID_NUMERIC(0, UA_NS0ID_HASSUBTYPE));
  55. /* MethodNode */
  56. UA_MethodNode *methodtest = UA_MethodNode_new();
  57. copyNames((UA_Node*)methodtest, "Methodtest");
  58. methodtest->nodeId.identifier.numeric = UA_NS0ID_METHODNODE;
  59. UA_Server_addNode(server, (UA_Node*)methodtest,
  60. UA_EXPANDEDNODEID_NUMERIC(0, 3),
  61. UA_NODEID_NUMERIC(0, UA_NS0ID_HASSUBTYPE));
  62. return server;
  63. }
  64. static UA_VariableNode* makeCompareSequence(void) {
  65. UA_VariableNode *node = UA_VariableNode_new();
  66. UA_Variant *myIntegerVariant = UA_Variant_new();
  67. UA_Int32 myInteger = 42;
  68. UA_Variant_setScalarCopy(myIntegerVariant, &myInteger,
  69. &UA_TYPES[UA_TYPES_INT32]);
  70. const UA_QualifiedName myIntegerName = UA_QUALIFIEDNAME(1, "the answer");
  71. const UA_LocalizedText myIntegerDisplName = UA_LOCALIZEDTEXT("en_US", "the answer");
  72. const UA_NodeId myIntegerNodeId = UA_NODEID_STRING(1, "the.answer");
  73. UA_NodeId parentNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER);
  74. //UA_NodeId parentReferenceNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES);
  75. node->value.variant=*myIntegerVariant;
  76. UA_NodeId_copy(&myIntegerNodeId,&node->nodeId);
  77. UA_QualifiedName_copy(&myIntegerName,&node->browseName);
  78. UA_LocalizedText_copy(&myIntegerDisplName, &node->displayName);
  79. UA_LocalizedText_copy(&myIntegerDisplName, &node->description);
  80. UA_ExpandedNodeId parentId;
  81. UA_ExpandedNodeId_init(&parentId);
  82. UA_NodeId_copy(&parentNodeId,&parentId.nodeId);
  83. return node;
  84. }
  85. START_TEST(ReadSingleAttributeValueWithoutTimestamp) {
  86. UA_Server *server = makeTestSequence();
  87. UA_DataValue resp;
  88. UA_DataValue_init(&resp);
  89. UA_ReadRequest rReq;
  90. UA_ReadRequest_init(&rReq);
  91. rReq.nodesToRead = UA_ReadValueId_new();
  92. rReq.nodesToReadSize = 1;
  93. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  94. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_VALUE;
  95. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  96. ck_assert_int_eq(-1, resp.value.arrayLength);
  97. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_INT32], resp.value.type);
  98. ck_assert_int_eq(42, *(UA_Int32* )resp.value.data);
  99. } END_TEST
  100. START_TEST(ReadSingleAttributeNodeIdWithoutTimestamp) {
  101. UA_Server *server = makeTestSequence();
  102. UA_DataValue resp;
  103. UA_DataValue_init(&resp);
  104. UA_ReadRequest rReq;
  105. UA_ReadRequest_init(&rReq);
  106. rReq.nodesToRead = UA_ReadValueId_new();
  107. rReq.nodesToReadSize = 1;
  108. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  109. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_NODEID;
  110. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  111. UA_NodeId* respval = (UA_NodeId*) resp.value.data;
  112. const UA_NodeId myIntegerNodeId = UA_NODEID_STRING(1, "the.answer");
  113. ck_assert_int_eq(-1, resp.value.arrayLength);
  114. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_NODEID], resp.value.type);
  115. ck_assert_int_eq(1, respval->namespaceIndex);
  116. for (int var = 0; var < respval->identifier.string.length; ++var) {
  117. ck_assert_int_eq(myIntegerNodeId.identifier.string.data[var],
  118. respval->identifier.string.data[var]);
  119. }
  120. UA_free(respval);
  121. } END_TEST
  122. START_TEST(ReadSingleAttributeNodeClassWithoutTimestamp) {
  123. UA_Server *server = makeTestSequence();
  124. UA_DataValue resp;
  125. UA_DataValue_init(&resp);
  126. UA_ReadRequest rReq;
  127. UA_ReadRequest_init(&rReq);
  128. rReq.nodesToRead = UA_ReadValueId_new();
  129. rReq.nodesToReadSize = 1;
  130. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  131. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_NODECLASS;
  132. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  133. ck_assert_int_eq(-1, resp.value.arrayLength);
  134. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_INT32],resp.value.type);
  135. ck_assert_int_eq(*(UA_Int32*)resp.value.data,UA_NODECLASS_VARIABLE);
  136. } END_TEST
  137. START_TEST(ReadSingleAttributeBrowseNameWithoutTimestamp) {
  138. UA_Server *server = makeTestSequence();
  139. UA_DataValue resp;
  140. UA_DataValue_init(&resp);
  141. UA_ReadRequest rReq;
  142. UA_ReadRequest_init(&rReq);
  143. rReq.nodesToRead = UA_ReadValueId_new();
  144. rReq.nodesToReadSize = 1;
  145. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  146. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_BROWSENAME;
  147. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  148. UA_QualifiedName* respval = (UA_QualifiedName*) resp.value.data;
  149. const UA_QualifiedName myIntegerName = UA_QUALIFIEDNAME(1, "the answer");
  150. ck_assert_int_eq(-1, resp.value.arrayLength);
  151. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_QUALIFIEDNAME], resp.value.type);
  152. ck_assert_int_eq(1, respval->namespaceIndex);
  153. ck_assert_int_eq(myIntegerName.name.length, respval->name.length);
  154. for (int var = 0; var < respval->name.length - 1; ++var) {
  155. ck_assert_int_eq(myIntegerName.name.data[var],
  156. respval->name.data[var]);
  157. }
  158. UA_free(respval);
  159. } END_TEST
  160. START_TEST(ReadSingleAttributeDisplayNameWithoutTimestamp) {
  161. UA_Server *server = makeTestSequence();
  162. UA_DataValue resp;
  163. UA_DataValue_init(&resp);
  164. UA_ReadRequest rReq;
  165. UA_ReadRequest_init(&rReq);
  166. rReq.nodesToRead = UA_ReadValueId_new();
  167. rReq.nodesToReadSize = 1;
  168. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  169. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_DISPLAYNAME;
  170. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  171. UA_LocalizedText* respval = (UA_LocalizedText*) resp.value.data;
  172. const UA_LocalizedText comp = UA_LOCALIZEDTEXT("locale", "the answer");
  173. UA_VariableNode* compNode = makeCompareSequence();
  174. ck_assert_int_eq(-1, resp.value.arrayLength);
  175. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_LOCALIZEDTEXT], resp.value.type);
  176. ck_assert_int_eq(comp.text.length, respval->text.length);
  177. for (int var = 0; var < respval->text.length - 1; ++var) {
  178. ck_assert_int_eq(comp.text.data[var], respval->text.data[var]);
  179. }
  180. ck_assert_int_eq(compNode->displayName.locale.length, respval->locale.length);
  181. for (int var = 0; var < respval->locale.length - 1; ++var) {
  182. ck_assert_int_eq(compNode->displayName.locale.data[var], respval->locale.data[var]);
  183. }
  184. UA_free(respval);
  185. } END_TEST
  186. START_TEST(ReadSingleAttributeDescriptionWithoutTimestamp) {
  187. UA_Server *server = makeTestSequence();
  188. UA_DataValue resp;
  189. UA_DataValue_init(&resp);
  190. UA_ReadRequest rReq;
  191. UA_ReadRequest_init(&rReq);
  192. rReq.nodesToRead = UA_ReadValueId_new();
  193. rReq.nodesToReadSize = 1;
  194. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  195. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_DESCRIPTION;
  196. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  197. UA_LocalizedText* respval = (UA_LocalizedText*) resp.value.data;
  198. UA_VariableNode* compNode = makeCompareSequence();
  199. ck_assert_int_eq(-1, resp.value.arrayLength);
  200. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_LOCALIZEDTEXT], resp.value.type);
  201. ck_assert_int_eq(compNode->description.locale.length, respval->locale.length);
  202. for (int var = 0; var < respval->locale.length - 1; ++var) {
  203. ck_assert_int_eq(compNode->description.locale.data[var], respval->locale.data[var]);
  204. }
  205. ck_assert_int_eq(compNode->description.text.length, respval->text.length);
  206. for (int var = 0; var < respval->text.length - 1; ++var) {
  207. ck_assert_int_eq(compNode->description.text.data[var], respval->text.data[var]);
  208. }
  209. UA_free(respval);
  210. } END_TEST
  211. START_TEST(ReadSingleAttributeWriteMaskWithoutTimestamp) {
  212. UA_Server *server = makeTestSequence();
  213. UA_DataValue resp;
  214. UA_DataValue_init(&resp);
  215. UA_ReadRequest rReq;
  216. UA_ReadRequest_init(&rReq);
  217. rReq.nodesToRead = UA_ReadValueId_new();
  218. rReq.nodesToReadSize = 1;
  219. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  220. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_WRITEMASK;
  221. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  222. UA_UInt32* respval = (UA_UInt32*) resp.value.data;
  223. ck_assert_int_eq(-1, resp.value.arrayLength);
  224. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_UINT32], resp.value.type);
  225. ck_assert_int_eq(0,*respval);
  226. } END_TEST
  227. START_TEST(ReadSingleAttributeUserWriteMaskWithoutTimestamp) {
  228. UA_Server *server = makeTestSequence();
  229. UA_DataValue resp;
  230. UA_DataValue_init(&resp);
  231. UA_ReadRequest rReq;
  232. UA_ReadRequest_init(&rReq);
  233. rReq.nodesToRead = UA_ReadValueId_new();
  234. rReq.nodesToReadSize = 1;
  235. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  236. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_USERWRITEMASK;
  237. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  238. UA_UInt32* respval = (UA_UInt32*) resp.value.data;
  239. ck_assert_int_eq(-1, resp.value.arrayLength);
  240. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_UINT32], resp.value.type);
  241. ck_assert_int_eq(0,*respval);
  242. } END_TEST
  243. START_TEST(ReadSingleAttributeIsAbstractWithoutTimestamp) {
  244. UA_Server *server = makeTestSequence();
  245. UA_DataValue resp;
  246. UA_DataValue_init(&resp);
  247. UA_ReadRequest rReq;
  248. UA_ReadRequest_init(&rReq);
  249. rReq.nodesToRead = UA_ReadValueId_new();
  250. rReq.nodesToReadSize = 1;
  251. rReq.nodesToRead[0].nodeId.identifier.numeric = UA_NS0ID_ORGANIZES;
  252. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_ISABSTRACT;
  253. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  254. ck_assert_int_eq(-1, resp.value.arrayLength);
  255. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BOOLEAN], resp.value.type);
  256. ck_assert(*(UA_Boolean* )resp.value.data==UA_FALSE);
  257. } END_TEST
  258. START_TEST(ReadSingleAttributeSymmetricWithoutTimestamp) {
  259. UA_Server *server = makeTestSequence();
  260. UA_DataValue resp;
  261. UA_DataValue_init(&resp);
  262. UA_ReadRequest rReq;
  263. UA_ReadRequest_init(&rReq);
  264. rReq.nodesToRead = UA_ReadValueId_new();
  265. rReq.nodesToReadSize = 1;
  266. rReq.nodesToRead[0].nodeId.identifier.numeric = UA_NS0ID_ORGANIZES;
  267. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_SYMMETRIC;
  268. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  269. ck_assert_int_eq(-1, resp.value.arrayLength);
  270. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BOOLEAN], resp.value.type);
  271. ck_assert(*(UA_Boolean* )resp.value.data==UA_FALSE);
  272. } END_TEST
  273. START_TEST(ReadSingleAttributeInverseNameWithoutTimestamp) {
  274. UA_Server *server = makeTestSequence();
  275. UA_DataValue resp;
  276. UA_DataValue_init(&resp);
  277. UA_ReadRequest rReq;
  278. UA_ReadRequest_init(&rReq);
  279. rReq.nodesToRead = UA_ReadValueId_new();
  280. rReq.nodesToReadSize = 1;
  281. rReq.nodesToRead[0].nodeId.identifier.numeric = UA_NS0ID_ORGANIZES;
  282. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_INVERSENAME;
  283. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  284. UA_LocalizedText* respval = (UA_LocalizedText*) resp.value.data;
  285. const UA_LocalizedText comp = UA_LOCALIZEDTEXT("en_US", "OrganizedBy");
  286. ck_assert_int_eq(-1, resp.value.arrayLength);
  287. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_LOCALIZEDTEXT],resp.value.type);
  288. ck_assert_int_eq(comp.text.length, respval->text.length);
  289. for (int var = 0; var < respval->text.length - 1; ++var) {
  290. ck_assert_int_eq(comp.text.data[var], respval->text.data[var]);
  291. }
  292. ck_assert_int_eq(comp.locale.length, respval->locale.length);
  293. for (int var = 0; var < respval->locale.length - 1; ++var) {
  294. ck_assert_int_eq(comp.locale.data[var], respval->locale.data[var]);
  295. }
  296. UA_free(respval);
  297. } END_TEST
  298. START_TEST(ReadSingleAttributeContainsNoLoopsWithoutTimestamp) {
  299. UA_Server *server = makeTestSequence();
  300. UA_DataValue resp;
  301. UA_DataValue_init(&resp);
  302. UA_ReadRequest rReq;
  303. UA_ReadRequest_init(&rReq);
  304. rReq.nodesToRead = UA_ReadValueId_new();
  305. rReq.nodesToReadSize = 1;
  306. rReq.nodesToRead[0].nodeId.identifier.numeric = UA_NS0ID_VIEWNODE;
  307. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_CONTAINSNOLOOPS;
  308. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  309. ck_assert_int_eq(-1, resp.value.arrayLength);
  310. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BOOLEAN], resp.value.type);
  311. ck_assert(*(UA_Boolean* )resp.value.data==UA_FALSE);
  312. } END_TEST
  313. START_TEST(ReadSingleAttributeEventNotifierWithoutTimestamp) {
  314. UA_Server *server = makeTestSequence();
  315. UA_DataValue resp;
  316. UA_DataValue_init(&resp);
  317. UA_ReadRequest rReq;
  318. UA_ReadRequest_init(&rReq);
  319. rReq.nodesToRead = UA_ReadValueId_new();
  320. rReq.nodesToReadSize = 1;
  321. rReq.nodesToRead[0].nodeId = UA_NODEID_NUMERIC(1, 50);
  322. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_EVENTNOTIFIER;
  323. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  324. ck_assert_int_eq(-1, resp.value.arrayLength);
  325. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BYTE],resp.value.type);
  326. ck_assert_int_eq(*(UA_Byte*)resp.value.data, 0);
  327. } END_TEST
  328. START_TEST(ReadSingleAttributeDataTypeWithoutTimestamp) {
  329. UA_Server *server = makeTestSequence();
  330. UA_DataValue resp;
  331. UA_DataValue_init(&resp);
  332. UA_ReadRequest rReq;
  333. UA_ReadRequest_init(&rReq);
  334. rReq.nodesToRead = UA_ReadValueId_new();
  335. rReq.nodesToReadSize = 1;
  336. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  337. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_DATATYPE;
  338. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  339. UA_NodeId* respval = (UA_NodeId*) resp.value.data;
  340. ck_assert_int_eq(-1, resp.value.arrayLength);
  341. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_NODEID], resp.value.type);
  342. ck_assert_int_eq(respval->namespaceIndex,0);
  343. ck_assert_int_eq(respval->identifier.numeric,UA_NS0ID_INT32);
  344. } END_TEST
  345. START_TEST(ReadSingleAttributeValueRankWithoutTimestamp) {
  346. UA_Server *server = makeTestSequence();
  347. UA_DataValue resp;
  348. UA_DataValue_init(&resp);
  349. UA_ReadRequest rReq;
  350. UA_ReadRequest_init(&rReq);
  351. rReq.nodesToRead = UA_ReadValueId_new();
  352. rReq.nodesToReadSize = 1;
  353. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  354. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_VALUERANK;
  355. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  356. ck_assert_int_eq(-1, resp.value.arrayLength);
  357. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_INT32], resp.value.type);
  358. ck_assert_int_eq(-2, *(UA_Int32* )resp.value.data);
  359. } END_TEST
  360. START_TEST(ReadSingleAttributeArrayDimensionsWithoutTimestamp) {
  361. UA_Server *server = makeTestSequence();
  362. UA_DataValue resp;
  363. UA_DataValue_init(&resp);
  364. UA_ReadRequest rReq;
  365. UA_ReadRequest_init(&rReq);
  366. rReq.nodesToRead = UA_ReadValueId_new();
  367. rReq.nodesToReadSize = 1;
  368. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  369. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_ARRAYDIMENSIONS;
  370. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  371. ck_assert_int_eq(-1, resp.value.arrayLength);
  372. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_INT32], resp.value.type);
  373. ck_assert_ptr_eq((UA_Int32*)resp.value.data,0);
  374. } END_TEST
  375. START_TEST(ReadSingleAttributeAccessLevelWithoutTimestamp) {
  376. UA_Server *server = makeTestSequence();
  377. UA_DataValue resp;
  378. UA_DataValue_init(&resp);
  379. UA_ReadRequest rReq;
  380. UA_ReadRequest_init(&rReq);
  381. rReq.nodesToRead = UA_ReadValueId_new();
  382. rReq.nodesToReadSize = 1;
  383. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  384. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_ACCESSLEVEL;
  385. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  386. ck_assert_int_eq(-1, resp.value.arrayLength);
  387. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BYTE], resp.value.type);
  388. ck_assert_int_eq(*(UA_Byte*)resp.value.data, 0);
  389. } END_TEST
  390. START_TEST(ReadSingleAttributeUserAccessLevelWithoutTimestamp) {
  391. UA_Server *server = makeTestSequence();
  392. UA_DataValue resp;
  393. UA_DataValue_init(&resp);
  394. UA_ReadRequest rReq;
  395. UA_ReadRequest_init(&rReq);
  396. rReq.nodesToRead = UA_ReadValueId_new();
  397. rReq.nodesToReadSize = 1;
  398. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  399. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_USERACCESSLEVEL;
  400. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  401. UA_VariableNode* compNode = makeCompareSequence();
  402. ck_assert_int_eq(-1, resp.value.arrayLength);
  403. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BYTE], resp.value.type);
  404. ck_assert_int_eq(*(UA_Byte*)resp.value.data, compNode->userAccessLevel);
  405. } END_TEST
  406. START_TEST(ReadSingleAttributeMinimumSamplingIntervalWithoutTimestamp) {
  407. UA_Server *server = makeTestSequence();
  408. UA_DataValue resp;
  409. UA_DataValue_init(&resp);
  410. UA_ReadRequest rReq;
  411. UA_ReadRequest_init(&rReq);
  412. rReq.nodesToRead = UA_ReadValueId_new();
  413. rReq.nodesToReadSize = 1;
  414. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  415. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_MINIMUMSAMPLINGINTERVAL;
  416. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  417. UA_Double* respval = (UA_Double*) resp.value.data;
  418. UA_VariableNode *compNode = makeCompareSequence();
  419. UA_Double comp = (UA_Double) compNode->minimumSamplingInterval;
  420. ck_assert_int_eq(-1, resp.value.arrayLength);
  421. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_DOUBLE], resp.value.type);
  422. ck_assert(*respval == comp);
  423. } END_TEST
  424. START_TEST(ReadSingleAttributeHistorizingWithoutTimestamp) {
  425. UA_Server *server = makeTestSequence();
  426. UA_DataValue resp;
  427. UA_DataValue_init(&resp);
  428. UA_ReadRequest rReq;
  429. UA_ReadRequest_init(&rReq);
  430. rReq.nodesToRead = UA_ReadValueId_new();
  431. rReq.nodesToReadSize = 1;
  432. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  433. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_HISTORIZING;
  434. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  435. ck_assert_int_eq(-1, resp.value.arrayLength);
  436. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BOOLEAN], resp.value.type);
  437. ck_assert(*(UA_Boolean* )resp.value.data==UA_FALSE);
  438. } END_TEST
  439. START_TEST(ReadSingleAttributeExecutableWithoutTimestamp) {
  440. UA_Server *server = makeTestSequence();
  441. UA_DataValue resp;
  442. UA_DataValue_init(&resp);
  443. UA_ReadRequest rReq;
  444. UA_ReadRequest_init(&rReq);
  445. rReq.nodesToRead = UA_ReadValueId_new();
  446. rReq.nodesToReadSize = 1;
  447. rReq.nodesToRead[0].nodeId.identifier.numeric = UA_NS0ID_METHODNODE;
  448. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_EXECUTABLE;
  449. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  450. ck_assert_int_eq(-1, resp.value.arrayLength);
  451. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BOOLEAN], resp.value.type);
  452. ck_assert(*(UA_Boolean*)resp.value.data==UA_FALSE);
  453. } END_TEST
  454. START_TEST(ReadSingleAttributeUserExecutableWithoutTimestamp) {
  455. UA_Server *server = makeTestSequence();
  456. UA_DataValue resp;
  457. UA_DataValue_init(&resp);
  458. UA_ReadRequest rReq;
  459. UA_ReadRequest_init(&rReq);
  460. rReq.nodesToRead = UA_ReadValueId_new();
  461. rReq.nodesToReadSize = 1;
  462. rReq.nodesToRead[0].nodeId.identifier.numeric = UA_NS0ID_METHODNODE;
  463. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_USEREXECUTABLE;
  464. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  465. ck_assert_int_eq(-1, resp.value.arrayLength);
  466. ck_assert_ptr_eq(&UA_TYPES[UA_TYPES_BOOLEAN], resp.value.type);
  467. ck_assert(*(UA_Boolean*)resp.value.data==UA_FALSE);
  468. } END_TEST
  469. /* Tests for writeValue method */
  470. START_TEST(WriteSingleAttributeNodeId) {
  471. UA_Server *server = makeTestSequence();
  472. UA_WriteValue wValue;
  473. UA_WriteValue_init(&wValue);
  474. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  475. wValue.attributeId = UA_ATTRIBUTEID_NODEID;
  476. wValue.value.hasValue = UA_TRUE;
  477. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  478. ck_assert_int_eq(retval, UA_STATUSCODE_BADWRITENOTSUPPORTED);
  479. } END_TEST
  480. START_TEST(WriteSingleAttributeNodeclass) {
  481. UA_Server *server = makeTestSequence();
  482. UA_WriteValue wValue;
  483. UA_WriteValue_init(&wValue);
  484. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  485. wValue.attributeId = UA_ATTRIBUTEID_NODECLASS;
  486. wValue.value.hasValue = UA_TRUE;
  487. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  488. ck_assert_int_eq(retval, UA_STATUSCODE_BADWRITENOTSUPPORTED);
  489. } END_TEST
  490. START_TEST(WriteSingleAttributeBrowseName) {
  491. UA_Server *server = makeTestSequence();
  492. UA_WriteValue wValue;
  493. UA_WriteValue_init(&wValue);
  494. UA_QualifiedName testValue = UA_QUALIFIEDNAME(1, "the.answer");
  495. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_QUALIFIEDNAME]);
  496. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  497. wValue.attributeId = UA_ATTRIBUTEID_BROWSENAME;
  498. wValue.value.hasValue = UA_TRUE;
  499. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  500. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  501. } END_TEST
  502. START_TEST(WriteSingleAttributeDisplayName) {
  503. UA_Server *server = makeTestSequence();
  504. UA_WriteValue wValue;
  505. UA_WriteValue_init(&wValue);
  506. UA_LocalizedText testValue = UA_LOCALIZEDTEXT("en_EN", "the.answer");
  507. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_LOCALIZEDTEXT]);
  508. wValue.value.hasValue = UA_TRUE;
  509. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  510. wValue.attributeId = UA_ATTRIBUTEID_DISPLAYNAME;
  511. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  512. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  513. } END_TEST
  514. START_TEST(WriteSingleAttributeDescription) {
  515. UA_Server *server = makeTestSequence();
  516. UA_WriteValue wValue;
  517. UA_WriteValue_init(&wValue);
  518. UA_LocalizedText testValue = UA_LOCALIZEDTEXT("en_EN", "the.answer");
  519. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_LOCALIZEDTEXT]);
  520. wValue.value.hasValue = UA_TRUE;
  521. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  522. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  523. wValue.attributeId = UA_ATTRIBUTEID_DESCRIPTION;
  524. wValue.value.hasValue = UA_TRUE;
  525. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  526. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  527. } END_TEST
  528. START_TEST(WriteSingleAttributeWriteMask) {
  529. UA_Server *server = makeTestSequence();
  530. UA_WriteValue wValue;
  531. UA_WriteValue_init(&wValue);
  532. UA_Int32 testValue = 0;
  533. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_UINT32]);
  534. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  535. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  536. wValue.attributeId = UA_ATTRIBUTEID_WRITEMASK;
  537. wValue.value.hasValue = UA_TRUE;
  538. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  539. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  540. } END_TEST
  541. START_TEST(WriteSingleAttributeUserWriteMask) {
  542. UA_Server *server = makeTestSequence();
  543. UA_WriteValue wValue;
  544. UA_WriteValue_init(&wValue);
  545. UA_Int32 testValue = 0;
  546. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_UINT32]);
  547. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  548. wValue.attributeId = UA_ATTRIBUTEID_USERWRITEMASK;
  549. wValue.value.hasValue = UA_TRUE;
  550. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  551. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  552. } END_TEST
  553. START_TEST(WriteSingleAttributeIsAbstract) {
  554. UA_Server *server = makeTestSequence();
  555. UA_WriteValue wValue;
  556. UA_WriteValue_init(&wValue);
  557. UA_Boolean testValue = UA_TRUE;
  558. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BOOLEAN]);
  559. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  560. wValue.attributeId = UA_ATTRIBUTEID_ISABSTRACT;
  561. wValue.value.hasValue = UA_TRUE;
  562. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  563. ck_assert_int_eq(retval, UA_STATUSCODE_BADNODECLASSINVALID);
  564. } END_TEST
  565. START_TEST(WriteSingleAttributeSymmetric) {
  566. UA_Server *server = makeTestSequence();
  567. UA_WriteValue wValue;
  568. UA_WriteValue_init(&wValue);
  569. UA_Boolean testValue = UA_TRUE;
  570. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BOOLEAN]);
  571. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  572. wValue.attributeId = UA_ATTRIBUTEID_SYMMETRIC;
  573. wValue.value.hasValue = UA_TRUE;
  574. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  575. ck_assert_int_eq(retval, UA_STATUSCODE_BADNODECLASSINVALID);
  576. } END_TEST
  577. START_TEST(WriteSingleAttributeInverseName) {
  578. UA_Server *server = makeTestSequence();
  579. UA_WriteValue wValue;
  580. UA_WriteValue_init(&wValue);
  581. UA_LocalizedText testValue = UA_LOCALIZEDTEXT("en_US", "not.the.answer");
  582. UA_Variant_setScalar(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_LOCALIZEDTEXT]);
  583. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  584. wValue.attributeId = UA_ATTRIBUTEID_INVERSENAME;
  585. wValue.value.hasValue = UA_TRUE;
  586. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  587. ck_assert_int_eq(retval, UA_STATUSCODE_BADNODECLASSINVALID);
  588. } END_TEST
  589. START_TEST(WriteSingleAttributeContainsNoLoops) {
  590. UA_Server *server = makeTestSequence();
  591. UA_WriteValue wValue;
  592. UA_WriteValue_init(&wValue);
  593. UA_Boolean testValue = UA_TRUE;
  594. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BOOLEAN]);
  595. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  596. wValue.attributeId = UA_ATTRIBUTEID_CONTAINSNOLOOPS;
  597. wValue.value.hasValue = UA_TRUE;
  598. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  599. ck_assert_int_eq(retval, UA_STATUSCODE_BADNODECLASSINVALID);
  600. } END_TEST
  601. START_TEST(WriteSingleAttributeEventNotifier) {
  602. UA_Server *server = makeTestSequence();
  603. UA_WriteValue wValue;
  604. UA_WriteValue_init(&wValue);
  605. UA_Byte testValue = 0;
  606. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BYTE]);
  607. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  608. wValue.attributeId = UA_ATTRIBUTEID_EVENTNOTIFIER;
  609. wValue.value.hasValue = UA_TRUE;
  610. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  611. ck_assert_int_eq(retval, UA_STATUSCODE_BADNODECLASSINVALID);
  612. } END_TEST
  613. START_TEST(WriteSingleAttributeValue) {
  614. UA_Server *server = makeTestSequence();
  615. UA_WriteValue wValue;
  616. UA_WriteValue_init(&wValue);
  617. UA_Variant *myIntegerVariant = UA_Variant_new();
  618. UA_Int32 myInteger = 20;
  619. UA_Variant_setScalarCopy(myIntegerVariant, &myInteger, &UA_TYPES[UA_TYPES_INT32]);
  620. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  621. wValue.attributeId = UA_ATTRIBUTEID_VALUE;
  622. wValue.value.hasValue = UA_TRUE;
  623. wValue.value.value = *myIntegerVariant;
  624. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  625. UA_DataValue resp;
  626. UA_DataValue_init(&resp);
  627. UA_ReadRequest rReq;
  628. UA_ReadRequest_init(&rReq);
  629. rReq.nodesToRead = UA_ReadValueId_new();
  630. rReq.nodesToReadSize = 1;
  631. rReq.nodesToRead[0].nodeId = UA_NODEID_STRING(1, "the.answer");
  632. rReq.nodesToRead[0].attributeId = UA_ATTRIBUTEID_VALUE;
  633. Service_Read_single(server, &adminSession, UA_TIMESTAMPSTORETURN_NEITHER, &rReq.nodesToRead[0], &resp);
  634. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  635. ck_assert(wValue.value.hasValue);
  636. const UA_Node *node = UA_NodeStore_get(server->nodestore, &wValue.nodeId);
  637. ck_assert_int_eq(node->nodeClass, UA_NODECLASS_VARIABLE);
  638. const UA_VariableNode *vn = (const UA_VariableNode*)node;
  639. const UA_Variant *oldV = &vn->value.variant;
  640. ck_assert_ptr_eq(&oldV->type->typeId, &wValue.value.value.type->typeId);
  641. ck_assert_int_eq(20, *(UA_Int32* )resp.value.data);
  642. } END_TEST
  643. START_TEST(WriteSingleAttributeDataType) {
  644. UA_Server *server = makeTestSequence();
  645. UA_WriteValue wValue;
  646. UA_WriteValue_init(&wValue);
  647. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  648. wValue.attributeId = UA_ATTRIBUTEID_DATATYPE;
  649. wValue.value.hasValue = UA_TRUE;
  650. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  651. ck_assert_int_eq(retval, UA_STATUSCODE_BADWRITENOTSUPPORTED);
  652. } END_TEST
  653. START_TEST(WriteSingleAttributeValueRank) {
  654. UA_Server *server = makeTestSequence();
  655. UA_WriteValue wValue;
  656. UA_WriteValue_init(&wValue);
  657. UA_Int32 testValue = -1;
  658. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_INT32]);
  659. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  660. wValue.attributeId = UA_ATTRIBUTEID_VALUERANK;
  661. wValue.value.hasValue = UA_TRUE;
  662. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  663. // Returns attributeInvalid, since variant/value may be writable
  664. ck_assert_int_eq(retval, UA_STATUSCODE_BADATTRIBUTEIDINVALID);
  665. } END_TEST
  666. START_TEST(WriteSingleAttributeArrayDimensions) {
  667. UA_Server *server = makeTestSequence();
  668. UA_WriteValue wValue;
  669. UA_WriteValue_init(&wValue);
  670. UA_Int32 testValue[] = {-1,-1,-1};
  671. UA_Variant_setArrayCopy(&wValue.value.value, &testValue, 3, &UA_TYPES[UA_TYPES_INT32]);
  672. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  673. wValue.attributeId = UA_ATTRIBUTEID_ARRAYDIMENSIONS;
  674. wValue.value.hasValue = UA_TRUE;
  675. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  676. // Returns attributeInvalid, since variant/value may be writable
  677. ck_assert_int_eq(retval, UA_STATUSCODE_BADATTRIBUTEIDINVALID);
  678. } END_TEST
  679. START_TEST(WriteSingleAttributeAccessLevel) {
  680. UA_Server *server = makeTestSequence();
  681. UA_WriteValue wValue;
  682. UA_WriteValue_init(&wValue);
  683. UA_Byte testValue = 0;
  684. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BYTE]);
  685. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  686. wValue.attributeId = UA_ATTRIBUTEID_ACCESSLEVEL;
  687. wValue.value.hasValue = UA_TRUE;
  688. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  689. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  690. } END_TEST
  691. START_TEST(WriteSingleAttributeUserAccessLevel) {
  692. UA_Server *server = makeTestSequence();
  693. UA_WriteValue wValue;
  694. UA_WriteValue_init(&wValue);
  695. UA_Byte testValue = 0;
  696. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BYTE]);
  697. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  698. wValue.attributeId = UA_ATTRIBUTEID_USERACCESSLEVEL;
  699. wValue.value.hasValue = UA_TRUE;
  700. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  701. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  702. } END_TEST
  703. START_TEST(WriteSingleAttributeMinimumSamplingInterval) {
  704. UA_Server *server = makeTestSequence();
  705. UA_WriteValue wValue;
  706. UA_WriteValue_init(&wValue);
  707. UA_Double testValue = 0.0;
  708. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_DOUBLE]);
  709. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  710. wValue.attributeId = UA_ATTRIBUTEID_MINIMUMSAMPLINGINTERVAL;
  711. wValue.value.hasValue = UA_TRUE;
  712. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  713. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  714. } END_TEST
  715. START_TEST(WriteSingleAttributeHistorizing) {
  716. UA_Server *server = makeTestSequence();
  717. UA_WriteValue wValue;
  718. UA_WriteValue_init(&wValue);
  719. UA_Boolean testValue = UA_TRUE;
  720. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BOOLEAN]);
  721. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  722. wValue.attributeId = UA_ATTRIBUTEID_HISTORIZING;
  723. wValue.value.hasValue = UA_TRUE;
  724. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  725. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  726. } END_TEST
  727. START_TEST(WriteSingleAttributeExecutable) {
  728. UA_Server *server = makeTestSequence();
  729. UA_WriteValue wValue;
  730. UA_WriteValue_init(&wValue);
  731. UA_Boolean testValue = UA_TRUE;
  732. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BOOLEAN]);
  733. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  734. wValue.attributeId = UA_ATTRIBUTEID_EXECUTABLE;
  735. wValue.value.hasValue = UA_TRUE;
  736. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  737. ck_assert_int_eq(retval, UA_STATUSCODE_BADNODECLASSINVALID);
  738. } END_TEST
  739. START_TEST(WriteSingleAttributeUserExecutable) {
  740. UA_Server *server = makeTestSequence();
  741. UA_WriteValue wValue;
  742. UA_WriteValue_init(&wValue);
  743. UA_Boolean testValue = UA_TRUE;
  744. UA_Variant_setScalarCopy(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_BOOLEAN]);
  745. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  746. wValue.attributeId = UA_ATTRIBUTEID_USEREXECUTABLE;
  747. wValue.value.hasValue = UA_TRUE;
  748. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  749. ck_assert_int_eq(retval, UA_STATUSCODE_BADNODECLASSINVALID);
  750. } END_TEST
  751. START_TEST(WriteSingleAttributeNoValue) {
  752. UA_Server *server = makeTestSequence();
  753. UA_WriteValue wValue;
  754. UA_WriteValue_init(&wValue);
  755. wValue.nodeId = UA_NODEID_STRING(1, "the.answer");
  756. wValue.attributeId = UA_ATTRIBUTEID_USEREXECUTABLE;
  757. wValue.value.hasValue = UA_FALSE;
  758. UA_StatusCode retval = Service_Write_single(server, &adminSession, &wValue);
  759. ck_assert_int_eq(retval, UA_STATUSCODE_BADTYPEMISMATCH);
  760. } END_TEST
  761. START_TEST(numericRange) {
  762. UA_NumericRange range;
  763. const UA_String str = (UA_String){9, (UA_Byte*)"1:2,0:3,5"};
  764. UA_StatusCode retval = parse_numericrange(str, &range);
  765. ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
  766. ck_assert_int_eq(range.dimensionsSize,3);
  767. ck_assert_int_eq(range.dimensions[0].min,1);
  768. ck_assert_int_eq(range.dimensions[0].max,2);
  769. ck_assert_int_eq(range.dimensions[1].min,0);
  770. ck_assert_int_eq(range.dimensions[1].max,3);
  771. ck_assert_int_eq(range.dimensions[2].min,5);
  772. ck_assert_int_eq(range.dimensions[2].max,5);
  773. } END_TEST
  774. static Suite * testSuite_services_attributes(void) {
  775. Suite *s = suite_create("services_attributes_read");
  776. TCase *tc_readSingleAttributes = tcase_create("readSingleAttributes");
  777. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeValueWithoutTimestamp);
  778. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeNodeIdWithoutTimestamp);
  779. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeNodeClassWithoutTimestamp);
  780. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeBrowseNameWithoutTimestamp);
  781. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeDisplayNameWithoutTimestamp);
  782. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeDescriptionWithoutTimestamp);
  783. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeWriteMaskWithoutTimestamp);
  784. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeUserWriteMaskWithoutTimestamp);
  785. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeIsAbstractWithoutTimestamp);
  786. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeSymmetricWithoutTimestamp);
  787. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeInverseNameWithoutTimestamp);
  788. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeContainsNoLoopsWithoutTimestamp);
  789. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeEventNotifierWithoutTimestamp);
  790. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeDataTypeWithoutTimestamp);
  791. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeValueRankWithoutTimestamp);
  792. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeArrayDimensionsWithoutTimestamp);
  793. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeAccessLevelWithoutTimestamp);
  794. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeUserAccessLevelWithoutTimestamp);
  795. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeMinimumSamplingIntervalWithoutTimestamp);
  796. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeHistorizingWithoutTimestamp);
  797. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeExecutableWithoutTimestamp);
  798. tcase_add_test(tc_readSingleAttributes, ReadSingleAttributeUserExecutableWithoutTimestamp);
  799. suite_add_tcase(s, tc_readSingleAttributes);
  800. TCase *tc_writeSingleAttributes = tcase_create("writeSingleAttributes");
  801. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeNodeId);
  802. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeNodeclass);
  803. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeBrowseName);
  804. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeDisplayName);
  805. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeDescription);
  806. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeWriteMask);
  807. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeUserWriteMask);
  808. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeIsAbstract);
  809. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeSymmetric);
  810. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeInverseName);
  811. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeContainsNoLoops);
  812. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeEventNotifier);
  813. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeValue);
  814. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeDataType);
  815. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeValueRank);
  816. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeArrayDimensions);
  817. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeAccessLevel);
  818. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeUserAccessLevel);
  819. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeMinimumSamplingInterval);
  820. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeHistorizing);
  821. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeExecutable);
  822. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeUserExecutable);
  823. tcase_add_test(tc_writeSingleAttributes, WriteSingleAttributeNoValue);
  824. suite_add_tcase(s, tc_writeSingleAttributes);
  825. TCase *tc_parseNumericRange = tcase_create("parseNumericRange");
  826. tcase_add_test(tc_parseNumericRange, numericRange);
  827. suite_add_tcase(s, tc_parseNumericRange);
  828. return s;
  829. }
  830. int main(void) {
  831. int number_failed = 0;
  832. Suite *s;
  833. s = testSuite_services_attributes();
  834. SRunner *sr = srunner_create(s);
  835. srunner_set_log(sr, "test.log");
  836. void srunner_set_fork_status(SRunner * sr, enum fork_status CK_NOFORK);
  837. srunner_run_all(sr, CK_NORMAL);
  838. number_failed += srunner_ntests_failed(sr);
  839. srunner_free(sr);
  840. return (number_failed == 0) ? EXIT_SUCCESS : EXIT_FAILURE;
  841. }