israeli-bank-scrapers 6.1.1 → 6.1.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/assertNever.js +7 -5
- package/lib/constants.js +16 -13
- package/lib/definitions.js +113 -109
- package/lib/helpers/browser.js +13 -9
- package/lib/helpers/dates.js +19 -18
- package/lib/helpers/debug.js +9 -9
- package/lib/helpers/elements-interactions.js +82 -78
- package/lib/helpers/fetch.js +85 -82
- package/lib/helpers/navigation.js +28 -24
- package/lib/helpers/storage.js +11 -10
- package/lib/helpers/transactions.js +32 -33
- package/lib/helpers/waiting.js +42 -45
- package/lib/index.js +82 -15
- package/lib/scrapers/amex.js +13 -11
- package/lib/scrapers/amex.test.d.ts +1 -0
- package/lib/scrapers/amex.test.js +49 -0
- package/lib/scrapers/base-beinleumi-group.d.ts +32 -0
- package/lib/scrapers/base-beinleumi-group.js +350 -234
- package/lib/scrapers/base-isracard-amex.js +273 -273
- package/lib/scrapers/base-scraper-with-browser.js +263 -241
- package/lib/scrapers/base-scraper-with-browser.test.d.ts +1 -0
- package/lib/scrapers/base-scraper-with-browser.test.js +53 -0
- package/lib/scrapers/base-scraper.js +82 -82
- package/lib/scrapers/behatsdaa.js +103 -98
- package/lib/scrapers/behatsdaa.test.d.ts +1 -0
- package/lib/scrapers/behatsdaa.test.js +46 -0
- package/lib/scrapers/beinleumi.js +13 -11
- package/lib/scrapers/beinleumi.test.d.ts +1 -0
- package/lib/scrapers/beinleumi.test.js +47 -0
- package/lib/scrapers/beyahad-bishvilha.js +132 -132
- package/lib/scrapers/beyahad-bishvilha.test.d.ts +1 -0
- package/lib/scrapers/beyahad-bishvilha.test.js +47 -0
- package/lib/scrapers/discount.js +101 -97
- package/lib/scrapers/discount.test.d.ts +1 -0
- package/lib/scrapers/discount.test.js +49 -0
- package/lib/scrapers/errors.js +25 -22
- package/lib/scrapers/factory.js +67 -66
- package/lib/scrapers/factory.test.d.ts +1 -0
- package/lib/scrapers/factory.test.js +19 -0
- package/lib/scrapers/hapoalim.js +175 -162
- package/lib/scrapers/hapoalim.test.d.ts +1 -0
- package/lib/scrapers/hapoalim.test.js +47 -0
- package/lib/scrapers/interface.js +5 -2
- package/lib/scrapers/isracard.js +13 -11
- package/lib/scrapers/isracard.test.d.ts +1 -0
- package/lib/scrapers/isracard.test.js +49 -0
- package/lib/scrapers/leumi.js +170 -167
- package/lib/scrapers/leumi.test.d.ts +1 -0
- package/lib/scrapers/leumi.test.js +47 -0
- package/lib/scrapers/massad.js +13 -11
- package/lib/scrapers/max.js +261 -261
- package/lib/scrapers/max.test.d.ts +1 -0
- package/lib/scrapers/max.test.js +65 -0
- package/lib/scrapers/mercantile.js +16 -14
- package/lib/scrapers/mercantile.test.d.ts +1 -0
- package/lib/scrapers/mercantile.test.js +45 -0
- package/lib/scrapers/mizrahi.js +154 -158
- package/lib/scrapers/mizrahi.test.d.ts +1 -0
- package/lib/scrapers/mizrahi.test.js +53 -0
- package/lib/scrapers/one-zero-queries.js +7 -4
- package/lib/scrapers/one-zero.js +221 -176
- package/lib/scrapers/one-zero.test.d.ts +1 -0
- package/lib/scrapers/one-zero.test.js +51 -0
- package/lib/scrapers/otsar-hahayal.js +13 -11
- package/lib/scrapers/otsar-hahayal.test.d.ts +1 -0
- package/lib/scrapers/otsar-hahayal.test.js +47 -0
- package/lib/scrapers/pagi.js +13 -11
- package/lib/scrapers/pagi.test.d.ts +1 -0
- package/lib/scrapers/pagi.test.js +47 -0
- package/lib/scrapers/union-bank.js +173 -172
- package/lib/scrapers/union-bank.test.d.ts +1 -0
- package/lib/scrapers/union-bank.test.js +47 -0
- package/lib/scrapers/visa-cal.d.ts +2 -0
- package/lib/scrapers/visa-cal.js +263 -254
- package/lib/scrapers/visa-cal.test.d.ts +1 -0
- package/lib/scrapers/visa-cal.test.js +49 -0
- package/lib/scrapers/yahav.js +206 -190
- package/lib/scrapers/yahav.test.d.ts +1 -0
- package/lib/scrapers/yahav.test.js +49 -0
- package/lib/transactions.js +16 -13
- package/package.json +9 -4
|
@@ -1,7 +1,10 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
3
6
|
exports.GET_MOVEMENTS = exports.GET_CUSTOMER = void 0;
|
|
4
|
-
exports.GET_CUSTOMER = `
|
|
7
|
+
const GET_CUSTOMER = exports.GET_CUSTOMER = `
|
|
5
8
|
query GetCustomer {
|
|
6
9
|
customer {
|
|
7
10
|
__typename
|
|
@@ -67,7 +70,7 @@ fragment Portfolio on Portfolio {
|
|
|
67
70
|
onboardingCompleted
|
|
68
71
|
}
|
|
69
72
|
`;
|
|
70
|
-
exports.GET_MOVEMENTS = `query GetMovements(
|
|
73
|
+
const GET_MOVEMENTS = exports.GET_MOVEMENTS = `query GetMovements(
|
|
71
74
|
$portfolioId: String!
|
|
72
75
|
$accountId: String!
|
|
73
76
|
$pagination: PaginationInput!
|
|
@@ -554,4 +557,4 @@ fragment MovementsFragment on Movements {
|
|
|
554
557
|
...PaginationFragment
|
|
555
558
|
}
|
|
556
559
|
}`;
|
|
557
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"one-zero-queries.js","sourceRoot":"","sources":["../../src/scrapers/one-zero-queries.ts"],"names":[],"mappings":";;;AAAa,QAAA,YAAY,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAiE3B,CAAC;AAEW,QAAA,aAAa,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAse3B,CAAC","sourcesContent":["export const GET_CUSTOMER = `\nquery GetCustomer {\n  customer {\n    __typename\n    customerId\n    userId\n    idType\n    idNumber\n    hebrewFirstName\n    hebrewLastName\n    latinFirstName\n    latinLastName\n    dateOfBirth\n    lastLoginDate\n    userEmail\n    gender\n    portfolioRelations {\n      __typename\n      customerId\n      customerRole\n      portfolioId\n      initiator\n      relationToInitiator\n      status\n    }\n    portfolios {\n      __typename\n      ...Portfolio\n    }\n    status\n  }\n}\nfragment Portfolio on Portfolio {\n  __typename\n  accounts {\n    __typename\n    accountId\n    accountType\n    closingDate\n    currency\n    openingDate\n    status\n    subType\n  }\n  activationDate\n  bank\n  baseCurrency\n  branch\n  club\n  clubDescription\n  iban\n  imageURL\n  isJointAccount\n  partnerName {\n    __typename\n    partnerFirstName\n    partnerLastName\n  }\n  portfolioId\n  portfolioNum\n  portfolioType\n  status\n  subType\n  onboardingCompleted\n}\n`;\n\nexport const GET_MOVEMENTS = `query GetMovements(\n  $portfolioId: String!\n  $accountId: String!\n  $pagination: PaginationInput!\n  $language: BffLanguage!\n) {\n  movements(\n    portfolioId: $portfolioId\n    accountId: $accountId\n    pagination: $pagination\n    language: $language\n  ) {\n    __typename\n    ...MovementsFragment\n  }\n}\nfragment TransactionInstrumentAmountFragment on TransactionInstrumentAmount {\n  __typename\n  instrumentAmount\n  instrumentSymbol\n  instrumentType\n}\nfragment CounterPartyReferenceFragment on CounterPartyReference {\n  __typename\n  bankId\n  bic\n  branchCode\n  id\n  name\n  type\n}\nfragment BaseTransactionFragment on BaseTransaction {\n  __typename\n  accountId\n  betweenOwnAccounts\n  bookDate\n  calculatedStatus\n  chargeAmount {\n    __typename\n    ...TransactionInstrumentAmountFragment\n  }\n  clearingSystem\n  counterParty {\n    __typename\n    ...CounterPartyReferenceFragment\n  }\n  currentPaymentNumber\n  direction\n  domainType\n  isReversal\n  method\n  originalAmount {\n    __typename\n    ...TransactionInstrumentAmountFragment\n  }\n  portfolioId\n  totalPaymentsCount\n  transactionId\n  transactionType\n  valueDate\n}\nfragment CategoryFragment on Category {\n  __typename\n  categoryId\n  dataSource\n  subCategoryId\n}\nfragment RecurrenceFragment on Recurrence {\n  __typename\n  dataSource\n  isRecurrent\n}\nfragment TransactionEnrichmentFragment on TransactionEnrichment {\n  __typename\n  categories {\n    __typename\n    ...CategoryFragment\n  }\n  recurrences {\n    __typename\n    ...RecurrenceFragment\n  }\n}\nfragment TransactionEventMetadataFragment on TransactionEventMetadata {\n  __typename\n  correlationId\n  processingOrder\n}\nfragment CounterPartyTransferData on CounterPartyTransfer {\n  __typename\n  accountId\n  bank_id\n  branch_code\n  counter_party_name\n}\nfragment BankTransferDetailsData on BankTransferDetails {\n  __typename\n  ... on CashBlockTransfer {\n    counterParty {\n      __typename\n      ...CounterPartyTransferData\n    }\n    transferDescriptionKey\n  }\n  ... on RTGSReturnTransfer {\n    transferDescriptionKey\n  }\n  ... on RTGSTransfer {\n    transferDescriptionKey\n  }\n  ... on SwiftReturnTransfer {\n    transferConversionRate\n    transferDescriptionKey\n  }\n  ... on SwiftTransfer {\n    transferConversionRate\n    transferDescriptionKey\n  }\n  ... on Transfer {\n    counterParty {\n      __typename\n      ...CounterPartyTransferData\n    }\n    transferDescriptionKey\n  }\n}\nfragment CategoryData on Category {\n  __typename\n  categoryId\n  dataSource\n  subCategoryId\n}\nfragment RecurrenceData on Recurrence {\n  __typename\n  dataSource\n  isRecurrent\n}\nfragment CardDetailsData on CardDetails {\n  __typename\n  ... on CardCharge {\n    book_date\n    cardDescriptionKey\n  }\n  ... on CardChargeFCY {\n    book_date\n    cardConversionRate\n    cardDescriptionKey\n    cardFCYAmount\n    cardFCYCurrency\n  }\n  ... on CardMonthlySettlement {\n    cardDescriptionKey\n  }\n  ... on CardRefund {\n    cardDescriptionKey\n  }\n  ... on CashBlockCardCharge {\n    cardDescriptionKey\n  }\n}\nfragment CashDetailsData on CashDetails {\n  __typename\n  ... on CashWithdrawal {\n    cashDescriptionKey\n  }\n  ... on CashWithdrawalFCY {\n    FCYAmount\n    FCYCurrency\n    cashDescriptionKey\n    conversionRate\n  }\n}\nfragment ChequesDetailsData on ChequesDetails {\n  __typename\n  ... on CashBlockChequeDeposit {\n    bookDate\n    chequesDescriptionKey\n  }\n  ... on ChequeDeposit {\n    bookDate\n    chequesDescriptionKey\n  }\n  ... on ChequeReturn {\n    bookDate\n    chequeReturnReason\n    chequesDescriptionKey\n  }\n  ... on ChequeWithdrawal {\n    chequesDescriptionKey\n  }\n}\nfragment DefaultDetailsData on DefaultDetails {\n  __typename\n  ... on DefaultWithTransaction {\n    defaultDescriptionKey\n  }\n  ... on DefaultWithoutTransaction {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    defaultDescriptionKey\n  }\n}\nfragment FeeDetailsData on FeeDetails {\n  __typename\n  ... on GeneralFee {\n    feeDescriptionKey\n  }\n}\nfragment LoanDetailsData on LoanDetails {\n  __typename\n  ... on FullPrePayment {\n    loanDescriptionKey\n  }\n  ... on Initiate {\n    loanDescriptionKey\n  }\n  ... on MonthlyPayment {\n    loanDescriptionKey\n    loanPaymentNumber\n    loanTotalPaymentsCount\n  }\n  ... on PartialPrePayment {\n    loanDescriptionKey\n  }\n}\nfragment MandateDetailsData on MandateDetails {\n  __typename\n  ... on MandatePayment {\n    mandateDescriptionKey\n  }\n  ... on MandateReturnPayment {\n    mandateDescriptionKey\n  }\n}\nfragment SavingsDetailsData on SavingsDetails {\n  __typename\n  ... on FullSavingsWithdrawal {\n    savingsDescriptionKey\n  }\n  ... on MonthlySavingsDeposit {\n    savingsDepositNumber\n    savingsDescriptionKey\n    savingsTotalDepositCount\n  }\n  ... on PartialSavingsWithdrawal {\n    savingsDescriptionKey\n  }\n  ... on SavingsClosing {\n    savingsDescriptionKey\n  }\n  ... on SavingsDeposit {\n    savingsDescriptionKey\n  }\n  ... on SavingsInterest {\n    savingsDescriptionKey\n  }\n  ... on SavingsPenalty {\n    savingsDescriptionKey\n  }\n  ... on SavingsTax {\n    savingsDescriptionKey\n  }\n}\nfragment SubscriptionDetailsData on SubscriptionDetails {\n  __typename\n  ... on SubscriptionPayment {\n    subscriptionDescriptionKey\n  }\n  ... on SubscriptionReturnPayment {\n    subscriptionDescriptionKey\n  }\n}\nfragment TransactionsDetailsData on TransactionDetails {\n  __typename\n  ... on BankTransfer {\n    bank_transfer_details {\n      __typename\n      ...BankTransferDetailsData\n    }\n    book_date\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Card {\n    card_details {\n      __typename\n      ...CardDetailsData\n    }\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Cash {\n    cash_details {\n      __typename\n      ...CashDetailsData\n    }\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Cheques {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    chequesDetails {\n      __typename\n      ...ChequesDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    valueDate\n    referenceNumber\n    frontImageUrl\n    backImageUrl\n  }\n  ... on Default {\n    default_details {\n      __typename\n      ...DefaultDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Fee {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    fee_details {\n      __typename\n      ...FeeDetailsData\n    }\n    value_date\n  }\n  ... on Loans {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    loan_details {\n      __typename\n      ...LoanDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Mandate {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    mandate_details {\n      __typename\n      ...MandateDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Savings {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    savings_details {\n      __typename\n      ...SavingsDetailsData\n    }\n    value_date\n  }\n  ... on SubscriptionTransaction {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    subscription_details {\n      __typename\n      ...SubscriptionDetailsData\n    }\n    value_date\n  }\n}\nfragment TransactionFragment on Transaction {\n  __typename\n  baseTransaction {\n    __typename\n    ...BaseTransactionFragment\n  }\n  enrichment {\n    __typename\n    ...TransactionEnrichmentFragment\n  }\n  metadata {\n    __typename\n    ...TransactionEventMetadataFragment\n  }\n  referenceNumber\n  transactionDetails {\n    __typename\n    ...TransactionsDetailsData\n  }\n}\nfragment MovementFragment on Movement {\n  __typename\n  accountId\n  bankCurrencyAmount\n  bookingDate\n  conversionRate\n  creditDebit\n  description\n  isReversed\n  linkTransaction {\n    __typename\n    ...TransactionFragment\n  }\n  movementAmount\n  movementCurrency\n  movementId\n  movementReversedId\n  movementTimestamp\n  movementType\n  portfolioId\n  runningBalance\n  transaction {\n    __typename\n    ...TransactionFragment\n  }\n  valueDate\n}\nfragment PaginationFragment on Pagination {\n  __typename\n  cursor\n  hasMore\n}\nfragment MovementsFragment on Movements {\n  __typename\n  isRunningBalanceInSync\n  movements {\n    __typename\n    ...MovementFragment\n  }\n  pagination {\n    __typename\n    ...PaginationFragment\n  }\n}`;\n"]}
|
|
560
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["GET_CUSTOMER","exports","GET_MOVEMENTS"],"sources":["../../src/scrapers/one-zero-queries.ts"],"sourcesContent":["export const GET_CUSTOMER = `\nquery GetCustomer {\n  customer {\n    __typename\n    customerId\n    userId\n    idType\n    idNumber\n    hebrewFirstName\n    hebrewLastName\n    latinFirstName\n    latinLastName\n    dateOfBirth\n    lastLoginDate\n    userEmail\n    gender\n    portfolioRelations {\n      __typename\n      customerId\n      customerRole\n      portfolioId\n      initiator\n      relationToInitiator\n      status\n    }\n    portfolios {\n      __typename\n      ...Portfolio\n    }\n    status\n  }\n}\nfragment Portfolio on Portfolio {\n  __typename\n  accounts {\n    __typename\n    accountId\n    accountType\n    closingDate\n    currency\n    openingDate\n    status\n    subType\n  }\n  activationDate\n  bank\n  baseCurrency\n  branch\n  club\n  clubDescription\n  iban\n  imageURL\n  isJointAccount\n  partnerName {\n    __typename\n    partnerFirstName\n    partnerLastName\n  }\n  portfolioId\n  portfolioNum\n  portfolioType\n  status\n  subType\n  onboardingCompleted\n}\n`;\n\nexport const GET_MOVEMENTS = `query GetMovements(\n  $portfolioId: String!\n  $accountId: String!\n  $pagination: PaginationInput!\n  $language: BffLanguage!\n) {\n  movements(\n    portfolioId: $portfolioId\n    accountId: $accountId\n    pagination: $pagination\n    language: $language\n  ) {\n    __typename\n    ...MovementsFragment\n  }\n}\nfragment TransactionInstrumentAmountFragment on TransactionInstrumentAmount {\n  __typename\n  instrumentAmount\n  instrumentSymbol\n  instrumentType\n}\nfragment CounterPartyReferenceFragment on CounterPartyReference {\n  __typename\n  bankId\n  bic\n  branchCode\n  id\n  name\n  type\n}\nfragment BaseTransactionFragment on BaseTransaction {\n  __typename\n  accountId\n  betweenOwnAccounts\n  bookDate\n  calculatedStatus\n  chargeAmount {\n    __typename\n    ...TransactionInstrumentAmountFragment\n  }\n  clearingSystem\n  counterParty {\n    __typename\n    ...CounterPartyReferenceFragment\n  }\n  currentPaymentNumber\n  direction\n  domainType\n  isReversal\n  method\n  originalAmount {\n    __typename\n    ...TransactionInstrumentAmountFragment\n  }\n  portfolioId\n  totalPaymentsCount\n  transactionId\n  transactionType\n  valueDate\n}\nfragment CategoryFragment on Category {\n  __typename\n  categoryId\n  dataSource\n  subCategoryId\n}\nfragment RecurrenceFragment on Recurrence {\n  __typename\n  dataSource\n  isRecurrent\n}\nfragment TransactionEnrichmentFragment on TransactionEnrichment {\n  __typename\n  categories {\n    __typename\n    ...CategoryFragment\n  }\n  recurrences {\n    __typename\n    ...RecurrenceFragment\n  }\n}\nfragment TransactionEventMetadataFragment on TransactionEventMetadata {\n  __typename\n  correlationId\n  processingOrder\n}\nfragment CounterPartyTransferData on CounterPartyTransfer {\n  __typename\n  accountId\n  bank_id\n  branch_code\n  counter_party_name\n}\nfragment BankTransferDetailsData on BankTransferDetails {\n  __typename\n  ... on CashBlockTransfer {\n    counterParty {\n      __typename\n      ...CounterPartyTransferData\n    }\n    transferDescriptionKey\n  }\n  ... on RTGSReturnTransfer {\n    transferDescriptionKey\n  }\n  ... on RTGSTransfer {\n    transferDescriptionKey\n  }\n  ... on SwiftReturnTransfer {\n    transferConversionRate\n    transferDescriptionKey\n  }\n  ... on SwiftTransfer {\n    transferConversionRate\n    transferDescriptionKey\n  }\n  ... on Transfer {\n    counterParty {\n      __typename\n      ...CounterPartyTransferData\n    }\n    transferDescriptionKey\n  }\n}\nfragment CategoryData on Category {\n  __typename\n  categoryId\n  dataSource\n  subCategoryId\n}\nfragment RecurrenceData on Recurrence {\n  __typename\n  dataSource\n  isRecurrent\n}\nfragment CardDetailsData on CardDetails {\n  __typename\n  ... on CardCharge {\n    book_date\n    cardDescriptionKey\n  }\n  ... on CardChargeFCY {\n    book_date\n    cardConversionRate\n    cardDescriptionKey\n    cardFCYAmount\n    cardFCYCurrency\n  }\n  ... on CardMonthlySettlement {\n    cardDescriptionKey\n  }\n  ... on CardRefund {\n    cardDescriptionKey\n  }\n  ... on CashBlockCardCharge {\n    cardDescriptionKey\n  }\n}\nfragment CashDetailsData on CashDetails {\n  __typename\n  ... on CashWithdrawal {\n    cashDescriptionKey\n  }\n  ... on CashWithdrawalFCY {\n    FCYAmount\n    FCYCurrency\n    cashDescriptionKey\n    conversionRate\n  }\n}\nfragment ChequesDetailsData on ChequesDetails {\n  __typename\n  ... on CashBlockChequeDeposit {\n    bookDate\n    chequesDescriptionKey\n  }\n  ... on ChequeDeposit {\n    bookDate\n    chequesDescriptionKey\n  }\n  ... on ChequeReturn {\n    bookDate\n    chequeReturnReason\n    chequesDescriptionKey\n  }\n  ... on ChequeWithdrawal {\n    chequesDescriptionKey\n  }\n}\nfragment DefaultDetailsData on DefaultDetails {\n  __typename\n  ... on DefaultWithTransaction {\n    defaultDescriptionKey\n  }\n  ... on DefaultWithoutTransaction {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    defaultDescriptionKey\n  }\n}\nfragment FeeDetailsData on FeeDetails {\n  __typename\n  ... on GeneralFee {\n    feeDescriptionKey\n  }\n}\nfragment LoanDetailsData on LoanDetails {\n  __typename\n  ... on FullPrePayment {\n    loanDescriptionKey\n  }\n  ... on Initiate {\n    loanDescriptionKey\n  }\n  ... on MonthlyPayment {\n    loanDescriptionKey\n    loanPaymentNumber\n    loanTotalPaymentsCount\n  }\n  ... on PartialPrePayment {\n    loanDescriptionKey\n  }\n}\nfragment MandateDetailsData on MandateDetails {\n  __typename\n  ... on MandatePayment {\n    mandateDescriptionKey\n  }\n  ... on MandateReturnPayment {\n    mandateDescriptionKey\n  }\n}\nfragment SavingsDetailsData on SavingsDetails {\n  __typename\n  ... on FullSavingsWithdrawal {\n    savingsDescriptionKey\n  }\n  ... on MonthlySavingsDeposit {\n    savingsDepositNumber\n    savingsDescriptionKey\n    savingsTotalDepositCount\n  }\n  ... on PartialSavingsWithdrawal {\n    savingsDescriptionKey\n  }\n  ... on SavingsClosing {\n    savingsDescriptionKey\n  }\n  ... on SavingsDeposit {\n    savingsDescriptionKey\n  }\n  ... on SavingsInterest {\n    savingsDescriptionKey\n  }\n  ... on SavingsPenalty {\n    savingsDescriptionKey\n  }\n  ... on SavingsTax {\n    savingsDescriptionKey\n  }\n}\nfragment SubscriptionDetailsData on SubscriptionDetails {\n  __typename\n  ... on SubscriptionPayment {\n    subscriptionDescriptionKey\n  }\n  ... on SubscriptionReturnPayment {\n    subscriptionDescriptionKey\n  }\n}\nfragment TransactionsDetailsData on TransactionDetails {\n  __typename\n  ... on BankTransfer {\n    bank_transfer_details {\n      __typename\n      ...BankTransferDetailsData\n    }\n    book_date\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Card {\n    card_details {\n      __typename\n      ...CardDetailsData\n    }\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Cash {\n    cash_details {\n      __typename\n      ...CashDetailsData\n    }\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Cheques {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    chequesDetails {\n      __typename\n      ...ChequesDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    valueDate\n    referenceNumber\n    frontImageUrl\n    backImageUrl\n  }\n  ... on Default {\n    default_details {\n      __typename\n      ...DefaultDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Fee {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    fee_details {\n      __typename\n      ...FeeDetailsData\n    }\n    value_date\n  }\n  ... on Loans {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    loan_details {\n      __typename\n      ...LoanDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Mandate {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    mandate_details {\n      __typename\n      ...MandateDetailsData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    value_date\n  }\n  ... on Savings {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    savings_details {\n      __typename\n      ...SavingsDetailsData\n    }\n    value_date\n  }\n  ... on SubscriptionTransaction {\n    categories {\n      __typename\n      ...CategoryData\n    }\n    recurrences {\n      __typename\n      ...RecurrenceData\n    }\n    subscription_details {\n      __typename\n      ...SubscriptionDetailsData\n    }\n    value_date\n  }\n}\nfragment TransactionFragment on Transaction {\n  __typename\n  baseTransaction {\n    __typename\n    ...BaseTransactionFragment\n  }\n  enrichment {\n    __typename\n    ...TransactionEnrichmentFragment\n  }\n  metadata {\n    __typename\n    ...TransactionEventMetadataFragment\n  }\n  referenceNumber\n  transactionDetails {\n    __typename\n    ...TransactionsDetailsData\n  }\n}\nfragment MovementFragment on Movement {\n  __typename\n  accountId\n  bankCurrencyAmount\n  bookingDate\n  conversionRate\n  creditDebit\n  description\n  isReversed\n  linkTransaction {\n    __typename\n    ...TransactionFragment\n  }\n  movementAmount\n  movementCurrency\n  movementId\n  movementReversedId\n  movementTimestamp\n  movementType\n  portfolioId\n  runningBalance\n  transaction {\n    __typename\n    ...TransactionFragment\n  }\n  valueDate\n}\nfragment PaginationFragment on Pagination {\n  __typename\n  cursor\n  hasMore\n}\nfragment MovementsFragment on Movements {\n  __typename\n  isRunningBalanceInSync\n  movements {\n    __typename\n    ...MovementFragment\n  }\n  pagination {\n    __typename\n    ...PaginationFragment\n  }\n}`;\n"],"mappings":";;;;;;AAAO,MAAMA,YAAY,GAAAC,OAAA,CAAAD,YAAA,GAAG;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AAEM,MAAME,aAAa,GAAAD,OAAA,CAAAC,aAAA,GAAG;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE","ignoreList":[]}
|
package/lib/scrapers/one-zero.js
CHANGED
|
@@ -1,193 +1,238 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.default = void 0;
|
|
7
|
+
var _moment = _interopRequireDefault(require("moment/moment"));
|
|
8
|
+
var _debug = require("../helpers/debug");
|
|
9
|
+
var _fetch = require("../helpers/fetch");
|
|
10
|
+
var _transactions = require("../transactions");
|
|
11
|
+
var _baseScraper = require("./base-scraper");
|
|
12
|
+
var _errors = require("./errors");
|
|
13
|
+
var _oneZeroQueries = require("./one-zero-queries");
|
|
14
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
13
15
|
const HEBREW_WORDS_REGEX = /[\u0590-\u05FF][\u0590-\u05FF"'\-_ /\\]*[\u0590-\u05FF]/g;
|
|
14
|
-
const debug = (0,
|
|
16
|
+
const debug = (0, _debug.getDebug)('one-zero');
|
|
15
17
|
const IDENTITY_SERVER_URL = 'https://identity.tfd-bank.com/v1/';
|
|
16
18
|
const GRAPHQL_API_URL = 'https://mobile.tfd-bank.com/mobile-graph/graphql';
|
|
17
|
-
class OneZeroScraper extends
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
if (!phoneNumber.startsWith('+')) {
|
|
22
|
-
return (0, errors_1.createGenericError)('A full international phone number starting with + and a three digit country code is required');
|
|
23
|
-
}
|
|
24
|
-
debug('Fetching device token');
|
|
25
|
-
const deviceTokenResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/devices/token`, {
|
|
26
|
-
extClientId: 'mobile',
|
|
27
|
-
os: 'Android',
|
|
28
|
-
});
|
|
29
|
-
const { resultData: { deviceToken }, } = deviceTokenResponse;
|
|
30
|
-
debug(`Sending OTP to phone number ${phoneNumber}`);
|
|
31
|
-
const otpPrepareResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/otp/prepare`, {
|
|
32
|
-
factorValue: phoneNumber,
|
|
33
|
-
deviceToken,
|
|
34
|
-
otpChannel: 'SMS_OTP',
|
|
35
|
-
});
|
|
36
|
-
const { resultData: { otpContext }, } = otpPrepareResponse;
|
|
37
|
-
this.otpContext = otpContext;
|
|
38
|
-
return {
|
|
39
|
-
success: true,
|
|
40
|
-
};
|
|
19
|
+
class OneZeroScraper extends _baseScraper.BaseScraper {
|
|
20
|
+
async triggerTwoFactorAuth(phoneNumber) {
|
|
21
|
+
if (!phoneNumber.startsWith('+')) {
|
|
22
|
+
return (0, _errors.createGenericError)('A full international phone number starting with + and a three digit country code is required');
|
|
41
23
|
}
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
24
|
+
debug('Fetching device token');
|
|
25
|
+
const deviceTokenResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/devices/token`, {
|
|
26
|
+
extClientId: 'mobile',
|
|
27
|
+
os: 'Android'
|
|
28
|
+
});
|
|
29
|
+
const {
|
|
30
|
+
resultData: {
|
|
31
|
+
deviceToken
|
|
32
|
+
}
|
|
33
|
+
} = deviceTokenResponse;
|
|
34
|
+
debug(`Sending OTP to phone number ${phoneNumber}`);
|
|
35
|
+
const otpPrepareResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/otp/prepare`, {
|
|
36
|
+
factorValue: phoneNumber,
|
|
37
|
+
deviceToken,
|
|
38
|
+
otpChannel: 'SMS_OTP'
|
|
39
|
+
});
|
|
40
|
+
const {
|
|
41
|
+
resultData: {
|
|
42
|
+
otpContext
|
|
43
|
+
}
|
|
44
|
+
} = otpPrepareResponse;
|
|
45
|
+
this.otpContext = otpContext;
|
|
46
|
+
return {
|
|
47
|
+
success: true
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
async getLongTermTwoFactorToken(otpCode) {
|
|
51
|
+
if (!this.otpContext) {
|
|
52
|
+
return (0, _errors.createGenericError)('triggerOtp was not called before calling getPermenantOtpToken()');
|
|
53
53
|
}
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
if (!otpTokenResult.success) {
|
|
79
|
-
return otpTokenResult;
|
|
80
|
-
}
|
|
81
|
-
return { success: true, longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken };
|
|
54
|
+
debug('Requesting OTP token');
|
|
55
|
+
const otpVerifyResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/otp/verify`, {
|
|
56
|
+
otpContext: this.otpContext,
|
|
57
|
+
otpCode
|
|
58
|
+
});
|
|
59
|
+
const {
|
|
60
|
+
resultData: {
|
|
61
|
+
otpToken
|
|
62
|
+
}
|
|
63
|
+
} = otpVerifyResponse;
|
|
64
|
+
return {
|
|
65
|
+
success: true,
|
|
66
|
+
longTermTwoFactorAuthToken: otpToken
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
async resolveOtpToken(credentials) {
|
|
70
|
+
if ('otpLongTermToken' in credentials) {
|
|
71
|
+
if (!credentials.otpLongTermToken) {
|
|
72
|
+
return (0, _errors.createGenericError)('Invalid otpLongTermToken');
|
|
73
|
+
}
|
|
74
|
+
return {
|
|
75
|
+
success: true,
|
|
76
|
+
longTermTwoFactorAuthToken: credentials.otpLongTermToken
|
|
77
|
+
};
|
|
82
78
|
}
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
const getIdTokenResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/getIdToken`, {
|
|
90
|
-
otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,
|
|
91
|
-
email: credentials.email,
|
|
92
|
-
pass: credentials.password,
|
|
93
|
-
pinCode: '',
|
|
94
|
-
});
|
|
95
|
-
const { resultData: { idToken }, } = getIdTokenResponse;
|
|
96
|
-
debug('Requesting session token');
|
|
97
|
-
const getSessionTokenResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/sessions/token`, {
|
|
98
|
-
idToken,
|
|
99
|
-
pass: credentials.password,
|
|
100
|
-
});
|
|
101
|
-
const { resultData: { accessToken }, } = getSessionTokenResponse;
|
|
102
|
-
this.accessToken = accessToken;
|
|
103
|
-
return {
|
|
104
|
-
success: true,
|
|
105
|
-
persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken,
|
|
106
|
-
};
|
|
79
|
+
if (!credentials.otpCodeRetriever) {
|
|
80
|
+
return {
|
|
81
|
+
success: false,
|
|
82
|
+
errorType: _errors.ScraperErrorTypes.TwoFactorRetrieverMissing,
|
|
83
|
+
errorMessage: 'otpCodeRetriever is required when otpPermanentToken is not provided'
|
|
84
|
+
};
|
|
107
85
|
}
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
const { movements: { movements: newMovements, pagination }, } = await (0, fetch_1.fetchGraphql)(GRAPHQL_API_URL, one_zero_queries_1.GET_MOVEMENTS, {
|
|
116
|
-
portfolioId: portfolio.portfolioId,
|
|
117
|
-
accountId: account.accountId,
|
|
118
|
-
language: 'HEBREW',
|
|
119
|
-
pagination: {
|
|
120
|
-
cursor,
|
|
121
|
-
limit: 50,
|
|
122
|
-
},
|
|
123
|
-
}, { authorization: `Bearer ${this.accessToken}` });
|
|
124
|
-
movements.unshift(...newMovements);
|
|
125
|
-
cursor = pagination.cursor;
|
|
126
|
-
if (!pagination.hasMore) {
|
|
127
|
-
break;
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());
|
|
131
|
-
const matchingMovements = movements.filter(movement => new Date(movement.movementTimestamp) >= startDate);
|
|
132
|
-
return {
|
|
133
|
-
accountNumber: portfolio.portfolioNum,
|
|
134
|
-
balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),
|
|
135
|
-
txns: matchingMovements.map((movement) => {
|
|
136
|
-
const hasInstallments = movement.transaction?.enrichment?.recurrences?.some(x => x.isRecurrent);
|
|
137
|
-
const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;
|
|
138
|
-
return {
|
|
139
|
-
identifier: movement.movementId,
|
|
140
|
-
date: movement.valueDate,
|
|
141
|
-
chargedAmount: +movement.movementAmount * modifier,
|
|
142
|
-
chargedCurrency: movement.movementCurrency,
|
|
143
|
-
originalAmount: +movement.movementAmount * modifier,
|
|
144
|
-
originalCurrency: movement.movementCurrency,
|
|
145
|
-
description: this.sanitizeHebrew(movement.description),
|
|
146
|
-
processedDate: movement.movementTimestamp,
|
|
147
|
-
status: transactions_1.TransactionStatuses.Completed,
|
|
148
|
-
type: hasInstallments ? transactions_1.TransactionTypes.Installments : transactions_1.TransactionTypes.Normal,
|
|
149
|
-
};
|
|
150
|
-
}),
|
|
151
|
-
};
|
|
86
|
+
if (!credentials.phoneNumber) {
|
|
87
|
+
return (0, _errors.createGenericError)('phoneNumber is required when providing a otpCodeRetriever callback');
|
|
88
|
+
}
|
|
89
|
+
debug('Triggering user supplied otpCodeRetriever callback');
|
|
90
|
+
const triggerResult = await this.triggerTwoFactorAuth(credentials.phoneNumber);
|
|
91
|
+
if (!triggerResult.success) {
|
|
92
|
+
return triggerResult;
|
|
152
93
|
}
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
94
|
+
const otpCode = await credentials.otpCodeRetriever();
|
|
95
|
+
const otpTokenResult = await this.getLongTermTwoFactorToken(otpCode);
|
|
96
|
+
if (!otpTokenResult.success) {
|
|
97
|
+
return otpTokenResult;
|
|
98
|
+
}
|
|
99
|
+
return {
|
|
100
|
+
success: true,
|
|
101
|
+
longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
async login(credentials) {
|
|
105
|
+
const otpTokenResult = await this.resolveOtpToken(credentials);
|
|
106
|
+
if (!otpTokenResult.success) {
|
|
107
|
+
return otpTokenResult;
|
|
108
|
+
}
|
|
109
|
+
debug('Requesting id token');
|
|
110
|
+
const getIdTokenResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/getIdToken`, {
|
|
111
|
+
otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,
|
|
112
|
+
email: credentials.email,
|
|
113
|
+
pass: credentials.password,
|
|
114
|
+
pinCode: ''
|
|
115
|
+
});
|
|
116
|
+
const {
|
|
117
|
+
resultData: {
|
|
118
|
+
idToken
|
|
119
|
+
}
|
|
120
|
+
} = getIdTokenResponse;
|
|
121
|
+
debug('Requesting session token');
|
|
122
|
+
const getSessionTokenResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/sessions/token`, {
|
|
123
|
+
idToken,
|
|
124
|
+
pass: credentials.password
|
|
125
|
+
});
|
|
126
|
+
const {
|
|
127
|
+
resultData: {
|
|
128
|
+
accessToken
|
|
129
|
+
}
|
|
130
|
+
} = getSessionTokenResponse;
|
|
131
|
+
this.accessToken = accessToken;
|
|
132
|
+
return {
|
|
133
|
+
success: true,
|
|
134
|
+
persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
async fetchPortfolioMovements(portfolio, startDate) {
|
|
138
|
+
// TODO: Find out if we need the other accounts, there seems to always be one
|
|
139
|
+
const account = portfolio.accounts[0];
|
|
140
|
+
let cursor = null;
|
|
141
|
+
const movements = [];
|
|
142
|
+
while (!movements.length || new Date(movements[0].movementTimestamp) >= startDate) {
|
|
143
|
+
debug(`Fetching transactions for account ${portfolio.portfolioNum}...`);
|
|
144
|
+
const {
|
|
145
|
+
movements: {
|
|
146
|
+
movements: newMovements,
|
|
147
|
+
pagination
|
|
160
148
|
}
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
index += start - index;
|
|
169
|
-
const reversed = [...plainString.substring(start, end)].reverse();
|
|
170
|
-
out.push(...reversed);
|
|
171
|
-
index += end - start;
|
|
149
|
+
} = await (0, _fetch.fetchGraphql)(GRAPHQL_API_URL, _oneZeroQueries.GET_MOVEMENTS, {
|
|
150
|
+
portfolioId: portfolio.portfolioId,
|
|
151
|
+
accountId: account.accountId,
|
|
152
|
+
language: 'HEBREW',
|
|
153
|
+
pagination: {
|
|
154
|
+
cursor,
|
|
155
|
+
limit: 50
|
|
172
156
|
}
|
|
173
|
-
|
|
174
|
-
|
|
157
|
+
}, {
|
|
158
|
+
authorization: `Bearer ${this.accessToken}`
|
|
159
|
+
});
|
|
160
|
+
movements.unshift(...newMovements);
|
|
161
|
+
cursor = pagination.cursor;
|
|
162
|
+
if (!pagination.hasMore) {
|
|
163
|
+
break;
|
|
164
|
+
}
|
|
175
165
|
}
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
const
|
|
183
|
-
|
|
184
|
-
const result = await (0, fetch_1.fetchGraphql)(GRAPHQL_API_URL, one_zero_queries_1.GET_CUSTOMER, {}, { authorization: `Bearer ${this.accessToken}` });
|
|
185
|
-
const portfolios = result.customer.flatMap(customer => customer.portfolios || []);
|
|
166
|
+
movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());
|
|
167
|
+
const matchingMovements = movements.filter(movement => new Date(movement.movementTimestamp) >= startDate);
|
|
168
|
+
return {
|
|
169
|
+
accountNumber: portfolio.portfolioNum,
|
|
170
|
+
balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),
|
|
171
|
+
txns: matchingMovements.map(movement => {
|
|
172
|
+
const hasInstallments = movement.transaction?.enrichment?.recurrences?.some(x => x.isRecurrent);
|
|
173
|
+
const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;
|
|
186
174
|
return {
|
|
187
|
-
|
|
188
|
-
|
|
175
|
+
identifier: movement.movementId,
|
|
176
|
+
date: movement.valueDate,
|
|
177
|
+
chargedAmount: +movement.movementAmount * modifier,
|
|
178
|
+
chargedCurrency: movement.movementCurrency,
|
|
179
|
+
originalAmount: +movement.movementAmount * modifier,
|
|
180
|
+
originalCurrency: movement.movementCurrency,
|
|
181
|
+
description: this.sanitizeHebrew(movement.description),
|
|
182
|
+
processedDate: movement.movementTimestamp,
|
|
183
|
+
status: _transactions.TransactionStatuses.Completed,
|
|
184
|
+
type: hasInstallments ? _transactions.TransactionTypes.Installments : _transactions.TransactionTypes.Normal
|
|
189
185
|
};
|
|
186
|
+
})
|
|
187
|
+
};
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
/**
|
|
191
|
+
* one zero hebrew strings are reversed with a unicode control character that forces display in LTR order
|
|
192
|
+
* We need to remove the unicode control character, and then reverse hebrew substrings inside the string
|
|
193
|
+
*/
|
|
194
|
+
sanitizeHebrew(text) {
|
|
195
|
+
if (!text.includes('\u202d')) {
|
|
196
|
+
return text.trim();
|
|
197
|
+
}
|
|
198
|
+
const plainString = text.replace(/\u202d/gi, '').trim();
|
|
199
|
+
const hebrewSubStringsRanges = [...plainString.matchAll(HEBREW_WORDS_REGEX)];
|
|
200
|
+
const rangesToReverse = hebrewSubStringsRanges.map(str => ({
|
|
201
|
+
start: str.index,
|
|
202
|
+
end: str.index + str[0].length
|
|
203
|
+
}));
|
|
204
|
+
const out = [];
|
|
205
|
+
let index = 0;
|
|
206
|
+
for (const {
|
|
207
|
+
start,
|
|
208
|
+
end
|
|
209
|
+
} of rangesToReverse) {
|
|
210
|
+
out.push(...plainString.substring(index, start));
|
|
211
|
+
index += start - index;
|
|
212
|
+
const reversed = [...plainString.substring(start, end)].reverse();
|
|
213
|
+
out.push(...reversed);
|
|
214
|
+
index += end - start;
|
|
215
|
+
}
|
|
216
|
+
out.push(...plainString.substring(index, plainString.length));
|
|
217
|
+
return out.join('');
|
|
218
|
+
}
|
|
219
|
+
async fetchData() {
|
|
220
|
+
if (!this.accessToken) {
|
|
221
|
+
return (0, _errors.createGenericError)('login() was not called');
|
|
190
222
|
}
|
|
223
|
+
const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
|
|
224
|
+
const startDate = this.options.startDate || defaultStartMoment.toDate();
|
|
225
|
+
const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
|
|
226
|
+
debug('Fetching account list');
|
|
227
|
+
const result = await (0, _fetch.fetchGraphql)(GRAPHQL_API_URL, _oneZeroQueries.GET_CUSTOMER, {}, {
|
|
228
|
+
authorization: `Bearer ${this.accessToken}`
|
|
229
|
+
});
|
|
230
|
+
const portfolios = result.customer.flatMap(customer => customer.portfolios || []);
|
|
231
|
+
return {
|
|
232
|
+
success: true,
|
|
233
|
+
accounts: await Promise.all(portfolios.map(portfolio => this.fetchPortfolioMovements(portfolio, startMoment.toDate())))
|
|
234
|
+
};
|
|
235
|
+
}
|
|
191
236
|
}
|
|
192
237
|
exports.default = OneZeroScraper;
|
|
193
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"one-zero.js","sourceRoot":"","sources":["../../src/scrapers/one-zero.ts"],"names":[],"mappings":"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","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"]}
|
|
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":";;;;;;AAAA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AACA,IAAAC,MAAA,GAAAD,OAAA;AACA,IAAAE,MAAA,GAAAF,OAAA;AACA,IAAAG,aAAA,GAAAH,OAAA;AAMA,IAAAI,YAAA,GAAAJ,OAAA;AACA,IAAAK,OAAA,GAAAL,OAAA;AAOA,IAAAM,eAAA,GAAAN,OAAA;AAAiE,SAAAD,uBAAAQ,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAEjE,MAAMG,kBAAkB,GAAG,0DAA0D;AAErF,MAAMC,KAAK,GAAG,IAAAC,eAAQ,EAAC,UAAU,CAAC;AA6DlC,MAAMC,mBAAmB,GAAG,mCAAmC;AAE/D,MAAMC,eAAe,GAAG,kDAAkD;AAY3D,MAAMC,cAAc,SAASC,wBAAW,CAA6B;EAKlF,MAAMC,oBAAoBA,CAACC,WAAmB,EAA8C;IAC1F,IAAI,CAACA,WAAW,CAACC,UAAU,CAAC,GAAG,CAAC,EAAE;MAChC,OAAO,IAAAC,0BAAkB,EACvB,8FACF,CAAC;IACH;IAEAT,KAAK,CAAC,uBAAuB,CAAC;IAC9B,MAAMU,mBAAmB,GAAG,MAAM,IAAAC,gBAAS,EAAC,GAAGT,mBAAmB,gBAAgB,EAAE;MAClFU,WAAW,EAAE,QAAQ;MACrBC,EAAE,EAAE;IACN,CAAC,CAAC;IAEF,MAAM;MACJC,UAAU,EAAE;QAAEC;MAAY;IAC5B,CAAC,GAAGL,mBAAmB;IAEvBV,KAAK,CAAC,+BAA+BO,WAAW,EAAE,CAAC;IAEnD,MAAMS,kBAAkB,GAAG,MAAM,IAAAL,gBAAS,EAAC,GAAGT,mBAAmB,cAAc,EAAE;MAC/Ee,WAAW,EAAEV,WAAW;MACxBQ,WAAW;MACXG,UAAU,EAAE;IACd,CAAC,CAAC;IAEF,MAAM;MACJJ,UAAU,EAAE;QAAEK;MAAW;IAC3B,CAAC,GAAGH,kBAAkB;IAEtB,IAAI,CAACG,UAAU,GAAGA,UAAU;IAE5B,OAAO;MACLC,OAAO,EAAE;IACX,CAAC;EACH;EAEA,MAAaC,yBAAyBA,CAACC,OAAe,EAAmD;IACvG,IAAI,CAAC,IAAI,CAACH,UAAU,EAAE;MACpB,OAAO,IAAAV,0BAAkB,EAAC,iEAAiE,CAAC;IAC9F;IAEAT,KAAK,CAAC,sBAAsB,CAAC;IAC7B,MAAMuB,iBAAiB,GAAG,MAAM,IAAAZ,gBAAS,EAAC,GAAGT,mBAAmB,aAAa,EAAE;MAC7EiB,UAAU,EAAE,IAAI,CAACA,UAAU;MAC3BG;IACF,CAAC,CAAC;IAEF,MAAM;MACJR,UAAU,EAAE;QAAEU;MAAS;IACzB,CAAC,GAAGD,iBAAiB;IACrB,OAAO;MAAEH,OAAO,EAAE,IAAI;MAAEK,0BAA0B,EAAED;IAAS,CAAC;EAChE;EAEA,MAAcE,eAAeA,CAC3BC,WAAuC,EACU;IACjD,IAAI,kBAAkB,IAAIA,WAAW,EAAE;MACrC,IAAI,CAACA,WAAW,CAACC,gBAAgB,EAAE;QACjC,OAAO,IAAAnB,0BAAkB,EAAC,0BAA0B,CAAC;MACvD;MACA,OAAO;QAAEW,OAAO,EAAE,IAAI;QAAEK,0BAA0B,EAAEE,WAAW,CAACC;MAAiB,CAAC;IACpF;IAEA,IAAI,CAACD,WAAW,CAACE,gBAAgB,EAAE;MACjC,OAAO;QACLT,OAAO,EAAE,KAAK;QACdU,SAAS,EAAEC,yBAAiB,CAACC,yBAAyB;QACtDC,YAAY,EAAE;MAChB,CAAC;IACH;IAEA,IAAI,CAACN,WAAW,CAACpB,WAAW,EAAE;MAC5B,OAAO,IAAAE,0BAAkB,EAAC,oEAAoE,CAAC;IACjG;IAEAT,KAAK,CAAC,oDAAoD,CAAC;IAC3D,MAAMkC,aAAa,GAAG,MAAM,IAAI,CAAC5B,oBAAoB,CAACqB,WAAW,CAACpB,WAAW,CAAC;IAE9E,IAAI,CAAC2B,aAAa,CAACd,OAAO,EAAE;MAC1B,OAAOc,aAAa;IACtB;IAEA,MAAMZ,OAAO,GAAG,MAAMK,WAAW,CAACE,gBAAgB,CAAC,CAAC;IAEpD,MAAMM,cAAc,GAAG,MAAM,IAAI,CAACd,yBAAyB,CAACC,OAAO,CAAC;IACpE,IAAI,CAACa,cAAc,CAACf,OAAO,EAAE;MAC3B,OAAOe,cAAc;IACvB;IAEA,OAAO;MAAEf,OAAO,EAAE,IAAI;MAAEK,0BAA0B,EAAEU,cAAc,CAACV;IAA2B,CAAC;EACjG;EAEA,MAAMW,KAAKA,CAACT,WAAuC,EAA+B;IAChF,MAAMQ,cAAc,GAAG,MAAM,IAAI,CAACT,eAAe,CAACC,WAAW,CAAC;IAC9D,IAAI,CAACQ,cAAc,CAACf,OAAO,EAAE;MAC3B,OAAOe,cAAc;IACvB;IAEAnC,KAAK,CAAC,qBAAqB,CAAC;IAC5B,MAAMqC,kBAAkB,GAAG,MAAM,IAAA1B,gBAAS,EAAC,GAAGT,mBAAmB,aAAa,EAAE;MAC9EoC,WAAW,EAAEH,cAAc,CAACV,0BAA0B;MACtDc,KAAK,EAAEZ,WAAW,CAACY,KAAK;MACxBC,IAAI,EAAEb,WAAW,CAACc,QAAQ;MAC1BC,OAAO,EAAE;IACX,CAAC,CAAC;IAEF,MAAM;MACJ5B,UAAU,EAAE;QAAE6B;MAAQ;IACxB,CAAC,GAAGN,kBAAkB;IAEtBrC,KAAK,CAAC,0BAA0B,CAAC;IAEjC,MAAM4C,uBAAuB,GAAG,MAAM,IAAAjC,gBAAS,EAAC,GAAGT,mBAAmB,iBAAiB,EAAE;MACvFyC,OAAO;MACPH,IAAI,EAAEb,WAAW,CAACc;IACpB,CAAC,CAAC;IAEF,MAAM;MACJ3B,UAAU,EAAE;QAAE+B;MAAY;IAC5B,CAAC,GAAGD,uBAAuB;IAE3B,IAAI,CAACC,WAAW,GAAGA,WAAW;IAE9B,OAAO;MACLzB,OAAO,EAAE,IAAI;MACb0B,kBAAkB,EAAEX,cAAc,CAACV;IACrC,CAAC;EACH;EAEA,MAAcsB,uBAAuBA,CAACC,SAAoB,EAAEC,SAAe,EAAgC;IACzG;IACA,MAAMC,OAAO,GAAGF,SAAS,CAACG,QAAQ,CAAC,CAAC,CAAC;IACrC,IAAIC,MAAM,GAAG,IAAI;IACjB,MAAMC,SAAS,GAAG,EAAE;IAEpB,OAAO,CAACA,SAAS,CAACC,MAAM,IAAI,IAAIC,IAAI,CAACF,SAAS,CAAC,CAAC,CAAC,CAACG,iBAAiB,CAAC,IAAIP,SAAS,EAAE;MACjFjD,KAAK,CAAC,qCAAqCgD,SAAS,CAACS,YAAY,KAAK,CAAC;MACvE,MAAM;QACJJ,SAAS,EAAE;UAAEA,SAAS,EAAEK,YAAY;UAAEC;QAAW;MACoB,CAAC,GAAG,MAAM,IAAAC,mBAAY,EAC3FzD,eAAe,EACf0D,6BAAa,EACb;QACEC,WAAW,EAAEd,SAAS,CAACc,WAAW;QAClCC,SAAS,EAAEb,OAAO,CAACa,SAAS;QAC5BC,QAAQ,EAAE,QAAQ;QAClBL,UAAU,EAAE;UACVP,MAAM;UACNa,KAAK,EAAE;QACT;MACF,CAAC,EACD;QAAEC,aAAa,EAAE,UAAU,IAAI,CAACrB,WAAW;MAAG,CAChD,CAAC;MAEDQ,SAAS,CAACc,OAAO,CAAC,GAAGT,YAAY,CAAC;MAClCN,MAAM,GAAGO,UAAU,CAACP,MAAM;MAC1B,IAAI,CAACO,UAAU,CAACS,OAAO,EAAE;QACvB;MACF;IACF;IAEAf,SAAS,CAACgB,IAAI,CAAC,CAACC,CAAC,EAAEC,CAAC,KAAK,IAAIhB,IAAI,CAACe,CAAC,CAACd,iBAAiB,CAAC,CAACgB,OAAO,CAAC,CAAC,GAAG,IAAIjB,IAAI,CAACgB,CAAC,CAACf,iBAAiB,CAAC,CAACgB,OAAO,CAAC,CAAC,CAAC;IAE3G,MAAMC,iBAAiB,GAAGpB,SAAS,CAACqB,MAAM,CAACC,QAAQ,IAAI,IAAIpB,IAAI,CAACoB,QAAQ,CAACnB,iBAAiB,CAAC,IAAIP,SAAS,CAAC;IACzG,OAAO;MACL2B,aAAa,EAAE5B,SAAS,CAACS,YAAY;MACrCoB,OAAO,EAAE,CAACxB,SAAS,CAACC,MAAM,GAAG,CAAC,GAAGwB,UAAU,CAACzB,SAAS,CAACA,SAAS,CAACC,MAAM,GAAG,CAAC,CAAC,CAACyB,cAAc,CAAC;MAC3FC,IAAI,EAAEP,iBAAiB,CAACQ,GAAG,CAAEN,QAAQ,IAA0B;QAC7D,MAAMO,eAAe,GAAGP,QAAQ,CAACQ,WAAW,EAAEC,UAAU,EAAEC,WAAW,EAAEC,IAAI,CAAChB,CAAC,IAAIA,CAAC,CAACiB,WAAW,CAAC;QAC/F,MAAMC,QAAQ,GAAGb,QAAQ,CAACc,WAAW,KAAK,OAAO,GAAG,CAAC,CAAC,GAAG,CAAC;QAC1D,OAAO;UACLC,UAAU,EAAEf,QAAQ,CAACgB,UAAU;UAC/BC,IAAI,EAAEjB,QAAQ,CAACkB,SAAS;UACxBC,aAAa,EAAE,CAACnB,QAAQ,CAACoB,cAAc,GAAGP,QAAQ;UAClDQ,eAAe,EAAErB,QAAQ,CAACsB,gBAAgB;UAC1CC,cAAc,EAAE,CAACvB,QAAQ,CAACoB,cAAc,GAAGP,QAAQ;UACnDW,gBAAgB,EAAExB,QAAQ,CAACsB,gBAAgB;UAC3CG,WAAW,EAAE,IAAI,CAACC,cAAc,CAAC1B,QAAQ,CAACyB,WAAW,CAAC;UACtDE,aAAa,EAAE3B,QAAQ,CAACnB,iBAAiB;UACzC+C,MAAM,EAAEC,iCAAmB,CAACC,SAAS;UACrCC,IAAI,EAAExB,eAAe,GAAGyB,8BAAgB,CAACC,YAAY,GAAGD,8BAAgB,CAACE;QAC3E,CAAC;MACH,CAAC;IACH,CAAC;EACH;;EAEA;AACF;AACA;AACA;EACUR,cAAcA,CAACS,IAAY,EAAE;IACnC,IAAI,CAACA,IAAI,CAACC,QAAQ,CAAC,QAAQ,CAAC,EAAE;MAC5B,OAAOD,IAAI,CAACE,IAAI,CAAC,CAAC;IACpB;IAEA,MAAMC,WAAW,GAAGH,IAAI,CAACI,OAAO,CAAC,UAAU,EAAE,EAAE,CAAC,CAACF,IAAI,CAAC,CAAC;IACvD,MAAMG,sBAAsB,GAAG,CAAC,GAAGF,WAAW,CAACG,QAAQ,CAACrH,kBAAkB,CAAC,CAAC;IAC5E,MAAMsH,eAAe,GAAGF,sBAAsB,CAAClC,GAAG,CAACqC,GAAG,KAAK;MAAEC,KAAK,EAAED,GAAG,CAACE,KAAM;MAAEC,GAAG,EAAEH,GAAG,CAACE,KAAK,GAAIF,GAAG,CAAC,CAAC,CAAC,CAAChE;IAAO,CAAC,CAAC,CAAC;IACnH,MAAMoE,GAAG,GAAG,EAAE;IACd,IAAIF,KAAK,GAAG,CAAC;IAEb,KAAK,MAAM;MAAED,KAAK;MAAEE;IAAI,CAAC,IAAIJ,eAAe,EAAE;MAC5CK,GAAG,CAACC,IAAI,CAAC,GAAGV,WAAW,CAACW,SAAS,CAACJ,KAAK,EAAED,KAAK,CAAC,CAAC;MAChDC,KAAK,IAAID,KAAK,GAAGC,KAAK;MACtB,MAAMK,QAAQ,GAAG,CAAC,GAAGZ,WAAW,CAACW,SAAS,CAACL,KAAK,EAAEE,GAAG,CAAC,CAAC,CAACK,OAAO,CAAC,CAAC;MACjEJ,GAAG,CAACC,IAAI,CAAC,GAAGE,QAAQ,CAAC;MACrBL,KAAK,IAAIC,GAAG,GAAGF,KAAK;IACtB;IAEAG,GAAG,CAACC,IAAI,CAAC,GAAGV,WAAW,CAACW,SAAS,CAACJ,KAAK,EAAEP,WAAW,CAAC3D,MAAM,CAAC,CAAC;IAE7D,OAAOoE,GAAG,CAACK,IAAI,CAAC,EAAE,CAAC;EACrB;EAEA,MAAMC,SAASA,CAAA,EAAmC;IAChD,IAAI,CAAC,IAAI,CAACnF,WAAW,EAAE;MACrB,OAAO,IAAApC,0BAAkB,EAAC,wBAAwB,CAAC;IACrD;IAEA,MAAMwH,kBAAkB,GAAG,IAAAC,eAAM,EAAC,CAAC,CAACC,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC,CAACC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC;IACtE,MAAMnF,SAAS,GAAG,IAAI,CAACoF,OAAO,CAACpF,SAAS,IAAIgF,kBAAkB,CAACK,MAAM,CAAC,CAAC;IACvE,MAAMC,WAAW,GAAGL,eAAM,CAACM,GAAG,CAACP,kBAAkB,EAAE,IAAAC,eAAM,EAACjF,SAAS,CAAC,CAAC;IAErEjD,KAAK,CAAC,uBAAuB,CAAC;IAC9B,MAAMyI,MAAM,GAAG,MAAM,IAAA7E,mBAAY,EAC/BzD,eAAe,EACfuI,4BAAY,EACZ,CAAC,CAAC,EACF;MAAExE,aAAa,EAAE,UAAU,IAAI,CAACrB,WAAW;IAAG,CAChD,CAAC;IACD,MAAM8F,UAAU,GAAGF,MAAM,CAACG,QAAQ,CAACC,OAAO,CAACD,QAAQ,IAAIA,QAAQ,CAACD,UAAU,IAAI,EAAE,CAAC;IAEjF,OAAO;MACLvH,OAAO,EAAE,IAAI;MACb+B,QAAQ,EAAE,MAAM2F,OAAO,CAACC,GAAG,CACzBJ,UAAU,CAAC1D,GAAG,CAACjC,SAAS,IAAI,IAAI,CAACD,uBAAuB,CAACC,SAAS,EAAEuF,WAAW,CAACD,MAAM,CAAC,CAAC,CAAC,CAC3F;IACF,CAAC;EACH;AACF;AAACU,OAAA,CAAAlJ,OAAA,GAAAM,cAAA","ignoreList":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|