israeli-bank-scrapers 6.4.2 → 6.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -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';\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":";;;;;;AAAA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AAEA,IAAAC,UAAA,GAAAD,OAAA;AACA,IAAAE,qBAAA,GAAAF,OAAA;AAMA,IAAAG,MAAA,GAAAH,OAAA;AACA,IAAAI,WAAA,GAAAJ,OAAA;AACA,IAAAK,aAAA,GAAAL,OAAA;AACA,IAAAM,uBAAA,GAAAN,OAAA;AACA,IAAAO,OAAA,GAAAP,OAAA;AACA,IAAAQ,MAAA,GAAAR,OAAA;AAA4C,SAAAD,uBAAAU,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAE5C,MAAMG,KAAK,GAAG,IAAAC,eAAQ,EAAC,SAAS,CAAC;AA0DjC,MAAMC,gBAAgB,GAAG,mCAAmC;AAC5D,MAAMC,SAAS,GAAG,GAAGD,gBAAgB,iCAAiC;AACtE,MAAME,YAAY,GAAG,mCAAmC;AACxD,MAAMC,oBAAoB,GAAG,uDAAuD;AACpF,MAAMC,QAAQ,GAAG,6BAA6B;AAC9C,MAAMC,iBAAiB,GAAG,mCAAmC;AAC7D,MAAMC,yBAAyB,GAAG,CAChC,GAAGJ,YAAY,qCAAqC,EACpD,GAAGA,YAAY,gCAAgC,CAChD;AACD,MAAMK,yBAAyB,GAAG,6BAA6B;AAC/D,MAAMC,2BAA2B,GAAG,WAAW;AAC/C,MAAMC,gBAAgB,GAAG,GAAGP,YAAY,oCAAoC;AAC5E,MAAMQ,mBAAmB,GAAG,0EAA0E;AACtG,MAAMC,WAAW,GAAG,YAAY;AAChC,MAAMC,oBAAoB,GAAG,WAAW;AAExC,MAAMC,gBAAgB,GAAG,uBAAuB;AAChD,MAAMC,gBAAgB,GAAG,qBAAqB;AAC9C,MAAMC,oBAAoB,GAAG,wBAAwB;AACrD,MAAMC,uBAAuB,GAAG,qEAAqE;AACrG,MAAMC,kBAAkB,GAAG,gBAAgB;AAC3C,MAAMC,oBAAoB,GAAG,oCAAoC;AACjE,MAAMC,2BAA2B,GAAG,sBAAsB;AAC1D,MAAMC,sBAAsB,GAAG,mCAAmC;AAClE,MAAMC,4BAA4B,GAAG,UAAU;AAC/C,MAAMC,6BAA6B,GAAG,kBAAkB;AACxD,MAAMC,mBAAmB,GAAG,CAAC,6BAA6B,CAAC;AAE3D,SAASC,iBAAiBA,CAACC,WAAuC,EAAE;EAClE,OAAO,CACL;IAAEC,QAAQ,EAAEb,gBAAgB;IAAEc,KAAK,EAAEF,WAAW,CAACG;EAAS,CAAC,EAC3D;IAAEF,QAAQ,EAAEZ,gBAAgB;IAAEa,KAAK,EAAEF,WAAW,CAACI;EAAS,CAAC,CAC5D;AACH;AAEA,eAAeC,UAAUA,CAACC,OAAgD,EAAE;EAC1E,IAAI,CAACA,OAAO,EAAEC,IAAI,EAAE;IAClB,OAAO,KAAK;EACd;EACA,MAAMC,QAAQ,GAAG,0BAA0BZ,4BAA4B,sBAAsBC,6BAA6B,KAAK;EAC/H,MAAMY,MAAM,GAAG,MAAMH,OAAO,CAACC,IAAI,CAACG,EAAE,CAAC,QAAQF,QAAQ,EAAE,CAAC;EACxD,OAAOC,MAAM,CAACE,MAAM,GAAG,CAAC;AAC1B;AAEA,SAASC,uBAAuBA,CAACL,IAAU,EAAwB;EACjE,OAAO;IACL,CAACM,oCAAY,CAACC,OAAO,GAAG,CAACpC,oBAAoB,EAAE2B,UAAU,CAAC;IAC1D,CAACQ,oCAAY,CAACE,eAAe,GAAG,CAAC,YAAY,CAAC,EAAE,MAAMR,IAAI,CAACS,CAAC,CAACzB,uBAAuB,CAAC,CAAC,CAAC;IACvF,CAACsB,oCAAY,CAACI,cAAc,GAAG,CAAChC,mBAAmB;EACrD,CAAC;AACH;AAEA,SAASiC,cAAcA,CAACC,gBAAsB,EAAE;EAC9C,MAAMC,kBAAkB,GAAG,IAAAC,eAAM,EAAC,CAAC,CAACC,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC;EACxD,MAAMC,SAAS,GAAGJ,gBAAgB,IAAIC,kBAAkB,CAACI,MAAM,CAAC,CAAC;EACjE,OAAOH,eAAM,CAACI,GAAG,CAACL,kBAAkB,EAAE,IAAAC,eAAM,EAACE,SAAS,CAAC,CAAC;AAC1D;AAEA,eAAeG,0BAA0BA,CACvCnB,IAAU,EACVoB,IAAwB,EACxBC,UAAkC,EACZ;EACtB,IAAI;IACFvD,KAAK,CAAC,sCAAsC,EAAEsD,IAAI,CAAC;IACnD,IAAIA,IAAI,CAACE,iBAAiB,KAAK,GAAG,EAAE;MAClC,MAAMC,QAAQ,GAAG,IAAAT,eAAM,EAACM,IAAI,CAACI,cAAc,CAAC;MAC5C,MAAMC,QAAQ,GAAG,IAAAX,eAAM,EAACM,IAAI,CAACM,aAAa,CAAC;MAE3C,MAAMC,MAAM,GAAG;QACbC,UAAU,EAAER,IAAI,CAACS,cAAc;QAC/BC,UAAU,EAAEV,IAAI,CAACW,oBAAoB;QACrCC,OAAO,EAAEZ,IAAI,CAACa,WAAW;QACzBC,UAAU,EAAEd,IAAI,CAACS,cAAc;QAC/BM,SAAS,EAAEf,IAAI,CAACgB,mBAAmB;QACnCC,MAAM,EAAEjB,IAAI,CAACkB,UAAU;QACvBC,mBAAmB,EAAEhB,QAAQ,CAACiB,MAAM,CAAC7D,WAAW,CAAC;QACjD8D,mBAAmB,EAAE,CAAChB,QAAQ,CAACiB,IAAI,CAAC,CAAC,GAAG,IAAI,GAAGjB,QAAQ,GAAGF,QAAQ,EAAEiB,MAAM,CAAC7D,WAAW,CAAC;QACvFgE,SAAS,EAAEvB,IAAI,CAACwB,eAAe;QAC/BC,YAAY,EAAEzB,IAAI,CAAC0B,eAAe;QAClCC,mBAAmB,EAAE3B,IAAI,CAAC4B;MAC5B,CAAC;MAED,MAAMC,QAAQ,GAAG,MAAM,IAAAC,0BAAmB,EAAsBlD,IAAI,EAAEvB,gBAAgB,EAAEkD,MAAM,EAAEN,UAAU,CAAC;MAC3G,MAAM8B,OAAO,GAAGF,QAAQ,EAAEG,IAAI,CAACC,MAAM,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,EAAEC,OAAO,GAAG,CAAC,CAAC,CAACC,MAAM;MACpEzF,KAAK,CAAC,mBAAmB,EAAE6D,MAAM,EAAE,UAAU,EAAEwB,OAAO,CAAC;MACvD,IAAIK,KAAK,CAACC,OAAO,CAACN,OAAO,CAAC,IAAIA,OAAO,CAAC/C,MAAM,GAAG,CAAC,EAAE;QAChD,MAAMsD,OAAO,GAAGP,OAAO,CAACQ,GAAG,CAACC,MAAM,IAAI,CAACA,MAAM,CAACC,KAAK,CAACC,IAAI,CAAC,CAAC,EAAEF,MAAM,CAACG,KAAK,CAACD,IAAI,CAAC,CAAC,CAAC,CAAC;QACjF,OAAO;UACLJ,OAAO,EAAEM,MAAM,CAACC,WAAW,CAACP,OAAO,CAAC;UACpCQ,IAAI,EAAER,OAAO,CACVS,MAAM,CAAC,CAAC,CAACC,KAAK,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,CAACC,IAAI,CAACC,GAAG,IAAIF,KAAK,CAACG,UAAU,CAACD,GAAG,CAAC,CAAC,CAAC,CAC/EX,GAAG,CAAC,CAAC,CAACS,KAAK,EAAEzE,KAAK,CAAC,KAAK,GAAGyE,KAAK,IAAIzE,KAAK,EAAE,CAAC,CAC5C6E,IAAI,CAAC,IAAI;QACd,CAAC;MACH;IACF;EACF,CAAC,CAAC,OAAOC,KAAK,EAAE;IACd3G,KAAK,CAAC,2CAA2C,EAAE2G,KAAK,CAAC;EAC3D;EAEA,OAAO;IACLf,OAAO,EAAE,CAAC,CAAC;IACXQ,IAAI,EAAEQ;EACR,CAAC;AACH;AAEA,SAASC,qBAAqBA,CAACC,OAAoB,EAAEhE,gBAAsB,EAAE;EAC3E,MAAMiE,IAAI,GAAGC,IAAI,CAACC,KAAK,CAACH,OAAO,CAACI,QAAQ,CAAC,CAAC,IAAI,IAAI,CAAC;EAEnDH,IAAI,CAACI,UAAU,GAAGtE,cAAc,CAACC,gBAAgB,CAAC,CAAC4B,MAAM,CAAC7D,WAAW,CAAC;EACtEkG,IAAI,CAACK,QAAQ,GAAG,IAAApE,eAAM,EAAC,CAAC,CAAC0B,MAAM,CAAC7D,WAAW,CAAC;EAC5CkG,IAAI,CAACM,KAAK,CAACC,MAAM,GAAGxG,oBAAoB;EAExC,OAAOiG,IAAI;AACb;AAEA,SAASQ,wBAAwBA,CAACT,OAAoB,EAAE;EACtD,OAAO;IACLU,gBAAgB,EAAEV,OAAO,CAACW,OAAO,CAAC,CAAC,CAACD,gBAAgB;IACpD,cAAc,EAAEV,OAAO,CAACW,OAAO,CAAC,CAAC,CAAC,cAAc;EAClD,CAAC;AACH;AAEA,eAAeC,mBAAmBA,CAChCC,IAA0B,EAC1BC,cAAiE,EACjEC,yBAAkC,GAAG,KAAK,EAClB;EACxB,OAAOC,OAAO,CAACC,GAAG,CAChBJ,IAAI,CAAC9B,GAAG,CAAC,MAAMmC,GAAG,IAAI;IACpB,MAAMC,WAAW,GAAG,MAAML,cAAc,CAACI,GAAG,CAAC;IAE7C,MAAME,OAAO,GAAG,IAAAlF,eAAM,EAACgF,GAAG,CAACtE,cAAc,EAAEV,eAAM,CAACmF,SAAS,CAACC,sBAAsB,CAAC,CAACC,WAAW,CAAC,CAAC;IAEjG,OAAO;MACLC,IAAI,EAAEC,8BAAgB,CAACC,MAAM;MAC7BC,UAAU,EAAET,GAAG,CAAC/D,oBAAoB,GAAGyE,QAAQ,CAACV,GAAG,CAAC/D,oBAAoB,EAAE,EAAE,CAAC,GAAG2C,SAAS;MACzF+B,IAAI,EAAET,OAAO;MACbU,aAAa,EAAEV,OAAO;MACtBW,cAAc,EAAEb,GAAG,CAAC7D,WAAW;MAC/B2E,gBAAgB,EAAEC,0BAAe;MACjCC,aAAa,EAAEhB,GAAG,CAAC7D,WAAW;MAC9B8E,WAAW,EAAEjB,GAAG,CAACkB,cAAc;MAC/B9C,IAAI,EAAE6B,WAAW,EAAE7B,IAAI;MACvB+C,MAAM,EACJtB,yBAAyB,IAAIG,GAAG,CAACoB,kBAAkB,GAC/CC,iCAAmB,CAACC,OAAO,GAC3BD,iCAAmB,CAACE;IAC5B,CAAC;EACH,CAAC,CACH,CAAC;AACH;AAEA,eAAeC,0BAA0BA,CAACtH,IAAW,EAA0B;EAC7E,MAAMuH,UAAU,GAAG,MAAM,IAAAC,iCAAW,EAACxH,IAAI,EAAE,uBAAuB,EAAE,EAAE,EAAEyH,GAAG,IAAI;IAC7E,OAAOA,GAAG,CAAC9D,GAAG,CAAC+D,EAAE,IAAIlE,KAAK,CAACmE,IAAI,CAACD,EAAE,CAACE,gBAAgB,CAAC,IAAI,CAAC,EAAEC,EAAE,IAAIA,EAAE,CAACC,WAAW,IAAI,EAAE,CAAC,CAAC;EACzF,CAAC,CAAC;EAEF,OAAOP,UAAU,CACd5D,GAAG,CAAC,CAAC,CAACoE,OAAO,EAAEhB,WAAW,EAAEiB,eAAe,EAAEC,SAAS,CAAC,MAAM;IAC5DxB,IAAI,EAAE,IAAA3F,eAAM,EAACiH,OAAO,EAAE,UAAU,CAAC,CAAC5B,WAAW,CAAC,CAAC;IAC/C+B,MAAM,EAAEC,UAAU,CAACF,SAAS,CAACG,UAAU,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACjDrB,WAAW;IACXiB,eAAe,CAAE;EACnB,CAAC,CAAC,CAAC,CACF7D,MAAM,CAACkE,GAAG,IAAIA,GAAG,CAAC5B,IAAI,CAAC,CACvB9C,GAAG,CAAC,CAAC;IAAE8C,IAAI;IAAEM,WAAW;IAAEmB;EAAO,CAAC,MAAM;IACvC9B,IAAI,EAAEC,8BAAgB,CAACC,MAAM;IAC7BG,IAAI;IACJC,aAAa,EAAED,IAAI;IACnBE,cAAc,EAAEuB,MAAM;IACtBtB,gBAAgB,EAAEC,0BAAe;IACjCC,aAAa,EAAEoB,MAAM;IACrBnB,WAAW;IACXE,MAAM,EAAEE,iCAAmB,CAACC;EAC9B,CAAC,CAAC,CAAC;AACP;AAEA,eAAekB,SAASA,CAACtI,IAAU,EAAE;EACnC,MAAM4F,OAAO,CAAC2C,IAAI,CAAC,CACjB,IAAAC,2CAAqB,EAACxI,IAAI,EAAEf,kBAAkB,CAAC,EAC/C,IAAAuJ,2CAAqB,EAACxI,IAAI,EAAEhB,uBAAuB,CAAC,EACpD,IAAAyJ,sBAAU,EAACzI,IAAI,EAAEtB,mBAAmB,CAAC,CACtC,CAAC;AACJ;AAIA,MAAMgK,cAAc,SAASC,8CAAsB,CAA6B;EAC9EC,eAAeA,CAACnJ,WAAuC,EAAE;IACvD,OAAO;MACLoJ,QAAQ,EAAE5K,SAAS;MACnBoF,MAAM,EAAE7D,iBAAiB,CAACC,WAAW,CAAC;MACtCV,oBAAoB;MACpB+J,cAAc,EAAE,MAAAA,CAAA,KAAY,IAAAC,+CAAyB,EAAC,IAAI,CAAC/I,IAAI,EAAEd,oBAAoB,CAAC;MACtF8J,UAAU,EAAE,MAAAA,CAAA,KAAYV,SAAS,CAAC,IAAI,CAACtI,IAAI,CAAC;MAC5CiJ,eAAe,EAAE5I,uBAAuB,CAAC,IAAI,CAACL,IAAI;IACpD,CAAC;EACH;EAEA,MAAMkJ,SAASA,CAAA,EAAG;IAChB,MAAM,IAAI,CAAClJ,IAAI,CAACmJ,KAAK,CAAC,uBAAuB,EAAEC,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IAEjF,MAAMC,aAAa,GAAG,CAAC,MAAM,IAAI,CAACtJ,IAAI,CAACG,EAAE,CAAChB,2BAA2B,CAAC,EAAEiB,MAAM;IAE9E,IAAI;MACF,MAAMmJ,OAA8B,GAAG,EAAE;MAEzC,KAAK,IAAIC,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGF,aAAa,EAAEE,CAAC,IAAI,CAAC,EAAE;QACzC,IAAIA,CAAC,GAAG,CAAC,EAAE;UACT,MAAM,IAAI,CAACxJ,IAAI,CAACmJ,KAAK,CAAC,uBAAuB,EAAEC,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;QACnF;QAEA,MAAM,IAAI,CAACrJ,IAAI,CAACmJ,KAAK,CAAC,GAAGhK,2BAA2B,cAAcqK,CAAC,GAAG,CAAC,GAAG,EAAEJ,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;QAC9GE,OAAO,CAACE,IAAI,CAAC,MAAM,IAAI,CAACC,YAAY,CAAC,CAAC,CAAC;MACzC;MAEA,OAAO;QACLC,OAAO,EAAE,IAAI;QACbC,QAAQ,EAAEL;MACZ,CAAC;IACH,CAAC,CAAC,OAAO5L,CAAC,EAAE;MACV,OAAO;QACLgM,OAAO,EAAE,KAAK;QACdE,SAAS,EAAEC,yBAAiB,CAACC,OAAO;QACpCC,YAAY,EAAGrM,CAAC,CAAWsM;MAC7B,CAAC;IACH;EACF;EAEA,MAAcC,sBAAsBA,CAAA,EAA2B;IAC7D,MAAM,IAAI,CAAClK,IAAI,CAACmJ,KAAK,CAAC,YAAY5K,yBAAyB,IAAI,EAAE6K,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IACnG,MAAMc,KAAK,GAAG,MAAM,IAAAC,0CAAoB,EAAC,IAAI,CAACpK,IAAI,EAAEqK,CAAC,IAAIA,CAAC,CAACC,GAAG,CAAC,CAAC,CAACC,QAAQ,CAAC/L,2BAA2B,CAAC,CAAC;IACvG,MAAMgM,SAAS,GAAG,MAAM,IAAAhC,2CAAqB,EAAC2B,KAAK,EAAE/K,sBAAsB,CAAC,CACzEqL,IAAI,CAAC,MAAM,IAAI,CAAC,CAChBC,KAAK,CAAC,MAAM,KAAK,CAAC;IACrB,IAAI,CAACF,SAAS,EAAE;MACd,OAAO,EAAE;IACX;IAEA,MAAMjD,UAAU,GAAG,MAAMD,0BAA0B,CAAC6C,KAAK,CAAC;IAC1D,OAAO5C,UAAU;EACnB;EAEA,MAAcmC,YAAYA,CAAA,EAAG;IAC3B,MAAM,IAAI,CAAC1J,IAAI,CAAC2K,eAAe,CAAC,YAAYvM,QAAQ,IAAI,CAAC;IACzD,MAAM,IAAI,CAAC4B,IAAI,CAACmJ,KAAK,CAAC,YAAY/K,QAAQ,IAAI,EAAEgL,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IAClF,MAAM,IAAAb,2CAAqB,EAAC,IAAI,CAACxI,IAAI,EAAE,YAAY3B,iBAAiB,IAAI,CAAC;IACzE,MAAM,IAAI,CAAC2B,IAAI,CAACmJ,KAAK,CAAC,YAAY9K,iBAAiB,IAAI,EAAE+K,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IAE3F,MAAMuB,oBAAoB,GAAG,MAAM,IAAI,CAAC5K,IAAI,CAACS,CAAC,CAAC,uBAAuB,CAAC;IACvE,MAAMoK,mBAAmB,GAAG,MAAMD,oBAAoB,EAAEE,WAAW,CAAC,OAAO,CAAC;IAC5E,MAAMC,aAAa,GAAI,MAAMF,mBAAmB,EAAEG,SAAS,CAAC,CAAY;IACxE,IAAI,CAACD,aAAa,EAAE;MAClB,MAAM,IAAIE,KAAK,CAAC,0BAA0B,CAAC;IAC7C;IAEA,MAAM,CAAChI,QAAQ,EAAE5B,UAAU,CAAC,GAAG,MAAMuE,OAAO,CAACsF,GAAG,CAC9C5M,yBAAyB,CAACqF,GAAG,CAAC,MAAM2G,GAAG,IAAI;MACzC,MAAM1F,OAAO,GAAG,MAAM,IAAI,CAAC5E,IAAI,CAACmL,cAAc,CAACb,GAAG,CAAC;MACnD,MAAMzF,IAAI,GAAGF,qBAAqB,CAACC,OAAO,EAAE,IAAI,CAAC7E,OAAO,CAACiB,SAAS,CAAC;MACnE,MAAMuE,OAAO,GAAGF,wBAAwB,CAACT,OAAO,CAAC;MAEjD,OAAO,CAAC,MAAM,IAAA1B,0BAAmB,EAA4B,IAAI,CAAClD,IAAI,EAAEsK,GAAG,EAAEzF,IAAI,EAAEU,OAAO,CAAC,EAAEA,OAAO,CAAC;IACvG,CAAC,CACH,CAAC;IAED,IAAI,CAACtC,QAAQ,IAAIA,QAAQ,CAACmI,MAAM,CAACzB,OAAO,KAAK,KAAK,EAAE;MAClD,MAAM,IAAIsB,KAAK,CACb,iDAAiDhI,QAAQ,GAAGA,QAAQ,CAACmI,MAAM,CAACC,QAAQ,CAAC,CAAC,CAAC,CAACC,IAAI,GAAG,EAAE,EACnG,CAAC;IACH;IAEA,MAAMC,YAAY,GAAGtI,QAAQ,CAACG,IAAI,CAAC+B,KAAK,CAACqG,IAAI,CAACrH,MAAM,CAAC2B,GAAG,IAAIA,GAAG,CAAC2F,gBAAgB,CAAC;IACjF,MAAMC,MAAM,GAAG,MAAMlG,mBAAmB,CACtC+F,YAAY,EACZ,IAAI,CAACxL,OAAO,CAAC4L,gCAAgC,GACzC7F,GAAG,IAAI3E,0BAA0B,CAAC,IAAI,CAACnB,IAAI,EAAE8F,GAAG,EAAEzE,UAAU,CAAC,GAC7D,MAAMuE,OAAO,CAACgG,OAAO,CAAC;MAAElI,OAAO,EAAE,CAAC,CAAC;MAAEQ,IAAI,EAAEQ;IAAU,CAAC,CAAC,EAC3D,IAAI,CAAC3E,OAAO,CAAC8L,aAAa,EAAEtB,QAAQ,CAAC,mCAAmC,CAC1E,CAAC;IAEDmB,MAAM,CACHvH,MAAM,CAACkE,GAAG,IAAI,IAAI,CAACyD,mBAAmB,CAACzD,GAAG,CAAC,CAAC,CAC5C0D,OAAO,CAAC1D,GAAG,IAAI;MACdA,GAAG,CAACpB,MAAM,GAAGE,iCAAmB,CAACC,OAAO;IAC1C,CAAC,CAAC;;IAEJ;IACA,MAAM4E,WAAW,GAAGrL,cAAc,CAAC,IAAI,CAACZ,OAAO,CAACiB,SAAS,CAAC;IAC1D,MAAMiL,oBAAoB,GAAGP,MAAM,CAACvH,MAAM,CAACkE,GAAG,IAAI,IAAAvH,eAAM,EAACuH,GAAG,CAAC5B,IAAI,CAAC,CAACyF,aAAa,CAACF,WAAW,CAAC,CAAC;IAE9F,MAAMzE,UAAU,GAAG,MAAM,IAAI,CAAC2C,sBAAsB,CAAC,CAAC;IACtD,MAAMiC,MAAM,GAAGF,oBAAoB,CAACG,MAAM,CAAC7E,UAAU,CAAC;IAEtD,OAAO;MACLwD,aAAa;MACbtF,IAAI,EAAE0G,MAAM;MACZE,OAAO,EAAE,CAACpJ,QAAQ,CAACG,IAAI,CAACC,MAAM,EAAEiJ;IAClC,CAAC;EACH;EAEQR,mBAAmBA,CAACzD,GAAgB,EAAW;IACrD,IAAI,IAAI,CAACtI,OAAO,CAAC8L,aAAa,EAAEtB,QAAQ,CAAC,+BAA+B,CAAC,IAAI,CAAClC,GAAG,CAAC9B,UAAU,EAAE;MAC5FzI,KAAK,CAAC,wBAAwBuK,GAAG,CAACtB,WAAW,oCAAoC,CAAC;MAClF,OAAO,IAAI;IACb;IAEA,IACE,IAAI,CAAChH,OAAO,CAAC8L,aAAa,EAAEtB,QAAQ,CAAC,wCAAwC,CAAC,IAC9EhL,mBAAmB,CAACgL,QAAQ,CAAClC,GAAG,CAACtB,WAAW,CAAC,EAC7C;MACAjJ,KAAK,CAAC,wBAAwBuK,GAAG,CAACtB,WAAW,0CAA0C,CAAC;MACxF,OAAO,IAAI;IACb;IAEA,OAAO,KAAK;EACd;AACF;AAAC,IAAAwF,QAAA,GAAAC,OAAA,CAAA3O,OAAA,GAEc6K,cAAc","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';\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":";;;;;;AAAA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AAEA,IAAAC,UAAA,GAAAD,OAAA;AACA,IAAAE,qBAAA,GAAAF,OAAA;AAMA,IAAAG,MAAA,GAAAH,OAAA;AACA,IAAAI,WAAA,GAAAJ,OAAA;AACA,IAAAK,aAAA,GAAAL,OAAA;AACA,IAAAM,uBAAA,GAAAN,OAAA;AACA,IAAAO,OAAA,GAAAP,OAAA;AACA,IAAAQ,MAAA,GAAAR,OAAA;AAA4C,SAAAD,uBAAAU,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAG5C,MAAMG,KAAK,GAAG,IAAAC,eAAQ,EAAC,SAAS,CAAC;AA0DjC,MAAMC,gBAAgB,GAAG,mCAAmC;AAC5D,MAAMC,SAAS,GAAG,GAAGD,gBAAgB,iCAAiC;AACtE,MAAME,YAAY,GAAG,mCAAmC;AACxD,MAAMC,oBAAoB,GAAG,uDAAuD;AACpF,MAAMC,QAAQ,GAAG,6BAA6B;AAC9C,MAAMC,iBAAiB,GAAG,mCAAmC;AAC7D,MAAMC,yBAAyB,GAAG,CAChC,GAAGJ,YAAY,qCAAqC,EACpD,GAAGA,YAAY,gCAAgC,CAChD;AACD,MAAMK,yBAAyB,GAAG,6BAA6B;AAC/D,MAAMC,2BAA2B,GAAG,WAAW;AAC/C,MAAMC,gBAAgB,GAAG,GAAGP,YAAY,oCAAoC;AAC5E,MAAMQ,mBAAmB,GAAG,0EAA0E;AACtG,MAAMC,WAAW,GAAG,YAAY;AAChC,MAAMC,oBAAoB,GAAG,WAAW;AAExC,MAAMC,gBAAgB,GAAG,uBAAuB;AAChD,MAAMC,gBAAgB,GAAG,qBAAqB;AAC9C,MAAMC,oBAAoB,GAAG,wBAAwB;AACrD,MAAMC,uBAAuB,GAAG,qEAAqE;AACrG,MAAMC,kBAAkB,GAAG,gBAAgB;AAC3C,MAAMC,oBAAoB,GAAG,oCAAoC;AACjE,MAAMC,2BAA2B,GAAG,sBAAsB;AAC1D,MAAMC,sBAAsB,GAAG,mCAAmC;AAClE,MAAMC,4BAA4B,GAAG,UAAU;AAC/C,MAAMC,6BAA6B,GAAG,kBAAkB;AACxD,MAAMC,mBAAmB,GAAG,CAAC,6BAA6B,CAAC;AAE3D,SAASC,iBAAiBA,CAACC,WAAuC,EAAE;EAClE,OAAO,CACL;IAAEC,QAAQ,EAAEb,gBAAgB;IAAEc,KAAK,EAAEF,WAAW,CAACG;EAAS,CAAC,EAC3D;IAAEF,QAAQ,EAAEZ,gBAAgB;IAAEa,KAAK,EAAEF,WAAW,CAACI;EAAS,CAAC,CAC5D;AACH;AAEA,eAAeC,UAAUA,CAACC,OAAgD,EAAE;EAC1E,IAAI,CAACA,OAAO,EAAEC,IAAI,EAAE;IAClB,OAAO,KAAK;EACd;EACA,MAAMC,QAAQ,GAAG,0BAA0BZ,4BAA4B,sBAAsBC,6BAA6B,KAAK;EAC/H,MAAMY,MAAM,GAAG,MAAMH,OAAO,CAACC,IAAI,CAACG,EAAE,CAAC,QAAQF,QAAQ,EAAE,CAAC;EACxD,OAAOC,MAAM,CAACE,MAAM,GAAG,CAAC;AAC1B;AAEA,SAASC,uBAAuBA,CAACL,IAAU,EAAwB;EACjE,OAAO;IACL,CAACM,oCAAY,CAACC,OAAO,GAAG,CAACpC,oBAAoB,EAAE2B,UAAU,CAAC;IAC1D,CAACQ,oCAAY,CAACE,eAAe,GAAG,CAAC,YAAY,CAAC,EAAE,MAAMR,IAAI,CAACS,CAAC,CAACzB,uBAAuB,CAAC,CAAC,CAAC;IACvF,CAACsB,oCAAY,CAACI,cAAc,GAAG,CAAChC,mBAAmB;EACrD,CAAC;AACH;AAEA,SAASiC,cAAcA,CAACC,gBAAsB,EAAE;EAC9C,MAAMC,kBAAkB,GAAG,IAAAC,eAAM,EAAC,CAAC,CAACC,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC;EACxD,MAAMC,SAAS,GAAGJ,gBAAgB,IAAIC,kBAAkB,CAACI,MAAM,CAAC,CAAC;EACjE,OAAOH,eAAM,CAACI,GAAG,CAACL,kBAAkB,EAAE,IAAAC,eAAM,EAACE,SAAS,CAAC,CAAC;AAC1D;AAEA,eAAeG,0BAA0BA,CACvCnB,IAAU,EACVoB,IAAwB,EACxBC,UAAkC,EACZ;EACtB,IAAI;IACFvD,KAAK,CAAC,sCAAsC,EAAEsD,IAAI,CAAC;IACnD,IAAIA,IAAI,CAACE,iBAAiB,KAAK,GAAG,EAAE;MAClC,MAAMC,QAAQ,GAAG,IAAAT,eAAM,EAACM,IAAI,CAACI,cAAc,CAAC;MAC5C,MAAMC,QAAQ,GAAG,IAAAX,eAAM,EAACM,IAAI,CAACM,aAAa,CAAC;MAE3C,MAAMC,MAAM,GAAG;QACbC,UAAU,EAAER,IAAI,CAACS,cAAc;QAC/BC,UAAU,EAAEV,IAAI,CAACW,oBAAoB;QACrCC,OAAO,EAAEZ,IAAI,CAACa,WAAW;QACzBC,UAAU,EAAEd,IAAI,CAACS,cAAc;QAC/BM,SAAS,EAAEf,IAAI,CAACgB,mBAAmB;QACnCC,MAAM,EAAEjB,IAAI,CAACkB,UAAU;QACvBC,mBAAmB,EAAEhB,QAAQ,CAACiB,MAAM,CAAC7D,WAAW,CAAC;QACjD8D,mBAAmB,EAAE,CAAChB,QAAQ,CAACiB,IAAI,CAAC,CAAC,GAAG,IAAI,GAAGjB,QAAQ,GAAGF,QAAQ,EAAEiB,MAAM,CAAC7D,WAAW,CAAC;QACvFgE,SAAS,EAAEvB,IAAI,CAACwB,eAAe;QAC/BC,YAAY,EAAEzB,IAAI,CAAC0B,eAAe;QAClCC,mBAAmB,EAAE3B,IAAI,CAAC4B;MAC5B,CAAC;MAED,MAAMC,QAAQ,GAAG,MAAM,IAAAC,0BAAmB,EAAsBlD,IAAI,EAAEvB,gBAAgB,EAAEkD,MAAM,EAAEN,UAAU,CAAC;MAC3G,MAAM8B,OAAO,GAAGF,QAAQ,EAAEG,IAAI,CAACC,MAAM,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,EAAEC,OAAO,GAAG,CAAC,CAAC,CAACC,MAAM;MACpEzF,KAAK,CAAC,mBAAmB,EAAE6D,MAAM,EAAE,UAAU,EAAEwB,OAAO,CAAC;MACvD,IAAIK,KAAK,CAACC,OAAO,CAACN,OAAO,CAAC,IAAIA,OAAO,CAAC/C,MAAM,GAAG,CAAC,EAAE;QAChD,MAAMsD,OAAO,GAAGP,OAAO,CAACQ,GAAG,CAACC,MAAM,IAAI,CAACA,MAAM,CAACC,KAAK,CAACC,IAAI,CAAC,CAAC,EAAEF,MAAM,CAACG,KAAK,CAACD,IAAI,CAAC,CAAC,CAAC,CAAC;QACjF,OAAO;UACLJ,OAAO,EAAEM,MAAM,CAACC,WAAW,CAACP,OAAO,CAAC;UACpCQ,IAAI,EAAER,OAAO,CACVS,MAAM,CAAC,CAAC,CAACC,KAAK,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,CAACC,IAAI,CAACC,GAAG,IAAIF,KAAK,CAACG,UAAU,CAACD,GAAG,CAAC,CAAC,CAAC,CAC/EX,GAAG,CAAC,CAAC,CAACS,KAAK,EAAEzE,KAAK,CAAC,KAAK,GAAGyE,KAAK,IAAIzE,KAAK,EAAE,CAAC,CAC5C6E,IAAI,CAAC,IAAI;QACd,CAAC;MACH;IACF;EACF,CAAC,CAAC,OAAOC,KAAK,EAAE;IACd3G,KAAK,CAAC,2CAA2C,EAAE2G,KAAK,CAAC;EAC3D;EAEA,OAAO;IACLf,OAAO,EAAE,CAAC,CAAC;IACXQ,IAAI,EAAEQ;EACR,CAAC;AACH;AAEA,SAASC,qBAAqBA,CAACC,OAAoB,EAAEhE,gBAAsB,EAAE;EAC3E,MAAMiE,IAAI,GAAGC,IAAI,CAACC,KAAK,CAACH,OAAO,CAACI,QAAQ,CAAC,CAAC,IAAI,IAAI,CAAC;EAEnDH,IAAI,CAACI,UAAU,GAAGtE,cAAc,CAACC,gBAAgB,CAAC,CAAC4B,MAAM,CAAC7D,WAAW,CAAC;EACtEkG,IAAI,CAACK,QAAQ,GAAG,IAAApE,eAAM,EAAC,CAAC,CAAC0B,MAAM,CAAC7D,WAAW,CAAC;EAC5CkG,IAAI,CAACM,KAAK,CAACC,MAAM,GAAGxG,oBAAoB;EAExC,OAAOiG,IAAI;AACb;AAEA,SAASQ,wBAAwBA,CAACT,OAAoB,EAAE;EACtD,OAAO;IACLU,gBAAgB,EAAEV,OAAO,CAACW,OAAO,CAAC,CAAC,CAACD,gBAAgB;IACpD,cAAc,EAAEV,OAAO,CAACW,OAAO,CAAC,CAAC,CAAC,cAAc;EAClD,CAAC;AACH;AAEA,eAAeC,mBAAmBA,CAChCC,IAA0B,EAC1BC,cAAiE,EACjEC,yBAAkC,GAAG,KAAK,EAC1C5F,OAAwB,EACA;EACxB,OAAO6F,OAAO,CAACC,GAAG,CAChBJ,IAAI,CAAC9B,GAAG,CAAC,MAAMmC,GAAG,IAAI;IACpB,MAAMC,WAAW,GAAG,MAAML,cAAc,CAACI,GAAG,CAAC;IAE7C,MAAME,OAAO,GAAG,IAAAlF,eAAM,EAACgF,GAAG,CAACtE,cAAc,EAAEV,eAAM,CAACmF,SAAS,CAACC,sBAAsB,CAAC,CAACC,WAAW,CAAC,CAAC;IAEjG,MAAMC,MAAmB,GAAG;MAC1BC,IAAI,EAAEC,8BAAgB,CAACC,MAAM;MAC7BC,UAAU,EAAEV,GAAG,CAAC/D,oBAAoB,GAAG0E,QAAQ,CAACX,GAAG,CAAC/D,oBAAoB,EAAE,EAAE,CAAC,GAAG2C,SAAS;MACzFgC,IAAI,EAAEV,OAAO;MACbW,aAAa,EAAEX,OAAO;MACtBY,cAAc,EAAEd,GAAG,CAAC7D,WAAW;MAC/B4E,gBAAgB,EAAEC,0BAAe;MACjCC,aAAa,EAAEjB,GAAG,CAAC7D,WAAW;MAC9B+E,WAAW,EAAElB,GAAG,CAACmB,cAAc;MAC/B/C,IAAI,EAAE6B,WAAW,EAAE7B,IAAI;MACvBgD,MAAM,EACJvB,yBAAyB,IAAIG,GAAG,CAACqB,kBAAkB,GAC/CC,iCAAmB,CAACC,OAAO,GAC3BD,iCAAmB,CAACE;IAC5B,CAAC;IAED,IAAIvH,OAAO,EAAEwH,qBAAqB,EAAE;MAClCnB,MAAM,CAACoB,cAAc,GAAG1B,GAAG;IAC7B;IAEA,OAAOM,MAAM;EACf,CAAC,CACH,CAAC;AACH;AAEA,eAAeqB,0BAA0BA,CAACzH,IAAW,EAA0B;EAC7E,MAAM0H,UAAU,GAAG,MAAM,IAAAC,iCAAW,EAAC3H,IAAI,EAAE,uBAAuB,EAAE,EAAE,EAAE4H,GAAG,IAAI;IAC7E,OAAOA,GAAG,CAACjE,GAAG,CAACkE,EAAE,IAAIrE,KAAK,CAACsE,IAAI,CAACD,EAAE,CAACE,gBAAgB,CAAC,IAAI,CAAC,EAAEC,EAAE,IAAIA,EAAE,CAACC,WAAW,IAAI,EAAE,CAAC,CAAC;EACzF,CAAC,CAAC;EAEF,OAAOP,UAAU,CACd/D,GAAG,CAAC,CAAC,CAACuE,OAAO,EAAElB,WAAW,EAAEmB,eAAe,EAAEC,SAAS,CAAC,MAAM;IAC5D1B,IAAI,EAAE,IAAA5F,eAAM,EAACoH,OAAO,EAAE,UAAU,CAAC,CAAC/B,WAAW,CAAC,CAAC;IAC/CkC,MAAM,EAAEC,UAAU,CAACF,SAAS,CAACG,UAAU,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACjDvB,WAAW;IACXmB,eAAe,CAAE;EACnB,CAAC,CAAC,CAAC,CACFhE,MAAM,CAACqE,GAAG,IAAIA,GAAG,CAAC9B,IAAI,CAAC,CACvB/C,GAAG,CAAC,CAAC;IAAE+C,IAAI;IAAEM,WAAW;IAAEqB;EAAO,CAAC,MAAM;IACvChC,IAAI,EAAEC,8BAAgB,CAACC,MAAM;IAC7BG,IAAI;IACJC,aAAa,EAAED,IAAI;IACnBE,cAAc,EAAEyB,MAAM;IACtBxB,gBAAgB,EAAEC,0BAAe;IACjCC,aAAa,EAAEsB,MAAM;IACrBrB,WAAW;IACXE,MAAM,EAAEE,iCAAmB,CAACC;EAC9B,CAAC,CAAC,CAAC;AACP;AAEA,eAAeoB,SAASA,CAACzI,IAAU,EAAE;EACnC,MAAM4F,OAAO,CAAC8C,IAAI,CAAC,CACjB,IAAAC,2CAAqB,EAAC3I,IAAI,EAAEf,kBAAkB,CAAC,EAC/C,IAAA0J,2CAAqB,EAAC3I,IAAI,EAAEhB,uBAAuB,CAAC,EACpD,IAAA4J,sBAAU,EAAC5I,IAAI,EAAEtB,mBAAmB,CAAC,CACtC,CAAC;AACJ;AAIA,MAAMmK,cAAc,SAASC,8CAAsB,CAA6B;EAC9EC,eAAeA,CAACtJ,WAAuC,EAAE;IACvD,OAAO;MACLuJ,QAAQ,EAAE/K,SAAS;MACnBoF,MAAM,EAAE7D,iBAAiB,CAACC,WAAW,CAAC;MACtCV,oBAAoB;MACpBkK,cAAc,EAAE,MAAAA,CAAA,KAAY,IAAAC,+CAAyB,EAAC,IAAI,CAAClJ,IAAI,EAAEd,oBAAoB,CAAC;MACtFiK,UAAU,EAAE,MAAAA,CAAA,KAAYV,SAAS,CAAC,IAAI,CAACzI,IAAI,CAAC;MAC5CoJ,eAAe,EAAE/I,uBAAuB,CAAC,IAAI,CAACL,IAAI;IACpD,CAAC;EACH;EAEA,MAAMqJ,SAASA,CAAA,EAAG;IAChB,MAAM,IAAI,CAACrJ,IAAI,CAACsJ,KAAK,CAAC,uBAAuB,EAAEC,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IAEjF,MAAMC,aAAa,GAAG,CAAC,MAAM,IAAI,CAACzJ,IAAI,CAACG,EAAE,CAAChB,2BAA2B,CAAC,EAAEiB,MAAM;IAE9E,IAAI;MACF,MAAMsJ,OAA8B,GAAG,EAAE;MAEzC,KAAK,IAAIC,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGF,aAAa,EAAEE,CAAC,IAAI,CAAC,EAAE;QACzC,IAAIA,CAAC,GAAG,CAAC,EAAE;UACT,MAAM,IAAI,CAAC3J,IAAI,CAACsJ,KAAK,CAAC,uBAAuB,EAAEC,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;QACnF;QAEA,MAAM,IAAI,CAACxJ,IAAI,CAACsJ,KAAK,CAAC,GAAGnK,2BAA2B,cAAcwK,CAAC,GAAG,CAAC,GAAG,EAAEJ,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;QAC9GE,OAAO,CAACE,IAAI,CAAC,MAAM,IAAI,CAACC,YAAY,CAAC,CAAC,CAAC;MACzC;MAEA,OAAO;QACLC,OAAO,EAAE,IAAI;QACbC,QAAQ,EAAEL;MACZ,CAAC;IACH,CAAC,CAAC,OAAO/L,CAAC,EAAE;MACV,OAAO;QACLmM,OAAO,EAAE,KAAK;QACdE,SAAS,EAAEC,yBAAiB,CAACC,OAAO;QACpCC,YAAY,EAAGxM,CAAC,CAAWyM;MAC7B,CAAC;IACH;EACF;EAEA,MAAcC,sBAAsBA,CAAA,EAA2B;IAC7D,MAAM,IAAI,CAACrK,IAAI,CAACsJ,KAAK,CAAC,YAAY/K,yBAAyB,IAAI,EAAEgL,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IACnG,MAAMc,KAAK,GAAG,MAAM,IAAAC,0CAAoB,EAAC,IAAI,CAACvK,IAAI,EAAEwK,CAAC,IAAIA,CAAC,CAACC,GAAG,CAAC,CAAC,CAACC,QAAQ,CAAClM,2BAA2B,CAAC,CAAC;IACvG,MAAMmM,SAAS,GAAG,MAAM,IAAAhC,2CAAqB,EAAC2B,KAAK,EAAElL,sBAAsB,CAAC,CACzEwL,IAAI,CAAC,MAAM,IAAI,CAAC,CAChBC,KAAK,CAAC,MAAM,KAAK,CAAC;IACrB,IAAI,CAACF,SAAS,EAAE;MACd,OAAO,EAAE;IACX;IAEA,MAAMjD,UAAU,GAAG,MAAMD,0BAA0B,CAAC6C,KAAK,CAAC;IAC1D,OAAO5C,UAAU;EACnB;EAEA,MAAcmC,YAAYA,CAAA,EAAG;IAC3B,MAAM,IAAI,CAAC7J,IAAI,CAAC8K,eAAe,CAAC,YAAY1M,QAAQ,IAAI,CAAC;IACzD,MAAM,IAAI,CAAC4B,IAAI,CAACsJ,KAAK,CAAC,YAAYlL,QAAQ,IAAI,EAAEmL,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IAClF,MAAM,IAAAb,2CAAqB,EAAC,IAAI,CAAC3I,IAAI,EAAE,YAAY3B,iBAAiB,IAAI,CAAC;IACzE,MAAM,IAAI,CAAC2B,IAAI,CAACsJ,KAAK,CAAC,YAAYjL,iBAAiB,IAAI,EAAEkL,EAAE,IAAKA,EAAE,CAAiBC,KAAK,CAAC,CAAC,CAAC;IAE3F,MAAMuB,oBAAoB,GAAG,MAAM,IAAI,CAAC/K,IAAI,CAACS,CAAC,CAAC,uBAAuB,CAAC;IACvE,MAAMuK,mBAAmB,GAAG,MAAMD,oBAAoB,EAAEE,WAAW,CAAC,OAAO,CAAC;IAC5E,MAAMC,aAAa,GAAI,MAAMF,mBAAmB,EAAEG,SAAS,CAAC,CAAY;IACxE,IAAI,CAACD,aAAa,EAAE;MAClB,MAAM,IAAIE,KAAK,CAAC,0BAA0B,CAAC;IAC7C;IAEA,MAAM,CAACnI,QAAQ,EAAE5B,UAAU,CAAC,GAAG,MAAMuE,OAAO,CAACyF,GAAG,CAC9C/M,yBAAyB,CAACqF,GAAG,CAAC,MAAM8G,GAAG,IAAI;MACzC,MAAM7F,OAAO,GAAG,MAAM,IAAI,CAAC5E,IAAI,CAACsL,cAAc,CAACb,GAAG,CAAC;MACnD,MAAM5F,IAAI,GAAGF,qBAAqB,CAACC,OAAO,EAAE,IAAI,CAAC7E,OAAO,CAACiB,SAAS,CAAC;MACnE,MAAMuE,OAAO,GAAGF,wBAAwB,CAACT,OAAO,CAAC;MAEjD,OAAO,CAAC,MAAM,IAAA1B,0BAAmB,EAA4B,IAAI,CAAClD,IAAI,EAAEyK,GAAG,EAAE5F,IAAI,EAAEU,OAAO,CAAC,EAAEA,OAAO,CAAC;IACvG,CAAC,CACH,CAAC;IAED,IAAI,CAACtC,QAAQ,IAAIA,QAAQ,CAACsI,MAAM,CAACzB,OAAO,KAAK,KAAK,EAAE;MAClD,MAAM,IAAIsB,KAAK,CACb,iDAAiDnI,QAAQ,GAAGA,QAAQ,CAACsI,MAAM,CAACC,QAAQ,CAAC,CAAC,CAAC,CAACC,IAAI,GAAG,EAAE,EACnG,CAAC;IACH;IAEA,MAAMC,YAAY,GAAGzI,QAAQ,CAACG,IAAI,CAAC+B,KAAK,CAACwG,IAAI,CAACxH,MAAM,CAAC2B,GAAG,IAAIA,GAAG,CAAC8F,gBAAgB,CAAC;IACjF,MAAMC,MAAM,GAAG,MAAMrG,mBAAmB,CACtCkG,YAAY,EACZ,IAAI,CAAC3L,OAAO,CAAC+L,gCAAgC,GACzChG,GAAG,IAAI3E,0BAA0B,CAAC,IAAI,CAACnB,IAAI,EAAE8F,GAAG,EAAEzE,UAAU,CAAC,GAC7D,MAAMuE,OAAO,CAACmG,OAAO,CAAC;MAAErI,OAAO,EAAE,CAAC,CAAC;MAAEQ,IAAI,EAAEQ;IAAU,CAAC,CAAC,EAC3D,IAAI,CAAC3E,OAAO,CAACiM,aAAa,EAAEtB,QAAQ,CAAC,mCAAmC,CAAC,EACzE,IAAI,CAAC3K,OACP,CAAC;IAED8L,MAAM,CACH1H,MAAM,CAACqE,GAAG,IAAI,IAAI,CAACyD,mBAAmB,CAACzD,GAAG,CAAC,CAAC,CAC5C0D,OAAO,CAAC1D,GAAG,IAAI;MACdA,GAAG,CAACtB,MAAM,GAAGE,iCAAmB,CAACC,OAAO;IAC1C,CAAC,CAAC;;IAEJ;IACA,MAAM8E,WAAW,GAAGxL,cAAc,CAAC,IAAI,CAACZ,OAAO,CAACiB,SAAS,CAAC;IAC1D,MAAMoL,oBAAoB,GAAGP,MAAM,CAAC1H,MAAM,CAACqE,GAAG,IAAI,IAAA1H,eAAM,EAAC0H,GAAG,CAAC9B,IAAI,CAAC,CAAC2F,aAAa,CAACF,WAAW,CAAC,CAAC;IAE9F,MAAMzE,UAAU,GAAG,MAAM,IAAI,CAAC2C,sBAAsB,CAAC,CAAC;IACtD,MAAMiC,MAAM,GAAGF,oBAAoB,CAACG,MAAM,CAAC7E,UAAU,CAAC;IAEtD,OAAO;MACLwD,aAAa;MACbzF,IAAI,EAAE6G,MAAM;MACZE,OAAO,EAAE,CAACvJ,QAAQ,CAACG,IAAI,CAACC,MAAM,EAAEoJ;IAClC,CAAC;EACH;EAEQR,mBAAmBA,CAACzD,GAAgB,EAAW;IACrD,IAAI,IAAI,CAACzI,OAAO,CAACiM,aAAa,EAAEtB,QAAQ,CAAC,+BAA+B,CAAC,IAAI,CAAClC,GAAG,CAAChC,UAAU,EAAE;MAC5F1I,KAAK,CAAC,wBAAwB0K,GAAG,CAACxB,WAAW,oCAAoC,CAAC;MAClF,OAAO,IAAI;IACb;IAEA,IACE,IAAI,CAACjH,OAAO,CAACiM,aAAa,EAAEtB,QAAQ,CAAC,wCAAwC,CAAC,IAC9EnL,mBAAmB,CAACmL,QAAQ,CAAClC,GAAG,CAACxB,WAAW,CAAC,EAC7C;MACAlJ,KAAK,CAAC,wBAAwB0K,GAAG,CAACxB,WAAW,0CAA0C,CAAC;MACxF,OAAO,IAAI;IACb;IAEA,OAAO,KAAK;EACd;AACF;AAAC,IAAA0F,QAAA,GAAAC,OAAA,CAAA9O,OAAA,GAEcgL,cAAc","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":[]}
@@ -48,11 +48,11 @@ function getTxnAmount(txn) {
48
48
  const debit = getAmountData(txn.debit);
49
49
  return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);
50
50
  }
51
- function convertTransactions(txns) {
51
+ function convertTransactions(txns, options) {
52
52
  return txns.map(txn => {
53
53
  const convertedDate = (0, _moment.default)(txn.date, DATE_FORMAT).toISOString();
54
54
  const convertedAmount = getTxnAmount(txn);
55
- return {
55
+ const result = {
56
56
  type: _transactions.TransactionTypes.Normal,
57
57
  identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,
58
58
  date: convertedDate,
@@ -64,6 +64,10 @@ function convertTransactions(txns) {
64
64
  description: txn.description,
65
65
  memo: txn.memo
66
66
  };
67
+ if (options?.includeRawTransaction) {
68
+ result.rawTransaction = txn;
69
+ }
70
+ return result;
67
71
  });
68
72
  }
69
73
  function getTransactionDate(tds, txnsTableHeaders) {
@@ -176,38 +180,38 @@ async function expandTransactionsTable(page) {
176
180
  await (0, _elementsInteractions.clickButton)(page, "a[id*='lnkCtlExpandAll']");
177
181
  }
178
182
  }
179
- async function scrapeTransactionsFromTable(page) {
183
+ async function scrapeTransactionsFromTable(page, options) {
180
184
  const pendingTxns = await extractTransactionsFromTable(page, PENDING_TRANSACTIONS_TABLE_ID, _transactions.TransactionStatuses.Pending);
181
185
  const completedTxns = await extractTransactionsFromTable(page, COMPLETED_TRANSACTIONS_TABLE_ID, _transactions.TransactionStatuses.Completed);
182
186
  const txns = [...pendingTxns, ...completedTxns];
183
- return convertTransactions(txns);
187
+ return convertTransactions(txns, options);
184
188
  }
185
- async function getAccountTransactions(page) {
189
+ async function getAccountTransactions(page, options) {
186
190
  await Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, `#${COMPLETED_TRANSACTIONS_TABLE_ID}`, false), (0, _elementsInteractions.waitUntilElementFound)(page, `.${ERROR_MESSAGE_CLASS}`, false)]);
187
191
  const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);
188
192
  if (noTransactionInRangeError) {
189
193
  return [];
190
194
  }
191
195
  await expandTransactionsTable(page);
192
- return scrapeTransactionsFromTable(page);
196
+ return scrapeTransactionsFromTable(page, options);
193
197
  }
194
- async function fetchAccountData(page, startDate, accountId) {
198
+ async function fetchAccountData(page, startDate, accountId, options) {
195
199
  await chooseAccount(page, accountId);
196
200
  await searchByDates(page, startDate);
197
201
  const accountNumber = await getAccountNumber(page);
198
- const txns = await getAccountTransactions(page);
202
+ const txns = await getAccountTransactions(page, options);
199
203
  return {
200
204
  accountNumber,
201
205
  txns
202
206
  };
203
207
  }
204
- async function fetchAccounts(page, startDate) {
208
+ async function fetchAccounts(page, startDate, options) {
205
209
  const accounts = [];
206
210
  const accountsList = await (0, _elementsInteractions.dropdownElements)(page, ACCOUNTS_DROPDOWN_SELECTOR);
207
211
  for (const account of accountsList) {
208
212
  if (account.value !== '-1') {
209
213
  // Skip "All accounts" option
210
- const accountData = await fetchAccountData(page, startDate, account.value);
214
+ const accountData = await fetchAccountData(page, startDate, account.value, options);
211
215
  accounts.push(accountData);
212
216
  }
213
217
  }
@@ -231,7 +235,7 @@ class UnionBankScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
231
235
  const startDate = this.options.startDate || defaultStartMoment.toDate();
232
236
  const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
233
237
  await this.navigateTo(TRANSACTIONS_URL);
234
- const accounts = await fetchAccounts(this.page, startMoment);
238
+ const accounts = await fetchAccounts(this.page, startMoment, this.options);
235
239
  return {
236
240
  success: true,
237
241
  accounts
@@ -239,4 +243,4 @@ class UnionBankScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
239
243
  }
240
244
  }
241
245
  var _default = exports.default = UnionBankScraper;
242
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_navigation","_transactions","_baseScraperWithBrowser","e","__esModule","default","BASE_URL","TRANSACTIONS_URL","DATE_FORMAT","NO_TRANSACTION_IN_DATE_RANGE_TEXT","DATE_HEADER","DESCRIPTION_HEADER","REFERENCE_HEADER","DEBIT_HEADER","CREDIT_HEADER","PENDING_TRANSACTIONS_TABLE_ID","COMPLETED_TRANSACTIONS_TABLE_ID","ERROR_MESSAGE_CLASS","ACCOUNTS_DROPDOWN_SELECTOR","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","amountStrCopy","replace","parseFloat","getTxnAmount","txn","credit","debit","Number","isNaN","convertTransactions","txns","map","convertedDate","moment","date","toISOString","convertedAmount","type","TransactionTypes","Normal","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","status","description","memo","getTransactionDate","tds","txnsTableHeaders","trim","getTransactionDescription","getTransactionReference","getTransactionDebit","getTransactionCredit","extractTransactionDetails","txnRow","txnStatus","innerTds","isExpandedDescRow","id","editLastTransactionDesc","lastTxn","handleTransactionRow","txnType","lastTransaction","pop","push","Error","getTransactionsTableHeaders","page","tableTypeId","headersMap","headersObjs","pageEvalAll","ths","th","index","text","innerText","headerObj","extractTransactionsFromTable","transactionsTableHeaders","transactionsRows","trs","tr","getAttribute","Array","from","getElementsByTagName","td","isNoTransactionInDateRangeError","hasErrorInfoElement","elementPresentOnPage","errorText","$eval","errorElement","chooseAccount","accountId","hasDropDownList","dropdownSelect","searchByDates","startDate","waitUntilElementFound","fillInput","format","clickButton","waitForNavigation","getAccountNumber","selectedSnifAccount","option","expandTransactionsTable","hasExpandAllButton","scrapeTransactionsFromTable","pendingTxns","TransactionStatuses","Pending","completedTxns","Completed","getAccountTransactions","Promise","race","noTransactionInRangeError","fetchAccountData","accountNumber","fetchAccounts","accounts","accountsList","dropdownElements","account","accountData","waitForPostLogin","UnionBankScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","options","toDate","startMoment","max","navigateTo","success","_default","exports"],"sources":["../../src/scrapers/union-bank.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  clickButton,\n  dropdownElements,\n  dropdownSelect,\n  elementPresentOnPage,\n  fillInput,\n  pageEvalAll,\n  waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\n\nconst BASE_URL = 'https://hb.unionbank.co.il';\nconst TRANSACTIONS_URL = `${BASE_URL}/eBanking/Accounts/ExtendedActivity.aspx#/`;\nconst DATE_FORMAT = 'DD/MM/YY';\nconst NO_TRANSACTION_IN_DATE_RANGE_TEXT = 'לא קיימות תנועות מתאימות על פי הסינון שהוגדר';\nconst DATE_HEADER = 'תאריך';\nconst DESCRIPTION_HEADER = 'תיאור';\nconst REFERENCE_HEADER = 'אסמכתא';\nconst DEBIT_HEADER = 'חובה';\nconst CREDIT_HEADER = 'זכות';\nconst PENDING_TRANSACTIONS_TABLE_ID = 'trTodayActivityNapaTableUpper';\nconst COMPLETED_TRANSACTIONS_TABLE_ID = 'ctlActivityTable';\nconst ERROR_MESSAGE_CLASS = 'errInfo';\nconst ACCOUNTS_DROPDOWN_SELECTOR = 'select#ddlAccounts_m_ddl';\n\nfunction getPossibleLoginResults() {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [/eBanking\\/Accounts/];\n  urls[LoginResults.InvalidPassword] = [/InternalSite\\/CustomUpdate\\/leumi\\/LoginPage.ASP/];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#uid', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string) {\n  const amountStrCopy = amountStr.replace(',', '');\n  return parseFloat(amountStrCopy);\n}\n\ninterface ScrapedTransaction {\n  credit: string;\n  debit: string;\n  date: string;\n  reference?: string;\n  description: string;\n  memo: string;\n  status: TransactionStatuses;\n}\n\nfunction getTxnAmount(txn: ScrapedTransaction) {\n  const credit = getAmountData(txn.credit);\n  const debit = getAmountData(txn.debit);\n  return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map(txn => {\n    const convertedDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const convertedAmount = getTxnAmount(txn);\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: convertedDate,\n      processedDate: convertedDate,\n      originalAmount: convertedAmount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: convertedAmount,\n      status: txn.status,\n      description: txn.description,\n      memo: txn.memo,\n    };\n  });\n}\n\ntype TransactionsTr = { id: string; innerTds: TransactionsTrTds };\ntype TransactionTableHeaders = Record<string, number>;\ntype TransactionsTrTds = string[];\n\nfunction getTransactionDate(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[DATE_HEADER]] || '').trim();\n}\n\nfunction getTransactionDescription(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[DESCRIPTION_HEADER]] || '').trim();\n}\n\nfunction getTransactionReference(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[REFERENCE_HEADER]] || '').trim();\n}\n\nfunction getTransactionDebit(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[DEBIT_HEADER]] || '').trim();\n}\n\nfunction getTransactionCredit(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[CREDIT_HEADER]] || '').trim();\n}\n\nfunction extractTransactionDetails(\n  txnRow: TransactionsTr,\n  txnsTableHeaders: TransactionTableHeaders,\n  txnStatus: TransactionStatuses,\n): ScrapedTransaction {\n  const tds = txnRow.innerTds;\n  return {\n    status: txnStatus,\n    date: getTransactionDate(tds, txnsTableHeaders),\n    description: getTransactionDescription(tds, txnsTableHeaders),\n    reference: getTransactionReference(tds, txnsTableHeaders),\n    debit: getTransactionDebit(tds, txnsTableHeaders),\n    credit: getTransactionCredit(tds, txnsTableHeaders),\n    memo: '',\n  };\n}\n\nfunction isExpandedDescRow(txnRow: TransactionsTr) {\n  return txnRow.id === 'rowAdded';\n}\n\n/* eslint-disable no-param-reassign */\nfunction editLastTransactionDesc(txnRow: TransactionsTr, lastTxn: ScrapedTransaction): ScrapedTransaction {\n  lastTxn.description = `${lastTxn.description} ${txnRow.innerTds[0]}`;\n  return lastTxn;\n}\n\nfunction handleTransactionRow(\n  txns: ScrapedTransaction[],\n  txnsTableHeaders: TransactionTableHeaders,\n  txnRow: TransactionsTr,\n  txnType: TransactionStatuses,\n) {\n  if (isExpandedDescRow(txnRow)) {\n    const lastTransaction = txns.pop();\n    if (lastTransaction) {\n      txns.push(editLastTransactionDesc(txnRow, lastTransaction));\n    } else {\n      throw new Error('internal union-bank error');\n    }\n  } else {\n    txns.push(extractTransactionDetails(txnRow, txnsTableHeaders, txnType));\n  }\n}\n\nasync function getTransactionsTableHeaders(page: Page, tableTypeId: string) {\n  const headersMap: Record<string, any> = [];\n  const headersObjs = await pageEvalAll(page, `#WorkSpaceBox #${tableTypeId} tr[class='header'] th`, null, ths => {\n    return ths.map((th, index) => ({\n      text: (th as HTMLElement).innerText.trim(),\n      index,\n    }));\n  });\n\n  for (const headerObj of headersObjs) {\n    headersMap[headerObj.text] = headerObj.index;\n  }\n  return headersMap;\n}\n\nasync function extractTransactionsFromTable(\n  page: Page,\n  tableTypeId: string,\n  txnType: TransactionStatuses,\n): Promise<ScrapedTransaction[]> {\n  const txns: ScrapedTransaction[] = [];\n  const transactionsTableHeaders = await getTransactionsTableHeaders(page, tableTypeId);\n\n  const transactionsRows = await pageEvalAll<TransactionsTr[]>(\n    page,\n    `#WorkSpaceBox #${tableTypeId} tr[class]:not([class='header'])`,\n    [],\n    trs => {\n      return (trs as HTMLElement[]).map(tr => ({\n        id: tr.getAttribute('id') || '',\n        innerTds: Array.from(tr.getElementsByTagName('td')).map(td => (td as HTMLElement).innerText),\n      }));\n    },\n  );\n\n  for (const txnRow of transactionsRows) {\n    handleTransactionRow(txns, transactionsTableHeaders, txnRow, txnType);\n  }\n  return txns;\n}\n\nasync function isNoTransactionInDateRangeError(page: Page) {\n  const hasErrorInfoElement = await elementPresentOnPage(page, `.${ERROR_MESSAGE_CLASS}`);\n  if (hasErrorInfoElement) {\n    const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, errorElement => {\n      return (errorElement as HTMLElement).innerText;\n    });\n    return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;\n  }\n  return false;\n}\n\nasync function chooseAccount(page: Page, accountId: string) {\n  const hasDropDownList = await elementPresentOnPage(page, ACCOUNTS_DROPDOWN_SELECTOR);\n  if (hasDropDownList) {\n    await dropdownSelect(page, ACCOUNTS_DROPDOWN_SELECTOR, accountId);\n  }\n}\n\nasync function searchByDates(page: Page, startDate: Moment) {\n  await dropdownSelect(page, 'select#ddlTransactionPeriod', '004');\n  await waitUntilElementFound(page, 'select#ddlTransactionPeriod');\n  await fillInput(page, 'input#dtFromDate_textBox', startDate.format(DATE_FORMAT));\n  await clickButton(page, 'input#btnDisplayDates');\n  await waitForNavigation(page);\n}\n\nasync function getAccountNumber(page: Page) {\n  const selectedSnifAccount = await page.$eval('#ddlAccounts_m_ddl option[selected=\"selected\"]', option => {\n    return (option as HTMLElement).innerText;\n  });\n\n  return selectedSnifAccount.replace('/', '_');\n}\n\nasync function expandTransactionsTable(page: Page) {\n  const hasExpandAllButton = await elementPresentOnPage(page, \"a[id*='lnkCtlExpandAll']\");\n  if (hasExpandAllButton) {\n    await clickButton(page, \"a[id*='lnkCtlExpandAll']\");\n  }\n}\n\nasync function scrapeTransactionsFromTable(page: Page): Promise<Transaction[]> {\n  const pendingTxns = await extractTransactionsFromTable(\n    page,\n    PENDING_TRANSACTIONS_TABLE_ID,\n    TransactionStatuses.Pending,\n  );\n  const completedTxns = await extractTransactionsFromTable(\n    page,\n    COMPLETED_TRANSACTIONS_TABLE_ID,\n    TransactionStatuses.Completed,\n  );\n  const txns = [...pendingTxns, ...completedTxns];\n  return convertTransactions(txns);\n}\n\nasync function getAccountTransactions(page: Page): Promise<Transaction[]> {\n  await Promise.race([\n    waitUntilElementFound(page, `#${COMPLETED_TRANSACTIONS_TABLE_ID}`, false),\n    waitUntilElementFound(page, `.${ERROR_MESSAGE_CLASS}`, false),\n  ]);\n\n  const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);\n  if (noTransactionInRangeError) {\n    return [];\n  }\n\n  await expandTransactionsTable(page);\n  return scrapeTransactionsFromTable(page);\n}\n\nasync function fetchAccountData(page: Page, startDate: Moment, accountId: string): Promise<TransactionsAccount> {\n  await chooseAccount(page, accountId);\n  await searchByDates(page, startDate);\n  const accountNumber = await getAccountNumber(page);\n  const txns = await getAccountTransactions(page);\n  return {\n    accountNumber,\n    txns,\n  };\n}\n\nasync function fetchAccounts(page: Page, startDate: Moment) {\n  const accounts: TransactionsAccount[] = [];\n  const accountsList = await dropdownElements(page, ACCOUNTS_DROPDOWN_SELECTOR);\n  for (const account of accountsList) {\n    if (account.value !== '-1') {\n      // Skip \"All accounts\" option\n      const accountData = await fetchAccountData(page, startDate, account.value);\n      accounts.push(accountData);\n    }\n  }\n  return accounts;\n}\n\nasync function waitForPostLogin(page: Page) {\n  return Promise.race([waitUntilElementFound(page, '#signoff', true), waitUntilElementFound(page, '#restore', true)]);\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass UnionBankScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${BASE_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#enter',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    await this.navigateTo(TRANSACTIONS_URL);\n\n    const accounts = await fetchAccounts(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default UnionBankScraper;\n"],"mappings":"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","ignoreList":[]}
246
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_navigation","_transactions","_baseScraperWithBrowser","e","__esModule","default","BASE_URL","TRANSACTIONS_URL","DATE_FORMAT","NO_TRANSACTION_IN_DATE_RANGE_TEXT","DATE_HEADER","DESCRIPTION_HEADER","REFERENCE_HEADER","DEBIT_HEADER","CREDIT_HEADER","PENDING_TRANSACTIONS_TABLE_ID","COMPLETED_TRANSACTIONS_TABLE_ID","ERROR_MESSAGE_CLASS","ACCOUNTS_DROPDOWN_SELECTOR","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","amountStrCopy","replace","parseFloat","getTxnAmount","txn","credit","debit","Number","isNaN","convertTransactions","txns","options","map","convertedDate","moment","date","toISOString","convertedAmount","result","type","TransactionTypes","Normal","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","status","description","memo","includeRawTransaction","rawTransaction","getTransactionDate","tds","txnsTableHeaders","trim","getTransactionDescription","getTransactionReference","getTransactionDebit","getTransactionCredit","extractTransactionDetails","txnRow","txnStatus","innerTds","isExpandedDescRow","id","editLastTransactionDesc","lastTxn","handleTransactionRow","txnType","lastTransaction","pop","push","Error","getTransactionsTableHeaders","page","tableTypeId","headersMap","headersObjs","pageEvalAll","ths","th","index","text","innerText","headerObj","extractTransactionsFromTable","transactionsTableHeaders","transactionsRows","trs","tr","getAttribute","Array","from","getElementsByTagName","td","isNoTransactionInDateRangeError","hasErrorInfoElement","elementPresentOnPage","errorText","$eval","errorElement","chooseAccount","accountId","hasDropDownList","dropdownSelect","searchByDates","startDate","waitUntilElementFound","fillInput","format","clickButton","waitForNavigation","getAccountNumber","selectedSnifAccount","option","expandTransactionsTable","hasExpandAllButton","scrapeTransactionsFromTable","pendingTxns","TransactionStatuses","Pending","completedTxns","Completed","getAccountTransactions","Promise","race","noTransactionInRangeError","fetchAccountData","accountNumber","fetchAccounts","accounts","accountsList","dropdownElements","account","accountData","waitForPostLogin","UnionBankScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","toDate","startMoment","max","navigateTo","success","_default","exports"],"sources":["../../src/scrapers/union-bank.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  clickButton,\n  dropdownElements,\n  dropdownSelect,\n  elementPresentOnPage,\n  fillInput,\n  pageEvalAll,\n  waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst BASE_URL = 'https://hb.unionbank.co.il';\nconst TRANSACTIONS_URL = `${BASE_URL}/eBanking/Accounts/ExtendedActivity.aspx#/`;\nconst DATE_FORMAT = 'DD/MM/YY';\nconst NO_TRANSACTION_IN_DATE_RANGE_TEXT = 'לא קיימות תנועות מתאימות על פי הסינון שהוגדר';\nconst DATE_HEADER = 'תאריך';\nconst DESCRIPTION_HEADER = 'תיאור';\nconst REFERENCE_HEADER = 'אסמכתא';\nconst DEBIT_HEADER = 'חובה';\nconst CREDIT_HEADER = 'זכות';\nconst PENDING_TRANSACTIONS_TABLE_ID = 'trTodayActivityNapaTableUpper';\nconst COMPLETED_TRANSACTIONS_TABLE_ID = 'ctlActivityTable';\nconst ERROR_MESSAGE_CLASS = 'errInfo';\nconst ACCOUNTS_DROPDOWN_SELECTOR = 'select#ddlAccounts_m_ddl';\n\nfunction getPossibleLoginResults() {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [/eBanking\\/Accounts/];\n  urls[LoginResults.InvalidPassword] = [/InternalSite\\/CustomUpdate\\/leumi\\/LoginPage.ASP/];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#uid', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string) {\n  const amountStrCopy = amountStr.replace(',', '');\n  return parseFloat(amountStrCopy);\n}\n\ninterface ScrapedTransaction {\n  credit: string;\n  debit: string;\n  date: string;\n  reference?: string;\n  description: string;\n  memo: string;\n  status: TransactionStatuses;\n}\n\nfunction getTxnAmount(txn: ScrapedTransaction) {\n  const credit = getAmountData(txn.credit);\n  const debit = getAmountData(txn.debit);\n  return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[], options?: ScraperOptions): Transaction[] {\n  return txns.map(txn => {\n    const convertedDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const convertedAmount = getTxnAmount(txn);\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: convertedDate,\n      processedDate: convertedDate,\n      originalAmount: convertedAmount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: convertedAmount,\n      status: txn.status,\n      description: txn.description,\n      memo: txn.memo,\n    };\n\n    if (options?.includeRawTransaction) {\n      result.rawTransaction = txn;\n    }\n\n    return result;\n  });\n}\n\ntype TransactionsTr = { id: string; innerTds: TransactionsTrTds };\ntype TransactionTableHeaders = Record<string, number>;\ntype TransactionsTrTds = string[];\n\nfunction getTransactionDate(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[DATE_HEADER]] || '').trim();\n}\n\nfunction getTransactionDescription(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[DESCRIPTION_HEADER]] || '').trim();\n}\n\nfunction getTransactionReference(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[REFERENCE_HEADER]] || '').trim();\n}\n\nfunction getTransactionDebit(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[DEBIT_HEADER]] || '').trim();\n}\n\nfunction getTransactionCredit(tds: TransactionsTrTds, txnsTableHeaders: TransactionTableHeaders) {\n  return (tds[txnsTableHeaders[CREDIT_HEADER]] || '').trim();\n}\n\nfunction extractTransactionDetails(\n  txnRow: TransactionsTr,\n  txnsTableHeaders: TransactionTableHeaders,\n  txnStatus: TransactionStatuses,\n): ScrapedTransaction {\n  const tds = txnRow.innerTds;\n  return {\n    status: txnStatus,\n    date: getTransactionDate(tds, txnsTableHeaders),\n    description: getTransactionDescription(tds, txnsTableHeaders),\n    reference: getTransactionReference(tds, txnsTableHeaders),\n    debit: getTransactionDebit(tds, txnsTableHeaders),\n    credit: getTransactionCredit(tds, txnsTableHeaders),\n    memo: '',\n  };\n}\n\nfunction isExpandedDescRow(txnRow: TransactionsTr) {\n  return txnRow.id === 'rowAdded';\n}\n\n/* eslint-disable no-param-reassign */\nfunction editLastTransactionDesc(txnRow: TransactionsTr, lastTxn: ScrapedTransaction): ScrapedTransaction {\n  lastTxn.description = `${lastTxn.description} ${txnRow.innerTds[0]}`;\n  return lastTxn;\n}\n\nfunction handleTransactionRow(\n  txns: ScrapedTransaction[],\n  txnsTableHeaders: TransactionTableHeaders,\n  txnRow: TransactionsTr,\n  txnType: TransactionStatuses,\n) {\n  if (isExpandedDescRow(txnRow)) {\n    const lastTransaction = txns.pop();\n    if (lastTransaction) {\n      txns.push(editLastTransactionDesc(txnRow, lastTransaction));\n    } else {\n      throw new Error('internal union-bank error');\n    }\n  } else {\n    txns.push(extractTransactionDetails(txnRow, txnsTableHeaders, txnType));\n  }\n}\n\nasync function getTransactionsTableHeaders(page: Page, tableTypeId: string) {\n  const headersMap: Record<string, any> = [];\n  const headersObjs = await pageEvalAll(page, `#WorkSpaceBox #${tableTypeId} tr[class='header'] th`, null, ths => {\n    return ths.map((th, index) => ({\n      text: (th as HTMLElement).innerText.trim(),\n      index,\n    }));\n  });\n\n  for (const headerObj of headersObjs) {\n    headersMap[headerObj.text] = headerObj.index;\n  }\n  return headersMap;\n}\n\nasync function extractTransactionsFromTable(\n  page: Page,\n  tableTypeId: string,\n  txnType: TransactionStatuses,\n): Promise<ScrapedTransaction[]> {\n  const txns: ScrapedTransaction[] = [];\n  const transactionsTableHeaders = await getTransactionsTableHeaders(page, tableTypeId);\n\n  const transactionsRows = await pageEvalAll<TransactionsTr[]>(\n    page,\n    `#WorkSpaceBox #${tableTypeId} tr[class]:not([class='header'])`,\n    [],\n    trs => {\n      return (trs as HTMLElement[]).map(tr => ({\n        id: tr.getAttribute('id') || '',\n        innerTds: Array.from(tr.getElementsByTagName('td')).map(td => (td as HTMLElement).innerText),\n      }));\n    },\n  );\n\n  for (const txnRow of transactionsRows) {\n    handleTransactionRow(txns, transactionsTableHeaders, txnRow, txnType);\n  }\n  return txns;\n}\n\nasync function isNoTransactionInDateRangeError(page: Page) {\n  const hasErrorInfoElement = await elementPresentOnPage(page, `.${ERROR_MESSAGE_CLASS}`);\n  if (hasErrorInfoElement) {\n    const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, errorElement => {\n      return (errorElement as HTMLElement).innerText;\n    });\n    return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;\n  }\n  return false;\n}\n\nasync function chooseAccount(page: Page, accountId: string) {\n  const hasDropDownList = await elementPresentOnPage(page, ACCOUNTS_DROPDOWN_SELECTOR);\n  if (hasDropDownList) {\n    await dropdownSelect(page, ACCOUNTS_DROPDOWN_SELECTOR, accountId);\n  }\n}\n\nasync function searchByDates(page: Page, startDate: Moment) {\n  await dropdownSelect(page, 'select#ddlTransactionPeriod', '004');\n  await waitUntilElementFound(page, 'select#ddlTransactionPeriod');\n  await fillInput(page, 'input#dtFromDate_textBox', startDate.format(DATE_FORMAT));\n  await clickButton(page, 'input#btnDisplayDates');\n  await waitForNavigation(page);\n}\n\nasync function getAccountNumber(page: Page) {\n  const selectedSnifAccount = await page.$eval('#ddlAccounts_m_ddl option[selected=\"selected\"]', option => {\n    return (option as HTMLElement).innerText;\n  });\n\n  return selectedSnifAccount.replace('/', '_');\n}\n\nasync function expandTransactionsTable(page: Page) {\n  const hasExpandAllButton = await elementPresentOnPage(page, \"a[id*='lnkCtlExpandAll']\");\n  if (hasExpandAllButton) {\n    await clickButton(page, \"a[id*='lnkCtlExpandAll']\");\n  }\n}\n\nasync function scrapeTransactionsFromTable(page: Page, options?: ScraperOptions): Promise<Transaction[]> {\n  const pendingTxns = await extractTransactionsFromTable(\n    page,\n    PENDING_TRANSACTIONS_TABLE_ID,\n    TransactionStatuses.Pending,\n  );\n  const completedTxns = await extractTransactionsFromTable(\n    page,\n    COMPLETED_TRANSACTIONS_TABLE_ID,\n    TransactionStatuses.Completed,\n  );\n  const txns = [...pendingTxns, ...completedTxns];\n  return convertTransactions(txns, options);\n}\n\nasync function getAccountTransactions(page: Page, options?: ScraperOptions): Promise<Transaction[]> {\n  await Promise.race([\n    waitUntilElementFound(page, `#${COMPLETED_TRANSACTIONS_TABLE_ID}`, false),\n    waitUntilElementFound(page, `.${ERROR_MESSAGE_CLASS}`, false),\n  ]);\n\n  const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);\n  if (noTransactionInRangeError) {\n    return [];\n  }\n\n  await expandTransactionsTable(page);\n  return scrapeTransactionsFromTable(page, options);\n}\n\nasync function fetchAccountData(\n  page: Page,\n  startDate: Moment,\n  accountId: string,\n  options?: ScraperOptions,\n): Promise<TransactionsAccount> {\n  await chooseAccount(page, accountId);\n  await searchByDates(page, startDate);\n  const accountNumber = await getAccountNumber(page);\n  const txns = await getAccountTransactions(page, options);\n  return {\n    accountNumber,\n    txns,\n  };\n}\n\nasync function fetchAccounts(page: Page, startDate: Moment, options?: ScraperOptions) {\n  const accounts: TransactionsAccount[] = [];\n  const accountsList = await dropdownElements(page, ACCOUNTS_DROPDOWN_SELECTOR);\n  for (const account of accountsList) {\n    if (account.value !== '-1') {\n      // Skip \"All accounts\" option\n      const accountData = await fetchAccountData(page, startDate, account.value, options);\n      accounts.push(accountData);\n    }\n  }\n  return accounts;\n}\n\nasync function waitForPostLogin(page: Page) {\n  return Promise.race([waitUntilElementFound(page, '#signoff', true), waitUntilElementFound(page, '#restore', true)]);\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass UnionBankScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${BASE_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#enter',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    await this.navigateTo(TRANSACTIONS_URL);\n\n    const accounts = await fetchAccounts(this.page, startMoment, this.options);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default UnionBankScraper;\n"],"mappings":"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","ignoreList":[]}