sharee.bike-App/SharedBusinessLogic/Services/CopriApi/Polling.cs

257 lines
8.5 KiB
C#
Raw Permalink Normal View History

2023-02-22 14:03:35 +01:00
using System;
2022-04-25 22:15:15 +02:00
using System.Diagnostics;
using System.Linq;
using System.Threading.Tasks;
2022-08-30 15:42:25 +02:00
using Serilog;
2024-04-09 12:53:23 +02:00
using ShareeBike.Model;
using ShareeBike.Model.Bikes.BikeInfoNS.CopriLock;
using ShareeBike.Model.Connector;
using ShareeBike.Model.Connector.Updater;
using ShareeBike.Model.Device;
using ShareeBike.Model.Services.CopriApi;
using ShareeBike.Repository;
using ShareeBike.Repository.Response;
using ShareeBike.Services.CopriApi.Exception;
2022-04-25 22:15:15 +02:00
2024-04-09 12:53:23 +02:00
namespace ShareeBike.Services.CopriApi
2022-04-25 22:15:15 +02:00
{
2022-09-06 16:08:19 +02:00
public static class Polling
{
/// <summary> Timeout for open/ close operations.</summary>
private const int OPEN_CLOSE_TIMEOUT_MS = 50000;
/// <summary> Opens lock.</summary>
2023-02-22 14:03:35 +01:00
/// <param name="copriServer"> Instance to communicate with backend.</param>
2022-09-06 16:08:19 +02:00
/// <param name="bike">Bike object holding id of bike to open. Lock state of object is updated after open request.</param>
public static async Task OpenAync(
2023-02-22 14:03:35 +01:00
this ICopriServerBase copriServer,
2022-09-06 16:08:19 +02:00
IBikeInfoMutable bike)
{
2023-02-22 14:03:35 +01:00
if (!(copriServer is ICachedCopriServer cachedServer))
throw new ArgumentNullException(nameof(copriServer));
2022-09-06 16:08:19 +02:00
2023-02-22 14:03:35 +01:00
await cachedServer.OpenAync(bike);
}
/// <summary> Opens lock.</summary>
2023-05-09 08:47:52 +02:00
/// <param name="cachedServer"> Instance to communicate with backend.</param>
2023-02-22 14:03:35 +01:00
/// <param name="bike">Bike object holding id of bike to open. Lock state of object is updated after open request.</param>
public static async Task OpenAync(
this ICachedCopriServer cachedServer,
IBikeInfoMutable bike)
{
2022-09-06 16:08:19 +02:00
// Send command to close lock
2023-02-22 14:03:35 +01:00
await cachedServer.UpdateLockingStateAsync(
2022-09-06 16:08:19 +02:00
bike.Id,
Repository.Request.lock_state.unlocking,
bike.OperatorUri);
var lockingState = await cachedServer.GetLockStateAsync(bike.Id);
var watch = new Stopwatch();
watch.Start();
while (lockingState != LockingState.Open
&& lockingState != LockingState.UnknownDisconnected
&& watch.Elapsed < TimeSpan.FromMilliseconds(OPEN_CLOSE_TIMEOUT_MS))
{
// Delay a litte to reduce load on backend.
await Task.Delay(3000);
lockingState = await cachedServer.GetLockStateAsync(bike.Id);
Log.Information($"Current lock state is {lockingState}.");
}
// Update locking state.
bike.LockInfo.State = lockingState;
}
/// <summary>
/// Books a bike and opens the lock.
/// </summary>
2023-05-09 08:47:52 +02:00
/// <param name="copriServer"> Instance to communicate with backend.</param>
2022-09-06 16:08:19 +02:00
/// <param name="bike">Bike to book and open.</param>
/// <param name="mailAddress">Mail address of user which books bike.</param>
public static async Task BookAndOpenAync(
2023-05-09 08:47:52 +02:00
this ICopriServerBase copriServer,
2022-09-06 16:08:19 +02:00
IBikeInfoMutable bike,
string mailAddress)
{
if (bike == null)
{
throw new ArgumentNullException(nameof(bike), "Can not book bike and open lock. No bike object available.");
}
2023-05-09 08:47:52 +02:00
if (!(copriServer is ICachedCopriServer cachedServer))
throw new ArgumentNullException(nameof(copriServer));
2022-09-06 16:08:19 +02:00
// Send command to open lock
var response = bike.State.Value == Model.State.InUseStateEnum.Disposable
2023-05-09 08:47:52 +02:00
? (await copriServer.BookAvailableAndStartOpeningAsync(bike.Id, bike.OperatorUri)).GetIsBookingResponseOk(bike.Id)
: (await copriServer.BookReservedAndStartOpeningAsync(bike.Id, bike.OperatorUri)).GetIsBookingResponseOk(bike.Id);
2022-09-06 16:08:19 +02:00
2023-07-04 11:06:38 +02:00
// Updated locking state.
2022-09-06 16:08:19 +02:00
var lockingState = await cachedServer.GetOccupiedBikeLockStateAsync(bike.Id);
var watch = new Stopwatch();
watch.Start();
while (lockingState.HasValue /* if null bike is no more occupied*/ &&
lockingState.Value != LockingState.Open
&& watch.Elapsed < TimeSpan.FromMilliseconds(OPEN_CLOSE_TIMEOUT_MS))
{
// Delay a litte to reduce load on backend.
await Task.Delay(3000);
lockingState = await cachedServer.GetOccupiedBikeLockStateAsync(bike.Id);
Log.Debug($"Current lock state of bike {bike.Id} is {(lockingState.HasValue ? lockingState.Value.ToString() : "-")}.");
}
// Check if bike is still occupied.
if (lockingState == null)
{
// User did not take bike out of the station
2023-04-19 12:14:14 +02:00
throw new BikeStillInStationException("Booking was canceled because bike is still in station.");
2022-09-06 16:08:19 +02:00
}
2023-06-06 12:00:24 +02:00
// Update booking state.
2022-09-06 16:08:19 +02:00
bike.Load(
response,
mailAddress,
Model.Bikes.BikeInfoNS.BC.NotifyPropertyChangedLevel.None);
// Update locking state.
bike.LockInfo.State = lockingState.Value;
}
/// <summary>
/// Returns a bike and closes the lock.
/// </summary>
2023-05-09 08:47:52 +02:00
/// <param name="copriServer"> Instance to communicate with backend.</param>
2022-09-06 16:08:19 +02:00
/// <param name="bike">Bike to close.</param>
public static async Task CloseAync(
2023-05-09 08:47:52 +02:00
this ICopriServerBase copriServer,
2022-09-06 16:08:19 +02:00
IBikeInfoMutable bike)
{
2023-05-09 08:47:52 +02:00
if (!(copriServer is ICachedCopriServer cachedServer))
throw new ArgumentNullException(nameof(copriServer));
2022-09-06 16:08:19 +02:00
// Send command to close lock
2023-05-09 08:47:52 +02:00
await copriServer.UpdateLockingStateAsync(
2022-09-06 16:08:19 +02:00
bike.Id,
Repository.Request.lock_state.locking,
bike.OperatorUri);
var lockingState = await cachedServer.GetLockStateAsync(bike.Id);
var watch = new Stopwatch();
watch.Start();
while (lockingState != LockingState.Closed
&& lockingState != LockingState.UnknownDisconnected
&& watch.Elapsed < TimeSpan.FromMilliseconds(OPEN_CLOSE_TIMEOUT_MS))
{
// Delay a litte to reduce load on backend.
await Task.Delay(3000);
lockingState = await cachedServer.GetLockStateAsync(bike.Id);
Log.Information($"Current lock state is {lockingState}.");
}
// Update locking state.
bike.LockInfo.State = lockingState;
}
/// <summary>
/// Returns a bike and closes the lock.
/// </summary>
2023-05-09 08:47:52 +02:00
/// <param name="copriServer"> Instance to communicate with backend.</param>
2022-09-06 16:08:19 +02:00
/// <param name="smartDevice">Smart device on which app runs on.</param>
/// <param name="mailAddress">Mail address of user which books bike.</param>
public static async Task<BookingFinishedModel> ReturnAndCloseAync(
2023-05-09 08:47:52 +02:00
this ICopriServerBase copriServer,
2022-09-06 16:08:19 +02:00
ISmartDevice smartDevice,
IBikeInfoMutable bike)
{
2023-05-09 08:47:52 +02:00
if (!(copriServer is ICachedCopriServer cachedServer))
throw new ArgumentNullException(nameof(copriServer));
2022-09-06 16:08:19 +02:00
// Send command to open lock
DoReturnResponse response =
2023-05-09 08:47:52 +02:00
await copriServer.ReturnAndStartClosingAsync(bike.Id, bike.OperatorUri);
2022-09-06 16:08:19 +02:00
2023-07-04 11:06:38 +02:00
// Update booking state
bike.Load(
Model.Bikes.BikeInfoNS.BC.NotifyPropertyChangedLevel.None,
response.bike_returned.station ?? string.Empty);
2022-09-06 16:08:19 +02:00
var lockingState = await cachedServer.GetLockStateAsync(bike.Id);
var watch = new Stopwatch();
watch.Start();
while (lockingState != LockingState.Closed
&& lockingState != LockingState.UnknownDisconnected
&& watch.Elapsed < TimeSpan.FromMilliseconds(OPEN_CLOSE_TIMEOUT_MS))
{
// Delay a litte to reduce load on backend.
await Task.Delay(3000);
lockingState = await cachedServer.GetLockStateAsync(bike.Id);
Log.Information($"Current lock state is {lockingState}.");
}
// Update locking state.
bike.LockInfo.State = lockingState;
return response?.Create() ?? new BookingFinishedModel();
}
/// <summary>
/// Queries the locking state from copri.
/// </summary>
2023-05-09 08:47:52 +02:00
/// <param name="copriServer">Service to use.</param>
2022-09-06 16:08:19 +02:00
/// <param name="bikeId">Bike id to query lock state for.</param>
/// <returns>Locking state</returns>
private static async Task<LockingState> GetLockStateAsync(
2023-05-09 08:47:52 +02:00
this ICachedCopriServer copriServer,
2022-09-06 16:08:19 +02:00
string bikeId)
{
// Querry reserved or booked bikes first for performance reasons.
2023-05-09 08:47:52 +02:00
var bikeReservedOrBooked = (await copriServer.GetBikesOccupied(false))?.Response.bikes_occupied?.Values?.FirstOrDefault(x => x.bike == bikeId);
2022-09-06 16:08:19 +02:00
if (bikeReservedOrBooked != null)
{
return bikeReservedOrBooked.GetCopriLockingState();
}
2023-05-09 08:47:52 +02:00
var bikeAvailable = (await copriServer.GetBikesAvailable(false))?.Response.bikes?.Values?.FirstOrDefault(x => x.bike == bikeId);
2022-09-06 16:08:19 +02:00
if (bikeAvailable != null)
{
return bikeAvailable.GetCopriLockingState();
}
return LockingState.UnknownDisconnected;
}
/// <summary>
/// Queries the locking state of a occupied bike from copri.
/// </summary>
2023-05-09 08:47:52 +02:00
/// <param name="copriServer">Service to use.</param>
2022-09-06 16:08:19 +02:00
/// <param name="bikeId">Bike id to query lock state for.</param>
/// <returns>Locking state if bike is still occupied, null otherwise.</returns>
private static async Task<LockingState?> GetOccupiedBikeLockStateAsync(
2023-05-09 08:47:52 +02:00
this ICachedCopriServer copriServer,
2022-09-06 16:08:19 +02:00
string bikeId)
{
2023-05-09 08:47:52 +02:00
var bikeReservedOrBooked = (await copriServer.GetBikesOccupied(false))?.Response.bikes_occupied?.Values?.FirstOrDefault(x => x.bike == bikeId);
2022-09-06 16:08:19 +02:00
if (bikeReservedOrBooked != null)
{
return bikeReservedOrBooked.GetCopriLockingState();
}
return null;
}
}
2022-04-25 22:15:15 +02:00
}