israeli-bank-scrapers 6.4.1 → 6.5.0

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.
@@ -16,10 +16,10 @@ const BASE_URL = 'https://www.behatsdaa.org.il';
16
16
  const LOGIN_URL = `${BASE_URL}/login`;
17
17
  const PURCHASE_HISTORY_URL = 'https://back.behatsdaa.org.il/api/purchases/purchaseHistory';
18
18
  const debug = (0, _debug.getDebug)('behatsdaa');
19
- function variantToTransaction(variant) {
19
+ function variantToTransaction(variant, options) {
20
20
  // The price is positive, make it negative as it's an expense
21
21
  const originalAmount = -variant.customerPrice;
22
- return {
22
+ const result = {
23
23
  type: _transactions.TransactionTypes.Normal,
24
24
  identifier: variant.tTransactionID,
25
25
  date: (0, _moment.default)(variant.orderDate).format('YYYY-MM-DD'),
@@ -32,6 +32,10 @@ function variantToTransaction(variant) {
32
32
  status: _transactions.TransactionStatuses.Completed,
33
33
  memo: variant.variantName
34
34
  };
35
+ if (options?.includeRawTransaction) {
36
+ result.rawTransaction = variant;
37
+ }
38
+ return result;
35
39
  }
36
40
  class BehatsdaaScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
37
41
  getLoginOptions(credentials) {
@@ -104,10 +108,10 @@ class BehatsdaaScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
104
108
  success: true,
105
109
  accounts: [{
106
110
  accountNumber: res.data.memberId,
107
- txns: res.data.variants.map(variantToTransaction)
111
+ txns: res.data.variants.map(variant => variantToTransaction(variant, this.options))
108
112
  }]
109
113
  };
110
114
  }
111
115
  }
112
116
  var _default = exports.default = BehatsdaaScraper;
113
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_elementsInteractions","_fetch","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","BASE_URL","LOGIN_URL","PURCHASE_HISTORY_URL","debug","getDebug","variantToTransaction","variant","originalAmount","customerPrice","type","TransactionTypes","Normal","identifier","tTransactionID","date","moment","orderDate","format","processedDate","originalCurrency","chargedAmount","chargedCurrency","description","name","status","TransactionStatuses","Completed","memo","variantName","BehatsdaaScraper","BaseScraperWithBrowser","getLoginOptions","credentials","loginUrl","fields","selector","value","id","password","checkReadiness","Promise","all","waitUntilElementFound","page","possibleResults","LoginResults","Success","InvalidPassword","submitButtonSelector","sleep","button","$","click","fetchData","token","evaluate","window","localStorage","getItem","success","errorMessage","body","FromDate","options","startDate","ToDate","BenefitStatusId","res","fetchPostWithinPage","authorization","organizationid","errorDescription","data","accounts","accountNumber","memberId","txns","variants","map","_default","exports"],"sources":["../../src/scrapers/behatsdaa.ts"],"sourcesContent":["import moment from 'moment';\nimport { getDebug } from '../helpers/debug';\nimport { waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { sleep } from '../helpers/waiting';\nimport { type Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { BaseScraperWithBrowser, type LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport { type ScraperScrapingResult } from './interface';\n\nconst BASE_URL = 'https://www.behatsdaa.org.il';\nconst LOGIN_URL = `${BASE_URL}/login`;\nconst PURCHASE_HISTORY_URL = 'https://back.behatsdaa.org.il/api/purchases/purchaseHistory';\n\nconst debug = getDebug('behatsdaa');\n\ntype ScraperSpecificCredentials = { id: string; password: string };\n\ntype Variant = {\n  name: string;\n  variantName: string;\n  customerPrice: number;\n  orderDate: string; // ISO timestamp with no timezone\n  tTransactionID: string;\n};\n\ntype PurchaseHistoryResponse = {\n  data?: {\n    errorDescription?: string;\n    memberId: string;\n    variants: Variant[];\n  };\n  errorDescription?: string;\n};\n\nfunction variantToTransaction(variant: Variant): Transaction {\n  // The price is positive, make it negative as it's an expense\n  const originalAmount = -variant.customerPrice;\n  return {\n    type: TransactionTypes.Normal,\n    identifier: variant.tTransactionID,\n    date: moment(variant.orderDate).format('YYYY-MM-DD'),\n    processedDate: moment(variant.orderDate).format('YYYY-MM-DD'),\n    originalAmount,\n    originalCurrency: 'ILS',\n    chargedAmount: originalAmount,\n    chargedCurrency: 'ILS',\n    description: variant.name,\n    status: TransactionStatuses.Completed,\n    memo: variant.variantName,\n  };\n}\n\nclass BehatsdaaScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  public getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: [\n        { selector: '#loginId', value: credentials.id },\n        { selector: '#loginPassword', value: credentials.password },\n      ],\n      checkReadiness: async () => {\n        await Promise.all([\n          waitUntilElementFound(this.page, '#loginPassword'),\n          waitUntilElementFound(this.page, '#loginId'),\n        ]);\n      },\n      possibleResults: {\n        [LoginResults.Success]: [`${BASE_URL}/`],\n        [LoginResults.InvalidPassword]: ['.custom-input-error-label'],\n      },\n      submitButtonSelector: async () => {\n        await sleep(1000);\n        debug('Trying to find submit button');\n        const button = await this.page.$('xpath=//button[contains(., \"התחברות\")]');\n        if (button) {\n          debug('Submit button found');\n          await button.click();\n        } else {\n          debug('Submit button not found');\n        }\n      },\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const token = await this.page.evaluate(() => window.localStorage.getItem('userToken'));\n    if (!token) {\n      debug('Token not found in local storage');\n      return {\n        success: false,\n        errorMessage: 'TokenNotFound',\n      };\n    }\n\n    const body = {\n      FromDate: moment(this.options.startDate).format('YYYY-MM-DDTHH:mm:ss'),\n      ToDate: moment().format('YYYY-MM-DDTHH:mm:ss'),\n      BenefitStatusId: null,\n    };\n\n    debug('Fetching data');\n\n    const res = await fetchPostWithinPage<PurchaseHistoryResponse>(this.page, PURCHASE_HISTORY_URL, body, {\n      authorization: `Bearer ${token}`,\n      'Content-Type': 'application/json',\n      organizationid: '20',\n    });\n\n    debug('Data fetched');\n\n    if (res?.errorDescription || res?.data?.errorDescription) {\n      debug('Error fetching data', res.errorDescription || res.data?.errorDescription);\n      return { success: false, errorMessage: res.errorDescription };\n    }\n\n    if (!res?.data) {\n      debug('No data found');\n      return { success: false, errorMessage: 'NoData' };\n    }\n\n    debug('Data fetched successfully');\n    return {\n      success: true,\n      accounts: [\n        {\n          accountNumber: res.data.memberId,\n          txns: res.data.variants.map(variantToTransaction),\n        },\n      ],\n    };\n  }\n}\n\nexport default BehatsdaaScraper;\n"],"mappings":"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","ignoreList":[]}
117
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_elementsInteractions","_fetch","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","BASE_URL","LOGIN_URL","PURCHASE_HISTORY_URL","debug","getDebug","variantToTransaction","variant","options","originalAmount","customerPrice","result","type","TransactionTypes","Normal","identifier","tTransactionID","date","moment","orderDate","format","processedDate","originalCurrency","chargedAmount","chargedCurrency","description","name","status","TransactionStatuses","Completed","memo","variantName","includeRawTransaction","rawTransaction","BehatsdaaScraper","BaseScraperWithBrowser","getLoginOptions","credentials","loginUrl","fields","selector","value","id","password","checkReadiness","Promise","all","waitUntilElementFound","page","possibleResults","LoginResults","Success","InvalidPassword","submitButtonSelector","sleep","button","$","click","fetchData","token","evaluate","window","localStorage","getItem","success","errorMessage","body","FromDate","startDate","ToDate","BenefitStatusId","res","fetchPostWithinPage","authorization","organizationid","errorDescription","data","accounts","accountNumber","memberId","txns","variants","map","_default","exports"],"sources":["../../src/scrapers/behatsdaa.ts"],"sourcesContent":["import moment from 'moment';\nimport { getDebug } from '../helpers/debug';\nimport { waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { sleep } from '../helpers/waiting';\nimport { type Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { BaseScraperWithBrowser, type LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions, type ScraperScrapingResult } from './interface';\n\nconst BASE_URL = 'https://www.behatsdaa.org.il';\nconst LOGIN_URL = `${BASE_URL}/login`;\nconst PURCHASE_HISTORY_URL = 'https://back.behatsdaa.org.il/api/purchases/purchaseHistory';\n\nconst debug = getDebug('behatsdaa');\n\ntype ScraperSpecificCredentials = { id: string; password: string };\n\ntype Variant = {\n  name: string;\n  variantName: string;\n  customerPrice: number;\n  orderDate: string; // ISO timestamp with no timezone\n  tTransactionID: string;\n};\n\ntype PurchaseHistoryResponse = {\n  data?: {\n    errorDescription?: string;\n    memberId: string;\n    variants: Variant[];\n  };\n  errorDescription?: string;\n};\n\nfunction variantToTransaction(variant: Variant, options?: ScraperOptions): Transaction {\n  // The price is positive, make it negative as it's an expense\n  const originalAmount = -variant.customerPrice;\n  const result: Transaction = {\n    type: TransactionTypes.Normal,\n    identifier: variant.tTransactionID,\n    date: moment(variant.orderDate).format('YYYY-MM-DD'),\n    processedDate: moment(variant.orderDate).format('YYYY-MM-DD'),\n    originalAmount,\n    originalCurrency: 'ILS',\n    chargedAmount: originalAmount,\n    chargedCurrency: 'ILS',\n    description: variant.name,\n    status: TransactionStatuses.Completed,\n    memo: variant.variantName,\n  };\n\n  if (options?.includeRawTransaction) {\n    result.rawTransaction = variant;\n  }\n\n  return result;\n}\n\nclass BehatsdaaScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  public getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: [\n        { selector: '#loginId', value: credentials.id },\n        { selector: '#loginPassword', value: credentials.password },\n      ],\n      checkReadiness: async () => {\n        await Promise.all([\n          waitUntilElementFound(this.page, '#loginPassword'),\n          waitUntilElementFound(this.page, '#loginId'),\n        ]);\n      },\n      possibleResults: {\n        [LoginResults.Success]: [`${BASE_URL}/`],\n        [LoginResults.InvalidPassword]: ['.custom-input-error-label'],\n      },\n      submitButtonSelector: async () => {\n        await sleep(1000);\n        debug('Trying to find submit button');\n        const button = await this.page.$('xpath=//button[contains(., \"התחברות\")]');\n        if (button) {\n          debug('Submit button found');\n          await button.click();\n        } else {\n          debug('Submit button not found');\n        }\n      },\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const token = await this.page.evaluate(() => window.localStorage.getItem('userToken'));\n    if (!token) {\n      debug('Token not found in local storage');\n      return {\n        success: false,\n        errorMessage: 'TokenNotFound',\n      };\n    }\n\n    const body = {\n      FromDate: moment(this.options.startDate).format('YYYY-MM-DDTHH:mm:ss'),\n      ToDate: moment().format('YYYY-MM-DDTHH:mm:ss'),\n      BenefitStatusId: null,\n    };\n\n    debug('Fetching data');\n\n    const res = await fetchPostWithinPage<PurchaseHistoryResponse>(this.page, PURCHASE_HISTORY_URL, body, {\n      authorization: `Bearer ${token}`,\n      'Content-Type': 'application/json',\n      organizationid: '20',\n    });\n\n    debug('Data fetched');\n\n    if (res?.errorDescription || res?.data?.errorDescription) {\n      debug('Error fetching data', res.errorDescription || res.data?.errorDescription);\n      return { success: false, errorMessage: res.errorDescription };\n    }\n\n    if (!res?.data) {\n      debug('No data found');\n      return { success: false, errorMessage: 'NoData' };\n    }\n\n    debug('Data fetched successfully');\n    return {\n      success: true,\n      accounts: [\n        {\n          accountNumber: res.data.memberId,\n          txns: res.data.variants.map(variant => variantToTransaction(variant, this.options)),\n        },\n      ],\n    };\n  }\n}\n\nexport default BehatsdaaScraper;\n"],"mappings":"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","ignoreList":[]}
@@ -40,7 +40,7 @@ function getAmountData(amountStr) {
40
40
  currency
41
41
  };
42
42
  }
43
- function convertTransactions(txns) {
43
+ function convertTransactions(txns, options) {
44
44
  debug(`convert ${txns.length} raw transactions to official Transaction structure`);
45
45
  return txns.map(txn => {
46
46
  const chargedAmountTuple = getAmountData(txn.chargedAmount || '');
@@ -58,6 +58,9 @@ function convertTransactions(txns) {
58
58
  memo: '',
59
59
  identifier: txn.identifier
60
60
  };
61
+ if (options?.includeRawTransaction) {
62
+ result.rawTransaction = txn;
63
+ }
61
64
  return result;
62
65
  });
63
66
  }
@@ -90,7 +93,7 @@ async function fetchTransactions(page, options) {
90
93
  });
91
94
  });
92
95
  debug(`fetched ${rawTransactions.length} raw transactions from page`);
93
- const accountTransactions = convertTransactions(rawTransactions.filter(item => !!item));
96
+ const accountTransactions = convertTransactions(rawTransactions.filter(item => !!item), options);
94
97
  debug('filer out old transactions');
95
98
  const txns = options.outputData?.enableTransactionsFilterByDate ?? true ? (0, _transactions.filterOldTransactions)(accountTransactions, startMoment, false) : accountTransactions;
96
99
  debug(`found ${txns.length} valid transactions out of ${accountTransactions.length} transactions for account ending with ${accountNumber.substring(accountNumber.length - 2)}`);
@@ -146,4 +149,4 @@ class BeyahadBishvilhaScraper extends _baseScraperWithBrowser.BaseScraperWithBro
146
149
  }
147
150
  }
148
151
  var _default = exports.default = BeyahadBishvilhaScraper;
149
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_debug","_elementsInteractions","_transactions","_transactions2","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","DATE_FORMAT","LOGIN_URL","SUCCESS_URL","CARD_URL","getAmountData","amountStr","amountStrCln","replace","currency","amount","includes","SHEKEL_CURRENCY_SYMBOL","parseFloat","SHEKEL_CURRENCY","DOLLAR_CURRENCY_SYMBOL","DOLLAR_CURRENCY","EURO_CURRENCY_SYMBOL","EURO_CURRENCY","parts","split","convertTransactions","txns","length","map","txn","chargedAmountTuple","chargedAmount","txnProcessedDate","moment","date","result","type","TransactionTypes","Normal","status","TransactionStatuses","Completed","toISOString","processedDate","originalAmount","originalCurrency","chargedCurrency","description","memo","identifier","fetchTransactions","page","options","goto","waitUntilElementFound","defaultStartMoment","subtract","startDate","toDate","startMoment","max","accountNumber","pageEval","element","innerText","balance","rawTransactions","pageEvalAll","items","el","columns","querySelectorAll","accountTransactions","filter","item","outputData","enableTransactionsFilterByDate","filterOldTransactions","substring","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","id","password","BeyahadBishvilhaScraper","BaseScraperWithBrowser","getViewPort","width","height","getLoginOptions","loginUrl","fields","submitButtonSelector","button","$","click","possibleResults","fetchData","account","success","accounts","_default","exports"],"sources":["../../src/scrapers/beyahad-bishvilha.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport {\n  DOLLAR_CURRENCY,\n  DOLLAR_CURRENCY_SYMBOL,\n  EURO_CURRENCY,\n  EURO_CURRENCY_SYMBOL,\n  SHEKEL_CURRENCY,\n  SHEKEL_CURRENCY_SYMBOL,\n} from '../constants';\nimport { getDebug } from '../helpers/debug';\nimport { pageEval, pageEvalAll, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('beyahadBishvilha');\n\nconst DATE_FORMAT = 'DD/MM/YY';\nconst LOGIN_URL = 'https://www.hist.org.il/login';\nconst SUCCESS_URL = 'https://www.hist.org.il/';\nconst CARD_URL = 'https://www.hist.org.il/card/balanceAndUses';\n\ninterface ScrapedTransaction {\n  date: string;\n  description: string;\n  type: string;\n  chargedAmount: string;\n  identifier: string;\n}\n\nfunction getAmountData(amountStr: string) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(DOLLAR_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(DOLLAR_CURRENCY_SYMBOL, ''));\n    currency = DOLLAR_CURRENCY;\n  } else if (amountStrCln.includes(EURO_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(EURO_CURRENCY_SYMBOL, ''));\n    currency = EURO_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    [currency] = parts;\n    amount = parseFloat(parts[1]);\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  debug(`convert ${txns.length} raw transactions to official Transaction structure`);\n  return txns.map(txn => {\n    const chargedAmountTuple = getAmountData(txn.chargedAmount || '');\n    const txnProcessedDate = moment(txn.date, DATE_FORMAT);\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      date: txnProcessedDate.toISOString(),\n      processedDate: txnProcessedDate.toISOString(),\n      originalAmount: chargedAmountTuple.amount,\n      originalCurrency: chargedAmountTuple.currency,\n      chargedAmount: chargedAmountTuple.amount,\n      chargedCurrency: chargedAmountTuple.currency,\n      description: txn.description || '',\n      memo: '',\n      identifier: txn.identifier,\n    };\n\n    return result;\n  });\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  await page.goto(CARD_URL);\n  await waitUntilElementFound(page, '.react-loading.hide', false);\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n  const accountNumber = await pageEval(page, '.wallet-details div:nth-of-type(2)', null, element => {\n    return (element as any).innerText.replace('מספר כרטיס ', '');\n  });\n\n  const balance = await pageEval(page, '.wallet-details div:nth-of-type(4) > span:nth-of-type(2)', null, element => {\n    return (element as any).innerText;\n  });\n\n  debug('fetch raw transactions from page');\n\n  const rawTransactions: (ScrapedTransaction | null)[] = await pageEvalAll<(ScrapedTransaction | null)[]>(\n    page,\n    '.transaction-container, .transaction-component-container',\n    [],\n    items => {\n      return items.map(el => {\n        const columns: NodeListOf<HTMLSpanElement> = el.querySelectorAll('.transaction-item > span');\n        if (columns.length === 7) {\n          return {\n            date: columns[0].innerText,\n            identifier: columns[1].innerText,\n            description: columns[3].innerText,\n            type: columns[5].innerText,\n            chargedAmount: columns[6].innerText,\n          };\n        }\n        return null;\n      });\n    },\n  );\n  debug(`fetched ${rawTransactions.length} raw transactions from page`);\n\n  const accountTransactions = convertTransactions(rawTransactions.filter(item => !!item) as ScrapedTransaction[]);\n\n  debug('filer out old transactions');\n  const txns =\n    (options.outputData?.enableTransactionsFilterByDate ?? true)\n      ? filterOldTransactions(accountTransactions, startMoment, false)\n      : accountTransactions;\n  debug(\n    `found ${txns.length} valid transactions out of ${accountTransactions.length} transactions for account ending with ${accountNumber.substring(accountNumber.length - 2)}`,\n  );\n\n  return {\n    accountNumber,\n    balance: getAmountData(balance).amount,\n    txns,\n  };\n}\n\nfunction getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = []; // TODO\n  urls[LoginResults.InvalidPassword] = []; // TODO\n  urls[LoginResults.UnknownError] = []; // TODO\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#loginId', value: credentials.id },\n    { selector: '#loginPassword', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { id: string; password: string };\n\nclass BeyahadBishvilhaScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  protected getViewPort(): { width: number; height: number } {\n    return {\n      width: 1500,\n      height: 800,\n    };\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: async () => {\n        const button = await this.page.$('xpath//button[contains(., \"התחבר\")]');\n        if (button) {\n          await button.click();\n        }\n      },\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    const account = await fetchTransactions(this.page, this.options);\n    return {\n      success: true,\n      accounts: [account],\n    };\n  }\n}\n\nexport default BeyahadBishvilhaScraper;\n"],"mappings":"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","ignoreList":[]}
152
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_debug","_elementsInteractions","_transactions","_transactions2","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","DATE_FORMAT","LOGIN_URL","SUCCESS_URL","CARD_URL","getAmountData","amountStr","amountStrCln","replace","currency","amount","includes","SHEKEL_CURRENCY_SYMBOL","parseFloat","SHEKEL_CURRENCY","DOLLAR_CURRENCY_SYMBOL","DOLLAR_CURRENCY","EURO_CURRENCY_SYMBOL","EURO_CURRENCY","parts","split","convertTransactions","txns","options","length","map","txn","chargedAmountTuple","chargedAmount","txnProcessedDate","moment","date","result","type","TransactionTypes","Normal","status","TransactionStatuses","Completed","toISOString","processedDate","originalAmount","originalCurrency","chargedCurrency","description","memo","identifier","includeRawTransaction","rawTransaction","fetchTransactions","page","goto","waitUntilElementFound","defaultStartMoment","subtract","startDate","toDate","startMoment","max","accountNumber","pageEval","element","innerText","balance","rawTransactions","pageEvalAll","items","el","columns","querySelectorAll","accountTransactions","filter","item","outputData","enableTransactionsFilterByDate","filterOldTransactions","substring","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","id","password","BeyahadBishvilhaScraper","BaseScraperWithBrowser","getViewPort","width","height","getLoginOptions","loginUrl","fields","submitButtonSelector","button","$","click","possibleResults","fetchData","account","success","accounts","_default","exports"],"sources":["../../src/scrapers/beyahad-bishvilha.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport {\n  DOLLAR_CURRENCY,\n  DOLLAR_CURRENCY_SYMBOL,\n  EURO_CURRENCY,\n  EURO_CURRENCY_SYMBOL,\n  SHEKEL_CURRENCY,\n  SHEKEL_CURRENCY_SYMBOL,\n} from '../constants';\nimport { getDebug } from '../helpers/debug';\nimport { pageEval, pageEvalAll, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('beyahadBishvilha');\n\nconst DATE_FORMAT = 'DD/MM/YY';\nconst LOGIN_URL = 'https://www.hist.org.il/login';\nconst SUCCESS_URL = 'https://www.hist.org.il/';\nconst CARD_URL = 'https://www.hist.org.il/card/balanceAndUses';\n\ninterface ScrapedTransaction {\n  date: string;\n  description: string;\n  type: string;\n  chargedAmount: string;\n  identifier: string;\n}\n\nfunction getAmountData(amountStr: string) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(DOLLAR_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(DOLLAR_CURRENCY_SYMBOL, ''));\n    currency = DOLLAR_CURRENCY;\n  } else if (amountStrCln.includes(EURO_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(EURO_CURRENCY_SYMBOL, ''));\n    currency = EURO_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    [currency] = parts;\n    amount = parseFloat(parts[1]);\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[], options?: ScraperOptions): Transaction[] {\n  debug(`convert ${txns.length} raw transactions to official Transaction structure`);\n  return txns.map(txn => {\n    const chargedAmountTuple = getAmountData(txn.chargedAmount || '');\n    const txnProcessedDate = moment(txn.date, DATE_FORMAT);\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      date: txnProcessedDate.toISOString(),\n      processedDate: txnProcessedDate.toISOString(),\n      originalAmount: chargedAmountTuple.amount,\n      originalCurrency: chargedAmountTuple.currency,\n      chargedAmount: chargedAmountTuple.amount,\n      chargedCurrency: chargedAmountTuple.currency,\n      description: txn.description || '',\n      memo: '',\n      identifier: txn.identifier,\n    };\n\n    if (options?.includeRawTransaction) {\n      result.rawTransaction = txn;\n    }\n\n    return result;\n  });\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  await page.goto(CARD_URL);\n  await waitUntilElementFound(page, '.react-loading.hide', false);\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n  const accountNumber = await pageEval(page, '.wallet-details div:nth-of-type(2)', null, element => {\n    return (element as any).innerText.replace('מספר כרטיס ', '');\n  });\n\n  const balance = await pageEval(page, '.wallet-details div:nth-of-type(4) > span:nth-of-type(2)', null, element => {\n    return (element as any).innerText;\n  });\n\n  debug('fetch raw transactions from page');\n\n  const rawTransactions: (ScrapedTransaction | null)[] = await pageEvalAll<(ScrapedTransaction | null)[]>(\n    page,\n    '.transaction-container, .transaction-component-container',\n    [],\n    items => {\n      return items.map(el => {\n        const columns: NodeListOf<HTMLSpanElement> = el.querySelectorAll('.transaction-item > span');\n        if (columns.length === 7) {\n          return {\n            date: columns[0].innerText,\n            identifier: columns[1].innerText,\n            description: columns[3].innerText,\n            type: columns[5].innerText,\n            chargedAmount: columns[6].innerText,\n          };\n        }\n        return null;\n      });\n    },\n  );\n  debug(`fetched ${rawTransactions.length} raw transactions from page`);\n\n  const accountTransactions = convertTransactions(\n    rawTransactions.filter(item => !!item) as ScrapedTransaction[],\n    options,\n  );\n\n  debug('filer out old transactions');\n  const txns =\n    (options.outputData?.enableTransactionsFilterByDate ?? true)\n      ? filterOldTransactions(accountTransactions, startMoment, false)\n      : accountTransactions;\n  debug(\n    `found ${txns.length} valid transactions out of ${accountTransactions.length} transactions for account ending with ${accountNumber.substring(accountNumber.length - 2)}`,\n  );\n\n  return {\n    accountNumber,\n    balance: getAmountData(balance).amount,\n    txns,\n  };\n}\n\nfunction getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = []; // TODO\n  urls[LoginResults.InvalidPassword] = []; // TODO\n  urls[LoginResults.UnknownError] = []; // TODO\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#loginId', value: credentials.id },\n    { selector: '#loginPassword', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { id: string; password: string };\n\nclass BeyahadBishvilhaScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  protected getViewPort(): { width: number; height: number } {\n    return {\n      width: 1500,\n      height: 800,\n    };\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: async () => {\n        const button = await this.page.$('xpath//button[contains(., \"התחבר\")]');\n        if (button) {\n          await button.click();\n        }\n      },\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    const account = await fetchTransactions(this.page, this.options);\n    return {\n      success: true,\n      accounts: [account],\n    };\n  }\n}\n\nexport default BeyahadBishvilhaScraper;\n"],"mappings":"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","ignoreList":[]}
@@ -15,12 +15,12 @@ var _errors = require("./errors");
15
15
  function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
16
16
  const BASE_URL = 'https://start.telebank.co.il';
17
17
  const DATE_FORMAT = 'YYYYMMDD';
18
- function convertTransactions(txns, txnStatus) {
18
+ function convertTransactions(txns, txnStatus, options) {
19
19
  if (!txns) {
20
20
  return [];
21
21
  }
22
22
  return txns.map(txn => {
23
- return {
23
+ const result = {
24
24
  type: _transactions.TransactionTypes.Normal,
25
25
  identifier: txn.OperationNumber,
26
26
  date: (0, _moment.default)(txn.OperationDate, DATE_FORMAT).toISOString(),
@@ -31,6 +31,10 @@ function convertTransactions(txns, txnStatus) {
31
31
  description: txn.OperationDescriptionToDisplay,
32
32
  status: txnStatus
33
33
  };
34
+ if (options?.includeRawTransaction) {
35
+ result.rawTransaction = txn;
36
+ }
37
+ return result;
34
38
  });
35
39
  }
36
40
  async function fetchAccountData(page, options) {
@@ -60,9 +64,9 @@ async function fetchAccountData(page, options) {
60
64
  errorMessage: txnsResult && txnsResult.Error ? txnsResult.Error.MsgText : 'unknown error'
61
65
  };
62
66
  }
63
- const accountCompletedTxns = convertTransactions(txnsResult.CurrentAccountLastTransactions.OperationEntry, _transactions.TransactionStatuses.Completed);
67
+ const accountCompletedTxns = convertTransactions(txnsResult.CurrentAccountLastTransactions.OperationEntry, _transactions.TransactionStatuses.Completed, options);
64
68
  const rawFutureTxns = _lodash.default.get(txnsResult, 'CurrentAccountLastTransactions.FutureTransactionsBlock.FutureTransactionEntry');
65
- const accountPendingTxns = convertTransactions(rawFutureTxns, _transactions.TransactionStatuses.Pending);
69
+ const accountPendingTxns = convertTransactions(rawFutureTxns, _transactions.TransactionStatuses.Pending, options);
66
70
  accountsData.push({
67
71
  accountNumber,
68
72
  balance: txnsResult.CurrentAccountLastTransactions.CurrentAccountInfo.AccountBalance,
@@ -117,4 +121,4 @@ class DiscountScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
117
121
  }
118
122
  }
119
123
  var _default = exports.default = DiscountScraper;
120
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","_interopRequireDefault","require","_moment","_elementsInteractions","_fetch","_navigation","_transactions","_baseScraperWithBrowser","_errors","e","__esModule","default","BASE_URL","DATE_FORMAT","convertTransactions","txns","txnStatus","map","txn","type","TransactionTypes","Normal","identifier","OperationNumber","date","moment","OperationDate","toISOString","processedDate","ValueDate","originalAmount","OperationAmount","originalCurrency","chargedAmount","description","OperationDescriptionToDisplay","status","fetchAccountData","page","options","apiSiteUrl","accountDataUrl","accountInfo","fetchGetWithinPage","success","errorType","ScraperErrorTypes","Generic","errorMessage","defaultStartMoment","subtract","add","startDate","toDate","startMoment","max","startDateStr","format","accounts","UserAccountsData","UserAccounts","acc","NewAccountInfo","AccountID","accountsData","accountNumber","txnsUrl","txnsResult","Error","CurrentAccountLastTransactions","MsgText","accountCompletedTxns","OperationEntry","TransactionStatuses","Completed","rawFutureTxns","_","get","accountPendingTxns","Pending","push","balance","CurrentAccountInfo","AccountBalance","accountData","navigateOrErrorLabel","waitForNavigation","waitUntilElementFound","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","value","id","password","num","DiscountScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","checkReadiness","fields","submitButtonSelector","postAction","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/discount.ts"],"sourcesContent":["import _ from 'lodash';\nimport moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport { waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { type Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\nimport { type ScraperOptions, type ScraperScrapingResult } from './interface';\n\nconst BASE_URL = 'https://start.telebank.co.il';\nconst DATE_FORMAT = 'YYYYMMDD';\n\ninterface ScrapedTransaction {\n  OperationNumber: number;\n  OperationDate: string;\n  ValueDate: string;\n  OperationAmount: number;\n  OperationDescriptionToDisplay: string;\n}\n\ninterface CurrentAccountInfo {\n  AccountBalance: number;\n}\n\ninterface ScrapedAccountData {\n  UserAccountsData: {\n    DefaultAccountNumber: string;\n    UserAccounts: Array<{\n      NewAccountInfo: {\n        AccountID: string;\n      };\n    }>;\n  };\n}\n\ninterface ScrapedTransactionData {\n  Error?: { MsgText: string };\n  CurrentAccountLastTransactions?: {\n    OperationEntry: ScrapedTransaction[];\n    CurrentAccountInfo: CurrentAccountInfo;\n    FutureTransactionsBlock: {\n      FutureTransactionEntry: ScrapedTransaction[];\n    };\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[], txnStatus: TransactionStatuses): Transaction[] {\n  if (!txns) {\n    return [];\n  }\n  return txns.map(txn => {\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.OperationNumber,\n      date: moment(txn.OperationDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.ValueDate, DATE_FORMAT).toISOString(),\n      originalAmount: txn.OperationAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: txn.OperationAmount,\n      description: txn.OperationDescriptionToDisplay,\n      status: txnStatus,\n    };\n  });\n}\n\nasync function fetchAccountData(page: Page, options: ScraperOptions): Promise<ScraperScrapingResult> {\n  const apiSiteUrl = `${BASE_URL}/Titan/gatewayAPI`;\n\n  const accountDataUrl = `${apiSiteUrl}/userAccountsData`;\n  const accountInfo = await fetchGetWithinPage<ScrapedAccountData>(page, accountDataUrl);\n\n  if (!accountInfo) {\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.Generic,\n      errorMessage: 'failed to get account data',\n    };\n  }\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(2, '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\n  const accounts: string[] = accountInfo.UserAccountsData.UserAccounts.map(acc => acc.NewAccountInfo.AccountID);\n  const accountsData: Array<{ accountNumber: string; balance: number; txns: Transaction[] }> = [];\n\n  for (const accountNumber of accounts) {\n    const txnsUrl = `${apiSiteUrl}/lastTransactions/${accountNumber}/Date?IsCategoryDescCode=True&IsTransactionDetails=True&IsEventNames=True&IsFutureTransactionFlag=True&FromDate=${startDateStr}`;\n    const txnsResult = await fetchGetWithinPage<ScrapedTransactionData>(page, txnsUrl);\n    if (!txnsResult || txnsResult.Error || !txnsResult.CurrentAccountLastTransactions) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage: txnsResult && txnsResult.Error ? txnsResult.Error.MsgText : 'unknown error',\n      };\n    }\n\n    const accountCompletedTxns = convertTransactions(\n      txnsResult.CurrentAccountLastTransactions.OperationEntry,\n      TransactionStatuses.Completed,\n    );\n    const rawFutureTxns = _.get(\n      txnsResult,\n      'CurrentAccountLastTransactions.FutureTransactionsBlock.FutureTransactionEntry',\n    ) as ScrapedTransaction[];\n    const accountPendingTxns = convertTransactions(rawFutureTxns, TransactionStatuses.Pending);\n\n    accountsData.push({\n      accountNumber,\n      balance: txnsResult.CurrentAccountLastTransactions.CurrentAccountInfo.AccountBalance,\n      txns: [...accountCompletedTxns, ...accountPendingTxns],\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts: accountsData,\n  };\n\n  return accountData;\n}\n\nasync function navigateOrErrorLabel(page: Page) {\n  try {\n    await waitForNavigation(page);\n  } catch (e) {\n    await waitUntilElementFound(page, '#general-error', false, 100);\n  }\n}\n\nfunction getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${BASE_URL}/apollo/retail/#/MY_ACCOUNT_HOMEPAGE`,\n    `${BASE_URL}/apollo/retail2/#/MY_ACCOUNT_HOMEPAGE`,\n    `${BASE_URL}/apollo/retail2/`,\n  ];\n  urls[LoginResults.InvalidPassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/LOGIN_PAGE`];\n  urls[LoginResults.ChangePassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/PWD_RENEW`];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#tzId', value: credentials.id },\n    { selector: '#tzPassword', value: credentials.password },\n    { selector: '#aidnum', value: credentials.num },\n  ];\n}\n\ntype ScraperSpecificCredentials = { id: string; password: string; num: string };\n\nclass DiscountScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/login/#/LOGIN_PAGE`,\n      checkReadiness: async () => waitUntilElementFound(this.page, '#tzId'),\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.sendBtn',\n      postAction: async () => navigateOrErrorLabel(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.options);\n  }\n}\n\nexport default DiscountScraper;\n"],"mappings":"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","ignoreList":[]}
124
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","_interopRequireDefault","require","_moment","_elementsInteractions","_fetch","_navigation","_transactions","_baseScraperWithBrowser","_errors","e","__esModule","default","BASE_URL","DATE_FORMAT","convertTransactions","txns","txnStatus","options","map","txn","result","type","TransactionTypes","Normal","identifier","OperationNumber","date","moment","OperationDate","toISOString","processedDate","ValueDate","originalAmount","OperationAmount","originalCurrency","chargedAmount","description","OperationDescriptionToDisplay","status","includeRawTransaction","rawTransaction","fetchAccountData","page","apiSiteUrl","accountDataUrl","accountInfo","fetchGetWithinPage","success","errorType","ScraperErrorTypes","Generic","errorMessage","defaultStartMoment","subtract","add","startDate","toDate","startMoment","max","startDateStr","format","accounts","UserAccountsData","UserAccounts","acc","NewAccountInfo","AccountID","accountsData","accountNumber","txnsUrl","txnsResult","Error","CurrentAccountLastTransactions","MsgText","accountCompletedTxns","OperationEntry","TransactionStatuses","Completed","rawFutureTxns","_","get","accountPendingTxns","Pending","push","balance","CurrentAccountInfo","AccountBalance","accountData","navigateOrErrorLabel","waitForNavigation","waitUntilElementFound","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","value","id","password","num","DiscountScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","checkReadiness","fields","submitButtonSelector","postAction","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/discount.ts"],"sourcesContent":["import _ from 'lodash';\nimport moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport { waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { type Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\nimport { type ScraperOptions, type ScraperScrapingResult } from './interface';\n\nconst BASE_URL = 'https://start.telebank.co.il';\nconst DATE_FORMAT = 'YYYYMMDD';\n\ninterface ScrapedTransaction {\n  OperationNumber: number;\n  OperationDate: string;\n  ValueDate: string;\n  OperationAmount: number;\n  OperationDescriptionToDisplay: string;\n}\n\ninterface CurrentAccountInfo {\n  AccountBalance: number;\n}\n\ninterface ScrapedAccountData {\n  UserAccountsData: {\n    DefaultAccountNumber: string;\n    UserAccounts: Array<{\n      NewAccountInfo: {\n        AccountID: string;\n      };\n    }>;\n  };\n}\n\ninterface ScrapedTransactionData {\n  Error?: { MsgText: string };\n  CurrentAccountLastTransactions?: {\n    OperationEntry: ScrapedTransaction[];\n    CurrentAccountInfo: CurrentAccountInfo;\n    FutureTransactionsBlock: {\n      FutureTransactionEntry: ScrapedTransaction[];\n    };\n  };\n}\n\nfunction convertTransactions(\n  txns: ScrapedTransaction[],\n  txnStatus: TransactionStatuses,\n  options?: ScraperOptions,\n): Transaction[] {\n  if (!txns) {\n    return [];\n  }\n  return txns.map(txn => {\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.OperationNumber,\n      date: moment(txn.OperationDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.ValueDate, DATE_FORMAT).toISOString(),\n      originalAmount: txn.OperationAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: txn.OperationAmount,\n      description: txn.OperationDescriptionToDisplay,\n      status: txnStatus,\n    };\n\n    if (options?.includeRawTransaction) {\n      result.rawTransaction = txn;\n    }\n\n    return result;\n  });\n}\n\nasync function fetchAccountData(page: Page, options: ScraperOptions): Promise<ScraperScrapingResult> {\n  const apiSiteUrl = `${BASE_URL}/Titan/gatewayAPI`;\n\n  const accountDataUrl = `${apiSiteUrl}/userAccountsData`;\n  const accountInfo = await fetchGetWithinPage<ScrapedAccountData>(page, accountDataUrl);\n\n  if (!accountInfo) {\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.Generic,\n      errorMessage: 'failed to get account data',\n    };\n  }\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(2, '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\n  const accounts: string[] = accountInfo.UserAccountsData.UserAccounts.map(acc => acc.NewAccountInfo.AccountID);\n  const accountsData: Array<{ accountNumber: string; balance: number; txns: Transaction[] }> = [];\n\n  for (const accountNumber of accounts) {\n    const txnsUrl = `${apiSiteUrl}/lastTransactions/${accountNumber}/Date?IsCategoryDescCode=True&IsTransactionDetails=True&IsEventNames=True&IsFutureTransactionFlag=True&FromDate=${startDateStr}`;\n    const txnsResult = await fetchGetWithinPage<ScrapedTransactionData>(page, txnsUrl);\n    if (!txnsResult || txnsResult.Error || !txnsResult.CurrentAccountLastTransactions) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage: txnsResult && txnsResult.Error ? txnsResult.Error.MsgText : 'unknown error',\n      };\n    }\n\n    const accountCompletedTxns = convertTransactions(\n      txnsResult.CurrentAccountLastTransactions.OperationEntry,\n      TransactionStatuses.Completed,\n      options,\n    );\n    const rawFutureTxns = _.get(\n      txnsResult,\n      'CurrentAccountLastTransactions.FutureTransactionsBlock.FutureTransactionEntry',\n    ) as ScrapedTransaction[];\n    const accountPendingTxns = convertTransactions(rawFutureTxns, TransactionStatuses.Pending, options);\n\n    accountsData.push({\n      accountNumber,\n      balance: txnsResult.CurrentAccountLastTransactions.CurrentAccountInfo.AccountBalance,\n      txns: [...accountCompletedTxns, ...accountPendingTxns],\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts: accountsData,\n  };\n\n  return accountData;\n}\n\nasync function navigateOrErrorLabel(page: Page) {\n  try {\n    await waitForNavigation(page);\n  } catch (e) {\n    await waitUntilElementFound(page, '#general-error', false, 100);\n  }\n}\n\nfunction getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${BASE_URL}/apollo/retail/#/MY_ACCOUNT_HOMEPAGE`,\n    `${BASE_URL}/apollo/retail2/#/MY_ACCOUNT_HOMEPAGE`,\n    `${BASE_URL}/apollo/retail2/`,\n  ];\n  urls[LoginResults.InvalidPassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/LOGIN_PAGE`];\n  urls[LoginResults.ChangePassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/PWD_RENEW`];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#tzId', value: credentials.id },\n    { selector: '#tzPassword', value: credentials.password },\n    { selector: '#aidnum', value: credentials.num },\n  ];\n}\n\ntype ScraperSpecificCredentials = { id: string; password: string; num: string };\n\nclass DiscountScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/login/#/LOGIN_PAGE`,\n      checkReadiness: async () => waitUntilElementFound(this.page, '#tzId'),\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.sendBtn',\n      postAction: async () => navigateOrErrorLabel(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.options);\n  }\n}\n\nexport default DiscountScraper;\n"],"mappings":"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","ignoreList":[]}
@@ -18,7 +18,7 @@ const DATE_FORMAT = 'YYYYMMDD';
18
18
 
19
19
  // eslint-disable-next-line @typescript-eslint/no-namespace
20
20
 
21
- function convertTransactions(txns) {
21
+ function convertTransactions(txns, options) {
22
22
  return txns.map(txn => {
23
23
  const isOutbound = txn.eventActivityTypeCode === 2;
24
24
  let memo = '';
@@ -58,6 +58,9 @@ function convertTransactions(txns) {
58
58
  status: txn.serialNumber === 0 ? _transactions.TransactionStatuses.Pending : _transactions.TransactionStatuses.Completed,
59
59
  memo
60
60
  };
61
+ if (options?.includeRawTransaction) {
62
+ result.rawTransaction = txn;
63
+ }
61
64
  return result;
62
65
  });
63
66
  }
@@ -110,11 +113,11 @@ async function getExtraScrap(txnsResult, baseUrl, page, accountNumber) {
110
113
  transactions: res
111
114
  };
112
115
  }
113
- async function getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDate, endDate, additionalTransactionInformation = false) {
116
+ async function getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDate, endDate, additionalTransactionInformation = false, options) {
114
117
  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;
115
118
  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');
116
119
  const finalResult = additionalTransactionInformation && txnsResult?.transactions.length ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber) : txnsResult;
117
- return convertTransactions(finalResult?.transactions ?? []);
120
+ return convertTransactions(finalResult?.transactions ?? [], options);
118
121
  }
119
122
  async function getAccountBalance(apiSiteUrl, page, accountNumber) {
120
123
  const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;
@@ -127,7 +130,8 @@ async function fetchAccountData(page, baseUrl, options) {
127
130
  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;
128
131
  debug('fetching accounts data');
129
132
  const accountsInfo = (await (0, _fetch.fetchGetWithinPage)(page, accountDataUrl)) || [];
130
- debug('got %d accounts, fetching txns and balance', accountsInfo.length);
133
+ const openAccountsInfo = accountsInfo.filter(account => account.accountClosingReasonCode === 0);
134
+ debug('got %d open accounts from %d total accounts, fetching txns and balance', openAccountsInfo.length, accountsInfo.length);
131
135
  const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
132
136
  const startDate = options.startDate || defaultStartMoment.toDate();
133
137
  const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
@@ -137,16 +141,11 @@ async function fetchAccountData(page, baseUrl, options) {
137
141
  const startDateStr = startMoment.format(DATE_FORMAT);
138
142
  const endDateStr = (0, _moment.default)().format(DATE_FORMAT);
139
143
  const accounts = [];
140
- for (const account of accountsInfo) {
141
- let balance;
144
+ for (const account of openAccountsInfo) {
145
+ debug('getting information for account %s', account.accountNumber);
142
146
  const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;
143
- const isActiveAccount = account.accountClosingReasonCode === 0;
144
- if (isActiveAccount) {
145
- balance = await getAccountBalance(apiSiteUrl, page, accountNumber);
146
- } else {
147
- debug('Skipping balance for a closed account, balance will be undefined');
148
- }
149
- const txns = await getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDateStr, endDateStr, additionalTransactionInformation);
147
+ const balance = await getAccountBalance(apiSiteUrl, page, accountNumber);
148
+ const txns = await getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDateStr, endDateStr, additionalTransactionInformation, options);
150
149
  accounts.push({
151
150
  accountNumber,
152
151
  balance,
@@ -195,4 +194,4 @@ class HapoalimScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
195
194
  }
196
195
  }
197
196
  var _default = exports.default = HapoalimScraper;
198
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_uuid","_debug","_fetch","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","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","moment","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","getRestContext","page","waitUntil","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","uuid4","fetchPostWithinPage","getExtraScrap","txnsResult","baseUrl","accountNumber","promises","transactions","transaction","pfmDetails","extraTransactionDetails","fetchGetWithinPage","transactionNumber","res","Promise","all","getAccountTransactions","apiSiteUrl","startDate","endDate","additionalTransactionInformation","txnsUrl","finalResult","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","fetchAccountData","options","accountDataUrl","accountsInfo","defaultStartMoment","subtract","add","toDate","startMoment","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","waitForRedirect","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/hapoalim.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport { v4 as uuid4 } from 'uuid';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\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  pfmDetails: string;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ninterface ScrapedPfmTransaction {\n  transactionNumber: number;\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 { partyHeadline, partyName, messageHeadline, messageDetail } = 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(() => {\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(\n  page: Page,\n  url: string,\n  pageUuid: string,\n): 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 getExtraScrap(\n  txnsResult: FetchedAccountTransactionsData,\n  baseUrl: string,\n  page: Page,\n  accountNumber: string,\n): Promise<FetchedAccountTransactionsData> {\n  const promises = txnsResult.transactions.map(async (transaction: ScrapedTransaction): Promise<ScrapedTransaction> => {\n    const { pfmDetails, serialNumber } = transaction;\n    if (serialNumber !== 0) {\n      const url = `${baseUrl}${pfmDetails}&accountId=${accountNumber}&lang=he`;\n      const extraTransactionDetails = (await fetchGetWithinPage<ScrapedPfmTransaction[]>(page, url)) || [];\n      if (extraTransactionDetails && extraTransactionDetails.length) {\n        const { transactionNumber } = extraTransactionDetails[0];\n        if (transactionNumber) {\n          return { ...transaction, referenceNumber: transactionNumber };\n        }\n      }\n    }\n    return transaction;\n  });\n  const res = await Promise.all(promises);\n  return { transactions: res };\n}\n\nasync function getAccountTransactions(\n  baseUrl: string,\n  apiSiteUrl: string,\n  page: Page,\n  accountNumber: string,\n  startDate: string,\n  endDate: string,\n  additionalTransactionInformation = false,\n) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  const finalResult =\n    additionalTransactionInformation && txnsResult?.transactions.length\n      ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber)\n      : txnsResult;\n\n  return convertTransactions(finalResult?.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: ScraperOptions) {\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  const { additionalTransactionInformation } = options;\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(\n      baseUrl,\n      apiSiteUrl,\n      page,\n      accountNumber,\n      startDateStr,\n      endDateStr,\n      additionalTransactionInformation,\n    );\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  ];\n  urls[LoginResults.InvalidPassword] = [\n    `${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`,\n  ];\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: ScraperSpecificCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { userCode: string; password: string };\n\nclass HapoalimScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\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"],"mappings":";;;;;;AAAA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AAEA,IAAAC,KAAA,GAAAD,OAAA;AACA,IAAAE,MAAA,GAAAF,OAAA;AACA,IAAAG,MAAA,GAAAH,OAAA;AACA,IAAAI,WAAA,GAAAJ,OAAA;AACA,IAAAK,QAAA,GAAAL,OAAA;AACA,IAAAM,aAAA,GAAAN,OAAA;AACA,IAAAO,uBAAA,GAAAP,OAAA;AAA8G,SAAAD,uBAAAS,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAG9G,MAAMG,KAAK,GAAG,IAAAC,eAAQ,EAAC,UAAU,CAAC;AAElC,MAAMC,WAAW,GAAG,UAAU;;AAE9B;;AAkDA,SAASC,mBAAmBA,CAACC,IAA0B,EAAiB;EACtE,OAAOA,IAAI,CAACC,GAAG,CAACC,GAAG,IAAI;IACrB,MAAMC,UAAU,GAAGD,GAAG,CAACE,qBAAqB,KAAK,CAAC;IAElD,IAAIC,IAAI,GAAG,EAAE;IACb,IAAIH,GAAG,CAACI,sBAAsB,EAAE;MAC9B,MAAM;QAAEC,aAAa;QAAEC,SAAS;QAAEC,eAAe;QAAEC;MAAc,CAAC,GAAGR,GAAG,CAACI,sBAAsB;MAC/F,MAAMK,SAAmB,GAAG,EAAE;MAC9B,IAAIJ,aAAa,EAAE;QACjBI,SAAS,CAACC,IAAI,CAACL,aAAa,CAAC;MAC/B;MAEA,IAAIC,SAAS,EAAE;QACbG,SAAS,CAACC,IAAI,CAAC,GAAGJ,SAAS,GAAG,CAAC;MACjC;MAEA,IAAIC,eAAe,EAAE;QACnBE,SAAS,CAACC,IAAI,CAACH,eAAe,CAAC;MACjC;MAEA,IAAIC,aAAa,EAAE;QACjBC,SAAS,CAACC,IAAI,CAAC,GAAGF,aAAa,GAAG,CAAC;MACrC;MAEA,IAAIC,SAAS,CAACE,MAAM,EAAE;QACpBR,IAAI,GAAGM,SAAS,CAACG,IAAI,CAAC,GAAG,CAAC;MAC5B;IACF;IAEA,MAAMC,MAAmB,GAAG;MAC1BC,IAAI,EAAEC,8BAAgB,CAACC,MAAM;MAC7BC,UAAU,EAAEjB,GAAG,CAACkB,eAAe;MAC/BC,IAAI,EAAE,IAAAC,eAAM,EAACpB,GAAG,CAACqB,SAAS,EAAEzB,WAAW,CAAC,CAAC0B,WAAW,CAAC,CAAC;MACtDC,aAAa,EAAE,IAAAH,eAAM,EAACpB,GAAG,CAACwB,SAAS,EAAE5B,WAAW,CAAC,CAAC0B,WAAW,CAAC,CAAC;MAC/DG,cAAc,EAAExB,UAAU,GAAG,CAACD,GAAG,CAAC0B,WAAW,GAAG1B,GAAG,CAAC0B,WAAW;MAC/DC,gBAAgB,EAAE,KAAK;MACvBC,aAAa,EAAE3B,UAAU,GAAG,CAACD,GAAG,CAAC0B,WAAW,GAAG1B,GAAG,CAAC0B,WAAW;MAC9DG,WAAW,EAAE7B,GAAG,CAAC8B,mBAAmB,IAAI,EAAE;MAC1CC,MAAM,EAAE/B,GAAG,CAACgC,YAAY,KAAK,CAAC,GAAGC,iCAAmB,CAACC,OAAO,GAAGD,iCAAmB,CAACE,SAAS;MAC5FhC;IACF,CAAC;IAED,OAAOU,MAAM;EACf,CAAC,CAAC;AACJ;AAEA,eAAeuB,cAAcA,CAACC,IAAU,EAAE;EACxC,MAAM,IAAAC,kBAAS,EAAC,MAAM;IACpB,OAAOD,IAAI,CAACE,QAAQ,CAAC,MAAM,CAAC,CAACC,MAAM,CAACC,OAAO,CAAC;EAC9C,CAAC,EAAE,2BAA2B,CAAC;EAE/B,MAAM5B,MAAM,GAAG,MAAMwB,IAAI,CAACE,QAAQ,CAAC,MAAM;IACvC,OAAOC,MAAM,CAACC,OAAO,CAACC,WAAW;EACnC,CAAC,CAAC;EAEF,OAAO7B,MAAM,CAAC8B,KAAK,CAAC,CAAC,CAAC;AACxB;AAEA,eAAeC,yBAAyBA,CACtCP,IAAU,EACVQ,GAAW,EACXC,QAAgB,EACgC;EAChD,MAAMC,OAAO,GAAG,MAAMV,IAAI,CAACU,OAAO,CAAC,CAAC;EACpC,MAAMC,UAAU,GAAGD,OAAO,CAACE,IAAI,CAACC,MAAM,IAAIA,MAAM,CAACC,IAAI,KAAK,YAAY,CAAC;EACvE,MAAMC,OAA4B,GAAG,CAAC,CAAC;EACvC,IAAIJ,UAAU,IAAI,IAAI,EAAE;IACtBI,OAAO,CAAC,cAAc,CAAC,GAAGJ,UAAU,CAACK,KAAK;EAC5C;EACAD,OAAO,CAACN,QAAQ,GAAGA,QAAQ;EAC3BM,OAAO,CAACE,IAAI,GAAG,IAAAC,QAAK,EAAC,CAAC;EACtBH,OAAO,CAAC,cAAc,CAAC,GAAG,gCAAgC;EAC1D,OAAO,IAAAI,0BAAmB,EAAiCnB,IAAI,EAAEQ,GAAG,EAAE,EAAE,EAAEO,OAAO,CAAC;AACpF;AAEA,eAAeK,aAAaA,CAC1BC,UAA0C,EAC1CC,OAAe,EACftB,IAAU,EACVuB,aAAqB,EACoB;EACzC,MAAMC,QAAQ,GAAGH,UAAU,CAACI,YAAY,CAAC/D,GAAG,CAAC,MAAOgE,WAA+B,IAAkC;IACnH,MAAM;MAAEC,UAAU;MAAEhC;IAAa,CAAC,GAAG+B,WAAW;IAChD,IAAI/B,YAAY,KAAK,CAAC,EAAE;MACtB,MAAMa,GAAG,GAAG,GAAGc,OAAO,GAAGK,UAAU,cAAcJ,aAAa,UAAU;MACxE,MAAMK,uBAAuB,GAAG,CAAC,MAAM,IAAAC,yBAAkB,EAA0B7B,IAAI,EAAEQ,GAAG,CAAC,KAAK,EAAE;MACpG,IAAIoB,uBAAuB,IAAIA,uBAAuB,CAACtD,MAAM,EAAE;QAC7D,MAAM;UAAEwD;QAAkB,CAAC,GAAGF,uBAAuB,CAAC,CAAC,CAAC;QACxD,IAAIE,iBAAiB,EAAE;UACrB,OAAO;YAAE,GAAGJ,WAAW;YAAE7C,eAAe,EAAEiD;UAAkB,CAAC;QAC/D;MACF;IACF;IACA,OAAOJ,WAAW;EACpB,CAAC,CAAC;EACF,MAAMK,GAAG,GAAG,MAAMC,OAAO,CAACC,GAAG,CAACT,QAAQ,CAAC;EACvC,OAAO;IAAEC,YAAY,EAAEM;EAAI,CAAC;AAC9B;AAEA,eAAeG,sBAAsBA,CACnCZ,OAAe,EACfa,UAAkB,EAClBnC,IAAU,EACVuB,aAAqB,EACrBa,SAAiB,EACjBC,OAAe,EACfC,gCAAgC,GAAG,KAAK,EACxC;EACA,MAAMC,OAAO,GAAG,GAAGJ,UAAU,2CAA2CZ,aAAa,0CAA0Cc,OAAO,uBAAuBD,SAAS,aAAa;EACnL,MAAMf,UAAU,GAAG,MAAMd,yBAAyB,CAACP,IAAI,EAAEuC,OAAO,EAAE,+BAA+B,CAAC;EAElG,MAAMC,WAAW,GACfF,gCAAgC,IAAIjB,UAAU,EAAEI,YAAY,CAACnD,MAAM,GAC/D,MAAM8C,aAAa,CAACC,UAAU,EAAEC,OAAO,EAAEtB,IAAI,EAAEuB,aAAa,CAAC,GAC7DF,UAAU;EAEhB,OAAO7D,mBAAmB,CAACgF,WAAW,EAAEf,YAAY,IAAI,EAAE,CAAC;AAC7D;AAEA,eAAegB,iBAAiBA,CAACN,UAAkB,EAAEnC,IAAU,EAAEuB,aAAqB,EAAE;EACtF,MAAMmB,wBAAwB,GAAG,GAAGP,UAAU,8DAA8DZ,aAAa,uBAAuB;EAChJ,MAAMoB,qBAAqB,GAAG,MAAM,IAAAd,yBAAkB,EAAwB7B,IAAI,EAAE0C,wBAAwB,CAAC;EAE7G,OAAOC,qBAAqB,EAAEC,cAAc;AAC9C;AAEA,eAAeC,gBAAgBA,CAAC7C,IAAU,EAAEsB,OAAe,EAAEwB,OAAuB,EAAE;EACpF,MAAMzC,WAAW,GAAG,MAAMN,cAAc,CAACC,IAAI,CAAC;EAC9C,MAAMmC,UAAU,GAAG,GAAGb,OAAO,IAAIjB,WAAW,EAAE;EAC9C,MAAM0C,cAAc,GAAG,GAAGzB,OAAO,kCAAkC;EAEnEjE,KAAK,CAAC,wBAAwB,CAAC;EAC/B,MAAM2F,YAAY,GAAG,CAAC,MAAM,IAAAnB,yBAAkB,EAAqB7B,IAAI,EAAE+C,cAAc,CAAC,KAAK,EAAE;EAC/F1F,KAAK,CAAC,4CAA4C,EAAE2F,YAAY,CAAC1E,MAAM,CAAC;EAExE,MAAM2E,kBAAkB,GAAG,IAAAlE,eAAM,EAAC,CAAC,CAACmE,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC,CAACC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC;EACtE,MAAMf,SAAS,GAAGU,OAAO,CAACV,SAAS,IAAIa,kBAAkB,CAACG,MAAM,CAAC,CAAC;EAClE,MAAMC,WAAW,GAAGtE,eAAM,CAACuE,GAAG,CAACL,kBAAkB,EAAE,IAAAlE,eAAM,EAACqD,SAAS,CAAC,CAAC;EACrE,MAAM;IAAEE;EAAiC,CAAC,GAAGQ,OAAO;EAEpD,MAAMS,YAAY,GAAGF,WAAW,CAACG,MAAM,CAACjG,WAAW,CAAC;EACpD,MAAMkG,UAAU,GAAG,IAAA1E,eAAM,EAAC,CAAC,CAACyE,MAAM,CAACjG,WAAW,CAAC;EAE/C,MAAMmG,QAA+B,GAAG,EAAE;EAE1C,KAAK,MAAMC,OAAO,IAAIX,YAAY,EAAE;IAClC,IAAIY,OAA2B;IAC/B,MAAMrC,aAAa,GAAG,GAAGoC,OAAO,CAACE,UAAU,IAAIF,OAAO,CAACG,YAAY,IAAIH,OAAO,CAACpC,aAAa,EAAE;IAE9F,MAAMwC,eAAe,GAAGJ,OAAO,CAACK,wBAAwB,KAAK,CAAC;IAC9D,IAAID,eAAe,EAAE;MACnBH,OAAO,GAAG,MAAMnB,iBAAiB,CAACN,UAAU,EAAEnC,IAAI,EAAEuB,aAAa,CAAC;IACpE,CAAC,MAAM;MACLlE,KAAK,CAAC,kEAAkE,CAAC;IAC3E;IAEA,MAAMI,IAAI,GAAG,MAAMyE,sBAAsB,CACvCZ,OAAO,EACPa,UAAU,EACVnC,IAAI,EACJuB,aAAa,EACbgC,YAAY,EACZE,UAAU,EACVnB,gCACF,CAAC;IAEDoB,QAAQ,CAACrF,IAAI,CAAC;MACZkD,aAAa;MACbqC,OAAO;MACPnG;IACF,CAAC,CAAC;EACJ;EAEA,MAAMwG,WAAW,GAAG;IAClBC,OAAO,EAAE,IAAI;IACbR;EACF,CAAC;EACDrG,KAAK,CAAC,gBAAgB,CAAC;EACvB,OAAO4G,WAAW;AACpB;AAEA,SAASE,uBAAuBA,CAAC7C,OAAe,EAAE;EAChD,MAAM8C,IAA0B,GAAG,CAAC,CAAC;EACrCA,IAAI,CAACC,oCAAY,CAACC,OAAO,CAAC,GAAG,CAC3B,GAAGhD,OAAO,wBAAwB,EAClC,GAAGA,OAAO,+BAA+B,EACzC,GAAGA,OAAO,4BAA4B,CACvC;EACD8C,IAAI,CAACC,oCAAY,CAACE,eAAe,CAAC,GAAG,CACnC,GAAGjD,OAAO,8EAA8E,CACzF;EACD8C,IAAI,CAACC,oCAAY,CAACG,cAAc,CAAC,GAAG,CAClC,GAAGlD,OAAO,kDAAkD,EAC5D,yBAAyB,CAC1B;EACD,OAAO8C,IAAI;AACb;AAEA,SAASK,iBAAiBA,CAACC,WAAuC,EAAE;EAClE,OAAO,CACL;IAAEC,QAAQ,EAAE,WAAW;IAAE3D,KAAK,EAAE0D,WAAW,CAACE;EAAS,CAAC,EACtD;IAAED,QAAQ,EAAE,WAAW;IAAE3D,KAAK,EAAE0D,WAAW,CAACG;EAAS,CAAC,CACvD;AACH;AAIA,MAAMC,eAAe,SAASC,8CAAsB,CAA6B;EAC/E;EACA,IAAIzD,OAAOA,CAAA,EAAG;IACZ,OAAO,kCAAkC;EAC3C;EAEA0D,eAAeA,CAACN,WAAuC,EAAE;IACvD,OAAO;MACLO,QAAQ,EAAE,GAAG,IAAI,CAAC3D,OAAO,wCAAwC;MACjE4D,MAAM,EAAET,iBAAiB,CAACC,WAAW,CAAC;MACtCS,oBAAoB,EAAE,YAAY;MAClCC,UAAU,EAAE,MAAAA,CAAA,KAAY,IAAAC,2BAAe,EAAC,IAAI,CAACrF,IAAI,CAAC;MAClDsF,eAAe,EAAEnB,uBAAuB,CAAC,IAAI,CAAC7C,OAAO;IACvD,CAAC;EACH;EAEA,MAAMiE,SAASA,CAAA,EAAG;IAChB,OAAO1C,gBAAgB,CAAC,IAAI,CAAC7C,IAAI,EAAE,IAAI,CAACsB,OAAO,EAAE,IAAI,CAACwB,OAAO,CAAC;EAChE;AACF;AAAC,IAAA0C,QAAA,GAAAC,OAAA,CAAArI,OAAA,GAEc0H,eAAe","ignoreList":[]}
197
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_uuid","_debug","_fetch","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","DATE_FORMAT","convertTransactions","txns","options","map","txn","isOutbound","eventActivityTypeCode","memo","beneficiaryDetailsData","partyHeadline","partyName","messageHeadline","messageDetail","memoLines","push","length","join","result","type","TransactionTypes","Normal","identifier","referenceNumber","date","moment","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","includeRawTransaction","rawTransaction","getRestContext","page","waitUntil","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","uuid4","fetchPostWithinPage","getExtraScrap","txnsResult","baseUrl","accountNumber","promises","transactions","transaction","pfmDetails","extraTransactionDetails","fetchGetWithinPage","transactionNumber","res","Promise","all","getAccountTransactions","apiSiteUrl","startDate","endDate","additionalTransactionInformation","txnsUrl","finalResult","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","fetchAccountData","accountDataUrl","accountsInfo","openAccountsInfo","filter","account","accountClosingReasonCode","defaultStartMoment","subtract","add","toDate","startMoment","max","startDateStr","format","endDateStr","accounts","bankNumber","branchNumber","balance","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","waitForRedirect","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/hapoalim.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport { v4 as uuid4 } from 'uuid';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\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  pfmDetails: string;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ninterface ScrapedPfmTransaction {\n  transactionNumber: number;\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[], options?: ScraperOptions): Transaction[] {\n  return txns.map(txn => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n\n    let memo = '';\n    if (txn.beneficiaryDetailsData) {\n      const { partyHeadline, partyName, messageHeadline, messageDetail } = 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    if (options?.includeRawTransaction) {\n      result.rawTransaction = txn;\n    }\n\n    return result;\n  });\n}\n\nasync function getRestContext(page: Page) {\n  await waitUntil(() => {\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(\n  page: Page,\n  url: string,\n  pageUuid: string,\n): 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 getExtraScrap(\n  txnsResult: FetchedAccountTransactionsData,\n  baseUrl: string,\n  page: Page,\n  accountNumber: string,\n): Promise<FetchedAccountTransactionsData> {\n  const promises = txnsResult.transactions.map(async (transaction: ScrapedTransaction): Promise<ScrapedTransaction> => {\n    const { pfmDetails, serialNumber } = transaction;\n    if (serialNumber !== 0) {\n      const url = `${baseUrl}${pfmDetails}&accountId=${accountNumber}&lang=he`;\n      const extraTransactionDetails = (await fetchGetWithinPage<ScrapedPfmTransaction[]>(page, url)) || [];\n      if (extraTransactionDetails && extraTransactionDetails.length) {\n        const { transactionNumber } = extraTransactionDetails[0];\n        if (transactionNumber) {\n          return { ...transaction, referenceNumber: transactionNumber };\n        }\n      }\n    }\n    return transaction;\n  });\n  const res = await Promise.all(promises);\n  return { transactions: res };\n}\n\nasync function getAccountTransactions(\n  baseUrl: string,\n  apiSiteUrl: string,\n  page: Page,\n  accountNumber: string,\n  startDate: string,\n  endDate: string,\n  additionalTransactionInformation = false,\n  options?: ScraperOptions,\n) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  const finalResult =\n    additionalTransactionInformation && txnsResult?.transactions.length\n      ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber)\n      : txnsResult;\n\n  return convertTransactions(finalResult?.transactions ?? [], options);\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: ScraperOptions) {\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  const openAccountsInfo = accountsInfo.filter(account => account.accountClosingReasonCode === 0);\n  debug(\n    'got %d open accounts from %d total accounts, fetching txns and balance',\n    openAccountsInfo.length,\n    accountsInfo.length,\n  );\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  const { additionalTransactionInformation } = options;\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 openAccountsInfo) {\n    debug('getting information for account %s', account.accountNumber);\n    const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;\n\n    const balance = await getAccountBalance(apiSiteUrl, page, accountNumber);\n    const txns = await getAccountTransactions(\n      baseUrl,\n      apiSiteUrl,\n      page,\n      accountNumber,\n      startDateStr,\n      endDateStr,\n      additionalTransactionInformation,\n      options,\n    );\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  ];\n  urls[LoginResults.InvalidPassword] = [\n    `${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`,\n  ];\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: ScraperSpecificCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { userCode: string; password: string };\n\nclass HapoalimScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\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"],"mappings":"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","ignoreList":[]}
@@ -132,6 +132,11 @@ export type ScraperOptions = ScraperBrowserOptions & {
132
132
  * Please note: It will take more time to finish the process.
133
133
  */
134
134
  additionalTransactionInformation?: boolean;
135
+ /**
136
+ * Include the raw transaction object as received from the scraper source for debugging purposes.
137
+ * @default false
138
+ */
139
+ includeRawTransaction?: boolean;
135
140
  /**
136
141
  * Adjust the viewport size of the browser page.
137
142
  * If not set, the default viewport size of 1024x768 will be used.
@@ -3,4 +3,4 @@
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":[],"sources":["../../src/scrapers/interface.ts"],"sourcesContent":["import { type BrowserContext, type Browser, type Page } from 'puppeteer';\nimport { type CompanyTypes, type ScraperProgressTypes } from '../definitions';\nimport { type TransactionsAccount } from '../transactions';\nimport { type ErrorResult, type ScraperErrorTypes } from './errors';\n\n// TODO: Remove this type when the scraper 'factory' will return concrete scraper types\n// Instead of a generic interface (which in turn uses this type)\nexport type ScraperCredentials =\n  | { userCode: string; password: string }\n  | { username: string; password: string }\n  | { id: string; password: string }\n  | { id: string; password: string; num: string }\n  | { id: string; password: string; card6Digits: string }\n  | { username: string; nationalID: string; password: string }\n  | ({ email: string; password: string } & (\n      | {\n          otpCodeRetriever: () => Promise<string>;\n          phoneNumber: string;\n        }\n      | {\n          otpLongTermToken: string;\n        }\n    ));\n\nexport type OptInFeatures =\n  | 'isracard-amex:skipAdditionalTransactionInformation'\n  | 'mizrahi:pendingIfNoIdentifier'\n  | 'mizrahi:pendingIfHasGenericDescription'\n  | 'mizrahi:pendingIfTodayTransaction';\n\nexport interface FutureDebit {\n  amount: number;\n  amountCurrency: string;\n  chargeDate?: string;\n  bankAccountNumber?: string;\n}\n\ninterface ExternalBrowserOptions {\n  /**\n   * An externally created browser instance.\n   * you can get a browser directly from puppeteer via `puppeteer.launch()`\n   *\n   * Note: The browser will be closed by the library after the scraper finishes unless `skipCloseBrowser` is set to true\n   */\n  browser: Browser;\n\n  /**\n   * If true, the browser will not be closed by the library after the scraper finishes\n   */\n  skipCloseBrowser?: boolean;\n}\n\ninterface ExternalBrowserContextOptions {\n  /**\n   * An externally managed browser context. This is useful when you want to manage the browser\n   */\n  browserContext: BrowserContext;\n}\n\ninterface DefaultBrowserOptions {\n  /**\n   * shows the browser while scraping, good for debugging (default false)\n   */\n  showBrowser?: boolean;\n\n  /**\n   * provide a patch to local chromium to be used by puppeteer. Relevant when using\n   * `israeli-bank-scrapers-core` library\n   */\n  executablePath?: string;\n\n  /**\n   * additional arguments to pass to the browser instance. The list of flags can be found in\n   *\n   * https://developer.mozilla.org/en-US/docs/Mozilla/Command_Line_Options\n   * https://peter.sh/experiments/chromium-command-line-switches/\n   */\n  args?: string[];\n\n  /**\n   * Maximum navigation time in milliseconds, pass 0 to disable timeout.\n   * @default 30000\n   */\n  timeout?: number;\n\n  /**\n   * adjust the browser instance before it is being used\n   *\n   * @param browser\n   */\n  prepareBrowser?: (browser: Browser) => Promise<void>;\n}\n\ntype ScraperBrowserOptions = ExternalBrowserOptions | ExternalBrowserContextOptions | DefaultBrowserOptions;\n\nexport type ScraperOptions = ScraperBrowserOptions & {\n  /**\n   * The company you want to scrape\n   */\n  companyId: CompanyTypes;\n\n  /**\n   * include more debug info about in the output\n   */\n  verbose?: boolean;\n\n  /**\n   * the date to fetch transactions from (can't be before the minimum allowed time difference for the scraper)\n   */\n  startDate: Date;\n\n  /**\n   * scrape transactions to be processed X months in the future\n   */\n  futureMonthsToScrape?: number;\n\n  /**\n   * if set to true, all installment transactions will be combine into the first one\n   */\n  combineInstallments?: boolean;\n\n  /**\n   * adjust the page instance before it is being used.\n   *\n   * @param page\n   */\n  preparePage?: (page: Page) => Promise<void>;\n\n  /**\n   * if set, store a screenshot if failed to scrape. Used for debug purposes\n   */\n  storeFailureScreenShotPath?: string;\n\n  /**\n   * if set, will set the timeout in milliseconds of puppeteer's `page.setDefaultTimeout`.\n   */\n  defaultTimeout?: number;\n\n  /**\n   * Options for manipulation of output data\n   */\n  outputData?: OutputDataOptions;\n\n  /**\n   * Perform additional operation for each transaction to get more information (Like category) about it.\n   * Please note: It will take more time to finish the process.\n   */\n  additionalTransactionInformation?: boolean;\n\n  /**\n   * Adjust the viewport size of the browser page.\n   * If not set, the default viewport size of 1024x768 will be used.\n   */\n  viewportSize?: {\n    width: number;\n    height: number;\n  };\n\n  /**\n   * The number of times to retry the navigation in case of a failure (default 0)\n   */\n  navigationRetryCount?: number;\n\n  /**\n   * Opt-in features for the scrapers, allowing safe rollout of new breaking changes.\n   */\n  optInFeatures?: Array<OptInFeatures>;\n};\n\nexport interface OutputDataOptions {\n  /**\n   * if true, the result wouldn't be filtered out by date, and you will return unfiltered scrapped data.\n   */\n  enableTransactionsFilterByDate?: boolean;\n}\n\nexport interface ScraperScrapingResult {\n  success: boolean;\n  accounts?: TransactionsAccount[];\n  futureDebits?: FutureDebit[];\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport interface Scraper<TCredentials extends ScraperCredentials> {\n  scrape(credentials: TCredentials): Promise<ScraperScrapingResult>;\n  onProgress(func: (companyId: CompanyTypes, payload: { type: ScraperProgressTypes }) => void): void;\n  triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult>;\n  getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult>;\n}\n\nexport type ScraperTwoFactorAuthTriggerResult =\n  | ErrorResult\n  | {\n      success: true;\n    };\n\nexport type ScraperGetLongTermTwoFactorTokenResult =\n  | ErrorResult\n  | {\n      success: true;\n      longTermTwoFactorAuthToken: string;\n    };\n\nexport interface ScraperLoginResult {\n  success: boolean;\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n  persistentOtpToken?: string;\n}\n"],"mappings":"","ignoreList":[]}
6
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":[],"sources":["../../src/scrapers/interface.ts"],"sourcesContent":["import { type BrowserContext, type Browser, type Page } from 'puppeteer';\nimport { type CompanyTypes, type ScraperProgressTypes } from '../definitions';\nimport { type TransactionsAccount } from '../transactions';\nimport { type ErrorResult, type ScraperErrorTypes } from './errors';\n\n// TODO: Remove this type when the scraper 'factory' will return concrete scraper types\n// Instead of a generic interface (which in turn uses this type)\nexport type ScraperCredentials =\n  | { userCode: string; password: string }\n  | { username: string; password: string }\n  | { id: string; password: string }\n  | { id: string; password: string; num: string }\n  | { id: string; password: string; card6Digits: string }\n  | { username: string; nationalID: string; password: string }\n  | ({ email: string; password: string } & (\n      | {\n          otpCodeRetriever: () => Promise<string>;\n          phoneNumber: string;\n        }\n      | {\n          otpLongTermToken: string;\n        }\n    ));\n\nexport type OptInFeatures =\n  | 'isracard-amex:skipAdditionalTransactionInformation'\n  | 'mizrahi:pendingIfNoIdentifier'\n  | 'mizrahi:pendingIfHasGenericDescription'\n  | 'mizrahi:pendingIfTodayTransaction';\n\nexport interface FutureDebit {\n  amount: number;\n  amountCurrency: string;\n  chargeDate?: string;\n  bankAccountNumber?: string;\n}\n\ninterface ExternalBrowserOptions {\n  /**\n   * An externally created browser instance.\n   * you can get a browser directly from puppeteer via `puppeteer.launch()`\n   *\n   * Note: The browser will be closed by the library after the scraper finishes unless `skipCloseBrowser` is set to true\n   */\n  browser: Browser;\n\n  /**\n   * If true, the browser will not be closed by the library after the scraper finishes\n   */\n  skipCloseBrowser?: boolean;\n}\n\ninterface ExternalBrowserContextOptions {\n  /**\n   * An externally managed browser context. This is useful when you want to manage the browser\n   */\n  browserContext: BrowserContext;\n}\n\ninterface DefaultBrowserOptions {\n  /**\n   * shows the browser while scraping, good for debugging (default false)\n   */\n  showBrowser?: boolean;\n\n  /**\n   * provide a patch to local chromium to be used by puppeteer. Relevant when using\n   * `israeli-bank-scrapers-core` library\n   */\n  executablePath?: string;\n\n  /**\n   * additional arguments to pass to the browser instance. The list of flags can be found in\n   *\n   * https://developer.mozilla.org/en-US/docs/Mozilla/Command_Line_Options\n   * https://peter.sh/experiments/chromium-command-line-switches/\n   */\n  args?: string[];\n\n  /**\n   * Maximum navigation time in milliseconds, pass 0 to disable timeout.\n   * @default 30000\n   */\n  timeout?: number;\n\n  /**\n   * adjust the browser instance before it is being used\n   *\n   * @param browser\n   */\n  prepareBrowser?: (browser: Browser) => Promise<void>;\n}\n\ntype ScraperBrowserOptions = ExternalBrowserOptions | ExternalBrowserContextOptions | DefaultBrowserOptions;\n\nexport type ScraperOptions = ScraperBrowserOptions & {\n  /**\n   * The company you want to scrape\n   */\n  companyId: CompanyTypes;\n\n  /**\n   * include more debug info about in the output\n   */\n  verbose?: boolean;\n\n  /**\n   * the date to fetch transactions from (can't be before the minimum allowed time difference for the scraper)\n   */\n  startDate: Date;\n\n  /**\n   * scrape transactions to be processed X months in the future\n   */\n  futureMonthsToScrape?: number;\n\n  /**\n   * if set to true, all installment transactions will be combine into the first one\n   */\n  combineInstallments?: boolean;\n\n  /**\n   * adjust the page instance before it is being used.\n   *\n   * @param page\n   */\n  preparePage?: (page: Page) => Promise<void>;\n\n  /**\n   * if set, store a screenshot if failed to scrape. Used for debug purposes\n   */\n  storeFailureScreenShotPath?: string;\n\n  /**\n   * if set, will set the timeout in milliseconds of puppeteer's `page.setDefaultTimeout`.\n   */\n  defaultTimeout?: number;\n\n  /**\n   * Options for manipulation of output data\n   */\n  outputData?: OutputDataOptions;\n\n  /**\n   * Perform additional operation for each transaction to get more information (Like category) about it.\n   * Please note: It will take more time to finish the process.\n   */\n  additionalTransactionInformation?: boolean;\n\n  /**\n   * Include the raw transaction object as received from the scraper source for debugging purposes.\n   * @default false\n   */\n  includeRawTransaction?: boolean;\n\n  /**\n   * Adjust the viewport size of the browser page.\n   * If not set, the default viewport size of 1024x768 will be used.\n   */\n  viewportSize?: {\n    width: number;\n    height: number;\n  };\n\n  /**\n   * The number of times to retry the navigation in case of a failure (default 0)\n   */\n  navigationRetryCount?: number;\n\n  /**\n   * Opt-in features for the scrapers, allowing safe rollout of new breaking changes.\n   */\n  optInFeatures?: Array<OptInFeatures>;\n};\n\nexport interface OutputDataOptions {\n  /**\n   * if true, the result wouldn't be filtered out by date, and you will return unfiltered scrapped data.\n   */\n  enableTransactionsFilterByDate?: boolean;\n}\n\nexport interface ScraperScrapingResult {\n  success: boolean;\n  accounts?: TransactionsAccount[];\n  futureDebits?: FutureDebit[];\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport interface Scraper<TCredentials extends ScraperCredentials> {\n  scrape(credentials: TCredentials): Promise<ScraperScrapingResult>;\n  onProgress(func: (companyId: CompanyTypes, payload: { type: ScraperProgressTypes }) => void): void;\n  triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult>;\n  getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult>;\n}\n\nexport type ScraperTwoFactorAuthTriggerResult =\n  | ErrorResult\n  | {\n      success: true;\n    };\n\nexport type ScraperGetLongTermTwoFactorTokenResult =\n  | ErrorResult\n  | {\n      success: true;\n      longTermTwoFactorAuthToken: string;\n    };\n\nexport interface ScraperLoginResult {\n  success: boolean;\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n  persistentOtpToken?: string;\n}\n"],"mappings":"","ignoreList":[]}