corpus_generator.c 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708
  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. * This code is used to generate a binary file for every request type
  6. * which can be sent from a client to the server.
  7. * These files form the basic corpus for fuzzing the server.
  8. * This script is intended to be executed manually and then commit the new
  9. * corpus to the repository.
  10. */
  11. #ifndef UA_DEBUG_DUMP_PKGS_FILE
  12. #error UA_DEBUG_DUMP_PKGS_FILE must be defined
  13. #endif
  14. #include <stdio.h>
  15. #include <stdlib.h>
  16. #include <pthread.h>
  17. #include <ua_types.h>
  18. #include <sys/stat.h>
  19. #include <server/ua_server_internal.h>
  20. #include <dirent.h>
  21. #include <src_generated/ua_types_generated.h>
  22. #include <fcntl.h>
  23. #include <unistd.h>
  24. #include <ua_client_highlevel.h>
  25. #include "ua_server.h"
  26. #include "ua_client.h"
  27. #include "ua_config_default.h"
  28. #include "ua_client_highlevel.h"
  29. #include "ua_network_tcp.h"
  30. #include "ua_transport_generated.h"
  31. #include "ua_transport_generated_encoding_binary.h"
  32. #include "ua_securechannel.h"
  33. #include "ua_types_generated_encoding_binary.h"
  34. #include "server/ua_services.h"
  35. unsigned int chunkCount = 0;
  36. char dumpOutputFile[255];
  37. char *messageTypes[] = {"ack", "hel", "msg", "opn", "clo", "err", "unk"};
  38. char *messageTypePrefix = NULL;
  39. char requestServiceName[200];
  40. void UA_debug_dumpCompleteChunk(UA_Server *const server, UA_Connection *const connection, UA_ByteString *messageBuffer);
  41. static void setMessageTypePrefix(UA_UInt32 messageType) {
  42. switch(messageType & 0x00ffffff) {
  43. case UA_MESSAGETYPE_ACK:
  44. messageTypePrefix = messageTypes[0];
  45. break;
  46. case UA_MESSAGETYPE_HEL:
  47. messageTypePrefix = messageTypes[1];
  48. break;
  49. case UA_MESSAGETYPE_MSG:
  50. messageTypePrefix = messageTypes[2];
  51. break;
  52. case UA_MESSAGETYPE_OPN:
  53. messageTypePrefix = messageTypes[3];
  54. break;
  55. case UA_MESSAGETYPE_CLO:
  56. messageTypePrefix = messageTypes[4];
  57. break;
  58. case UA_MESSAGETYPE_ERR:
  59. messageTypePrefix = messageTypes[5];
  60. break;
  61. default:
  62. messageTypePrefix = messageTypes[6];
  63. }
  64. }
  65. static UA_StatusCode setRequestedServiceName(const UA_ByteString *msg) {
  66. /* At 0, the nodeid starts... */
  67. size_t offset = 0;
  68. /* Decode the nodeid */
  69. UA_NodeId requestTypeId;
  70. UA_StatusCode retval = UA_NodeId_decodeBinary(msg, &offset, &requestTypeId);
  71. if(retval != UA_STATUSCODE_GOOD)
  72. return retval;
  73. if(requestTypeId.identifierType != UA_NODEIDTYPE_NUMERIC || requestTypeId.namespaceIndex != 0) {
  74. snprintf(requestServiceName, 200, "invalid_request_id");
  75. return UA_STATUSCODE_BADUNEXPECTEDERROR;
  76. }
  77. const UA_DataType *requestType = NULL;
  78. for (size_t i=0; i<UA_TYPES_COUNT; i++) {
  79. if (UA_TYPES[i].binaryEncodingId == requestTypeId.identifier.numeric) {
  80. requestType = &UA_TYPES[i];
  81. break;
  82. }
  83. }
  84. if (requestType == NULL) {
  85. snprintf(requestServiceName, 200, "invalid_request_no_type");
  86. return UA_STATUSCODE_BADUNEXPECTEDERROR;
  87. }
  88. snprintf(requestServiceName, 200, "_%s", requestType->typeName);
  89. return UA_STATUSCODE_GOOD;
  90. }
  91. static UA_StatusCode
  92. processCompleteChunkWithoutChannel(UA_Server *server, UA_Connection *connection,
  93. UA_ByteString *message) {
  94. size_t offset = 0;
  95. UA_TcpMessageHeader tcpMessageHeader;
  96. UA_StatusCode retval =
  97. UA_TcpMessageHeader_decodeBinary(message, &offset, &tcpMessageHeader);
  98. if(retval != UA_STATUSCODE_GOOD)
  99. return retval;
  100. setMessageTypePrefix(tcpMessageHeader.messageTypeAndChunkType & 0x00ffffff);
  101. if ((tcpMessageHeader.messageTypeAndChunkType & 0x00ffffff) == UA_MESSAGETYPE_MSG) {
  102. // this should not happen in normal operation
  103. UA_LOG_ERROR(server->config.logger, UA_LOGCATEGORY_SERVER, "Got MSG package without channel.");
  104. return UA_STATUSCODE_BADUNEXPECTEDERROR;
  105. }
  106. return UA_STATUSCODE_GOOD;
  107. }
  108. /* Takes decoded messages starting at the nodeid of the content type. */
  109. static UA_StatusCode
  110. processSecureChannelMessage(void *application, UA_SecureChannel *channel,
  111. UA_MessageType messagetype, UA_UInt32 requestId,
  112. const UA_ByteString *message) {
  113. //UA_Server *server = (UA_Server*)application;
  114. UA_StatusCode retval = UA_STATUSCODE_GOOD;
  115. setMessageTypePrefix(messagetype);
  116. if (messagetype == UA_MESSAGETYPE_MSG) {
  117. setRequestedServiceName(message);
  118. }
  119. return retval;
  120. }
  121. void UA_debug_dumpCompleteChunk(UA_Server *const server, UA_Connection *const connection, UA_ByteString *messageBuffer) {
  122. messageTypePrefix = NULL;
  123. requestServiceName[0] = 0;
  124. UA_StatusCode retval;
  125. if(!connection->channel)
  126. retval = processCompleteChunkWithoutChannel(server, connection, messageBuffer);
  127. else {
  128. // make a backup of the sequence number and reset it, because processChunk increases it
  129. UA_UInt32 seqBackup = connection->channel->receiveSequenceNumber;
  130. retval = UA_SecureChannel_processChunk(connection->channel, messageBuffer,
  131. processSecureChannelMessage,
  132. server);
  133. connection->channel->receiveSequenceNumber = seqBackup;
  134. }
  135. snprintf(dumpOutputFile, 255, "%s/%05d_%s%s.bin", UA_CORPUS_OUTPUT_DIR, ++chunkCount, messageTypePrefix ? messageTypePrefix : "", requestServiceName);
  136. UA_LOG_INFO(server->config.logger, UA_LOGCATEGORY_SERVER, "Dumping package %s", dumpOutputFile);
  137. FILE *write_ptr;
  138. write_ptr = fopen(dumpOutputFile, "ab");
  139. fwrite(messageBuffer->data, messageBuffer->length, 1, write_ptr); // write 10 bytes from our buffer
  140. fclose(write_ptr);
  141. }
  142. UA_Server *server;
  143. UA_ServerConfig *config;
  144. UA_Boolean *running;
  145. pthread_t server_thread;
  146. static void * serverloop(void *_) {
  147. while(*running)
  148. UA_Server_run_iterate(server, true);
  149. return NULL;
  150. }
  151. static void start_server(void) {
  152. running = UA_Boolean_new();
  153. *running = true;
  154. config = UA_ServerConfig_new_default();
  155. server = UA_Server_new(config);
  156. UA_Server_run_startup(server);
  157. pthread_create(&server_thread, NULL, serverloop, NULL);
  158. }
  159. static void teardown_server(void) {
  160. *running = false;
  161. pthread_join(server_thread, NULL);
  162. UA_Server_run_shutdown(server);
  163. UA_Boolean_delete(running);
  164. UA_Server_delete(server);
  165. UA_ServerConfig_delete(config);
  166. }
  167. static void emptyCorpusDir() {
  168. DIR *theFolder = opendir(UA_CORPUS_OUTPUT_DIR);
  169. struct dirent *next_file;
  170. char filepath[256];
  171. while ( (next_file = readdir(theFolder)) != NULL )
  172. {
  173. // build the path for each file in the folder
  174. sprintf(filepath, "%s/%s", UA_CORPUS_OUTPUT_DIR, next_file->d_name);
  175. remove(filepath);
  176. }
  177. closedir(theFolder);
  178. }
  179. #define ASSERT_GOOD(X) if (X != UA_STATUSCODE_GOOD) return X;
  180. /*************************************************
  181. * The following list of client requests is based
  182. * on ua_server_binary.c:getServicePointers to
  183. * cover all possible services and their inputs
  184. ************************************************/
  185. static UA_StatusCode findServersRequest(UA_Client *client) {
  186. UA_ApplicationDescription* applicationDescriptionArray = NULL;
  187. size_t applicationDescriptionArraySize = 0;
  188. size_t serverUrisSize = 1;
  189. UA_String *serverUris = UA_String_new();
  190. serverUris[0] = UA_String_fromChars("urn:some:server:uri");
  191. size_t localeIdsSize = 1;
  192. UA_String *localeIds = UA_String_new();
  193. localeIds[0] = UA_String_fromChars("en");
  194. ASSERT_GOOD(UA_Client_findServers(client, "opc.tcp://localhost:4840",
  195. serverUrisSize, serverUris, localeIdsSize, localeIds,
  196. &applicationDescriptionArraySize, &applicationDescriptionArray));
  197. UA_Array_delete(serverUris, serverUrisSize, &UA_TYPES[UA_TYPES_STRING]);
  198. UA_Array_delete(localeIds, localeIdsSize, &UA_TYPES[UA_TYPES_STRING]);
  199. UA_Array_delete(applicationDescriptionArray, applicationDescriptionArraySize,
  200. &UA_TYPES[UA_TYPES_APPLICATIONDESCRIPTION]);
  201. return UA_STATUSCODE_GOOD;
  202. }
  203. static UA_StatusCode findServersOnNetworkRequest(UA_Client *client) {
  204. UA_ServerOnNetwork* serverOnNetwork = NULL;
  205. size_t serverOnNetworkSize = 0;
  206. size_t serverCapabilityFilterSize = 2;
  207. UA_String *serverCapabilityFilter = (UA_String*)UA_malloc(sizeof(UA_String) * serverCapabilityFilterSize);
  208. serverCapabilityFilter[0] = UA_String_fromChars("LDS");
  209. serverCapabilityFilter[1] = UA_String_fromChars("NA");
  210. ASSERT_GOOD(UA_Client_findServersOnNetwork(client, "opc.tcp://localhost:4840", 0, 0,
  211. serverCapabilityFilterSize, serverCapabilityFilter,
  212. &serverOnNetworkSize, &serverOnNetwork));
  213. UA_Array_delete(serverCapabilityFilter, serverCapabilityFilterSize,
  214. &UA_TYPES[UA_TYPES_STRING]);
  215. UA_Array_delete(serverOnNetwork, serverOnNetworkSize, &UA_TYPES[UA_TYPES_SERVERONNETWORK]);
  216. return UA_STATUSCODE_GOOD;
  217. }
  218. static void initUaRegisterServer(UA_RegisteredServer *requestServer) {
  219. requestServer->isOnline = UA_TRUE;
  220. requestServer->serverUri = server->config.applicationDescription.applicationUri;
  221. requestServer->productUri = server->config.applicationDescription.productUri;
  222. requestServer->serverType = server->config.applicationDescription.applicationType;
  223. requestServer->gatewayServerUri = server->config.applicationDescription.gatewayServerUri;
  224. // create the semaphore
  225. int fd = open("/tmp/open62541-corpus-semaphore", O_RDWR|O_CREAT, S_IRWXU | S_IRWXG | S_IRWXO);
  226. close(fd);
  227. requestServer->semaphoreFilePath = UA_STRING_ALLOC("/tmp/open62541-corpus-semaphore");
  228. requestServer->serverNames = &server->config.applicationDescription.applicationName;
  229. requestServer->serverNamesSize = 1;
  230. size_t nl_discurls = server->config.networkLayersSize;
  231. requestServer->discoveryUrls = (UA_String*)UA_malloc(sizeof(UA_String) * nl_discurls);
  232. requestServer->discoveryUrlsSize = nl_discurls;
  233. for(size_t i = 0; i < nl_discurls; ++i) {
  234. UA_ServerNetworkLayer *nl = &server->config.networkLayers[i];
  235. requestServer->discoveryUrls[i] = nl->discoveryUrl;
  236. }
  237. }
  238. static UA_StatusCode registerServerRequest(UA_Client *client) {
  239. /* Prepare the request. Do not cleanup the request after the service call,
  240. * as the members are stack-allocated or point into the server config. */
  241. UA_RegisterServerRequest request;
  242. UA_RegisterServerRequest_init(&request);
  243. /* Copy from RegisterServer2 request */
  244. request.requestHeader.timestamp = UA_DateTime_now();
  245. request.requestHeader.timeoutHint = 10000;
  246. initUaRegisterServer(&request.server);
  247. UA_RegisterServerResponse response;
  248. UA_RegisterServerResponse_init(&response);
  249. __UA_Client_Service(client, &request,
  250. &UA_TYPES[UA_TYPES_REGISTERSERVERREQUEST],
  251. &response,
  252. &UA_TYPES[UA_TYPES_REGISTERSERVERRESPONSE]);
  253. UA_free(request.server.discoveryUrls);
  254. ASSERT_GOOD(response.responseHeader.serviceResult);
  255. return UA_STATUSCODE_GOOD;
  256. }
  257. static UA_StatusCode registerServer2Request(UA_Client *client) {
  258. /* Prepare the request. Do not cleanup the request after the service call,
  259. * as the members are stack-allocated or point into the server config. */
  260. UA_RegisterServer2Request request;
  261. UA_RegisterServer2Request_init(&request);
  262. request.requestHeader.timestamp = UA_DateTime_now();
  263. request.requestHeader.timeoutHint = 10000;
  264. initUaRegisterServer(&request.server);
  265. UA_MdnsDiscoveryConfiguration mdnsConfig;
  266. UA_MdnsDiscoveryConfiguration_init(&mdnsConfig);
  267. request.discoveryConfigurationSize = 1;
  268. request.discoveryConfiguration = UA_ExtensionObject_new();
  269. UA_ExtensionObject_init(&request.discoveryConfiguration[0]);
  270. request.discoveryConfiguration[0].encoding = UA_EXTENSIONOBJECT_DECODED_NODELETE;
  271. request.discoveryConfiguration[0].content.decoded.type = &UA_TYPES[UA_TYPES_MDNSDISCOVERYCONFIGURATION];
  272. request.discoveryConfiguration[0].content.decoded.data = &mdnsConfig;
  273. mdnsConfig.mdnsServerName = server->config.mdnsServerName;
  274. mdnsConfig.serverCapabilities = server->config.serverCapabilities;
  275. mdnsConfig.serverCapabilitiesSize = server->config.serverCapabilitiesSize;
  276. // First try with RegisterServer2, if that isn't implemented, use RegisterServer
  277. UA_RegisterServer2Response response;
  278. UA_RegisterServer2Response_init(&response);
  279. __UA_Client_Service(client, &request, &UA_TYPES[UA_TYPES_REGISTERSERVER2REQUEST],
  280. &response, &UA_TYPES[UA_TYPES_REGISTERSERVER2RESPONSE]);
  281. ASSERT_GOOD(response.responseHeader.serviceResult);
  282. UA_free(request.server.discoveryUrls);
  283. UA_ExtensionObject_delete(request.discoveryConfiguration);
  284. return UA_STATUSCODE_GOOD;
  285. }
  286. static UA_StatusCode readValueRequest(UA_Client *client) {
  287. UA_ReadValueId rvi;
  288. UA_ReadValueId_init(&rvi);
  289. rvi.nodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_SERVER_SERVERSTATUS_STARTTIME);
  290. rvi.attributeId = UA_ATTRIBUTEID_VALUE;
  291. UA_DataValue resp = UA_Server_read(server, &rvi, UA_TIMESTAMPSTORETURN_BOTH);
  292. ASSERT_GOOD(resp.status);
  293. UA_DataValue_deleteMembers(&resp);
  294. return UA_STATUSCODE_GOOD;
  295. }
  296. static UA_StatusCode writeValueRequest(UA_Client *client) {
  297. UA_WriteValue wValue;
  298. UA_WriteValue_init(&wValue);
  299. UA_LocalizedText testValue = UA_LOCALIZEDTEXT("en-EN", "MyServer");
  300. UA_Variant_setScalar(&wValue.value.value, &testValue, &UA_TYPES[UA_TYPES_LOCALIZEDTEXT]);
  301. wValue.value.hasValue = true;
  302. wValue.nodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_SERVER);
  303. wValue.attributeId = UA_ATTRIBUTEID_DISPLAYNAME;
  304. ASSERT_GOOD(UA_Server_write(server, &wValue));
  305. return UA_STATUSCODE_GOOD;
  306. }
  307. static UA_StatusCode browseAndBrowseNextRequest(UA_Client *client) {
  308. // Browse node in server folder
  309. UA_BrowseRequest bReq;
  310. UA_BrowseRequest_init(&bReq);
  311. // normally is set to 0, to get all the nodes, but we want to test browse next
  312. bReq.requestedMaxReferencesPerNode = 1;
  313. bReq.nodesToBrowse = UA_BrowseDescription_new();
  314. bReq.nodesToBrowseSize = 1;
  315. bReq.nodesToBrowse[0].nodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_SERVER);
  316. bReq.nodesToBrowse[0].resultMask = UA_BROWSERESULTMASK_ALL;
  317. UA_BrowseResponse bResp = UA_Client_Service_browse(client, bReq);
  318. ASSERT_GOOD(bResp.responseHeader.serviceResult);
  319. // browse next
  320. UA_BrowseNextRequest bNextReq;
  321. UA_BrowseNextRequest_init(&bNextReq);
  322. // normally is set to 0, to get all the nodes, but we want to test browse next
  323. bNextReq.releaseContinuationPoints = UA_FALSE;
  324. bNextReq.continuationPoints = &bResp.results[0].continuationPoint;
  325. bNextReq.continuationPointsSize = 1;
  326. UA_BrowseNextResponse bNextResp = UA_Client_Service_browseNext(client, bNextReq);
  327. ASSERT_GOOD(bNextResp.responseHeader.serviceResult);
  328. UA_BrowseNextResponse_deleteMembers(&bNextResp);
  329. bNextResp = UA_Client_Service_browseNext(client, bNextReq);
  330. ASSERT_GOOD(bNextResp.responseHeader.serviceResult);
  331. UA_BrowseNextResponse_deleteMembers(&bNextResp);
  332. // release continuation point. Result is then empty
  333. bNextReq.releaseContinuationPoints = UA_TRUE;
  334. bNextResp = UA_Client_Service_browseNext(client, bNextReq);
  335. UA_BrowseNextResponse_deleteMembers(&bNextResp);
  336. ASSERT_GOOD(bNextResp.responseHeader.serviceResult);
  337. UA_BrowseRequest_deleteMembers(&bReq);
  338. UA_BrowseResponse_deleteMembers(&bResp);
  339. // already deleted by browse request
  340. bNextReq.continuationPoints = NULL;
  341. bNextReq.continuationPointsSize = 0;
  342. UA_BrowseNextRequest_deleteMembers(&bNextReq);
  343. return UA_STATUSCODE_GOOD;
  344. }
  345. static UA_StatusCode registerUnregisterNodesRequest(UA_Client *client) {
  346. UA_RegisterNodesRequest req;
  347. UA_RegisterNodesRequest_init(&req);
  348. req.nodesToRegister = UA_NodeId_new();
  349. req.nodesToRegister[0] = UA_NODEID_NUMERIC(0, UA_NS0ID_SERVER_SERVERSTATUS_CURRENTTIME);
  350. req.nodesToRegisterSize = 1;
  351. UA_RegisterNodesResponse res = UA_Client_Service_registerNodes(client, req);
  352. ASSERT_GOOD(res.responseHeader.serviceResult);
  353. UA_UnregisterNodesRequest reqUn;
  354. UA_UnregisterNodesRequest_init(&reqUn);
  355. reqUn.nodesToUnregister = UA_NodeId_new();
  356. reqUn.nodesToUnregister[0] = res.registeredNodeIds[0];
  357. reqUn.nodesToUnregisterSize = 1;
  358. UA_UnregisterNodesResponse resUn = UA_Client_Service_unregisterNodes(client, reqUn);
  359. ASSERT_GOOD(resUn.responseHeader.serviceResult);
  360. UA_UnregisterNodesRequest_deleteMembers(&reqUn);
  361. UA_UnregisterNodesResponse_deleteMembers(&resUn);
  362. UA_RegisterNodesRequest_deleteMembers(&req);
  363. UA_RegisterNodesResponse_deleteMembers(&res);
  364. return UA_STATUSCODE_GOOD;
  365. }
  366. static UA_StatusCode translateBrowsePathsToNodeIdsRequest(UA_Client *client) {
  367. // Just for testing we want to translate the following path to its corresponding node id
  368. // /Objects/Server/ServerStatus/State
  369. // Equals the following node IDs:
  370. // /85/2253/2256/2259
  371. #define BROWSE_PATHS_SIZE 3
  372. char *paths[BROWSE_PATHS_SIZE] = {"Server", "ServerStatus", "State"};
  373. UA_UInt32 ids[BROWSE_PATHS_SIZE] = {UA_NS0ID_ORGANIZES, UA_NS0ID_HASCOMPONENT, UA_NS0ID_HASCOMPONENT};
  374. UA_BrowsePath browsePath;
  375. UA_BrowsePath_init(&browsePath);
  376. browsePath.startingNode = UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER);
  377. browsePath.relativePath.elements = (UA_RelativePathElement*)UA_Array_new(BROWSE_PATHS_SIZE, &UA_TYPES[UA_TYPES_RELATIVEPATHELEMENT]);
  378. browsePath.relativePath.elementsSize = BROWSE_PATHS_SIZE;
  379. for(size_t i = 0; i < BROWSE_PATHS_SIZE; i++) {
  380. UA_RelativePathElement *elem = &browsePath.relativePath.elements[i];
  381. elem->referenceTypeId = UA_NODEID_NUMERIC(0, ids[i]);
  382. elem->targetName = UA_QUALIFIEDNAME_ALLOC(0, paths[i]);
  383. }
  384. UA_TranslateBrowsePathsToNodeIdsRequest request;
  385. UA_TranslateBrowsePathsToNodeIdsRequest_init(&request);
  386. request.browsePaths = &browsePath;
  387. request.browsePathsSize = 1;
  388. UA_TranslateBrowsePathsToNodeIdsResponse response = UA_Client_Service_translateBrowsePathsToNodeIds(client, request);
  389. ASSERT_GOOD(response.responseHeader.serviceResult);
  390. UA_BrowsePath_deleteMembers(&browsePath);
  391. UA_TranslateBrowsePathsToNodeIdsResponse_deleteMembers(&response);
  392. return UA_STATUSCODE_GOOD;
  393. }
  394. static void monitoredItemHandler(UA_UInt32 monId, UA_DataValue *value, void *context) {
  395. }
  396. static UA_StatusCode subscriptionRequests(UA_Client *client) {
  397. UA_UInt32 subId;
  398. // createSubscriptionRequest
  399. ASSERT_GOOD(UA_Client_Subscriptions_new(client, UA_SubscriptionSettings_default, &subId));
  400. // modifySubscription
  401. UA_ModifySubscriptionRequest modifySubscriptionRequest;
  402. UA_ModifySubscriptionRequest_init(&modifySubscriptionRequest);
  403. modifySubscriptionRequest.subscriptionId = subId;
  404. modifySubscriptionRequest.maxNotificationsPerPublish = UA_SubscriptionSettings_default.maxNotificationsPerPublish;
  405. modifySubscriptionRequest.priority = UA_SubscriptionSettings_default.priority;
  406. modifySubscriptionRequest.requestedLifetimeCount = UA_SubscriptionSettings_default.requestedLifetimeCount;
  407. modifySubscriptionRequest.requestedMaxKeepAliveCount = UA_SubscriptionSettings_default.requestedMaxKeepAliveCount;
  408. modifySubscriptionRequest.requestedPublishingInterval = UA_SubscriptionSettings_default.requestedPublishingInterval;
  409. UA_ModifySubscriptionResponse modifySubscriptionResponse;
  410. __UA_Client_Service(client, &modifySubscriptionRequest, &UA_TYPES[UA_TYPES_MODIFYSUBSCRIPTIONREQUEST],
  411. &modifySubscriptionResponse, &UA_TYPES[UA_TYPES_MODIFYSUBSCRIPTIONRESPONSE]);
  412. ASSERT_GOOD(modifySubscriptionResponse.responseHeader.serviceResult);
  413. UA_ModifySubscriptionRequest_deleteMembers(&modifySubscriptionRequest);
  414. UA_ModifySubscriptionResponse_deleteMembers(&modifySubscriptionResponse);
  415. // setPublishingMode
  416. UA_SetPublishingModeRequest setPublishingModeRequest;
  417. UA_SetPublishingModeRequest_init(&setPublishingModeRequest);
  418. setPublishingModeRequest.subscriptionIdsSize = 1;
  419. setPublishingModeRequest.subscriptionIds = UA_malloc(sizeof(UA_UInt32));
  420. setPublishingModeRequest.subscriptionIds[0] = subId;
  421. setPublishingModeRequest.publishingEnabled = UA_TRUE;
  422. UA_SetPublishingModeResponse setPublishingModeResponse;
  423. __UA_Client_Service(client, &setPublishingModeRequest, &UA_TYPES[UA_TYPES_SETPUBLISHINGMODEREQUEST],
  424. &setPublishingModeResponse, &UA_TYPES[UA_TYPES_SETPUBLISHINGMODERESPONSE]);
  425. ASSERT_GOOD(setPublishingModeResponse.responseHeader.serviceResult);
  426. UA_SetPublishingModeRequest_deleteMembers(&setPublishingModeRequest);
  427. UA_SetPublishingModeResponse_deleteMembers(&setPublishingModeResponse);
  428. // createMonitoredItemsRequest
  429. UA_UInt32 monId;
  430. ASSERT_GOOD(UA_Client_Subscriptions_addMonitoredItem(client, subId, UA_NODEID_NUMERIC(0, 2259),
  431. UA_ATTRIBUTEID_VALUE, monitoredItemHandler,
  432. NULL, &monId));
  433. // publishRequest
  434. ASSERT_GOOD(UA_Client_Subscriptions_manuallySendPublishRequest(client));
  435. // republishRequest
  436. UA_RepublishRequest republishRequest;
  437. UA_RepublishRequest_init(&republishRequest);
  438. republishRequest.retransmitSequenceNumber = 0;
  439. republishRequest.subscriptionId = subId;
  440. UA_RepublishResponse republishResponse;
  441. __UA_Client_Service(client, &republishRequest, &UA_TYPES[UA_TYPES_REPUBLISHREQUEST],
  442. &republishResponse, &UA_TYPES[UA_TYPES_REPUBLISHRESPONSE]);
  443. // here we don't care about the return value since it may be UA_STATUSCODE_BADMESSAGENOTAVAILABLE
  444. // ASSERT_GOOD(republishResponse.responseHeader.serviceResult);
  445. UA_RepublishRequest_deleteMembers(&republishRequest);
  446. UA_RepublishResponse_deleteMembers(&republishResponse);
  447. // modifyMonitoredItems
  448. UA_ModifyMonitoredItemsRequest modifyMonitoredItemsRequest;
  449. UA_ModifyMonitoredItemsRequest_init(&modifyMonitoredItemsRequest);
  450. modifyMonitoredItemsRequest.subscriptionId = subId;
  451. modifyMonitoredItemsRequest.itemsToModifySize = 1;
  452. modifyMonitoredItemsRequest.itemsToModify = UA_malloc(sizeof(UA_MonitoredItemModifyRequest));
  453. modifyMonitoredItemsRequest.itemsToModify[0].monitoredItemId = monId;
  454. UA_MonitoringParameters_init(&modifyMonitoredItemsRequest.itemsToModify[0].requestedParameters);
  455. UA_ModifyMonitoredItemsResponse modifyMonitoredItemsResponse;
  456. __UA_Client_Service(client, &modifyMonitoredItemsRequest, &UA_TYPES[UA_TYPES_MODIFYMONITOREDITEMSREQUEST],
  457. &modifyMonitoredItemsResponse, &UA_TYPES[UA_TYPES_MODIFYMONITOREDITEMSRESPONSE]);
  458. ASSERT_GOOD(modifyMonitoredItemsResponse.responseHeader.serviceResult);
  459. UA_ModifyMonitoredItemsRequest_deleteMembers(&modifyMonitoredItemsRequest);
  460. UA_ModifyMonitoredItemsResponse_deleteMembers(&modifyMonitoredItemsResponse);
  461. // setMonitoringMode
  462. UA_SetMonitoringModeRequest setMonitoringModeRequest;
  463. UA_SetMonitoringModeRequest_init(&setMonitoringModeRequest);
  464. setMonitoringModeRequest.subscriptionId = subId;
  465. setMonitoringModeRequest.monitoredItemIdsSize = 1;
  466. setMonitoringModeRequest.monitoredItemIds = UA_malloc(sizeof(UA_UInt32));
  467. setMonitoringModeRequest.monitoredItemIds[0] = monId;
  468. setMonitoringModeRequest.monitoringMode = UA_MONITORINGMODE_REPORTING;
  469. UA_SetMonitoringModeResponse setMonitoringModeResponse;
  470. __UA_Client_Service(client, &setMonitoringModeRequest, &UA_TYPES[UA_TYPES_SETMONITORINGMODEREQUEST],
  471. &setMonitoringModeResponse, &UA_TYPES[UA_TYPES_SETMONITORINGMODERESPONSE]);
  472. ASSERT_GOOD(setMonitoringModeResponse.responseHeader.serviceResult);
  473. UA_SetMonitoringModeRequest_deleteMembers(&setMonitoringModeRequest);
  474. UA_SetMonitoringModeResponse_deleteMembers(&setMonitoringModeResponse);
  475. // deleteMonitoredItemsRequest
  476. ASSERT_GOOD(UA_Client_Subscriptions_removeMonitoredItem(client, subId, monId));
  477. // deleteSubscriptionRequest
  478. ASSERT_GOOD(UA_Client_Subscriptions_remove(client, subId));
  479. return UA_STATUSCODE_GOOD;
  480. }
  481. static UA_StatusCode callRequest(UA_Client *client) {
  482. /* Set up the request */
  483. UA_CallRequest request;
  484. UA_CallRequest_init(&request);
  485. UA_CallMethodRequest item;
  486. UA_CallMethodRequest_init(&item);
  487. item.methodId = UA_NODEID_NUMERIC(0, UA_NS0ID_SERVER_GETMONITOREDITEMS);
  488. item.objectId = UA_NODEID_NUMERIC(0, UA_NS0ID_SERVER);
  489. UA_Variant input;
  490. UA_UInt32 subId = 12345;
  491. UA_Variant_init(&input);
  492. UA_Variant_setScalarCopy(&input, &subId, &UA_TYPES[UA_TYPES_UINT32]);
  493. item.inputArgumentsSize = 1;
  494. item.inputArguments = &input;
  495. request.methodsToCall = &item;
  496. request.methodsToCallSize = 1;
  497. /* Call the service */
  498. UA_CallResponse response = UA_Client_Service_call(client, request);
  499. ASSERT_GOOD(response.responseHeader.serviceResult);
  500. UA_CallResponse_deleteMembers(&response);
  501. return UA_STATUSCODE_GOOD;
  502. }
  503. static UA_StatusCode nodemanagementRequests(UA_Client *client) {
  504. UA_ObjectAttributes attr = UA_ObjectAttributes_default;
  505. attr.description = UA_LOCALIZEDTEXT("en-US", "Some Coordinates");
  506. attr.displayName = UA_LOCALIZEDTEXT("en-US", "Coordinates");
  507. UA_NodeId newObjectId;
  508. ASSERT_GOOD(UA_Client_addObjectNode(client, UA_NODEID_NULL,
  509. UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER),
  510. UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES),
  511. UA_QUALIFIEDNAME(1, "Coordinates"),
  512. UA_NODEID_NUMERIC(0, UA_NS0ID_FOLDERTYPE), attr, &newObjectId));
  513. UA_ExpandedNodeId target = UA_EXPANDEDNODEID_NULL;
  514. target.nodeId = newObjectId;
  515. ASSERT_GOOD(UA_Client_addReference(client, UA_NODEID_NUMERIC(0, UA_NS0ID_ROOTFOLDER), UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES),
  516. UA_TRUE, UA_STRING_NULL, target, UA_NODECLASS_OBJECT));
  517. ASSERT_GOOD(UA_Client_deleteReference(client, UA_NODEID_NUMERIC(0, UA_NS0ID_ROOTFOLDER),
  518. UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES),
  519. true, target, true));
  520. ASSERT_GOOD(UA_Client_deleteNode(client, newObjectId, UA_TRUE));
  521. return UA_STATUSCODE_GOOD;
  522. }
  523. static UA_StatusCode executeClientServices(UA_Client *client) {
  524. ASSERT_GOOD(findServersRequest(client));
  525. ASSERT_GOOD(findServersOnNetworkRequest(client));
  526. ASSERT_GOOD(registerServerRequest(client));
  527. ASSERT_GOOD(registerServer2Request(client));
  528. ASSERT_GOOD(readValueRequest(client));
  529. ASSERT_GOOD(writeValueRequest(client));
  530. ASSERT_GOOD(browseAndBrowseNextRequest(client));
  531. ASSERT_GOOD(registerUnregisterNodesRequest(client));
  532. ASSERT_GOOD(translateBrowsePathsToNodeIdsRequest(client));
  533. ASSERT_GOOD(subscriptionRequests(client));
  534. ASSERT_GOOD(callRequest(client));
  535. ASSERT_GOOD(nodemanagementRequests(client));
  536. return UA_STATUSCODE_GOOD;
  537. }
  538. int main(void) {
  539. emptyCorpusDir();
  540. start_server();
  541. UA_StatusCode retval;
  542. UA_Client *client = UA_Client_new(UA_ClientConfig_default);
  543. // this will also call getEndpointsRequest
  544. retval = UA_Client_connect(client, "opc.tcp://localhost:4840");
  545. if (retval == UA_STATUSCODE_GOOD)
  546. retval = executeClientServices(client);
  547. UA_Client_disconnect(client);
  548. UA_Client_delete(client);
  549. if (retval == UA_STATUSCODE_GOOD) {
  550. // now also connect with user/pass so that fuzzer also knows how to do that
  551. client = UA_Client_new(UA_ClientConfig_default);
  552. retval = UA_Client_connect_username(client, "opc.tcp://localhost:4840", "user", "password");
  553. retval = retval == UA_STATUSCODE_BADUSERACCESSDENIED ? UA_STATUSCODE_GOOD : retval;
  554. UA_Client_disconnect(client);
  555. UA_Client_delete(client);
  556. }
  557. teardown_server();
  558. if (retval != UA_STATUSCODE_GOOD) {
  559. printf("\n--------- AN ERROR OCCURED ----------\nStatus = %s\n", UA_StatusCode_name(retval));
  560. } else {
  561. printf("\n--------- SUCCESS -------\nThe corpus is stored in %s\nYou manually need to copy them into the direcotry tests/fuss/fuzz_binary_message_corpus/generated", UA_CORPUS_OUTPUT_DIR);
  562. }
  563. return (int) retval;
  564. }