Merged lloyds/get-transactions into main
This commit is contained in:
commit
b35c4765de
11 changed files with 200 additions and 0 deletions
|
@ -27,9 +27,18 @@ namespace BankingBot.ActionManagers.AccountManagers
|
||||||
providerAccountManager = (IProviderAccountManager)Activator.CreateInstance(providerAccountManagerType, BrowserBot, scriptManager);
|
providerAccountManager = (IProviderAccountManager)Activator.CreateInstance(providerAccountManagerType, BrowserBot, scriptManager);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#region Behaviours
|
||||||
|
|
||||||
public IEnumerable<Account> GetAccounts()
|
public IEnumerable<Account> GetAccounts()
|
||||||
{
|
{
|
||||||
return providerAccountManager.GetAccounts();
|
return providerAccountManager.GetAccounts();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public IEnumerable<Transaction> GetTransactions(string accountNumber)
|
||||||
|
{
|
||||||
|
return providerAccountManager.GetTransactions(accountNumber);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
using BankingBot.Attributes;
|
using BankingBot.Attributes;
|
||||||
using BankingBot.Contracts;
|
using BankingBot.Contracts;
|
||||||
using BankingBot.Enums;
|
using BankingBot.Enums;
|
||||||
|
using BankingBot.Extensions;
|
||||||
using BankingBot.Models;
|
using BankingBot.Models;
|
||||||
using BankingBot.Urls;
|
using BankingBot.Urls;
|
||||||
using OpenQA.Selenium;
|
using OpenQA.Selenium;
|
||||||
|
@ -19,6 +20,16 @@ namespace BankingBot.ActionManagers.AccountManagers
|
||||||
readonly IBrowserBot browserBot;
|
readonly IBrowserBot browserBot;
|
||||||
readonly IScriptManager scriptManager;
|
readonly IScriptManager scriptManager;
|
||||||
|
|
||||||
|
private enum TransactionTableColumn
|
||||||
|
{
|
||||||
|
Date = 0,
|
||||||
|
Description = 1,
|
||||||
|
Type = 2,
|
||||||
|
In = 3,
|
||||||
|
Out = 4,
|
||||||
|
Balance = 5
|
||||||
|
}
|
||||||
|
|
||||||
public LloydsAccountManager(
|
public LloydsAccountManager(
|
||||||
IBrowserBot browserBot,
|
IBrowserBot browserBot,
|
||||||
IScriptManager scriptManager)
|
IScriptManager scriptManager)
|
||||||
|
@ -61,5 +72,70 @@ namespace BankingBot.ActionManagers.AccountManagers
|
||||||
|
|
||||||
return accounts;
|
return accounts;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public IEnumerable<Transaction> GetTransactions(string accountNumber)
|
||||||
|
{
|
||||||
|
var transactions = new List<Transaction>();
|
||||||
|
|
||||||
|
var ajaxIdentifier = GetAjaxIdentifierForAccountNumber(accountNumber);
|
||||||
|
var accountDetailsUrl = $"{LloydsUrls.AccountDetails}/{ajaxIdentifier}";
|
||||||
|
browserBot.WebDriver.Url = accountDetailsUrl;
|
||||||
|
|
||||||
|
var transactionsTableBody = browserBot.WebDriver
|
||||||
|
.FindElement(By.ClassName("statements-wrap"))
|
||||||
|
.FindElement(By.ClassName("cwa-tbody"));
|
||||||
|
|
||||||
|
var transactionRows = transactionsTableBody.FindElements(By.CssSelector("tr[aria-expanded='false']"));
|
||||||
|
foreach (var row in transactionRows)
|
||||||
|
{
|
||||||
|
var rowCells = row.FindElements(By.TagName("td"));
|
||||||
|
|
||||||
|
var transaction = new Transaction
|
||||||
|
{
|
||||||
|
Date = DateTime.Parse(rowCells[(int)TransactionTableColumn.Date].Text),
|
||||||
|
IsPending = false,
|
||||||
|
Description = rowCells[(int)TransactionTableColumn.Description].Text
|
||||||
|
};
|
||||||
|
|
||||||
|
var amountIn = rowCells[(int)TransactionTableColumn.In].Text;
|
||||||
|
var amountOut = rowCells[(int)TransactionTableColumn.Out].Text;
|
||||||
|
if (amountIn != "")
|
||||||
|
{
|
||||||
|
transaction.Amount = decimal.Parse(amountIn);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
transaction.Amount = -(decimal.Parse(amountOut));
|
||||||
|
}
|
||||||
|
|
||||||
|
transactions.Add(transaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
return transactions;
|
||||||
|
}
|
||||||
|
|
||||||
|
private string GetAjaxIdentifierForAccountNumber(string accountNumber)
|
||||||
|
{
|
||||||
|
var currentUrl = browserBot.WebDriver.Url;
|
||||||
|
|
||||||
|
browserBot.WebDriver.Url = LloydsUrls.AccountOverview;
|
||||||
|
|
||||||
|
IWebElement accountContainer = null;
|
||||||
|
var accountContainers = browserBot.WebDriver.FindElements(By.ClassName("des-m-sat-xx-account-tile"));
|
||||||
|
foreach (var cnt in accountContainers)
|
||||||
|
{
|
||||||
|
if (cnt.HasElement(By.ClassName("account-number")))
|
||||||
|
{
|
||||||
|
accountContainer = cnt;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (accountContainer == null)
|
||||||
|
throw new InvalidOperationException($"Could not find account with account number '{accountNumber}'");
|
||||||
|
|
||||||
|
return accountContainer.GetAttribute("data-ajax-identifier");
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -67,7 +67,9 @@
|
||||||
<Compile Include="Contracts\IScriptManager.cs" />
|
<Compile Include="Contracts\IScriptManager.cs" />
|
||||||
<Compile Include="Enums\Provider.cs" />
|
<Compile Include="Enums\Provider.cs" />
|
||||||
<Compile Include="Enums\ResponseStatus.cs" />
|
<Compile Include="Enums\ResponseStatus.cs" />
|
||||||
|
<Compile Include="Enums\TransactionType.cs" />
|
||||||
<Compile Include="Exceptions\InvalidCredentialsException.cs" />
|
<Compile Include="Exceptions\InvalidCredentialsException.cs" />
|
||||||
|
<Compile Include="Extensions\IWebElementExtensions.cs" />
|
||||||
<Compile Include="Extensions\StringExtensions.cs" />
|
<Compile Include="Extensions\StringExtensions.cs" />
|
||||||
<Compile Include="Helpers\AccountHelpers.cs" />
|
<Compile Include="Helpers\AccountHelpers.cs" />
|
||||||
<Compile Include="LoginCredentials\BarclaysLoginCredentials.cs" />
|
<Compile Include="LoginCredentials\BarclaysLoginCredentials.cs" />
|
||||||
|
@ -76,6 +78,7 @@
|
||||||
<Compile Include="ActionManagers\LoginManagers\LloydsLoginManager.cs" />
|
<Compile Include="ActionManagers\LoginManagers\LloydsLoginManager.cs" />
|
||||||
<Compile Include="ActionManagers\LoginManagers\LoginManager.cs" />
|
<Compile Include="ActionManagers\LoginManagers\LoginManager.cs" />
|
||||||
<Compile Include="Models\Account.cs" />
|
<Compile Include="Models\Account.cs" />
|
||||||
|
<Compile Include="Models\Transaction.cs" />
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
<Compile Include="ScriptManagement\ScriptBundles.cs" />
|
<Compile Include="ScriptManagement\ScriptBundles.cs" />
|
||||||
<Compile Include="ScriptManagement\ScriptManager.cs" />
|
<Compile Include="ScriptManagement\ScriptManager.cs" />
|
||||||
|
|
|
@ -58,6 +58,11 @@ namespace BankingBot
|
||||||
return accountManager.GetAccounts();
|
return accountManager.GetAccounts();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public IEnumerable<Transaction> GetTransactions(string accountNumber)
|
||||||
|
{
|
||||||
|
return accountManager.GetTransactions(accountNumber);
|
||||||
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region IDisposable Support
|
#region IDisposable Support
|
||||||
|
@ -93,6 +98,8 @@ namespace BankingBot
|
||||||
// TODO: uncomment the following line if the finalizer is overridden above.
|
// TODO: uncomment the following line if the finalizer is overridden above.
|
||||||
// GC.SuppressFinalize(this);
|
// GC.SuppressFinalize(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,9 @@ namespace BankingBot.Contracts
|
||||||
public interface IAccountManager
|
public interface IAccountManager
|
||||||
{
|
{
|
||||||
void Init(Provider provider);
|
void Init(Provider provider);
|
||||||
|
|
||||||
IEnumerable<Account> GetAccounts();
|
IEnumerable<Account> GetAccounts();
|
||||||
|
|
||||||
|
IEnumerable<Transaction> GetTransactions(string accountNumber);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,5 +10,7 @@ namespace BankingBot.Contracts
|
||||||
decimal GetBalance();
|
decimal GetBalance();
|
||||||
|
|
||||||
IEnumerable<Account> GetAccounts();
|
IEnumerable<Account> GetAccounts();
|
||||||
|
|
||||||
|
IEnumerable<Transaction> GetTransactions(string accountNumber);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -10,5 +10,7 @@ namespace BankingBot.Contracts
|
||||||
public interface IProviderAccountManager
|
public interface IProviderAccountManager
|
||||||
{
|
{
|
||||||
IEnumerable<Account> GetAccounts();
|
IEnumerable<Account> GetAccounts();
|
||||||
|
|
||||||
|
IEnumerable<Transaction> GetTransactions(string accountNumber);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
54
BankingBot/Enums/TransactionType.cs
Normal file
54
BankingBot/Enums/TransactionType.cs
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace BankingBot.Enums
|
||||||
|
{
|
||||||
|
public enum TransactionType
|
||||||
|
{
|
||||||
|
BGC,
|
||||||
|
BNS,
|
||||||
|
BP,
|
||||||
|
CHG,
|
||||||
|
CHQ,
|
||||||
|
COM,
|
||||||
|
COR,
|
||||||
|
CPT,
|
||||||
|
CSH,
|
||||||
|
CSQ,
|
||||||
|
DD,
|
||||||
|
DEB,
|
||||||
|
DEP,
|
||||||
|
EFT,
|
||||||
|
EUR,
|
||||||
|
FE,
|
||||||
|
FEE,
|
||||||
|
FPC,
|
||||||
|
FPI,
|
||||||
|
FPO,
|
||||||
|
IB,
|
||||||
|
INT,
|
||||||
|
MPI,
|
||||||
|
MPO,
|
||||||
|
MTG,
|
||||||
|
NS,
|
||||||
|
NSC,
|
||||||
|
OTH,
|
||||||
|
PAY,
|
||||||
|
PSB,
|
||||||
|
PSV,
|
||||||
|
SAL,
|
||||||
|
SPB,
|
||||||
|
SO,
|
||||||
|
STK,
|
||||||
|
TD,
|
||||||
|
TDG,
|
||||||
|
TDI,
|
||||||
|
TDN,
|
||||||
|
TFR,
|
||||||
|
UT,
|
||||||
|
SUR
|
||||||
|
}
|
||||||
|
}
|
25
BankingBot/Extensions/IWebElementExtensions.cs
Normal file
25
BankingBot/Extensions/IWebElementExtensions.cs
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
using OpenQA.Selenium;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace BankingBot.Extensions
|
||||||
|
{
|
||||||
|
public static class IWebElementExtensions
|
||||||
|
{
|
||||||
|
public static bool HasElement(this IWebElement element, By by)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
element.FindElement(by);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch (NoSuchElementException)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
18
BankingBot/Models/Transaction.cs
Normal file
18
BankingBot/Models/Transaction.cs
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
using BankingBot.Enums;
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace BankingBot.Models
|
||||||
|
{
|
||||||
|
public class Transaction
|
||||||
|
{
|
||||||
|
public decimal Amount { get; set; }
|
||||||
|
|
||||||
|
public TransactionType Type { get; set; }
|
||||||
|
|
||||||
|
public string Description { get; set; }
|
||||||
|
|
||||||
|
public DateTime Date { get; set; }
|
||||||
|
|
||||||
|
public bool IsPending { get; set; }
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,5 +5,6 @@
|
||||||
public const string Login = "https://online.lloydsbank.co.uk/personal/logon/login.jsp";
|
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 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/";
|
public const string AccountOverview = "https://secure.lloydsbank.co.uk/personal/a/account_overview_personal/";
|
||||||
|
public const string AccountDetails = "https://secure.lloydsbank.co.uk/personal/a/account_details_ress";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue