2023-01-18 14:22:51 +01:00
using System ;
2023-05-09 08:47:52 +02:00
using System.Collections.Generic ;
2021-05-13 20:03:07 +02:00
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.Bikes ;
using ShareeBike.Model.Connector.Updater ;
using ShareeBike.Model.Services.CopriApi ;
using ShareeBike.Repository ;
using ShareeBike.Repository.Response ;
2021-05-13 20:03:07 +02:00
2024-04-09 12:53:23 +02:00
namespace ShareeBike.Model.Connector
2021-05-13 20:03:07 +02:00
{
2022-09-06 16:08:19 +02:00
/// <summary> Provides query functionality for a logged in user. </summary>
public class CachedQueryLoggedIn : BaseLoggedIn , IQuery
{
2023-01-18 14:22:51 +01:00
/// <summary> Cached copri server (connection to copri backed up by cache). </summary>
2022-09-06 16:08:19 +02:00
private ICachedCopriServer Server { get ; }
/// <summary>Constructs a copri query object.</summary>
/// <param name="copriServer">Server which implements communication.</param>
public CachedQueryLoggedIn ( ICopriServerBase copriServer ,
string sessionCookie ,
string mail ,
Func < DateTime > dateTimeProvider ) : base ( copriServer , sessionCookie , mail , dateTimeProvider )
{
Server = copriServer as ICachedCopriServer ;
if ( Server = = null )
{
2023-05-09 08:47:52 +02:00
throw new ArgumentException ( $"Copri server is not of expected type. Type detected is {copriServer.GetType()}." ) ;
2022-09-06 16:08:19 +02:00
}
}
2023-04-19 12:14:14 +02:00
/// <summary> Gets all stations including positions.</summary>
2022-09-06 16:08:19 +02:00
public async Task < Result < StationsAndBikesContainer > > GetBikesAndStationsAsync ( )
{
2023-05-09 08:47:52 +02:00
BikeCollection GetBikeCollection ( IEnumerable < BikeInfoReservedOrBooked > bikeInfoEnumerable , Bikes . BikeInfoNS . BC . DataSource dataSource ) = >
BikeCollectionFactory . GetBikesAll (
null , // Bikes available are no more of interest because count of available bikes at each given station is was added to station object.
bikeInfoEnumerable ? ? new Dictionary < string , BikeInfoReservedOrBooked > ( ) . Values ,
Mail ,
DateTimeProvider ,
dataSource ) ;
2022-09-06 16:08:19 +02:00
var stationsResponse = await Server . GetStations ( ) ;
if ( stationsResponse . Source = = typeof ( CopriCallsMonkeyStore )
| | stationsResponse . Exception ! = null )
{
2023-05-09 08:47:52 +02:00
// Stations were read from cache ==> get bikes available and occupied from cache as well to avoid inconsistencies
2022-09-06 16:08:19 +02:00
return new Result < StationsAndBikesContainer > (
stationsResponse . Source ,
new StationsAndBikesContainer (
stationsResponse . Response . GetStationsAllMutable ( ) ,
2023-05-09 08:47:52 +02:00
GetBikeCollection ( stationsResponse . Response . bikes_occupied ? . Values , Bikes . BikeInfoNS . BC . DataSource . Cache ) ) ,
2022-09-06 16:08:19 +02:00
stationsResponse . GeneralData ,
stationsResponse . Exception ) ;
}
// Both types bikes could read from copri => update cache
Server . AddToCache ( stationsResponse ) ;
return new Result < StationsAndBikesContainer > (
stationsResponse . Source ,
2023-05-09 08:47:52 +02:00
new StationsAndBikesContainer (
stationsResponse . Response . GetStationsAllMutable ( ) ,
GetBikeCollection ( stationsResponse . Response . bikes_occupied ? . Values , Bikes . BikeInfoNS . BC . DataSource . Copri ) ) ,
2022-09-06 16:08:19 +02:00
stationsResponse . GeneralData ,
2023-05-09 08:47:52 +02:00
stationsResponse ? . Exception ) ;
2022-09-06 16:08:19 +02:00
}
/// <summary> Gets bikes occupied. </summary>
/// <returns>Collection of bikes.</returns>
public async Task < Result < BikeCollection > > GetBikesOccupiedAsync ( )
{
var bikesAvailableResponse = await Server . GetBikesAvailable ( false ) ;
if ( bikesAvailableResponse . Source = = typeof ( CopriCallsMonkeyStore )
| | bikesAvailableResponse . Exception ! = null )
{
// Bikes available were read from cache ==> get bikes occupied from cache as well to avoid inconsistencies.
Log . ForContext < CachedQueryLoggedIn > ( ) . Debug ( "Bikes available read from cache. Reading bikes occupied from cache as well." ) ;
return new Result < BikeCollection > (
bikesAvailableResponse . Source ,
2023-01-18 14:22:51 +01:00
BikeCollectionFactory . GetBikesAll (
2022-09-06 16:08:19 +02:00
bikesAvailableResponse . Response ? . bikes ? . Values ? . Where ( bike = > bike . GetState ( ) = = State . InUseStateEnum . FeedbackPending ) ,
( await Server . GetBikesOccupied ( true ) ) ? . Response ? . bikes_occupied ? . Values ,
Mail ,
2023-01-18 14:22:51 +01:00
DateTimeProvider ,
Bikes . BikeInfoNS . BC . DataSource . Cache ) ,
2022-09-06 16:08:19 +02:00
bikesAvailableResponse . GeneralData ,
bikesAvailableResponse . Exception ) ;
}
var bikesOccupiedResponse = await Server . GetBikesOccupied ( false ) ;
if ( bikesOccupiedResponse . Source = = typeof ( CopriCallsMonkeyStore )
| | bikesOccupiedResponse . Exception ! = null )
{
// Bikes occupied were read from cache ==> get bikes available from cache as well to avoid inconsistencies
Log . ForContext < CachedQueryLoggedIn > ( ) . Debug ( "Bikes occupied read from cache. Reread bikes available from cache as well." ) ;
return new Result < BikeCollection > (
bikesOccupiedResponse . Source ,
2023-01-18 14:22:51 +01:00
BikeCollectionFactory . GetBikesAll (
2022-09-06 16:08:19 +02:00
( await Server . GetBikesAvailable ( true ) ) . Response ? . bikes ? . Values ? . Where ( bike = > bike . GetState ( ) = = State . InUseStateEnum . FeedbackPending ) ,
bikesOccupiedResponse . Response ? . bikes_occupied ? . Values ,
Mail ,
2023-01-18 14:22:51 +01:00
DateTimeProvider ,
Bikes . BikeInfoNS . BC . DataSource . Cache ) ,
2022-09-06 16:08:19 +02:00
bikesOccupiedResponse . GeneralData ,
bikesOccupiedResponse . Exception ) ;
}
// Both types bikes could read from copri => update bikes occupied cache.
// // Do not add bikes available to cache because this might lead to conflicts calls GetBikesAsync() and bikes with FeedbackPending state are of no use offline.
Server . AddToCache ( bikesOccupiedResponse ) ;
return new Result < BikeCollection > (
bikesOccupiedResponse . Source ,
2023-01-18 14:22:51 +01:00
BikeCollectionFactory . GetBikesAll (
2022-09-06 16:08:19 +02:00
bikesAvailableResponse ? . Response . bikes ? . Values ? . Select ( bike = > bike ) ? . Where ( bike = > bike . GetState ( ) = = State . InUseStateEnum . FeedbackPending ) ,
bikesOccupiedResponse ? . Response ? . bikes_occupied ? . Values ,
Mail ,
2023-01-18 14:22:51 +01:00
DateTimeProvider ,
Bikes . BikeInfoNS . BC . DataSource . Copri ) ,
2022-09-06 16:08:19 +02:00
bikesOccupiedResponse . GeneralData ,
bikesOccupiedResponse . Exception ) ;
}
/// <summary> Gets bikes available and bikes occupied. </summary>
2023-11-06 12:23:09 +01:00
/// <param name="operatorUri">Uri of the operator host to get bikes from or null if bikes have to be gotten form primary host.</param>
2023-11-21 15:26:57 +01:00
/// <param name="stationId"> Id of station which is used for filtering bikes. Null if no filtering should be applied.</param>
/// <param name="bikeId"> Id of bike which is used for filtering bikes. Null if no filtering should be applied.</param>
2022-09-06 16:08:19 +02:00
/// <returns>Collection of bikes.</returns>
2023-11-21 15:26:57 +01:00
public async Task < Result < BikeCollection > > GetBikesAsync ( Uri operatorUri = null , string stationId = null , string bikeId = null )
2022-09-06 16:08:19 +02:00
{
2023-11-21 15:26:57 +01:00
var bikesAvailableResponse = await Server . GetBikesAvailable ( operatorUri : operatorUri , stationId : stationId , bikeId : bikeId ) ;
2022-09-06 16:08:19 +02:00
if ( bikesAvailableResponse . Source = = typeof ( CopriCallsMonkeyStore )
| | bikesAvailableResponse . Exception ! = null )
{
2023-11-06 12:23:09 +01:00
// Bikes were read from cache.
Log . ForContext < CachedQueryLoggedIn > ( ) . Debug ( "Bikes available and bikes occupied from cache invoking one single call." ) ;
2022-09-06 16:08:19 +02:00
return new Result < BikeCollection > (
bikesAvailableResponse . Source ,
2023-01-18 14:22:51 +01:00
BikeCollectionFactory . GetBikesAll (
2022-09-06 16:08:19 +02:00
bikesAvailableResponse . Response ? . bikes ? . Values ,
2023-11-06 12:23:09 +01:00
operatorUri ? . AbsoluteUri = = null ?
( await Server . GetBikesOccupied ( true ) ) . Response ? . bikes_occupied ? . Values // Get bikes occupied from cache as well to avoid inconsistencies.
: bikesAvailableResponse . Response ? . bikes_occupied ? . Values ,
2022-09-06 16:08:19 +02:00
Mail ,
2023-01-18 14:22:51 +01:00
DateTimeProvider ,
Bikes . BikeInfoNS . BC . DataSource . Cache ) ,
2022-09-06 16:08:19 +02:00
bikesAvailableResponse . GeneralData ,
bikesAvailableResponse . Exception ) ;
}
2023-11-06 12:23:09 +01:00
if ( operatorUri ? . AbsoluteUri ! = null )
{
// Both types bikes could read from copri successfully => update cache
2023-11-21 15:26:57 +01:00
Server . AddToCache ( bikesAvailableResponse , operatorUri , stationId , bikeId ) ;
2023-11-06 12:23:09 +01:00
Log . ForContext < CachedQueryLoggedIn > ( ) . Debug ( "Bikes available and occupied read successfully from server invoking one single request." ) ;
return new Result < BikeCollection > (
bikesAvailableResponse . Source ,
BikeCollectionFactory . GetBikesAll (
bikesAvailableResponse . Response ? . bikes ? . Values ,
bikesAvailableResponse . Response ? . bikes_occupied ? . Values ,
Mail ,
DateTimeProvider ,
Bikes . BikeInfoNS . BC . DataSource . Copri ) ,
bikesAvailableResponse . GeneralData ,
bikesAvailableResponse . Exception ! = null ? new AggregateException ( new [ ] { bikesAvailableResponse . Exception } ) : null ) ;
}
/// Legacy implementation: GetBikesOccupied are not returned in <see cref="ICachedCopriServer.GetBikesAvailable"/> call.
/// A separate call <see cref="ICachedCopriServer.GetBikesOccupied"/> is required to retrieve all bikes.
var bikesOccupiedResponse = await Server . GetBikesOccupied ( ) ; /* Only query bikes occupied if operator uri is unknown. */
2022-09-06 16:08:19 +02:00
if ( bikesOccupiedResponse . Source = = typeof ( CopriCallsMonkeyStore )
| | bikesOccupiedResponse . Exception ! = null )
{
// Bikes occupied were read from cache ==> get bikes available from cache as well to avoid inconsistencies
Log . ForContext < CachedQueryLoggedIn > ( ) . Debug ( "Bikes occupied read from cache. Reread bikes available from cache as well." ) ;
return new Result < BikeCollection > (
bikesOccupiedResponse . Source ,
2023-01-18 14:22:51 +01:00
BikeCollectionFactory . GetBikesAll (
2023-11-21 15:26:57 +01:00
( await Server . GetBikesAvailable ( true , operatorUri , stationId , bikeId ) ) . Response ? . bikes ? . Values ,
2022-09-06 16:08:19 +02:00
bikesOccupiedResponse . Response ? . bikes_occupied ? . Values ,
Mail ,
2023-01-18 14:22:51 +01:00
DateTimeProvider ,
Bikes . BikeInfoNS . BC . DataSource . Cache ) ,
2022-09-06 16:08:19 +02:00
bikesOccupiedResponse . GeneralData ,
bikesOccupiedResponse . Exception ) ;
}
// Both types bikes could read from copri => update cache
2023-11-21 15:26:57 +01:00
Server . AddToCache ( bikesAvailableResponse , operatorUri , stationId , bikeId ) ;
2022-09-06 16:08:19 +02:00
Server . AddToCache ( bikesOccupiedResponse ) ;
Log . ForContext < CachedQueryLoggedIn > ( ) . Debug ( "Bikes available and occupied read successfully from server." ) ;
return new Result < BikeCollection > (
bikesAvailableResponse . Source ,
2023-01-18 14:22:51 +01:00
BikeCollectionFactory . GetBikesAll (
2022-09-06 16:08:19 +02:00
bikesAvailableResponse . Response ? . bikes ? . Values ,
bikesOccupiedResponse . Response ? . bikes_occupied ? . Values ,
Mail ,
2023-01-18 14:22:51 +01:00
DateTimeProvider ,
Bikes . BikeInfoNS . BC . DataSource . Copri ) ,
2022-09-06 16:08:19 +02:00
bikesAvailableResponse . GeneralData ,
bikesAvailableResponse . Exception ! = null | | bikesOccupiedResponse . Exception ! = null ? new AggregateException ( new [ ] { bikesAvailableResponse . Exception , bikesOccupiedResponse . Exception } ) : null ) ;
}
}
2021-05-13 20:03:07 +02:00
}