private async Task SendMethodAndRespondPoolOverAmqp( Client.TransportType transport, int poolSize, int devicesCount, Func <DeviceClient, string, Task <Task> > setDeviceReceiveMethod, ConnectionStringAuthScope authScope = ConnectionStringAuthScope.Device) { Func <DeviceClient, TestDevice, Task> initOperation = async(deviceClient, testDevice) => { _log.WriteLine($"{nameof(MethodE2EPoolAmqpTests)}: Setting method for device {testDevice.Id}"); Task methodReceivedTask = await setDeviceReceiveMethod(deviceClient, MethodName).ConfigureAwait(false); }; Func <DeviceClient, TestDevice, Task> testOperation = async(deviceClient, testDevice) => { _log.WriteLine($"{nameof(MethodE2EPoolAmqpTests)}: Preparing to receive method for device {testDevice.Id}"); await MethodE2ETests.ServiceSendMethodAndVerifyResponse( testDevice.Id, MethodName, MethodE2ETests.DeviceResponseJson, MethodE2ETests.ServiceRequestJson).ConfigureAwait(false); }; Func <IList <DeviceClient>, Task> cleanupOperation = async(deviceClients) => { foreach (DeviceClient deviceClient in deviceClients) { deviceClient.Dispose(); } await Task.FromResult <bool>(false).ConfigureAwait(false); }; await PoolingOverAmqp.TestPoolAmqpAsync( DevicePrefix, transport, poolSize, devicesCount, initOperation, testOperation, cleanupOperation, authScope, true).ConfigureAwait(false); }
private async Task SendMethodAndRespondRecoveryPoolOverAmqpAsync( TestDeviceType type, Client.TransportType transport, int poolSize, int devicesCount, Func <DeviceClient, string, Task <Task> > setDeviceReceiveMethod, string faultType, string reason, int delayInSec = FaultInjection.DefaultDelayInSec, int durationInSec = FaultInjection.DefaultDurationInSec, ConnectionStringAuthScope authScope = ConnectionStringAuthScope.Device) { var testDevicesWithCallbackHandler = new Dictionary <string, TestDeviceCallbackHandler>(); Func <DeviceClient, TestDevice, Task> initOperation = async(deviceClient, testDevice) => { var testDeviceCallbackHandler = new TestDeviceCallbackHandler(deviceClient); testDevicesWithCallbackHandler.Add(testDevice.Id, testDeviceCallbackHandler); _log.WriteLine($"{nameof(MethodE2EPoolAmqpTests)}: Setting method callback handler for device {testDevice.Id}"); await testDeviceCallbackHandler .SetDeviceReceiveMethodAsync(MethodName, MethodE2ETests.DeviceResponseJson, MethodE2ETests.ServiceRequestJson) .ConfigureAwait(false); }; Func <DeviceClient, TestDevice, Task> testOperation = async(deviceClient, testDevice) => { TestDeviceCallbackHandler testDeviceCallbackHandler = testDevicesWithCallbackHandler[testDevice.Id]; using var cts = new CancellationTokenSource(FaultInjection.RecoveryTimeMilliseconds); _log.WriteLine($"{nameof(MethodE2EPoolAmqpTests)}: Preparing to receive method for device {testDevice.Id}"); Task serviceSendTask = MethodE2ETests.ServiceSendMethodAndVerifyResponseAsync( testDevice.Id, MethodName, MethodE2ETests.DeviceResponseJson, MethodE2ETests.ServiceRequestJson); Task methodReceivedTask = testDeviceCallbackHandler.WaitForMethodCallbackAsync(cts.Token); await Task.WhenAll(serviceSendTask, methodReceivedTask).ConfigureAwait(false); }; Func <IList <DeviceClient>, Task> cleanupOperation = async(deviceClients) => { foreach (DeviceClient deviceClient in deviceClients) { deviceClient.Dispose(); } testDevicesWithCallbackHandler.Clear(); await Task.FromResult <bool>(false).ConfigureAwait(false); }; await FaultInjectionPoolingOverAmqp .TestFaultInjectionPoolAmqpAsync( MethodDevicePrefix, transport, poolSize, devicesCount, faultType, reason, delayInSec, durationInSec, initOperation, testOperation, cleanupOperation, authScope) .ConfigureAwait(false); }
private async Task DeviceCombinedClientOperationsAsync( Client.TransportType transport, int poolSize, int devicesCount, ConnectionStringAuthScope authScope = ConnectionStringAuthScope.Device) { // Initialize service client for service-side operations using var serviceClient = ServiceClient.CreateFromConnectionString(TestConfiguration.IoTHub.ConnectionString); // Message payload and properties for C2D operation var messagesSent = new Dictionary <string, Tuple <Message, string> >(); // Twin properties var twinPropertyMap = new Dictionary <string, List <string> >(); async Task InitOperationAsync(DeviceClient deviceClient, TestDevice testDevice, TestDeviceCallbackHandler _) { IList <Task> initOperations = new List <Task>(); // Send C2D Message Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Send C2D for device={testDevice.Id}"); (Message msg, string payload, string p1Value) = MessageReceiveE2ETests.ComposeC2dTestMessage(Logger); using (msg) { messagesSent.Add(testDevice.Id, Tuple.Create(msg, payload)); Task sendC2dMessage = serviceClient.SendAsync(testDevice.Id, msg); initOperations.Add(sendC2dMessage); // Set method handler Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Set direct method {MethodName} for device={testDevice.Id}"); Task <Task> methodReceivedTask = MethodE2ETests.SetDeviceReceiveMethodAsync(deviceClient, MethodName, Logger); initOperations.Add(methodReceivedTask); // Set the twin desired properties callback Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Set desired property callback for device={testDevice.Id}"); string propName = Guid.NewGuid().ToString(); string propValue = Guid.NewGuid().ToString(); twinPropertyMap.Add(testDevice.Id, new List <string> { propName, propValue }); Task <Task> updateReceivedTask = TwinE2ETests.SetTwinPropertyUpdateCallbackHandlerAsync(deviceClient, propName, propValue, Logger); initOperations.Add(updateReceivedTask); await Task.WhenAll(initOperations).ConfigureAwait(false); } } async Task TestOperationAsync(DeviceClient deviceClient, TestDevice testDevice, TestDeviceCallbackHandler _) { IList <Task> clientOperations = new List <Task>(); await deviceClient.OpenAsync().ConfigureAwait(false); // D2C Operation Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Operation 1: Send D2C for device={testDevice.Id}"); Task sendD2cMessage = MessageSendE2ETests.SendSingleMessageAsync(deviceClient, testDevice.Id, Logger); clientOperations.Add(sendD2cMessage); // C2D Operation Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Operation 2: Receive C2D for device={testDevice.Id}"); Tuple <Message, string> msgSent = messagesSent[testDevice.Id]; Message msg = msgSent.Item1; string payload = msgSent.Item2; Task verifyDeviceClientReceivesMessage = MessageReceiveE2ETests.VerifyReceivedC2DMessageAsync(transport, deviceClient, testDevice.Id, msg, payload, Logger); clientOperations.Add(verifyDeviceClientReceivesMessage); // Invoke direct methods Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Operation 3: Direct methods test for device={testDevice.Id}"); Task serviceInvokeMethod = MethodE2ETests.ServiceSendMethodAndVerifyResponseAsync(testDevice.Id, MethodName, MethodE2ETests.DeviceResponseJson, MethodE2ETests.ServiceRequestJson, Logger); clientOperations.Add(serviceInvokeMethod); // Set reported twin properties Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Operation 4: Set reported property for device={testDevice.Id}"); Task setReportedProperties = TwinE2ETests.Twin_DeviceSetsReportedPropertyAndGetsItBackAsync(deviceClient, testDevice.Id, Guid.NewGuid().ToString(), Logger); clientOperations.Add(setReportedProperties); // Receive set desired twin properties Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: Operation 5: Receive desired property for device={testDevice.Id}"); List <string> twinProperties = twinPropertyMap[testDevice.Id]; string propName = twinProperties[0]; string propValue = twinProperties[1]; Task updateDesiredProperties = TwinE2ETests.RegistryManagerUpdateDesiredPropertyAsync(testDevice.Id, propName, propValue); clientOperations.Add(updateDesiredProperties); await Task.WhenAll(clientOperations).ConfigureAwait(false); Logger.Trace($"{nameof(CombinedClientOperationsPoolAmqpTests)}: All operations completed for device={testDevice.Id}"); } Task CleanupOperationAsync() { messagesSent.Clear(); twinPropertyMap.Clear(); return(Task.FromResult(0)); } await PoolingOverAmqp .TestPoolAmqpAsync( _devicePrefix, transport, poolSize, devicesCount, InitOperationAsync, TestOperationAsync, CleanupOperationAsync, authScope, false, Logger) .ConfigureAwait(false); }
private async Task DeviceCombinedClientOperations( Client.TransportType transport, int poolSize, int devicesCount, ConnectionStringAuthScope authScope) { // Initialize service client for service-side operations ServiceClient serviceClient = ServiceClient.CreateFromConnectionString(Configuration.IoTHub.ConnectionString); // Message payload for C2D operation Dictionary <string, List <string> > messagesSent = new Dictionary <string, List <string> >(); // Twin properties Dictionary <string, List <string> > twinPropertyMap = new Dictionary <string, List <string> >(); Func <DeviceClient, TestDevice, Task> initOperation = async(deviceClient, testDevice) => { IList <Task> initOperations = new List <Task>(); // Send C2D Message _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Send C2D for device={testDevice.Id}"); (Message msg, string messageId, string payload, string p1Value) = MessageReceiveE2ETests.ComposeC2DTestMessage(); messagesSent.Add(testDevice.Id, new List <string> { payload, p1Value }); var sendC2DMessage = serviceClient.SendAsync(testDevice.Id, msg); initOperations.Add(sendC2DMessage); // Set method handler _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Set direct method {MethodName} for device={testDevice.Id}"); var methodReceivedTask = MethodE2ETests.SetDeviceReceiveMethod(deviceClient, MethodName); initOperations.Add(methodReceivedTask); // Set the twin desired properties callback _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Set desired property callback for device={testDevice.Id}"); var propName = Guid.NewGuid().ToString(); var propValue = Guid.NewGuid().ToString(); twinPropertyMap.Add(testDevice.Id, new List <string> { propName, propValue }); var updateReceivedTask = TwinE2ETests.SetTwinPropertyUpdateCallbackHandlerAsync(deviceClient, propName, propValue); initOperations.Add(updateReceivedTask); await Task.WhenAll(initOperations).ConfigureAwait(false); }; Func <DeviceClient, TestDevice, Task> testOperation = async(deviceClient, testDevice) => { IList <Task> clientOperations = new List <Task>(); await deviceClient.OpenAsync().ConfigureAwait(false); // D2C Operation _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Operation 1: Send D2C for device={testDevice.Id}"); var sendD2CMessage = MessageSendE2ETests.SendSingleMessageAndVerifyAsync(deviceClient, testDevice.Id); clientOperations.Add(sendD2CMessage); // C2D Operation _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Operation 2: Receive C2D for device={testDevice.Id}"); List <string> msgSent = messagesSent[testDevice.Id]; var payload = msgSent[0]; var p1Value = msgSent[1]; var verifyDeviceClientReceivesMessage = MessageReceiveE2ETests.VerifyReceivedC2DMessageAsync(transport, deviceClient, payload, p1Value); clientOperations.Add(verifyDeviceClientReceivesMessage); // Invoke direct methods _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Operation 3: Direct methods test for device={testDevice.Id}"); var serviceInvokeMethod = MethodE2ETests.ServiceSendMethodAndVerifyResponse(testDevice.Id, MethodName, MethodE2ETests.DeviceResponseJson, MethodE2ETests.ServiceRequestJson); clientOperations.Add(serviceInvokeMethod); // Set reported twin properties _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Operation 4: Set reported property for device={testDevice.Id}"); var setReportedProperties = TwinE2ETests.Twin_DeviceSetsReportedPropertyAndGetsItBack(deviceClient); clientOperations.Add(setReportedProperties); // Receive set desired twin properties _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: Operation 5: Receive desired property for device={testDevice.Id}"); List <string> twinProperties = twinPropertyMap[testDevice.Id]; var propName = twinProperties[0]; var propValue = twinProperties[1]; var updateDesiredProperties = TwinE2ETests.RegistryManagerUpdateDesiredPropertyAsync(testDevice.Id, propName, propValue); clientOperations.Add(updateDesiredProperties); await Task.WhenAll(clientOperations).ConfigureAwait(false); _log.WriteLine($"{nameof(CombinedClientOperationsMultiplexingOverAmqpTests)}: All operations completed for device={testDevice.Id}"); }; Func <IList <DeviceClient>, Task> cleanupOperation = async(deviceClients) => { await serviceClient.CloseAsync().ConfigureAwait(false); serviceClient.Dispose(); foreach (DeviceClient deviceClient in deviceClients) { deviceClient.Dispose(); } messagesSent.Clear(); twinPropertyMap.Clear(); }; await MultiplexingOverAmqp.TestMultiplexingOperationAsync( DevicePrefix, transport, poolSize, devicesCount, initOperation, testOperation, cleanupOperation).ConfigureAwait(false); }
private async Task SendMethodAndRespondRecoveryPoolOverAmqpAsync( TestDeviceType type, Client.TransportType transport, int poolSize, int devicesCount, Func <DeviceClient, string, MsTestLogger, Task <Task> > setDeviceReceiveMethod, string faultType, string reason, TimeSpan delayInSec = default, TimeSpan durationInSec = default, ConnectionStringAuthScope authScope = ConnectionStringAuthScope.Device, string proxyAddress = null) { async Task InitOperationAsync(DeviceClient deviceClient, TestDevice testDevice, TestDeviceCallbackHandler testDeviceCallbackHandler) { Logger.Trace($"{nameof(MethodE2EPoolAmqpTests)}: Setting method callback handler for device {testDevice.Id}"); await testDeviceCallbackHandler .SetDeviceReceiveMethodAsync(MethodName, MethodE2ETests.DeviceResponseJson, MethodE2ETests.ServiceRequestJson) .ConfigureAwait(false); } async Task TestOperationAsync(DeviceClient deviceClient, TestDevice testDevice, TestDeviceCallbackHandler testDeviceCallbackHandler) { using var cts = new CancellationTokenSource(FaultInjection.RecoveryTime); Logger.Trace($"{nameof(MethodE2EPoolAmqpTests)}: Preparing to receive method for device {testDevice.Id}"); Task serviceSendTask = MethodE2ETests .ServiceSendMethodAndVerifyResponseAsync( testDevice.Id, MethodName, MethodE2ETests.DeviceResponseJson, MethodE2ETests.ServiceRequestJson, Logger); Task methodReceivedTask = testDeviceCallbackHandler.WaitForMethodCallbackAsync(cts.Token); await Task.WhenAll(serviceSendTask, methodReceivedTask).ConfigureAwait(false); } async Task CleanupOperationAsync(List <DeviceClient> deviceClients, List <TestDeviceCallbackHandler> testDeviceCallbackHandlers) { deviceClients.ForEach(deviceClient => deviceClient.Dispose()); testDeviceCallbackHandlers.ForEach(testDeviceCallbackHandler => testDeviceCallbackHandler.Dispose()); await Task.FromResult <bool>(false).ConfigureAwait(false); } await FaultInjectionPoolingOverAmqp .TestFaultInjectionPoolAmqpAsync( MethodDevicePrefix, transport, proxyAddress, poolSize, devicesCount, faultType, reason, delayInSec == TimeSpan.Zero?FaultInjection.DefaultFaultDelay : delayInSec, durationInSec == TimeSpan.Zero?FaultInjection.DefaultFaultDuration : durationInSec, InitOperationAsync, TestOperationAsync, CleanupOperationAsync, authScope, Logger) .ConfigureAwait(false); }