@novha/calc-engines 1.6.2 → 1.6.4

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.
@@ -3,7 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.CanadaMortgageServiceImpl = void 0;
4
4
  class CanadaMortgageServiceImpl {
5
5
  calculate(input, rules) {
6
- const { propertyPrice, downPayment, interestRate, amortizationYears, paymentFrequency } = input;
6
+ const { propertyPrice, downPayment, amortizationYears, paymentFrequency } = input;
7
+ const interestRate = input.interestRate / 100;
7
8
  /* -----------------------------
8
9
  1. Loan Amount & LTV
9
10
  ------------------------------ */
@@ -53,10 +54,24 @@ class CanadaMortgageServiceImpl {
53
54
  loanAmount: loanAmount,
54
55
  insurancePremium: insurancePremium,
55
56
  totalMortgage: totalMortgage,
56
- paymentAmount: paymentAmount,
57
+ monthlyPaymentAmount: paymentAmount,
57
58
  totalInterestPaid: totalInterestPaid,
58
59
  totalPaid: totalPaid,
59
- amortizationSchedule: amortizationSchedule
60
+ amortizationSchedule: amortizationSchedule,
61
+ otherFees: {
62
+ notaryFees: {
63
+ value: 0,
64
+ label: 'NOTARY_FEES'
65
+ },
66
+ bankFees: {
67
+ value: 0,
68
+ label: 'BANK_FEES'
69
+ },
70
+ monthlyInsuranceFees: {
71
+ value: insurancePremium,
72
+ label: 'INSURANCE_PREMIUM'
73
+ }
74
+ }
60
75
  };
61
76
  }
62
77
  /* ======================================================
@@ -106,4 +121,4 @@ class CanadaMortgageServiceImpl {
106
121
  }
107
122
  }
108
123
  exports.CanadaMortgageServiceImpl = CanadaMortgageServiceImpl;
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CanadaMortgageServiceImpl.js","sourceRoot":"","sources":["../../../src/mortgage/canada/CanadaMortgageServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { CanadaMortgageService } from './CanadaMortgageService';\nimport {\n  MortgageRules,\n  MortgageCalculationInput,\n  MortgageCalculationResult,\n  AmortizationScheduleItem\n} from './domain/types';\n\nexport class CanadaMortgageServiceImpl implements CanadaMortgageService {\n\n  public calculate(input: MortgageCalculationInput, rules: MortgageRules): MortgageCalculationResult {\n\n    const {\n      propertyPrice,\n      downPayment,\n      interestRate,\n      amortizationYears,\n      paymentFrequency\n    } = input;\n\n    /* -----------------------------\n       1. Loan Amount & LTV\n    ------------------------------ */\n\n    const loanAmount = propertyPrice - downPayment;\n    if (loanAmount <= 0) {\n      throw new Error('Invalid loan amount');\n    }\n\n    const ltv = loanAmount / propertyPrice;\n\n    /* -----------------------------\n       2. CMHC Insurance\n    ------------------------------ */\n\n    let insurancePremium = 0;\n\n    if (ltv > rules.mortgageInsurance.requiredBelowLtv) {\n      const premiumRule = rules.mortgageInsurance.premiumRates\n        .find(r => ltv <= r.maxLtv);\n\n      if (!premiumRule) {\n        throw new Error('LTV exceeds maximum insurable limit');\n      }\n\n      insurancePremium = loanAmount * premiumRule.rate;\n    }\n\n    const totalMortgage = rules.mortgageInsurance.premiumAddedToLoan\n      ? loanAmount + insurancePremium\n      : loanAmount;\n\n    /* -----------------------------\n       3. Interest Rate Conversion\n       (Canada semi-annual compounding)\n    ------------------------------ */\n\n    const periodicRate = this.convertCanadianRate(\n      interestRate,\n      rules.interest.compounding,\n      rules.paymentFrequencyRules[paymentFrequency].paymentsPerYear\n    );\n\n    /* -----------------------------\n       4. Payment Frequency\n    ------------------------------ */\n\n    const frequencyRule = rules.paymentFrequencyRules[paymentFrequency];\n    const paymentsPerYear = frequencyRule.paymentsPerYear;\n    const totalPayments = amortizationYears * paymentsPerYear;\n\n    /* -----------------------------\n       5. Mortgage Payment Formula\n       P = L × [ r(1+r)^n ] / [ (1+r)^n − 1 ]\n    ------------------------------ */\n\n    const paymentAmount = this.calculatePayment(\n      totalMortgage,\n      periodicRate,\n      totalPayments\n    );\n\n    /* -----------------------------\n       6. Totals\n    ------------------------------ */\n\n    const totalPaid = paymentAmount * totalPayments;\n    const totalInterestPaid = totalPaid - totalMortgage;\n\n    const amortizationSchedule = this.calculateAmortizationSchedule(\n      totalMortgage,\n      periodicRate,\n      paymentAmount,\n      totalPayments,\n      amortizationYears,\n      paymentsPerYear\n    );\n\n    return {\n      loanAmount: loanAmount,\n      insurancePremium: insurancePremium,\n      totalMortgage: totalMortgage,\n      paymentAmount: paymentAmount,\n      totalInterestPaid: totalInterestPaid,\n      totalPaid: totalPaid,\n      amortizationSchedule: amortizationSchedule\n    };\n  }\n\n  /* ======================================================\n     Helper Methods\n  ====================================================== */\n\n  private calculateAmortizationSchedule(\n    principal: number,\n    periodicRate: number,\n    paymentAmount: number,\n    totalPayments: number,\n    amortizationYears: number,\n    paymentsPerYear: number\n  ): AmortizationScheduleItem[] {\n    const schedule: AmortizationScheduleItem[] = [];\n    let balance = principal;\n\n    for (let year = 1; year <= amortizationYears; year++) {\n      let yearlyPrincipal = 0;\n      let yearlyInterest = 0;\n\n      const paymentsInYear = Math.min(paymentsPerYear, totalPayments - (year - 1) * paymentsPerYear);\n\n      for (let payment = 1; payment <= paymentsInYear; payment++) {\n        const interestPayment = balance * periodicRate;\n        const principalPayment = paymentAmount - interestPayment;\n\n        yearlyInterest += interestPayment;\n        yearlyPrincipal += principalPayment;\n        balance -= principalPayment;\n      }\n\n      schedule.push({\n        year,\n        principal: yearlyPrincipal,\n        interest: yearlyInterest,\n        balance: Math.max(0, balance)\n      });\n\n      if (balance <= 0) break;\n    }\n\n    return schedule;\n  }\n\n  private convertCanadianRate(\n    annualRate: number,\n    compounding: 'SEMI_ANNUAL' | 'ANNUAL' | 'MONTHLY',\n    paymentsPerYear: number\n  ): number {\n\n    if (compounding !== 'SEMI_ANNUAL') {\n      throw new Error('Only Canadian semi-annual compounding supported');\n    }\n\n    // Canadian standard conversion\n    const semiAnnualRate = annualRate / 2;\n    const effectiveAnnualRate = Math.pow(1 + semiAnnualRate, 2) - 1;\n\n    return Math.pow(1 + effectiveAnnualRate, 1 / paymentsPerYear) - 1;\n  }\n\n  private calculatePayment(principal: number, rate: number, periods: number): number {\n    if (rate === 0) {\n      return principal / periods;\n    }\n\n    return principal *\n      (rate * Math.pow(1 + rate, periods)) /\n      (Math.pow(1 + rate, periods) - 1);\n  }\n}\n"]}
124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CanadaMortgageServiceImpl.js","sourceRoot":"","sources":["../../../src/mortgage/canada/CanadaMortgageServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { CanadaMortgageService } from './CanadaMortgageService';\nimport {\n  MortgageRules,\n  MortgageCalculationInput,\n  MortgageCalculationResult,\n  AmortizationScheduleItem\n} from './domain/types';\n\nexport class CanadaMortgageServiceImpl implements CanadaMortgageService {\n\n  public calculate(input: MortgageCalculationInput, rules: MortgageRules): MortgageCalculationResult {\n\n    const {\n      propertyPrice,\n      downPayment,\n      amortizationYears,\n      paymentFrequency\n    } = input;\n\n    const interestRate = input.interestRate / 100;\n\n    /* -----------------------------\n       1. Loan Amount & LTV\n    ------------------------------ */\n\n    const loanAmount = propertyPrice - downPayment;\n    if (loanAmount <= 0) {\n      throw new Error('Invalid loan amount');\n    }\n\n    const ltv = loanAmount / propertyPrice;\n\n    /* -----------------------------\n       2. CMHC Insurance\n    ------------------------------ */\n\n    let insurancePremium = 0;\n\n    if (ltv > rules.mortgageInsurance.requiredBelowLtv) {\n      const premiumRule = rules.mortgageInsurance.premiumRates\n        .find(r => ltv <= r.maxLtv);\n\n      if (!premiumRule) {\n        throw new Error('LTV exceeds maximum insurable limit');\n      }\n\n      insurancePremium = loanAmount * premiumRule.rate;\n    }\n\n    const totalMortgage = rules.mortgageInsurance.premiumAddedToLoan\n      ? loanAmount + insurancePremium\n      : loanAmount;\n\n    /* -----------------------------\n       3. Interest Rate Conversion\n       (Canada semi-annual compounding)\n    ------------------------------ */\n\n    const periodicRate = this.convertCanadianRate(\n      interestRate,\n      rules.interest.compounding,\n      rules.paymentFrequencyRules[paymentFrequency].paymentsPerYear\n    );\n\n    /* -----------------------------\n       4. Payment Frequency\n    ------------------------------ */\n\n    const frequencyRule = rules.paymentFrequencyRules[paymentFrequency];\n    const paymentsPerYear = frequencyRule.paymentsPerYear;\n    const totalPayments = amortizationYears * paymentsPerYear;\n\n    /* -----------------------------\n       5. Mortgage Payment Formula\n       P = L × [ r(1+r)^n ] / [ (1+r)^n − 1 ]\n    ------------------------------ */\n\n    const paymentAmount = this.calculatePayment(\n      totalMortgage,\n      periodicRate,\n      totalPayments\n    );\n\n    /* -----------------------------\n       6. Totals\n    ------------------------------ */\n\n    const totalPaid = paymentAmount * totalPayments;\n    const totalInterestPaid = totalPaid - totalMortgage;\n\n    const amortizationSchedule = this.calculateAmortizationSchedule(\n      totalMortgage,\n      periodicRate,\n      paymentAmount,\n      totalPayments,\n      amortizationYears,\n      paymentsPerYear\n    );\n\n    return {\n      loanAmount: loanAmount,\n      insurancePremium: insurancePremium,\n      totalMortgage: totalMortgage,\n      monthlyPaymentAmount: paymentAmount,\n      totalInterestPaid: totalInterestPaid,\n      totalPaid: totalPaid,\n      amortizationSchedule: amortizationSchedule,\n      otherFees: {\n        notaryFees: {\n            value: 0,\n            label: 'NOTARY_FEES'\n        },\n        bankFees: {\n            value: 0,\n            label: 'BANK_FEES'\n        },\n        monthlyInsuranceFees: {\n            value: insurancePremium,\n            label: 'INSURANCE_PREMIUM'\n        }\n    }\n    };\n  }\n\n  /* ======================================================\n     Helper Methods\n  ====================================================== */\n\n  private calculateAmortizationSchedule(\n    principal: number,\n    periodicRate: number,\n    paymentAmount: number,\n    totalPayments: number,\n    amortizationYears: number,\n    paymentsPerYear: number\n  ): AmortizationScheduleItem[] {\n    const schedule: AmortizationScheduleItem[] = [];\n    let balance = principal;\n\n    for (let year = 1; year <= amortizationYears; year++) {\n      let yearlyPrincipal = 0;\n      let yearlyInterest = 0;\n\n      const paymentsInYear = Math.min(paymentsPerYear, totalPayments - (year - 1) * paymentsPerYear);\n\n      for (let payment = 1; payment <= paymentsInYear; payment++) {\n        const interestPayment = balance * periodicRate;\n        const principalPayment = paymentAmount - interestPayment;\n\n        yearlyInterest += interestPayment;\n        yearlyPrincipal += principalPayment;\n        balance -= principalPayment;\n      }\n\n      schedule.push({\n        year,\n        principal: yearlyPrincipal,\n        interest: yearlyInterest,\n        balance: Math.max(0, balance)\n      });\n\n      if (balance <= 0) break;\n    }\n\n    return schedule;\n  }\n\n  private convertCanadianRate(\n    annualRate: number,\n    compounding: 'SEMI_ANNUAL' | 'ANNUAL' | 'MONTHLY',\n    paymentsPerYear: number\n  ): number {\n\n    if (compounding !== 'SEMI_ANNUAL') {\n      throw new Error('Only Canadian semi-annual compounding supported');\n    }\n\n    // Canadian standard conversion\n    const semiAnnualRate = annualRate / 2;\n    const effectiveAnnualRate = Math.pow(1 + semiAnnualRate, 2) - 1;\n\n    return Math.pow(1 + effectiveAnnualRate, 1 / paymentsPerYear) - 1;\n  }\n\n  private calculatePayment(principal: number, rate: number, periods: number): number {\n    if (rate === 0) {\n      return principal / periods;\n    }\n\n    return principal *\n      (rate * Math.pow(1 + rate, periods)) /\n      (Math.pow(1 + rate, periods) - 1);\n  }\n}\n"]}
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2UvY2FuYWRhL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUnVsZU1ldGEgfSBmcm9tIFwiLi4vLi4vLi4vc2hhcmVkL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFJ1bGVJbnB1dERlZmluaXRpb24ge1xuICAgIHR5cGU6ICdudW1iZXInIHwgJ3N0cmluZyc7XG4gICAgbWluPzogbnVtYmVyO1xuICAgIG1heD86IG51bWJlcjtcbiAgICBlbnVtPzogc3RyaW5nW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUnVsZUlucHV0cyB7XG4gICAgcHJvcGVydHlQcmljZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgICBkb3duUGF5bWVudDogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgICBpbnRlcmVzdFJhdGU6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gICAgYW1vcnRpemF0aW9uWWVhcnM6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gICAgcGF5bWVudEZyZXF1ZW5jeTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSdWxlT3V0cHV0cyB7XG4gICAgbG9hbkFtb3VudDogJ251bWJlcic7XG4gICAgaW5zdXJhbmNlUHJlbWl1bTogJ251bWJlcic7XG4gICAgdG90YWxNb3J0Z2FnZTogJ251bWJlcic7XG4gICAgcGF5bWVudEFtb3VudDogJ251bWJlcic7XG4gICAgdG90YWxJbnRlcmVzdFBhaWQ6ICdudW1iZXInO1xuICAgIHRvdGFsUGFpZDogJ251bWJlcic7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTG9hbkNvbnN0cmFpbnRzIHtcbiAgICBtYXhBbW9ydGl6YXRpb25ZZWFyczogbnVtYmVyO1xuICAgIGluc3VyZWRNYXhBbW9ydGl6YXRpb25ZZWFyczogbnVtYmVyO1xuICAgIG1pbkRvd25QYXltZW50OiB7XG4gICAgICAgIHVwVG81MDBrOiBudW1iZXI7XG4gICAgICAgIGFib3ZlNTAwazogbnVtYmVyO1xuICAgIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VJbnN1cmFuY2VSYXRlIHtcbiAgICBtYXhMdHY6IG51bWJlcjtcbiAgICByYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VJbnN1cmFuY2VSdWxlcyB7XG4gICAgcmVxdWlyZWRCZWxvd0x0djogbnVtYmVyO1xuICAgIHByZW1pdW1SYXRlczogTW9ydGdhZ2VJbnN1cmFuY2VSYXRlW107XG4gICAgcHJlbWl1bUFkZGVkVG9Mb2FuOiBib29sZWFuO1xufVxuXG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VSdWxlcyB7XG4gICAgbG9hbkNvbnN0cmFpbnRzOiBMb2FuQ29uc3RyYWludHM7XG4gICAgbW9ydGdhZ2VJbnN1cmFuY2U6IE1vcnRnYWdlSW5zdXJhbmNlUnVsZXM7XG4gICAgaW50ZXJlc3Q6IEludGVyZXN0UnVsZXM7XG4gICAgcGF5bWVudEZyZXF1ZW5jeVJ1bGVzOiBQYXltZW50RnJlcXVlbmN5UnVsZXM7XG4gICAgc3RyZXNzVGVzdDogU3RyZXNzVGVzdFJ1bGVzO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEludGVyZXN0UnVsZXMge1xuICAgIGNvbXBvdW5kaW5nOiAnU0VNSV9BTk5VQUwnIHwgJ0FOTlVBTCcgfCAnTU9OVEhMWSc7XG4gICAgY29udmVyc2lvbkZvcm11bGE6ICdDQU5BREFfU1RBTkRBUkQnIHwgc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBheW1lbnRGcmVxdWVuY3lSdWxlIHtcbiAgICBwYXltZW50c1BlclllYXI6IG51bWJlcjtcbiAgICBhY2NlbGVyYXRpb24/OiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBheW1lbnRGcmVxdWVuY3lSdWxlcyB7XG4gICAgTU9OVEhMWTogUGF5bWVudEZyZXF1ZW5jeVJ1bGU7XG4gICAgQklfV0VFS0xZOiBQYXltZW50RnJlcXVlbmN5UnVsZTtcbiAgICBBQ0NFTEVSQVRFRF9CSV9XRUVLTFk6IFBheW1lbnRGcmVxdWVuY3lSdWxlO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN0cmVzc1Rlc3RSdWxlcyB7XG4gICAgYXBwbHk6IGJvb2xlYW47XG4gICAgbWluaW11bVJhdGVCdWZmZXI6IG51bWJlcjtcbiAgICBtaW5pbXVtUXVhbGlmeWluZ1JhdGU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVTZXQge1xuICAgIG1ldGE6IFJ1bGVNZXRhO1xuICAgIGlucHV0czogUnVsZUlucHV0cztcbiAgICBydWxlczogTW9ydGdhZ2VSdWxlcztcbiAgICBvdXRwdXRzOiBSdWxlT3V0cHV0cztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUNhbGN1bGF0aW9uSW5wdXQge1xuICBwcm9wZXJ0eVByaWNlOiBudW1iZXI7XG4gIGRvd25QYXltZW50OiBudW1iZXI7XG4gIGludGVyZXN0UmF0ZTogbnVtYmVyOyAvLyBBbm51YWwgbm9taW5hbCByYXRlIChlLmcuIDAuMDUyKVxuICBhbW9ydGl6YXRpb25ZZWFyczogbnVtYmVyO1xuICBwYXltZW50RnJlcXVlbmN5OiAnTU9OVEhMWScgfCAnQklfV0VFS0xZJyB8ICdBQ0NFTEVSQVRFRF9CSV9XRUVLTFknO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEFtb3J0aXphdGlvblNjaGVkdWxlSXRlbSB7XG4gIHllYXI6IG51bWJlcjtcbiAgcHJpbmNpcGFsOiBudW1iZXI7XG4gIGludGVyZXN0OiBudW1iZXI7XG4gIGJhbGFuY2U6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUNhbGN1bGF0aW9uUmVzdWx0IHtcbiAgbG9hbkFtb3VudDogbnVtYmVyO1xuICBpbnN1cmFuY2VQcmVtaXVtOiBudW1iZXI7XG4gIHRvdGFsTW9ydGdhZ2U6IG51bWJlcjtcbiAgcGF5bWVudEFtb3VudDogbnVtYmVyO1xuICB0b3RhbEludGVyZXN0UGFpZDogbnVtYmVyO1xuICB0b3RhbFBhaWQ6IG51bWJlcjtcbiAgYW1vcnRpemF0aW9uU2NoZWR1bGU6IEFtb3J0aXphdGlvblNjaGVkdWxlSXRlbVtdO1xufVxuIl19
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2UvY2FuYWRhL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUnVsZU1ldGEgfSBmcm9tIFwiLi4vLi4vLi4vc2hhcmVkL2RvbWFpbi90eXBlc1wiO1xuaW1wb3J0IHsgT3RoZXJGZWVzIH0gZnJvbSBcIi4uLy4uL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFJ1bGVJbnB1dERlZmluaXRpb24ge1xuICAgIHR5cGU6ICdudW1iZXInIHwgJ3N0cmluZyc7XG4gICAgbWluPzogbnVtYmVyO1xuICAgIG1heD86IG51bWJlcjtcbiAgICBlbnVtPzogc3RyaW5nW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUnVsZUlucHV0cyB7XG4gICAgcHJvcGVydHlQcmljZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgICBkb3duUGF5bWVudDogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgICBpbnRlcmVzdFJhdGU6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gICAgYW1vcnRpemF0aW9uWWVhcnM6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gICAgcGF5bWVudEZyZXF1ZW5jeTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSdWxlT3V0cHV0cyB7XG4gICAgbG9hbkFtb3VudDogJ251bWJlcic7XG4gICAgaW5zdXJhbmNlUHJlbWl1bTogJ251bWJlcic7XG4gICAgdG90YWxNb3J0Z2FnZTogJ251bWJlcic7XG4gICAgcGF5bWVudEFtb3VudDogJ251bWJlcic7XG4gICAgdG90YWxJbnRlcmVzdFBhaWQ6ICdudW1iZXInO1xuICAgIHRvdGFsUGFpZDogJ251bWJlcic7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTG9hbkNvbnN0cmFpbnRzIHtcbiAgICBtYXhBbW9ydGl6YXRpb25ZZWFyczogbnVtYmVyO1xuICAgIGluc3VyZWRNYXhBbW9ydGl6YXRpb25ZZWFyczogbnVtYmVyO1xuICAgIG1pbkRvd25QYXltZW50OiB7XG4gICAgICAgIHVwVG81MDBrOiBudW1iZXI7XG4gICAgICAgIGFib3ZlNTAwazogbnVtYmVyO1xuICAgIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VJbnN1cmFuY2VSYXRlIHtcbiAgICBtYXhMdHY6IG51bWJlcjtcbiAgICByYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VJbnN1cmFuY2VSdWxlcyB7XG4gICAgcmVxdWlyZWRCZWxvd0x0djogbnVtYmVyO1xuICAgIHByZW1pdW1SYXRlczogTW9ydGdhZ2VJbnN1cmFuY2VSYXRlW107XG4gICAgcHJlbWl1bUFkZGVkVG9Mb2FuOiBib29sZWFuO1xufVxuXG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VSdWxlcyB7XG4gICAgbG9hbkNvbnN0cmFpbnRzOiBMb2FuQ29uc3RyYWludHM7XG4gICAgbW9ydGdhZ2VJbnN1cmFuY2U6IE1vcnRnYWdlSW5zdXJhbmNlUnVsZXM7XG4gICAgaW50ZXJlc3Q6IEludGVyZXN0UnVsZXM7XG4gICAgcGF5bWVudEZyZXF1ZW5jeVJ1bGVzOiBQYXltZW50RnJlcXVlbmN5UnVsZXM7XG4gICAgc3RyZXNzVGVzdDogU3RyZXNzVGVzdFJ1bGVzO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEludGVyZXN0UnVsZXMge1xuICAgIGNvbXBvdW5kaW5nOiAnU0VNSV9BTk5VQUwnIHwgJ0FOTlVBTCcgfCAnTU9OVEhMWSc7XG4gICAgY29udmVyc2lvbkZvcm11bGE6ICdDQU5BREFfU1RBTkRBUkQnIHwgc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBheW1lbnRGcmVxdWVuY3lSdWxlIHtcbiAgICBwYXltZW50c1BlclllYXI6IG51bWJlcjtcbiAgICBhY2NlbGVyYXRpb24/OiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBheW1lbnRGcmVxdWVuY3lSdWxlcyB7XG4gICAgTU9OVEhMWTogUGF5bWVudEZyZXF1ZW5jeVJ1bGU7XG4gICAgQklfV0VFS0xZOiBQYXltZW50RnJlcXVlbmN5UnVsZTtcbiAgICBBQ0NFTEVSQVRFRF9CSV9XRUVLTFk6IFBheW1lbnRGcmVxdWVuY3lSdWxlO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN0cmVzc1Rlc3RSdWxlcyB7XG4gICAgYXBwbHk6IGJvb2xlYW47XG4gICAgbWluaW11bVJhdGVCdWZmZXI6IG51bWJlcjtcbiAgICBtaW5pbXVtUXVhbGlmeWluZ1JhdGU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVTZXQge1xuICAgIG1ldGE6IFJ1bGVNZXRhO1xuICAgIGlucHV0czogUnVsZUlucHV0cztcbiAgICBydWxlczogTW9ydGdhZ2VSdWxlcztcbiAgICBvdXRwdXRzOiBSdWxlT3V0cHV0cztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUNhbGN1bGF0aW9uSW5wdXQge1xuICBwcm9wZXJ0eVByaWNlOiBudW1iZXI7XG4gIGRvd25QYXltZW50OiBudW1iZXI7XG4gIGludGVyZXN0UmF0ZTogbnVtYmVyOyAvLyBBbm51YWwgbm9taW5hbCByYXRlIChlLmcuIDAuMDUyKVxuICBhbW9ydGl6YXRpb25ZZWFyczogbnVtYmVyO1xuICBwYXltZW50RnJlcXVlbmN5OiAnTU9OVEhMWScgfCAnQklfV0VFS0xZJyB8ICdBQ0NFTEVSQVRFRF9CSV9XRUVLTFknO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEFtb3J0aXphdGlvblNjaGVkdWxlSXRlbSB7XG4gIHllYXI6IG51bWJlcjtcbiAgcHJpbmNpcGFsOiBudW1iZXI7XG4gIGludGVyZXN0OiBudW1iZXI7XG4gIGJhbGFuY2U6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUNhbGN1bGF0aW9uUmVzdWx0IHtcbiAgbG9hbkFtb3VudDogbnVtYmVyO1xuICBpbnN1cmFuY2VQcmVtaXVtOiBudW1iZXI7XG4gIHRvdGFsTW9ydGdhZ2U6IG51bWJlcjtcbiAgbW9udGhseVBheW1lbnRBbW91bnQ6IG51bWJlcjtcbiAgdG90YWxJbnRlcmVzdFBhaWQ6IG51bWJlcjtcbiAgdG90YWxQYWlkOiBudW1iZXI7XG4gIGFtb3J0aXphdGlvblNjaGVkdWxlOiBBbW9ydGl6YXRpb25TY2hlZHVsZUl0ZW1bXTtcbiAgb3RoZXJGZWVzOiBPdGhlckZlZXM7XG59XG4iXX0=
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbW9ydGdhZ2UvZG9tYWluL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIE90aGVyRmVlcyB7XG4gICAgbm90YXJ5RmVlczoge1xuICAgICAgICB2YWx1ZTogbnVtYmVyO1xuICAgICAgICBsYWJlbDogc3RyaW5nO1xuICAgIH07XG4gICAgYmFua0ZlZXM6IHtcbiAgICAgICAgdmFsdWU6IG51bWJlcjtcbiAgICAgICAgbGFiZWw6IHN0cmluZztcbiAgICB9O1xuICAgIG1vbnRobHlJbnN1cmFuY2VGZWVzOiB7XG4gICAgICAgIHZhbHVlOiBudW1iZXI7XG4gICAgICAgIGxhYmVsOiBzdHJpbmc7XG4gICAgfTtcbn0iXX0=
@@ -39,27 +39,55 @@ class FranceMortgageServiceImpl {
39
39
  maxMonthlyPayment -= monthlyInsuranceCost;
40
40
  }
41
41
  const maxLoanAmount = this.calculateLoanAmount(maxMonthlyPayment, stressedRate, loanDurationYears);
42
- const amortizationSchedule = this.calculateAmortizationSchedule(requiredLoanAmount, this._input.annualInterestRate, loanDurationYears);
42
+ const monthlyPayment = this.calculateMonthlyPayment(requiredLoanAmount, this._input.annualInterestRate, loanDurationYears);
43
+ const totalPaid = monthlyPayment * loanDurationYears * 12;
44
+ const totalInterestPaid = totalPaid - requiredLoanAmount;
45
+ const amortizationSchedule = this.calculateAmortizationSchedule(requiredLoanAmount, this._input.annualInterestRate, loanDurationYears, monthlyPayment);
43
46
  return {
47
+ loanAmount: requiredLoanAmount,
48
+ totalPaid,
49
+ totalInterestPaid,
50
+ monthlyPayment,
51
+ requiredLoanAmount,
44
52
  maxMonthlyPayment,
45
53
  maxLoanAmount,
46
54
  totalProjectCost,
47
- requiredLoanAmount,
48
55
  loanDurationYears,
49
56
  debtRatio: debtRatio,
50
57
  monthlyInsuranceCost,
51
58
  isEligible: maxLoanAmount >= requiredLoanAmount,
52
- amortizationSchedule
59
+ amortizationSchedule,
60
+ otherFees: {
61
+ notaryFees: {
62
+ value: notaryFees,
63
+ label: 'NOTARY_FEES'
64
+ },
65
+ bankFees: {
66
+ value: bankFees,
67
+ label: 'BANK_FEES'
68
+ },
69
+ monthlyInsuranceFees: {
70
+ value: monthlyInsuranceCost,
71
+ label: 'MONTHLY_INSURANCE_FEES'
72
+ }
73
+ }
53
74
  };
54
75
  }
55
- calculateAmortizationSchedule(loanAmount, annualRate, years) {
76
+ calculateMonthlyPayment(principal, annualRate, years) {
77
+ const monthlyRate = annualRate / 100 / 12;
78
+ const payments = years * 12;
79
+ if (monthlyRate === 0) {
80
+ return payments === 0 ? 0 : principal / payments;
81
+ }
82
+ return (principal *
83
+ (monthlyRate / (1 - Math.pow(1 + monthlyRate, -payments))));
84
+ }
85
+ calculateAmortizationSchedule(loanAmount, annualRate, years, monthlyPayment) {
56
86
  const monthlyRate = annualRate / 100 / 12;
57
87
  const totalPayments = years * 12;
58
- if (monthlyRate === 0 || loanAmount <= 0) {
88
+ if (monthlyRate === 0 || loanAmount <= 0 || monthlyPayment <= 0) {
59
89
  return [];
60
90
  }
61
- const monthlyPayment = (loanAmount * monthlyRate) /
62
- (1 - Math.pow(1 + monthlyRate, -totalPayments));
63
91
  const schedule = [];
64
92
  let balance = loanAmount;
65
93
  for (let year = 1; year <= years; year++) {
@@ -102,10 +130,28 @@ class FranceMortgageServiceImpl {
102
130
  loanDurationYears: 0,
103
131
  debtRatio: 0,
104
132
  monthlyInsuranceCost: 0,
133
+ loanAmount: 0,
134
+ monthlyPayment: 0,
135
+ totalInterestPaid: 0,
136
+ totalPaid: 0,
105
137
  isEligible: false,
106
- amortizationSchedule: []
138
+ amortizationSchedule: [],
139
+ otherFees: {
140
+ notaryFees: {
141
+ value: 0,
142
+ label: 'NOTARY_FEES'
143
+ },
144
+ bankFees: {
145
+ value: 0,
146
+ label: 'BANK_FEES'
147
+ },
148
+ monthlyInsuranceFees: {
149
+ value: 0,
150
+ label: 'MONTHLY_INSURANCE_FEES'
151
+ }
152
+ }
107
153
  };
108
154
  }
109
155
  }
110
156
  exports.FranceMortgageServiceImpl = FranceMortgageServiceImpl;
111
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FranceMortgageServiceImpl.js","sourceRoot":"","sources":["../../../src/mortgage/france/FranceMortgageServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { FranceMortgageService } from '../..';\nimport { MortgageRules, MortgageInput, MortgageOutput, AmortizationScheduleItem } from './domain/types';\n\nexport class FranceMortgageServiceImpl implements FranceMortgageService {\n    private _input: MortgageInput;\n    private _rules: MortgageRules;\n\n    constructor(input: MortgageInput, rules: MortgageRules) {\n        this._input = input;\n        this._rules = rules;\n    }\n\n    calculate(): MortgageOutput {\n\n        if (\n            this._input.isFirstTimeBuyer &&\n            this._rules.firstTimeBuyer &&\n            this._rules.firstTimeBuyer.requiresPrimaryResidence &&\n            !this._input.isPrimaryResidence\n        ) {\n        return this.ineligibleResult(this._input);\n        }\n\n        const debtRatio =\n            this._input.isFirstTimeBuyer && this._rules.firstTimeBuyer &&this._rules.firstTimeBuyer.enabled\n                ? this._rules.firstTimeBuyer.maxDebtRatio\n                : this._rules.maxDebtRatio;\n\n        const loanDurationYears =\n            this._input.isFirstTimeBuyer && this._rules.firstTimeBuyer && this._rules.firstTimeBuyer.enabled\n                ? this._rules.firstTimeBuyer.maxLoanDurationYears\n                : this._input.isNewBuild\n                ? this._rules.maxLoanDurationNewBuildYears\n                : this._rules.maxLoanDurationYears;\n\n        const minDownPayment = this._input.propertyPrice * this._rules.minDownPaymentRate;\n\n        if (this._input.downPayment < minDownPayment) {\n            return this.ineligibleResult(this._input);\n        }\n\n        const notaryRate = this._input.isNewBuild\n        ? this._rules.fees.notaryRateNewProperty\n        : this._rules.fees.notaryRateOldProperty;\n\n        const notaryFees = this._input.propertyPrice * notaryRate;\n        const bankFees = this._input.propertyPrice * this._rules.fees.bankFeesRate;\n        const totalProjectCost = this._input.propertyPrice + notaryFees + bankFees;\n        const requiredLoanAmount = totalProjectCost - this._input.downPayment;\n        const stressedRate = this._input.annualInterestRate + this._rules.stressTest.interestRateBuffer;\n\n        const monthlyInsuranceCost = (requiredLoanAmount * this._rules.insurance.averageRate) / 12;\n\n        let maxMonthlyPayment = this._input.netMonthlyIncome * debtRatio;\n\n        if (this._rules.insurance.includedInDebtRatio) {\n            maxMonthlyPayment -= monthlyInsuranceCost;\n        }\n\n        const maxLoanAmount = this.calculateLoanAmount(\n            maxMonthlyPayment,\n            stressedRate,\n            loanDurationYears\n        );\n\n        const amortizationSchedule = this.calculateAmortizationSchedule(\n            requiredLoanAmount,\n            this._input.annualInterestRate,\n            loanDurationYears\n        );\n\n        return {\n            maxMonthlyPayment,\n            maxLoanAmount,\n            totalProjectCost,\n            requiredLoanAmount,\n            loanDurationYears,\n            debtRatio: debtRatio,\n            monthlyInsuranceCost,\n            isEligible: maxLoanAmount >= requiredLoanAmount,\n            amortizationSchedule\n        };\n    }\n\n    private calculateAmortizationSchedule(\n        loanAmount: number,\n        annualRate: number,\n        years: number\n    ): AmortizationScheduleItem[] {\n        const monthlyRate = annualRate / 100 / 12;\n        const totalPayments = years * 12;\n\n        if (monthlyRate === 0 || loanAmount <= 0) {\n            return [];\n        }\n\n        const monthlyPayment =\n            (loanAmount * monthlyRate) /\n            (1 - Math.pow(1 + monthlyRate, -totalPayments));\n\n        const schedule: AmortizationScheduleItem[] = [];\n        let balance = loanAmount;\n\n        for (let year = 1; year <= years; year++) {\n            let yearlyPrincipal = 0;\n            let yearlyInterest = 0;\n\n            const paymentsInYear = Math.min(12, totalPayments - (year - 1) * 12);\n\n            for (let month = 1; month <= paymentsInYear; month++) {\n                const interestPayment = balance * monthlyRate;\n                const principalPayment = monthlyPayment - interestPayment;\n\n                yearlyInterest += interestPayment;\n                yearlyPrincipal += principalPayment;\n                balance -= principalPayment;\n            }\n\n            schedule.push({\n                year,\n                principal: yearlyPrincipal,\n                interest: yearlyInterest,\n                balance: Math.max(0, balance)\n            });\n\n            if (balance <= 0) break;\n        }\n\n        return schedule;\n    }\n\n    private calculateLoanAmount(monthlyPayment: number, annualRate: number, years: number): number {\n        const monthlyRate = annualRate / 100 / 12;\n        const payments = years * 12;\n\n        if (monthlyRate === 0) {\n            return monthlyPayment * payments;\n        }\n\n        return (\n            monthlyPayment *\n            ((1 - Math.pow(1 + monthlyRate, -payments)) / monthlyRate)\n        );\n    }\n\n    private ineligibleResult(input: MortgageInput): MortgageOutput {\n        return {\n            maxMonthlyPayment: 0,\n            maxLoanAmount: 0,\n            totalProjectCost: input.propertyPrice,\n            requiredLoanAmount: input.propertyPrice - input.downPayment,\n            loanDurationYears: 0,\n            debtRatio: 0,\n            monthlyInsuranceCost: 0,\n            isEligible: false,\n            amortizationSchedule: []\n        };\n    }\n}"]}
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FranceMortgageServiceImpl.js","sourceRoot":"","sources":["../../../src/mortgage/france/FranceMortgageServiceImpl.ts"],"names":[],"mappings":";;;AAGA,MAAa,yBAAyB;IAIlC,YAAY,KAAoB,EAAE,KAAoB;QAClD,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QACpB,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;IACxB,CAAC;IAED,SAAS;QAEL,IACI,IAAI,CAAC,MAAM,CAAC,gBAAgB;YAC5B,IAAI,CAAC,MAAM,CAAC,cAAc;YAC1B,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,wBAAwB;YACnD,CAAC,IAAI,CAAC,MAAM,CAAC,kBAAkB,EACjC,CAAC;YACH,OAAO,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC1C,CAAC;QAED,MAAM,SAAS,GACX,IAAI,CAAC,MAAM,CAAC,gBAAgB,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,IAAG,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,OAAO;YAC3F,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,YAAY;YACzC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC;QAEnC,MAAM,iBAAiB,GACnB,IAAI,CAAC,MAAM,CAAC,gBAAgB,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,OAAO;YAC5F,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,oBAAoB;YACjD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU;gBACxB,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,4BAA4B;gBAC1C,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,oBAAoB,CAAC;QAE3C,MAAM,cAAc,GAAG,IAAI,CAAC,MAAM,CAAC,aAAa,GAAG,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC;QAElF,IAAI,IAAI,CAAC,MAAM,CAAC,WAAW,GAAG,cAAc,EAAE,CAAC;YAC3C,OAAO,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC9C,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU;YACzC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,qBAAqB;YACxC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,qBAAqB,CAAC;QAEzC,MAAM,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,aAAa,GAAG,UAAU,CAAC;QAC1D,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,aAAa,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC;QAC3E,MAAM,gBAAgB,GAAG,IAAI,CAAC,MAAM,CAAC,aAAa,GAAG,UAAU,GAAG,QAAQ,CAAC;QAC3E,MAAM,kBAAkB,GAAG,gBAAgB,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC;QACtE,MAAM,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,kBAAkB,CAAC;QAEhG,MAAM,oBAAoB,GAAG,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,WAAW,CAAC,GAAG,EAAE,CAAC;QAE3F,IAAI,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,gBAAgB,GAAG,SAAS,CAAC;QAEjE,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,mBAAmB,EAAE,CAAC;YAC5C,iBAAiB,IAAI,oBAAoB,CAAC;QAC9C,CAAC;QAED,MAAM,aAAa,GAAG,IAAI,CAAC,mBAAmB,CAC1C,iBAAiB,EACjB,YAAY,EACZ,iBAAiB,CACpB,CAAC;QAEF,MAAM,cAAc,GAAG,IAAI,CAAC,uBAAuB,CAC/C,kBAAkB,EAClB,IAAI,CAAC,MAAM,CAAC,kBAAkB,EAC9B,iBAAiB,CACpB,CAAC;QAEF,MAAM,SAAS,GAAG,cAAc,GAAG,iBAAiB,GAAG,EAAE,CAAC;QAC1D,MAAM,iBAAiB,GAAG,SAAS,GAAG,kBAAkB,CAAC;QAEzD,MAAM,oBAAoB,GAAG,IAAI,CAAC,6BAA6B,CAC3D,kBAAkB,EAClB,IAAI,CAAC,MAAM,CAAC,kBAAkB,EAC9B,iBAAiB,EACjB,cAAc,CACjB,CAAC;QAEF,OAAO;YACH,UAAU,EAAE,kBAAkB;YAC9B,SAAS;YACT,iBAAiB;YACjB,cAAc;YACd,kBAAkB;YAClB,iBAAiB;YACjB,aAAa;YACb,gBAAgB;YAChB,iBAAiB;YACjB,SAAS,EAAE,SAAS;YACpB,oBAAoB;YACpB,UAAU,EAAE,aAAa,IAAI,kBAAkB;YAC/C,oBAAoB;YACpB,SAAS,EAAE;gBACP,UAAU,EAAE;oBACR,KAAK,EAAE,UAAU;oBACjB,KAAK,EAAE,aAAa;iBACvB;gBACD,QAAQ,EAAE;oBACN,KAAK,EAAE,QAAQ;oBACf,KAAK,EAAE,WAAW;iBACrB;gBACD,oBAAoB,EAAE;oBAClB,KAAK,EAAE,oBAAoB;oBAC3B,KAAK,EAAE,wBAAwB;iBAClC;aACJ;SACJ,CAAC;IACN,CAAC;IAEO,uBAAuB,CAAC,SAAiB,EAAE,UAAkB,EAAE,KAAa;QAChF,MAAM,WAAW,GAAG,UAAU,GAAG,GAAG,GAAG,EAAE,CAAC;QAC1C,MAAM,QAAQ,GAAG,KAAK,GAAG,EAAE,CAAC;QAE5B,IAAI,WAAW,KAAK,CAAC,EAAE,CAAC;YACpB,OAAO,QAAQ,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,GAAG,QAAQ,CAAC;QACrD,CAAC;QAED,OAAO,CACH,SAAS;YACT,CAAC,WAAW,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,WAAW,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,CAC7D,CAAC;IACN,CAAC;IAEO,6BAA6B,CACjC,UAAkB,EAClB,UAAkB,EAClB,KAAa,EACb,cAAsB;QAEtB,MAAM,WAAW,GAAG,UAAU,GAAG,GAAG,GAAG,EAAE,CAAC;QAC1C,MAAM,aAAa,GAAG,KAAK,GAAG,EAAE,CAAC;QAEjC,IAAI,WAAW,KAAK,CAAC,IAAI,UAAU,IAAI,CAAC,IAAI,cAAc,IAAI,CAAC,EAAE,CAAC;YAC9D,OAAO,EAAE,CAAC;QACd,CAAC;QAED,MAAM,QAAQ,GAA+B,EAAE,CAAC;QAChD,IAAI,OAAO,GAAG,UAAU,CAAC;QAEzB,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,IAAI,KAAK,EAAE,IAAI,EAAE,EAAE,CAAC;YACvC,IAAI,eAAe,GAAG,CAAC,CAAC;YACxB,IAAI,cAAc,GAAG,CAAC,CAAC;YAEvB,MAAM,cAAc,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,aAAa,GAAG,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;YAErE,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,IAAI,cAAc,EAAE,KAAK,EAAE,EAAE,CAAC;gBACnD,MAAM,eAAe,GAAG,OAAO,GAAG,WAAW,CAAC;gBAC9C,MAAM,gBAAgB,GAAG,cAAc,GAAG,eAAe,CAAC;gBAE1D,cAAc,IAAI,eAAe,CAAC;gBAClC,eAAe,IAAI,gBAAgB,CAAC;gBACpC,OAAO,IAAI,gBAAgB,CAAC;YAChC,CAAC;YAED,QAAQ,CAAC,IAAI,CAAC;gBACV,IAAI;gBACJ,SAAS,EAAE,eAAe;gBAC1B,QAAQ,EAAE,cAAc;gBACxB,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,CAAC;aAChC,CAAC,CAAC;YAEH,IAAI,OAAO,IAAI,CAAC;gBAAE,MAAM;QAC5B,CAAC;QAED,OAAO,QAAQ,CAAC;IACpB,CAAC;IAEO,mBAAmB,CAAC,cAAsB,EAAE,UAAkB,EAAE,KAAa;QACjF,MAAM,WAAW,GAAG,UAAU,GAAG,GAAG,GAAG,EAAE,CAAC;QAC1C,MAAM,QAAQ,GAAG,KAAK,GAAG,EAAE,CAAC;QAE5B,IAAI,WAAW,KAAK,CAAC,EAAE,CAAC;YACpB,OAAO,cAAc,GAAG,QAAQ,CAAC;QACrC,CAAC;QAED,OAAO,CACH,cAAc;YACd,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,WAAW,EAAE,CAAC,QAAQ,CAAC,CAAC,GAAG,WAAW,CAAC,CAC7D,CAAC;IACN,CAAC;IAEO,gBAAgB,CAAC,KAAoB;QACzC,OAAO;YACH,iBAAiB,EAAE,CAAC;YACpB,aAAa,EAAE,CAAC;YAChB,gBAAgB,EAAE,KAAK,CAAC,aAAa;YACrC,kBAAkB,EAAE,KAAK,CAAC,aAAa,GAAG,KAAK,CAAC,WAAW;YAC3D,iBAAiB,EAAE,CAAC;YACpB,SAAS,EAAE,CAAC;YACZ,oBAAoB,EAAE,CAAC;YACvB,UAAU,EAAE,CAAC;YACb,cAAc,EAAE,CAAC;YACjB,iBAAiB,EAAE,CAAC;YACpB,SAAS,EAAE,CAAC;YACZ,UAAU,EAAE,KAAK;YACjB,oBAAoB,EAAE,EAAE;YACxB,SAAS,EAAE;gBACP,UAAU,EAAE;oBACR,KAAK,EAAE,CAAC;oBACR,KAAK,EAAE,aAAa;iBACvB;gBACD,QAAQ,EAAE;oBACN,KAAK,EAAE,CAAC;oBACR,KAAK,EAAE,WAAW;iBACrB;gBACD,oBAAoB,EAAE;oBAClB,KAAK,EAAE,CAAC;oBACR,KAAK,EAAE,wBAAwB;iBAClC;aACJ;SACJ,CAAC;IACN,CAAC;CACJ;AApND,8DAoNC","sourcesContent":["import { FranceMortgageService } from '../..';\nimport { MortgageRules, MortgageInput, MortgageOutput, AmortizationScheduleItem } from './domain/types';\n\nexport class FranceMortgageServiceImpl implements FranceMortgageService {\n    private _input: MortgageInput;\n    private _rules: MortgageRules;\n\n    constructor(input: MortgageInput, rules: MortgageRules) {\n        this._input = input;\n        this._rules = rules;\n    }\n\n    calculate(): MortgageOutput {\n\n        if (\n            this._input.isFirstTimeBuyer &&\n            this._rules.firstTimeBuyer &&\n            this._rules.firstTimeBuyer.requiresPrimaryResidence &&\n            !this._input.isPrimaryResidence\n        ) {\n        return this.ineligibleResult(this._input);\n        }\n\n        const debtRatio =\n            this._input.isFirstTimeBuyer && this._rules.firstTimeBuyer &&this._rules.firstTimeBuyer.enabled\n                ? this._rules.firstTimeBuyer.maxDebtRatio\n                : this._rules.maxDebtRatio;\n\n        const loanDurationYears =\n            this._input.isFirstTimeBuyer && this._rules.firstTimeBuyer && this._rules.firstTimeBuyer.enabled\n                ? this._rules.firstTimeBuyer.maxLoanDurationYears\n                : this._input.isNewBuild\n                ? this._rules.maxLoanDurationNewBuildYears\n                : this._rules.maxLoanDurationYears;\n\n        const minDownPayment = this._input.propertyPrice * this._rules.minDownPaymentRate;\n\n        if (this._input.downPayment < minDownPayment) {\n            return this.ineligibleResult(this._input);\n        }\n\n        const notaryRate = this._input.isNewBuild\n        ? this._rules.fees.notaryRateNewProperty\n        : this._rules.fees.notaryRateOldProperty;\n\n        const notaryFees = this._input.propertyPrice * notaryRate;\n        const bankFees = this._input.propertyPrice * this._rules.fees.bankFeesRate;\n        const totalProjectCost = this._input.propertyPrice + notaryFees + bankFees;\n        const requiredLoanAmount = totalProjectCost - this._input.downPayment;\n        const stressedRate = this._input.annualInterestRate + this._rules.stressTest.interestRateBuffer;\n\n        const monthlyInsuranceCost = (requiredLoanAmount * this._rules.insurance.averageRate) / 12;\n\n        let maxMonthlyPayment = this._input.netMonthlyIncome * debtRatio;\n\n        if (this._rules.insurance.includedInDebtRatio) {\n            maxMonthlyPayment -= monthlyInsuranceCost;\n        }\n\n        const maxLoanAmount = this.calculateLoanAmount(\n            maxMonthlyPayment,\n            stressedRate,\n            loanDurationYears\n        );\n\n        const monthlyPayment = this.calculateMonthlyPayment(\n            requiredLoanAmount,\n            this._input.annualInterestRate,\n            loanDurationYears\n        );\n\n        const totalPaid = monthlyPayment * loanDurationYears * 12;\n        const totalInterestPaid = totalPaid - requiredLoanAmount;\n\n        const amortizationSchedule = this.calculateAmortizationSchedule(\n            requiredLoanAmount,\n            this._input.annualInterestRate,\n            loanDurationYears,\n            monthlyPayment\n        );\n\n        return {\n            loanAmount: requiredLoanAmount,\n            totalPaid,\n            totalInterestPaid,\n            monthlyPayment,\n            requiredLoanAmount,\n            maxMonthlyPayment,\n            maxLoanAmount,\n            totalProjectCost,\n            loanDurationYears,\n            debtRatio: debtRatio,\n            monthlyInsuranceCost,\n            isEligible: maxLoanAmount >= requiredLoanAmount,\n            amortizationSchedule,\n            otherFees: {\n                notaryFees: {\n                    value: notaryFees,\n                    label: 'NOTARY_FEES'\n                },\n                bankFees: {\n                    value: bankFees,\n                    label: 'BANK_FEES'\n                },\n                monthlyInsuranceFees: {\n                    value: monthlyInsuranceCost,\n                    label: 'MONTHLY_INSURANCE_FEES'\n                }\n            }\n        };\n    }\n\n    private calculateMonthlyPayment(principal: number, annualRate: number, years: number): number {\n        const monthlyRate = annualRate / 100 / 12;\n        const payments = years * 12;\n\n        if (monthlyRate === 0) {\n            return payments === 0 ? 0 : principal / payments;\n        }\n\n        return (\n            principal *\n            (monthlyRate / (1 - Math.pow(1 + monthlyRate, -payments)))\n        );\n    }\n\n    private calculateAmortizationSchedule(\n        loanAmount: number,\n        annualRate: number,\n        years: number,\n        monthlyPayment: number\n    ): AmortizationScheduleItem[] {\n        const monthlyRate = annualRate / 100 / 12;\n        const totalPayments = years * 12;\n\n        if (monthlyRate === 0 || loanAmount <= 0 || monthlyPayment <= 0) {\n            return [];\n        }\n\n        const schedule: AmortizationScheduleItem[] = [];\n        let balance = loanAmount;\n\n        for (let year = 1; year <= years; year++) {\n            let yearlyPrincipal = 0;\n            let yearlyInterest = 0;\n\n            const paymentsInYear = Math.min(12, totalPayments - (year - 1) * 12);\n\n            for (let month = 1; month <= paymentsInYear; month++) {\n                const interestPayment = balance * monthlyRate;\n                const principalPayment = monthlyPayment - interestPayment;\n\n                yearlyInterest += interestPayment;\n                yearlyPrincipal += principalPayment;\n                balance -= principalPayment;\n            }\n\n            schedule.push({\n                year,\n                principal: yearlyPrincipal,\n                interest: yearlyInterest,\n                balance: Math.max(0, balance)\n            });\n\n            if (balance <= 0) break;\n        }\n\n        return schedule;\n    }\n\n    private calculateLoanAmount(monthlyPayment: number, annualRate: number, years: number): number {\n        const monthlyRate = annualRate / 100 / 12;\n        const payments = years * 12;\n\n        if (monthlyRate === 0) {\n            return monthlyPayment * payments;\n        }\n\n        return (\n            monthlyPayment *\n            ((1 - Math.pow(1 + monthlyRate, -payments)) / monthlyRate)\n        );\n    }\n\n    private ineligibleResult(input: MortgageInput): MortgageOutput {\n        return {\n            maxMonthlyPayment: 0,\n            maxLoanAmount: 0,\n            totalProjectCost: input.propertyPrice,\n            requiredLoanAmount: input.propertyPrice - input.downPayment,\n            loanDurationYears: 0,\n            debtRatio: 0,\n            monthlyInsuranceCost: 0,\n            loanAmount: 0,\n            monthlyPayment: 0,\n            totalInterestPaid: 0,\n            totalPaid: 0,\n            isEligible: false,\n            amortizationSchedule: [],\n            otherFees: {\n                notaryFees: {\n                    value: 0,\n                    label: 'NOTARY_FEES'\n                },\n                bankFees: {\n                    value: 0,\n                    label: 'BANK_FEES'\n                },\n                monthlyInsuranceFees: {\n                    value: 0,\n                    label: 'MONTHLY_INSURANCE_FEES'\n                }\n            }\n        };\n    }\n}"]}
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2UvZnJhbmNlL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVzIHtcbiAgICBtYXhEZWJ0UmF0aW86IG51bWJlcjtcbiAgICBtYXhMb2FuRHVyYXRpb25ZZWFyczogbnVtYmVyO1xuICAgIG1heExvYW5EdXJhdGlvbk5ld0J1aWxkWWVhcnM6IG51bWJlcjtcbiAgICBtaW5Eb3duUGF5bWVudFJhdGU6IG51bWJlcjtcblxuICAgIGZpcnN0VGltZUJ1eWVyPzogRmlyc3RUaW1lQnV5ZXJSdWxlcztcblxuICAgIGluc3VyYW5jZTogTW9ydGdhZ2VJbnN1cmFuY2VSdWxlcztcbiAgICBmZWVzOiBNb3J0Z2FnZUZlZXNSdWxlcztcbiAgICBzdHJlc3NUZXN0OiBTdHJlc3NUZXN0UnVsZXM7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmlyc3RUaW1lQnV5ZXJSdWxlcyB7XG4gICAgZW5hYmxlZDogYm9vbGVhbjtcbiAgICBtYXhEZWJ0UmF0aW86IG51bWJlcjtcbiAgICBtYXhMb2FuRHVyYXRpb25ZZWFyczogbnVtYmVyO1xuICAgIHF1b3RhRGlzY2xhaW1lcjogc3RyaW5nO1xuICAgIHJlcXVpcmVzUHJpbWFyeVJlc2lkZW5jZTogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUluc3VyYW5jZVJ1bGVzIHtcbiAgICBhdmVyYWdlUmF0ZTogbnVtYmVyO1xuICAgIGluY2x1ZGVkSW5EZWJ0UmF0aW86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VGZWVzUnVsZXMge1xuICBub3RhcnlSYXRlT2xkUHJvcGVydHk6IG51bWJlcjtcbiAgbm90YXJ5UmF0ZU5ld1Byb3BlcnR5OiBudW1iZXI7XG4gIGJhbmtGZWVzUmF0ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN0cmVzc1Rlc3RSdWxlcyB7XG4gICAgaW50ZXJlc3RSYXRlQnVmZmVyOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW1vcnRpemF0aW9uU2NoZWR1bGVJdGVtIHtcbiAgICB5ZWFyOiBudW1iZXI7XG4gICAgcHJpbmNpcGFsOiBudW1iZXI7XG4gICAgaW50ZXJlc3Q6IG51bWJlcjtcbiAgICBiYWxhbmNlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VJbnB1dCB7XG4gICAgcHJvcGVydHlQcmljZTogbnVtYmVyO1xuICAgIGRvd25QYXltZW50OiBudW1iZXI7XG4gICAgbmV0TW9udGhseUluY29tZTogbnVtYmVyO1xuICAgIGxvYW5EdXJhdGlvblllYXJzOiBudW1iZXI7XG4gICAgYW5udWFsSW50ZXJlc3RSYXRlOiBudW1iZXI7XG4gICAgaXNQcmltYXJ5UmVzaWRlbmNlOiBib29sZWFuO1xuICAgIGlzRmlyc3RUaW1lQnV5ZXI6IGJvb2xlYW47XG4gICAgaXNOZXdCdWlsZDogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZU91dHB1dCB7XG4gICAgbWF4TW9udGhseVBheW1lbnQ6IG51bWJlcjtcbiAgICBtYXhMb2FuQW1vdW50OiBudW1iZXI7XG4gICAgbG9hbkR1cmF0aW9uWWVhcnM6IG51bWJlcjtcbiAgICBkZWJ0UmF0aW86IG51bWJlcjtcbiAgICB0b3RhbFByb2plY3RDb3N0OiBudW1iZXI7XG4gICAgcmVxdWlyZWRMb2FuQW1vdW50OiBudW1iZXI7XG4gICAgbW9udGhseUluc3VyYW5jZUNvc3Q6IG51bWJlcjtcbiAgICBpc0VsaWdpYmxlOiBib29sZWFuO1xuICAgIGFtb3J0aXphdGlvblNjaGVkdWxlOiBBbW9ydGl6YXRpb25TY2hlZHVsZUl0ZW1bXTtcbn1cbiJdfQ==
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2UvZnJhbmNlL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgT3RoZXJGZWVzIH0gZnJvbSBcIi4uLy4uL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIE1vcnRnYWdlUnVsZXMge1xuICAgIG1heERlYnRSYXRpbzogbnVtYmVyO1xuICAgIG1heExvYW5EdXJhdGlvblllYXJzOiBudW1iZXI7XG4gICAgbWF4TG9hbkR1cmF0aW9uTmV3QnVpbGRZZWFyczogbnVtYmVyO1xuICAgIG1pbkRvd25QYXltZW50UmF0ZTogbnVtYmVyO1xuXG4gICAgZmlyc3RUaW1lQnV5ZXI/OiBGaXJzdFRpbWVCdXllclJ1bGVzO1xuXG4gICAgaW5zdXJhbmNlOiBNb3J0Z2FnZUluc3VyYW5jZVJ1bGVzO1xuICAgIGZlZXM6IE1vcnRnYWdlRmVlc1J1bGVzO1xuICAgIHN0cmVzc1Rlc3Q6IFN0cmVzc1Rlc3RSdWxlcztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBGaXJzdFRpbWVCdXllclJ1bGVzIHtcbiAgICBlbmFibGVkOiBib29sZWFuO1xuICAgIG1heERlYnRSYXRpbzogbnVtYmVyO1xuICAgIG1heExvYW5EdXJhdGlvblllYXJzOiBudW1iZXI7XG4gICAgcXVvdGFEaXNjbGFpbWVyOiBzdHJpbmc7XG4gICAgcmVxdWlyZXNQcmltYXJ5UmVzaWRlbmNlOiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vcnRnYWdlSW5zdXJhbmNlUnVsZXMge1xuICAgIGF2ZXJhZ2VSYXRlOiBudW1iZXI7XG4gICAgaW5jbHVkZWRJbkRlYnRSYXRpbzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUZlZXNSdWxlcyB7XG4gIG5vdGFyeVJhdGVPbGRQcm9wZXJ0eTogbnVtYmVyO1xuICBub3RhcnlSYXRlTmV3UHJvcGVydHk6IG51bWJlcjtcbiAgYmFua0ZlZXNSYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3RyZXNzVGVzdFJ1bGVzIHtcbiAgICBpbnRlcmVzdFJhdGVCdWZmZXI6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBbW9ydGl6YXRpb25TY2hlZHVsZUl0ZW0ge1xuICAgIHllYXI6IG51bWJlcjtcbiAgICBwcmluY2lwYWw6IG51bWJlcjtcbiAgICBpbnRlcmVzdDogbnVtYmVyO1xuICAgIGJhbGFuY2U6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUlucHV0IHtcbiAgICBwcm9wZXJ0eVByaWNlOiBudW1iZXI7XG4gICAgZG93blBheW1lbnQ6IG51bWJlcjtcbiAgICBuZXRNb250aGx5SW5jb21lOiBudW1iZXI7XG4gICAgbG9hbkR1cmF0aW9uWWVhcnM6IG51bWJlcjtcbiAgICBhbm51YWxJbnRlcmVzdFJhdGU6IG51bWJlcjtcbiAgICBpc1ByaW1hcnlSZXNpZGVuY2U6IGJvb2xlYW47XG4gICAgaXNGaXJzdFRpbWVCdXllcjogYm9vbGVhbjtcbiAgICBpc05ld0J1aWxkOiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vcnRnYWdlT3V0cHV0IHtcbiAgICBsb2FuQW1vdW50OiBudW1iZXI7XG4gICAgdG90YWxQYWlkOiBudW1iZXI7XG4gICAgbW9udGhseVBheW1lbnQ6IG51bWJlcjtcbiAgICB0b3RhbEludGVyZXN0UGFpZDogbnVtYmVyO1xuICAgIG1heE1vbnRobHlQYXltZW50OiBudW1iZXI7XG4gICAgbWF4TG9hbkFtb3VudDogbnVtYmVyO1xuICAgIGxvYW5EdXJhdGlvblllYXJzOiBudW1iZXI7XG4gICAgZGVidFJhdGlvOiBudW1iZXI7XG4gICAgdG90YWxQcm9qZWN0Q29zdDogbnVtYmVyO1xuICAgIHJlcXVpcmVkTG9hbkFtb3VudDogbnVtYmVyO1xuICAgIG1vbnRobHlJbnN1cmFuY2VDb3N0OiBudW1iZXI7XG4gICAgaXNFbGlnaWJsZTogYm9vbGVhbjtcbiAgICBhbW9ydGl6YXRpb25TY2hlZHVsZTogQW1vcnRpemF0aW9uU2NoZWR1bGVJdGVtW107XG4gICAgb3RoZXJGZWVzOiBPdGhlckZlZXM7XG59XG4iXX0=
@@ -32,7 +32,21 @@ class SouthAfricaMortgageServiceImpl {
32
32
  isAffordable,
33
33
  transferDuty,
34
34
  bondRegistrationFee,
35
- amortizationSchedule
35
+ amortizationSchedule,
36
+ otherFees: {
37
+ notaryFees: {
38
+ value: bondRegistrationFee,
39
+ label: 'BOND_REGISTRATION_FEES'
40
+ },
41
+ bankFees: {
42
+ value: transferDuty,
43
+ label: 'TRANSFER_DUTY'
44
+ },
45
+ monthlyInsuranceFees: {
46
+ value: 0,
47
+ label: 'MONTHLY_INSURANCE_FEES'
48
+ }
49
+ }
36
50
  };
37
51
  }
38
52
  calculateAmortizationSchedule(loanAmount, monthlyRate, monthlyPayment, totalPayments) {
@@ -79,4 +93,4 @@ class SouthAfricaMortgageServiceImpl {
79
93
  }
80
94
  }
81
95
  exports.SouthAfricaMortgageServiceImpl = SouthAfricaMortgageServiceImpl;
82
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SouthAfricaMortgageServiceImpl.js","sourceRoot":"","sources":["../../../src/mortgage/south-africa/SouthAfricaMortgageServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { MortgageRules, MortgageInput, MortgageOutput, AmortizationScheduleItem } from './domain/types';\nimport { SouthAfricaMortgageService } from './SouthAfricaMortgageService';\n\nexport class SouthAfricaMortgageServiceImpl implements SouthAfricaMortgageService {\n  constructor(\n    private readonly input: MortgageInput,\n    private readonly rules: MortgageRules\n) {}\n\n\tcalculate(): MortgageOutput {\n\t\tconst loanAmount = this.input.propertyPrice - this.input.downPayment;\n\n\t\t// Monthly interest rate\n\t\tconst monthlyRate = this.input.annualInterestRate / 100 / 12;\n\t\tconst totalPayments = this.input.amortizationYears * 12;\n\n\t\t// Standard amortized monthly payment formula\n\t\tconst monthlyPayment =\n\t\t(loanAmount * monthlyRate) /\n\t\t(1 - Math.pow(1 + monthlyRate, -totalPayments));\n\n\t\tconst totalPaid = monthlyPayment * totalPayments;\n\t\tconst totalInterestPaid = totalPaid - loanAmount;\n\n\t\t// Debt-to-income ratio\n\t\tconst debtToIncomeRatio =\n\t\t(monthlyPayment / this.input.grossMonthlyIncome) * 100;\n\n\t\tconst isAffordable =\n\t\tdebtToIncomeRatio <= this.rules.loanConstraints.maxDebtToIncomePercent;\n\n\t\t// Fees\n\t\tconst bondRegistrationFee =\n\t\tloanAmount * (this.rules.fees.bondRegistrationPercent / 100);\n\n\t\tconst transferDuty = this.calculateTransferDuty(this.input.propertyPrice);\n\t\tconst amortizationSchedule = this.calculateAmortizationSchedule(\n\t\t\tloanAmount,\n\t\t\tmonthlyRate,\n\t\t\tmonthlyPayment,\n\t\t\ttotalPayments\n\t\t);\n\n\t\treturn {\n\t\t\tloanAmount,\n\t\t\tmonthlyPayment,\n\t\t\ttotalInterestPaid,\n\t\t\ttotalPaid,\n\t\t\tdebtToIncomeRatio,\n\t\t\tisAffordable,\n\t\t\ttransferDuty,\n\t\t\tbondRegistrationFee,\n\t\t\tamortizationSchedule\n\t\t};\n\t}\n\n\tprivate calculateAmortizationSchedule(\n\t\tloanAmount: number,\n\t\tmonthlyRate: number,\n\t\tmonthlyPayment: number,\n\t\ttotalPayments: number\n\t): AmortizationScheduleItem[] {\n\t\tconst schedule: AmortizationScheduleItem[] = [];\n\t\tlet balance = loanAmount;\n\n\t\tfor (let year = 1; year <= this.input.amortizationYears; year++) {\n\t\t\tlet yearlyPrincipal = 0;\n\t\t\tlet yearlyInterest = 0;\n\n\t\t\t// Calculate for 12 months or remaining payments\n\t\t\tconst paymentsInYear = Math.min(12, totalPayments - (year - 1) * 12);\n\n\t\t\tfor (let month = 1; month <= paymentsInYear; month++) {\n\t\t\t\tconst interestPayment = balance * monthlyRate;\n\t\t\t\tconst principalPayment = monthlyPayment - interestPayment;\n\n\t\t\t\tyearlyInterest += interestPayment;\n\t\t\t\tyearlyPrincipal += principalPayment;\n\t\t\t\tbalance -= principalPayment;\n\t\t\t}\n\n\t\t\tschedule.push({\n\t\t\t\tyear,\n\t\t\t\tprincipal: yearlyPrincipal,\n\t\t\t\tinterest: yearlyInterest,\n\t\t\t\tbalance: Math.max(0, balance)\n\t\t\t});\n\n\t\t\tif (balance <= 0) break;\n\t\t}\n\n\t\treturn schedule;\n\t}\n\n\tprivate calculateTransferDuty(propertyPrice: number): number {\n\t\tlet duty = 0;\n\t\tlet previousLimit = 0;\n\n\t\tfor (const bracket of this.rules.fees.transferDuty.brackets) {\n\t\t\tif (bracket.upTo && propertyPrice > previousLimit) {\n\t\t\t\tconst taxable = Math.min(propertyPrice, bracket.upTo) - previousLimit;\n\t\t\t\tduty += taxable * bracket.rate;\n\t\t\t\tpreviousLimit = bracket.upTo;\n\t\t\t}\n\n\t\t\tif (bracket.above && propertyPrice > bracket.above) {\n\t\t\t\tduty += (propertyPrice - bracket.above) * bracket.rate;\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn duty;\n\t}\n}\n"]}
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SouthAfricaMortgageServiceImpl.js","sourceRoot":"","sources":["../../../src/mortgage/south-africa/SouthAfricaMortgageServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { MortgageRules, MortgageInput, MortgageOutput, AmortizationScheduleItem } from './domain/types';\nimport { SouthAfricaMortgageService } from './SouthAfricaMortgageService';\n\nexport class SouthAfricaMortgageServiceImpl implements SouthAfricaMortgageService {\n  constructor(\n    private readonly input: MortgageInput,\n    private readonly rules: MortgageRules\n) {}\n\n\tcalculate(): MortgageOutput {\n\t\tconst loanAmount = this.input.propertyPrice - this.input.downPayment;\n\n\t\t// Monthly interest rate\n\t\tconst monthlyRate = this.input.annualInterestRate / 100 / 12;\n\t\tconst totalPayments = this.input.amortizationYears * 12;\n\n\t\t// Standard amortized monthly payment formula\n\t\tconst monthlyPayment =\n\t\t(loanAmount * monthlyRate) /\n\t\t(1 - Math.pow(1 + monthlyRate, -totalPayments));\n\n\t\tconst totalPaid = monthlyPayment * totalPayments;\n\t\tconst totalInterestPaid = totalPaid - loanAmount;\n\n\t\t// Debt-to-income ratio\n\t\tconst debtToIncomeRatio =\n\t\t(monthlyPayment / this.input.grossMonthlyIncome) * 100;\n\n\t\tconst isAffordable =\n\t\tdebtToIncomeRatio <= this.rules.loanConstraints.maxDebtToIncomePercent;\n\n\t\t// Fees\n\t\tconst bondRegistrationFee =\n\t\tloanAmount * (this.rules.fees.bondRegistrationPercent / 100);\n\n\t\tconst transferDuty = this.calculateTransferDuty(this.input.propertyPrice);\n\t\tconst amortizationSchedule = this.calculateAmortizationSchedule(\n\t\t\tloanAmount,\n\t\t\tmonthlyRate,\n\t\t\tmonthlyPayment,\n\t\t\ttotalPayments\n\t\t);\n\n\t\treturn {\n\t\t\tloanAmount,\n\t\t\tmonthlyPayment,\n\t\t\ttotalInterestPaid,\n\t\t\ttotalPaid,\n\t\t\tdebtToIncomeRatio,\n\t\t\tisAffordable,\n\t\t\ttransferDuty,\n\t\t\tbondRegistrationFee,\n\t\t\tamortizationSchedule,\n\t\t\totherFees: {\n                notaryFees: {\n                    value: bondRegistrationFee,\n                    label: 'BOND_REGISTRATION_FEES'\n                },\n                bankFees: {\n                    value: transferDuty,\n                    label: 'TRANSFER_DUTY'\n                },\n                monthlyInsuranceFees: {\n                    value: 0,\n                    label: 'MONTHLY_INSURANCE_FEES'\n                }\n            }\n\t\t};\n\t}\n\n\tprivate calculateAmortizationSchedule(\n\t\tloanAmount: number,\n\t\tmonthlyRate: number,\n\t\tmonthlyPayment: number,\n\t\ttotalPayments: number\n\t): AmortizationScheduleItem[] {\n\t\tconst schedule: AmortizationScheduleItem[] = [];\n\t\tlet balance = loanAmount;\n\n\t\tfor (let year = 1; year <= this.input.amortizationYears; year++) {\n\t\t\tlet yearlyPrincipal = 0;\n\t\t\tlet yearlyInterest = 0;\n\n\t\t\t// Calculate for 12 months or remaining payments\n\t\t\tconst paymentsInYear = Math.min(12, totalPayments - (year - 1) * 12);\n\n\t\t\tfor (let month = 1; month <= paymentsInYear; month++) {\n\t\t\t\tconst interestPayment = balance * monthlyRate;\n\t\t\t\tconst principalPayment = monthlyPayment - interestPayment;\n\n\t\t\t\tyearlyInterest += interestPayment;\n\t\t\t\tyearlyPrincipal += principalPayment;\n\t\t\t\tbalance -= principalPayment;\n\t\t\t}\n\n\t\t\tschedule.push({\n\t\t\t\tyear,\n\t\t\t\tprincipal: yearlyPrincipal,\n\t\t\t\tinterest: yearlyInterest,\n\t\t\t\tbalance: Math.max(0, balance)\n\t\t\t});\n\n\t\t\tif (balance <= 0) break;\n\t\t}\n\n\t\treturn schedule;\n\t}\n\n\tprivate calculateTransferDuty(propertyPrice: number): number {\n\t\tlet duty = 0;\n\t\tlet previousLimit = 0;\n\n\t\tfor (const bracket of this.rules.fees.transferDuty.brackets) {\n\t\t\tif (bracket.upTo && propertyPrice > previousLimit) {\n\t\t\t\tconst taxable = Math.min(propertyPrice, bracket.upTo) - previousLimit;\n\t\t\t\tduty += taxable * bracket.rate;\n\t\t\t\tpreviousLimit = bracket.upTo;\n\t\t\t}\n\n\t\t\tif (bracket.above && propertyPrice > bracket.above) {\n\t\t\t\tduty += (propertyPrice - bracket.above) * bracket.rate;\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn duty;\n\t}\n}\n"]}
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2Uvc291dGgtYWZyaWNhL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUnVsZU1ldGEgfSBmcm9tIFwiLi4vLi4vLi4vc2hhcmVkL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFJ1bGVJbnB1dERlZmluaXRpb24ge1xuICB0eXBlOiBcIm51bWJlclwiIHwgXCJzdHJpbmdcIjtcbiAgbWluPzogbnVtYmVyO1xuICBtYXg/OiBudW1iZXI7XG4gIGVudW0/OiBzdHJpbmdbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSdWxlSW5wdXRzIHtcbiAgcHJvcGVydHlQcmljZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgZG93blBheW1lbnQ6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gIGludGVyZXN0UmF0ZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgYW1vcnRpemF0aW9uWWVhcnM6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gIGdyb3NzTW9udGhseUluY29tZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBMb2FuQ29uc3RyYWludHMge1xuICBtYXhMdHY6IG51bWJlcjtcbiAgbWluRG93blBheW1lbnRQZXJjZW50OiBudW1iZXI7XG4gIG1heEFtb3J0aXphdGlvblllYXJzOiBudW1iZXI7XG4gIG1heERlYnRUb0luY29tZVBlcmNlbnQ6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBJbnRlcmVzdFJ1bGVzIHtcbiAgdHlwZTogXCJmaXhlZFwiIHwgXCJ2YXJpYWJsZVwiO1xuICByYXRlUmFuZ2VQZXJjZW50OiB7IG1pbjogbnVtYmVyOyBtYXg6IG51bWJlciB9O1xuICBzdHJlc3NUZXN0QnVmZmVyUGVyY2VudDogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEluc3VyYW5jZVJ1bGVzIHtcbiAgcmVxdWlyZWQ6IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVHJhbnNmZXJEdXR5QnJhY2tldCB7XG4gIHVwVG8/OiBudW1iZXI7XG4gIGFib3ZlPzogbnVtYmVyO1xuICByYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmVlc1J1bGVzIHtcbiAgYm9uZFJlZ2lzdHJhdGlvblBlcmNlbnQ6IG51bWJlcjtcbiAgdHJhbnNmZXJEdXR5OiB7XG4gICAgYnJhY2tldHM6IFRyYW5zZmVyRHV0eUJyYWNrZXRbXTtcbiAgfTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVzIHtcbiAgbG9hbkNvbnN0cmFpbnRzOiBMb2FuQ29uc3RyYWludHM7XG4gIGludGVyZXN0OiBJbnRlcmVzdFJ1bGVzO1xuICBpbnN1cmFuY2U6IEluc3VyYW5jZVJ1bGVzO1xuICBmZWVzOiBGZWVzUnVsZXM7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUnVsZU91dHB1dHMge1xuICBsb2FuQW1vdW50OiBcIm51bWJlclwiO1xuICBtb250aGx5UGF5bWVudDogXCJudW1iZXJcIjtcbiAgdG90YWxJbnRlcmVzdFBhaWQ6IFwibnVtYmVyXCI7XG4gIHRvdGFsUGFpZDogXCJudW1iZXJcIjtcbiAgZGVidFRvSW5jb21lUmF0aW86IFwibnVtYmVyXCI7XG4gIGlzQWZmb3JkYWJsZTogXCJib29sZWFuXCI7XG4gIHRyYW5zZmVyRHV0eTogXCJudW1iZXJcIjtcbiAgYm9uZFJlZ2lzdHJhdGlvbkZlZTogXCJudW1iZXJcIjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVTZXQge1xuICBtZXRhOiBSdWxlTWV0YTtcbiAgaW5wdXRzOiBSdWxlSW5wdXRzO1xuICBydWxlczogTW9ydGdhZ2VSdWxlcztcbiAgb3V0cHV0czogUnVsZU91dHB1dHM7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW1vcnRpemF0aW9uU2NoZWR1bGVJdGVtIHtcbiAgeWVhcjogbnVtYmVyO1xuICBwcmluY2lwYWw6IG51bWJlcjtcbiAgaW50ZXJlc3Q6IG51bWJlcjtcbiAgYmFsYW5jZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vcnRnYWdlSW5wdXQge1xuICBwcm9wZXJ0eVByaWNlOiBudW1iZXI7XG4gIGRvd25QYXltZW50OiBudW1iZXI7XG4gIGFubnVhbEludGVyZXN0UmF0ZTogbnVtYmVyOyAvLyBlLmcuLCAxMS43NSVcbiAgYW1vcnRpemF0aW9uWWVhcnM6IG51bWJlcjtcbiAgZ3Jvc3NNb250aGx5SW5jb21lOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VPdXRwdXQge1xuICBsb2FuQW1vdW50OiBudW1iZXI7XG4gIG1vbnRobHlQYXltZW50OiBudW1iZXI7XG4gIHRvdGFsSW50ZXJlc3RQYWlkOiBudW1iZXI7XG4gIHRvdGFsUGFpZDogbnVtYmVyO1xuICBkZWJ0VG9JbmNvbWVSYXRpbzogbnVtYmVyO1xuICBpc0FmZm9yZGFibGU6IGJvb2xlYW47XG4gIHRyYW5zZmVyRHV0eTogbnVtYmVyO1xuICBib25kUmVnaXN0cmF0aW9uRmVlOiBudW1iZXI7XG4gIGFtb3J0aXphdGlvblNjaGVkdWxlOiBBbW9ydGl6YXRpb25TY2hlZHVsZUl0ZW1bXTtcbn1cbiJdfQ==
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2Uvc291dGgtYWZyaWNhL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUnVsZU1ldGEgfSBmcm9tIFwiLi4vLi4vLi4vc2hhcmVkL2RvbWFpbi90eXBlc1wiO1xuaW1wb3J0IHsgT3RoZXJGZWVzIH0gZnJvbSBcIi4uLy4uL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFJ1bGVJbnB1dERlZmluaXRpb24ge1xuICB0eXBlOiBcIm51bWJlclwiIHwgXCJzdHJpbmdcIjtcbiAgbWluPzogbnVtYmVyO1xuICBtYXg/OiBudW1iZXI7XG4gIGVudW0/OiBzdHJpbmdbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSdWxlSW5wdXRzIHtcbiAgcHJvcGVydHlQcmljZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgZG93blBheW1lbnQ6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gIGludGVyZXN0UmF0ZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbiAgYW1vcnRpemF0aW9uWWVhcnM6IFJ1bGVJbnB1dERlZmluaXRpb247XG4gIGdyb3NzTW9udGhseUluY29tZTogUnVsZUlucHV0RGVmaW5pdGlvbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBMb2FuQ29uc3RyYWludHMge1xuICBtYXhMdHY6IG51bWJlcjtcbiAgbWluRG93blBheW1lbnRQZXJjZW50OiBudW1iZXI7XG4gIG1heEFtb3J0aXphdGlvblllYXJzOiBudW1iZXI7XG4gIG1heERlYnRUb0luY29tZVBlcmNlbnQ6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBJbnRlcmVzdFJ1bGVzIHtcbiAgdHlwZTogXCJmaXhlZFwiIHwgXCJ2YXJpYWJsZVwiO1xuICByYXRlUmFuZ2VQZXJjZW50OiB7IG1pbjogbnVtYmVyOyBtYXg6IG51bWJlciB9O1xuICBzdHJlc3NUZXN0QnVmZmVyUGVyY2VudDogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEluc3VyYW5jZVJ1bGVzIHtcbiAgcmVxdWlyZWQ6IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVHJhbnNmZXJEdXR5QnJhY2tldCB7XG4gIHVwVG8/OiBudW1iZXI7XG4gIGFib3ZlPzogbnVtYmVyO1xuICByYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmVlc1J1bGVzIHtcbiAgYm9uZFJlZ2lzdHJhdGlvblBlcmNlbnQ6IG51bWJlcjtcbiAgdHJhbnNmZXJEdXR5OiB7XG4gICAgYnJhY2tldHM6IFRyYW5zZmVyRHV0eUJyYWNrZXRbXTtcbiAgfTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVzIHtcbiAgbG9hbkNvbnN0cmFpbnRzOiBMb2FuQ29uc3RyYWludHM7XG4gIGludGVyZXN0OiBJbnRlcmVzdFJ1bGVzO1xuICBpbnN1cmFuY2U6IEluc3VyYW5jZVJ1bGVzO1xuICBmZWVzOiBGZWVzUnVsZXM7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUnVsZU91dHB1dHMge1xuICBsb2FuQW1vdW50OiBcIm51bWJlclwiO1xuICBtb250aGx5UGF5bWVudDogXCJudW1iZXJcIjtcbiAgdG90YWxJbnRlcmVzdFBhaWQ6IFwibnVtYmVyXCI7XG4gIHRvdGFsUGFpZDogXCJudW1iZXJcIjtcbiAgZGVidFRvSW5jb21lUmF0aW86IFwibnVtYmVyXCI7XG4gIGlzQWZmb3JkYWJsZTogXCJib29sZWFuXCI7XG4gIHRyYW5zZmVyRHV0eTogXCJudW1iZXJcIjtcbiAgYm9uZFJlZ2lzdHJhdGlvbkZlZTogXCJudW1iZXJcIjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVTZXQge1xuICBtZXRhOiBSdWxlTWV0YTtcbiAgaW5wdXRzOiBSdWxlSW5wdXRzO1xuICBydWxlczogTW9ydGdhZ2VSdWxlcztcbiAgb3V0cHV0czogUnVsZU91dHB1dHM7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW1vcnRpemF0aW9uU2NoZWR1bGVJdGVtIHtcbiAgeWVhcjogbnVtYmVyO1xuICBwcmluY2lwYWw6IG51bWJlcjtcbiAgaW50ZXJlc3Q6IG51bWJlcjtcbiAgYmFsYW5jZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vcnRnYWdlSW5wdXQge1xuICBwcm9wZXJ0eVByaWNlOiBudW1iZXI7XG4gIGRvd25QYXltZW50OiBudW1iZXI7XG4gIGFubnVhbEludGVyZXN0UmF0ZTogbnVtYmVyOyAvLyBlLmcuLCAxMS43NSVcbiAgYW1vcnRpemF0aW9uWWVhcnM6IG51bWJlcjtcbiAgZ3Jvc3NNb250aGx5SW5jb21lOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VPdXRwdXQge1xuICBsb2FuQW1vdW50OiBudW1iZXI7XG4gIG1vbnRobHlQYXltZW50OiBudW1iZXI7XG4gIHRvdGFsSW50ZXJlc3RQYWlkOiBudW1iZXI7XG4gIHRvdGFsUGFpZDogbnVtYmVyO1xuICBkZWJ0VG9JbmNvbWVSYXRpbzogbnVtYmVyO1xuICBpc0FmZm9yZGFibGU6IGJvb2xlYW47XG4gIHRyYW5zZmVyRHV0eTogbnVtYmVyO1xuICBib25kUmVnaXN0cmF0aW9uRmVlOiBudW1iZXI7XG4gIGFtb3J0aXphdGlvblNjaGVkdWxlOiBBbW9ydGl6YXRpb25TY2hlZHVsZUl0ZW1bXTtcbiAgb3RoZXJGZWVzOiBPdGhlckZlZXM7XG59XG4iXX0=
@@ -1,4 +1,5 @@
1
1
  import { RuleMeta } from "../../../shared/domain/types";
2
+ import { OtherFees } from "../../domain/types";
2
3
  export interface RuleInputDefinition {
3
4
  type: 'number' | 'string';
4
5
  min?: number;
@@ -85,8 +86,9 @@ export interface MortgageCalculationResult {
85
86
  loanAmount: number;
86
87
  insurancePremium: number;
87
88
  totalMortgage: number;
88
- paymentAmount: number;
89
+ monthlyPaymentAmount: number;
89
90
  totalInterestPaid: number;
90
91
  totalPaid: number;
91
92
  amortizationSchedule: AmortizationScheduleItem[];
93
+ otherFees: OtherFees;
92
94
  }
@@ -0,0 +1,14 @@
1
+ export interface OtherFees {
2
+ notaryFees: {
3
+ value: number;
4
+ label: string;
5
+ };
6
+ bankFees: {
7
+ value: number;
8
+ label: string;
9
+ };
10
+ monthlyInsuranceFees: {
11
+ value: number;
12
+ label: string;
13
+ };
14
+ }
@@ -5,6 +5,7 @@ export declare class FranceMortgageServiceImpl implements FranceMortgageService
5
5
  private _rules;
6
6
  constructor(input: MortgageInput, rules: MortgageRules);
7
7
  calculate(): MortgageOutput;
8
+ private calculateMonthlyPayment;
8
9
  private calculateAmortizationSchedule;
9
10
  private calculateLoanAmount;
10
11
  private ineligibleResult;
@@ -1,3 +1,4 @@
1
+ import { OtherFees } from "../../domain/types";
1
2
  export interface MortgageRules {
2
3
  maxDebtRatio: number;
3
4
  maxLoanDurationYears: number;
@@ -44,6 +45,10 @@ export interface MortgageInput {
44
45
  isNewBuild: boolean;
45
46
  }
46
47
  export interface MortgageOutput {
48
+ loanAmount: number;
49
+ totalPaid: number;
50
+ monthlyPayment: number;
51
+ totalInterestPaid: number;
47
52
  maxMonthlyPayment: number;
48
53
  maxLoanAmount: number;
49
54
  loanDurationYears: number;
@@ -53,4 +58,5 @@ export interface MortgageOutput {
53
58
  monthlyInsuranceCost: number;
54
59
  isEligible: boolean;
55
60
  amortizationSchedule: AmortizationScheduleItem[];
61
+ otherFees: OtherFees;
56
62
  }
@@ -1,4 +1,5 @@
1
1
  import { RuleMeta } from "../../../shared/domain/types";
2
+ import { OtherFees } from "../../domain/types";
2
3
  export interface RuleInputDefinition {
3
4
  type: "number" | "string";
4
5
  min?: number;
@@ -85,4 +86,5 @@ export interface MortgageOutput {
85
86
  transferDuty: number;
86
87
  bondRegistrationFee: number;
87
88
  amortizationSchedule: AmortizationScheduleItem[];
89
+ otherFees: OtherFees;
88
90
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@novha/calc-engines",
3
- "version": "1.6.2",
3
+ "version": "1.6.4",
4
4
  "main": "dist/index.js",
5
5
  "types": "dist/types/index.d.ts",
6
6
  "scripts": {
@@ -13,11 +13,12 @@ export class CanadaMortgageServiceImpl implements CanadaMortgageService {
13
13
  const {
14
14
  propertyPrice,
15
15
  downPayment,
16
- interestRate,
17
16
  amortizationYears,
18
17
  paymentFrequency
19
18
  } = input;
20
19
 
20
+ const interestRate = input.interestRate / 100;
21
+
21
22
  /* -----------------------------
22
23
  1. Loan Amount & LTV
23
24
  ------------------------------ */
@@ -100,10 +101,24 @@ export class CanadaMortgageServiceImpl implements CanadaMortgageService {
100
101
  loanAmount: loanAmount,
101
102
  insurancePremium: insurancePremium,
102
103
  totalMortgage: totalMortgage,
103
- paymentAmount: paymentAmount,
104
+ monthlyPaymentAmount: paymentAmount,
104
105
  totalInterestPaid: totalInterestPaid,
105
106
  totalPaid: totalPaid,
106
- amortizationSchedule: amortizationSchedule
107
+ amortizationSchedule: amortizationSchedule,
108
+ otherFees: {
109
+ notaryFees: {
110
+ value: 0,
111
+ label: 'NOTARY_FEES'
112
+ },
113
+ bankFees: {
114
+ value: 0,
115
+ label: 'BANK_FEES'
116
+ },
117
+ monthlyInsuranceFees: {
118
+ value: insurancePremium,
119
+ label: 'INSURANCE_PREMIUM'
120
+ }
121
+ }
107
122
  };
108
123
  }
109
124
 
@@ -1,4 +1,5 @@
1
1
  import { RuleMeta } from "../../../shared/domain/types";
2
+ import { OtherFees } from "../../domain/types";
2
3
 
3
4
  export interface RuleInputDefinition {
4
5
  type: 'number' | 'string';
@@ -101,8 +102,9 @@ export interface MortgageCalculationResult {
101
102
  loanAmount: number;
102
103
  insurancePremium: number;
103
104
  totalMortgage: number;
104
- paymentAmount: number;
105
+ monthlyPaymentAmount: number;
105
106
  totalInterestPaid: number;
106
107
  totalPaid: number;
107
108
  amortizationSchedule: AmortizationScheduleItem[];
109
+ otherFees: OtherFees;
108
110
  }
@@ -0,0 +1,14 @@
1
+ export interface OtherFees {
2
+ notaryFees: {
3
+ value: number;
4
+ label: string;
5
+ };
6
+ bankFees: {
7
+ value: number;
8
+ label: string;
9
+ };
10
+ monthlyInsuranceFees: {
11
+ value: number;
12
+ label: string;
13
+ };
14
+ }
@@ -63,41 +63,80 @@ export class FranceMortgageServiceImpl implements FranceMortgageService {
63
63
  loanDurationYears
64
64
  );
65
65
 
66
- const amortizationSchedule = this.calculateAmortizationSchedule(
66
+ const monthlyPayment = this.calculateMonthlyPayment(
67
67
  requiredLoanAmount,
68
68
  this._input.annualInterestRate,
69
69
  loanDurationYears
70
70
  );
71
71
 
72
+ const totalPaid = monthlyPayment * loanDurationYears * 12;
73
+ const totalInterestPaid = totalPaid - requiredLoanAmount;
74
+
75
+ const amortizationSchedule = this.calculateAmortizationSchedule(
76
+ requiredLoanAmount,
77
+ this._input.annualInterestRate,
78
+ loanDurationYears,
79
+ monthlyPayment
80
+ );
81
+
72
82
  return {
83
+ loanAmount: requiredLoanAmount,
84
+ totalPaid,
85
+ totalInterestPaid,
86
+ monthlyPayment,
87
+ requiredLoanAmount,
73
88
  maxMonthlyPayment,
74
89
  maxLoanAmount,
75
90
  totalProjectCost,
76
- requiredLoanAmount,
77
91
  loanDurationYears,
78
92
  debtRatio: debtRatio,
79
93
  monthlyInsuranceCost,
80
94
  isEligible: maxLoanAmount >= requiredLoanAmount,
81
- amortizationSchedule
95
+ amortizationSchedule,
96
+ otherFees: {
97
+ notaryFees: {
98
+ value: notaryFees,
99
+ label: 'NOTARY_FEES'
100
+ },
101
+ bankFees: {
102
+ value: bankFees,
103
+ label: 'BANK_FEES'
104
+ },
105
+ monthlyInsuranceFees: {
106
+ value: monthlyInsuranceCost,
107
+ label: 'MONTHLY_INSURANCE_FEES'
108
+ }
109
+ }
82
110
  };
83
111
  }
84
112
 
113
+ private calculateMonthlyPayment(principal: number, annualRate: number, years: number): number {
114
+ const monthlyRate = annualRate / 100 / 12;
115
+ const payments = years * 12;
116
+
117
+ if (monthlyRate === 0) {
118
+ return payments === 0 ? 0 : principal / payments;
119
+ }
120
+
121
+ return (
122
+ principal *
123
+ (monthlyRate / (1 - Math.pow(1 + monthlyRate, -payments)))
124
+ );
125
+ }
126
+
85
127
  private calculateAmortizationSchedule(
86
128
  loanAmount: number,
87
129
  annualRate: number,
88
- years: number
130
+ years: number,
131
+ monthlyPayment: number
89
132
  ): AmortizationScheduleItem[] {
90
133
  const monthlyRate = annualRate / 100 / 12;
91
134
  const totalPayments = years * 12;
92
135
 
93
- if (monthlyRate === 0 || loanAmount <= 0) {
136
+ if (monthlyRate === 0 || loanAmount <= 0 || monthlyPayment <= 0) {
94
137
  return [];
95
138
  }
96
139
 
97
- const monthlyPayment =
98
- (loanAmount * monthlyRate) /
99
- (1 - Math.pow(1 + monthlyRate, -totalPayments));
100
-
101
140
  const schedule: AmortizationScheduleItem[] = [];
102
141
  let balance = loanAmount;
103
142
 
@@ -152,8 +191,26 @@ export class FranceMortgageServiceImpl implements FranceMortgageService {
152
191
  loanDurationYears: 0,
153
192
  debtRatio: 0,
154
193
  monthlyInsuranceCost: 0,
194
+ loanAmount: 0,
195
+ monthlyPayment: 0,
196
+ totalInterestPaid: 0,
197
+ totalPaid: 0,
155
198
  isEligible: false,
156
- amortizationSchedule: []
199
+ amortizationSchedule: [],
200
+ otherFees: {
201
+ notaryFees: {
202
+ value: 0,
203
+ label: 'NOTARY_FEES'
204
+ },
205
+ bankFees: {
206
+ value: 0,
207
+ label: 'BANK_FEES'
208
+ },
209
+ monthlyInsuranceFees: {
210
+ value: 0,
211
+ label: 'MONTHLY_INSURANCE_FEES'
212
+ }
213
+ }
157
214
  };
158
215
  }
159
216
  }
@@ -1,3 +1,5 @@
1
+ import { OtherFees } from "../../domain/types";
2
+
1
3
  export interface MortgageRules {
2
4
  maxDebtRatio: number;
3
5
  maxLoanDurationYears: number;
@@ -53,6 +55,10 @@ export interface MortgageInput {
53
55
  }
54
56
 
55
57
  export interface MortgageOutput {
58
+ loanAmount: number;
59
+ totalPaid: number;
60
+ monthlyPayment: number;
61
+ totalInterestPaid: number;
56
62
  maxMonthlyPayment: number;
57
63
  maxLoanAmount: number;
58
64
  loanDurationYears: number;
@@ -62,4 +68,5 @@ export interface MortgageOutput {
62
68
  monthlyInsuranceCost: number;
63
69
  isEligible: boolean;
64
70
  amortizationSchedule: AmortizationScheduleItem[];
71
+ otherFees: OtherFees;
65
72
  }
@@ -50,7 +50,21 @@ export class SouthAfricaMortgageServiceImpl implements SouthAfricaMortgageServic
50
50
  isAffordable,
51
51
  transferDuty,
52
52
  bondRegistrationFee,
53
- amortizationSchedule
53
+ amortizationSchedule,
54
+ otherFees: {
55
+ notaryFees: {
56
+ value: bondRegistrationFee,
57
+ label: 'BOND_REGISTRATION_FEES'
58
+ },
59
+ bankFees: {
60
+ value: transferDuty,
61
+ label: 'TRANSFER_DUTY'
62
+ },
63
+ monthlyInsuranceFees: {
64
+ value: 0,
65
+ label: 'MONTHLY_INSURANCE_FEES'
66
+ }
67
+ }
54
68
  };
55
69
  }
56
70
 
@@ -1,4 +1,5 @@
1
1
  import { RuleMeta } from "../../../shared/domain/types";
2
+ import { OtherFees } from "../../domain/types";
2
3
 
3
4
  export interface RuleInputDefinition {
4
5
  type: "number" | "string";
@@ -95,4 +96,5 @@ export interface MortgageOutput {
95
96
  transferDuty: number;
96
97
  bondRegistrationFee: number;
97
98
  amortizationSchedule: AmortizationScheduleItem[];
99
+ otherFees: OtherFees;
98
100
  }