mirror of
https://dev.azure.com/TeilRad/sharee.bike%20App/_git/Code
synced 2025-06-21 21:46:27 +02:00
Initial version.
This commit is contained in:
parent
193aaa1a56
commit
b72c67a53e
228 changed files with 25924 additions and 0 deletions
9
TINKLib/Services/BluetoothLock/ILocksServiceFake.cs
Normal file
9
TINKLib/Services/BluetoothLock/ILocksServiceFake.cs
Normal file
|
@ -0,0 +1,9 @@
|
|||
using TINK.Model.Bike;
|
||||
|
||||
namespace TINK.Services.BluetoothLock
|
||||
{
|
||||
public interface ILocksServiceFake : ILocksService
|
||||
{
|
||||
void UpdateSimulation(BikeCollection bikes);
|
||||
}
|
||||
}
|
157
TINKLib/Services/BluetoothLock/LocksServiceInReach.cs
Normal file
157
TINKLib/Services/BluetoothLock/LocksServiceInReach.cs
Normal file
|
@ -0,0 +1,157 @@
|
|||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Bike;
|
||||
using TINK.Model.Bike.BluetoothLock;
|
||||
using TINK.Services.BluetoothLock.Tdo;
|
||||
using TINK.Model.State;
|
||||
using System;
|
||||
|
||||
namespace TINK.Services.BluetoothLock
|
||||
{
|
||||
/// <summary>
|
||||
/// Facke locks service implementation which simulates locks which are in reach.
|
||||
/// </summary>
|
||||
public class LocksServiceInReach : ILocksServiceFake
|
||||
{
|
||||
private IEnumerable<LockInfoTdo> LocksInfo { get; set; } = new List<LockInfoTdo>();
|
||||
|
||||
/// <summary> Holds timeout values for series of connecting attemps to a lock or multiple locks. </summary>
|
||||
public ITimeOutProvider TimeOut { get; set; }
|
||||
|
||||
/// <summary> Connects to lock.</summary>
|
||||
/// <param name="authInfo"> Info required to connect to lock.</param>
|
||||
/// <param name="connectTimeout">Timeout for connect operation.</param>
|
||||
public async Task<LockInfoTdo> ConnectAsync(LockInfoAuthTdo authInfo, TimeSpan connectTimeout)
|
||||
{
|
||||
return await Task.FromResult(new LockInfoTdo.Builder { Id = authInfo.Id, Guid = authInfo.Guid, State = LockitLockingState.Closed }.Build());
|
||||
}
|
||||
|
||||
/// <param name="connectTimeout">Timeout for connect operation of a single lock.</param>
|
||||
public async Task<IEnumerable<LockInfoTdo>> GetLocksStateAsync(IEnumerable<LockInfoAuthTdo> locksInfo, TimeSpan connectTimeout)
|
||||
{
|
||||
return await Task.FromResult(LocksInfo);
|
||||
}
|
||||
|
||||
public void UpdateSimulation(BikeCollection bikes)
|
||||
{
|
||||
var locksInfo = new List<LockInfoTdo> ();
|
||||
|
||||
// Add and process locks info object
|
||||
foreach (var bikeInfo in bikes.OfType<BikeInfo>())
|
||||
{
|
||||
var lockInfo = bikeInfo.LockInfo;
|
||||
|
||||
switch (bikeInfo.State.Value)
|
||||
{
|
||||
case InUseStateEnum.Disposable:
|
||||
switch (lockInfo.State )
|
||||
{
|
||||
case LockingState.Open:
|
||||
case LockingState.Disconnected:
|
||||
case LockingState.Unknown:
|
||||
// Open bikes are never disposable because as soon as a they are open they get booked.
|
||||
if (LocksInfo.FirstOrDefault(x => x.Id == lockInfo.Id) != null)
|
||||
{
|
||||
continue; // Lock was already added.
|
||||
}
|
||||
locksInfo.Add(new LockInfoTdo.Builder { Id = lockInfo.Id, State = LockitLockingState.Closed }.Build());
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case InUseStateEnum.Reserved:
|
||||
switch (lockInfo.State)
|
||||
{
|
||||
case LockingState.Open:
|
||||
case LockingState.Disconnected:
|
||||
case LockingState.Unknown:
|
||||
// Closed bikes are never reserved because as soon as they are open they get booked.
|
||||
if (LocksInfo.FirstOrDefault(x => x.Id == lockInfo.Id) != null)
|
||||
{
|
||||
continue; // Lock was already added.
|
||||
}
|
||||
locksInfo.Add(new LockInfoTdo.Builder { Id = lockInfo.Id, State = LockitLockingState.Closed }.Build());
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case InUseStateEnum.Booked:
|
||||
switch (lockInfo.State)
|
||||
{
|
||||
case LockingState.Disconnected:
|
||||
case LockingState.Unknown:
|
||||
if (LocksInfo.FirstOrDefault(x => x.Id == lockInfo.Id) != null)
|
||||
{
|
||||
continue; // Lock was already added.
|
||||
}
|
||||
locksInfo.Add(new LockInfoTdo.Builder { Id = lockInfo.Id, State = LockitLockingState.Closed }.Build());
|
||||
break;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
LocksInfo = locksInfo;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Opens a lock.</summary>
|
||||
/// <param name="lockId">Id of lock to open.</param>
|
||||
public async Task<LockitLockingState?> OpenAsync(int bikeId, byte[] copriKey)
|
||||
{
|
||||
return await Task.FromResult(LockitLockingState.Open);
|
||||
}
|
||||
|
||||
/// <summary> Closes a lock.</summary>
|
||||
/// <param name="lockId">Id of lock to close.</param>
|
||||
public async Task<LockitLockingState?> CloseAsync(int bikeId, byte[] copriKey)
|
||||
{
|
||||
return await Task.FromResult(LockitLockingState.Closed);
|
||||
}
|
||||
|
||||
/// <summary> Set sound settings.</summary>
|
||||
/// <param name="lockId">Id of lock to set sound settings.</param>
|
||||
public async Task<bool> SetSoundAsync(int lockId, SoundSettings settings)
|
||||
{
|
||||
return await Task.FromResult(true);
|
||||
}
|
||||
|
||||
/// <summary> Sets whether alarm is on or off.</summary>
|
||||
/// <param name="lockId">Id of lock to get info from.</param>
|
||||
public async Task SetIsAlarmOffAsync(int lockId, bool activated)
|
||||
{
|
||||
await Task.FromResult(true);
|
||||
}
|
||||
|
||||
/// <summary> Gets battery percentage.</summary>
|
||||
/// <param name="lockId">Id of lock to get info for.</param>
|
||||
public Task<double> GetBatteryPercentageAsync(int lockId)
|
||||
{
|
||||
throw new NotSupportedException();
|
||||
}
|
||||
|
||||
/// <summary> Gets whether alarm is on or off.</summary>
|
||||
/// <param name="lockId">Id of lock to get info for.</param>
|
||||
public async Task<bool> GetIsAlarmOffAsync(int lockId)
|
||||
{
|
||||
return await Task.FromResult(true);
|
||||
}
|
||||
|
||||
/// <summary>Gets a lock by bike Id.</summary>
|
||||
/// <param name="bikeId"></param>
|
||||
/// <returns>Lock object</returns>
|
||||
public ILockService this[int bikeId]
|
||||
{
|
||||
get
|
||||
{
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// <summary> Disconnects lock.</summary>
|
||||
/// <param name="bikeId"> Id of lock to disconnect.</param>
|
||||
/// <param name="bikeGuid"> Guid of lock to disconnect.</param>
|
||||
public async Task<LockingState> DisconnectAsync(int bikeId, Guid bikeGuid) => await Task.FromResult(LockingState.Disconnected);
|
||||
}
|
||||
}
|
65
TINKLib/Services/BluetoothLock/LocksServiceOutOfReach.cs
Normal file
65
TINKLib/Services/BluetoothLock/LocksServiceOutOfReach.cs
Normal file
|
@ -0,0 +1,65 @@
|
|||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Bike;
|
||||
using TINK.Model.Bike.BluetoothLock;
|
||||
using TINK.Services.BluetoothLock.Tdo;
|
||||
using System;
|
||||
|
||||
namespace TINK.Services.BluetoothLock
|
||||
{
|
||||
/// <summary>
|
||||
/// Facke locks service implementation which simulates locks which are out of reach.
|
||||
/// </summary>
|
||||
public class LocksServiceOutOfReach : ILocksServiceFake
|
||||
{
|
||||
private IEnumerable<LockInfoTdo> LocksInfo { get; set; } = new List<LockInfoTdo>();
|
||||
|
||||
/// <summary> Holds timeout values for series of connecting attemps to a lock or multiple locks. </summary>
|
||||
public ITimeOutProvider TimeOut { get; set; }
|
||||
|
||||
/// <summary> Connects to lock.</summary>
|
||||
/// <param name="authInfo"> Info required to connect to lock.</param>
|
||||
/// <param name="connectTimeout">Timeout for connect operation.</param>
|
||||
public async Task<LockInfoTdo> ConnectAsync(LockInfoAuthTdo authInfo, TimeSpan connectTimeout)
|
||||
{
|
||||
return await Task.FromResult(new LockInfoTdo.Builder { Id = authInfo.Id, Guid = authInfo.Guid, State = null }.Build());
|
||||
}
|
||||
|
||||
/// <summary> No info availalbe because no lock is in reach.</summary>
|
||||
/// <param name="connectTimeout">Timeout for connect operation of a single lock.</param>
|
||||
/// <returns>Empty collection.</returns>
|
||||
public async Task<IEnumerable<LockInfoTdo>> GetLocksStateAsync(IEnumerable<LockInfoAuthTdo> locksInfo, TimeSpan connectTimeout) => await Task.FromResult(LocksInfo);
|
||||
|
||||
|
||||
/// <summary> No changes required because lock state is unknown.</summary>
|
||||
public void UpdateSimulation(BikeCollection bikes)
|
||||
{
|
||||
var locksInfo = new List<LockInfoTdo>();
|
||||
|
||||
// Add and process locks info object
|
||||
foreach (var bikeInfo in bikes.OfType<BikeInfo>())
|
||||
{
|
||||
locksInfo.Add(new LockInfoTdo.Builder { Id = bikeInfo.LockInfo.Id, State = null }.Build());
|
||||
}
|
||||
|
||||
LocksInfo = locksInfo;
|
||||
}
|
||||
|
||||
/// <summary>Gets a lock by bike Id.</summary>
|
||||
/// <param name="bikeId"></param>
|
||||
/// <returns>Lock object</returns>
|
||||
public ILockService this[int bikeId]
|
||||
{
|
||||
get
|
||||
{
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Disconnects lock.</summary>
|
||||
/// <param name="bikeId"> Id of lock to disconnect.</param>
|
||||
/// <param name="bikeGuid"> Guid of lock to disconnect.</param>
|
||||
public async Task<LockingState> DisconnectAsync(int bikeId, Guid bikeGuid) => await Task.FromResult(LockingState.Disconnected);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,49 @@
|
|||
using System;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
|
||||
namespace TINK.Services.BluetoothLock
|
||||
{
|
||||
/// <summary> Manages Locks services and related configuration parameter. </summary>
|
||||
public class LocksServicesContainerMutable : IEnumerable<string>
|
||||
{
|
||||
/// <summary> Manages the different types of LocksService objects.</summary>
|
||||
private ServicesContainerMutable<ILocksService> LocksServices { get; }
|
||||
|
||||
/// <summary> Holds the name of the default locks service to use. </summary>
|
||||
public static string DefaultLocksservice => typeof(BLE.LockItByScanServicePolling).FullName;
|
||||
|
||||
/// <summary></summary>
|
||||
/// <param name="activeLockService">Name of active lock service implementation to use.</param>
|
||||
/// <param name="locksServices">Null for production (set of lock service implentations and some fake implementation will created) hash set of services for testing purposes. </param>
|
||||
public LocksServicesContainerMutable(
|
||||
string activeLockService,
|
||||
HashSet<ILocksService> locksServices)
|
||||
{
|
||||
LocksServices = new ServicesContainerMutable<ILocksService>(
|
||||
locksServices,
|
||||
activeLockService);
|
||||
}
|
||||
|
||||
/// <summary> Active locks service.</summary>
|
||||
public ILocksService Active => LocksServices.Active;
|
||||
|
||||
/// <summary> Sets a lock service as active locks service by name. </summary>
|
||||
/// <param name="active">Name of the new locks service.</param>
|
||||
public void SetActive(string active) => LocksServices.SetActive(active);
|
||||
|
||||
IEnumerator<string> IEnumerable<string>.GetEnumerator()
|
||||
=> LocksServices.Select(x => x.GetType().FullName).GetEnumerator();
|
||||
IEnumerator IEnumerable.GetEnumerator()
|
||||
=> LocksServices.GetEnumerator();
|
||||
|
||||
public void SetTimeOut(TimeSpan connectTimeout)
|
||||
{
|
||||
foreach (var locksService in LocksServices)
|
||||
{
|
||||
locksService.TimeOut = new TimeOutProvider(new List<TimeSpan> { connectTimeout });
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
41
TINKLib/Services/BluetoothLock/StateChecker.cs
Normal file
41
TINKLib/Services/BluetoothLock/StateChecker.cs
Normal file
|
@ -0,0 +1,41 @@
|
|||
using Plugin.BLE.Abstractions.Contracts;
|
||||
using System;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace TINK.Services.BluetoothLock
|
||||
{
|
||||
public static class StateChecker
|
||||
{
|
||||
/// <summary>
|
||||
/// Get current bluetooth state
|
||||
/// </summary>
|
||||
/// <remarks>See https://github.com/xabre/xamarin-bluetooth-le/issues/112#issuecomment-380994887.</remarks>
|
||||
/// <param name="ble">Crossplatform bluetooth implementation object</param>
|
||||
/// <returns>BluetoothState</returns>
|
||||
public static Task<BluetoothState> GetBluetoothState(this IBluetoothLE ble)
|
||||
{
|
||||
var tcs = new TaskCompletionSource<BluetoothState>();
|
||||
|
||||
if (ble.State != BluetoothState.Unknown)
|
||||
{
|
||||
// If we can detect state out of box just returning in
|
||||
tcs.SetResult(ble.State);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Otherwise let's setup dynamic event handler and wait for first state update
|
||||
EventHandler<Plugin.BLE.Abstractions.EventArgs.BluetoothStateChangedArgs> handler = null;
|
||||
handler = (o, e) =>
|
||||
{
|
||||
ble.StateChanged -= handler;
|
||||
// and return it as our state
|
||||
// we can have an 'Unknown' check here, but in normal situation it should never occur
|
||||
tcs.SetResult(e.NewState);
|
||||
};
|
||||
ble.StateChanged += handler;
|
||||
}
|
||||
|
||||
return tcs.Task;
|
||||
}
|
||||
}
|
||||
}
|
246
TINKLib/Services/CopriApi/CopriProviderHttps.cs
Normal file
246
TINKLib/Services/CopriApi/CopriProviderHttps.cs
Normal file
|
@ -0,0 +1,246 @@
|
|||
using Serilog;
|
||||
using System;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Repository;
|
||||
using TINK.Model.Repository.Request;
|
||||
using TINK.Model.Repository.Response;
|
||||
using TINK.Repository.Response;
|
||||
|
||||
namespace TINK.Model.Services.CopriApi
|
||||
{
|
||||
/// <summary> Object which manages calls to copri in a thread safe way inclding cache functionality. </summary>
|
||||
public class CopriProviderHttps : ICachedCopriServer
|
||||
{
|
||||
/// <summary> Object which manages stored copri answers. </summary>
|
||||
private ICopriCache CacheServer { get; }
|
||||
|
||||
/// <summary> Communicates whith copri server. </summary>
|
||||
private ICopriServer HttpsServer { get; }
|
||||
|
||||
/// <summary> True if connector has access to copri server, false if cached values are used. </summary>
|
||||
public bool IsConnected => HttpsServer.IsConnected;
|
||||
|
||||
/// <summary> Gets the session cookie if user is logged in, an empty string otherwise. </summary>
|
||||
public string SessionCookie => HttpsServer.SessionCookie;
|
||||
|
||||
/// <summary> Gets the merchant id.</summary>
|
||||
public string MerchantId => HttpsServer.MerchantId;
|
||||
|
||||
/// <summary> Constructs copri provider object to connet to https using a cache objet. </summary>
|
||||
/// <param name="copriHost"></param>
|
||||
/// <param name="merchantId"></param>
|
||||
/// <param name="userAgent">Holds the name and version of the TINKApp.</param>
|
||||
/// <param name="sessionCookie">Cookie of user if a user is logged in, false otherwise.</param>
|
||||
/// <param name="expiresAfter">Timespan which holds value after which cache expires.</param>
|
||||
/// <param name="isExpired">Delegate which returns if cache conted is out of date or not.</param>
|
||||
public CopriProviderHttps(
|
||||
Uri copriHost,
|
||||
string merchantId,
|
||||
string userAgent,
|
||||
string sessionCookie = null,
|
||||
TimeSpan? expiresAfter = null,
|
||||
ICopriCache cacheServer = null,
|
||||
ICopriServer httpsServer = null)
|
||||
{
|
||||
CacheServer = cacheServer ?? new CopriCallsMonkeyStore(merchantId, sessionCookie, expiresAfter);
|
||||
HttpsServer = httpsServer ?? new CopriCallsHttps(copriHost, merchantId, userAgent, sessionCookie);
|
||||
}
|
||||
|
||||
/// <summary>Gets bikes available.</summary>
|
||||
/// <param name="p_strMerchantId">Id of the merchant.</param>
|
||||
/// <param name="sessionCookie">Auto cookie of user if user is logged in.</param>
|
||||
/// <returns>Response holding list of bikes.</returns>
|
||||
public async Task<Result<BikesAvailableResponse>> GetBikesAvailable(bool fromCache = false)
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Request to get bikes available{(fromCache ? " from cache" : "")}...");
|
||||
if (!CacheServer.IsBikesAvailableExpired
|
||||
|| fromCache)
|
||||
{
|
||||
// No need to query because previous answer is not yet outdated.
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Returning bikes available from cache.");
|
||||
return new Result<BikesAvailableResponse>(typeof(CopriCallsMonkeyStore), await CacheServer.GetBikesAvailableAsync());
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Querrying bikes available from copri.");
|
||||
return new Result<BikesAvailableResponse>(
|
||||
typeof(CopriCallsHttps),
|
||||
(await HttpsServer.GetBikesAvailableAsync()).GetIsResponseOk("Abfrage der verfügbaren Räder fehlgeschlagen."));
|
||||
|
||||
}
|
||||
catch (Exception exception)
|
||||
{
|
||||
// Return response from cache.
|
||||
Log.ForContext<CopriProviderHttps>().Debug("An error occurred querrying bikes available. {Exception}.", exception);
|
||||
return new Result<BikesAvailableResponse>(typeof(CopriCallsMonkeyStore), await CacheServer.GetBikesAvailableAsync(), exception);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Gets a list of bikes reserved/ booked by acctive user. </summary>
|
||||
/// <param name="sessionCookie">Cookie to authenticate user.</param>
|
||||
/// <returns>Response holding list of bikes.</returns>
|
||||
public async Task<Result<BikesReservedOccupiedResponse>> GetBikesOccupied(bool fromCache = false)
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Request to get bikes occupied{(fromCache ? " from cache" : "")}...");
|
||||
if (!CacheServer.IsBikesOccupiedExpired
|
||||
|| fromCache)
|
||||
{
|
||||
// No need to query because previous answer is not yet outdated.
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Returning bikes occupied from cache.");
|
||||
return new Result<BikesReservedOccupiedResponse>(typeof(CopriCallsMonkeyStore), await CacheServer.GetBikesOccupiedAsync());
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Querrying bikes occupied from copri.");
|
||||
return new Result<BikesReservedOccupiedResponse>(
|
||||
typeof(CopriCallsHttps),
|
||||
(await HttpsServer.GetBikesOccupiedAsync()).GetIsResponseOk("Abfrage der reservierten/ gebuchten Räder fehlgeschlagen."));
|
||||
|
||||
}
|
||||
catch (Exception exception)
|
||||
{
|
||||
// Return response from cache.
|
||||
Log.ForContext<CopriProviderHttps>().Debug("An error occurred querrying bikes occupied. {Exception}.", exception);
|
||||
return new Result<BikesReservedOccupiedResponse>(typeof(CopriCallsMonkeyStore), await CacheServer.GetBikesOccupiedAsync(), exception);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Get list of stations. </summary>
|
||||
/// <returns>List of files.</returns>
|
||||
public async Task<Result<StationsAllResponse>> GetStations(bool fromCache = false)
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Request to get stations{(fromCache ? " from cache" : "")}...");
|
||||
if (!CacheServer.IsStationsExpired
|
||||
|| fromCache)
|
||||
{
|
||||
// No need to query because previous answer is not yet outdated.
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Returning stations from cache.");
|
||||
return new Result<StationsAllResponse>(typeof(CopriCallsMonkeyStore), await CacheServer.GetStationsAsync());
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Querrying stations from copri.");
|
||||
return new Result<StationsAllResponse>(
|
||||
typeof(CopriCallsHttps),
|
||||
(await HttpsServer.GetStationsAsync()).GetIsResponseOk("Abfrage der Stationen fehlsgeschlagen."));
|
||||
}
|
||||
catch (Exception exception)
|
||||
{
|
||||
// Return response from cache.
|
||||
Log.ForContext<CopriProviderHttps>().Debug("An error occurred querrying stations. {Exception}.", exception);
|
||||
return new Result<StationsAllResponse>(typeof(CopriCallsMonkeyStore), await CacheServer.GetStationsAsync(), exception);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>Adds https--response to cache if response is ok. </summary>
|
||||
/// <param name="response">Response to add to cache.</param>
|
||||
/// <returns></returns>
|
||||
public void AddToCache(Result<StationsAllResponse> result)
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Request to add stations all response to cache...");
|
||||
if (result.Source == typeof(CopriCallsMonkeyStore)
|
||||
|| result.Exception != null)
|
||||
{
|
||||
// Do not add responses form cache or invalid responses to cache.
|
||||
return;
|
||||
}
|
||||
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Add bikes available response to cache.");
|
||||
CacheServer.AddToCache(result.Response);
|
||||
}
|
||||
|
||||
/// <summary>Adds https--response to cache if response is ok. </summary>
|
||||
/// <param name="response">Response to add to cache.</param>
|
||||
/// <returns></returns>
|
||||
public void AddToCache(Result<BikesAvailableResponse> result)
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Request to add bikes available response to cache...");
|
||||
if (result.Source == typeof(CopriCallsMonkeyStore)
|
||||
|| result.Exception != null)
|
||||
{
|
||||
// Do not add responses form cache or invalid responses to cache.
|
||||
return;
|
||||
}
|
||||
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Add bikes available response to cache.");
|
||||
CacheServer.AddToCache(result.Response);
|
||||
|
||||
}
|
||||
|
||||
/// <summary>Adds https--response to cache if response is ok. </summary>
|
||||
/// <param name="response">Response to add to cache.</param>
|
||||
/// <returns></returns>
|
||||
public void AddToCache(Result<BikesReservedOccupiedResponse> result)
|
||||
{
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Request to add bikes occupied response to cache...");
|
||||
if (result.Source == typeof(CopriCallsMonkeyStore)
|
||||
|| result.Exception != null)
|
||||
{
|
||||
// Do not add responses form cache or invalid responses to cache.
|
||||
return;
|
||||
}
|
||||
|
||||
Log.ForContext<CopriProviderHttps>().Debug($"Add bikes occupied response to cache.");
|
||||
CacheServer.AddToCache(result.Response);
|
||||
}
|
||||
|
||||
public async Task<AuthorizationResponse> DoAuthorizationAsync(string p_strMailAddress, string p_strPassword, string p_strDeviceId)
|
||||
{
|
||||
return await HttpsServer.DoAuthorizationAsync(p_strMailAddress, p_strPassword, p_strDeviceId);
|
||||
}
|
||||
|
||||
public async Task<AuthorizationoutResponse> DoAuthoutAsync()
|
||||
{
|
||||
return await HttpsServer.DoAuthoutAsync();
|
||||
}
|
||||
|
||||
public async Task<ReservationBookingResponse> DoReserveAsync(int p_iBikeId, Uri operatorUri)
|
||||
{
|
||||
return await HttpsServer.DoReserveAsync(p_iBikeId, operatorUri);
|
||||
}
|
||||
|
||||
public async Task<ReservationCancelReturnResponse> DoCancelReservationAsync(int p_iBikeId, Uri operatorUri)
|
||||
{
|
||||
return await HttpsServer.DoCancelReservationAsync(p_iBikeId, operatorUri);
|
||||
}
|
||||
|
||||
public async Task<ReservationBookingResponse> CalculateAuthKeysAsync(int bikeId, Uri operatorUri)
|
||||
{
|
||||
return await HttpsServer.CalculateAuthKeysAsync(bikeId, operatorUri);
|
||||
}
|
||||
|
||||
public async Task<ReservationBookingResponse> UpdateLockingStateAsync(
|
||||
int bikeId,
|
||||
LocationDto location,
|
||||
lock_state state,
|
||||
double batteryLevel,
|
||||
Uri operatorUri)
|
||||
=> await HttpsServer.UpdateLockingStateAsync(bikeId, location, state, batteryLevel, operatorUri);
|
||||
|
||||
/// <summary> Books a bike. </summary>
|
||||
/// <param name="bikeId">Id of the bike to book.</param>
|
||||
/// <param name="guid">Used to publish GUID from app to copri. Used for initial setup of bike in copri.</param>
|
||||
/// <param name="batteryPercentage">Holds the filling level percentage of the battery.</param>
|
||||
/// <returns>Response on booking request.</returns>
|
||||
public async Task<ReservationBookingResponse> DoBookAsync(int bikeId, Guid guid, double batteryPercentage, Uri operatorUri)
|
||||
{
|
||||
return await HttpsServer.DoBookAsync(bikeId, guid, batteryPercentage, operatorUri);
|
||||
}
|
||||
|
||||
public async Task<ReservationCancelReturnResponse> DoReturn(int bikeId, LocationDto location, Uri operatorUri)
|
||||
{
|
||||
return await HttpsServer.DoReturn(bikeId, location, operatorUri);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Submits feedback to copri server.
|
||||
/// </summary>
|
||||
/// <param name="message">General purpose message or error description.</param>
|
||||
/// <param name="isBikeBroken">True if bike is broken.</param>
|
||||
public async Task<SubmitFeedbackResponse> DoSubmitFeedback(string message, bool isBikeBroken, Uri opertorUri) =>
|
||||
await HttpsServer.DoSubmitFeedback(message, isBikeBroken, opertorUri);
|
||||
}
|
||||
}
|
87
TINKLib/Services/CopriApi/CopriProviderMonkeyStore.cs
Normal file
87
TINKLib/Services/CopriApi/CopriProviderMonkeyStore.cs
Normal file
|
@ -0,0 +1,87 @@
|
|||
using System;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Repository;
|
||||
using TINK.Model.Repository.Request;
|
||||
using TINK.Model.Repository.Response;
|
||||
using TINK.Repository.Response;
|
||||
|
||||
namespace TINK.Model.Services.CopriApi
|
||||
{
|
||||
public class CopriProviderMonkeyStore : ICopriServer
|
||||
{
|
||||
/// <summary> Object which manages stored copri answers. </summary>
|
||||
private readonly CopriCallsMonkeyStore monkeyStore;
|
||||
|
||||
/// <summary> True if connector has access to copri server, false if cached values are used. </summary>
|
||||
public bool IsConnected => monkeyStore.IsConnected;
|
||||
|
||||
/// <summary> Gets the session cookie if user is logged in, an empty string otherwise. </summary>
|
||||
public string SessionCookie => monkeyStore.SessionCookie;
|
||||
|
||||
/// <summary> Constructs object which Object which manages stored copri answers in a thread save way. </summary>
|
||||
/// <param name="merchantId">Id of the merchant TINK-App.</param>
|
||||
public CopriProviderMonkeyStore(
|
||||
string merchantId,
|
||||
string sessionCookie)
|
||||
{
|
||||
monkeyStore = new CopriCallsMonkeyStore(merchantId, sessionCookie);
|
||||
}
|
||||
|
||||
/// <summary> Gets the merchant id.</summary>
|
||||
public string MerchantId => monkeyStore.MerchantId;
|
||||
|
||||
public Task<ReservationBookingResponse> DoReserveAsync(int p_iBikeId, Uri operatorUri)
|
||||
=> throw new NotSupportedException($"{nameof(DoReserveAsync)} is not cachable.");
|
||||
|
||||
public Task<ReservationCancelReturnResponse> DoCancelReservationAsync(int p_iBikeId, Uri operatorUri)
|
||||
=> throw new NotSupportedException($"{nameof(DoCancelReservationAsync)} is not cachable.");
|
||||
|
||||
public Task<ReservationBookingResponse> CalculateAuthKeysAsync(int bikeId, Uri operatorUri)
|
||||
=> throw new NotSupportedException($"{nameof(CalculateAuthKeysAsync)} is not cachable.");
|
||||
|
||||
public async Task<ReservationBookingResponse> UpdateLockingStateAsync(
|
||||
int bikeId,
|
||||
LocationDto geolocation,
|
||||
lock_state state,
|
||||
double batteryLevel,
|
||||
Uri operatorUri)
|
||||
=> await monkeyStore.UpdateLockingStateAsync(bikeId, geolocation, state, batteryLevel, operatorUri);
|
||||
|
||||
public async Task<ReservationBookingResponse> DoBookAsync(int bikeId, Guid guid, double batteryPercentage, Uri operatorUri)
|
||||
{
|
||||
return await monkeyStore.DoBookAsync(bikeId, guid, batteryPercentage, operatorUri);
|
||||
}
|
||||
|
||||
public async Task<ReservationCancelReturnResponse> DoReturn(int bikeId, LocationDto geolocation, Uri operatorUri)
|
||||
{
|
||||
return await monkeyStore.DoReturn(bikeId, geolocation, operatorUri);
|
||||
}
|
||||
|
||||
public Task<SubmitFeedbackResponse> DoSubmitFeedback(string messge, bool bIsBikeBroke, Uri operatorUri) => throw new NotImplementedException();
|
||||
|
||||
public async Task<AuthorizationResponse> DoAuthorizationAsync(string p_strMailAddress, string p_strPassword, string p_strDeviceId)
|
||||
{
|
||||
return await monkeyStore.DoAuthorizationAsync(p_strMailAddress, p_strPassword, p_strDeviceId);
|
||||
}
|
||||
|
||||
public async Task<AuthorizationoutResponse> DoAuthoutAsync()
|
||||
{
|
||||
return await monkeyStore.DoAuthoutAsync();
|
||||
}
|
||||
|
||||
public async Task<BikesAvailableResponse> GetBikesAvailableAsync()
|
||||
{
|
||||
return await monkeyStore.GetBikesAvailableAsync();
|
||||
}
|
||||
|
||||
public async Task<BikesReservedOccupiedResponse> GetBikesOccupiedAsync()
|
||||
{
|
||||
return await monkeyStore.GetBikesOccupiedAsync();
|
||||
}
|
||||
|
||||
public async Task<StationsAllResponse> GetStationsAsync()
|
||||
{
|
||||
return await monkeyStore.GetStationsAsync();
|
||||
}
|
||||
}
|
||||
}
|
37
TINKLib/Services/CopriApi/ICachedCopriServer.cs
Normal file
37
TINKLib/Services/CopriApi/ICachedCopriServer.cs
Normal file
|
@ -0,0 +1,37 @@
|
|||
|
||||
using System;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Repository;
|
||||
using TINK.Model.Repository.Response;
|
||||
|
||||
namespace TINK.Model.Services.CopriApi
|
||||
{
|
||||
/// <summary> Manages cache which holds copri data.</summary>
|
||||
public interface ICachedCopriServer : ICopriServerBase
|
||||
{
|
||||
/// <summary> Get list of stations. </summary>
|
||||
/// <returns>List of all stations.</returns>
|
||||
Task<Result<StationsAllResponse>> GetStations(bool fromCache = false);
|
||||
|
||||
/// <summary> Gets a list of bikes from Copri. </summary>
|
||||
/// <returns>Response holding list of bikes.</returns>
|
||||
Task<Result<BikesAvailableResponse>> GetBikesAvailable(bool fromCache = false);
|
||||
|
||||
/// <summary> Gets a list of bikes reserved/ booked by acctive user from Copri.</summary>
|
||||
/// <returns>Response holding list of bikes.</returns>
|
||||
Task<Result<BikesReservedOccupiedResponse>> GetBikesOccupied(bool fromCache = false);
|
||||
|
||||
/// <summary>Adds https--response to cache if response is ok. </summary>
|
||||
/// <param name="response">Response to add to cache.</param>
|
||||
void AddToCache(Result<StationsAllResponse> result);
|
||||
|
||||
/// <summary>Adds https--response to cache if response is ok. </summary>
|
||||
/// <param name="response">Response to add to cache.</param>
|
||||
void AddToCache(Result<BikesAvailableResponse> result);
|
||||
|
||||
/// <summary>Adds https--response to cache if response is ok. </summary>
|
||||
/// <param name="response">Response to add to cache.</param>
|
||||
/// <returns></returns>
|
||||
void AddToCache(Result<BikesReservedOccupiedResponse> result);
|
||||
}
|
||||
}
|
29
TINKLib/Services/CopriApi/ICopriCache.cs
Normal file
29
TINKLib/Services/CopriApi/ICopriCache.cs
Normal file
|
@ -0,0 +1,29 @@
|
|||
using TINK.Model.Repository;
|
||||
using TINK.Model.Repository.Response;
|
||||
|
||||
namespace TINK.Model.Services.CopriApi
|
||||
{
|
||||
public interface ICopriCache : ICopriServer
|
||||
{
|
||||
/// <summary> Gets a value indicating whether stations are expired or not.</summary>
|
||||
bool IsStationsExpired { get; }
|
||||
|
||||
/// <summary> Adds a stations all response to cache.</summary>
|
||||
/// <param name="stations">Stations to add.</param>
|
||||
void AddToCache(StationsAllResponse stations);
|
||||
|
||||
/// <summary> Gets a value indicating whether stations are expired or not.</summary>
|
||||
bool IsBikesAvailableExpired { get; }
|
||||
|
||||
/// <summary> Adds a bikes response to cache.</summary>
|
||||
/// <param name="bikes">Bikes to add.</param>
|
||||
void AddToCache(BikesAvailableResponse bikes);
|
||||
|
||||
/// <summary> Gets a value indicating whether stations are expired or not.</summary>
|
||||
bool IsBikesOccupiedExpired { get; }
|
||||
|
||||
/// <summary> Adds a bikes response to cache.</summary>
|
||||
/// <param name="bikes">Bikes to add.</param>
|
||||
void AddToCache(BikesReservedOccupiedResponse bikes);
|
||||
}
|
||||
}
|
23
TINKLib/Services/CopriApi/Result.cs
Normal file
23
TINKLib/Services/CopriApi/Result.cs
Normal file
|
@ -0,0 +1,23 @@
|
|||
using System;
|
||||
|
||||
namespace TINK.Model.Services.CopriApi
|
||||
{
|
||||
public class Result<T> where T : class
|
||||
{
|
||||
public Result(Type source, T response, System.Exception exception = null)
|
||||
{
|
||||
Source = source ?? throw new ArgumentException(nameof(source));
|
||||
Response = response ?? throw new ArgumentException(nameof(response));
|
||||
Exception = exception;
|
||||
}
|
||||
|
||||
/// <summary> Holds the copri respsonse</summary>
|
||||
public T Response { get; }
|
||||
|
||||
/// <summary> Specifies the souce of the copri response.</summary>
|
||||
public Type Source { get; }
|
||||
|
||||
/// <summary> Holds the exception if a communication error occurred.</summary>
|
||||
public System.Exception Exception { get; private set; }
|
||||
}
|
||||
}
|
45
TINKLib/Services/CopriApi/ServerUris/CopriHelper.cs
Normal file
45
TINKLib/Services/CopriApi/ServerUris/CopriHelper.cs
Normal file
|
@ -0,0 +1,45 @@
|
|||
using System;
|
||||
using TINK.Model.Services.CopriApi.ServerUris;
|
||||
|
||||
namespace TINK.Services.CopriApi.ServerUris
|
||||
{
|
||||
public static class CopriHelper
|
||||
{
|
||||
public const string SHAREE_SILTEFOLDERNAME = "site";
|
||||
|
||||
/// <summary> Gets a value indicating whether a host is copri or not. </summary>
|
||||
/// <param name="hostName">Host name.</param>
|
||||
/// <returns>True if server is copri, fals if not.</returns>
|
||||
public static bool GetIsCopri(this string hostName)
|
||||
=> new Uri(CopriServerUriList.TINK_DEVEL).Host == hostName
|
||||
|| new Uri(CopriServerUriList.TINK_LIVE).Host == hostName;
|
||||
|
||||
/// <summary> Get folder name depending on host name. </summary>
|
||||
/// <param name="hostName">Host name.</param>
|
||||
/// <returns>Folder name.</returns>
|
||||
public static string GetAppFolderName(this string hostName)
|
||||
=> hostName.GetIsCopri()
|
||||
? "tinkapp" /* resource tree is more complex for TINK/ konrad*/
|
||||
: "app";
|
||||
|
||||
/// <summary> Get folder name for a static site depending on host name. </summary>
|
||||
/// <param name="hostName">Host name.</param>
|
||||
/// <returns>Folder name.</returns>
|
||||
public static string GetSiteFolderName(this string hostName)
|
||||
=> hostName.GetIsCopri()
|
||||
? "tinkapp" /* resource tree is more complex for TINK/ konrad*/
|
||||
: SHAREE_SILTEFOLDERNAME;
|
||||
|
||||
/// <summary> Get the agb resource name name depending on host name. </summary>
|
||||
/// <param name="hostName">Host name.</param>
|
||||
/// <returns>AGB resource..</returns>
|
||||
public static string GetAGBResource(this string hostName)
|
||||
=> $"{hostName.GetSiteFolderName()}/{(hostName.GetIsCopri()? "konrad-TINK-AGB" : "agb.html")}";
|
||||
|
||||
/// <summary> Get the agb resource name name depending on host name. </summary>
|
||||
/// <param name="hostName">Host name.</param>
|
||||
/// <returns>AGB resource..</returns>
|
||||
public static string GetPrivacyResource(this string hostName)
|
||||
=> $"{hostName.GetSiteFolderName()}/{(hostName.GetIsCopri() ? "Datenschutz" : "privacy.html")}";
|
||||
}
|
||||
}
|
88
TINKLib/Services/CopriApi/ServerUris/CopriServerUriList.cs
Normal file
88
TINKLib/Services/CopriApi/ServerUris/CopriServerUriList.cs
Normal file
|
@ -0,0 +1,88 @@
|
|||
using Newtonsoft.Json;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
|
||||
namespace TINK.Model.Services.CopriApi.ServerUris
|
||||
{
|
||||
[JsonObject(MemberSerialization.OptIn)]
|
||||
sealed public class CopriServerUriList
|
||||
{
|
||||
/// <summary>
|
||||
/// Holds the rest resource root name.
|
||||
/// </summary>
|
||||
public const string REST_RESOURCE_ROOT = "APIjsonserver";
|
||||
|
||||
/// <summary> Holds the URL of the TINK/Konrad test server.</summary>
|
||||
public const string TINK_DEVEL = @"https://tinkwwp.copri-bike.de/APIjsonserver";
|
||||
|
||||
/// <summary> Holds the URL the TINK/Konrad productive server.</summary>
|
||||
public const string TINK_LIVE = @"https://app.tink-konstanz.de/APIjsonserver";
|
||||
|
||||
/// <summary> Holds the URL the Sharee server.</summary>
|
||||
public const string SHAREE_DEVEL = @"https://shareeapp-primary.copri-bike.de/APIjsonserver";
|
||||
|
||||
/// <summary> Holds the URL the Sharee server.</summary>
|
||||
public const string SHAREE_LIVE = @"https://shareeapp-primary.copri.eu/APIjsonserver";
|
||||
|
||||
/// <summary>Constructs default uri list.</summary>
|
||||
public CopriServerUriList(Uri activeUri = null) : this (
|
||||
new List<Uri>
|
||||
{
|
||||
new Uri(SHAREE_LIVE),
|
||||
new Uri(TINK_LIVE),
|
||||
new Uri(SHAREE_DEVEL),
|
||||
new Uri(TINK_DEVEL)
|
||||
}.ToArray(),
|
||||
activeUri ?? new Uri(SHAREE_LIVE)) // Default URi which is used after install of app
|
||||
{
|
||||
}
|
||||
|
||||
/// <summary> Constructs uris object. </summary>
|
||||
/// <param name="p_oSource">Object to copy from.</param>
|
||||
public CopriServerUriList(CopriServerUriList p_oSource) : this(
|
||||
p_oSource.Uris.ToArray(),
|
||||
p_oSource.ActiveUri)
|
||||
{
|
||||
}
|
||||
|
||||
/// <summary> Constructs a valid uris object. </summary>
|
||||
/// <param name="uris">Known uris.</param>
|
||||
/// <param name="p_oActiveUri">Zero based index of active uri.</param>
|
||||
/// <param name="p_oDevelUri">Uri of the development server.</param>
|
||||
public CopriServerUriList(
|
||||
Uri[] uris,
|
||||
Uri p_oActiveUri)
|
||||
{
|
||||
if (uris == null || uris.Length < 1)
|
||||
{
|
||||
throw new ArgumentException($"Can not construct {typeof(CopriServerUriList)}- object. Array of uris must not be null and contain at least one element.");
|
||||
}
|
||||
|
||||
if (!uris.Contains(p_oActiveUri))
|
||||
{
|
||||
throw new ArgumentException($"Active uri {p_oActiveUri} not contained in ({string.Join("; ", uris.Select(x => x.AbsoluteUri))}).");
|
||||
}
|
||||
|
||||
Uris = new List<Uri>(uris);
|
||||
ActiveUri = p_oActiveUri;
|
||||
}
|
||||
|
||||
/// <summary> Gets the active uri. </summary>
|
||||
[JsonProperty]
|
||||
public Uri ActiveUri { get; }
|
||||
|
||||
/// <summary> Gets the active uri. </summary>
|
||||
public static Uri DevelopUri => new(TINK_DEVEL);
|
||||
|
||||
/// <summary> Gets the known uris. </summary>
|
||||
[JsonProperty]
|
||||
public IList<Uri> Uris { get ; }
|
||||
|
||||
/// <summary> Gets the default uri which is active after first installation. </summary>
|
||||
public static Uri DefaultActiveUri
|
||||
{
|
||||
get { return new CopriServerUriList().ActiveUri; }
|
||||
}
|
||||
}
|
||||
}
|
18
TINKLib/Services/CopriApi/StationsAndBikesContainer.cs
Normal file
18
TINKLib/Services/CopriApi/StationsAndBikesContainer.cs
Normal file
|
@ -0,0 +1,18 @@
|
|||
using TINK.Model.Bike;
|
||||
using TINK.Model.Station;
|
||||
|
||||
namespace TINK.Model.Services.CopriApi
|
||||
{
|
||||
public class StationsAndBikesContainer
|
||||
{
|
||||
public StationsAndBikesContainer(StationDictionary stations, BikeCollection bikes)
|
||||
{
|
||||
StationsAll = stations;
|
||||
Bikes = bikes;
|
||||
}
|
||||
|
||||
public StationDictionary StationsAll { get; }
|
||||
|
||||
public BikeCollection Bikes { get; }
|
||||
}
|
||||
}
|
59
TINKLib/Services/Geolocation/GeolocationService.cs
Normal file
59
TINKLib/Services/Geolocation/GeolocationService.cs
Normal file
|
@ -0,0 +1,59 @@
|
|||
using Serilog;
|
||||
using System;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Device;
|
||||
using Xamarin.Essentials;
|
||||
|
||||
namespace TINK.Model.Services.Geolocation
|
||||
{
|
||||
public class GeolocationService : IGeolocation
|
||||
{
|
||||
/// <summary> Timeout for geolocation request operations.</summary>
|
||||
private const int GEOLOCATIONREQUEST_TIMEOUT_MS = 5000;
|
||||
|
||||
private IGeolodationDependent Dependent { get; }
|
||||
|
||||
public GeolocationService(IGeolodationDependent dependent)
|
||||
{
|
||||
Dependent = dependent;
|
||||
}
|
||||
|
||||
public bool IsSimulation => false;
|
||||
|
||||
public bool IsGeolcationEnabled => Dependent.IsGeolcationEnabled;
|
||||
|
||||
/// <param name="timeStamp">Time when geolocation is of interest. Is used to determine whether cached geoloation can be used or not.</param>
|
||||
public async Task<Location> GetAsync(DateTime? timeStamp = null)
|
||||
{
|
||||
try
|
||||
{
|
||||
var request = new GeolocationRequest(GeolocationAccuracy.Medium, TimeSpan.FromMilliseconds(GEOLOCATIONREQUEST_TIMEOUT_MS));
|
||||
return await Xamarin.Essentials.Geolocation.GetLocationAsync(request);
|
||||
}
|
||||
catch (FeatureNotSupportedException fnsEx)
|
||||
{
|
||||
// Handle not supported on device exception
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation not supported on device. {Exception}", fnsEx);
|
||||
throw new Exception("Abfrage Standort nicht möglich auf Gerät.", fnsEx);
|
||||
}
|
||||
catch (FeatureNotEnabledException fneEx)
|
||||
{
|
||||
// Handle not enabled on device exception
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation not enabled on device. {Exception}", fneEx);
|
||||
throw new Exception("Abfrage Standort nicht aktiviert auf Gerät.", fneEx);
|
||||
}
|
||||
catch (PermissionException pEx)
|
||||
{
|
||||
// Handle permission exception
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation not permitted on device. {Exception}", pEx);
|
||||
throw new Exception("Berechtiung für Abfrage Standort nicht erteilt für App.", pEx);
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
// Unable to get location
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation failed. {Exception}", ex);
|
||||
throw new Exception("Abfrage Standort fehlgeschlagen.", ex);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
19
TINKLib/Services/Geolocation/IGeolocation.cs
Normal file
19
TINKLib/Services/Geolocation/IGeolocation.cs
Normal file
|
@ -0,0 +1,19 @@
|
|||
using System;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Device;
|
||||
using Xamarin.Essentials;
|
||||
|
||||
namespace TINK.Model.Services.Geolocation
|
||||
{
|
||||
/// <summary> Query geolocation. </summary>
|
||||
public interface IGeolocation : IGeolodationDependent
|
||||
{
|
||||
/// <summary> Gets the current location.</summary>
|
||||
/// <param name="timeStamp">Time when geolocation is of interest. Is used to determine for some implementations whether cached geoloation can be used or not.</param>
|
||||
/// <returns></returns>
|
||||
Task<Location> GetAsync(DateTime? timeStamp = null);
|
||||
|
||||
/// <summary> If true location data returned is simulated.</summary>
|
||||
bool IsSimulation { get; }
|
||||
}
|
||||
}
|
72
TINKLib/Services/Geolocation/LastKnownGeolocationService.cs
Normal file
72
TINKLib/Services/Geolocation/LastKnownGeolocationService.cs
Normal file
|
@ -0,0 +1,72 @@
|
|||
using Serilog;
|
||||
using System;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Device;
|
||||
using Xamarin.Essentials;
|
||||
|
||||
namespace TINK.Model.Services.Geolocation
|
||||
{
|
||||
public class LastKnownGeolocationService : IGeolocation
|
||||
{
|
||||
/// <summary> Timeout for geolocation request operations.</summary>
|
||||
private const int GEOLOCATIONREQUEST_TIMEOUT_MS = 5000;
|
||||
|
||||
private IGeolodationDependent Dependent { get; }
|
||||
|
||||
public LastKnownGeolocationService(IGeolodationDependent dependent)
|
||||
{
|
||||
Dependent = dependent;
|
||||
}
|
||||
|
||||
/// <param name="timeStamp">Time when geolocation is of interest. Is used to determine whether cached geoloation can be used or not.</param>
|
||||
public async Task<Location> GetAsync(DateTime? timeStamp = null)
|
||||
{
|
||||
Location location;
|
||||
try
|
||||
{
|
||||
var request = new GeolocationRequest(GeolocationAccuracy.Medium, TimeSpan.FromMilliseconds(GEOLOCATIONREQUEST_TIMEOUT_MS));
|
||||
location = await Xamarin.Essentials.Geolocation.GetLocationAsync(request);
|
||||
}
|
||||
catch (FeatureNotSupportedException fnsEx)
|
||||
{
|
||||
// Handle not supported on device exception
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation not supported on device. {Exception}", fnsEx);
|
||||
throw new Exception("Abfrage Standort nicht möglich auf Gerät.", fnsEx);
|
||||
}
|
||||
catch (FeatureNotEnabledException fneEx)
|
||||
{
|
||||
// Handle not enabled on device exception
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation not enabled on device. {Exception}", fneEx);
|
||||
throw new Exception("Abfrage Standort nicht aktiviert auf Gerät.", fneEx);
|
||||
}
|
||||
catch (PermissionException pEx)
|
||||
{
|
||||
// Handle permission exception
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation not permitted on device. {Exception}", pEx);
|
||||
throw new Exception("Berechtiung für Abfrage Standort nicht erteilt für App.", pEx);
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
// Unable to get location
|
||||
Log.ForContext<LastKnownGeolocationService>().Error("Retrieving Geolocation failed. {Exception}", ex);
|
||||
throw new Exception("Abfrage Standort fehlgeschlagen.", ex);
|
||||
}
|
||||
|
||||
if (location != null // Cached location is available.
|
||||
&& (timeStamp == null || timeStamp.Value.Subtract(location.Timestamp.DateTime) < MaxAge))
|
||||
{
|
||||
// No time stamp available or location not too old.
|
||||
return location;
|
||||
}
|
||||
|
||||
return await Xamarin.Essentials.Geolocation.GetLocationAsync();
|
||||
}
|
||||
|
||||
/// <summary> If true location data returned is simulated.</summary>
|
||||
public bool IsSimulation { get => false; }
|
||||
|
||||
public TimeSpan MaxAge => new TimeSpan(0, 3, 0);
|
||||
|
||||
public bool IsGeolcationEnabled => Dependent.IsGeolcationEnabled;
|
||||
}
|
||||
}
|
27
TINKLib/Services/Geolocation/SimulatedGeolocationService.cs
Normal file
27
TINKLib/Services/Geolocation/SimulatedGeolocationService.cs
Normal file
|
@ -0,0 +1,27 @@
|
|||
using System;
|
||||
using System.Threading.Tasks;
|
||||
using TINK.Model.Device;
|
||||
using Xamarin.Essentials;
|
||||
|
||||
namespace TINK.Model.Services.Geolocation
|
||||
{
|
||||
public class SimulatedGeolocationService : IGeolocation
|
||||
{
|
||||
private IGeolodationDependent Dependent { get; }
|
||||
|
||||
public SimulatedGeolocationService(IGeolodationDependent dependent)
|
||||
{
|
||||
Dependent = dependent;
|
||||
}
|
||||
|
||||
public async Task<Location> GetAsync(DateTime? timeStamp = null)
|
||||
{
|
||||
return await Task.FromResult(new Location(47.976634, 7.825490) { Accuracy = 0, Timestamp = timeStamp ?? DateTime.Now }); ;
|
||||
}
|
||||
|
||||
/// <summary> If true location data returned is simulated.</summary>
|
||||
public bool IsSimulation { get => true; }
|
||||
|
||||
public bool IsGeolcationEnabled => Dependent.IsGeolcationEnabled;
|
||||
}
|
||||
}
|
57
TINKLib/Services/ServicesContainerMutable.cs
Normal file
57
TINKLib/Services/ServicesContainerMutable.cs
Normal file
|
@ -0,0 +1,57 @@
|
|||
using System;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using System.ComponentModel;
|
||||
using System.Linq;
|
||||
|
||||
namespace TINK.Services
|
||||
{
|
||||
/// <summary> Container of service objects (locks , geolocation, ...) where one service is active. </summary>
|
||||
/// <remarks> All service objects must be of different type. </remarks>
|
||||
public class ServicesContainerMutable<T>: IEnumerable<T>, INotifyPropertyChanged
|
||||
{
|
||||
private readonly Dictionary<string, T> serviceDict;
|
||||
|
||||
public event PropertyChangedEventHandler PropertyChanged;
|
||||
|
||||
/// <summary> Constructs object on startup of app.</summary>
|
||||
/// <param name="services">Fixed list of service- objects .</param>
|
||||
/// <param name="activeName">LocksService name which is activated on startup of app.</param>
|
||||
public ServicesContainerMutable(
|
||||
IEnumerable<T> services,
|
||||
string activeName)
|
||||
{
|
||||
serviceDict = services.Distinct().ToDictionary(x => x.GetType().FullName);
|
||||
|
||||
if (!serviceDict.ContainsKey(activeName))
|
||||
{
|
||||
throw new ArgumentException($"Can not instantiate {typeof(T).Name}- object. Active lock service {activeName} must be contained in [{String.Join(",", serviceDict)}].");
|
||||
}
|
||||
|
||||
Active = serviceDict[activeName];
|
||||
}
|
||||
|
||||
/// <summary> Active locks service.</summary>
|
||||
public T Active { get; private set; }
|
||||
|
||||
/// <summary> Sets a lock service as active locks service by name. </summary>
|
||||
/// <param name="active">Name of the new locks service.</param>
|
||||
public void SetActive(string active)
|
||||
{
|
||||
if (!serviceDict.ContainsKey(active))
|
||||
{
|
||||
throw new ArgumentException($"Can not set active lock service {active}. Service must be contained in [{String.Join(",", serviceDict)}].");
|
||||
}
|
||||
|
||||
Active = serviceDict[active];
|
||||
|
||||
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Active)));
|
||||
}
|
||||
|
||||
public IEnumerator<T> GetEnumerator()
|
||||
=> serviceDict.Values.GetEnumerator();
|
||||
|
||||
IEnumerator IEnumerable.GetEnumerator()
|
||||
=> serviceDict.Values.GetEnumerator();
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue