2023-08-31 12:20:06 +02:00
using System ;
using System.Collections.Generic ;
using System.Threading ;
using System.Threading.Tasks ;
using Serilog ;
2024-04-09 12:53:23 +02:00
using ShareeBike.Model.Connector ;
using ShareeBike.Repository.Exception ;
using ShareeBike.Repository.Request ;
using ShareeBike.Services.BluetoothLock ;
using ShareeBike.Services.BluetoothLock.Exception ;
using ShareeBike.Services.BluetoothLock.Tdo ;
using ShareeBike.Services.Geolocation ;
using ShareeBike.ViewModel.Bikes.Bike.BluetoothLock.RequestHandler ;
2023-08-31 12:20:06 +02:00
2024-04-09 12:53:23 +02:00
namespace ShareeBike.Model.Bikes.BikeInfoNS.BluetoothLock.Command
2023-08-31 12:20:06 +02:00
{
public static class CloseCommand
{
/// <summary>
/// Possible steps of closing a lock.
/// </summary>
public enum Step
{
StartStopingPolling ,
StartingQueryingLocation ,
ClosingLock ,
WaitStopPollingQueryLocation ,
/// <summary>
/// Notifies back end about modified lock state.
/// </summary>
UpdateLockingState ,
QueryLocationTerminated
}
/// <summary>
2023-08-31 12:31:38 +02:00
/// Possible states of closing a lock.
2023-08-31 12:20:06 +02:00
/// </summary>
public enum State
{
OutOfReachError ,
CouldntCloseMovingError ,
CouldntCloseBoltBlockedError ,
GeneralCloseError ,
StartGeolocationException ,
WaitGeolocationException ,
WebConnectFailed ,
ResponseIsInvalid ,
BackendUpdateFailed
}
/// <summary>
/// Interface to notify view model about steps/ state changes of closing process.
/// </summary>
public interface ICloseCommandListener
{
/// <summary>
/// Reports current step.
/// </summary>
/// <param name="currentStep">Current step to report.</param>
void ReportStep ( Step currentStep ) ;
/// <summary>
/// Reports current state.
/// </summary>
/// <param name="currentState">Current state to report.</param>
/// <param name="message">Message describing the current state.</param>
/// <returns></returns>
Task ReportStateAsync ( State currentState , string message ) ;
}
/// <summary>
/// Closes the lock and updates copri.
/// </summary>
/// <param name="listener">Interface to notify view model about steps/ state changes of closing process.</param>
/// <param name="stopPolling">Task which stops polling.</param>
public static async Task InvokeAsync < T > (
IBikeInfoMutable bike ,
IGeolocationService geolocation ,
ILocksService lockService ,
Func < bool > isConnectedDelegate ,
2024-04-09 12:53:23 +02:00
Func < bool , IConnector > connectorFactory ,
2023-08-31 12:20:06 +02:00
ICloseCommandListener listener ,
Task stopPollingTask )
{
// Invokes member to notify about step being started.
void InvokeCurrentStep ( Step step )
{
if ( listener = = null )
return ;
try
{
listener . ReportStep ( step ) ;
}
2023-11-06 12:23:09 +01:00
catch ( Exception exception )
2023-08-31 12:20:06 +02:00
{
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Error ( "An exception {@exception} was thrown invoking step-action for step {step} " , exception , step ) ;
2023-08-31 12:20:06 +02:00
}
}
// Invokes member to notify about state change.
async Task InvokeCurrentStateAsync ( State state , string message )
{
if ( listener = = null )
return ;
try
{
await listener . ReportStateAsync ( state , message ) ;
}
2023-11-06 12:23:09 +01:00
catch ( Exception exception )
2023-08-31 12:20:06 +02:00
{
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Error ( "An exception {@exception} was thrown invoking state-action for state {state} " , exception , state ) ;
2023-08-31 12:20:06 +02:00
}
}
// Wait for geolocation and polling task to stop (on finished or on canceled).
async Task < IGeolocation > WaitForPendingTasks ( Task < IGeolocation > locationTask )
{
// Step: Wait until getting geolocation has completed.
InvokeCurrentStep ( Step . WaitStopPollingQueryLocation ) ;
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( $"Waiting on steps {Step.StartingQueryingLocation} and {Step.StartStopingPolling} to finish..." ) ;
2023-08-31 12:20:06 +02:00
try
{
await Task . WhenAll ( new List < Task > { locationTask , stopPollingTask ? ? Task . CompletedTask } ) ;
}
2023-11-06 12:23:09 +01:00
catch ( Exception exception )
2023-08-31 12:20:06 +02:00
{
// No location information available.
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( "Canceling query location/ wait for polling task to finish failed. {@exception}" , exception ) ;
await InvokeCurrentStateAsync ( State . WaitGeolocationException , exception . Message ) ;
2023-08-31 12:20:06 +02:00
InvokeCurrentStep ( Step . QueryLocationTerminated ) ;
return null ;
}
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( $"Steps {Step.StartingQueryingLocation} and {Step.StartStopingPolling} finished." ) ;
2023-08-31 12:20:06 +02:00
InvokeCurrentStep ( Step . QueryLocationTerminated ) ;
return locationTask . Result ;
}
// Updates locking state
async Task UpdateLockingState ( IGeolocation location , DateTime timeStamp )
{
// Step: Update backend.
InvokeCurrentStep ( Step . UpdateLockingState ) ;
try
{
2024-04-09 12:53:23 +02:00
await connectorFactory ( true ) . Command . UpdateLockingStateAsync (
2023-08-31 12:20:06 +02:00
bike ,
location ! = null
? new LocationDto . Builder
{
Latitude = location . Latitude ,
Longitude = location . Longitude ,
Accuracy = location . Accuracy ? ? double . NaN ,
Age = timeStamp . Subtract ( location . Timestamp . DateTime ) ,
} . Build ( )
: null ) ;
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( "Backend updated for bike {bikeId} successfully." , bike . Id ) ;
2023-08-31 12:20:06 +02:00
}
catch ( Exception exception )
{
2023-11-06 12:23:09 +01:00
Log . ForContext < ReservedOpen > ( ) . Information ( "Updating backend for bike {bikeId} failed." , bike . Id ) ;
2023-08-31 12:20:06 +02:00
//BikesViewModel.RentalProcess.Result = CurrentStepStatus.Failed;
if ( exception is WebConnectFailureException )
{
// Copri server is not reachable.
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Debug ( "Copri server not reachable." ) ;
2023-08-31 12:20:06 +02:00
await InvokeCurrentStateAsync ( State . WebConnectFailed , exception . Message ) ;
return ;
}
else if ( exception is ResponseException copriException )
{
// Copri server is not reachable.
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Debug ( "Message: {Message} Details: {Details}" , copriException . Message , copriException . Response ) ;
2023-08-31 12:20:06 +02:00
await InvokeCurrentStateAsync ( State . ResponseIsInvalid , exception . Message ) ;
return ;
}
else
{
Log . ForContext < T > ( ) . Error ( "User locked bike {bike} in order to pause ride but updating failed. {@l_oException}" , bike . Id , exception ) ;
await InvokeCurrentStateAsync ( State . BackendUpdateFailed , exception . Message ) ;
return ;
}
}
}
2024-04-09 12:53:23 +02:00
//// Start Action
2023-08-31 12:31:38 +02:00
//// Step: Start query geolocation data.
2023-08-31 12:20:06 +02:00
Log . ForContext < T > ( ) . Debug ( $"Starting step {Step.StartingQueryingLocation}..." ) ;
InvokeCurrentStep ( Step . StartingQueryingLocation ) ;
var ctsLocation = new CancellationTokenSource ( ) ;
Task < IGeolocation > currentLocationTask = Task . FromResult < IGeolocation > ( null ) ;
var timeStampNow = DateTime . Now ;
try
{
currentLocationTask = geolocation . GetAsync ( ctsLocation . Token , timeStampNow ) ;
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( "Starting query location successful." ) ;
2023-08-31 12:20:06 +02:00
}
2023-11-06 12:23:09 +01:00
catch ( Exception exception )
2023-08-31 12:20:06 +02:00
{
// No location information available.
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( "Starting query location failed. {@exception}" , exception ) ;
await InvokeCurrentStateAsync ( State . StartGeolocationException , exception . Message ) ;
2023-08-31 12:20:06 +02:00
}
2023-08-31 12:31:38 +02:00
//// Step: Close lock.
2023-08-31 12:20:06 +02:00
IGeolocation currentLocation ;
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( $"Starting step {Step.ClosingLock}..." ) ;
2023-08-31 12:20:06 +02:00
InvokeCurrentStep ( Step . ClosingLock ) ;
LockitLockingState ? lockingState ;
try
{
lockingState = await lockService [ bike . LockInfo . Id ] . CloseAsync ( ) ;
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( "Lock of bike {bikeId} closed successfully." , bike . Id ) ;
2023-08-31 12:20:06 +02:00
}
catch ( Exception exception )
{
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Information ( "Lock of bike {bikeId} can not be closed." , bike . Id ) ;
2023-08-31 12:20:06 +02:00
if ( exception is OutOfReachException )
{
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Debug ( "Lock is out of reach" ) ;
2023-08-31 12:20:06 +02:00
await InvokeCurrentStateAsync ( State . OutOfReachError , exception . Message ) ;
}
else if ( exception is CouldntCloseMovingException )
{
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Debug ( "Lock is moving." ) ;
2023-08-31 12:20:06 +02:00
await InvokeCurrentStateAsync ( State . CouldntCloseMovingError , exception . Message ) ;
}
else if ( exception is CouldntCloseBoltBlockedException )
{
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Debug ( "Bold is blocked.}" ) ;
2023-08-31 12:20:06 +02:00
await InvokeCurrentStateAsync ( State . CouldntCloseBoltBlockedError , exception . Message ) ;
}
else
{
2023-11-06 12:23:09 +01:00
Log . ForContext < T > ( ) . Debug ( "{@exception}" , exception ) ;
2023-08-31 12:20:06 +02:00
await InvokeCurrentStateAsync ( State . GeneralCloseError , exception . Message ) ;
}
// Signal cts to cancel getting geolocation.
ctsLocation . Cancel ( ) ;
2023-08-31 12:31:38 +02:00
// Wait until getting geolocation and stop polling has completed.
2023-08-31 12:20:06 +02:00
currentLocation = await WaitForPendingTasks ( currentLocationTask ) ;
// Update current state from exception
bike . LockInfo . State = exception is StateAwareException stateAwareException
? stateAwareException . State
: LockingState . UnknownDisconnected ;
if ( ! isConnectedDelegate ( ) )
{
// Lock state can not be updated because there is no connected.
throw ;
}
if ( exception is OutOfReachException )
{
// Locking state can not be updated because lock is not connected.
throw ;
}
2023-08-31 12:31:38 +02:00
// Update backend.
2023-08-31 12:20:06 +02:00
// Do this even if current lock state is open (lock state must not necessarily be open before try to open, i.e. something undefined between open and closed).
await UpdateLockingState ( currentLocation , timeStampNow ) ;
throw ;
}
2023-08-31 12:31:38 +02:00
//// Step: Wait until getting geolocation and stop polling has completed.
2023-08-31 12:20:06 +02:00
currentLocation = await WaitForPendingTasks ( currentLocationTask ) ;
bike . LockInfo . State = lockingState ? . GetLockingState ( ) ? ? LockingState . UnknownDisconnected ;
// Keep geolocation where closing action occurred.
bike . LockInfo . Location = currentLocation ;
if ( ! isConnectedDelegate ( ) )
{
return ;
}
2023-08-31 12:31:38 +02:00
//// Step: Update backend.
2023-08-31 12:20:06 +02:00
await UpdateLockingState ( currentLocation , timeStampNow ) ;
}
}
}