israeli-bank-scrapers 1.7.0 → 1.10.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +13 -1
- package/lib/definitions.d.ts +6 -1
- package/lib/definitions.js +6 -1
- package/lib/helpers/elements-interactions.js +2 -2
- package/lib/scrapers/base-beinleumi-group.js +2 -2
- package/lib/scrapers/base-scraper-with-browser.js +6 -1
- package/lib/scrapers/factory.d.ts +2 -1
- package/lib/scrapers/factory.js +6 -1
- package/lib/scrapers/hapoalim.js +33 -12
- package/lib/scrapers/max.js +22 -1
- package/lib/scrapers/otsar-hahayal.js +9 -2
- package/lib/scrapers/visa-cal.js +66 -55
- package/lib/scrapers/yahav.d.ts +21 -0
- package/lib/scrapers/yahav.js +272 -0
- package/lib/scrapers/yahav.test.d.ts +1 -0
- package/lib/scrapers/yahav.test.js +65 -0
- package/lib/transactions.d.ts +1 -0
- package/lib/transactions.js +1 -1
- package/package.json +1 -1
package/lib/scrapers/hapoalim.js
CHANGED
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
|
|
3
|
+
require("core-js/modules/es.array.iterator");
|
|
4
|
+
|
|
3
5
|
require("core-js/modules/es.promise");
|
|
4
6
|
|
|
5
7
|
Object.defineProperty(exports, "__esModule", {
|
|
@@ -21,8 +23,11 @@ var _fetch = require("../helpers/fetch");
|
|
|
21
23
|
|
|
22
24
|
var _transactions = require("../transactions");
|
|
23
25
|
|
|
26
|
+
var _debug = require("../helpers/debug");
|
|
27
|
+
|
|
24
28
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
25
29
|
|
|
30
|
+
const debug = (0, _debug.getDebug)('hapoalim');
|
|
26
31
|
const DATE_FORMAT = 'YYYYMMDD'; // eslint-disable-next-line @typescript-eslint/no-namespace
|
|
27
32
|
|
|
28
33
|
function convertTransactions(txns) {
|
|
@@ -101,11 +106,27 @@ async function fetchPoalimXSRFWithinPage(page, url, pageUuid) {
|
|
|
101
106
|
return (0, _fetch.fetchPostWithinPage)(page, url, [], headers);
|
|
102
107
|
}
|
|
103
108
|
|
|
109
|
+
async function getAccountTransactions(apiSiteUrl, page, accountNumber, startDate, endDate) {
|
|
110
|
+
var _txnsResult$transacti;
|
|
111
|
+
|
|
112
|
+
const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=150&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;
|
|
113
|
+
const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');
|
|
114
|
+
return convertTransactions((_txnsResult$transacti = txnsResult === null || txnsResult === void 0 ? void 0 : txnsResult.transactions) !== null && _txnsResult$transacti !== void 0 ? _txnsResult$transacti : []);
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
async function getAccountBalance(apiSiteUrl, page, accountNumber) {
|
|
118
|
+
const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;
|
|
119
|
+
const balanceAndCreditLimit = await (0, _fetch.fetchGetWithinPage)(page, balanceAndCreditLimitUrl);
|
|
120
|
+
return balanceAndCreditLimit === null || balanceAndCreditLimit === void 0 ? void 0 : balanceAndCreditLimit.currentBalance;
|
|
121
|
+
}
|
|
122
|
+
|
|
104
123
|
async function fetchAccountData(page, baseUrl, options) {
|
|
105
124
|
const restContext = await getRestContext(page);
|
|
106
125
|
const apiSiteUrl = `${baseUrl}/${restContext}`;
|
|
107
126
|
const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;
|
|
127
|
+
debug('fetching accounts data');
|
|
108
128
|
const accountsInfo = (await (0, _fetch.fetchGetWithinPage)(page, accountDataUrl)) || [];
|
|
129
|
+
debug('got %d accounts, fetching txns and balance', accountsInfo.length);
|
|
109
130
|
const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
|
|
110
131
|
const startDate = options.startDate || defaultStartMoment.toDate();
|
|
111
132
|
|
|
@@ -115,19 +136,18 @@ async function fetchAccountData(page, baseUrl, options) {
|
|
|
115
136
|
const endDateStr = (0, _moment.default)().format(DATE_FORMAT);
|
|
116
137
|
const accounts = [];
|
|
117
138
|
|
|
118
|
-
for (
|
|
119
|
-
|
|
120
|
-
const
|
|
121
|
-
const
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
if (txnsResult) {
|
|
128
|
-
txns = convertTransactions(txnsResult.transactions);
|
|
139
|
+
for (const account of accountsInfo) {
|
|
140
|
+
let balance;
|
|
141
|
+
const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;
|
|
142
|
+
const isActiveAccount = account.accountClosingReasonCode === 0;
|
|
143
|
+
|
|
144
|
+
if (isActiveAccount) {
|
|
145
|
+
balance = await getAccountBalance(apiSiteUrl, page, accountNumber);
|
|
146
|
+
} else {
|
|
147
|
+
debug('Skipping balance for a closed account, balance will be undefined');
|
|
129
148
|
}
|
|
130
149
|
|
|
150
|
+
const txns = await getAccountTransactions(apiSiteUrl, page, accountNumber, startDateStr, endDateStr);
|
|
131
151
|
accounts.push({
|
|
132
152
|
accountNumber,
|
|
133
153
|
balance,
|
|
@@ -139,6 +159,7 @@ async function fetchAccountData(page, baseUrl, options) {
|
|
|
139
159
|
success: true,
|
|
140
160
|
accounts
|
|
141
161
|
};
|
|
162
|
+
debug('fetching ended');
|
|
142
163
|
return accountData;
|
|
143
164
|
}
|
|
144
165
|
|
|
@@ -184,4 +205,4 @@ class HapoalimScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
184
205
|
|
|
185
206
|
var _default = HapoalimScraper;
|
|
186
207
|
exports.default = _default;
|
|
187
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/hapoalim.ts"],"names":["DATE_FORMAT","convertTransactions","txns","map","txn","isOutbound","eventActivityTypeCode","memo","beneficiaryDetailsData","partyHeadline","partyName","messageHeadline","messageDetail","memoLines","push","length","join","result","type","TransactionTypes","Normal","identifier","referenceNumber","date","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","getRestContext","page","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","fetchAccountData","baseUrl","options","apiSiteUrl","accountDataUrl","accountsInfo","defaultStartMoment","subtract","add","startDate","toDate","startMoment","moment","max","startDateStr","format","endDateStr","accounts","accountIndex","accountNumber","bankNumber","branchNumber","balanceAndCreditLimitUrl","balanceAndCreditLimit","balance","currentBalance","txnsUrl","txnsResult","transactions","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport uuid4 from 'uuid/v4';\n\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport {\n  TransactionsAccount, Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScaperOptions, ScraperCredentials } from './base-scraper';\n\nconst DATE_FORMAT = 'YYYYMMDD';\n\n// eslint-disable-next-line @typescript-eslint/no-namespace\ndeclare namespace window {\n  const bnhpApp: any;\n}\n\ninterface ScrapedTransaction {\n  serialNumber?: number;\n  activityDescription?: string;\n  eventAmount: number;\n  valueDate?: string;\n  eventDate?: string;\n  referenceNumber?: number;\n  ScrapedTransaction?: string;\n  eventActivityTypeCode: number;\n  currentBalance: number;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ntype FetchedAccountData = {\n  bankNumber: string;\n  accountNumber: string;\n  branchNumber: string;\n}[];\n\ntype FetchedAccountTransactionsData = {\n  transactions: ScrapedTransaction[];\n};\n\ntype BalanceAndCreditLimit = {\n  creditLimitAmount: number;\n  creditLimitDescription: string;\n  creditLimitUtilizationAmount: number;\n  creditLimitUtilizationExistanceCode: number;\n  creditLimitUtilizationPercent: number;\n  currentAccountLimitsAmount: number;\n  currentBalance: number;\n  withdrawalBalance: number;\n};\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn) => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n\n    let memo = '';\n    if (txn.beneficiaryDetailsData) {\n      const {\n        partyHeadline,\n        partyName,\n        messageHeadline,\n        messageDetail,\n      } = txn.beneficiaryDetailsData;\n      const memoLines: string[] = [];\n      if (partyHeadline) {\n        memoLines.push(partyHeadline);\n      }\n\n      if (partyName) {\n        memoLines.push(`${partyName}.`);\n      }\n\n      if (messageHeadline) {\n        memoLines.push(messageHeadline);\n      }\n\n      if (messageDetail) {\n        memoLines.push(`${messageDetail}.`);\n      }\n\n      if (memoLines.length) {\n        memo = memoLines.join(' ');\n      }\n    }\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.referenceNumber,\n      date: moment(txn.eventDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.valueDate, DATE_FORMAT).toISOString(),\n      originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      description: txn.activityDescription || '',\n      status: txn.serialNumber === 0 ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      memo,\n    };\n\n    return result;\n  });\n}\n\nasync function getRestContext(page: Page) {\n  await waitUntil(async () => {\n    return page.evaluate(() => !!window.bnhpApp);\n  }, 'waiting for app data load');\n\n  const result = await page.evaluate(() => {\n    return window.bnhpApp.restContext;\n  });\n\n  return result.slice(1);\n}\n\nasync function fetchPoalimXSRFWithinPage(page: Page, url: string, pageUuid: string): Promise<FetchedAccountTransactionsData | null> {\n  const cookies = await page.cookies();\n  const XSRFCookie = cookies.find((cookie) => cookie.name === 'XSRF-TOKEN');\n  const headers: Record<string, any> = {};\n  if (XSRFCookie != null) {\n    headers['X-XSRF-TOKEN'] = XSRFCookie.value;\n  }\n  headers.pageUuid = pageUuid;\n  headers.uuid = uuid4();\n  headers['Content-Type'] = 'application/json;charset=UTF-8';\n  return fetchPostWithinPage<FetchedAccountTransactionsData>(page, url, [], headers);\n}\n\nasync function fetchAccountData(page: Page, baseUrl: string, options: ScaperOptions) {\n  const restContext = await getRestContext(page);\n  const apiSiteUrl = `${baseUrl}/${restContext}`;\n  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;\n  const accountsInfo = await fetchGetWithinPage<FetchedAccountData>(page, accountDataUrl) || [];\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const endDateStr = moment().format(DATE_FORMAT);\n\n  const accounts: TransactionsAccount[] = [];\n  for (let accountIndex = 0; accountIndex < accountsInfo.length; accountIndex += 1) {\n    const accountNumber = `${accountsInfo[accountIndex].bankNumber}-${accountsInfo[accountIndex].branchNumber}-${accountsInfo[accountIndex].accountNumber}`;\n\n    const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;\n    const balanceAndCreditLimit = await fetchGetWithinPage(page, balanceAndCreditLimitUrl) as BalanceAndCreditLimit;\n    const balance: number = balanceAndCreditLimit?.currentBalance;\n\n    const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=150&retrievalEndDate=${endDateStr}&retrievalStartDate=${startDateStr}&sortCode=1`;\n    const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n    let txns: Transaction[] = [];\n\n    if (txnsResult) {\n      txns = convertTransactions(txnsResult.transactions);\n    }\n\n    accounts.push({\n      accountNumber,\n      balance,\n      txns,\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts,\n  };\n\n  return accountData;\n}\n\nfunction getPossibleLoginResults(baseUrl: string) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${baseUrl}/portalserver/HomePage`,\n    `${baseUrl}/ng-portals-bt/rb/he/homepage`,\n    `${baseUrl}/ng-portals/rb/he/homepage`];\n  urls[LoginResults.InvalidPassword] = [`${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`];\n  urls[LoginResults.ChangePassword] = [\n    `${baseUrl}/MCP/START?flow=MCP&state=START&expiredDate=null`,\n    /\\/ABOUTTOEXPIRE\\/START/i,\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass HapoalimScraper extends BaseScraperWithBrowser {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${this.baseUrl}/cgi-bin/poalwwwc?reqName=getLogonPage`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.login-btn',\n      postAction: async () => waitForRedirect(this.page),\n      possibleResults: getPossibleLoginResults(this.baseUrl),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.baseUrl, this.options);\n  }\n}\n\nexport default HapoalimScraper;\n"]}
|
|
208
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/hapoalim.ts"],"names":["debug","DATE_FORMAT","convertTransactions","txns","map","txn","isOutbound","eventActivityTypeCode","memo","beneficiaryDetailsData","partyHeadline","partyName","messageHeadline","messageDetail","memoLines","push","length","join","result","type","TransactionTypes","Normal","identifier","referenceNumber","date","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","getRestContext","page","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","getAccountTransactions","apiSiteUrl","accountNumber","startDate","endDate","txnsUrl","txnsResult","transactions","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","fetchAccountData","baseUrl","options","accountDataUrl","accountsInfo","defaultStartMoment","subtract","add","toDate","startMoment","moment","max","startDateStr","format","endDateStr","accounts","account","balance","bankNumber","branchNumber","isActiveAccount","accountClosingReasonCode","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport uuid4 from 'uuid/v4';\n\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport {\n  TransactionsAccount, Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScaperOptions, ScraperCredentials } from './base-scraper';\nimport { getDebug } from '../helpers/debug';\n\nconst debug = getDebug('hapoalim');\n\nconst DATE_FORMAT = 'YYYYMMDD';\n\n// eslint-disable-next-line @typescript-eslint/no-namespace\ndeclare namespace window {\n  const bnhpApp: any;\n}\n\ninterface ScrapedTransaction {\n  serialNumber?: number;\n  activityDescription?: string;\n  eventAmount: number;\n  valueDate?: string;\n  eventDate?: string;\n  referenceNumber?: number;\n  ScrapedTransaction?: string;\n  eventActivityTypeCode: number;\n  currentBalance: number;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ntype FetchedAccountData = {\n  bankNumber: string;\n  accountNumber: string;\n  branchNumber: string;\n  accountClosingReasonCode: number;\n}[];\n\ntype FetchedAccountTransactionsData = {\n  transactions: ScrapedTransaction[];\n};\n\ntype BalanceAndCreditLimit = {\n  creditLimitAmount: number;\n  creditLimitDescription: string;\n  creditLimitUtilizationAmount: number;\n  creditLimitUtilizationExistanceCode: number;\n  creditLimitUtilizationPercent: number;\n  currentAccountLimitsAmount: number;\n  currentBalance: number;\n  withdrawalBalance: number;\n};\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn) => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n\n    let memo = '';\n    if (txn.beneficiaryDetailsData) {\n      const {\n        partyHeadline,\n        partyName,\n        messageHeadline,\n        messageDetail,\n      } = txn.beneficiaryDetailsData;\n      const memoLines: string[] = [];\n      if (partyHeadline) {\n        memoLines.push(partyHeadline);\n      }\n\n      if (partyName) {\n        memoLines.push(`${partyName}.`);\n      }\n\n      if (messageHeadline) {\n        memoLines.push(messageHeadline);\n      }\n\n      if (messageDetail) {\n        memoLines.push(`${messageDetail}.`);\n      }\n\n      if (memoLines.length) {\n        memo = memoLines.join(' ');\n      }\n    }\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.referenceNumber,\n      date: moment(txn.eventDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.valueDate, DATE_FORMAT).toISOString(),\n      originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      description: txn.activityDescription || '',\n      status: txn.serialNumber === 0 ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      memo,\n    };\n\n    return result;\n  });\n}\n\nasync function getRestContext(page: Page) {\n  await waitUntil(async () => {\n    return page.evaluate(() => !!window.bnhpApp);\n  }, 'waiting for app data load');\n\n  const result = await page.evaluate(() => {\n    return window.bnhpApp.restContext;\n  });\n\n  return result.slice(1);\n}\n\nasync function fetchPoalimXSRFWithinPage(page: Page, url: string, pageUuid: string): Promise<FetchedAccountTransactionsData | null> {\n  const cookies = await page.cookies();\n  const XSRFCookie = cookies.find((cookie) => cookie.name === 'XSRF-TOKEN');\n  const headers: Record<string, any> = {};\n  if (XSRFCookie != null) {\n    headers['X-XSRF-TOKEN'] = XSRFCookie.value;\n  }\n  headers.pageUuid = pageUuid;\n  headers.uuid = uuid4();\n  headers['Content-Type'] = 'application/json;charset=UTF-8';\n  return fetchPostWithinPage<FetchedAccountTransactionsData>(page, url, [], headers);\n}\n\nasync function getAccountTransactions(apiSiteUrl: string, page: Page, accountNumber: string, startDate: string, endDate: string) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=150&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  return convertTransactions(txnsResult?.transactions ?? []);\n}\n\nasync function getAccountBalance(apiSiteUrl: string, page: Page, accountNumber: string) {\n  const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;\n  const balanceAndCreditLimit = await fetchGetWithinPage<BalanceAndCreditLimit>(page, balanceAndCreditLimitUrl);\n\n  return balanceAndCreditLimit?.currentBalance;\n}\n\nasync function fetchAccountData(page: Page, baseUrl: string, options: ScaperOptions) {\n  const restContext = await getRestContext(page);\n  const apiSiteUrl = `${baseUrl}/${restContext}`;\n  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;\n\n  debug('fetching accounts data');\n  const accountsInfo = await fetchGetWithinPage<FetchedAccountData>(page, accountDataUrl) || [];\n  debug('got %d accounts, fetching txns and balance', accountsInfo.length);\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const endDateStr = moment().format(DATE_FORMAT);\n\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of accountsInfo) {\n    let balance: number | undefined;\n    const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;\n\n    const isActiveAccount = account.accountClosingReasonCode === 0;\n    if (isActiveAccount) {\n      balance = await getAccountBalance(apiSiteUrl, page, accountNumber);\n    } else {\n      debug('Skipping balance for a closed account, balance will be undefined');\n    }\n\n    const txns = await getAccountTransactions(apiSiteUrl, page, accountNumber, startDateStr, endDateStr);\n\n    accounts.push({\n      accountNumber,\n      balance,\n      txns,\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts,\n  };\n  debug('fetching ended');\n  return accountData;\n}\n\nfunction getPossibleLoginResults(baseUrl: string) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${baseUrl}/portalserver/HomePage`,\n    `${baseUrl}/ng-portals-bt/rb/he/homepage`,\n    `${baseUrl}/ng-portals/rb/he/homepage`];\n  urls[LoginResults.InvalidPassword] = [`${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`];\n  urls[LoginResults.ChangePassword] = [\n    `${baseUrl}/MCP/START?flow=MCP&state=START&expiredDate=null`,\n    /\\/ABOUTTOEXPIRE\\/START/i,\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass HapoalimScraper extends BaseScraperWithBrowser {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${this.baseUrl}/cgi-bin/poalwwwc?reqName=getLogonPage`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.login-btn',\n      postAction: async () => waitForRedirect(this.page),\n      possibleResults: getPossibleLoginResults(this.baseUrl),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.baseUrl, this.options);\n  }\n}\n\nexport default HapoalimScraper;\n"]}
|
package/lib/scrapers/max.js
CHANGED
|
@@ -31,6 +31,8 @@ var _transactions = require("../helpers/transactions");
|
|
|
31
31
|
|
|
32
32
|
var _transactions2 = require("../transactions");
|
|
33
33
|
|
|
34
|
+
var _debug = require("../helpers/debug");
|
|
35
|
+
|
|
34
36
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
35
37
|
|
|
36
38
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
|
|
@@ -39,6 +41,7 @@ function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { va
|
|
|
39
41
|
|
|
40
42
|
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
41
43
|
|
|
44
|
+
const debug = (0, _debug.getDebug)('max');
|
|
42
45
|
const BASE_ACTIONS_URL = 'https://online.max.co.il';
|
|
43
46
|
const BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';
|
|
44
47
|
const BASE_WELCOME_URL = 'https://www.max.co.il';
|
|
@@ -63,6 +66,7 @@ const ACCUMULATING_BASKET = 'סל מצטבר';
|
|
|
63
66
|
const POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';
|
|
64
67
|
const INVALID_DETAILS_SELECTOR = '#popupWrongDetails';
|
|
65
68
|
const LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';
|
|
69
|
+
const categories = new Map();
|
|
66
70
|
|
|
67
71
|
function redirectOrDialog(page) {
|
|
68
72
|
return Promise.race([(0, _navigation.waitForRedirect)(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]), (0, _elementsInteractions.waitUntilElementFound)(page, INVALID_DETAILS_SELECTOR, true), (0, _elementsInteractions.waitUntilElementFound)(page, LOGIN_ERROR_SELECTOR, true)]);
|
|
@@ -84,6 +88,21 @@ function getTransactionsUrl(monthMoment) {
|
|
|
84
88
|
});
|
|
85
89
|
}
|
|
86
90
|
|
|
91
|
+
async function loadCategories(page) {
|
|
92
|
+
debug('Loading categories');
|
|
93
|
+
const res = await (0, _fetch.fetchGetWithinPage)(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);
|
|
94
|
+
|
|
95
|
+
if (res && Array.isArray(res.result)) {
|
|
96
|
+
var _res$result;
|
|
97
|
+
|
|
98
|
+
debug(`${res.result.length} categories loaded`);
|
|
99
|
+
(_res$result = res.result) === null || _res$result === void 0 ? void 0 : _res$result.forEach(({
|
|
100
|
+
id,
|
|
101
|
+
name
|
|
102
|
+
}) => categories.set(id, name));
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
87
106
|
function getTransactionType(txnTypeStr) {
|
|
88
107
|
const cleanedUpTxnTypeStr = txnTypeStr.replace('\t', ' ').trim();
|
|
89
108
|
|
|
@@ -143,6 +162,7 @@ function mapTransaction(rawTransaction) {
|
|
|
143
162
|
chargedAmount: -rawTransaction.actualPaymentAmount,
|
|
144
163
|
description: rawTransaction.merchantName.trim(),
|
|
145
164
|
memo: rawTransaction.comments,
|
|
165
|
+
category: categories.get(rawTransaction === null || rawTransaction === void 0 ? void 0 : rawTransaction.categoryId),
|
|
146
166
|
installments: getInstallmentsInfo(rawTransaction.comments) || undefined,
|
|
147
167
|
status
|
|
148
168
|
};
|
|
@@ -200,6 +220,7 @@ async function fetchTransactions(page, options) {
|
|
|
200
220
|
const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
|
|
201
221
|
|
|
202
222
|
const allMonths = (0, _dates.default)(startMoment, futureMonthsToScrape);
|
|
223
|
+
await loadCategories(page);
|
|
203
224
|
let allResults = {};
|
|
204
225
|
|
|
205
226
|
for (let i = 0; i < allMonths.length; i += 1) {
|
|
@@ -280,4 +301,4 @@ class MaxScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
280
301
|
|
|
281
302
|
var _default = MaxScraper;
|
|
282
303
|
exports.default = _default;
|
|
283
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["BASE_ACTIONS_URL","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","NORMAL_TYPE_NAME","ATM_TYPE_NAME","INTERNET_SHOPPING_TYPE_NAME","INSTALLMENTS_TYPE_NAME","MONTHLY_CHARGE_TYPE_NAME","ONE_MONTH_POSTPONED_TYPE_NAME","MONTHLY_POSTPONED_TYPE_NAME","MONTHLY_PAYMENT_TYPE_NAME","FUTURE_PURCHASE_FINANCING","MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME","THIRTY_DAYS_PLUS_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME","MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME","CREDIT_TYPE_NAME","ACCUMULATING_BASKET","POSTPONED_TRANSACTION_INSTALLMENTS","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","matches","match","length","number","parseInt","total","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","description","merchantName","memo","installments","undefined","fetchTransactionsForMonth","url","data","transactionsByAccount","result","transactions","filter","transaction","forEach","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","clonedTxns","Array","from","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startDate","toDate","moment","max","allMonths","i","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","checkReadiness","postAction","possibleResults","fetchData","results","accounts","map","success"],"mappings":"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","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntilElementFound, elementPresentOnPage, clickButton } from '../helpers/elements-interactions';\nimport getAllMonthMoments from '../helpers/dates';\nimport { fixInstallments, sortTransactionsByDate, filterOldTransactions } from '../helpers/transactions';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScaperOptions, ScraperCredentials } from './base-scraper';\n\n\ninterface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  comments: string;\n  merchantName: string;\n}\n\nconst BASE_ACTIONS_URL = 'https://online.max.co.il';\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_ACTIONS_URL}/Anonymous/Login/PasswordExpired.aspx`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nconst NORMAL_TYPE_NAME = 'רגילה';\nconst ATM_TYPE_NAME = 'חיוב עסקות מיידי';\nconst INTERNET_SHOPPING_TYPE_NAME = 'אינטרנט/חו\"ל';\nconst INSTALLMENTS_TYPE_NAME = 'תשלומים';\nconst MONTHLY_CHARGE_TYPE_NAME = 'חיוב חודשי';\nconst ONE_MONTH_POSTPONED_TYPE_NAME = 'דחוי חודש';\nconst MONTHLY_POSTPONED_TYPE_NAME = 'דחוי לחיוב החודשי';\nconst MONTHLY_PAYMENT_TYPE_NAME = 'תשלום חודשי';\nconst FUTURE_PURCHASE_FINANCING = 'מימון לרכישה עתידית';\nconst MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME = 'דחוי חודש תשלומים';\nconst THIRTY_DAYS_PLUS_TYPE_NAME = 'עסקת 30 פלוס';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME = 'דחוי חודשיים';\nconst MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME = 'חודשי + ריבית';\nconst CREDIT_TYPE_NAME = 'קרדיט';\nconst ACCUMULATING_BASKET = 'סל מצטבר';\nconst POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n     * url explanation:\n     * userIndex: -1 for all account owners\n     * cardIndex: -1 for all cards under the account\n     * all other query params are static, beside the date which changes for request per month\n     */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1&v=V3.69-HF-CarLoanLeviModel.2.57`,\n  });\n}\n\nfunction getTransactionType(txnTypeStr: string) {\n  const cleanedUpTxnTypeStr = txnTypeStr.replace('\\t', ' ').trim();\n  switch (cleanedUpTxnTypeStr) {\n    case ATM_TYPE_NAME:\n    case NORMAL_TYPE_NAME:\n    case MONTHLY_CHARGE_TYPE_NAME:\n    case ONE_MONTH_POSTPONED_TYPE_NAME:\n    case MONTHLY_POSTPONED_TYPE_NAME:\n    case FUTURE_PURCHASE_FINANCING:\n    case MONTHLY_PAYMENT_TYPE_NAME:\n    case MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME:\n    case THIRTY_DAYS_PLUS_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME:\n    case ACCUMULATING_BASKET:\n    case INTERNET_SHOPPING_TYPE_NAME:\n    case MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME:\n    case POSTPONED_TRANSACTION_INSTALLMENTS:\n      return TransactionTypes.Normal;\n    case INSTALLMENTS_TYPE_NAME:\n    case CREDIT_TYPE_NAME:\n      return TransactionTypes.Installments;\n    default:\n      throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return null;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return null;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ?\n    rawTransaction.purchaseDate :\n    rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  return {\n    type: getTransactionType(rawTransaction.planName),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    description: rawTransaction.merchantName.trim(),\n    memo: rawTransaction.comments,\n    installments: getInstallmentsInfo(rawTransaction.comments) || undefined,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult{\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter((transaction) => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach((accountNumber) => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(txns: Transaction[], startMoment: moment.Moment, combineInstallments: boolean) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = filterOldTransactions(clonedTxns, startMoment, combineInstallments || false);\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScaperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach((accountNumber) => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(txns, startMoment, options.combineInstallments || false);\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [async () => {\n    return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n  }];\n  urls[LoginResults.UnknownError] = [async () => {\n    return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n  }];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass MaxScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#login-password #send-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map((accountNumber) => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"]}
|
|
304
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["debug","BASE_ACTIONS_URL","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","NORMAL_TYPE_NAME","ATM_TYPE_NAME","INTERNET_SHOPPING_TYPE_NAME","INSTALLMENTS_TYPE_NAME","MONTHLY_CHARGE_TYPE_NAME","ONE_MONTH_POSTPONED_TYPE_NAME","MONTHLY_POSTPONED_TYPE_NAME","MONTHLY_PAYMENT_TYPE_NAME","FUTURE_PURCHASE_FINANCING","MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME","THIRTY_DAYS_PLUS_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME","MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME","CREDIT_TYPE_NAME","ACCUMULATING_BASKET","POSTPONED_TRANSACTION_INSTALLMENTS","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","loadCategories","res","Array","isArray","result","length","forEach","id","name","set","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","matches","match","number","parseInt","total","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","description","merchantName","memo","category","get","categoryId","installments","undefined","fetchTransactionsForMonth","url","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","clonedTxns","from","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startDate","toDate","moment","max","allMonths","i","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","checkReadiness","postAction","possibleResults","fetchData","results","accounts","map","success"],"mappings":"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","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntilElementFound, elementPresentOnPage, clickButton } from '../helpers/elements-interactions';\nimport getAllMonthMoments from '../helpers/dates';\nimport { fixInstallments, sortTransactionsByDate, filterOldTransactions } from '../helpers/transactions';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScaperOptions, ScraperCredentials } from './base-scraper';\nimport { getDebug } from '../helpers/debug';\n\nconst debug = getDebug('max');\n\ninterface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n}\n\nconst BASE_ACTIONS_URL = 'https://online.max.co.il';\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_ACTIONS_URL}/Anonymous/Login/PasswordExpired.aspx`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nconst NORMAL_TYPE_NAME = 'רגילה';\nconst ATM_TYPE_NAME = 'חיוב עסקות מיידי';\nconst INTERNET_SHOPPING_TYPE_NAME = 'אינטרנט/חו\"ל';\nconst INSTALLMENTS_TYPE_NAME = 'תשלומים';\nconst MONTHLY_CHARGE_TYPE_NAME = 'חיוב חודשי';\nconst ONE_MONTH_POSTPONED_TYPE_NAME = 'דחוי חודש';\nconst MONTHLY_POSTPONED_TYPE_NAME = 'דחוי לחיוב החודשי';\nconst MONTHLY_PAYMENT_TYPE_NAME = 'תשלום חודשי';\nconst FUTURE_PURCHASE_FINANCING = 'מימון לרכישה עתידית';\nconst MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME = 'דחוי חודש תשלומים';\nconst THIRTY_DAYS_PLUS_TYPE_NAME = 'עסקת 30 פלוס';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME = 'דחוי חודשיים';\nconst MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME = 'חודשי + ריבית';\nconst CREDIT_TYPE_NAME = 'קרדיט';\nconst ACCUMULATING_BASKET = 'סל מצטבר';\nconst POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n     * url explanation:\n     * userIndex: -1 for all account owners\n     * cardIndex: -1 for all cards under the account\n     * all other query params are static, beside the date which changes for request per month\n     */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1&v=V3.69-HF-CarLoanLeviModel.2.57`,\n  });\n}\n\ninterface FetchCategoryResult {\n  result? : Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n      res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(txnTypeStr: string) {\n  const cleanedUpTxnTypeStr = txnTypeStr.replace('\\t', ' ').trim();\n  switch (cleanedUpTxnTypeStr) {\n    case ATM_TYPE_NAME:\n    case NORMAL_TYPE_NAME:\n    case MONTHLY_CHARGE_TYPE_NAME:\n    case ONE_MONTH_POSTPONED_TYPE_NAME:\n    case MONTHLY_POSTPONED_TYPE_NAME:\n    case FUTURE_PURCHASE_FINANCING:\n    case MONTHLY_PAYMENT_TYPE_NAME:\n    case MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME:\n    case THIRTY_DAYS_PLUS_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME:\n    case ACCUMULATING_BASKET:\n    case INTERNET_SHOPPING_TYPE_NAME:\n    case MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME:\n    case POSTPONED_TRANSACTION_INSTALLMENTS:\n      return TransactionTypes.Normal;\n    case INSTALLMENTS_TYPE_NAME:\n    case CREDIT_TYPE_NAME:\n      return TransactionTypes.Installments;\n    default:\n      throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return null;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return null;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ?\n    rawTransaction.purchaseDate :\n    rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  return {\n    type: getTransactionType(rawTransaction.planName),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    description: rawTransaction.merchantName.trim(),\n    memo: rawTransaction.comments,\n    category: categories.get(rawTransaction?.categoryId),\n    installments: getInstallmentsInfo(rawTransaction.comments) || undefined,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult{\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter((transaction) => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach((accountNumber) => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(txns: Transaction[], startMoment: moment.Moment, combineInstallments: boolean) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = filterOldTransactions(clonedTxns, startMoment, combineInstallments || false);\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScaperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach((accountNumber) => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(txns, startMoment, options.combineInstallments || false);\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [async () => {\n    return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n  }];\n  urls[LoginResults.UnknownError] = [async () => {\n    return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n  }];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass MaxScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#login-password #send-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map((accountNumber) => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"]}
|
|
@@ -28,6 +28,7 @@ var _transactions = require("../transactions");
|
|
|
28
28
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
29
29
|
|
|
30
30
|
const BASE_URL = 'https://online.bankotsar.co.il';
|
|
31
|
+
const LONG_DATE_FORMAT = 'DD/MM/YYYY';
|
|
31
32
|
const DATE_FORMAT = 'DD/MM/YY';
|
|
32
33
|
|
|
33
34
|
function getPossibleLoginResults(page) {
|
|
@@ -79,7 +80,13 @@ function getAmountData(amountStr, hasCurrency = false) {
|
|
|
79
80
|
|
|
80
81
|
function convertTransactions(txns) {
|
|
81
82
|
return txns.map(txn => {
|
|
82
|
-
const
|
|
83
|
+
const dateFormat = txn.date.length === 8 ? DATE_FORMAT : txn.date.length === 10 ? LONG_DATE_FORMAT : null;
|
|
84
|
+
|
|
85
|
+
if (!dateFormat) {
|
|
86
|
+
throw new Error('invalid date format');
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
const txnDate = (0, _moment.default)(txn.date, dateFormat).toISOString();
|
|
83
90
|
const credit = getAmountData(txn.credit || '').amount;
|
|
84
91
|
const debit = getAmountData(txn.debit || '').amount;
|
|
85
92
|
const amount = (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);
|
|
@@ -217,4 +224,4 @@ class OtsarHahayalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser
|
|
|
217
224
|
|
|
218
225
|
var _default = OtsarHahayalScraper;
|
|
219
226
|
exports.default = _default;
|
|
220
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/otsar-hahayal.ts"],"names":["BASE_URL","DATE_FORMAT","getPossibleLoginResults","page","urls","LoginResults","Success","InvalidPassword","getTransactionsUrl","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","hasCurrency","amountStrCln","replace","currency","amount","parseFloat","SHEKEL_CURRENCY","includes","SHEKEL_CURRENCY_SYMBOL","parts","split","convertTransactions","txns","map","txn","txnDate","date","toISOString","credit","debit","Number","isNaN","result","type","TransactionTypes","Normal","status","TransactionStatuses","Completed","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","chargedAmount","description","memo","parseTransactionPage","tdsValues","trs","el","querySelector","innerText","balance","getAccountSummary","balanceElm","$","balanceInnerTextElm","getProperty","balanceText","jsonValue","balanceValue","creditLimit","creditUtilization","balanceCurrency","fetchTransactionsForAccount","startDate","summary","branchNum","$eval","span","accountNmbr","accountNumber","format","hasNextPage","noTransactionElm","pageTxns","concat","button","slice","fetchTransactions","waitForPostLogin","Promise","race","OtsarHahayalScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","options","toDate","startMoment","moment","max","url","navigateTo","accounts","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForNavigation } from '../helpers/navigation';\nimport {\n  fillInput,\n  clickButton,\n  waitUntilElementFound,\n  pageEvalAll,\n  elementPresentOnPage,\n} from '../helpers/elements-interactions';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScraperCredentials } from './base-scraper';\n\nconst BASE_URL = 'https://online.bankotsar.co.il';\nconst DATE_FORMAT = 'DD/MM/YY';\n\ninterface ScrapedTransaction {\n  balance?: string;\n  debit?: string;\n  credit?: string;\n  memo?: string;\n  status?: string;\n  reference?: string;\n  description?: string;\n  date: string;\n}\n\nfunction getPossibleLoginResults(page: Page) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [`${BASE_URL}/wps/myportal/FibiMenu/Online`];\n  urls[LoginResults.InvalidPassword] = [() => elementPresentOnPage(page, '#validationMsg')];\n  // TODO: support change password\n  /* urls[LOGIN_RESULT.CHANGE_PASSWORD] = [``]; */\n  return urls;\n}\n\nfunction getTransactionsUrl() {\n  return `${BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string, hasCurrency = false) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (!hasCurrency) {\n    amount = parseFloat(amountStrCln);\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    amount = parseFloat(parts[0]);\n    [, currency] = parts;\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn) => {\n    const txnDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const credit = getAmountData(txn.credit || '').amount;\n    const debit = getAmountData(txn.debit || '').amount;\n    const amount = (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn.description || '',\n      memo: '',\n    };\n\n    return result;\n  });\n}\n\nasync function parseTransactionPage(page: Page): Promise<ScrapedTransaction[]> {\n  const tdsValues = await pageEvalAll(page, '#dataTable077 tbody tr', [], (trs) => {\n    return (trs).map((el) => ({\n      date: (el.querySelector('.date') as HTMLElement).innerText,\n      // reference and description have vice-versa class name\n      description: (el.querySelector('.reference') as HTMLElement).innerText,\n      reference: (el.querySelector('.details') as HTMLElement).innerText,\n      credit: (el.querySelector('.credit') as HTMLElement).innerText,\n      debit: (el.querySelector('.debit') as HTMLElement).innerText,\n      balance: (el.querySelector('.balance') as HTMLElement).innerText,\n    }));\n  });\n\n  return tdsValues;\n}\n\nasync function getAccountSummary(page: Page) {\n  const balanceElm = await page.$('.current_balance');\n  const balanceInnerTextElm = await balanceElm!.getProperty('innerText');\n  const balanceText = await balanceInnerTextElm.jsonValue();\n  const balanceValue = getAmountData(balanceText as string, true);\n  // TODO: Find the credit field in bank website (could see it in my account)\n  return {\n    balance: Number.isNaN(balanceValue.amount) ? 0 : balanceValue.amount,\n    creditLimit: 0.0,\n    creditUtilization: 0.0,\n    balanceCurrency: balanceValue.currency,\n  };\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment) {\n  const summary = await getAccountSummary(page);\n  await waitUntilElementFound(page, 'input#fromDate');\n  // Get account number\n  const branchNum = await page.$eval('.branch_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n\n  const accountNmbr = await page.$eval('.acc_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n  const accountNumber = `14-${branchNum}-${accountNmbr}`;\n  // Search for relavant transaction from startDate\n  await clickButton(page, '#tabHeader4');\n  await fillInput(\n    page,\n    'input#fromDate',\n    startDate.format('DD/MM/YYYY'),\n  );\n\n  await clickButton(page, '#fibi_tab_dates .fibi_btn:nth-child(2)');\n  await waitForNavigation(page);\n  await waitUntilElementFound(page, 'table#dataTable077, #NO_DATA077');\n  let hasNextPage = true;\n  let txns: ScrapedTransaction[] = [];\n\n  const noTransactionElm = await page.$('#NO_DATA077');\n  if (noTransactionElm == null) {\n    // Scape transactions (this maybe spanned on multiple pages)\n    while (hasNextPage) {\n      const pageTxns = await parseTransactionPage(page);\n      txns = txns.concat(pageTxns);\n      const button = await page.$('#Npage');\n      hasNextPage = false;\n      if (button != null) {\n        hasNextPage = true;\n      }\n      if (hasNextPage) {\n        await clickButton(page, '#Npage');\n        await waitForNavigation(page);\n        await waitUntilElementFound(page, 'table#dataTable077');\n      }\n    }\n  }\n\n  return {\n    accountNumber,\n    summary,\n    txns: convertTransactions(txns.slice(1)), // Remove first line which is \"opening balance\"\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment) {\n  // TODO need to extend to support multiple accounts and foreign accounts\n  return [await fetchTransactionsForAccount(page, startDate)];\n}\n\nasync function waitForPostLogin(page: Page) {\n  // TODO check for condition to provide new password\n  return Promise.race([\n    waitUntilElementFound(page, 'div.lotusFrame', true),\n    waitUntilElementFound(page, '#validationMsg'),\n  ]);\n}\n\nclass OtsarHahayalScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=OTSARPRTAL&site=Private&KODSAFA=HE`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    const url = getTransactionsUrl();\n    await this.navigateTo(url);\n\n    const accounts = await fetchTransactions(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default OtsarHahayalScraper;\n"]}
|
|
227
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/otsar-hahayal.ts"],"names":["BASE_URL","LONG_DATE_FORMAT","DATE_FORMAT","getPossibleLoginResults","page","urls","LoginResults","Success","InvalidPassword","getTransactionsUrl","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","hasCurrency","amountStrCln","replace","currency","amount","parseFloat","SHEKEL_CURRENCY","includes","SHEKEL_CURRENCY_SYMBOL","parts","split","convertTransactions","txns","map","txn","dateFormat","date","length","Error","txnDate","toISOString","credit","debit","Number","isNaN","result","type","TransactionTypes","Normal","status","TransactionStatuses","Completed","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","chargedAmount","description","memo","parseTransactionPage","tdsValues","trs","el","querySelector","innerText","balance","getAccountSummary","balanceElm","$","balanceInnerTextElm","getProperty","balanceText","jsonValue","balanceValue","creditLimit","creditUtilization","balanceCurrency","fetchTransactionsForAccount","startDate","summary","branchNum","$eval","span","accountNmbr","accountNumber","format","hasNextPage","noTransactionElm","pageTxns","concat","button","slice","fetchTransactions","waitForPostLogin","Promise","race","OtsarHahayalScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","options","toDate","startMoment","moment","max","url","navigateTo","accounts","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForNavigation } from '../helpers/navigation';\nimport {\n  fillInput,\n  clickButton,\n  waitUntilElementFound,\n  pageEvalAll,\n  elementPresentOnPage,\n} from '../helpers/elements-interactions';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScraperCredentials } from './base-scraper';\n\nconst BASE_URL = 'https://online.bankotsar.co.il';\nconst LONG_DATE_FORMAT = 'DD/MM/YYYY';\nconst DATE_FORMAT = 'DD/MM/YY';\n\ninterface ScrapedTransaction {\n  balance?: string;\n  debit?: string;\n  credit?: string;\n  memo?: string;\n  status?: string;\n  reference?: string;\n  description?: string;\n  date: string;\n}\n\nfunction getPossibleLoginResults(page: Page) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [`${BASE_URL}/wps/myportal/FibiMenu/Online`];\n  urls[LoginResults.InvalidPassword] = [() => elementPresentOnPage(page, '#validationMsg')];\n  // TODO: support change password\n  /* urls[LOGIN_RESULT.CHANGE_PASSWORD] = [``]; */\n  return urls;\n}\n\nfunction getTransactionsUrl() {\n  return `${BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string, hasCurrency = false) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (!hasCurrency) {\n    amount = parseFloat(amountStrCln);\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    amount = parseFloat(parts[0]);\n    [, currency] = parts;\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn) => {\n    const dateFormat =\n        txn.date.length === 8 ?\n          DATE_FORMAT :\n          txn.date.length === 10 ?\n            LONG_DATE_FORMAT :\n            null;\n    if (!dateFormat) {\n      throw new Error('invalid date format');\n    }\n    const txnDate = moment(txn.date, dateFormat).toISOString();\n    const credit = getAmountData(txn.credit || '').amount;\n    const debit = getAmountData(txn.debit || '').amount;\n    const amount = (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn.description || '',\n      memo: '',\n    };\n\n    return result;\n  });\n}\n\nasync function parseTransactionPage(page: Page): Promise<ScrapedTransaction[]> {\n  const tdsValues = await pageEvalAll(page, '#dataTable077 tbody tr', [], (trs) => {\n    return (trs).map((el) => ({\n      date: (el.querySelector('.date') as HTMLElement).innerText,\n      // reference and description have vice-versa class name\n      description: (el.querySelector('.reference') as HTMLElement).innerText,\n      reference: (el.querySelector('.details') as HTMLElement).innerText,\n      credit: (el.querySelector('.credit') as HTMLElement).innerText,\n      debit: (el.querySelector('.debit') as HTMLElement).innerText,\n      balance: (el.querySelector('.balance') as HTMLElement).innerText,\n    }));\n  });\n\n  return tdsValues;\n}\n\nasync function getAccountSummary(page: Page) {\n  const balanceElm = await page.$('.current_balance');\n  const balanceInnerTextElm = await balanceElm!.getProperty('innerText');\n  const balanceText = await balanceInnerTextElm.jsonValue();\n  const balanceValue = getAmountData(balanceText as string, true);\n  // TODO: Find the credit field in bank website (could see it in my account)\n  return {\n    balance: Number.isNaN(balanceValue.amount) ? 0 : balanceValue.amount,\n    creditLimit: 0.0,\n    creditUtilization: 0.0,\n    balanceCurrency: balanceValue.currency,\n  };\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment) {\n  const summary = await getAccountSummary(page);\n  await waitUntilElementFound(page, 'input#fromDate');\n  // Get account number\n  const branchNum = await page.$eval('.branch_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n\n  const accountNmbr = await page.$eval('.acc_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n  const accountNumber = `14-${branchNum}-${accountNmbr}`;\n  // Search for relavant transaction from startDate\n  await clickButton(page, '#tabHeader4');\n  await fillInput(\n    page,\n    'input#fromDate',\n    startDate.format('DD/MM/YYYY'),\n  );\n\n  await clickButton(page, '#fibi_tab_dates .fibi_btn:nth-child(2)');\n  await waitForNavigation(page);\n  await waitUntilElementFound(page, 'table#dataTable077, #NO_DATA077');\n  let hasNextPage = true;\n  let txns: ScrapedTransaction[] = [];\n\n  const noTransactionElm = await page.$('#NO_DATA077');\n  if (noTransactionElm == null) {\n    // Scape transactions (this maybe spanned on multiple pages)\n    while (hasNextPage) {\n      const pageTxns = await parseTransactionPage(page);\n      txns = txns.concat(pageTxns);\n      const button = await page.$('#Npage');\n      hasNextPage = false;\n      if (button != null) {\n        hasNextPage = true;\n      }\n      if (hasNextPage) {\n        await clickButton(page, '#Npage');\n        await waitForNavigation(page);\n        await waitUntilElementFound(page, 'table#dataTable077');\n      }\n    }\n  }\n\n  return {\n    accountNumber,\n    summary,\n    txns: convertTransactions(txns.slice(1)), // Remove first line which is \"opening balance\"\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment) {\n  // TODO need to extend to support multiple accounts and foreign accounts\n  return [await fetchTransactionsForAccount(page, startDate)];\n}\n\nasync function waitForPostLogin(page: Page) {\n  // TODO check for condition to provide new password\n  return Promise.race([\n    waitUntilElementFound(page, 'div.lotusFrame', true),\n    waitUntilElementFound(page, '#validationMsg'),\n  ]);\n}\n\nclass OtsarHahayalScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=OTSARPRTAL&site=Private&KODSAFA=HE`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    const url = getTransactionsUrl();\n    await this.navigateTo(url);\n\n    const accounts = await fetchTransactions(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default OtsarHahayalScraper;\n"]}
|