israeli-bank-scrapers-core 6.4.2 → 6.5.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -155,13 +155,13 @@ function getMemo({
155
155
  }
156
156
  return comments;
157
157
  }
158
- function mapTransaction(rawTransaction) {
158
+ function mapTransaction(rawTransaction, options) {
159
159
  const isPending = rawTransaction.paymentDate === null;
160
160
  const processedDate = (0, _moment.default)(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();
161
161
  const status = isPending ? _transactions2.TransactionStatuses.Pending : _transactions2.TransactionStatuses.Completed;
162
162
  const installments = getInstallmentsInfo(rawTransaction.comments);
163
163
  const identifier = installments ? `${rawTransaction.dealData?.arn}_${installments.number}` : rawTransaction.dealData?.arn;
164
- return {
164
+ const result = {
165
165
  type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),
166
166
  date: (0, _moment.default)(rawTransaction.purchaseDate).toISOString(),
167
167
  processedDate,
@@ -176,8 +176,12 @@ function mapTransaction(rawTransaction) {
176
176
  identifier,
177
177
  status
178
178
  };
179
+ if (options?.includeRawTransaction) {
180
+ result.rawTransaction = rawTransaction;
181
+ }
182
+ return result;
179
183
  }
180
- async function fetchTransactionsForMonth(page, monthMoment) {
184
+ async function fetchTransactionsForMonth(page, monthMoment, options) {
181
185
  const url = getTransactionsUrl(monthMoment);
182
186
  const data = await (0, _fetch.fetchGetWithinPage)(page, url);
183
187
  const transactionsByAccount = {};
@@ -188,7 +192,7 @@ async function fetchTransactionsForMonth(page, monthMoment) {
188
192
  if (!transactionsByAccount[transaction.shortCardNumber]) {
189
193
  transactionsByAccount[transaction.shortCardNumber] = [];
190
194
  }
191
- const mappedTransaction = mapTransaction(transaction);
195
+ const mappedTransaction = mapTransaction(transaction, options);
192
196
  transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);
193
197
  });
194
198
  return transactionsByAccount;
@@ -224,7 +228,7 @@ async function fetchTransactions(page, options) {
224
228
  await loadCategories(page);
225
229
  let allResults = {};
226
230
  for (let i = 0; i < allMonths.length; i += 1) {
227
- const result = await fetchTransactionsForMonth(page, allMonths[i]);
231
+ const result = await fetchTransactionsForMonth(page, allMonths[i], options);
228
232
  allResults = addResult(allResults, result);
229
233
  }
230
234
  Object.keys(allResults).forEach(accountNumber => {
@@ -296,4 +300,4 @@ class MaxScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
296
300
  }
297
301
  }
298
302
  var _default = exports.default = MaxScraper;
299
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_dates","_debug","_elementsInteractions","_fetch","_navigation","_transactions","_transactions2","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","MaxPlanName","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","waitForRedirect","waitUntilElementFound","getTransactionsUrl","monthMoment","month","year","date","url","URL","searchParams","set","toString","loadCategories","res","fetchGetWithinPage","Array","isArray","result","length","forEach","id","name","getTransactionType","planName","planTypeId","cleanedUpTxnTypeStr","replace","trim","ImmediateCharge","Normal","MonthlyCharge","OneMonthPostponed","MonthlyPostponed","FuturePurchaseFinancing","MonthlyPayment","MonthlyPostponedInstallments","ThirtyDaysPlus","TwoMonthsPostponed","TwoMonthsPostponed2","AccumulatingBasket","InternetShopping","MonthlyChargePlusInterest","PostponedTransactionInstallments","ReplacementCard","EarlyRepayment","MonthlyCardFee","CurrencyPocket","TransactionTypes","Installments","Credit","CreditOutsideTheLimit","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","getChargedCurrency","currencyId","SHEKEL_CURRENCY","DOLLAR_CURRENCY","EURO_CURRENCY","getMemo","fundsTransferReceiverOrTransfer","fundsTransferComment","memo","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","moment","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","chargedCurrency","paymentCurrency","description","merchantName","category","get","categoryId","fetchTransactionsForMonth","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","enableTransactionsFilterByDate","clonedTxns","from","fixInstallments","sortTransactionsByDate","filterOldTransactions","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startMomentLimit","startDate","toDate","max","allMonths","getAllMonthMoments","i","outputData","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","elementPresentOnPage","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","clickButton","checkReadiness","postAction","possibleResults","waitUntil","fetchData","results","accounts","map","success","_default","exports"],"sources":["../../src/scrapers/max.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer-core';\nimport { DOLLAR_CURRENCY, EURO_CURRENCY, SHEKEL_CURRENCY } from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { filterOldTransactions, fixInstallments, sortTransactionsByDate } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport {\n  BaseScraperWithBrowser,\n  LoginResults,\n  type LoginOptions,\n  type PossibleLoginResults,\n} from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('max');\n\nexport interface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  paymentCurrency: number | null;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  planTypeId: number;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  fundsTransferComment?: string;\n  fundsTransferReceiverOrTransfer?: string;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/login`;\nconst PASSWORD_EXPIRED_URL = `${BASE_WELCOME_URL}/renew-password`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nenum MaxPlanName {\n  Normal = 'רגילה',\n  ImmediateCharge = 'חיוב עסקות מיידי',\n  InternetShopping = 'אינטרנט/חו\"ל',\n  Installments = 'תשלומים',\n  MonthlyCharge = 'חיוב חודשי',\n  OneMonthPostponed = 'דחוי חודש',\n  MonthlyPostponed = 'דחוי לחיוב החודשי',\n  MonthlyPayment = 'תשלום חודשי',\n  FuturePurchaseFinancing = 'מימון לרכישה עתידית',\n  MonthlyPostponedInstallments = 'דחוי חודש תשלומים',\n  ThirtyDaysPlus = 'עסקת 30 פלוס',\n  TwoMonthsPostponed = 'דחוי חודשיים',\n  TwoMonthsPostponed2 = \"דחוי 2 ח' תשלומים\",\n  MonthlyChargePlusInterest = 'חודשי + ריבית',\n  Credit = 'קרדיט',\n  CreditOutsideTheLimit = 'קרדיט-מחוץ למסגרת',\n  AccumulatingBasket = 'סל מצטבר',\n  PostponedTransactionInstallments = 'פריסת העסקה הדחויה',\n  ReplacementCard = 'כרטיס חליפי',\n  EarlyRepayment = 'פרעון מוקדם',\n  MonthlyCardFee = 'דמי כרטיס',\n  CurrencyPocket = 'חיוב ארנק מטח',\n}\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n   * url explanation:\n   * userIndex: -1 for all account owners\n   * cardIndex: -1 for all cards under the account\n   * all other query params are static, beside the date which changes for request per month\n   */\n  const url = new URL(`${BASE_API_ACTIONS_URL}/api/registered/transactionDetails/getTransactionsAndGraphs`);\n  url.searchParams.set(\n    'filterData',\n    `{\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}`,\n  );\n  url.searchParams.set('firstCallCardIndex', '-1');\n  return url.toString();\n}\n\ninterface FetchCategoryResult {\n  result?: Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n    res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(planName: string, planTypeId: number) {\n  const cleanedUpTxnTypeStr = planName.replace('\\t', ' ').trim() as MaxPlanName;\n  switch (cleanedUpTxnTypeStr) {\n    case MaxPlanName.ImmediateCharge:\n    case MaxPlanName.Normal:\n    case MaxPlanName.MonthlyCharge:\n    case MaxPlanName.OneMonthPostponed:\n    case MaxPlanName.MonthlyPostponed:\n    case MaxPlanName.FuturePurchaseFinancing:\n    case MaxPlanName.MonthlyPayment:\n    case MaxPlanName.MonthlyPostponedInstallments:\n    case MaxPlanName.ThirtyDaysPlus:\n    case MaxPlanName.TwoMonthsPostponed:\n    case MaxPlanName.TwoMonthsPostponed2:\n    case MaxPlanName.AccumulatingBasket:\n    case MaxPlanName.InternetShopping:\n    case MaxPlanName.MonthlyChargePlusInterest:\n    case MaxPlanName.PostponedTransactionInstallments:\n    case MaxPlanName.ReplacementCard:\n    case MaxPlanName.EarlyRepayment:\n    case MaxPlanName.MonthlyCardFee:\n    case MaxPlanName.CurrencyPocket:\n      return TransactionTypes.Normal;\n    case MaxPlanName.Installments:\n    case MaxPlanName.Credit:\n    case MaxPlanName.CreditOutsideTheLimit:\n      return TransactionTypes.Installments;\n    default:\n      switch (planTypeId) {\n        case 2:\n        case 3:\n          return TransactionTypes.Installments;\n        case 5:\n          return TransactionTypes.Normal;\n        default:\n          throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr as string}`);\n      }\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\n\nfunction getChargedCurrency(currencyId: number | null) {\n  switch (currencyId) {\n    case 376:\n      return SHEKEL_CURRENCY;\n    case 840:\n      return DOLLAR_CURRENCY;\n    case 978:\n      return EURO_CURRENCY;\n    default:\n      return undefined;\n  }\n}\n\nexport function getMemo({\n  comments,\n  fundsTransferReceiverOrTransfer,\n  fundsTransferComment,\n}: Pick<ScrapedTransaction, 'comments' | 'fundsTransferReceiverOrTransfer' | 'fundsTransferComment'>) {\n  if (fundsTransferReceiverOrTransfer) {\n    const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;\n    return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;\n  }\n\n  return comments;\n}\n\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments\n    ? `${rawTransaction.dealData?.arn}_${installments.number}`\n    : rawTransaction.dealData?.arn;\n\n  return {\n    type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),\n    description: rawTransaction.merchantName.trim(),\n    memo: getMemo(rawTransaction),\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult {\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter(transaction => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach(accountNumber => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(\n  txns: Transaction[],\n  startMoment: moment.Moment,\n  combineInstallments: boolean,\n  enableTransactionsFilterByDate: boolean,\n) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = enableTransactionsFilterByDate\n    ? filterOldTransactions(clonedTxns, startMoment, combineInstallments || false)\n    : clonedTxns;\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startMomentLimit = moment().subtract(4, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(startMomentLimit, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach(accountNumber => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(\n      txns,\n      startMoment,\n      options.combineInstallments || false,\n      options.outputData?.enableTransactionsFilterByDate ?? true,\n    );\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [\n    async () => {\n      return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n    },\n  ];\n  urls[LoginResults.UnknownError] = [\n    async () => {\n      return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n    },\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MaxScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'app-user-login-form .general-button.send-me-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        if (await elementPresentOnPage(this.page, '.login-link#private')) {\n          await clickButton(this.page, '.login-link#private');\n        }\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n      waitUntil: 'domcontentloaded',\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map(accountNumber => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"],"mappings":"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","ignoreList":[]}
303
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_dates","_debug","_elementsInteractions","_fetch","_navigation","_transactions","_transactions2","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","MaxPlanName","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","waitForRedirect","waitUntilElementFound","getTransactionsUrl","monthMoment","month","year","date","url","URL","searchParams","set","toString","loadCategories","res","fetchGetWithinPage","Array","isArray","result","length","forEach","id","name","getTransactionType","planName","planTypeId","cleanedUpTxnTypeStr","replace","trim","ImmediateCharge","Normal","MonthlyCharge","OneMonthPostponed","MonthlyPostponed","FuturePurchaseFinancing","MonthlyPayment","MonthlyPostponedInstallments","ThirtyDaysPlus","TwoMonthsPostponed","TwoMonthsPostponed2","AccumulatingBasket","InternetShopping","MonthlyChargePlusInterest","PostponedTransactionInstallments","ReplacementCard","EarlyRepayment","MonthlyCardFee","CurrencyPocket","TransactionTypes","Installments","Credit","CreditOutsideTheLimit","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","getChargedCurrency","currencyId","SHEKEL_CURRENCY","DOLLAR_CURRENCY","EURO_CURRENCY","getMemo","fundsTransferReceiverOrTransfer","fundsTransferComment","memo","mapTransaction","rawTransaction","options","isPending","paymentDate","processedDate","moment","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","chargedCurrency","paymentCurrency","description","merchantName","category","get","categoryId","includeRawTransaction","fetchTransactionsForMonth","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","enableTransactionsFilterByDate","clonedTxns","from","fixInstallments","sortTransactionsByDate","filterOldTransactions","fetchTransactions","futureMonthsToScrape","defaultStartMoment","subtract","startMomentLimit","startDate","toDate","max","allMonths","getAllMonthMoments","i","outputData","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","elementPresentOnPage","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","clickButton","checkReadiness","postAction","possibleResults","waitUntil","fetchData","results","accounts","map","success","_default","exports"],"sources":["../../src/scrapers/max.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer-core';\nimport { DOLLAR_CURRENCY, EURO_CURRENCY, SHEKEL_CURRENCY } from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { filterOldTransactions, fixInstallments, sortTransactionsByDate } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport {\n  BaseScraperWithBrowser,\n  LoginResults,\n  type LoginOptions,\n  type PossibleLoginResults,\n} from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('max');\n\nexport interface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  paymentCurrency: number | null;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  planTypeId: number;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  fundsTransferComment?: string;\n  fundsTransferReceiverOrTransfer?: string;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/login`;\nconst PASSWORD_EXPIRED_URL = `${BASE_WELCOME_URL}/renew-password`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nenum MaxPlanName {\n  Normal = 'רגילה',\n  ImmediateCharge = 'חיוב עסקות מיידי',\n  InternetShopping = 'אינטרנט/חו\"ל',\n  Installments = 'תשלומים',\n  MonthlyCharge = 'חיוב חודשי',\n  OneMonthPostponed = 'דחוי חודש',\n  MonthlyPostponed = 'דחוי לחיוב החודשי',\n  MonthlyPayment = 'תשלום חודשי',\n  FuturePurchaseFinancing = 'מימון לרכישה עתידית',\n  MonthlyPostponedInstallments = 'דחוי חודש תשלומים',\n  ThirtyDaysPlus = 'עסקת 30 פלוס',\n  TwoMonthsPostponed = 'דחוי חודשיים',\n  TwoMonthsPostponed2 = \"דחוי 2 ח' תשלומים\",\n  MonthlyChargePlusInterest = 'חודשי + ריבית',\n  Credit = 'קרדיט',\n  CreditOutsideTheLimit = 'קרדיט-מחוץ למסגרת',\n  AccumulatingBasket = 'סל מצטבר',\n  PostponedTransactionInstallments = 'פריסת העסקה הדחויה',\n  ReplacementCard = 'כרטיס חליפי',\n  EarlyRepayment = 'פרעון מוקדם',\n  MonthlyCardFee = 'דמי כרטיס',\n  CurrencyPocket = 'חיוב ארנק מטח',\n}\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n   * url explanation:\n   * userIndex: -1 for all account owners\n   * cardIndex: -1 for all cards under the account\n   * all other query params are static, beside the date which changes for request per month\n   */\n  const url = new URL(`${BASE_API_ACTIONS_URL}/api/registered/transactionDetails/getTransactionsAndGraphs`);\n  url.searchParams.set(\n    'filterData',\n    `{\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}`,\n  );\n  url.searchParams.set('firstCallCardIndex', '-1');\n  return url.toString();\n}\n\ninterface FetchCategoryResult {\n  result?: Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n    res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(planName: string, planTypeId: number) {\n  const cleanedUpTxnTypeStr = planName.replace('\\t', ' ').trim() as MaxPlanName;\n  switch (cleanedUpTxnTypeStr) {\n    case MaxPlanName.ImmediateCharge:\n    case MaxPlanName.Normal:\n    case MaxPlanName.MonthlyCharge:\n    case MaxPlanName.OneMonthPostponed:\n    case MaxPlanName.MonthlyPostponed:\n    case MaxPlanName.FuturePurchaseFinancing:\n    case MaxPlanName.MonthlyPayment:\n    case MaxPlanName.MonthlyPostponedInstallments:\n    case MaxPlanName.ThirtyDaysPlus:\n    case MaxPlanName.TwoMonthsPostponed:\n    case MaxPlanName.TwoMonthsPostponed2:\n    case MaxPlanName.AccumulatingBasket:\n    case MaxPlanName.InternetShopping:\n    case MaxPlanName.MonthlyChargePlusInterest:\n    case MaxPlanName.PostponedTransactionInstallments:\n    case MaxPlanName.ReplacementCard:\n    case MaxPlanName.EarlyRepayment:\n    case MaxPlanName.MonthlyCardFee:\n    case MaxPlanName.CurrencyPocket:\n      return TransactionTypes.Normal;\n    case MaxPlanName.Installments:\n    case MaxPlanName.Credit:\n    case MaxPlanName.CreditOutsideTheLimit:\n      return TransactionTypes.Installments;\n    default:\n      switch (planTypeId) {\n        case 2:\n        case 3:\n          return TransactionTypes.Installments;\n        case 5:\n          return TransactionTypes.Normal;\n        default:\n          throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr as string}`);\n      }\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\n\nfunction getChargedCurrency(currencyId: number | null) {\n  switch (currencyId) {\n    case 376:\n      return SHEKEL_CURRENCY;\n    case 840:\n      return DOLLAR_CURRENCY;\n    case 978:\n      return EURO_CURRENCY;\n    default:\n      return undefined;\n  }\n}\n\nexport function getMemo({\n  comments,\n  fundsTransferReceiverOrTransfer,\n  fundsTransferComment,\n}: Pick<ScrapedTransaction, 'comments' | 'fundsTransferReceiverOrTransfer' | 'fundsTransferComment'>) {\n  if (fundsTransferReceiverOrTransfer) {\n    const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;\n    return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;\n  }\n\n  return comments;\n}\n\nfunction mapTransaction(rawTransaction: ScrapedTransaction, options?: ScraperOptions): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments\n    ? `${rawTransaction.dealData?.arn}_${installments.number}`\n    : rawTransaction.dealData?.arn;\n\n  const result: Transaction = {\n    type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),\n    description: rawTransaction.merchantName.trim(),\n    memo: getMemo(rawTransaction),\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n\n  if (options?.includeRawTransaction) {\n    result.rawTransaction = rawTransaction;\n  }\n\n  return result;\n}\ninterface ScrapedTransactionsResult {\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment, options?: ScraperOptions) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter(transaction => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction, options);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach(accountNumber => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(\n  txns: Transaction[],\n  startMoment: moment.Moment,\n  combineInstallments: boolean,\n  enableTransactionsFilterByDate: boolean,\n) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = enableTransactionsFilterByDate\n    ? filterOldTransactions(clonedTxns, startMoment, combineInstallments || false)\n    : clonedTxns;\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startMomentLimit = moment().subtract(4, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(startMomentLimit, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i], options);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach(accountNumber => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(\n      txns,\n      startMoment,\n      options.combineInstallments || false,\n      options.outputData?.enableTransactionsFilterByDate ?? true,\n    );\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [\n    async () => {\n      return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n    },\n  ];\n  urls[LoginResults.UnknownError] = [\n    async () => {\n      return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n    },\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MaxScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'app-user-login-form .general-button.send-me-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        if (await elementPresentOnPage(this.page, '.login-link#private')) {\n          await clickButton(this.page, '.login-link#private');\n        }\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n      waitUntil: 'domcontentloaded',\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map(accountNumber => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"],"mappings":"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","ignoreList":[]}
@@ -119,11 +119,11 @@ function createHeadersFromRequest(request) {
119
119
  'Content-Type': request.headers()['content-type']
120
120
  };
121
121
  }
122
- async function convertTransactions(txns, getMoreDetails, pendingIfTodayTransaction = false) {
122
+ async function convertTransactions(txns, getMoreDetails, pendingIfTodayTransaction = false, options) {
123
123
  return Promise.all(txns.map(async row => {
124
124
  const moreDetails = await getMoreDetails(row);
125
125
  const txnDate = (0, _moment.default)(row.MC02PeulaTaaEZ, _moment.default.HTML5_FMT.DATETIME_LOCAL_SECONDS).toISOString();
126
- return {
126
+ const result = {
127
127
  type: _transactions.TransactionTypes.Normal,
128
128
  identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,
129
129
  date: txnDate,
@@ -135,6 +135,10 @@ async function convertTransactions(txns, getMoreDetails, pendingIfTodayTransacti
135
135
  memo: moreDetails?.memo,
136
136
  status: pendingIfTodayTransaction && row.IsTodayTransaction ? _transactions.TransactionStatuses.Pending : _transactions.TransactionStatuses.Completed
137
137
  };
138
+ if (options?.includeRawTransaction) {
139
+ result.rawTransaction = row;
140
+ }
141
+ return result;
138
142
  }));
139
143
  }
140
144
  async function extractPendingTransactions(page) {
@@ -233,7 +237,7 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
233
237
  const oshTxn = await convertTransactions(relevantRows, this.options.additionalTransactionInformation ? row => getExtraTransactionDetails(this.page, row, apiHeaders) : () => Promise.resolve({
234
238
  entries: {},
235
239
  memo: undefined
236
- }), this.options.optInFeatures?.includes('mizrahi:pendingIfTodayTransaction'));
240
+ }), this.options.optInFeatures?.includes('mizrahi:pendingIfTodayTransaction'), this.options);
237
241
  oshTxn.filter(txn => this.shouldMarkAsPending(txn)).forEach(txn => {
238
242
  txn.status = _transactions.TransactionStatuses.Pending;
239
243
  });
@@ -262,4 +266,4 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
262
266
  }
263
267
  }
264
268
  var _default = exports.default = MizrahiScraper;
265
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_fetch","_navigation","_transactions","_baseScraperWithBrowser","_errors","_debug","e","__esModule","default","debug","getDebug","BASE_WEBSITE_URL","LOGIN_URL","BASE_APP_URL","AFTER_LOGIN_BASE_URL","OSH_PAGE","TRANSACTIONS_PAGE","TRANSACTIONS_REQUEST_URLS","PENDING_TRANSACTIONS_PAGE","PENDING_TRANSACTIONS_IFRAME","MORE_DETAILS_URL","CHANGE_PASSWORD_URL","DATE_FORMAT","MAX_ROWS_PER_REQUEST","usernameSelector","passwordSelector","submitButtonSelector","invalidPasswordSelector","afterLoginSelector","loginSpinnerSelector","accountDropDownItemSelector","pendingTrxIdentifierId","checkingAccountTabHebrewName","checkingAccountTabEnglishName","genericDescriptions","createLoginFields","credentials","selector","value","username","password","isLoggedIn","options","page","oshXPath","oshTab","$$","length","getPossibleLoginResults","LoginResults","Success","InvalidPassword","$","ChangePassword","getStartMoment","optionsStartDate","defaultStartMoment","moment","subtract","startDate","toDate","max","getExtraTransactionDetails","item","apiHeaders","MC02ShowDetailsEZ","tarPeula","MC02PeulaTaaEZ","tarErech","MC02ErehTaaEZ","params","inKodGorem","MC02KodGoremEZ","inAsmachta","MC02AsmahtaMekoritEZ","inSchum","MC02SchumEZ","inNakvanit","inSugTnua","MC02SugTnuaKaspitEZ","inAgid","MC02AgidEZ","inTarPeulaFormatted","format","inTarErechFormatted","year","inKodNose","MC02SeifMaralEZ","inKodTatNose","MC02NoseMaralEZ","inTransactionNumber","TransactionNumber","response","fetchPostWithinPage","details","body","fields","Records","Fields","Array","isArray","entries","map","record","Label","trim","Value","Object","fromEntries","memo","filter","label","some","key","startsWith","join","error","undefined","createDataFromRequest","request","data","JSON","parse","postData","inFromDate","inToDate","table","maxRow","createHeadersFromRequest","mizrahixsrftoken","headers","convertTransactions","txns","getMoreDetails","pendingIfTodayTransaction","Promise","all","row","moreDetails","txnDate","HTML5_FMT","DATETIME_LOCAL_SECONDS","toISOString","type","TransactionTypes","Normal","identifier","parseInt","date","processedDate","originalAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","description","MC02TnuaTeurEZ","status","IsTodayTransaction","TransactionStatuses","Pending","Completed","extractPendingTransactions","pendingTxn","pageEvalAll","trs","tr","from","querySelectorAll","td","textContent","dateStr","incomeAmountStr","amountStr","amount","parseFloat","replaceAll","txn","postLogin","race","waitUntilElementFound","waitForUrl","MizrahiScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","checkReadiness","waitUntilElementDisappear","postAction","possibleResults","fetchData","$eval","el","click","numOfAccounts","results","i","push","fetchAccount","success","accounts","errorType","ScraperErrorTypes","Generic","errorMessage","message","getPendingTransactions","frame","waitUntilIframeFound","f","url","includes","isPending","then","catch","waitForSelector","accountNumberElement","accountNumberHandle","getProperty","accountNumber","jsonValue","Error","any","waitForRequest","header","messages","text","relevantRows","rows","RecTypeSpecified","oshTxn","additionalTransactionInformation","resolve","optInFeatures","shouldMarkAsPending","forEach","startMoment","oshTxnAfterStartDate","isSameOrAfter","allTxn","concat","balance","Yitra","_default","exports"],"sources":["../../src/scrapers/mizrahi.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Frame, type HTTPRequest, type Page } from 'puppeteer-core';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  pageEvalAll,\n  waitUntilElementDisappear,\n  waitUntilElementFound,\n  waitUntilIframeFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForUrl } from '../helpers/navigation';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\nimport { getDebug } from '../helpers/debug';\n\nconst debug = getDebug('mizrahi');\n\ninterface ScrapedTransaction {\n  RecTypeSpecified: boolean;\n  MC02PeulaTaaEZ: string;\n  MC02SchumEZ: number;\n  MC02AsmahtaMekoritEZ: string;\n  MC02TnuaTeurEZ: string;\n  IsTodayTransaction: boolean;\n  MC02ErehTaaEZ: string;\n  MC02ShowDetailsEZ?: string;\n  MC02KodGoremEZ: any;\n  MC02SugTnuaKaspitEZ: any;\n  MC02AgidEZ: any;\n  MC02SeifMaralEZ: any;\n  MC02NoseMaralEZ: any;\n  TransactionNumber: any;\n}\n\ninterface ScrapedTransactionsResult {\n  header: {\n    success: boolean;\n    messages: { text: string }[];\n  };\n  body: {\n    fields: {\n      Yitra: string;\n    };\n    table: {\n      rows: ScrapedTransaction[];\n    };\n  };\n}\n\ntype MoreDetailsResponse = {\n  body: {\n    fields: [\n      [\n        {\n          Records: [\n            {\n              Fields: Array<{\n                Label: string;\n                Value: string;\n              }>;\n            },\n          ];\n        },\n      ],\n    ];\n  };\n};\n\ntype MoreDetails = {\n  entries: Record<string, string>;\n  memo: string | undefined;\n};\n\nconst BASE_WEBSITE_URL = 'https://www.mizrahi-tefahot.co.il';\nconst LOGIN_URL = `${BASE_WEBSITE_URL}/login/index.html#/auth-page-he`;\nconst BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';\nconst AFTER_LOGIN_BASE_URL = /https:\\/\\/mto\\.mizrahi-tefahot\\.co\\.il\\/OnlineApp\\/.*/;\nconst OSH_PAGE = '/osh/legacy/legacy-Osh-Main';\nconst TRANSACTIONS_PAGE = '/osh/legacy/root-main-osh-p428New';\nconst TRANSACTIONS_REQUEST_URLS = [\n  `${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`,\n  `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`,\n];\nconst PENDING_TRANSACTIONS_PAGE = '/osh/legacy/legacy-Osh-p420';\nconst PENDING_TRANSACTIONS_IFRAME = 'p420.aspx';\nconst MORE_DETAILS_URL = `${BASE_APP_URL}/Online/api/OSH/getMaherBerurimSMF`;\nconst CHANGE_PASSWORD_URL = /https:\\/\\/www\\.mizrahi-tefahot\\.co\\.il\\/login\\/index\\.html#\\/change-pass/;\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst MAX_ROWS_PER_REQUEST = 10000000000;\n\nconst usernameSelector = '#userNumberDesktopHeb';\nconst passwordSelector = '#passwordDesktopHeb';\nconst submitButtonSelector = 'button.btn.btn-primary';\nconst invalidPasswordSelector = 'a[href*=\"https://sc.mizrahi-tefahot.co.il/SCServices/SC/P010.aspx\"]';\nconst afterLoginSelector = '#dropdownBasic';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\nconst accountDropDownItemSelector = '#AccountPicker .item';\nconst pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';\nconst checkingAccountTabHebrewName = 'עובר ושב';\nconst checkingAccountTabEnglishName = 'Checking Account';\nconst genericDescriptions = ['העברת יומן לבנק זר מסניף זר'];\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: usernameSelector, value: credentials.username },\n    { selector: passwordSelector, value: credentials.password },\n  ];\n}\n\nasync function isLoggedIn(options: { page?: Page | undefined } | undefined) {\n  if (!options?.page) {\n    return false;\n  }\n  const oshXPath = `//a//span[contains(., \"${checkingAccountTabHebrewName}\") or contains(., \"${checkingAccountTabEnglishName}\")]`;\n  const oshTab = await options.page.$$(`xpath${oshXPath}`);\n  return oshTab.length > 0;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  return {\n    [LoginResults.Success]: [AFTER_LOGIN_BASE_URL, isLoggedIn],\n    [LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],\n    [LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL],\n  };\n}\n\nfunction getStartMoment(optionsStartDate: Date) {\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = optionsStartDate || defaultStartMoment.toDate();\n  return moment.max(defaultStartMoment, moment(startDate));\n}\n\nasync function getExtraTransactionDetails(\n  page: Page,\n  item: ScrapedTransaction,\n  apiHeaders: Record<string, string>,\n): Promise<MoreDetails> {\n  try {\n    debug('getExtraTransactionDetails for item:', item);\n    if (item.MC02ShowDetailsEZ === '1') {\n      const tarPeula = moment(item.MC02PeulaTaaEZ);\n      const tarErech = moment(item.MC02ErehTaaEZ);\n\n      const params = {\n        inKodGorem: item.MC02KodGoremEZ,\n        inAsmachta: item.MC02AsmahtaMekoritEZ,\n        inSchum: item.MC02SchumEZ,\n        inNakvanit: item.MC02KodGoremEZ,\n        inSugTnua: item.MC02SugTnuaKaspitEZ,\n        inAgid: item.MC02AgidEZ,\n        inTarPeulaFormatted: tarPeula.format(DATE_FORMAT),\n        inTarErechFormatted: (tarErech.year() > 2000 ? tarErech : tarPeula).format(DATE_FORMAT),\n        inKodNose: item.MC02SeifMaralEZ,\n        inKodTatNose: item.MC02NoseMaralEZ,\n        inTransactionNumber: item.TransactionNumber,\n      };\n\n      const response = await fetchPostWithinPage<MoreDetailsResponse>(page, MORE_DETAILS_URL, params, apiHeaders);\n      const details = response?.body.fields?.[0]?.[0]?.Records?.[0].Fields;\n      debug('fetch details for', params, 'details:', details);\n      if (Array.isArray(details) && details.length > 0) {\n        const entries = details.map(record => [record.Label.trim(), record.Value.trim()]);\n        return {\n          entries: Object.fromEntries(entries),\n          memo: entries\n            .filter(([label]) => ['שם', 'מהות', 'חשבון'].some(key => label.startsWith(key)))\n            .map(([label, value]) => `${label} ${value}`)\n            .join(', '),\n        };\n      }\n    }\n  } catch (error) {\n    debug('Error fetching extra transaction details:', error);\n  }\n\n  return {\n    entries: {},\n    memo: undefined,\n  };\n}\n\nfunction createDataFromRequest(request: HTTPRequest, optionsStartDate: Date) {\n  const data = JSON.parse(request.postData() || '{}');\n\n  data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);\n  data.inToDate = moment().format(DATE_FORMAT);\n  data.table.maxRow = MAX_ROWS_PER_REQUEST;\n\n  return data;\n}\n\nfunction createHeadersFromRequest(request: HTTPRequest) {\n  return {\n    mizrahixsrftoken: request.headers().mizrahixsrftoken,\n    'Content-Type': request.headers()['content-type'],\n  };\n}\n\nasync function convertTransactions(\n  txns: ScrapedTransaction[],\n  getMoreDetails: (row: ScrapedTransaction) => Promise<MoreDetails>,\n  pendingIfTodayTransaction: boolean = false,\n): Promise<Transaction[]> {\n  return Promise.all(\n    txns.map(async row => {\n      const moreDetails = await getMoreDetails(row);\n\n      const txnDate = moment(row.MC02PeulaTaaEZ, moment.HTML5_FMT.DATETIME_LOCAL_SECONDS).toISOString();\n\n      return {\n        type: TransactionTypes.Normal,\n        identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,\n        date: txnDate,\n        processedDate: txnDate,\n        originalAmount: row.MC02SchumEZ,\n        originalCurrency: SHEKEL_CURRENCY,\n        chargedAmount: row.MC02SchumEZ,\n        description: row.MC02TnuaTeurEZ,\n        memo: moreDetails?.memo,\n        status:\n          pendingIfTodayTransaction && row.IsTodayTransaction\n            ? TransactionStatuses.Pending\n            : TransactionStatuses.Completed,\n      };\n    }),\n  );\n}\n\nasync function extractPendingTransactions(page: Frame): Promise<Transaction[]> {\n  const pendingTxn = await pageEvalAll(page, 'tr.rgRow, tr.rgAltRow', [], trs => {\n    return trs.map(tr => Array.from(tr.querySelectorAll('td'), td => td.textContent || ''));\n  });\n\n  return pendingTxn\n    .map(([dateStr, description, incomeAmountStr, amountStr]) => ({\n      date: moment(dateStr, 'DD/MM/YY').toISOString(),\n      amount: parseFloat(amountStr.replaceAll(',', '')),\n      description,\n      incomeAmountStr, // TODO: handle incomeAmountStr once we know the sign of it\n    }))\n    .filter(txn => txn.date)\n    .map(({ date, description, amount }) => ({\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description,\n      status: TransactionStatuses.Pending,\n    }));\n}\n\nasync function postLogin(page: Page) {\n  await Promise.race([\n    waitUntilElementFound(page, afterLoginSelector),\n    waitUntilElementFound(page, invalidPasswordSelector),\n    waitForUrl(page, CHANGE_PASSWORD_URL),\n  ]);\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MizrahiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector,\n      checkReadiness: async () => waitUntilElementDisappear(this.page, loginSpinnerSelector),\n      postAction: async () => postLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n\n    const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;\n\n    try {\n      const results: TransactionsAccount[] = [];\n\n      for (let i = 0; i < numOfAccounts; i += 1) {\n        if (i > 0) {\n          await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n        }\n\n        await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, el => (el as HTMLElement).click());\n        results.push(await this.fetchAccount());\n      }\n\n      return {\n        success: true,\n        accounts: results,\n      };\n    } catch (e) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage: (e as Error).message,\n      };\n    }\n  }\n\n  private async getPendingTransactions(): Promise<Transaction[]> {\n    await this.page.$eval(`a[href*=\"${PENDING_TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n    const frame = await waitUntilIframeFound(this.page, f => f.url().includes(PENDING_TRANSACTIONS_IFRAME));\n    const isPending = await waitUntilElementFound(frame, pendingTrxIdentifierId)\n      .then(() => true)\n      .catch(() => false);\n    if (!isPending) {\n      return [];\n    }\n\n    const pendingTxn = await extractPendingTransactions(frame);\n    return pendingTxn;\n  }\n\n  private async fetchAccount() {\n    await this.page.waitForSelector(`a[href*=\"${OSH_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${OSH_PAGE}\"]`, el => (el as HTMLElement).click());\n    await waitUntilElementFound(this.page, `a[href*=\"${TRANSACTIONS_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n\n    const accountNumberElement = await this.page.$('#dropdownBasic b span');\n    const accountNumberHandle = await accountNumberElement?.getProperty('title');\n    const accountNumber = (await accountNumberHandle?.jsonValue()) as string;\n    if (!accountNumber) {\n      throw new Error('Account number not found');\n    }\n\n    const [response, apiHeaders] = await Promise.any(\n      TRANSACTIONS_REQUEST_URLS.map(async url => {\n        const request = await this.page.waitForRequest(url);\n        const data = createDataFromRequest(request, this.options.startDate);\n        const headers = createHeadersFromRequest(request);\n\n        return [await fetchPostWithinPage<ScrapedTransactionsResult>(this.page, url, data, headers), headers] as const;\n      }),\n    );\n\n    if (!response || response.header.success === false) {\n      throw new Error(\n        `Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`,\n      );\n    }\n\n    const relevantRows = response.body.table.rows.filter(row => row.RecTypeSpecified);\n    const oshTxn = await convertTransactions(\n      relevantRows,\n      this.options.additionalTransactionInformation\n        ? row => getExtraTransactionDetails(this.page, row, apiHeaders)\n        : () => Promise.resolve({ entries: {}, memo: undefined }),\n      this.options.optInFeatures?.includes('mizrahi:pendingIfTodayTransaction'),\n    );\n\n    oshTxn\n      .filter(txn => this.shouldMarkAsPending(txn))\n      .forEach(txn => {\n        txn.status = TransactionStatuses.Pending;\n      });\n\n    // workaround for a bug which the bank's API returns transactions before the requested start date\n    const startMoment = getStartMoment(this.options.startDate);\n    const oshTxnAfterStartDate = oshTxn.filter(txn => moment(txn.date).isSameOrAfter(startMoment));\n\n    const pendingTxn = await this.getPendingTransactions();\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      accountNumber,\n      txns: allTxn,\n      balance: +response.body.fields?.Yitra,\n    };\n  }\n\n  private shouldMarkAsPending(txn: Transaction): boolean {\n    if (this.options.optInFeatures?.includes('mizrahi:pendingIfNoIdentifier') && !txn.identifier) {\n      debug(`Marking transaction '${txn.description}' as pending due to no identifier.`);\n      return true;\n    }\n\n    if (\n      this.options.optInFeatures?.includes('mizrahi:pendingIfHasGenericDescription') &&\n      genericDescriptions.includes(txn.description)\n    ) {\n      debug(`Marking transaction '${txn.description}' as pending due to generic description.`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nexport default MizrahiScraper;\n"],"mappings":"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","ignoreList":[]}
269
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_fetch","_navigation","_transactions","_baseScraperWithBrowser","_errors","_debug","e","__esModule","default","debug","getDebug","BASE_WEBSITE_URL","LOGIN_URL","BASE_APP_URL","AFTER_LOGIN_BASE_URL","OSH_PAGE","TRANSACTIONS_PAGE","TRANSACTIONS_REQUEST_URLS","PENDING_TRANSACTIONS_PAGE","PENDING_TRANSACTIONS_IFRAME","MORE_DETAILS_URL","CHANGE_PASSWORD_URL","DATE_FORMAT","MAX_ROWS_PER_REQUEST","usernameSelector","passwordSelector","submitButtonSelector","invalidPasswordSelector","afterLoginSelector","loginSpinnerSelector","accountDropDownItemSelector","pendingTrxIdentifierId","checkingAccountTabHebrewName","checkingAccountTabEnglishName","genericDescriptions","createLoginFields","credentials","selector","value","username","password","isLoggedIn","options","page","oshXPath","oshTab","$$","length","getPossibleLoginResults","LoginResults","Success","InvalidPassword","$","ChangePassword","getStartMoment","optionsStartDate","defaultStartMoment","moment","subtract","startDate","toDate","max","getExtraTransactionDetails","item","apiHeaders","MC02ShowDetailsEZ","tarPeula","MC02PeulaTaaEZ","tarErech","MC02ErehTaaEZ","params","inKodGorem","MC02KodGoremEZ","inAsmachta","MC02AsmahtaMekoritEZ","inSchum","MC02SchumEZ","inNakvanit","inSugTnua","MC02SugTnuaKaspitEZ","inAgid","MC02AgidEZ","inTarPeulaFormatted","format","inTarErechFormatted","year","inKodNose","MC02SeifMaralEZ","inKodTatNose","MC02NoseMaralEZ","inTransactionNumber","TransactionNumber","response","fetchPostWithinPage","details","body","fields","Records","Fields","Array","isArray","entries","map","record","Label","trim","Value","Object","fromEntries","memo","filter","label","some","key","startsWith","join","error","undefined","createDataFromRequest","request","data","JSON","parse","postData","inFromDate","inToDate","table","maxRow","createHeadersFromRequest","mizrahixsrftoken","headers","convertTransactions","txns","getMoreDetails","pendingIfTodayTransaction","Promise","all","row","moreDetails","txnDate","HTML5_FMT","DATETIME_LOCAL_SECONDS","toISOString","result","type","TransactionTypes","Normal","identifier","parseInt","date","processedDate","originalAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","description","MC02TnuaTeurEZ","status","IsTodayTransaction","TransactionStatuses","Pending","Completed","includeRawTransaction","rawTransaction","extractPendingTransactions","pendingTxn","pageEvalAll","trs","tr","from","querySelectorAll","td","textContent","dateStr","incomeAmountStr","amountStr","amount","parseFloat","replaceAll","txn","postLogin","race","waitUntilElementFound","waitForUrl","MizrahiScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","checkReadiness","waitUntilElementDisappear","postAction","possibleResults","fetchData","$eval","el","click","numOfAccounts","results","i","push","fetchAccount","success","accounts","errorType","ScraperErrorTypes","Generic","errorMessage","message","getPendingTransactions","frame","waitUntilIframeFound","f","url","includes","isPending","then","catch","waitForSelector","accountNumberElement","accountNumberHandle","getProperty","accountNumber","jsonValue","Error","any","waitForRequest","header","messages","text","relevantRows","rows","RecTypeSpecified","oshTxn","additionalTransactionInformation","resolve","optInFeatures","shouldMarkAsPending","forEach","startMoment","oshTxnAfterStartDate","isSameOrAfter","allTxn","concat","balance","Yitra","_default","exports"],"sources":["../../src/scrapers/mizrahi.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Frame, type HTTPRequest, type Page } from 'puppeteer-core';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  pageEvalAll,\n  waitUntilElementDisappear,\n  waitUntilElementFound,\n  waitUntilIframeFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForUrl } from '../helpers/navigation';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\nimport { getDebug } from '../helpers/debug';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('mizrahi');\n\ninterface ScrapedTransaction {\n  RecTypeSpecified: boolean;\n  MC02PeulaTaaEZ: string;\n  MC02SchumEZ: number;\n  MC02AsmahtaMekoritEZ: string;\n  MC02TnuaTeurEZ: string;\n  IsTodayTransaction: boolean;\n  MC02ErehTaaEZ: string;\n  MC02ShowDetailsEZ?: string;\n  MC02KodGoremEZ: any;\n  MC02SugTnuaKaspitEZ: any;\n  MC02AgidEZ: any;\n  MC02SeifMaralEZ: any;\n  MC02NoseMaralEZ: any;\n  TransactionNumber: any;\n}\n\ninterface ScrapedTransactionsResult {\n  header: {\n    success: boolean;\n    messages: { text: string }[];\n  };\n  body: {\n    fields: {\n      Yitra: string;\n    };\n    table: {\n      rows: ScrapedTransaction[];\n    };\n  };\n}\n\ntype MoreDetailsResponse = {\n  body: {\n    fields: [\n      [\n        {\n          Records: [\n            {\n              Fields: Array<{\n                Label: string;\n                Value: string;\n              }>;\n            },\n          ];\n        },\n      ],\n    ];\n  };\n};\n\ntype MoreDetails = {\n  entries: Record<string, string>;\n  memo: string | undefined;\n};\n\nconst BASE_WEBSITE_URL = 'https://www.mizrahi-tefahot.co.il';\nconst LOGIN_URL = `${BASE_WEBSITE_URL}/login/index.html#/auth-page-he`;\nconst BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';\nconst AFTER_LOGIN_BASE_URL = /https:\\/\\/mto\\.mizrahi-tefahot\\.co\\.il\\/OnlineApp\\/.*/;\nconst OSH_PAGE = '/osh/legacy/legacy-Osh-Main';\nconst TRANSACTIONS_PAGE = '/osh/legacy/root-main-osh-p428New';\nconst TRANSACTIONS_REQUEST_URLS = [\n  `${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`,\n  `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`,\n];\nconst PENDING_TRANSACTIONS_PAGE = '/osh/legacy/legacy-Osh-p420';\nconst PENDING_TRANSACTIONS_IFRAME = 'p420.aspx';\nconst MORE_DETAILS_URL = `${BASE_APP_URL}/Online/api/OSH/getMaherBerurimSMF`;\nconst CHANGE_PASSWORD_URL = /https:\\/\\/www\\.mizrahi-tefahot\\.co\\.il\\/login\\/index\\.html#\\/change-pass/;\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst MAX_ROWS_PER_REQUEST = 10000000000;\n\nconst usernameSelector = '#userNumberDesktopHeb';\nconst passwordSelector = '#passwordDesktopHeb';\nconst submitButtonSelector = 'button.btn.btn-primary';\nconst invalidPasswordSelector = 'a[href*=\"https://sc.mizrahi-tefahot.co.il/SCServices/SC/P010.aspx\"]';\nconst afterLoginSelector = '#dropdownBasic';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\nconst accountDropDownItemSelector = '#AccountPicker .item';\nconst pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';\nconst checkingAccountTabHebrewName = 'עובר ושב';\nconst checkingAccountTabEnglishName = 'Checking Account';\nconst genericDescriptions = ['העברת יומן לבנק זר מסניף זר'];\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: usernameSelector, value: credentials.username },\n    { selector: passwordSelector, value: credentials.password },\n  ];\n}\n\nasync function isLoggedIn(options: { page?: Page | undefined } | undefined) {\n  if (!options?.page) {\n    return false;\n  }\n  const oshXPath = `//a//span[contains(., \"${checkingAccountTabHebrewName}\") or contains(., \"${checkingAccountTabEnglishName}\")]`;\n  const oshTab = await options.page.$$(`xpath${oshXPath}`);\n  return oshTab.length > 0;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  return {\n    [LoginResults.Success]: [AFTER_LOGIN_BASE_URL, isLoggedIn],\n    [LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],\n    [LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL],\n  };\n}\n\nfunction getStartMoment(optionsStartDate: Date) {\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = optionsStartDate || defaultStartMoment.toDate();\n  return moment.max(defaultStartMoment, moment(startDate));\n}\n\nasync function getExtraTransactionDetails(\n  page: Page,\n  item: ScrapedTransaction,\n  apiHeaders: Record<string, string>,\n): Promise<MoreDetails> {\n  try {\n    debug('getExtraTransactionDetails for item:', item);\n    if (item.MC02ShowDetailsEZ === '1') {\n      const tarPeula = moment(item.MC02PeulaTaaEZ);\n      const tarErech = moment(item.MC02ErehTaaEZ);\n\n      const params = {\n        inKodGorem: item.MC02KodGoremEZ,\n        inAsmachta: item.MC02AsmahtaMekoritEZ,\n        inSchum: item.MC02SchumEZ,\n        inNakvanit: item.MC02KodGoremEZ,\n        inSugTnua: item.MC02SugTnuaKaspitEZ,\n        inAgid: item.MC02AgidEZ,\n        inTarPeulaFormatted: tarPeula.format(DATE_FORMAT),\n        inTarErechFormatted: (tarErech.year() > 2000 ? tarErech : tarPeula).format(DATE_FORMAT),\n        inKodNose: item.MC02SeifMaralEZ,\n        inKodTatNose: item.MC02NoseMaralEZ,\n        inTransactionNumber: item.TransactionNumber,\n      };\n\n      const response = await fetchPostWithinPage<MoreDetailsResponse>(page, MORE_DETAILS_URL, params, apiHeaders);\n      const details = response?.body.fields?.[0]?.[0]?.Records?.[0].Fields;\n      debug('fetch details for', params, 'details:', details);\n      if (Array.isArray(details) && details.length > 0) {\n        const entries = details.map(record => [record.Label.trim(), record.Value.trim()]);\n        return {\n          entries: Object.fromEntries(entries),\n          memo: entries\n            .filter(([label]) => ['שם', 'מהות', 'חשבון'].some(key => label.startsWith(key)))\n            .map(([label, value]) => `${label} ${value}`)\n            .join(', '),\n        };\n      }\n    }\n  } catch (error) {\n    debug('Error fetching extra transaction details:', error);\n  }\n\n  return {\n    entries: {},\n    memo: undefined,\n  };\n}\n\nfunction createDataFromRequest(request: HTTPRequest, optionsStartDate: Date) {\n  const data = JSON.parse(request.postData() || '{}');\n\n  data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);\n  data.inToDate = moment().format(DATE_FORMAT);\n  data.table.maxRow = MAX_ROWS_PER_REQUEST;\n\n  return data;\n}\n\nfunction createHeadersFromRequest(request: HTTPRequest) {\n  return {\n    mizrahixsrftoken: request.headers().mizrahixsrftoken,\n    'Content-Type': request.headers()['content-type'],\n  };\n}\n\nasync function convertTransactions(\n  txns: ScrapedTransaction[],\n  getMoreDetails: (row: ScrapedTransaction) => Promise<MoreDetails>,\n  pendingIfTodayTransaction: boolean = false,\n  options?: ScraperOptions,\n): Promise<Transaction[]> {\n  return Promise.all(\n    txns.map(async row => {\n      const moreDetails = await getMoreDetails(row);\n\n      const txnDate = moment(row.MC02PeulaTaaEZ, moment.HTML5_FMT.DATETIME_LOCAL_SECONDS).toISOString();\n\n      const result: Transaction = {\n        type: TransactionTypes.Normal,\n        identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,\n        date: txnDate,\n        processedDate: txnDate,\n        originalAmount: row.MC02SchumEZ,\n        originalCurrency: SHEKEL_CURRENCY,\n        chargedAmount: row.MC02SchumEZ,\n        description: row.MC02TnuaTeurEZ,\n        memo: moreDetails?.memo,\n        status:\n          pendingIfTodayTransaction && row.IsTodayTransaction\n            ? TransactionStatuses.Pending\n            : TransactionStatuses.Completed,\n      };\n\n      if (options?.includeRawTransaction) {\n        result.rawTransaction = row;\n      }\n\n      return result;\n    }),\n  );\n}\n\nasync function extractPendingTransactions(page: Frame): Promise<Transaction[]> {\n  const pendingTxn = await pageEvalAll(page, 'tr.rgRow, tr.rgAltRow', [], trs => {\n    return trs.map(tr => Array.from(tr.querySelectorAll('td'), td => td.textContent || ''));\n  });\n\n  return pendingTxn\n    .map(([dateStr, description, incomeAmountStr, amountStr]) => ({\n      date: moment(dateStr, 'DD/MM/YY').toISOString(),\n      amount: parseFloat(amountStr.replaceAll(',', '')),\n      description,\n      incomeAmountStr, // TODO: handle incomeAmountStr once we know the sign of it\n    }))\n    .filter(txn => txn.date)\n    .map(({ date, description, amount }) => ({\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description,\n      status: TransactionStatuses.Pending,\n    }));\n}\n\nasync function postLogin(page: Page) {\n  await Promise.race([\n    waitUntilElementFound(page, afterLoginSelector),\n    waitUntilElementFound(page, invalidPasswordSelector),\n    waitForUrl(page, CHANGE_PASSWORD_URL),\n  ]);\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MizrahiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector,\n      checkReadiness: async () => waitUntilElementDisappear(this.page, loginSpinnerSelector),\n      postAction: async () => postLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n\n    const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;\n\n    try {\n      const results: TransactionsAccount[] = [];\n\n      for (let i = 0; i < numOfAccounts; i += 1) {\n        if (i > 0) {\n          await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n        }\n\n        await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, el => (el as HTMLElement).click());\n        results.push(await this.fetchAccount());\n      }\n\n      return {\n        success: true,\n        accounts: results,\n      };\n    } catch (e) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage: (e as Error).message,\n      };\n    }\n  }\n\n  private async getPendingTransactions(): Promise<Transaction[]> {\n    await this.page.$eval(`a[href*=\"${PENDING_TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n    const frame = await waitUntilIframeFound(this.page, f => f.url().includes(PENDING_TRANSACTIONS_IFRAME));\n    const isPending = await waitUntilElementFound(frame, pendingTrxIdentifierId)\n      .then(() => true)\n      .catch(() => false);\n    if (!isPending) {\n      return [];\n    }\n\n    const pendingTxn = await extractPendingTransactions(frame);\n    return pendingTxn;\n  }\n\n  private async fetchAccount() {\n    await this.page.waitForSelector(`a[href*=\"${OSH_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${OSH_PAGE}\"]`, el => (el as HTMLElement).click());\n    await waitUntilElementFound(this.page, `a[href*=\"${TRANSACTIONS_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n\n    const accountNumberElement = await this.page.$('#dropdownBasic b span');\n    const accountNumberHandle = await accountNumberElement?.getProperty('title');\n    const accountNumber = (await accountNumberHandle?.jsonValue()) as string;\n    if (!accountNumber) {\n      throw new Error('Account number not found');\n    }\n\n    const [response, apiHeaders] = await Promise.any(\n      TRANSACTIONS_REQUEST_URLS.map(async url => {\n        const request = await this.page.waitForRequest(url);\n        const data = createDataFromRequest(request, this.options.startDate);\n        const headers = createHeadersFromRequest(request);\n\n        return [await fetchPostWithinPage<ScrapedTransactionsResult>(this.page, url, data, headers), headers] as const;\n      }),\n    );\n\n    if (!response || response.header.success === false) {\n      throw new Error(\n        `Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`,\n      );\n    }\n\n    const relevantRows = response.body.table.rows.filter(row => row.RecTypeSpecified);\n    const oshTxn = await convertTransactions(\n      relevantRows,\n      this.options.additionalTransactionInformation\n        ? row => getExtraTransactionDetails(this.page, row, apiHeaders)\n        : () => Promise.resolve({ entries: {}, memo: undefined }),\n      this.options.optInFeatures?.includes('mizrahi:pendingIfTodayTransaction'),\n      this.options,\n    );\n\n    oshTxn\n      .filter(txn => this.shouldMarkAsPending(txn))\n      .forEach(txn => {\n        txn.status = TransactionStatuses.Pending;\n      });\n\n    // workaround for a bug which the bank's API returns transactions before the requested start date\n    const startMoment = getStartMoment(this.options.startDate);\n    const oshTxnAfterStartDate = oshTxn.filter(txn => moment(txn.date).isSameOrAfter(startMoment));\n\n    const pendingTxn = await this.getPendingTransactions();\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      accountNumber,\n      txns: allTxn,\n      balance: +response.body.fields?.Yitra,\n    };\n  }\n\n  private shouldMarkAsPending(txn: Transaction): boolean {\n    if (this.options.optInFeatures?.includes('mizrahi:pendingIfNoIdentifier') && !txn.identifier) {\n      debug(`Marking transaction '${txn.description}' as pending due to no identifier.`);\n      return true;\n    }\n\n    if (\n      this.options.optInFeatures?.includes('mizrahi:pendingIfHasGenericDescription') &&\n      genericDescriptions.includes(txn.description)\n    ) {\n      debug(`Marking transaction '${txn.description}' as pending due to generic description.`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nexport default MizrahiScraper;\n"],"mappings":"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","ignoreList":[]}
@@ -171,7 +171,7 @@ class OneZeroScraper extends _baseScraper.BaseScraper {
171
171
  txns: matchingMovements.map(movement => {
172
172
  const hasInstallments = movement.transaction?.enrichment?.recurrences?.some(x => x.isRecurrent);
173
173
  const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;
174
- return {
174
+ const result = {
175
175
  identifier: movement.movementId,
176
176
  date: movement.valueDate,
177
177
  chargedAmount: +movement.movementAmount * modifier,
@@ -183,6 +183,10 @@ class OneZeroScraper extends _baseScraper.BaseScraper {
183
183
  status: _transactions.TransactionStatuses.Completed,
184
184
  type: hasInstallments ? _transactions.TransactionTypes.Installments : _transactions.TransactionTypes.Normal
185
185
  };
186
+ if (this.options?.includeRawTransaction) {
187
+ result.rawTransaction = movement;
188
+ }
189
+ return result;
186
190
  })
187
191
  };
188
192
  }
@@ -235,4 +239,4 @@ class OneZeroScraper extends _baseScraper.BaseScraper {
235
239
  }
236
240
  }
237
241
  exports.default = OneZeroScraper;
238
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_fetch","_transactions","_baseScraper","_errors","_oneZeroQueries","e","__esModule","default","HEBREW_WORDS_REGEX","debug","getDebug","IDENTITY_SERVER_URL","GRAPHQL_API_URL","OneZeroScraper","BaseScraper","triggerTwoFactorAuth","phoneNumber","startsWith","createGenericError","deviceTokenResponse","fetchPost","extClientId","os","resultData","deviceToken","otpPrepareResponse","factorValue","otpChannel","otpContext","success","getLongTermTwoFactorToken","otpCode","otpVerifyResponse","otpToken","longTermTwoFactorAuthToken","resolveOtpToken","credentials","otpLongTermToken","otpCodeRetriever","errorType","ScraperErrorTypes","TwoFactorRetrieverMissing","errorMessage","triggerResult","otpTokenResult","login","getIdTokenResponse","otpSmsToken","email","pass","password","pinCode","idToken","getSessionTokenResponse","accessToken","persistentOtpToken","fetchPortfolioMovements","portfolio","startDate","account","accounts","cursor","movements","length","Date","movementTimestamp","portfolioNum","newMovements","pagination","fetchGraphql","GET_MOVEMENTS","portfolioId","accountId","language","limit","authorization","unshift","hasMore","sort","x","y","valueOf","matchingMovements","filter","movement","accountNumber","balance","parseFloat","runningBalance","txns","map","hasInstallments","transaction","enrichment","recurrences","some","isRecurrent","modifier","creditDebit","identifier","movementId","date","valueDate","chargedAmount","movementAmount","chargedCurrency","movementCurrency","originalAmount","originalCurrency","description","sanitizeHebrew","processedDate","status","TransactionStatuses","Completed","type","TransactionTypes","Installments","Normal","text","includes","trim","plainString","replace","hebrewSubStringsRanges","matchAll","rangesToReverse","str","start","index","end","out","push","substring","reversed","reverse","join","fetchData","defaultStartMoment","moment","subtract","add","options","toDate","startMoment","max","result","GET_CUSTOMER","portfolios","customer","flatMap","Promise","all","exports"],"sources":["../../src/scrapers/one-zero.ts"],"sourcesContent":["import moment from 'moment/moment';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGraphql, fetchPost } from '../helpers/fetch';\nimport {\n  type Transaction as ScrapingTransaction,\n  TransactionStatuses,\n  TransactionTypes,\n  type TransactionsAccount,\n} from '../transactions';\nimport { BaseScraper } from './base-scraper';\nimport { ScraperErrorTypes, createGenericError } from './errors';\nimport {\n  type ScraperGetLongTermTwoFactorTokenResult,\n  type ScraperLoginResult,\n  type ScraperScrapingResult,\n  type ScraperTwoFactorAuthTriggerResult,\n} from './interface';\nimport { GET_CUSTOMER, GET_MOVEMENTS } from './one-zero-queries';\n\nconst HEBREW_WORDS_REGEX = /[\\u0590-\\u05FF][\\u0590-\\u05FF\"'\\-_ /\\\\]*[\\u0590-\\u05FF]/g;\n\nconst debug = getDebug('one-zero');\n\ntype Account = {\n  accountId: string;\n};\n\ntype Portfolio = {\n  accounts: Array<Account>;\n  portfolioId: string;\n  portfolioNum: string;\n};\n\ntype Customer = {\n  customerId: string;\n  portfolios?: Array<Portfolio> | null;\n};\n\nexport type Category = {\n  categoryId: number;\n  dataSource: string;\n  subCategoryId?: number | null;\n};\n\nexport type Recurrence = {\n  dataSource: string;\n  isRecurrent: boolean;\n};\n\ntype TransactionEnrichment = {\n  categories?: Category[] | null;\n  recurrences?: Recurrence[] | null;\n};\n\ntype Transaction = {\n  enrichment?: TransactionEnrichment | null;\n  // TODO: Get installments information here\n  // transactionDetails: TransactionDetails;\n};\n\ntype Movement = {\n  accountId: string;\n  bankCurrencyAmount: string;\n  bookingDate: string;\n  conversionRate: string;\n  creditDebit: string;\n  description: string;\n  isReversed: boolean;\n  movementAmount: string;\n  movementCurrency: string;\n  movementId: string;\n  movementReversedId?: string | null;\n  movementTimestamp: string;\n  movementType: string;\n  portfolioId: string;\n  runningBalance: string;\n  transaction?: Transaction | null;\n  valueDate: string;\n};\n\ntype QueryPagination = { hasMore: boolean; cursor: string };\n\nconst IDENTITY_SERVER_URL = 'https://identity.tfd-bank.com/v1/';\n\nconst GRAPHQL_API_URL = 'https://mobile.tfd-bank.com/mobile-graph/graphql';\n\ntype ScraperSpecificCredentials = { email: string; password: string } & (\n  | {\n      otpCodeRetriever: () => Promise<string>;\n      phoneNumber: string;\n    }\n  | {\n      otpLongTermToken: string;\n    }\n);\n\nexport default class OneZeroScraper extends BaseScraper<ScraperSpecificCredentials> {\n  private otpContext?: string;\n\n  private accessToken?: string;\n\n  async triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult> {\n    if (!phoneNumber.startsWith('+')) {\n      return createGenericError(\n        'A full international phone number starting with + and a three digit country code is required',\n      );\n    }\n\n    debug('Fetching device token');\n    const deviceTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/devices/token`, {\n      extClientId: 'mobile',\n      os: 'Android',\n    });\n\n    const {\n      resultData: { deviceToken },\n    } = deviceTokenResponse;\n\n    debug(`Sending OTP to phone number ${phoneNumber}`);\n\n    const otpPrepareResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/prepare`, {\n      factorValue: phoneNumber,\n      deviceToken,\n      otpChannel: 'SMS_OTP',\n    });\n\n    const {\n      resultData: { otpContext },\n    } = otpPrepareResponse;\n\n    this.otpContext = otpContext;\n\n    return {\n      success: true,\n    };\n  }\n\n  public async getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if (!this.otpContext) {\n      return createGenericError('triggerOtp was not called before calling getPermenantOtpToken()');\n    }\n\n    debug('Requesting OTP token');\n    const otpVerifyResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/verify`, {\n      otpContext: this.otpContext,\n      otpCode,\n    });\n\n    const {\n      resultData: { otpToken },\n    } = otpVerifyResponse;\n    return { success: true, longTermTwoFactorAuthToken: otpToken };\n  }\n\n  private async resolveOtpToken(\n    credentials: ScraperSpecificCredentials,\n  ): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if ('otpLongTermToken' in credentials) {\n      if (!credentials.otpLongTermToken) {\n        return createGenericError('Invalid otpLongTermToken');\n      }\n      return { success: true, longTermTwoFactorAuthToken: credentials.otpLongTermToken };\n    }\n\n    if (!credentials.otpCodeRetriever) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.TwoFactorRetrieverMissing,\n        errorMessage: 'otpCodeRetriever is required when otpPermanentToken is not provided',\n      };\n    }\n\n    if (!credentials.phoneNumber) {\n      return createGenericError('phoneNumber is required when providing a otpCodeRetriever callback');\n    }\n\n    debug('Triggering user supplied otpCodeRetriever callback');\n    const triggerResult = await this.triggerTwoFactorAuth(credentials.phoneNumber);\n\n    if (!triggerResult.success) {\n      return triggerResult;\n    }\n\n    const otpCode = await credentials.otpCodeRetriever();\n\n    const otpTokenResult = await this.getLongTermTwoFactorToken(otpCode);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    return { success: true, longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken };\n  }\n\n  async login(credentials: ScraperSpecificCredentials): Promise<ScraperLoginResult> {\n    const otpTokenResult = await this.resolveOtpToken(credentials);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    debug('Requesting id token');\n    const getIdTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/getIdToken`, {\n      otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,\n      email: credentials.email,\n      pass: credentials.password,\n      pinCode: '',\n    });\n\n    const {\n      resultData: { idToken },\n    } = getIdTokenResponse;\n\n    debug('Requesting session token');\n\n    const getSessionTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/sessions/token`, {\n      idToken,\n      pass: credentials.password,\n    });\n\n    const {\n      resultData: { accessToken },\n    } = getSessionTokenResponse;\n\n    this.accessToken = accessToken;\n\n    return {\n      success: true,\n      persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken,\n    };\n  }\n\n  private async fetchPortfolioMovements(portfolio: Portfolio, startDate: Date): Promise<TransactionsAccount> {\n    // TODO: Find out if we need the other accounts, there seems to always be one\n    const account = portfolio.accounts[0];\n    let cursor = null;\n    const movements = [];\n\n    while (!movements.length || new Date(movements[0].movementTimestamp) >= startDate) {\n      debug(`Fetching transactions for account ${portfolio.portfolioNum}...`);\n      const {\n        movements: { movements: newMovements, pagination },\n      }: { movements: { movements: Movement[]; pagination: QueryPagination } } = await fetchGraphql(\n        GRAPHQL_API_URL,\n        GET_MOVEMENTS,\n        {\n          portfolioId: portfolio.portfolioId,\n          accountId: account.accountId,\n          language: 'HEBREW',\n          pagination: {\n            cursor,\n            limit: 50,\n          },\n        },\n        { authorization: `Bearer ${this.accessToken}` },\n      );\n\n      movements.unshift(...newMovements);\n      cursor = pagination.cursor;\n      if (!pagination.hasMore) {\n        break;\n      }\n    }\n\n    movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());\n\n    const matchingMovements = movements.filter(movement => new Date(movement.movementTimestamp) >= startDate);\n    return {\n      accountNumber: portfolio.portfolioNum,\n      balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),\n      txns: matchingMovements.map((movement): ScrapingTransaction => {\n        const hasInstallments = movement.transaction?.enrichment?.recurrences?.some(x => x.isRecurrent);\n        const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;\n        return {\n          identifier: movement.movementId,\n          date: movement.valueDate,\n          chargedAmount: +movement.movementAmount * modifier,\n          chargedCurrency: movement.movementCurrency,\n          originalAmount: +movement.movementAmount * modifier,\n          originalCurrency: movement.movementCurrency,\n          description: this.sanitizeHebrew(movement.description),\n          processedDate: movement.movementTimestamp,\n          status: TransactionStatuses.Completed,\n          type: hasInstallments ? TransactionTypes.Installments : TransactionTypes.Normal,\n        };\n      }),\n    };\n  }\n\n  /**\n   * one zero hebrew strings are reversed with a unicode control character that forces display in LTR order\n   * We need to remove the unicode control character, and then reverse hebrew substrings inside the string\n   */\n  private sanitizeHebrew(text: string) {\n    if (!text.includes('\\u202d')) {\n      return text.trim();\n    }\n\n    const plainString = text.replace(/\\u202d/gi, '').trim();\n    const hebrewSubStringsRanges = [...plainString.matchAll(HEBREW_WORDS_REGEX)];\n    const rangesToReverse = hebrewSubStringsRanges.map(str => ({ start: str.index!, end: str.index! + str[0].length }));\n    const out = [];\n    let index = 0;\n\n    for (const { start, end } of rangesToReverse) {\n      out.push(...plainString.substring(index, start));\n      index += start - index;\n      const reversed = [...plainString.substring(start, end)].reverse();\n      out.push(...reversed);\n      index += end - start;\n    }\n\n    out.push(...plainString.substring(index, plainString.length));\n\n    return out.join('');\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    if (!this.accessToken) {\n      return createGenericError('login() was not called');\n    }\n\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    debug('Fetching account list');\n    const result = await fetchGraphql<{ customer: Customer[] }>(\n      GRAPHQL_API_URL,\n      GET_CUSTOMER,\n      {},\n      { authorization: `Bearer ${this.accessToken}` },\n    );\n    const portfolios = result.customer.flatMap(customer => customer.portfolios || []);\n\n    return {\n      success: true,\n      accounts: await Promise.all(\n        portfolios.map(portfolio => this.fetchPortfolioMovements(portfolio, startMoment.toDate())),\n      ),\n    };\n  }\n}\n"],"mappings":"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","ignoreList":[]}
242
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_fetch","_transactions","_baseScraper","_errors","_oneZeroQueries","e","__esModule","default","HEBREW_WORDS_REGEX","debug","getDebug","IDENTITY_SERVER_URL","GRAPHQL_API_URL","OneZeroScraper","BaseScraper","triggerTwoFactorAuth","phoneNumber","startsWith","createGenericError","deviceTokenResponse","fetchPost","extClientId","os","resultData","deviceToken","otpPrepareResponse","factorValue","otpChannel","otpContext","success","getLongTermTwoFactorToken","otpCode","otpVerifyResponse","otpToken","longTermTwoFactorAuthToken","resolveOtpToken","credentials","otpLongTermToken","otpCodeRetriever","errorType","ScraperErrorTypes","TwoFactorRetrieverMissing","errorMessage","triggerResult","otpTokenResult","login","getIdTokenResponse","otpSmsToken","email","pass","password","pinCode","idToken","getSessionTokenResponse","accessToken","persistentOtpToken","fetchPortfolioMovements","portfolio","startDate","account","accounts","cursor","movements","length","Date","movementTimestamp","portfolioNum","newMovements","pagination","fetchGraphql","GET_MOVEMENTS","portfolioId","accountId","language","limit","authorization","unshift","hasMore","sort","x","y","valueOf","matchingMovements","filter","movement","accountNumber","balance","parseFloat","runningBalance","txns","map","hasInstallments","transaction","enrichment","recurrences","some","isRecurrent","modifier","creditDebit","result","identifier","movementId","date","valueDate","chargedAmount","movementAmount","chargedCurrency","movementCurrency","originalAmount","originalCurrency","description","sanitizeHebrew","processedDate","status","TransactionStatuses","Completed","type","TransactionTypes","Installments","Normal","options","includeRawTransaction","rawTransaction","text","includes","trim","plainString","replace","hebrewSubStringsRanges","matchAll","rangesToReverse","str","start","index","end","out","push","substring","reversed","reverse","join","fetchData","defaultStartMoment","moment","subtract","add","toDate","startMoment","max","GET_CUSTOMER","portfolios","customer","flatMap","Promise","all","exports"],"sources":["../../src/scrapers/one-zero.ts"],"sourcesContent":["import moment from 'moment/moment';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGraphql, fetchPost } from '../helpers/fetch';\nimport {\n  type Transaction as ScrapingTransaction,\n  TransactionStatuses,\n  TransactionTypes,\n  type TransactionsAccount,\n} from '../transactions';\nimport { BaseScraper } from './base-scraper';\nimport { ScraperErrorTypes, createGenericError } from './errors';\nimport {\n  type ScraperGetLongTermTwoFactorTokenResult,\n  type ScraperLoginResult,\n  type ScraperScrapingResult,\n  type ScraperTwoFactorAuthTriggerResult,\n} from './interface';\nimport { GET_CUSTOMER, GET_MOVEMENTS } from './one-zero-queries';\n\nconst HEBREW_WORDS_REGEX = /[\\u0590-\\u05FF][\\u0590-\\u05FF\"'\\-_ /\\\\]*[\\u0590-\\u05FF]/g;\n\nconst debug = getDebug('one-zero');\n\ntype Account = {\n  accountId: string;\n};\n\ntype Portfolio = {\n  accounts: Array<Account>;\n  portfolioId: string;\n  portfolioNum: string;\n};\n\ntype Customer = {\n  customerId: string;\n  portfolios?: Array<Portfolio> | null;\n};\n\nexport type Category = {\n  categoryId: number;\n  dataSource: string;\n  subCategoryId?: number | null;\n};\n\nexport type Recurrence = {\n  dataSource: string;\n  isRecurrent: boolean;\n};\n\ntype TransactionEnrichment = {\n  categories?: Category[] | null;\n  recurrences?: Recurrence[] | null;\n};\n\ntype Transaction = {\n  enrichment?: TransactionEnrichment | null;\n  // TODO: Get installments information here\n  // transactionDetails: TransactionDetails;\n};\n\ntype Movement = {\n  accountId: string;\n  bankCurrencyAmount: string;\n  bookingDate: string;\n  conversionRate: string;\n  creditDebit: string;\n  description: string;\n  isReversed: boolean;\n  movementAmount: string;\n  movementCurrency: string;\n  movementId: string;\n  movementReversedId?: string | null;\n  movementTimestamp: string;\n  movementType: string;\n  portfolioId: string;\n  runningBalance: string;\n  transaction?: Transaction | null;\n  valueDate: string;\n};\n\ntype QueryPagination = { hasMore: boolean; cursor: string };\n\nconst IDENTITY_SERVER_URL = 'https://identity.tfd-bank.com/v1/';\n\nconst GRAPHQL_API_URL = 'https://mobile.tfd-bank.com/mobile-graph/graphql';\n\ntype ScraperSpecificCredentials = { email: string; password: string } & (\n  | {\n      otpCodeRetriever: () => Promise<string>;\n      phoneNumber: string;\n    }\n  | {\n      otpLongTermToken: string;\n    }\n);\n\nexport default class OneZeroScraper extends BaseScraper<ScraperSpecificCredentials> {\n  private otpContext?: string;\n\n  private accessToken?: string;\n\n  async triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult> {\n    if (!phoneNumber.startsWith('+')) {\n      return createGenericError(\n        'A full international phone number starting with + and a three digit country code is required',\n      );\n    }\n\n    debug('Fetching device token');\n    const deviceTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/devices/token`, {\n      extClientId: 'mobile',\n      os: 'Android',\n    });\n\n    const {\n      resultData: { deviceToken },\n    } = deviceTokenResponse;\n\n    debug(`Sending OTP to phone number ${phoneNumber}`);\n\n    const otpPrepareResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/prepare`, {\n      factorValue: phoneNumber,\n      deviceToken,\n      otpChannel: 'SMS_OTP',\n    });\n\n    const {\n      resultData: { otpContext },\n    } = otpPrepareResponse;\n\n    this.otpContext = otpContext;\n\n    return {\n      success: true,\n    };\n  }\n\n  public async getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if (!this.otpContext) {\n      return createGenericError('triggerOtp was not called before calling getPermenantOtpToken()');\n    }\n\n    debug('Requesting OTP token');\n    const otpVerifyResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/verify`, {\n      otpContext: this.otpContext,\n      otpCode,\n    });\n\n    const {\n      resultData: { otpToken },\n    } = otpVerifyResponse;\n    return { success: true, longTermTwoFactorAuthToken: otpToken };\n  }\n\n  private async resolveOtpToken(\n    credentials: ScraperSpecificCredentials,\n  ): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if ('otpLongTermToken' in credentials) {\n      if (!credentials.otpLongTermToken) {\n        return createGenericError('Invalid otpLongTermToken');\n      }\n      return { success: true, longTermTwoFactorAuthToken: credentials.otpLongTermToken };\n    }\n\n    if (!credentials.otpCodeRetriever) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.TwoFactorRetrieverMissing,\n        errorMessage: 'otpCodeRetriever is required when otpPermanentToken is not provided',\n      };\n    }\n\n    if (!credentials.phoneNumber) {\n      return createGenericError('phoneNumber is required when providing a otpCodeRetriever callback');\n    }\n\n    debug('Triggering user supplied otpCodeRetriever callback');\n    const triggerResult = await this.triggerTwoFactorAuth(credentials.phoneNumber);\n\n    if (!triggerResult.success) {\n      return triggerResult;\n    }\n\n    const otpCode = await credentials.otpCodeRetriever();\n\n    const otpTokenResult = await this.getLongTermTwoFactorToken(otpCode);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    return { success: true, longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken };\n  }\n\n  async login(credentials: ScraperSpecificCredentials): Promise<ScraperLoginResult> {\n    const otpTokenResult = await this.resolveOtpToken(credentials);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    debug('Requesting id token');\n    const getIdTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/getIdToken`, {\n      otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,\n      email: credentials.email,\n      pass: credentials.password,\n      pinCode: '',\n    });\n\n    const {\n      resultData: { idToken },\n    } = getIdTokenResponse;\n\n    debug('Requesting session token');\n\n    const getSessionTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/sessions/token`, {\n      idToken,\n      pass: credentials.password,\n    });\n\n    const {\n      resultData: { accessToken },\n    } = getSessionTokenResponse;\n\n    this.accessToken = accessToken;\n\n    return {\n      success: true,\n      persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken,\n    };\n  }\n\n  private async fetchPortfolioMovements(portfolio: Portfolio, startDate: Date): Promise<TransactionsAccount> {\n    // TODO: Find out if we need the other accounts, there seems to always be one\n    const account = portfolio.accounts[0];\n    let cursor = null;\n    const movements = [];\n\n    while (!movements.length || new Date(movements[0].movementTimestamp) >= startDate) {\n      debug(`Fetching transactions for account ${portfolio.portfolioNum}...`);\n      const {\n        movements: { movements: newMovements, pagination },\n      }: { movements: { movements: Movement[]; pagination: QueryPagination } } = await fetchGraphql(\n        GRAPHQL_API_URL,\n        GET_MOVEMENTS,\n        {\n          portfolioId: portfolio.portfolioId,\n          accountId: account.accountId,\n          language: 'HEBREW',\n          pagination: {\n            cursor,\n            limit: 50,\n          },\n        },\n        { authorization: `Bearer ${this.accessToken}` },\n      );\n\n      movements.unshift(...newMovements);\n      cursor = pagination.cursor;\n      if (!pagination.hasMore) {\n        break;\n      }\n    }\n\n    movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());\n\n    const matchingMovements = movements.filter(movement => new Date(movement.movementTimestamp) >= startDate);\n    return {\n      accountNumber: portfolio.portfolioNum,\n      balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),\n      txns: matchingMovements.map((movement): ScrapingTransaction => {\n        const hasInstallments = movement.transaction?.enrichment?.recurrences?.some(x => x.isRecurrent);\n        const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;\n        const result: ScrapingTransaction = {\n          identifier: movement.movementId,\n          date: movement.valueDate,\n          chargedAmount: +movement.movementAmount * modifier,\n          chargedCurrency: movement.movementCurrency,\n          originalAmount: +movement.movementAmount * modifier,\n          originalCurrency: movement.movementCurrency,\n          description: this.sanitizeHebrew(movement.description),\n          processedDate: movement.movementTimestamp,\n          status: TransactionStatuses.Completed,\n          type: hasInstallments ? TransactionTypes.Installments : TransactionTypes.Normal,\n        };\n\n        if (this.options?.includeRawTransaction) {\n          result.rawTransaction = movement;\n        }\n\n        return result;\n      }),\n    };\n  }\n\n  /**\n   * one zero hebrew strings are reversed with a unicode control character that forces display in LTR order\n   * We need to remove the unicode control character, and then reverse hebrew substrings inside the string\n   */\n  private sanitizeHebrew(text: string) {\n    if (!text.includes('\\u202d')) {\n      return text.trim();\n    }\n\n    const plainString = text.replace(/\\u202d/gi, '').trim();\n    const hebrewSubStringsRanges = [...plainString.matchAll(HEBREW_WORDS_REGEX)];\n    const rangesToReverse = hebrewSubStringsRanges.map(str => ({ start: str.index!, end: str.index! + str[0].length }));\n    const out = [];\n    let index = 0;\n\n    for (const { start, end } of rangesToReverse) {\n      out.push(...plainString.substring(index, start));\n      index += start - index;\n      const reversed = [...plainString.substring(start, end)].reverse();\n      out.push(...reversed);\n      index += end - start;\n    }\n\n    out.push(...plainString.substring(index, plainString.length));\n\n    return out.join('');\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    if (!this.accessToken) {\n      return createGenericError('login() was not called');\n    }\n\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    debug('Fetching account list');\n    const result = await fetchGraphql<{ customer: Customer[] }>(\n      GRAPHQL_API_URL,\n      GET_CUSTOMER,\n      {},\n      { authorization: `Bearer ${this.accessToken}` },\n    );\n    const portfolios = result.customer.flatMap(customer => customer.portfolios || []);\n\n    return {\n      success: true,\n      accounts: await Promise.all(\n        portfolios.map(portfolio => this.fetchPortfolioMovements(portfolio, startMoment.toDate())),\n      ),\n    };\n  }\n}\n"],"mappings":"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","ignoreList":[]}