using System; using System.Collections.Generic; using System.Linq; using System.Security.Authentication; using System.Text; using System.Threading.Tasks; using System.Timers; using Tango.BL; using Tango.BL.Builders; using Tango.BL.Entities; using Tango.BL.Enumerations; using Tango.Core; using Tango.Core.DI; using Tango.Core.ExtensionMethods; using Tango.FSE.BL; using Tango.FSE.BL.Web; using Tango.FSE.Common; using Tango.FSE.Common.Authentication; using Tango.FSE.Common.FSEApplication; using Tango.FSE.Common.Navigation; using Tango.FSE.Common.Notifications; using Tango.FSE.UI.ViewModels; using Tango.FSE.Web.Messages; using Tango.MachineService.Gateway; using Tango.Settings; namespace Tango.FSE.UI.Authentication { /// /// Represents the default FSE authentication provider. /// /// /// public class DefaultAuthenticationProvider : ExtendedObject, IAuthenticationProvider { private const string WEB_DEBUG_ADDRESS = "http://localhost:1111"; private Timer _tokenRefreshTimer; [TangoInject(TangoInjectMode.WhenAvailable)] private IFSEApplicationManager ApplicationManager { get; set; } [TangoInject(TangoInjectMode.WhenAvailable)] private INavigationManager NavigationManager { get; set; } [TangoInject(TangoInjectMode.WhenAvailable)] private INotificationProvider NotificationProvider { get; set; } [TangoInject(TangoInjectMode.WhenAvailable)] private FSEServicesContainer Services { get; set; } [TangoInject(TangoInjectMode.WhenAvailable)] private FSEWebClient WebClient { get; set; } /// /// Occurs when the current logged-in user has changed. /// public event EventHandler CurrentUserChanged; private User _currentUser; /// /// Gets the current logged-in user. /// public User CurrentUser { get { return _currentUser; } private set { _currentUser = value; RaisePropertyChangedAuto(); } } private EnvironmentConfiguration _currentEnvironment; /// /// Gets the current environment configuration. /// public EnvironmentConfiguration CurrentEnvironment { get { return _currentEnvironment; } private set { _currentEnvironment = value; RaisePropertyChangedAuto(); } } /// /// Initializes a new instance of the class. /// public DefaultAuthenticationProvider() { _tokenRefreshTimer = new Timer(TimeSpan.FromMinutes(30).TotalMilliseconds); _tokenRefreshTimer.Elapsed += OnRefreshTokenTimer; _tokenRefreshTimer.Stop(); } /// /// Logins the specified email. /// /// The email. /// The password. /// The environment. /// public Task Login(String email, String password, EnvironmentConfiguration environment, Action onStatusChanged = null) { return Task.Factory.StartNew(() => { _tokenRefreshTimer.Stop(); LogManager.Log($"Authenticating user '{email}'..."); onStatusChanged?.Invoke($"Authenticating user {email}..."); var settings = SettingsManager.Default.GetOrCreate(); if (ApplicationManager.IsWebDebugMode) { LogManager.Log($"Application is currently in web debug mode. Authenticating through '{WEB_DEBUG_ADDRESS}'."); WebClient.Address = WEB_DEBUG_ADDRESS; } else { LogManager.Log($"Authenticating through machine service at '{environment.MachineServiceAddress}'..."); WebClient.Address = environment.MachineServiceAddress; } var appVersion = ApplicationManager.Version.ToString(); if (settings.ForceVersionUpdate) { LogManager.Log("Application settings are set to force version update. Setting reported application version to '1.0.0.0'."); appVersion = "1.0.0.0"; } LoginResponse response = null; try { onStatusChanged?.Invoke($"Contacting machine service at '{WebClient.Address}'..."); CurrentEnvironment = environment; response = Services.AuthenticationService.Authenticate(new LoginRequest() { Email = email, Password = password, Version = appVersion, Build = BuildProvider.Build }).Result; LogManager.Log($"Authentication response received:\n{response.ToJsonString()}"); settings.Save(); } catch (Exception ex) { LogManager.Log(ex, "Error authenticating user."); throw new AggregateException(new AuthenticationException("Error logging in to machine service."), ex); } if (settings.Environment == FSESettings.WorkingEnvironment.Remote) { ObservablesContext.OverrideSettingsDataSource(response.DataSource); } if (response.PasswordChangeRequired) { LogManager.Log("User password change required."); Services.AuthenticationService.DropCachedResponse(email); return new AuthenticationResult() { Response = response }; } LogManager.Log("Retrieving full user details..."); User user = Services.UsersService.GetUserFull(email.ToLower()).Result; if (user != null) { LogManager.Log($"Current user is now: {user.Contact.FullName}."); } else { throw LogManager.Log(new AuthenticationException("Login failed for some strange reason.")); } _tokenRefreshTimer.Start(); CurrentUser = user; settings.Save(); InvokeUI(() => { CurrentUserChanged?.Invoke(this, CurrentUser); }); return new AuthenticationResult() { Response = response, User = CurrentUser, }; }); } /// /// Logs-out the current logged-in user. /// public async void LogOut() { LogManager.Log("Logging out current user..."); using (NotificationProvider.PushTaskItem("Logging out...")) { await Task.Delay(1000); LogManager.Log($"Invoking {nameof(FSEViewModel.OnApplicationLogout)} on all FSE view models..."); foreach (var vm in TangoIOC.Default.GetAllInstancesByBase()) { try { if (!await vm.OnApplicationLogout()) { return; } } catch (Exception ex) { LogManager.Log(ex, $"An error occurred on ApplicationLogOut on {vm.GetType().FullName}."); } } await Task.Delay(1000); } LogManager.Log("Logging out current user."); CurrentUser = null; CurrentUserChanged?.Invoke(this, CurrentUser); LogManager.Log($"Navigating to {nameof(NavigationView.LoginView)}..."); _tokenRefreshTimer.Stop(); MemoryCacheManager.Default.ClearAll(); await NavigationManager.NavigateWithObject(NavigationView.LoginView, new LoginViewVM.NavigationObject() { IsLoggingOut = true, }, false); } /// /// Throws an exception if the current user does not have the specified permission. /// /// The permission. /// Optional message. /// /// public void ThrowIfNoPermission(Permissions permission, String message = null) { if (CurrentUser == null || !CurrentUser.HasPermission(permission)) { if (message == null) { throw new AuthorizationException(); } else { throw new AuthorizationException(message); } } } /// /// Changes the specified user password. /// /// The email. /// The old password. /// The new password. /// public Task ChangePassword(string email, string oldPassword, string newPassword) { return Services.AuthenticationService.ChangePassword(email, oldPassword, newPassword); } private async void OnRefreshTokenTimer(object sender, ElapsedEventArgs e) { _tokenRefreshTimer.Stop(); if (ObservablesContext.GetActualDataSource().Type == DataSourceType.AccessToken) { try { LogManager.Log("Refreshing database access token..."); var response = await WebClient.RefreshToken(new RefreshTokenRequest()); ObservablesContext.UpdateAccessToken(response.AccessToken, response.Expiration); } catch (Exception ex) { LogManager.Log(ex, "Error occurred while trying to refresh the database access token."); } } _tokenRefreshTimer.Start(); } } }