@novha/calc-engines 1.6.2 → 1.6.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.
@@ -39,12 +39,19 @@ 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,
@@ -52,14 +59,21 @@ class FranceMortgageServiceImpl {
52
59
  amortizationSchedule
53
60
  };
54
61
  }
55
- calculateAmortizationSchedule(loanAmount, annualRate, years) {
62
+ calculateMonthlyPayment(principal, annualRate, years) {
63
+ const monthlyRate = annualRate / 100 / 12;
64
+ const payments = years * 12;
65
+ if (monthlyRate === 0) {
66
+ return payments === 0 ? 0 : principal / payments;
67
+ }
68
+ return (principal *
69
+ (monthlyRate / (1 - Math.pow(1 + monthlyRate, -payments))));
70
+ }
71
+ calculateAmortizationSchedule(loanAmount, annualRate, years, monthlyPayment) {
56
72
  const monthlyRate = annualRate / 100 / 12;
57
73
  const totalPayments = years * 12;
58
- if (monthlyRate === 0 || loanAmount <= 0) {
74
+ if (monthlyRate === 0 || loanAmount <= 0 || monthlyPayment <= 0) {
59
75
  return [];
60
76
  }
61
- const monthlyPayment = (loanAmount * monthlyRate) /
62
- (1 - Math.pow(1 + monthlyRate, -totalPayments));
63
77
  const schedule = [];
64
78
  let balance = loanAmount;
65
79
  for (let year = 1; year <= years; year++) {
@@ -102,10 +116,14 @@ class FranceMortgageServiceImpl {
102
116
  loanDurationYears: 0,
103
117
  debtRatio: 0,
104
118
  monthlyInsuranceCost: 0,
119
+ loanAmount: 0,
120
+ monthlyPayment: 0,
121
+ totalInterestPaid: 0,
122
+ totalPaid: 0,
105
123
  isEligible: false,
106
124
  amortizationSchedule: []
107
125
  };
108
126
  }
109
127
  }
110
128
  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}"]}
129
+ //# 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 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        };\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        };\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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2UvZnJhbmNlL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZVJ1bGVzIHtcbiAgICBtYXhEZWJ0UmF0aW86IG51bWJlcjtcbiAgICBtYXhMb2FuRHVyYXRpb25ZZWFyczogbnVtYmVyO1xuICAgIG1heExvYW5EdXJhdGlvbk5ld0J1aWxkWWVhcnM6IG51bWJlcjtcbiAgICBtaW5Eb3duUGF5bWVudFJhdGU6IG51bWJlcjtcblxuICAgIGZpcnN0VGltZUJ1eWVyPzogRmlyc3RUaW1lQnV5ZXJSdWxlcztcblxuICAgIGluc3VyYW5jZTogTW9ydGdhZ2VJbnN1cmFuY2VSdWxlcztcbiAgICBmZWVzOiBNb3J0Z2FnZUZlZXNSdWxlcztcbiAgICBzdHJlc3NUZXN0OiBTdHJlc3NUZXN0UnVsZXM7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmlyc3RUaW1lQnV5ZXJSdWxlcyB7XG4gICAgZW5hYmxlZDogYm9vbGVhbjtcbiAgICBtYXhEZWJ0UmF0aW86IG51bWJlcjtcbiAgICBtYXhMb2FuRHVyYXRpb25ZZWFyczogbnVtYmVyO1xuICAgIHF1b3RhRGlzY2xhaW1lcjogc3RyaW5nO1xuICAgIHJlcXVpcmVzUHJpbWFyeVJlc2lkZW5jZTogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZUluc3VyYW5jZVJ1bGVzIHtcbiAgICBhdmVyYWdlUmF0ZTogbnVtYmVyO1xuICAgIGluY2x1ZGVkSW5EZWJ0UmF0aW86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VGZWVzUnVsZXMge1xuICBub3RhcnlSYXRlT2xkUHJvcGVydHk6IG51bWJlcjtcbiAgbm90YXJ5UmF0ZU5ld1Byb3BlcnR5OiBudW1iZXI7XG4gIGJhbmtGZWVzUmF0ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN0cmVzc1Rlc3RSdWxlcyB7XG4gICAgaW50ZXJlc3RSYXRlQnVmZmVyOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW1vcnRpemF0aW9uU2NoZWR1bGVJdGVtIHtcbiAgICB5ZWFyOiBudW1iZXI7XG4gICAgcHJpbmNpcGFsOiBudW1iZXI7XG4gICAgaW50ZXJlc3Q6IG51bWJlcjtcbiAgICBiYWxhbmNlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VJbnB1dCB7XG4gICAgcHJvcGVydHlQcmljZTogbnVtYmVyO1xuICAgIGRvd25QYXltZW50OiBudW1iZXI7XG4gICAgbmV0TW9udGhseUluY29tZTogbnVtYmVyO1xuICAgIGxvYW5EdXJhdGlvblllYXJzOiBudW1iZXI7XG4gICAgYW5udWFsSW50ZXJlc3RSYXRlOiBudW1iZXI7XG4gICAgaXNQcmltYXJ5UmVzaWRlbmNlOiBib29sZWFuO1xuICAgIGlzRmlyc3RUaW1lQnV5ZXI6IGJvb2xlYW47XG4gICAgaXNOZXdCdWlsZDogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb3J0Z2FnZU91dHB1dCB7XG4gICAgbG9hbkFtb3VudDogbnVtYmVyO1xuICAgIHRvdGFsUGFpZDogbnVtYmVyO1xuICAgIG1vbnRobHlQYXltZW50OiBudW1iZXI7XG4gICAgdG90YWxJbnRlcmVzdFBhaWQ6IG51bWJlcjtcbiAgICBtYXhNb250aGx5UGF5bWVudDogbnVtYmVyO1xuICAgIG1heExvYW5BbW91bnQ6IG51bWJlcjtcbiAgICBsb2FuRHVyYXRpb25ZZWFyczogbnVtYmVyO1xuICAgIGRlYnRSYXRpbzogbnVtYmVyO1xuICAgIHRvdGFsUHJvamVjdENvc3Q6IG51bWJlcjtcbiAgICByZXF1aXJlZExvYW5BbW91bnQ6IG51bWJlcjtcbiAgICBtb250aGx5SW5zdXJhbmNlQ29zdDogbnVtYmVyO1xuICAgIGlzRWxpZ2libGU6IGJvb2xlYW47XG4gICAgYW1vcnRpemF0aW9uU2NoZWR1bGU6IEFtb3J0aXphdGlvblNjaGVkdWxlSXRlbVtdO1xufVxuIl19
@@ -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;
@@ -44,6 +44,10 @@ export interface MortgageInput {
44
44
  isNewBuild: boolean;
45
45
  }
46
46
  export interface MortgageOutput {
47
+ loanAmount: number;
48
+ totalPaid: number;
49
+ monthlyPayment: number;
50
+ totalInterestPaid: number;
47
51
  maxMonthlyPayment: number;
48
52
  maxLoanAmount: number;
49
53
  loanDurationYears: number;
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.3",
4
4
  "main": "dist/index.js",
5
5
  "types": "dist/types/index.d.ts",
6
6
  "scripts": {
@@ -63,17 +63,31 @@ 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,
@@ -82,22 +96,33 @@ export class FranceMortgageServiceImpl implements FranceMortgageService {
82
96
  };
83
97
  }
84
98
 
99
+ private calculateMonthlyPayment(principal: number, annualRate: number, years: number): number {
100
+ const monthlyRate = annualRate / 100 / 12;
101
+ const payments = years * 12;
102
+
103
+ if (monthlyRate === 0) {
104
+ return payments === 0 ? 0 : principal / payments;
105
+ }
106
+
107
+ return (
108
+ principal *
109
+ (monthlyRate / (1 - Math.pow(1 + monthlyRate, -payments)))
110
+ );
111
+ }
112
+
85
113
  private calculateAmortizationSchedule(
86
114
  loanAmount: number,
87
115
  annualRate: number,
88
- years: number
116
+ years: number,
117
+ monthlyPayment: number
89
118
  ): AmortizationScheduleItem[] {
90
119
  const monthlyRate = annualRate / 100 / 12;
91
120
  const totalPayments = years * 12;
92
121
 
93
- if (monthlyRate === 0 || loanAmount <= 0) {
122
+ if (monthlyRate === 0 || loanAmount <= 0 || monthlyPayment <= 0) {
94
123
  return [];
95
124
  }
96
125
 
97
- const monthlyPayment =
98
- (loanAmount * monthlyRate) /
99
- (1 - Math.pow(1 + monthlyRate, -totalPayments));
100
-
101
126
  const schedule: AmortizationScheduleItem[] = [];
102
127
  let balance = loanAmount;
103
128
 
@@ -152,6 +177,10 @@ export class FranceMortgageServiceImpl implements FranceMortgageService {
152
177
  loanDurationYears: 0,
153
178
  debtRatio: 0,
154
179
  monthlyInsuranceCost: 0,
180
+ loanAmount: 0,
181
+ monthlyPayment: 0,
182
+ totalInterestPaid: 0,
183
+ totalPaid: 0,
155
184
  isEligible: false,
156
185
  amortizationSchedule: []
157
186
  };
@@ -53,6 +53,10 @@ export interface MortgageInput {
53
53
  }
54
54
 
55
55
  export interface MortgageOutput {
56
+ loanAmount: number;
57
+ totalPaid: number;
58
+ monthlyPayment: number;
59
+ totalInterestPaid: number;
56
60
  maxMonthlyPayment: number;
57
61
  maxLoanAmount: number;
58
62
  loanDurationYears: number;