blob: 68f985a650396c63d1dec9dd3087984532bfc10c [file] [log] [blame]
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include <open62541/server_config_default.h>
#include <open62541/plugin/log_stdout.h>
#include <open62541/client_config_default.h>
#include <open62541/client_highlevel.h>
#include <check.h>
#include "thread_wrapper.h"
/* While server initialization, value callbacks are called twice.
* This counter is used to ensure that the deletion of the variable is triggered by the client (not while the server initialization)*/
int counter = 0;
UA_Server *server;
UA_Boolean running;
UA_ServerNetworkLayer nl;
UA_NodeId temperatureNodeId = {1, UA_NODEIDTYPE_NUMERIC, {1001}};
UA_Int32 temperature;
UA_Boolean deleteNodeWhileWriting;
THREAD_HANDLE server_thread;
static void
updateCurrentTime(void) {
UA_DateTime now = UA_DateTime_now();
UA_Variant value;
UA_Variant_setScalar(&value, &now, &UA_TYPES[UA_TYPES_DATETIME]);
UA_NodeId currentNodeId = UA_NODEID_STRING(1, "current-time-value-callback");
UA_Server_writeValue(server, currentNodeId, value);
}
static void
addCurrentTimeVariable(void) {
UA_DateTime now = 0;
UA_VariableAttributes attr = UA_VariableAttributes_default;
attr.displayName = UA_LOCALIZEDTEXT("en-US", "Current time - value callback");
attr.accessLevel = UA_ACCESSLEVELMASK_READ | UA_ACCESSLEVELMASK_WRITE;
UA_Variant_setScalar(&attr.value, &now, &UA_TYPES[UA_TYPES_DATETIME]);
UA_NodeId currentNodeId = UA_NODEID_STRING(1, "current-time-value-callback");
UA_QualifiedName currentName = UA_QUALIFIEDNAME(1, "current-time-value-callback");
UA_NodeId parentNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER);
UA_NodeId parentReferenceNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES);
UA_NodeId variableTypeNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_BASEDATAVARIABLETYPE);
UA_Server_addVariableNode(server, currentNodeId, parentNodeId,
parentReferenceNodeId, currentName,
variableTypeNodeId, attr, NULL, NULL);
updateCurrentTime();
}
static void
beforeReadTime(UA_Server *tmpserver,
const UA_NodeId *sessionId, void *sessionContext,
const UA_NodeId *nodeid, void *nodeContext,
const UA_NumericRange *range, const UA_DataValue *data) {
updateCurrentTime();
}
static void
afterWriteTime(UA_Server *tmpServer,
const UA_NodeId *sessionId, void *sessionContext,
const UA_NodeId *nodeId, void *nodeContext,
const UA_NumericRange *range, const UA_DataValue *data) {
UA_LOG_INFO(UA_Log_Stdout, UA_LOGCATEGORY_USERLAND, "The variable was updated");
}
static void
addValueCallbackToCurrentTimeVariable(void) {
UA_NodeId currentNodeId = UA_NODEID_STRING(1, "current-time-value-callback");
UA_ValueCallback callback ;
callback.onRead = beforeReadTime;
callback.onWrite = afterWriteTime;
UA_Server_setVariableNode_valueCallback(server, currentNodeId, callback);
}
static UA_StatusCode
readTemperature(UA_Server *tmpServer,
const UA_NodeId *sessionId, void *sessionContext,
const UA_NodeId *nodeId, void *nodeContext,
UA_Boolean sourceTimeStamp, const UA_NumericRange *range,
UA_DataValue *dataValue) {
if (counter < 2)
counter++;
else
UA_Server_deleteNode(server, temperatureNodeId, true);
UA_Variant_setScalarCopy(&dataValue->value, &temperature, &UA_TYPES[UA_TYPES_INT32]);
dataValue->hasValue = true;
return UA_STATUSCODE_GOOD;
}
static UA_StatusCode
writeTemperature(UA_Server *tmpServer,
const UA_NodeId *sessionId, void *sessionContext,
const UA_NodeId *nodeId, void *nodeContext,
const UA_NumericRange *range, const UA_DataValue *data) {
temperature = *(UA_Int32 *) data->value.data;
if (deleteNodeWhileWriting)
UA_Server_deleteNode(server, temperatureNodeId, true);
return UA_STATUSCODE_GOOD;
}
static void
addDataSourceVariable(void) {
UA_VariableAttributes attr = UA_VariableAttributes_default;
attr.displayName = UA_LOCALIZEDTEXT("en-US", "Temperature");
attr.accessLevel = UA_ACCESSLEVELMASK_READ | UA_ACCESSLEVELMASK_WRITE;
UA_DataSource temperatureSource;
temperatureSource.read = readTemperature;
temperatureSource.write = writeTemperature;
UA_StatusCode retval = UA_Server_addDataSourceVariableNode(server, temperatureNodeId, UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER),
UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES), UA_QUALIFIEDNAME(1, "Temperature"),
UA_NODEID_NUMERIC(0, UA_NS0ID_BASEDATAVARIABLETYPE), attr,
temperatureSource, NULL, NULL);
ck_assert_int_eq(retval, UA_STATUSCODE_GOOD);
}
THREAD_CALLBACK(serverloop) {
while(running)
UA_Server_run_iterate(server, true);
return 0;
}
static void setup(void) {
running = true;
server = UA_Server_new();
UA_ServerConfig_setDefault(UA_Server_getConfig(server));
UA_Server_run_startup(server);
addCurrentTimeVariable();
addValueCallbackToCurrentTimeVariable();
addDataSourceVariable();
THREAD_CREATE(server_thread, serverloop);
}
static void teardown(void) {
running = false;
counter = 0;
THREAD_JOIN(server_thread);
UA_Server_run_shutdown(server);
UA_Server_delete(server);
}
START_TEST(client_readValueCallbackAttribute) {
UA_Client *client = UA_Client_new();
UA_ClientConfig_setDefault(UA_Client_getConfig(client));
UA_StatusCode retval = UA_Client_connect(client, "opc.tcp://localhost:4840");
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
UA_Variant val;
UA_Variant_init(&val);
UA_NodeId nodeId = UA_NODEID_STRING(1, "current-time-value-callback");
retval = UA_Client_readValueAttribute(client, nodeId, &val);
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
UA_Variant_deleteMembers(&val);
UA_Client_disconnect(client);
UA_Client_delete(client);
}
END_TEST
START_TEST(client_readMultipleAttributes) {
UA_Client *client = UA_Client_new();
UA_ClientConfig_setDefault(UA_Client_getConfig(client));
UA_StatusCode retval = UA_Client_connect(client, "opc.tcp://localhost:4840");
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
UA_ReadRequest request;
UA_ReadRequest_init(&request);
UA_ReadValueId ids[3];
UA_ReadValueId_init(&ids[0]);
ids[0].attributeId = UA_ATTRIBUTEID_DESCRIPTION;
ids[0].nodeId = temperatureNodeId;
UA_ReadValueId_init(&ids[1]);
ids[1].attributeId = UA_ATTRIBUTEID_VALUE;
ids[1].nodeId = temperatureNodeId;
UA_ReadValueId_init(&ids[2]);
ids[2].attributeId = UA_ATTRIBUTEID_BROWSENAME;
ids[2].nodeId = temperatureNodeId;
request.nodesToRead = ids;
request.nodesToReadSize = 3;
UA_ReadResponse response = UA_Client_Service_read(client, request);
retval = response.responseHeader.serviceResult;
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
ck_assert_int_eq(response.resultsSize, 3);
ck_assert_uint_eq(response.results[0].status, UA_STATUSCODE_GOOD);
ck_assert_uint_eq(response.results[1].status, UA_STATUSCODE_GOOD);
ck_assert_uint_eq(response.results[2].status, UA_STATUSCODE_BADNODEIDUNKNOWN);
UA_ReadResponse_deleteMembers(&response);
UA_Client_disconnect(client);
UA_Client_delete(client);
}
END_TEST
START_TEST(client_writeValueCallbackAttribute) {
UA_Client *client = UA_Client_new();
UA_ClientConfig_setDefault(UA_Client_getConfig(client));
UA_StatusCode retval = UA_Client_connect(client, "opc.tcp://localhost:4840");
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
UA_Variant *val = UA_Variant_new();
UA_Int32 value = 77;
UA_Variant_setScalarCopy(val, &value, &UA_TYPES[UA_TYPES_INT32]);
retval = UA_Client_writeValueAttribute(client, temperatureNodeId, val);
#ifdef UA_ENABLE_IMMUTABLE_NODES
ck_assert_uint_eq(retval, UA_STATUSCODE_BADNODEIDUNKNOWN);
#else
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
#endif
UA_Variant_delete(val);
UA_Client_disconnect(client);
UA_Client_delete(client);
}
END_TEST
START_TEST(client_writeMultipleAttributes) {
UA_Client *client = UA_Client_new();
UA_ClientConfig_setDefault(UA_Client_getConfig(client));
UA_StatusCode retval = UA_Client_connect(client, "opc.tcp://localhost:4840");
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
UA_Int32 value1 = 23;
UA_WriteRequest wReq;
UA_WriteRequest_init(&wReq);
UA_LocalizedText string = UA_LOCALIZEDTEXT("en-US", "Temperature");
UA_WriteValue wv[2];
UA_WriteValue_init(&wv[0]);
UA_WriteValue_init(&wv[1]);
wReq.nodesToWrite = wv;
wReq.nodesToWriteSize = 2;
wReq.nodesToWrite[0].nodeId = temperatureNodeId;
wReq.nodesToWrite[0].attributeId = UA_ATTRIBUTEID_DISPLAYNAME;
wReq.nodesToWrite[0].value.hasValue = true;
wReq.nodesToWrite[0].value.value.type = &UA_TYPES[UA_TYPES_LOCALIZEDTEXT];
wReq.nodesToWrite[0].value.value.storageType = UA_VARIANT_DATA_NODELETE;
wReq.nodesToWrite[0].value.value.data = &string;
wReq.nodesToWrite[1].nodeId = temperatureNodeId;
wReq.nodesToWrite[1].attributeId = UA_ATTRIBUTEID_VALUE;
wReq.nodesToWrite[1].value.hasValue = true;
wReq.nodesToWrite[1].value.value.type = &UA_TYPES[UA_TYPES_INT32];
wReq.nodesToWrite[1].value.value.storageType = UA_VARIANT_DATA_NODELETE;
wReq.nodesToWrite[1].value.value.data = &value1;
UA_WriteResponse wResp = UA_Client_Service_write(client, wReq);
ck_assert_uint_eq(wResp.responseHeader.serviceResult,UA_STATUSCODE_GOOD);
ck_assert_uint_eq(retval, UA_STATUSCODE_GOOD);
UA_WriteResponse_clear(&wResp);
UA_Client_disconnect(client);
UA_Client_delete(client);
}
END_TEST
static Suite* testSuite_immutableNodes(void) {
Suite *s = suite_create("Immutable Nodes");
TCase *valueCallback = tcase_create("ValueCallback");
deleteNodeWhileWriting = UA_FALSE;
tcase_add_checked_fixture(valueCallback, setup, teardown);
tcase_add_test(valueCallback, client_readValueCallbackAttribute);
tcase_add_test(valueCallback, client_readMultipleAttributes);
deleteNodeWhileWriting = UA_TRUE;
tcase_add_test(valueCallback, client_writeValueCallbackAttribute);
tcase_add_test(valueCallback, client_writeMultipleAttributes);
suite_add_tcase(s,valueCallback);
return s;
}
int main(void) {
Suite *s = testSuite_immutableNodes();
SRunner *sr = srunner_create(s);
srunner_set_fork_status(sr, CK_NOFORK);
srunner_run_all(sr, CK_NORMAL);
int number_failed = srunner_ntests_failed(sr);
srunner_free(sr);
return (number_failed == 0) ? EXIT_SUCCESS : EXIT_FAILURE;
}