diff --git a/BankingBot/ActionManagers/AccountManagers/AccountManager.cs b/BankingBot/ActionManagers/AccountManagers/AccountManager.cs index d75dc66..51f08b6 100644 --- a/BankingBot/ActionManagers/AccountManagers/AccountManager.cs +++ b/BankingBot/ActionManagers/AccountManagers/AccountManager.cs @@ -1,23 +1,35 @@ using BankingBot.Contracts; using System; using System.Collections.Generic; -using System.Linq; -using System.Text; -using System.Threading.Tasks; using BankingBot.Models; +using BankingBot.Enums; +using BankingBot.ScriptManagement; namespace BankingBot.ActionManagers.AccountManagers { public class AccountManager : ActionManager, IAccountManager { + protected IProviderAccountManager providerAccountManager; + private Provider _provider; + public AccountManager(IBrowserBot browserBot) : base(browserBot) + { } + + public void Init(Provider provider) { + _provider = provider; + + // TODO: Implement DI to get rid of this new() crap + var scriptManager = new ScriptManager(BrowserBot); + + var providerAccountManagerType = GetTypeFromInterface(provider, typeof(IProviderAccountManager)); + providerAccountManager = (IProviderAccountManager)Activator.CreateInstance(providerAccountManagerType, BrowserBot, scriptManager); } public IEnumerable GetAccounts() { - throw new NotImplementedException(); + return providerAccountManager.GetAccounts(); } } } diff --git a/BankingBot/ActionManagers/AccountManagers/LloydsAccountManager.cs b/BankingBot/ActionManagers/AccountManagers/LloydsAccountManager.cs new file mode 100644 index 0000000..dae8295 --- /dev/null +++ b/BankingBot/ActionManagers/AccountManagers/LloydsAccountManager.cs @@ -0,0 +1,53 @@ +using BankingBot.Attributes; +using BankingBot.Contracts; +using BankingBot.Enums; +using BankingBot.Models; +using BankingBot.Urls; +using OpenQA.Selenium; +using System; +using System.Collections.Generic; +using System.Globalization; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace BankingBot.ActionManagers.AccountManagers +{ + [ProviderIdentifier(Provider.Lloyds)] + public class LloydsAccountManager : IProviderAccountManager + { + readonly IBrowserBot browserBot; + readonly IScriptManager scriptManager; + + public LloydsAccountManager( + IBrowserBot browserBot, + IScriptManager scriptManager) + { + this.browserBot = browserBot; + this.scriptManager = scriptManager; + } + + public IEnumerable GetAccounts() + { + var accounts = new List(); + + var accountsContainer = browserBot.WebDriver.FindElements(By.ClassName("des-m-sat-xx-account-information")); + foreach (var container in accountsContainer) + { + var account = new Account(); + account.Name = container.FindElement(By.ClassName("account-name")).Text; + account.AccountNumber = container.FindElement(By.ClassName("account-number")).Text; + account.SortCode = container.FindElement(By.CssSelector("dd[aria-label='12 34 56']")).Text; + + var balance = 0m; + var balanceTxt = container.FindElement(By.ClassName("balance")).FindElement(By.TagName("span")).Text; + decimal.TryParse(balanceTxt, NumberStyles.Currency, new CultureInfo("en-GB"), out balance); + account.Balance = balance; + + accounts.Add(account); + } + + return accounts; + } + } +} diff --git a/BankingBot/ActionManagers/ActionDetail.cs b/BankingBot/ActionManagers/ActionDetail.cs new file mode 100644 index 0000000..0e72ddf --- /dev/null +++ b/BankingBot/ActionManagers/ActionDetail.cs @@ -0,0 +1,12 @@ +using BankingBot.Enums; +using System; + +namespace BankingBot.ActionManagers +{ + public class ActionDetail + { + public Type Type { get; set; } + + public Provider Provider { get; set; } + } +} diff --git a/BankingBot/ActionManagers/ActionManager.cs b/BankingBot/ActionManagers/ActionManager.cs index d64c30c..f0070b9 100644 --- a/BankingBot/ActionManagers/ActionManager.cs +++ b/BankingBot/ActionManagers/ActionManager.cs @@ -4,8 +4,6 @@ using BankingBot.Enums; using System; using System.Collections.Generic; using System.Linq; -using System.Text; -using System.Threading.Tasks; namespace BankingBot.ActionManagers { @@ -18,22 +16,38 @@ namespace BankingBot.ActionManagers BrowserBot = browserBot; } - protected Type GetActionTypeFromInterface(object identifyingType, Type interfaceType) + protected ActionDetail GetActionDetailFromInterface(object identifyingType, Type interfaceType) { var provider = ProviderIdentifier.GetProviderFromType(identifyingType.GetType()); + var type = GetTypeAssociatedWithProvider(provider, interfaceType); - // Get all types implementing the given interface - var typesImplementingInterface = AppDomain.CurrentDomain.GetAssemblies() + return new ActionDetail { Provider = provider, Type = type }; + } + + protected Type GetTypeFromInterface(Provider provider, Type interfaceType) + { + return GetTypeAssociatedWithProvider(provider, interfaceType); + } + + private static IEnumerable GetTypesImplementingInterface(Type interfaceType) + { + return AppDomain.CurrentDomain.GetAssemblies() .SelectMany(s => s.GetTypes()) .Where(p => interfaceType.IsAssignableFrom(p) && - p != interfaceType); + p != interfaceType).ToList(); + } + private static Type GetTypeAssociatedWithProvider(Provider provider, Type interfaceType) + { + var typesImplementingInterface = GetTypesImplementingInterface(interfaceType); foreach (var type in typesImplementingInterface) { var typeProvider = ProviderIdentifier.GetProviderFromType(type); if (typeProvider == provider) + { return type; + } } return null; diff --git a/BankingBot/ActionManagers/LoginManagers/LloydsLoginManager.cs b/BankingBot/ActionManagers/LoginManagers/LloydsLoginManager.cs index 13ccd1e..7a622c5 100644 --- a/BankingBot/ActionManagers/LoginManagers/LloydsLoginManager.cs +++ b/BankingBot/ActionManagers/LoginManagers/LloydsLoginManager.cs @@ -8,19 +8,13 @@ using OpenQA.Selenium; using BankingBot.Responses; using BankingBot.Enums; using BankingBot.ScriptManagement; +using BankingBot.Urls; namespace BankingBot.ActionManagers.LoginManagers { [ProviderIdentifier(Provider.Lloyds)] public class LloydsLoginManager : IProviderLoginManager { - private static class Urls - { - public const string Login = "https://online.lloydsbank.co.uk/personal/logon/login.jsp"; - public const string MemorableInfo = "https://secure.lloydsbank.co.uk/personal/a/logon/entermemorableinformation.jsp"; - public const string AccountOverview = "https://secure.lloydsbank.co.uk/personal/a/account_overview_personal/"; - } - private readonly IBrowserBot _browserBot; private readonly IScriptManager _scriptManager; @@ -41,12 +35,12 @@ namespace BankingBot.ActionManagers.LoginManagers { LoginStep1(lloydsCreds); - if (!_browserBot.WebDriver.Url.Contains(Urls.MemorableInfo)) + if (!_browserBot.WebDriver.Url.Contains(LloydsUrls.MemorableInfo)) throw new InvalidOperationException("Invalid login credentials"); LoginStep2(lloydsCreds); - if (!_browserBot.WebDriver.Url.Contains(Urls.AccountOverview)) + if (!_browserBot.WebDriver.Url.Contains(LloydsUrls.AccountOverview)) throw new InvalidOperationException("Invalid passphrase for account"); response.Status = ResponseStatus.Success; @@ -62,7 +56,7 @@ namespace BankingBot.ActionManagers.LoginManagers private void LoginStep1(LloydsLoginCredentials credentials) { - _browserBot.WebDriver.Url = Urls.Login; + _browserBot.WebDriver.Url = LloydsUrls.Login; _browserBot.WebDriver.Navigate(); _browserBot.WebDriver.FindElement(By.Id("frmLogin:strCustomerLogin_userID")).SendKeys(credentials.Username); @@ -93,7 +87,7 @@ namespace BankingBot.ActionManagers.LoginManagers private int[] GetPassphraseIndexes() { - if (_browserBot.WebDriver.Url != Urls.MemorableInfo) + if (_browserBot.WebDriver.Url != LloydsUrls.MemorableInfo) throw new InvalidOperationException("Must be on the memorable info page"); var charIndexes = new int[3]; diff --git a/BankingBot/ActionManagers/LoginManagers/LoginManager.cs b/BankingBot/ActionManagers/LoginManagers/LoginManager.cs index 4e98965..fa07ae0 100644 --- a/BankingBot/ActionManagers/LoginManagers/LoginManager.cs +++ b/BankingBot/ActionManagers/LoginManagers/LoginManager.cs @@ -18,8 +18,8 @@ namespace BankingBot.ActionManagers.LoginManagers // TODO: THIS NEEDS TO BE MOVED var scriptManager = new ScriptManager(BrowserBot); - var provLoginManagerType = GetActionTypeFromInterface(credentials, typeof(IProviderLoginManager)); - var provLoginManager = (IProviderLoginManager)Activator.CreateInstance(provLoginManagerType, BrowserBot, scriptManager); + var providerLoginManagerType = GetTypeFromInterface(credentials.GetProvider(), typeof(IProviderLoginManager)); + var provLoginManager = (IProviderLoginManager)Activator.CreateInstance(providerLoginManagerType, BrowserBot, scriptManager); return provLoginManager.Login(credentials); } diff --git a/BankingBot/Attributes/ProviderIdentifier.cs b/BankingBot/Attributes/ProviderIdentifier.cs index 8d5ced6..f813a47 100644 --- a/BankingBot/Attributes/ProviderIdentifier.cs +++ b/BankingBot/Attributes/ProviderIdentifier.cs @@ -12,7 +12,7 @@ namespace BankingBot.Attributes Provider = provider; } - public static Provider? GetProviderFromType(Type t) + public static Provider GetProviderFromType(Type t) { foreach (var attr in t.GetCustomAttributes(false)) { @@ -22,7 +22,7 @@ namespace BankingBot.Attributes } } - return null; + throw new InvalidOperationException("Could not find associated provider for given type"); } } } diff --git a/BankingBot/BankingBot.csproj b/BankingBot/BankingBot.csproj index 2eb2700..3a35506 100644 --- a/BankingBot/BankingBot.csproj +++ b/BankingBot/BankingBot.csproj @@ -49,6 +49,8 @@ + + @@ -59,6 +61,7 @@ + @@ -77,6 +80,7 @@ + diff --git a/BankingBot/Client.cs b/BankingBot/Client.cs index 5809a4b..5dbafd3 100644 --- a/BankingBot/Client.cs +++ b/BankingBot/Client.cs @@ -9,6 +9,7 @@ using BankingBot.Models; using OpenQA.Selenium; using BankingBot.ActionManagers.AccountManagers; using BankingBot.Responses; +using BankingBot.Enums; namespace BankingBot { @@ -16,13 +17,14 @@ namespace BankingBot where T : IWebDriver { #region Dependencies - private readonly ILoginManager _loginManager; - private readonly IAccountManager _accountManager; + readonly ILoginManager loginManager; + readonly IAccountManager accountManager; protected readonly IBrowserBot BrowserBot; #endregion public ILoginCredentials LoginCredentials { get; private set; } + public Provider Provider { get; private set; } public bool IsLoggedIn { @@ -33,8 +35,8 @@ namespace BankingBot { BrowserBot = new BrowserBot(); - _loginManager = new LoginManager(BrowserBot); - _accountManager = new AccountManager(BrowserBot); + loginManager = new LoginManager(BrowserBot); + accountManager = new AccountManager(BrowserBot); } #region Actions - Login Manager @@ -42,8 +44,15 @@ namespace BankingBot public Response Login(ILoginCredentials credentials) { LoginCredentials = credentials; + Provider = credentials.GetProvider(); - return _loginManager.Login(credentials); + var response = loginManager.Login(credentials); + if (response.Status == ResponseStatus.Success) + { + accountManager.Init(Provider); + } + + return response; } #endregion @@ -57,7 +66,7 @@ namespace BankingBot public IEnumerable GetAccounts() { - return _accountManager.GetAccounts(); + return accountManager.GetAccounts(); } #endregion diff --git a/BankingBot/Contracts/IAccountManager.cs b/BankingBot/Contracts/IAccountManager.cs index 6fcc63d..3b8edb9 100644 --- a/BankingBot/Contracts/IAccountManager.cs +++ b/BankingBot/Contracts/IAccountManager.cs @@ -1,10 +1,12 @@ -using BankingBot.Models; +using BankingBot.Enums; +using BankingBot.Models; using System.Collections.Generic; namespace BankingBot.Contracts { public interface IAccountManager { + void Init(Provider provider); IEnumerable GetAccounts(); } } diff --git a/BankingBot/Contracts/ILoginCredentials.cs b/BankingBot/Contracts/ILoginCredentials.cs index b504b5c..dea9b40 100644 --- a/BankingBot/Contracts/ILoginCredentials.cs +++ b/BankingBot/Contracts/ILoginCredentials.cs @@ -4,6 +4,6 @@ namespace BankingBot.Contracts { public interface ILoginCredentials { - Provider? GetProvider(); + Provider GetProvider(); } } \ No newline at end of file diff --git a/BankingBot/Contracts/IProviderAccountManager.cs b/BankingBot/Contracts/IProviderAccountManager.cs new file mode 100644 index 0000000..b2abd7e --- /dev/null +++ b/BankingBot/Contracts/IProviderAccountManager.cs @@ -0,0 +1,14 @@ +using BankingBot.Models; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace BankingBot.Contracts +{ + public interface IProviderAccountManager + { + IEnumerable GetAccounts(); + } +} diff --git a/BankingBot/LoginCredentials/LoginCredentials.cs b/BankingBot/LoginCredentials/LoginCredentials.cs index 15e4e39..ea501de 100644 --- a/BankingBot/LoginCredentials/LoginCredentials.cs +++ b/BankingBot/LoginCredentials/LoginCredentials.cs @@ -11,7 +11,7 @@ namespace BankingBot.LoginCredentials { public abstract class LoginCredentials : ILoginCredentials { - public Provider? GetProvider() + public Provider GetProvider() { return ProviderIdentifier.GetProviderFromType(GetType()); } diff --git a/BankingBot/Models/Account.cs b/BankingBot/Models/Account.cs index ecb7186..7133565 100644 --- a/BankingBot/Models/Account.cs +++ b/BankingBot/Models/Account.cs @@ -13,5 +13,7 @@ namespace BankingBot.Models public string AccountNumber { get; set; } public string Name { get; set; } + + public decimal Balance { get; set; } } } diff --git a/BankingBot/Urls/LloydsUrls.cs b/BankingBot/Urls/LloydsUrls.cs new file mode 100644 index 0000000..f8227ea --- /dev/null +++ b/BankingBot/Urls/LloydsUrls.cs @@ -0,0 +1,9 @@ +namespace BankingBot.Urls +{ + public static class LloydsUrls + { + public const string Login = "https://online.lloydsbank.co.uk/personal/logon/login.jsp"; + public const string MemorableInfo = "https://secure.lloydsbank.co.uk/personal/a/logon/entermemorableinformation.jsp"; + public const string AccountOverview = "https://secure.lloydsbank.co.uk/personal/a/account_overview_personal/"; + } +}