#include <uaclient_config.h>
#include <stdlib.h>
#include <opcua_datetime.h>
#include <opcua_trace.h>
#include <opcua_errorhandling.h>
#include <opcua_memory.h>
#include <opcua_string.h>
#include <uaclient_module.h>
#include <uaclient_session.h>
#include <uabase_utilities.h>
#include "samplesubscription.h"
#define UACLIENT_APPLICATION_NAME "UaSdkC - Client - Lesson06"
#define UACLIENT_APPLICATION_URI "urn:UnifiedAutomation:UaSdkC:Client:Lesson06"
#define UACLIENT_PRODUCT_URI "urn:UnifiedAutomation:UaSdkC:Client:Lesson06"
#define SERVER_ENDPOINT_URL "opc.tcp://localhost:48020"
typedef enum _SampleStateMachine
{
State_Idle,
State_Connect,
State_Connected,
State_Subscription,
State_MethodCall,
State_MethodCallDone,
State_Disconnect,
State_Disconnected,
State_Error
} SampleStateMachine;
typedef struct _SampleClientContext
{
SampleStateMachine State;
} SampleClientContext;
OpcUa_ProxyStubConfiguration *a_pProxyStubConfiguration)
{
OpcUa_InitializeStatus(OpcUa_Module_Client, "InitializeOpcUaStack");
printf("UA Client: Initializing Stack...\n");
a_pProxyStubConfiguration->bProxyStub_Trace_Enabled = OpcUa_True;
a_pProxyStubConfiguration->uProxyStub_Trace_Level = OPCUA_TRACE_OUTPUT_LEVEL_WARNING;
OpcUa_ReturnStatusCode;
OpcUa_BeginErrorHandling;
OpcUa_FinishErrorHandling;
}
{
OpcUa_InitializeStatus(OpcUa_Module_Client, "CleanupOpcUaStack");
OpcUa_ReturnStatusCode;
OpcUa_BeginErrorHandling;
OpcUa_FinishErrorHandling;
}
{
SampleClientContext *pClientContext = a_pSession->
pUserData;
const char *pStatus = "INVALID";
switch (a_status)
{
pStatus = "Disconnected";
if (pClientContext->State == State_Connect)
{
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "UA Client: failed to connect to server!\n");
}
if (pClientContext->State == State_Disconnect)
{
pClientContext->State = State_Disconnected;
}
else
{
pClientContext->State = State_Error;
}
break;
pStatus = "Connected";
if (pClientContext->State == State_Connect || pClientContext->State == State_Subscription)
{
pClientContext->State = State_Connected;
}
else
{
pClientContext->State = State_Error;
}
break;
pStatus = "Connecting";
break;
pStatus = "ConnectionWarningWatchdogTimeout";
break;
pStatus = "ConnectionErrorClientReconnect";
break;
pStatus = "SessionAutomaticallyRecreated";
break;
default: break;
}
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "\n--> Sample_ConnectionStatusChanged_CB: %s\n\n", pStatus);
}
OpcUa_Boolean a_overridable)
{
const char *pServiceType = "INVALID";
OpcUa_Boolean bRet = OpcUa_False;
OpcUa_ReferenceParameter(a_pSession);
switch (a_serviceType)
{
default: break;
}
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "Sample_ConnectError_CB:\n");
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, " ServiceType: %s\n", pServiceType);
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, " Error: 0x%08x\n", a_error);
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, " Overridable: %s\n", a_overridable == OpcUa_False ? "false" : "true");
if (a_overridable)
{
bRet = OpcUa_True;
}
return bRet;
}
OpcUa_Void *a_pUserData)
{
SampleClientContext *pClientContext = a_pSession->
pUserData;
OpcUa_ReferenceParameter(a_pResult);
OpcUa_ReferenceParameter(a_pDiagnosticInfo);
OpcUa_ReferenceParameter(a_pUserData);
{
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "Sample_MethodCall_CB succeeded\n");
if (pClientContext->State == State_MethodCall)
{
pClientContext->State = State_MethodCallDone;
}
else
{
pClientContext->State = State_Error;
}
}
else
{
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING,
"Sample_MethodCall_CB failed (0x%08x)\n", a_pResponseHeader->
ServiceResult);
pClientContext->State = State_Error;
}
}
{
OpcUa_InitializeStatus(OpcUa_Module_Client, "Sample_MethodCall");
OpcUa_CallMethodRequest_Initialize(&callMethodRequest);
callMethodRequest.NoOfInputArguments = 1;
OpcUa_Null,
&callMethodRequest,
Sample_MethodCall_CB,
OpcUa_Null);
OpcUa_GotoErrorIfBad(uStatus);
OpcUa_CallMethodRequest_Clear(&callMethodRequest);
OpcUa_ReturnStatusCode;
OpcUa_BeginErrorHandling;
OpcUa_CallMethodRequest_Clear(&callMethodRequest);
OpcUa_FinishErrorHandling;
}
{
OpcUa_Boolean bClientInitialized = OpcUa_False;
OpcUa_Boolean bComplete = OpcUa_False;
SampleClientContext clientContext;
SampleSubscription *pSampleSubscription = OpcUa_Null;
OpcUa_InitializeStatus(OpcUa_Module_Client, "ClientMain");
OpcUa_DeclareErrorTraceModule(OpcUa_Module_Client);
clientContext.State = State_Idle;
OpcUa_GotoErrorIfBad(uStatus);
bClientInitialized = OpcUa_True;
pClientConfiguration->
PkiConfig.strPkiType = (
char*)OPCUA_PKI_TYPE_NONE;
OpcUa_MemSet(&sessionCallback, 0, sizeof(sessionCallback));
sessionCallback.pfConnectionStatusChanged_CB = Sample_ConnectionStatusChanged_CB;
sessionCallback.pfConnectError_CB = Sample_ConnectError_CB;
OpcUa_GotoErrorIfBad(uStatus);
OpcUa_GotoErrorIfBad(uStatus);
{
switch (clientContext.State)
{
case State_Idle:
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "UA Client: Connecting to %s ...\n", SERVER_ENDPOINT_URL);
OpcUa_GotoErrorIfBad(uStatus);
clientContext.State = State_Connect;
break;
case State_Connected:
pSampleSubscription = SampleSubscription_Create(pSession);
OpcUa_GotoErrorIfArgumentNull(pSampleSubscription);
uStatus = SampleSubscription_CreateSubscription(pSampleSubscription);
OpcUa_GotoErrorIfBad(uStatus);
clientContext.State = State_Subscription;
break;
case State_Subscription:
switch (pSampleSubscription->State)
{
case State_SampleSubscription_SubscriptionCreateDone:
uStatus = SampleSubscription_CreateMonitoredItem(pSampleSubscription);
OpcUa_GotoErrorIfBad(uStatus);
break;
case State_SampleSubscription_MonitoredItemCreateDone:
uStatus = Sample_MethodCall(pSession);
OpcUa_GotoErrorIfBad(uStatus);
pSampleSubscription->State = State_SampleSubscription_WaitForNewEvent;
clientContext.State = State_MethodCall;
break;
case State_SampleSubscription_NewEventReceived:
uStatus = SampleSubscription_DeleteMonitoredItem(pSampleSubscription);
OpcUa_GotoErrorIfBad(uStatus);
break;
case State_SampleSubscription_MonitoredItemDeleteDone:
uStatus = SampleSubscription_DeleteSubscription(pSampleSubscription);
OpcUa_GotoErrorIfBad(uStatus);
break;
case State_SampleSubscription_SubscriptionDeleteDone:
SampleSubscription_Delete(&pSampleSubscription);
pSampleSubscription = OpcUa_Null;
OpcUa_GotoErrorIfBad(uStatus);
clientContext.State = State_Disconnect;
break;
case State_SampleSubscription_Error:
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "An error occured. Terminating now.\n");
bComplete = OpcUa_True;
break;
default:
break;
}
break;
case State_MethodCallDone:
clientContext.State = State_Subscription;
break;
case State_Disconnected:
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "Sample successfully completed. Terminating now.\n");
bComplete = OpcUa_True;
break;
case State_Error:
if (OpcUa_IsBad(uStatus) && uStatus != OpcUa_BadInvalidState) { OpcUa_GotoError; }
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "An error occured. Terminating now.\n");
bComplete = OpcUa_True;
break;
default:
break;
}
if (OpcUa_IsBad(uStatus))
{
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "UaBase_DoCom failed (0x%08x)\n", uStatus);
bComplete = OpcUa_True;
}
}
SampleSubscription_Delete(&pSampleSubscription);
OpcUa_CertificateStoreConfiguration_Initialize(&pClientConfiguration->
PkiConfig);
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "UA Client: Main stopped\n");
OpcUa_ReturnStatusCode;
OpcUa_BeginErrorHandling;
if (bClientInitialized != OpcUa_False)
{
}
OpcUa_Trace(OPCUA_TRACE_LEVEL_WARNING, "UA Client: Main stopped due ERROR! (0x%08x)\n", uStatus);
OpcUa_FinishErrorHandling;
}
int main(void)
{
int ret = EXIT_SUCCESS;
OpcUa_Handle hProxyStubPlatformLayer = OpcUa_Null;
OpcUa_ProxyStubConfiguration proxyStubConfiguration;
uStatus = InitializeOpcUaStack(&hProxyStubPlatformLayer, &proxyStubConfiguration);
if ( OpcUa_IsNotGood(uStatus) )
{
return EXIT_FAILURE;
}
uStatus = ClientMain();
if ( OpcUa_IsNotGood(uStatus) )
{
ret = EXIT_FAILURE;
}
uStatus = CleanupOpcUaStack(&hProxyStubPlatformLayer);
if ( OpcUa_IsNotGood(uStatus) )
{
ret = EXIT_FAILURE;
}
return ret;
}