@novha/calc-engines 2.0.2 → 3.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/dist/corporate/uk/UKCorporateTaxService.js +3 -0
  2. package/dist/corporate/uk/UKCorporateTaxServiceImpl.js +70 -0
  3. package/dist/corporate/uk/domain/types.js +3 -0
  4. package/dist/income-tax/uk/UKIncomeTaxService.js +3 -0
  5. package/dist/income-tax/uk/UKIncomeTaxServiceImpl.js +96 -0
  6. package/dist/income-tax/uk/domain/types.js +3 -0
  7. package/dist/index.js +9 -2
  8. package/dist/mortgage/uk/UKMortgageService.js +3 -0
  9. package/dist/mortgage/uk/UKMortgageServiceImpl.js +99 -0
  10. package/dist/mortgage/uk/domain/types.js +3 -0
  11. package/dist/types/corporate/uk/UKCorporateTaxService.d.ts +4 -0
  12. package/dist/types/corporate/uk/UKCorporateTaxServiceImpl.d.ts +10 -0
  13. package/dist/types/corporate/uk/domain/types.d.ts +32 -0
  14. package/dist/types/income-tax/uk/UKIncomeTaxService.d.ts +4 -0
  15. package/dist/types/income-tax/uk/UKIncomeTaxServiceImpl.d.ts +12 -0
  16. package/dist/types/income-tax/uk/domain/types.d.ts +27 -0
  17. package/dist/types/index.d.ts +6 -0
  18. package/dist/types/mortgage/uk/UKMortgageService.d.ts +4 -0
  19. package/dist/types/mortgage/uk/UKMortgageServiceImpl.d.ts +9 -0
  20. package/dist/types/mortgage/uk/domain/types.d.ts +49 -0
  21. package/package.json +1 -1
  22. package/src/corporate/uk/UKCorporateTaxService.ts +5 -0
  23. package/src/corporate/uk/UKCorporateTaxServiceImpl.ts +82 -0
  24. package/src/corporate/uk/domain/types.ts +38 -0
  25. package/src/income-tax/uk/UKIncomeTaxService.ts +5 -0
  26. package/src/income-tax/uk/UKIncomeTaxServiceImpl.ts +133 -0
  27. package/src/income-tax/uk/domain/types.ts +31 -0
  28. package/src/index.ts +21 -0
  29. package/src/mortgage/uk/UKMortgageService.ts +5 -0
  30. package/src/mortgage/uk/UKMortgageServiceImpl.ts +140 -0
  31. package/src/mortgage/uk/domain/types.ts +58 -0
  32. package/test/australia-income-tax.test.ts +10 -10
  33. package/test/uk-corporate-tax.test.ts +119 -0
  34. package/test/uk-income-tax.test.ts +146 -0
  35. package/test/uk-mortgage.test.ts +154 -0
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiVUtDb3Jwb3JhdGVUYXhTZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2NvcnBvcmF0ZS91ay9VS0NvcnBvcmF0ZVRheFNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlc3VsdCB9IGZyb20gXCIuL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFVLQ29ycG9yYXRlVGF4U2VydmljZSB7XG4gICAgY2FsY3VsYXRlKCk6IFJlc3VsdDtcbn1cbiJdfQ==
@@ -0,0 +1,70 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.UKCorporateTaxServiceImpl = void 0;
4
+ class UKCorporateTaxServiceImpl {
5
+ constructor(input, rules) {
6
+ this._input = input;
7
+ this._rules = rules;
8
+ }
9
+ calculate() {
10
+ const income = this._input.taxableIncome;
11
+ const { smallProfits, main, marginalRelief } = this._rules.regimes;
12
+ let tax = 0;
13
+ let breakdowns = [];
14
+ if (income <= 0) {
15
+ return { corporateTax: 0, effectiveTaxRate: 0, breakdowns: [] };
16
+ }
17
+ if (income <= marginalRelief.lowerLimit) {
18
+ tax = income * smallProfits.rate;
19
+ breakdowns = [{
20
+ from: '0',
21
+ to: `${marginalRelief.lowerLimit}`,
22
+ rate: smallProfits.rate,
23
+ amount: tax,
24
+ }];
25
+ }
26
+ else if (income >= marginalRelief.upperLimit) {
27
+ tax = income * main.rate;
28
+ breakdowns = [{
29
+ from: '0',
30
+ to: 'Above',
31
+ rate: main.rate,
32
+ amount: tax,
33
+ }];
34
+ }
35
+ else {
36
+ tax = this.applyMarginalRelief(income, marginalRelief);
37
+ breakdowns = this.buildMarginalReliefBreakdowns(income, tax, marginalRelief);
38
+ }
39
+ return {
40
+ corporateTax: tax,
41
+ effectiveTaxRate: income > 0 ? (tax / income) * 100 : 0,
42
+ breakdowns,
43
+ };
44
+ }
45
+ applyMarginalRelief(income, rules) {
46
+ const grossTax = income * rules.mainRate;
47
+ const relief = rules.standardFraction * (rules.upperLimit - income);
48
+ return grossTax - relief;
49
+ }
50
+ buildMarginalReliefBreakdowns(income, tax, rules) {
51
+ const grossTax = income * rules.mainRate;
52
+ const relief = rules.standardFraction * (rules.upperLimit - income);
53
+ return [
54
+ {
55
+ from: '0',
56
+ to: 'Above',
57
+ rate: rules.mainRate,
58
+ amount: grossTax,
59
+ },
60
+ {
61
+ from: `${rules.lowerLimit}`,
62
+ to: `${rules.upperLimit}`,
63
+ rate: -(rules.standardFraction),
64
+ amount: -relief,
65
+ },
66
+ ];
67
+ }
68
+ }
69
+ exports.UKCorporateTaxServiceImpl = UKCorporateTaxServiceImpl;
70
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiVUtDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb3Jwb3JhdGUvdWsvVUtDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFJQSxNQUFhLHlCQUF5QjtJQUlsQyxZQUFZLEtBQVksRUFBRSxLQUFZO1FBQ2xDLElBQUksQ0FBQyxNQUFNLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLElBQUksQ0FBQyxNQUFNLEdBQUcsS0FBSyxDQUFDO0lBQ3hCLENBQUM7SUFFRCxTQUFTO1FBQ0wsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxhQUFhLENBQUM7UUFDekMsTUFBTSxFQUFFLFlBQVksRUFBRSxJQUFJLEVBQUUsY0FBYyxFQUFFLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUM7UUFFbkUsSUFBSSxHQUFHLEdBQUcsQ0FBQyxDQUFDO1FBQ1osSUFBSSxVQUFVLEdBQWdCLEVBQUUsQ0FBQztRQUVqQyxJQUFJLE1BQU0sSUFBSSxDQUFDLEVBQUUsQ0FBQztZQUNkLE9BQU8sRUFBRSxZQUFZLEVBQUUsQ0FBQyxFQUFFLGdCQUFnQixFQUFFLENBQUMsRUFBRSxVQUFVLEVBQUUsRUFBRSxFQUFFLENBQUM7UUFDcEUsQ0FBQztRQUVELElBQUksTUFBTSxJQUFJLGNBQWMsQ0FBQyxVQUFVLEVBQUUsQ0FBQztZQUN0QyxHQUFHLEdBQUcsTUFBTSxHQUFHLFlBQVksQ0FBQyxJQUFJLENBQUM7WUFDakMsVUFBVSxHQUFHLENBQUM7b0JBQ1YsSUFBSSxFQUFFLEdBQUc7b0JBQ1QsRUFBRSxFQUFFLEdBQUcsY0FBYyxDQUFDLFVBQVUsRUFBRTtvQkFDbEMsSUFBSSxFQUFFLFlBQVksQ0FBQyxJQUFJO29CQUN2QixNQUFNLEVBQUUsR0FBRztpQkFDZCxDQUFDLENBQUM7UUFDUCxDQUFDO2FBQU0sSUFBSSxNQUFNLElBQUksY0FBYyxDQUFDLFVBQVUsRUFBRSxDQUFDO1lBQzdDLEdBQUcsR0FBRyxNQUFNLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQztZQUN6QixVQUFVLEdBQUcsQ0FBQztvQkFDVixJQUFJLEVBQUUsR0FBRztvQkFDVCxFQUFFLEVBQUUsT0FBTztvQkFDWCxJQUFJLEVBQUUsSUFBSSxDQUFDLElBQUk7b0JBQ2YsTUFBTSxFQUFFLEdBQUc7aUJBQ2QsQ0FBQyxDQUFDO1FBQ1AsQ0FBQzthQUFNLENBQUM7WUFDSixHQUFHLEdBQUcsSUFBSSxDQUFDLG1CQUFtQixDQUFDLE1BQU0sRUFBRSxjQUFjLENBQUMsQ0FBQztZQUN2RCxVQUFVLEdBQUcsSUFBSSxDQUFDLDZCQUE2QixDQUFDLE1BQU0sRUFBRSxHQUFHLEVBQUUsY0FBYyxDQUFDLENBQUM7UUFDakYsQ0FBQztRQUVELE9BQU87WUFDSCxZQUFZLEVBQUUsR0FBRztZQUNqQixnQkFBZ0IsRUFBRSxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsR0FBRyxNQUFNLENBQUMsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDdkQsVUFBVTtTQUNiLENBQUM7SUFDTixDQUFDO0lBRU8sbUJBQW1CLENBQUMsTUFBYyxFQUFFLEtBQXlCO1FBQ2pFLE1BQU0sUUFBUSxHQUFHLE1BQU0sR0FBRyxLQUFLLENBQUMsUUFBUSxDQUFDO1FBQ3pDLE1BQU0sTUFBTSxHQUFHLEtBQUssQ0FBQyxnQkFBZ0IsR0FBRyxDQUFDLEtBQUssQ0FBQyxVQUFVLEdBQUcsTUFBTSxDQUFDLENBQUM7UUFDcEUsT0FBTyxRQUFRLEdBQUcsTUFBTSxDQUFDO0lBQzdCLENBQUM7SUFFTyw2QkFBNkIsQ0FDakMsTUFBYyxFQUNkLEdBQVcsRUFDWCxLQUF5QjtRQUV6QixNQUFNLFFBQVEsR0FBRyxNQUFNLEdBQUcsS0FBSyxDQUFDLFFBQVEsQ0FBQztRQUN6QyxNQUFNLE1BQU0sR0FBRyxLQUFLLENBQUMsZ0JBQWdCLEdBQUcsQ0FBQyxLQUFLLENBQUMsVUFBVSxHQUFHLE1BQU0sQ0FBQyxDQUFDO1FBRXBFLE9BQU87WUFDSDtnQkFDSSxJQUFJLEVBQUUsR0FBRztnQkFDVCxFQUFFLEVBQUUsT0FBTztnQkFDWCxJQUFJLEVBQUUsS0FBSyxDQUFDLFFBQVE7Z0JBQ3BCLE1BQU0sRUFBRSxRQUFRO2FBQ25CO1lBQ0Q7Z0JBQ0ksSUFBSSxFQUFFLEdBQUcsS0FBSyxDQUFDLFVBQVUsRUFBRTtnQkFDM0IsRUFBRSxFQUFFLEdBQUcsS0FBSyxDQUFDLFVBQVUsRUFBRTtnQkFDekIsSUFBSSxFQUFFLENBQUMsQ0FBQyxLQUFLLENBQUMsZ0JBQWdCLENBQUM7Z0JBQy9CLE1BQU0sRUFBRSxDQUFDLE1BQU07YUFDbEI7U0FDSixDQUFDO0lBQ04sQ0FBQztDQUNKO0FBN0VELDhEQTZFQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEJyZWFrZG93biB9IGZyb20gXCIuLi9kb21haW4vdHlwZXNcIjtcbmltcG9ydCB7IFVLQ29ycG9yYXRlVGF4U2VydmljZSB9IGZyb20gXCIuL1VLQ29ycG9yYXRlVGF4U2VydmljZVwiO1xuaW1wb3J0IHsgSW5wdXQsIE1hcmdpbmFsUmVsaWVmUnVsZSwgUmVzdWx0LCBSdWxlcyB9IGZyb20gXCIuL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgY2xhc3MgVUtDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCBpbXBsZW1lbnRzIFVLQ29ycG9yYXRlVGF4U2VydmljZSB7XG4gICAgcHJpdmF0ZSBfaW5wdXQ6IElucHV0O1xuICAgIHByaXZhdGUgX3J1bGVzOiBSdWxlcztcblxuICAgIGNvbnN0cnVjdG9yKGlucHV0OiBJbnB1dCwgcnVsZXM6IFJ1bGVzKSB7XG4gICAgICAgIHRoaXMuX2lucHV0ID0gaW5wdXQ7XG4gICAgICAgIHRoaXMuX3J1bGVzID0gcnVsZXM7XG4gICAgfVxuXG4gICAgY2FsY3VsYXRlKCk6IFJlc3VsdCB7XG4gICAgICAgIGNvbnN0IGluY29tZSA9IHRoaXMuX2lucHV0LnRheGFibGVJbmNvbWU7XG4gICAgICAgIGNvbnN0IHsgc21hbGxQcm9maXRzLCBtYWluLCBtYXJnaW5hbFJlbGllZiB9ID0gdGhpcy5fcnVsZXMucmVnaW1lcztcblxuICAgICAgICBsZXQgdGF4ID0gMDtcbiAgICAgICAgbGV0IGJyZWFrZG93bnM6IEJyZWFrZG93bltdID0gW107XG5cbiAgICAgICAgaWYgKGluY29tZSA8PSAwKSB7XG4gICAgICAgICAgICByZXR1cm4geyBjb3Jwb3JhdGVUYXg6IDAsIGVmZmVjdGl2ZVRheFJhdGU6IDAsIGJyZWFrZG93bnM6IFtdIH07XG4gICAgICAgIH1cblxuICAgICAgICBpZiAoaW5jb21lIDw9IG1hcmdpbmFsUmVsaWVmLmxvd2VyTGltaXQpIHtcbiAgICAgICAgICAgIHRheCA9IGluY29tZSAqIHNtYWxsUHJvZml0cy5yYXRlO1xuICAgICAgICAgICAgYnJlYWtkb3ducyA9IFt7XG4gICAgICAgICAgICAgICAgZnJvbTogJzAnLFxuICAgICAgICAgICAgICAgIHRvOiBgJHttYXJnaW5hbFJlbGllZi5sb3dlckxpbWl0fWAsXG4gICAgICAgICAgICAgICAgcmF0ZTogc21hbGxQcm9maXRzLnJhdGUsXG4gICAgICAgICAgICAgICAgYW1vdW50OiB0YXgsXG4gICAgICAgICAgICB9XTtcbiAgICAgICAgfSBlbHNlIGlmIChpbmNvbWUgPj0gbWFyZ2luYWxSZWxpZWYudXBwZXJMaW1pdCkge1xuICAgICAgICAgICAgdGF4ID0gaW5jb21lICogbWFpbi5yYXRlO1xuICAgICAgICAgICAgYnJlYWtkb3ducyA9IFt7XG4gICAgICAgICAgICAgICAgZnJvbTogJzAnLFxuICAgICAgICAgICAgICAgIHRvOiAnQWJvdmUnLFxuICAgICAgICAgICAgICAgIHJhdGU6IG1haW4ucmF0ZSxcbiAgICAgICAgICAgICAgICBhbW91bnQ6IHRheCxcbiAgICAgICAgICAgIH1dO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgdGF4ID0gdGhpcy5hcHBseU1hcmdpbmFsUmVsaWVmKGluY29tZSwgbWFyZ2luYWxSZWxpZWYpO1xuICAgICAgICAgICAgYnJlYWtkb3ducyA9IHRoaXMuYnVpbGRNYXJnaW5hbFJlbGllZkJyZWFrZG93bnMoaW5jb21lLCB0YXgsIG1hcmdpbmFsUmVsaWVmKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBjb3Jwb3JhdGVUYXg6IHRheCxcbiAgICAgICAgICAgIGVmZmVjdGl2ZVRheFJhdGU6IGluY29tZSA+IDAgPyAodGF4IC8gaW5jb21lKSAqIDEwMCA6IDAsXG4gICAgICAgICAgICBicmVha2Rvd25zLFxuICAgICAgICB9O1xuICAgIH1cblxuICAgIHByaXZhdGUgYXBwbHlNYXJnaW5hbFJlbGllZihpbmNvbWU6IG51bWJlciwgcnVsZXM6IE1hcmdpbmFsUmVsaWVmUnVsZSk6IG51bWJlciB7XG4gICAgICAgIGNvbnN0IGdyb3NzVGF4ID0gaW5jb21lICogcnVsZXMubWFpblJhdGU7XG4gICAgICAgIGNvbnN0IHJlbGllZiA9IHJ1bGVzLnN0YW5kYXJkRnJhY3Rpb24gKiAocnVsZXMudXBwZXJMaW1pdCAtIGluY29tZSk7XG4gICAgICAgIHJldHVybiBncm9zc1RheCAtIHJlbGllZjtcbiAgICB9XG5cbiAgICBwcml2YXRlIGJ1aWxkTWFyZ2luYWxSZWxpZWZCcmVha2Rvd25zKFxuICAgICAgICBpbmNvbWU6IG51bWJlcixcbiAgICAgICAgdGF4OiBudW1iZXIsXG4gICAgICAgIHJ1bGVzOiBNYXJnaW5hbFJlbGllZlJ1bGUsXG4gICAgKTogQnJlYWtkb3duW10ge1xuICAgICAgICBjb25zdCBncm9zc1RheCA9IGluY29tZSAqIHJ1bGVzLm1haW5SYXRlO1xuICAgICAgICBjb25zdCByZWxpZWYgPSBydWxlcy5zdGFuZGFyZEZyYWN0aW9uICogKHJ1bGVzLnVwcGVyTGltaXQgLSBpbmNvbWUpO1xuXG4gICAgICAgIHJldHVybiBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgZnJvbTogJzAnLFxuICAgICAgICAgICAgICAgIHRvOiAnQWJvdmUnLFxuICAgICAgICAgICAgICAgIHJhdGU6IHJ1bGVzLm1haW5SYXRlLFxuICAgICAgICAgICAgICAgIGFtb3VudDogZ3Jvc3NUYXgsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIGZyb206IGAke3J1bGVzLmxvd2VyTGltaXR9YCxcbiAgICAgICAgICAgICAgICB0bzogYCR7cnVsZXMudXBwZXJMaW1pdH1gLFxuICAgICAgICAgICAgICAgIHJhdGU6IC0ocnVsZXMuc3RhbmRhcmRGcmFjdGlvbiksXG4gICAgICAgICAgICAgICAgYW1vdW50OiAtcmVsaWVmLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgXTtcbiAgICB9XG59XG4iXX0=
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvY29ycG9yYXRlL3VrL2RvbWFpbi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQnJlYWtkb3duIH0gZnJvbSBcIi4uLy4uL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIEZsYXRUYXhSdWxlIHtcbiAgICB0eXBlOiAnZmxhdCc7XG4gICAgcmF0ZTogbnVtYmVyO1xuICAgIGNvbmRpdGlvbnM/OiB7XG4gICAgICAgIG1heEluY29tZT86IG51bWJlcjtcbiAgICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1hcmdpbmFsUmVsaWVmUnVsZSB7XG4gICAgdHlwZTogJ21hcmdpbmFsX3JlbGllZic7XG4gICAgbWFpblJhdGU6IG51bWJlcjtcbiAgICBzbWFsbFByb2ZpdHNSYXRlOiBudW1iZXI7XG4gICAgdXBwZXJMaW1pdDogbnVtYmVyO1xuICAgIGxvd2VyTGltaXQ6IG51bWJlcjtcbiAgICBzdGFuZGFyZEZyYWN0aW9uOiBudW1iZXI7XG59XG5cbmV4cG9ydCB0eXBlIFRheFJ1bGUgPSBGbGF0VGF4UnVsZSB8IE1hcmdpbmFsUmVsaWVmUnVsZTtcblxuZXhwb3J0IGludGVyZmFjZSBSdWxlcyB7XG4gICAgcmVnaW1lczoge1xuICAgICAgICBzbWFsbFByb2ZpdHM6IEZsYXRUYXhSdWxlO1xuICAgICAgICBtYWluOiBGbGF0VGF4UnVsZTtcbiAgICAgICAgbWFyZ2luYWxSZWxpZWY6IE1hcmdpbmFsUmVsaWVmUnVsZTtcbiAgICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIElucHV0IHtcbiAgICB0YXhhYmxlSW5jb21lOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVzdWx0IHtcbiAgICBjb3Jwb3JhdGVUYXg6IG51bWJlcjtcbiAgICBlZmZlY3RpdmVUYXhSYXRlOiBudW1iZXI7XG4gICAgYnJlYWtkb3duczogQnJlYWtkb3duW107XG59XG4iXX0=
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiVUtJbmNvbWVUYXhTZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2luY29tZS10YXgvdWsvVUtJbmNvbWVUYXhTZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wdXRlZEluY29tZVRheFZhbHVlcyB9IGZyb20gXCIuL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFVLSW5jb21lVGF4U2VydmljZSB7XG4gICAgY2FsY3VsYXRlTmV0SW5jb21lKCk6IENvbXB1dGVkSW5jb21lVGF4VmFsdWVzO1xufVxuIl19
@@ -0,0 +1,96 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.UKIncomeTaxServiceImpl = void 0;
4
+ class UKIncomeTaxServiceImpl {
5
+ constructor(income, rules) {
6
+ this._income = income;
7
+ this._rules = rules;
8
+ }
9
+ calculateNetIncome() {
10
+ const effectivePA = this.computePersonalAllowance(this._income, this._rules.personalAllowance);
11
+ const taxableIncome = Math.max(0, this._income - effectivePA);
12
+ const { grossTax, bracketBreakdown } = this.computeTaxBrackets(taxableIncome);
13
+ const ni = this.computeNationalInsurance(this._income, this._rules.nationalInsurance);
14
+ const totalDeductions = grossTax + ni;
15
+ const netIncome = this._income - totalDeductions;
16
+ return {
17
+ grossIncome: this._income,
18
+ incomeTax: this.round(grossTax),
19
+ nationalInsurance: this.round(ni),
20
+ personalAllowance: effectivePA,
21
+ totalDeductions: this.round(totalDeductions),
22
+ netIncome: this.round(netIncome),
23
+ effectiveTaxRate: this._income > 0 ? this.round(grossTax / this._income, 4) : 0,
24
+ taxBracketBreakdown: bracketBreakdown,
25
+ };
26
+ }
27
+ computePersonalAllowance(income, rules) {
28
+ if (income <= rules.taperThreshold) {
29
+ return rules.amount;
30
+ }
31
+ const reduction = Math.floor((income - rules.taperThreshold) * rules.taperRate);
32
+ return Math.max(0, rules.amount - reduction);
33
+ }
34
+ computeTaxBrackets(taxableIncome) {
35
+ let tax = 0;
36
+ const bracketBreakdown = [];
37
+ for (let index = 0; index < this._rules.taxBrackets.length; index++) {
38
+ const bracket = this._rules.taxBrackets[index];
39
+ if (taxableIncome <= bracket.from) {
40
+ bracketBreakdown.push({
41
+ bracketIndex: index,
42
+ bracketName: `Bracket ${index + 1}`,
43
+ from: bracket.from,
44
+ to: bracket.to ?? null,
45
+ rate: bracket.rate,
46
+ amountInBracket: 0,
47
+ taxOnAmount: 0,
48
+ });
49
+ break;
50
+ }
51
+ const upper = bracket.to ?? taxableIncome;
52
+ const taxableAmount = Math.min(upper, taxableIncome) - bracket.from;
53
+ if (taxableAmount > 0) {
54
+ const taxOnAmount = taxableAmount * bracket.rate;
55
+ tax += taxOnAmount;
56
+ bracketBreakdown.push({
57
+ bracketIndex: index,
58
+ bracketName: `Bracket ${index + 1}`,
59
+ from: bracket.from,
60
+ to: bracket.to ?? null,
61
+ rate: bracket.rate,
62
+ amountInBracket: taxableAmount,
63
+ taxOnAmount,
64
+ });
65
+ }
66
+ else {
67
+ bracketBreakdown.push({
68
+ bracketIndex: index,
69
+ bracketName: `Bracket ${index + 1}`,
70
+ from: bracket.from,
71
+ to: bracket.to ?? null,
72
+ rate: bracket.rate,
73
+ amountInBracket: 0,
74
+ taxOnAmount: 0,
75
+ });
76
+ }
77
+ }
78
+ return { grossTax: tax, bracketBreakdown };
79
+ }
80
+ computeNationalInsurance(income, rules) {
81
+ if (income <= rules.primaryThreshold) {
82
+ return 0;
83
+ }
84
+ if (income <= rules.upperEarningsLimit) {
85
+ return (income - rules.primaryThreshold) * rules.mainRate;
86
+ }
87
+ const mainBand = (rules.upperEarningsLimit - rules.primaryThreshold) * rules.mainRate;
88
+ const upperBand = (income - rules.upperEarningsLimit) * rules.upperRate;
89
+ return mainBand + upperBand;
90
+ }
91
+ round(value, decimals = 2) {
92
+ return Number(value.toFixed(decimals));
93
+ }
94
+ }
95
+ exports.UKIncomeTaxServiceImpl = UKIncomeTaxServiceImpl;
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UKIncomeTaxServiceImpl.js","sourceRoot":"","sources":["../../../src/income-tax/uk/UKIncomeTaxServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { BracketAllocation } from \"../domain/types\";\nimport { UKIncomeTaxService } from \"./UKIncomeTaxService\";\nimport {\n    ComputedIncomeTaxValues,\n    IncomeTaxRules,\n    NationalInsuranceRules,\n    PersonalAllowanceRules,\n} from \"./domain/types\";\n\nexport class UKIncomeTaxServiceImpl implements UKIncomeTaxService {\n    private _income: number;\n    private _rules: IncomeTaxRules;\n\n    constructor(income: number, rules: IncomeTaxRules) {\n        this._income = income;\n        this._rules = rules;\n    }\n\n    public calculateNetIncome(): ComputedIncomeTaxValues {\n        const effectivePA = this.computePersonalAllowance(\n            this._income,\n            this._rules.personalAllowance,\n        );\n        const taxableIncome = Math.max(0, this._income - effectivePA);\n\n        const { grossTax, bracketBreakdown } = this.computeTaxBrackets(taxableIncome);\n\n        const ni = this.computeNationalInsurance(\n            this._income,\n            this._rules.nationalInsurance,\n        );\n\n        const totalDeductions = grossTax + ni;\n        const netIncome = this._income - totalDeductions;\n\n        return {\n            grossIncome: this._income,\n            incomeTax: this.round(grossTax),\n            nationalInsurance: this.round(ni),\n            personalAllowance: effectivePA,\n            totalDeductions: this.round(totalDeductions),\n            netIncome: this.round(netIncome),\n            effectiveTaxRate: this._income > 0 ? this.round(grossTax / this._income, 4) : 0,\n            taxBracketBreakdown: bracketBreakdown,\n        };\n    }\n\n    private computePersonalAllowance(\n        income: number,\n        rules: PersonalAllowanceRules,\n    ): number {\n        if (income <= rules.taperThreshold) {\n            return rules.amount;\n        }\n\n        const reduction = Math.floor((income - rules.taperThreshold) * rules.taperRate);\n        return Math.max(0, rules.amount - reduction);\n    }\n\n    private computeTaxBrackets(\n        taxableIncome: number,\n    ): { grossTax: number; bracketBreakdown: BracketAllocation[] } {\n        let tax = 0;\n        const bracketBreakdown: BracketAllocation[] = [];\n\n        for (let index = 0; index < this._rules.taxBrackets.length; index++) {\n            const bracket = this._rules.taxBrackets[index];\n\n            if (taxableIncome <= bracket.from) {\n                bracketBreakdown.push({\n                    bracketIndex: index,\n                    bracketName: `Bracket ${index + 1}`,\n                    from: bracket.from,\n                    to: bracket.to ?? null,\n                    rate: bracket.rate,\n                    amountInBracket: 0,\n                    taxOnAmount: 0,\n                });\n                break;\n            }\n\n            const upper = bracket.to ?? taxableIncome;\n            const taxableAmount = Math.min(upper, taxableIncome) - bracket.from;\n\n            if (taxableAmount > 0) {\n                const taxOnAmount = taxableAmount * bracket.rate;\n                tax += taxOnAmount;\n                bracketBreakdown.push({\n                    bracketIndex: index,\n                    bracketName: `Bracket ${index + 1}`,\n                    from: bracket.from,\n                    to: bracket.to ?? null,\n                    rate: bracket.rate,\n                    amountInBracket: taxableAmount,\n                    taxOnAmount,\n                });\n            } else {\n                bracketBreakdown.push({\n                    bracketIndex: index,\n                    bracketName: `Bracket ${index + 1}`,\n                    from: bracket.from,\n                    to: bracket.to ?? null,\n                    rate: bracket.rate,\n                    amountInBracket: 0,\n                    taxOnAmount: 0,\n                });\n            }\n        }\n\n        return { grossTax: tax, bracketBreakdown };\n    }\n\n    private computeNationalInsurance(\n        income: number,\n        rules: NationalInsuranceRules,\n    ): number {\n        if (income <= rules.primaryThreshold) {\n            return 0;\n        }\n\n        if (income <= rules.upperEarningsLimit) {\n            return (income - rules.primaryThreshold) * rules.mainRate;\n        }\n\n        const mainBand = (rules.upperEarningsLimit - rules.primaryThreshold) * rules.mainRate;\n        const upperBand = (income - rules.upperEarningsLimit) * rules.upperRate;\n        return mainBand + upperBand;\n    }\n\n    private round(value: number, decimals = 2): number {\n        return Number(value.toFixed(decimals));\n    }\n}\n"]}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvaW5jb21lLXRheC91ay9kb21haW4vdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEJyYWNrZXRBbGxvY2F0aW9uLCBUYXhCcmFja2V0IH0gZnJvbSBcIi4uLy4uL2RvbWFpbi90eXBlc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIEluY29tZVRheFJ1bGVzIHtcbiAgICB0YXhCcmFja2V0czogVGF4QnJhY2tldFtdO1xuICAgIHBlcnNvbmFsQWxsb3dhbmNlOiBQZXJzb25hbEFsbG93YW5jZVJ1bGVzO1xuICAgIG5hdGlvbmFsSW5zdXJhbmNlOiBOYXRpb25hbEluc3VyYW5jZVJ1bGVzO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBlcnNvbmFsQWxsb3dhbmNlUnVsZXMge1xuICAgIGFtb3VudDogbnVtYmVyO1xuICAgIHRhcGVyVGhyZXNob2xkOiBudW1iZXI7XG4gICAgdGFwZXJSYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTmF0aW9uYWxJbnN1cmFuY2VSdWxlcyB7XG4gICAgcHJpbWFyeVRocmVzaG9sZDogbnVtYmVyO1xuICAgIHVwcGVyRWFybmluZ3NMaW1pdDogbnVtYmVyO1xuICAgIG1haW5SYXRlOiBudW1iZXI7XG4gICAgdXBwZXJSYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29tcHV0ZWRJbmNvbWVUYXhWYWx1ZXMge1xuICAgIGdyb3NzSW5jb21lOiBudW1iZXI7XG4gICAgaW5jb21lVGF4OiBudW1iZXI7XG4gICAgbmF0aW9uYWxJbnN1cmFuY2U6IG51bWJlcjtcbiAgICBwZXJzb25hbEFsbG93YW5jZTogbnVtYmVyO1xuICAgIHRvdGFsRGVkdWN0aW9uczogbnVtYmVyO1xuICAgIG5ldEluY29tZTogbnVtYmVyO1xuICAgIGVmZmVjdGl2ZVRheFJhdGU6IG51bWJlcjtcbiAgICB0YXhCcmFja2V0QnJlYWtkb3duOiBCcmFja2V0QWxsb2NhdGlvbltdO1xufVxuIl19
package/dist/index.js CHANGED
@@ -14,7 +14,7 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
14
  for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
15
  };
16
16
  Object.defineProperty(exports, "__esModule", { value: true });
17
- exports.AustraliaCorporateTaxService = exports.AustraliaMortgageService = exports.AustraliaIncomeTaxService = exports.SouthAfricaCorporateTaxService = exports.SouthAfricaMortgageService = exports.SouthAfricaIncomeTaxService = exports.FranceCorporateTaxService = exports.FranceMortgageService = exports.FranceIncomeTaxService = exports.CanadaCorporateTaxService = exports.CanadaMortgageService = exports.CanadaIncomeTaxService = exports.CalculatorType = void 0;
17
+ exports.UKMortgageService = exports.UKCorporateTaxService = exports.UKIncomeTaxService = exports.AustraliaCorporateTaxService = exports.AustraliaMortgageService = exports.AustraliaIncomeTaxService = exports.SouthAfricaCorporateTaxService = exports.SouthAfricaMortgageService = exports.SouthAfricaIncomeTaxService = exports.FranceCorporateTaxService = exports.FranceMortgageService = exports.FranceIncomeTaxService = exports.CanadaCorporateTaxService = exports.CanadaMortgageService = exports.CanadaIncomeTaxService = exports.CalculatorType = void 0;
18
18
  var types_1 = require("./shared/domain/types");
19
19
  Object.defineProperty(exports, "CalculatorType", { enumerable: true, get: function () { return types_1.CalculatorType; } });
20
20
  // Canada
@@ -45,5 +45,12 @@ var AustraliaMortgageServiceImpl_1 = require("./mortgage/australia/AustraliaMort
45
45
  Object.defineProperty(exports, "AustraliaMortgageService", { enumerable: true, get: function () { return AustraliaMortgageServiceImpl_1.AustraliaMortgageServiceImpl; } });
46
46
  var AustraliaCorporateTaxServiceImpl_1 = require("./corporate/australia/AustraliaCorporateTaxServiceImpl");
47
47
  Object.defineProperty(exports, "AustraliaCorporateTaxService", { enumerable: true, get: function () { return AustraliaCorporateTaxServiceImpl_1.AustraliaCorporateTaxServiceImpl; } });
48
+ // United Kingdom
49
+ var UKIncomeTaxServiceImpl_1 = require("./income-tax/uk/UKIncomeTaxServiceImpl");
50
+ Object.defineProperty(exports, "UKIncomeTaxService", { enumerable: true, get: function () { return UKIncomeTaxServiceImpl_1.UKIncomeTaxServiceImpl; } });
51
+ var UKCorporateTaxServiceImpl_1 = require("./corporate/uk/UKCorporateTaxServiceImpl");
52
+ Object.defineProperty(exports, "UKCorporateTaxService", { enumerable: true, get: function () { return UKCorporateTaxServiceImpl_1.UKCorporateTaxServiceImpl; } });
53
+ var UKMortgageServiceImpl_1 = require("./mortgage/uk/UKMortgageServiceImpl");
54
+ Object.defineProperty(exports, "UKMortgageService", { enumerable: true, get: function () { return UKMortgageServiceImpl_1.UKMortgageServiceImpl; } });
48
55
  __exportStar(require("./income-tax/domain/types"), exports);
49
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSwrQ0FBdUQ7QUFBOUMsdUdBQUEsY0FBYyxPQUFBO0FBRXZCLFNBQVM7QUFDVCw2RkFBc0g7QUFBN0csb0lBQUEsMEJBQTBCLE9BQTBCO0FBQzdELHlGQUFpSDtBQUF4RyxrSUFBQSx5QkFBeUIsT0FBeUI7QUFXM0Qsa0dBQThIO0FBQXJILDBJQUFBLDZCQUE2QixPQUE2QjtBQU9uRSxTQUFTO0FBQ1QsNkZBQXNIO0FBQTdHLG9JQUFBLDBCQUEwQixPQUEwQjtBQUM3RCx5RkFBaUg7QUFBeEcsa0lBQUEseUJBQXlCLE9BQXlCO0FBVzNELGtHQUE4SDtBQUFySCwwSUFBQSw2QkFBNkIsT0FBNkI7QUFPbkUsZUFBZTtBQUNmLDZHQUEySTtBQUFsSSw4SUFBQSwrQkFBK0IsT0FBK0I7QUFDdkUseUdBQXNJO0FBQTdILDRJQUFBLDhCQUE4QixPQUE4QjtBQVdyRSxrSEFBbUo7QUFBMUksb0pBQUEsa0NBQWtDLE9BQWtDO0FBTzdFLFlBQVk7QUFDWixzR0FBa0k7QUFBekgsMElBQUEsNkJBQTZCLE9BQTZCO0FBTW5FLGtHQUE2SDtBQUFwSCx3SUFBQSw0QkFBNEIsT0FBNEI7QUFPakUsMkdBQTBJO0FBQWpJLGdKQUFBLGdDQUFnQyxPQUFnQztBQVF6RSw0REFBMEMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBDYWxjdWxhdG9yVHlwZSB9IGZyb20gJy4vc2hhcmVkL2RvbWFpbi90eXBlcyc7XG5cbi8vIENhbmFkYVxuZXhwb3J0IHsgQ2FuYWRhSW5jb21lVGF4U2VydmljZUltcGwgYXMgQ2FuYWRhSW5jb21lVGF4U2VydmljZSB9IGZyb20gJy4vaW5jb21lLXRheC9jYW5hZGEvQ2FuYWRhSW5jb21lVGF4U2VydmljZUltcGwnO1xuZXhwb3J0IHsgQ2FuYWRhTW9ydGdhZ2VTZXJ2aWNlSW1wbCBhcyBDYW5hZGFNb3J0Z2FnZVNlcnZpY2UgfSBmcm9tICcuL21vcnRnYWdlL2NhbmFkYS9DYW5hZGFNb3J0Z2FnZVNlcnZpY2VJbXBsJztcbmV4cG9ydCB7XG4gICAgQ29tcHV0ZWRJbmNvbWVUYXhWYWx1ZXMgYXMgQ2FuYWRhQ29tcHV0ZWRJbmNvbWVUYXhWYWx1ZXMsXG4gICAgSW5jb21lVGF4UnVsZXMgYXMgQ2FuYWRhSW5jb21lVGF4UnVsZXMsXG59IGZyb20gJy4vaW5jb21lLXRheC9jYW5hZGEvZG9tYWluL3R5cGVzJztcbmV4cG9ydCB7XG4gICAgTW9ydGdhZ2VSdWxlcyBhcyBDYW5hZGFNb3J0Z2FnZVJ1bGVzLFxuICAgIE1vcnRnYWdlQ2FsY3VsYXRpb25JbnB1dCBhcyBDYW5hZGFNb3J0Z2FnZUNhbGN1bGF0aW9uSW5wdXQsXG4gICAgTW9ydGdhZ2VDYWxjdWxhdGlvblJlc3VsdCBhcyBDYW5hZGFNb3J0Z2FnZUNhbGN1bGF0aW9uUmVzdWx0LFxufSBmcm9tICcuL21vcnRnYWdlL2NhbmFkYS9kb21haW4vdHlwZXMnO1xuXG5leHBvcnQgeyBDYW5hZGFDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCBhcyBDYW5hZGFDb3Jwb3JhdGVUYXhTZXJ2aWNlIH0gZnJvbSAnLi9jb3Jwb3JhdGUvY2FuYWRhL0NhbmFkYUNvcnBvcmF0ZVRheFNlcnZpY2VJbXBsJztcbmV4cG9ydCB7XG4gICAgSW5wdXQgYXMgQ2FuYWRhQ29ycG9yYXRlVGF4SW5wdXQsXG4gICAgUnVsZXMgYXMgQ2FuYWRhQ29ycG9yYXRlVGF4UnVsZXMsXG4gICAgUmVzdWx0IGFzIENhbmFkYUNvcnBvcmF0ZVRheFJlc3VsdCxcbn0gZnJvbSAnLi9jb3Jwb3JhdGUvY2FuYWRhL2RvbWFpbi90eXBlcyc7XG5cbi8vIEZyYW5jZVxuZXhwb3J0IHsgRnJhbmNlSW5jb21lVGF4U2VydmljZUltcGwgYXMgRnJhbmNlSW5jb21lVGF4U2VydmljZSB9IGZyb20gJy4vaW5jb21lLXRheC9mcmFuY2UvRnJhbmNlSW5jb21lVGF4U2VydmljZUltcGwnO1xuZXhwb3J0IHsgRnJhbmNlTW9ydGdhZ2VTZXJ2aWNlSW1wbCBhcyBGcmFuY2VNb3J0Z2FnZVNlcnZpY2UgfSBmcm9tICcuL21vcnRnYWdlL2ZyYW5jZS9GcmFuY2VNb3J0Z2FnZVNlcnZpY2VJbXBsJztcbmV4cG9ydCB7IFxuICAgIENvbXB1dGVkSW5jb21lVGF4VmFsdWVzIGFzIEZyYW5jZUNvbXB1dGVkSW5jb21lVGF4VmFsdWVzLFxuICAgIEluY29tZVRheFJ1bGVzIGFzIEZyYW5jZUluY29tZVRheFJ1bGVzLFxufSBmcm9tICcuL2luY29tZS10YXgvZnJhbmNlL2RvbWFpbi90eXBlcyc7XG5leHBvcnQge1xuICAgIE1vcnRnYWdlUnVsZXMgYXMgRnJhbmNlTW9ydGdhZ2VSdWxlcyxcbiAgICBNb3J0Z2FnZUlucHV0IGFzIEZyYW5jZU1vcnRnYWdlSW5wdXQsXG4gICAgTW9ydGdhZ2VPdXRwdXQgYXMgRnJhbmNlTW9ydGdhZ2VDYWxjdWxhdGlvblJlc3VsdCxcbn0gZnJvbSAnLi9tb3J0Z2FnZS9mcmFuY2UvZG9tYWluL3R5cGVzJztcblxuZXhwb3J0IHsgRnJhbmNlQ29ycG9yYXRlVGF4U2VydmljZUltcGwgYXMgRnJhbmNlQ29ycG9yYXRlVGF4U2VydmljZSB9IGZyb20gJy4vY29ycG9yYXRlL2ZyYW5jZS9GcmFuY2VDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIElucHV0IGFzIEZyYW5jZUNvcnBvcmF0ZVRheElucHV0LFxuICAgIFJ1bGVzIGFzIEZyYW5jZUNvcnBvcmF0ZVRheFJ1bGVzLFxuICAgIFJlc3VsdCBhcyBGcmFuY2VDb3Jwb3JhdGVUYXhSZXN1bHQsXG59IGZyb20gJy4vY29ycG9yYXRlL2ZyYW5jZS9kb21haW4vdHlwZXMnO1xuXG4vLyBTb3V0aCBBZnJpY2FcbmV4cG9ydCB7IFNvdXRoQWZyaWNhSW5jb21lVGF4U2VydmljZUltcGwgYXMgU291dGhBZnJpY2FJbmNvbWVUYXhTZXJ2aWNlIH0gZnJvbSAnLi9pbmNvbWUtdGF4L3NvdXRoLWFmcmljYS9Tb3V0aEFmcmljYUluY29tZVRheFNlcnZpY2VJbXBsJztcbmV4cG9ydCB7IFNvdXRoQWZyaWNhTW9ydGdhZ2VTZXJ2aWNlSW1wbCBhcyBTb3V0aEFmcmljYU1vcnRnYWdlU2VydmljZSB9IGZyb20gJy4vbW9ydGdhZ2Uvc291dGgtYWZyaWNhL1NvdXRoQWZyaWNhTW9ydGdhZ2VTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIENvbXB1dGVkSW5jb21lVGF4VmFsdWVzIGFzIFNvdXRoQWZyaWNhQ29tcHV0ZWRJbmNvbWVUYXhWYWx1ZXMsXG4gICAgSW5jb21lVGF4UnVsZXMgYXMgU291dGhBZnJpY2FJbmNvbWVUYXhSdWxlcyxcbn0gZnJvbSAnLi9pbmNvbWUtdGF4L3NvdXRoLWFmcmljYS9kb21haW4vdHlwZXMnO1xuZXhwb3J0IHtcbiAgICBNb3J0Z2FnZVJ1bGVzIGFzIFNvdXRoQWZyaWNhTW9ydGdhZ2VSdWxlcyxcbiAgICBNb3J0Z2FnZUlucHV0IGFzIFNvdXRoQWZyaWNhTW9ydGdhZ2VJbnB1dCxcbiAgICBNb3J0Z2FnZU91dHB1dCBhcyBTb3V0aEFmcmljYU1vcnRnYWdlT3V0cHV0LFxufSBmcm9tICcuL21vcnRnYWdlL3NvdXRoLWFmcmljYS9kb21haW4vdHlwZXMnO1xuXG5leHBvcnQgeyBTb3V0aEFmcmljYUNvcnBvcmF0ZVRheFNlcnZpY2VJbXBsIGFzIFNvdXRoQWZyaWNhQ29ycG9yYXRlVGF4U2VydmljZSB9IGZyb20gJy4vY29ycG9yYXRlL3NvdXRoLWFmcmljYS9Tb3V0aEFmcmljYUNvcnBvcmF0ZVRheFNlcnZpY2VJbXBsJztcbmV4cG9ydCB7XG4gICAgUnVsZXMgYXMgU291dGhBZnJpY2FDb3Jwb3JhdGVUYXhSdWxlcyxcbiAgICBJbnB1dCBhcyBTb3V0aEFmcmljYUNvcnBvcmF0ZVRheElucHV0LFxuICAgIFJlc3VsdCBhcyBTb3V0aEFmcmljYUNvcnBvcmF0ZVRheFJlc3VsdCxcbn0gZnJvbSAnLi9jb3Jwb3JhdGUvc291dGgtYWZyaWNhL2RvbWFpbi90eXBlcyc7XG5cbi8vIEF1c3RyYWxpYVxuZXhwb3J0IHsgQXVzdHJhbGlhSW5jb21lVGF4U2VydmljZUltcGwgYXMgQXVzdHJhbGlhSW5jb21lVGF4U2VydmljZSB9IGZyb20gJy4vaW5jb21lLXRheC9hdXN0cmFsaWEvQXVzdHJhbGlhSW5jb21lVGF4U2VydmljZUltcGwnO1xuZXhwb3J0IHtcbiAgICBDb21wdXRlZEluY29tZVRheFZhbHVlcyBhcyBBdXN0cmFsaWFDb21wdXRlZEluY29tZVRheFZhbHVlcyxcbiAgICBJbmNvbWVUYXhSdWxlcyBhcyBBdXN0cmFsaWFJbmNvbWVUYXhSdWxlcyxcbn0gZnJvbSAnLi9pbmNvbWUtdGF4L2F1c3RyYWxpYS9kb21haW4vdHlwZXMnO1xuXG5leHBvcnQgeyBBdXN0cmFsaWFNb3J0Z2FnZVNlcnZpY2VJbXBsIGFzIEF1c3RyYWxpYU1vcnRnYWdlU2VydmljZSB9IGZyb20gJy4vbW9ydGdhZ2UvYXVzdHJhbGlhL0F1c3RyYWxpYU1vcnRnYWdlU2VydmljZUltcGwnO1xuZXhwb3J0IHtcbiAgICBNb3J0Z2FnZVJ1bGVzIGFzIEF1c3RyYWxpYU1vcnRnYWdlUnVsZXMsXG4gICAgTW9ydGdhZ2VJbnB1dCBhcyBBdXN0cmFsaWFNb3J0Z2FnZUlucHV0LFxuICAgIE1vcnRnYWdlT3V0cHV0IGFzIEF1c3RyYWxpYU1vcnRnYWdlT3V0cHV0LFxufSBmcm9tICcuL21vcnRnYWdlL2F1c3RyYWxpYS9kb21haW4vdHlwZXMnO1xuXG5leHBvcnQgeyBBdXN0cmFsaWFDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCBhcyBBdXN0cmFsaWFDb3Jwb3JhdGVUYXhTZXJ2aWNlIH0gZnJvbSAnLi9jb3Jwb3JhdGUvYXVzdHJhbGlhL0F1c3RyYWxpYUNvcnBvcmF0ZVRheFNlcnZpY2VJbXBsJztcbmV4cG9ydCB7XG4gICAgSW5wdXQgYXMgQXVzdHJhbGlhQ29ycG9yYXRlVGF4SW5wdXQsXG4gICAgUnVsZXMgYXMgQXVzdHJhbGlhQ29ycG9yYXRlVGF4UnVsZXMsXG4gICAgUmVzdWx0IGFzIEF1c3RyYWxpYUNvcnBvcmF0ZVRheFJlc3VsdCxcbn0gZnJvbSAnLi9jb3Jwb3JhdGUvYXVzdHJhbGlhL2RvbWFpbi90eXBlcyc7XG5cbmV4cG9ydCB7IEluY29tZVRheENhbGN1bGF0b3JTY2hlbWEgfSBmcm9tICcuL2luY29tZS10YXgvZG9tYWluL3R5cGVzJztcbmV4cG9ydCAqIGZyb20gJy4vaW5jb21lLXRheC9kb21haW4vdHlwZXMnOyJdfQ==
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSwrQ0FBdUQ7QUFBOUMsdUdBQUEsY0FBYyxPQUFBO0FBRXZCLFNBQVM7QUFDVCw2RkFBc0g7QUFBN0csb0lBQUEsMEJBQTBCLE9BQTBCO0FBQzdELHlGQUFpSDtBQUF4RyxrSUFBQSx5QkFBeUIsT0FBeUI7QUFXM0Qsa0dBQThIO0FBQXJILDBJQUFBLDZCQUE2QixPQUE2QjtBQU9uRSxTQUFTO0FBQ1QsNkZBQXNIO0FBQTdHLG9JQUFBLDBCQUEwQixPQUEwQjtBQUM3RCx5RkFBaUg7QUFBeEcsa0lBQUEseUJBQXlCLE9BQXlCO0FBVzNELGtHQUE4SDtBQUFySCwwSUFBQSw2QkFBNkIsT0FBNkI7QUFPbkUsZUFBZTtBQUNmLDZHQUEySTtBQUFsSSw4SUFBQSwrQkFBK0IsT0FBK0I7QUFDdkUseUdBQXNJO0FBQTdILDRJQUFBLDhCQUE4QixPQUE4QjtBQVdyRSxrSEFBbUo7QUFBMUksb0pBQUEsa0NBQWtDLE9BQWtDO0FBTzdFLFlBQVk7QUFDWixzR0FBa0k7QUFBekgsMElBQUEsNkJBQTZCLE9BQTZCO0FBTW5FLGtHQUE2SDtBQUFwSCx3SUFBQSw0QkFBNEIsT0FBNEI7QUFPakUsMkdBQTBJO0FBQWpJLGdKQUFBLGdDQUFnQyxPQUFnQztBQU96RSxpQkFBaUI7QUFDakIsaUZBQXNHO0FBQTdGLDRIQUFBLHNCQUFzQixPQUFzQjtBQU1yRCxzRkFBOEc7QUFBckcsa0lBQUEseUJBQXlCLE9BQXlCO0FBTzNELDZFQUFpRztBQUF4RiwwSEFBQSxxQkFBcUIsT0FBcUI7QUFRbkQsNERBQTBDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgQ2FsY3VsYXRvclR5cGUgfSBmcm9tICcuL3NoYXJlZC9kb21haW4vdHlwZXMnO1xuXG4vLyBDYW5hZGFcbmV4cG9ydCB7IENhbmFkYUluY29tZVRheFNlcnZpY2VJbXBsIGFzIENhbmFkYUluY29tZVRheFNlcnZpY2UgfSBmcm9tICcuL2luY29tZS10YXgvY2FuYWRhL0NhbmFkYUluY29tZVRheFNlcnZpY2VJbXBsJztcbmV4cG9ydCB7IENhbmFkYU1vcnRnYWdlU2VydmljZUltcGwgYXMgQ2FuYWRhTW9ydGdhZ2VTZXJ2aWNlIH0gZnJvbSAnLi9tb3J0Z2FnZS9jYW5hZGEvQ2FuYWRhTW9ydGdhZ2VTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIENvbXB1dGVkSW5jb21lVGF4VmFsdWVzIGFzIENhbmFkYUNvbXB1dGVkSW5jb21lVGF4VmFsdWVzLFxuICAgIEluY29tZVRheFJ1bGVzIGFzIENhbmFkYUluY29tZVRheFJ1bGVzLFxufSBmcm9tICcuL2luY29tZS10YXgvY2FuYWRhL2RvbWFpbi90eXBlcyc7XG5leHBvcnQge1xuICAgIE1vcnRnYWdlUnVsZXMgYXMgQ2FuYWRhTW9ydGdhZ2VSdWxlcyxcbiAgICBNb3J0Z2FnZUNhbGN1bGF0aW9uSW5wdXQgYXMgQ2FuYWRhTW9ydGdhZ2VDYWxjdWxhdGlvbklucHV0LFxuICAgIE1vcnRnYWdlQ2FsY3VsYXRpb25SZXN1bHQgYXMgQ2FuYWRhTW9ydGdhZ2VDYWxjdWxhdGlvblJlc3VsdCxcbn0gZnJvbSAnLi9tb3J0Z2FnZS9jYW5hZGEvZG9tYWluL3R5cGVzJztcblxuZXhwb3J0IHsgQ2FuYWRhQ29ycG9yYXRlVGF4U2VydmljZUltcGwgYXMgQ2FuYWRhQ29ycG9yYXRlVGF4U2VydmljZSB9IGZyb20gJy4vY29ycG9yYXRlL2NhbmFkYS9DYW5hZGFDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIElucHV0IGFzIENhbmFkYUNvcnBvcmF0ZVRheElucHV0LFxuICAgIFJ1bGVzIGFzIENhbmFkYUNvcnBvcmF0ZVRheFJ1bGVzLFxuICAgIFJlc3VsdCBhcyBDYW5hZGFDb3Jwb3JhdGVUYXhSZXN1bHQsXG59IGZyb20gJy4vY29ycG9yYXRlL2NhbmFkYS9kb21haW4vdHlwZXMnO1xuXG4vLyBGcmFuY2VcbmV4cG9ydCB7IEZyYW5jZUluY29tZVRheFNlcnZpY2VJbXBsIGFzIEZyYW5jZUluY29tZVRheFNlcnZpY2UgfSBmcm9tICcuL2luY29tZS10YXgvZnJhbmNlL0ZyYW5jZUluY29tZVRheFNlcnZpY2VJbXBsJztcbmV4cG9ydCB7IEZyYW5jZU1vcnRnYWdlU2VydmljZUltcGwgYXMgRnJhbmNlTW9ydGdhZ2VTZXJ2aWNlIH0gZnJvbSAnLi9tb3J0Z2FnZS9mcmFuY2UvRnJhbmNlTW9ydGdhZ2VTZXJ2aWNlSW1wbCc7XG5leHBvcnQgeyBcbiAgICBDb21wdXRlZEluY29tZVRheFZhbHVlcyBhcyBGcmFuY2VDb21wdXRlZEluY29tZVRheFZhbHVlcyxcbiAgICBJbmNvbWVUYXhSdWxlcyBhcyBGcmFuY2VJbmNvbWVUYXhSdWxlcyxcbn0gZnJvbSAnLi9pbmNvbWUtdGF4L2ZyYW5jZS9kb21haW4vdHlwZXMnO1xuZXhwb3J0IHtcbiAgICBNb3J0Z2FnZVJ1bGVzIGFzIEZyYW5jZU1vcnRnYWdlUnVsZXMsXG4gICAgTW9ydGdhZ2VJbnB1dCBhcyBGcmFuY2VNb3J0Z2FnZUlucHV0LFxuICAgIE1vcnRnYWdlT3V0cHV0IGFzIEZyYW5jZU1vcnRnYWdlQ2FsY3VsYXRpb25SZXN1bHQsXG59IGZyb20gJy4vbW9ydGdhZ2UvZnJhbmNlL2RvbWFpbi90eXBlcyc7XG5cbmV4cG9ydCB7IEZyYW5jZUNvcnBvcmF0ZVRheFNlcnZpY2VJbXBsIGFzIEZyYW5jZUNvcnBvcmF0ZVRheFNlcnZpY2UgfSBmcm9tICcuL2NvcnBvcmF0ZS9mcmFuY2UvRnJhbmNlQ29ycG9yYXRlVGF4U2VydmljZUltcGwnO1xuZXhwb3J0IHtcbiAgICBJbnB1dCBhcyBGcmFuY2VDb3Jwb3JhdGVUYXhJbnB1dCxcbiAgICBSdWxlcyBhcyBGcmFuY2VDb3Jwb3JhdGVUYXhSdWxlcyxcbiAgICBSZXN1bHQgYXMgRnJhbmNlQ29ycG9yYXRlVGF4UmVzdWx0LFxufSBmcm9tICcuL2NvcnBvcmF0ZS9mcmFuY2UvZG9tYWluL3R5cGVzJztcblxuLy8gU291dGggQWZyaWNhXG5leHBvcnQgeyBTb3V0aEFmcmljYUluY29tZVRheFNlcnZpY2VJbXBsIGFzIFNvdXRoQWZyaWNhSW5jb21lVGF4U2VydmljZSB9IGZyb20gJy4vaW5jb21lLXRheC9zb3V0aC1hZnJpY2EvU291dGhBZnJpY2FJbmNvbWVUYXhTZXJ2aWNlSW1wbCc7XG5leHBvcnQgeyBTb3V0aEFmcmljYU1vcnRnYWdlU2VydmljZUltcGwgYXMgU291dGhBZnJpY2FNb3J0Z2FnZVNlcnZpY2UgfSBmcm9tICcuL21vcnRnYWdlL3NvdXRoLWFmcmljYS9Tb3V0aEFmcmljYU1vcnRnYWdlU2VydmljZUltcGwnO1xuZXhwb3J0IHtcbiAgICBDb21wdXRlZEluY29tZVRheFZhbHVlcyBhcyBTb3V0aEFmcmljYUNvbXB1dGVkSW5jb21lVGF4VmFsdWVzLFxuICAgIEluY29tZVRheFJ1bGVzIGFzIFNvdXRoQWZyaWNhSW5jb21lVGF4UnVsZXMsXG59IGZyb20gJy4vaW5jb21lLXRheC9zb3V0aC1hZnJpY2EvZG9tYWluL3R5cGVzJztcbmV4cG9ydCB7XG4gICAgTW9ydGdhZ2VSdWxlcyBhcyBTb3V0aEFmcmljYU1vcnRnYWdlUnVsZXMsXG4gICAgTW9ydGdhZ2VJbnB1dCBhcyBTb3V0aEFmcmljYU1vcnRnYWdlSW5wdXQsXG4gICAgTW9ydGdhZ2VPdXRwdXQgYXMgU291dGhBZnJpY2FNb3J0Z2FnZU91dHB1dCxcbn0gZnJvbSAnLi9tb3J0Z2FnZS9zb3V0aC1hZnJpY2EvZG9tYWluL3R5cGVzJztcblxuZXhwb3J0IHsgU291dGhBZnJpY2FDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCBhcyBTb3V0aEFmcmljYUNvcnBvcmF0ZVRheFNlcnZpY2UgfSBmcm9tICcuL2NvcnBvcmF0ZS9zb3V0aC1hZnJpY2EvU291dGhBZnJpY2FDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIFJ1bGVzIGFzIFNvdXRoQWZyaWNhQ29ycG9yYXRlVGF4UnVsZXMsXG4gICAgSW5wdXQgYXMgU291dGhBZnJpY2FDb3Jwb3JhdGVUYXhJbnB1dCxcbiAgICBSZXN1bHQgYXMgU291dGhBZnJpY2FDb3Jwb3JhdGVUYXhSZXN1bHQsXG59IGZyb20gJy4vY29ycG9yYXRlL3NvdXRoLWFmcmljYS9kb21haW4vdHlwZXMnO1xuXG4vLyBBdXN0cmFsaWFcbmV4cG9ydCB7IEF1c3RyYWxpYUluY29tZVRheFNlcnZpY2VJbXBsIGFzIEF1c3RyYWxpYUluY29tZVRheFNlcnZpY2UgfSBmcm9tICcuL2luY29tZS10YXgvYXVzdHJhbGlhL0F1c3RyYWxpYUluY29tZVRheFNlcnZpY2VJbXBsJztcbmV4cG9ydCB7XG4gICAgQ29tcHV0ZWRJbmNvbWVUYXhWYWx1ZXMgYXMgQXVzdHJhbGlhQ29tcHV0ZWRJbmNvbWVUYXhWYWx1ZXMsXG4gICAgSW5jb21lVGF4UnVsZXMgYXMgQXVzdHJhbGlhSW5jb21lVGF4UnVsZXMsXG59IGZyb20gJy4vaW5jb21lLXRheC9hdXN0cmFsaWEvZG9tYWluL3R5cGVzJztcblxuZXhwb3J0IHsgQXVzdHJhbGlhTW9ydGdhZ2VTZXJ2aWNlSW1wbCBhcyBBdXN0cmFsaWFNb3J0Z2FnZVNlcnZpY2UgfSBmcm9tICcuL21vcnRnYWdlL2F1c3RyYWxpYS9BdXN0cmFsaWFNb3J0Z2FnZVNlcnZpY2VJbXBsJztcbmV4cG9ydCB7XG4gICAgTW9ydGdhZ2VSdWxlcyBhcyBBdXN0cmFsaWFNb3J0Z2FnZVJ1bGVzLFxuICAgIE1vcnRnYWdlSW5wdXQgYXMgQXVzdHJhbGlhTW9ydGdhZ2VJbnB1dCxcbiAgICBNb3J0Z2FnZU91dHB1dCBhcyBBdXN0cmFsaWFNb3J0Z2FnZU91dHB1dCxcbn0gZnJvbSAnLi9tb3J0Z2FnZS9hdXN0cmFsaWEvZG9tYWluL3R5cGVzJztcblxuZXhwb3J0IHsgQXVzdHJhbGlhQ29ycG9yYXRlVGF4U2VydmljZUltcGwgYXMgQXVzdHJhbGlhQ29ycG9yYXRlVGF4U2VydmljZSB9IGZyb20gJy4vY29ycG9yYXRlL2F1c3RyYWxpYS9BdXN0cmFsaWFDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIElucHV0IGFzIEF1c3RyYWxpYUNvcnBvcmF0ZVRheElucHV0LFxuICAgIFJ1bGVzIGFzIEF1c3RyYWxpYUNvcnBvcmF0ZVRheFJ1bGVzLFxuICAgIFJlc3VsdCBhcyBBdXN0cmFsaWFDb3Jwb3JhdGVUYXhSZXN1bHQsXG59IGZyb20gJy4vY29ycG9yYXRlL2F1c3RyYWxpYS9kb21haW4vdHlwZXMnO1xuXG4vLyBVbml0ZWQgS2luZ2RvbVxuZXhwb3J0IHsgVUtJbmNvbWVUYXhTZXJ2aWNlSW1wbCBhcyBVS0luY29tZVRheFNlcnZpY2UgfSBmcm9tICcuL2luY29tZS10YXgvdWsvVUtJbmNvbWVUYXhTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIENvbXB1dGVkSW5jb21lVGF4VmFsdWVzIGFzIFVLQ29tcHV0ZWRJbmNvbWVUYXhWYWx1ZXMsXG4gICAgSW5jb21lVGF4UnVsZXMgYXMgVUtJbmNvbWVUYXhSdWxlcyxcbn0gZnJvbSAnLi9pbmNvbWUtdGF4L3VrL2RvbWFpbi90eXBlcyc7XG5cbmV4cG9ydCB7IFVLQ29ycG9yYXRlVGF4U2VydmljZUltcGwgYXMgVUtDb3Jwb3JhdGVUYXhTZXJ2aWNlIH0gZnJvbSAnLi9jb3Jwb3JhdGUvdWsvVUtDb3Jwb3JhdGVUYXhTZXJ2aWNlSW1wbCc7XG5leHBvcnQge1xuICAgIElucHV0IGFzIFVLQ29ycG9yYXRlVGF4SW5wdXQsXG4gICAgUnVsZXMgYXMgVUtDb3Jwb3JhdGVUYXhSdWxlcyxcbiAgICBSZXN1bHQgYXMgVUtDb3Jwb3JhdGVUYXhSZXN1bHQsXG59IGZyb20gJy4vY29ycG9yYXRlL3VrL2RvbWFpbi90eXBlcyc7XG5cbmV4cG9ydCB7IFVLTW9ydGdhZ2VTZXJ2aWNlSW1wbCBhcyBVS01vcnRnYWdlU2VydmljZSB9IGZyb20gJy4vbW9ydGdhZ2UvdWsvVUtNb3J0Z2FnZVNlcnZpY2VJbXBsJztcbmV4cG9ydCB7XG4gICAgTW9ydGdhZ2VSdWxlcyBhcyBVS01vcnRnYWdlUnVsZXMsXG4gICAgTW9ydGdhZ2VJbnB1dCBhcyBVS01vcnRnYWdlSW5wdXQsXG4gICAgTW9ydGdhZ2VPdXRwdXQgYXMgVUtNb3J0Z2FnZU91dHB1dCxcbn0gZnJvbSAnLi9tb3J0Z2FnZS91ay9kb21haW4vdHlwZXMnO1xuXG5leHBvcnQgeyBJbmNvbWVUYXhDYWxjdWxhdG9yU2NoZW1hIH0gZnJvbSAnLi9pbmNvbWUtdGF4L2RvbWFpbi90eXBlcyc7XG5leHBvcnQgKiBmcm9tICcuL2luY29tZS10YXgvZG9tYWluL3R5cGVzJzsiXX0=
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiVUtNb3J0Z2FnZVNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbW9ydGdhZ2UvdWsvVUtNb3J0Z2FnZVNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE1vcnRnYWdlSW5wdXQsIE1vcnRnYWdlT3V0cHV0LCBNb3J0Z2FnZVJ1bGVzIH0gZnJvbSBcIi4vZG9tYWluL3R5cGVzXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgVUtNb3J0Z2FnZVNlcnZpY2Uge1xuICAgIGNhbGN1bGF0ZShpbnB1dDogTW9ydGdhZ2VJbnB1dCwgcnVsZXM6IE1vcnRnYWdlUnVsZXMpOiBNb3J0Z2FnZU91dHB1dDtcbn1cbiJdfQ==
@@ -0,0 +1,99 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.UKMortgageServiceImpl = void 0;
4
+ class UKMortgageServiceImpl {
5
+ calculate(input, rules) {
6
+ const loanAmount = input.propertyPrice - input.downPayment;
7
+ if (loanAmount <= 0) {
8
+ throw new Error('Invalid loan amount: down payment must be less than property price');
9
+ }
10
+ const totalPayments = input.amortizationYears * 12;
11
+ const monthlyRate = (input.annualInterestRate / 100) / 12;
12
+ const monthlyPayment = this.calculatePayment(loanAmount, monthlyRate, totalPayments);
13
+ const totalPaid = monthlyPayment * totalPayments;
14
+ const totalInterestPaid = totalPaid - loanAmount;
15
+ const stampDuty = this.calculateStampDuty(input, rules);
16
+ const amortizationSchedule = this.calculateAmortizationSchedule(loanAmount, monthlyRate, monthlyPayment, totalPayments, input.amortizationYears);
17
+ return {
18
+ loanAmount,
19
+ totalMortgage: loanAmount,
20
+ monthlyPayment,
21
+ totalInterestPaid,
22
+ totalPaid,
23
+ stampDuty,
24
+ amortizationSchedule,
25
+ otherFees: {
26
+ notaryFees: {
27
+ value: stampDuty,
28
+ label: 'STAMP_DUTY',
29
+ },
30
+ bankFees: {
31
+ value: 0,
32
+ label: 'BANK_FEES',
33
+ },
34
+ monthlyInsuranceFees: {
35
+ value: 0,
36
+ label: 'MONTHLY_INSURANCE_FEES',
37
+ },
38
+ },
39
+ };
40
+ }
41
+ calculateStampDuty(input, rules) {
42
+ const { propertyPrice, isFirstTimeBuyer } = input;
43
+ const { firstTimeBuyer, standardBrackets } = rules.stampDuty;
44
+ const useFirstTimeBuyerRules = isFirstTimeBuyer && propertyPrice <= firstTimeBuyer.maxEligiblePropertyPrice;
45
+ const brackets = useFirstTimeBuyerRules ? firstTimeBuyer.brackets : standardBrackets;
46
+ return this.applyStampDutyBrackets(propertyPrice, brackets);
47
+ }
48
+ applyStampDutyBrackets(propertyPrice, brackets) {
49
+ let duty = 0;
50
+ let previousLimit = 0;
51
+ for (const bracket of brackets) {
52
+ if (bracket.upTo !== undefined && propertyPrice > previousLimit) {
53
+ const taxable = Math.min(propertyPrice, bracket.upTo) - previousLimit;
54
+ duty += taxable * bracket.rate;
55
+ previousLimit = bracket.upTo;
56
+ }
57
+ if (bracket.above !== undefined && propertyPrice > bracket.above) {
58
+ duty += (propertyPrice - bracket.above) * bracket.rate;
59
+ break;
60
+ }
61
+ }
62
+ return duty;
63
+ }
64
+ calculateAmortizationSchedule(principal, monthlyRate, monthlyPayment, totalPayments, amortizationYears) {
65
+ const schedule = [];
66
+ let balance = principal;
67
+ for (let year = 1; year <= amortizationYears; year++) {
68
+ let yearlyPrincipal = 0;
69
+ let yearlyInterest = 0;
70
+ const paymentsInYear = Math.min(12, totalPayments - (year - 1) * 12);
71
+ for (let payment = 1; payment <= paymentsInYear; payment++) {
72
+ const interestPayment = balance * monthlyRate;
73
+ const principalPayment = monthlyPayment - interestPayment;
74
+ yearlyInterest += interestPayment;
75
+ yearlyPrincipal += principalPayment;
76
+ balance -= principalPayment;
77
+ }
78
+ schedule.push({
79
+ year,
80
+ principal: yearlyPrincipal,
81
+ interest: yearlyInterest,
82
+ balance: Math.max(0, balance),
83
+ });
84
+ if (balance <= 0)
85
+ break;
86
+ }
87
+ return schedule;
88
+ }
89
+ calculatePayment(principal, rate, periods) {
90
+ if (rate === 0) {
91
+ return principal / periods;
92
+ }
93
+ return principal *
94
+ (rate * Math.pow(1 + rate, periods)) /
95
+ (Math.pow(1 + rate, periods) - 1);
96
+ }
97
+ }
98
+ exports.UKMortgageServiceImpl = UKMortgageServiceImpl;
99
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UKMortgageServiceImpl.js","sourceRoot":"","sources":["../../../src/mortgage/uk/UKMortgageServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { UKMortgageService } from \"./UKMortgageService\";\nimport {\n    AmortizationScheduleItem,\n    MortgageInput,\n    MortgageOutput,\n    MortgageRules,\n    StampDutyBracket,\n} from \"./domain/types\";\n\nexport class UKMortgageServiceImpl implements UKMortgageService {\n\n    public calculate(input: MortgageInput, rules: MortgageRules): MortgageOutput {\n        const loanAmount = input.propertyPrice - input.downPayment;\n\n        if (loanAmount <= 0) {\n            throw new Error('Invalid loan amount: down payment must be less than property price');\n        }\n\n        const totalPayments = input.amortizationYears * 12;\n        const monthlyRate = (input.annualInterestRate / 100) / 12;\n\n        const monthlyPayment = this.calculatePayment(loanAmount, monthlyRate, totalPayments);\n        const totalPaid = monthlyPayment * totalPayments;\n        const totalInterestPaid = totalPaid - loanAmount;\n\n        const stampDuty = this.calculateStampDuty(input, rules);\n\n        const amortizationSchedule = this.calculateAmortizationSchedule(\n            loanAmount,\n            monthlyRate,\n            monthlyPayment,\n            totalPayments,\n            input.amortizationYears,\n        );\n\n        return {\n            loanAmount,\n            totalMortgage: loanAmount,\n            monthlyPayment,\n            totalInterestPaid,\n            totalPaid,\n            stampDuty,\n            amortizationSchedule,\n            otherFees: {\n                notaryFees: {\n                    value: stampDuty,\n                    label: 'STAMP_DUTY',\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\n    private calculateStampDuty(input: MortgageInput, rules: MortgageRules): number {\n        const { propertyPrice, isFirstTimeBuyer } = input;\n        const { firstTimeBuyer, standardBrackets } = rules.stampDuty;\n\n        const useFirstTimeBuyerRules =\n            isFirstTimeBuyer && propertyPrice <= firstTimeBuyer.maxEligiblePropertyPrice;\n\n        const brackets = useFirstTimeBuyerRules ? firstTimeBuyer.brackets : standardBrackets;\n\n        return this.applyStampDutyBrackets(propertyPrice, brackets);\n    }\n\n    private applyStampDutyBrackets(propertyPrice: number, brackets: StampDutyBracket[]): number {\n        let duty = 0;\n        let previousLimit = 0;\n\n        for (const bracket of brackets) {\n            if (bracket.upTo !== undefined && propertyPrice > previousLimit) {\n                const taxable = Math.min(propertyPrice, bracket.upTo) - previousLimit;\n                duty += taxable * bracket.rate;\n                previousLimit = bracket.upTo;\n            }\n\n            if (bracket.above !== undefined && propertyPrice > bracket.above) {\n                duty += (propertyPrice - bracket.above) * bracket.rate;\n                break;\n            }\n        }\n\n        return duty;\n    }\n\n    private calculateAmortizationSchedule(\n        principal: number,\n        monthlyRate: number,\n        monthlyPayment: number,\n        totalPayments: number,\n        amortizationYears: 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(12, totalPayments - (year - 1) * 12);\n\n            for (let payment = 1; payment <= paymentsInYear; payment++) {\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 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"]}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbW9ydGdhZ2UvdWsvZG9tYWluL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBPdGhlckZlZXMgfSBmcm9tIFwiLi4vLi4vZG9tYWluL3R5cGVzXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VSdWxlcyB7XG4gICAgbG9hbkNvbnN0cmFpbnRzOiBMb2FuQ29uc3RyYWludHM7XG4gICAgaW50ZXJlc3Q6IEludGVyZXN0UnVsZXM7XG4gICAgc3RhbXBEdXR5OiBTdGFtcER1dHlSdWxlcztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBMb2FuQ29uc3RyYWludHMge1xuICAgIG1heEx0dlBlcmNlbnQ6IG51bWJlcjtcbiAgICBtYXhBbW9ydGl6YXRpb25ZZWFyczogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEludGVyZXN0UnVsZXMge1xuICAgIGNvbXBvdW5kaW5nOiAnTU9OVEhMWSc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3RhbXBEdXR5QnJhY2tldCB7XG4gICAgdXBUbz86IG51bWJlcjtcbiAgICBhYm92ZT86IG51bWJlcjtcbiAgICByYXRlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3RhbXBEdXR5UnVsZXMge1xuICAgIHN0YW5kYXJkQnJhY2tldHM6IFN0YW1wRHV0eUJyYWNrZXRbXTtcbiAgICBmaXJzdFRpbWVCdXllcjogRmlyc3RUaW1lQnV5ZXJTdGFtcER1dHk7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmlyc3RUaW1lQnV5ZXJTdGFtcER1dHkge1xuICAgIGJyYWNrZXRzOiBTdGFtcER1dHlCcmFja2V0W107XG4gICAgbWF4RWxpZ2libGVQcm9wZXJ0eVByaWNlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW1vcnRpemF0aW9uU2NoZWR1bGVJdGVtIHtcbiAgICB5ZWFyOiBudW1iZXI7XG4gICAgcHJpbmNpcGFsOiBudW1iZXI7XG4gICAgaW50ZXJlc3Q6IG51bWJlcjtcbiAgICBiYWxhbmNlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9ydGdhZ2VJbnB1dCB7XG4gICAgcHJvcGVydHlQcmljZTogbnVtYmVyO1xuICAgIGRvd25QYXltZW50OiBudW1iZXI7XG4gICAgYW5udWFsSW50ZXJlc3RSYXRlOiBudW1iZXI7XG4gICAgYW1vcnRpemF0aW9uWWVhcnM6IG51bWJlcjtcbiAgICBpc0ZpcnN0VGltZUJ1eWVyOiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vcnRnYWdlT3V0cHV0IHtcbiAgICBsb2FuQW1vdW50OiBudW1iZXI7XG4gICAgdG90YWxNb3J0Z2FnZTogbnVtYmVyO1xuICAgIG1vbnRobHlQYXltZW50OiBudW1iZXI7XG4gICAgdG90YWxJbnRlcmVzdFBhaWQ6IG51bWJlcjtcbiAgICB0b3RhbFBhaWQ6IG51bWJlcjtcbiAgICBzdGFtcER1dHk6IG51bWJlcjtcbiAgICBhbW9ydGl6YXRpb25TY2hlZHVsZTogQW1vcnRpemF0aW9uU2NoZWR1bGVJdGVtW107XG4gICAgb3RoZXJGZWVzOiBPdGhlckZlZXM7XG59XG4iXX0=
@@ -0,0 +1,4 @@
1
+ import { Result } from "./domain/types";
2
+ export interface UKCorporateTaxService {
3
+ calculate(): Result;
4
+ }
@@ -0,0 +1,10 @@
1
+ import { UKCorporateTaxService } from "./UKCorporateTaxService";
2
+ import { Input, Result, Rules } from "./domain/types";
3
+ export declare class UKCorporateTaxServiceImpl implements UKCorporateTaxService {
4
+ private _input;
5
+ private _rules;
6
+ constructor(input: Input, rules: Rules);
7
+ calculate(): Result;
8
+ private applyMarginalRelief;
9
+ private buildMarginalReliefBreakdowns;
10
+ }
@@ -0,0 +1,32 @@
1
+ import { Breakdown } from "../../domain/types";
2
+ export interface FlatTaxRule {
3
+ type: 'flat';
4
+ rate: number;
5
+ conditions?: {
6
+ maxIncome?: number;
7
+ };
8
+ }
9
+ export interface MarginalReliefRule {
10
+ type: 'marginal_relief';
11
+ mainRate: number;
12
+ smallProfitsRate: number;
13
+ upperLimit: number;
14
+ lowerLimit: number;
15
+ standardFraction: number;
16
+ }
17
+ export type TaxRule = FlatTaxRule | MarginalReliefRule;
18
+ export interface Rules {
19
+ regimes: {
20
+ smallProfits: FlatTaxRule;
21
+ main: FlatTaxRule;
22
+ marginalRelief: MarginalReliefRule;
23
+ };
24
+ }
25
+ export interface Input {
26
+ taxableIncome: number;
27
+ }
28
+ export interface Result {
29
+ corporateTax: number;
30
+ effectiveTaxRate: number;
31
+ breakdowns: Breakdown[];
32
+ }
@@ -0,0 +1,4 @@
1
+ import { ComputedIncomeTaxValues } from "./domain/types";
2
+ export interface UKIncomeTaxService {
3
+ calculateNetIncome(): ComputedIncomeTaxValues;
4
+ }
@@ -0,0 +1,12 @@
1
+ import { UKIncomeTaxService } from "./UKIncomeTaxService";
2
+ import { ComputedIncomeTaxValues, IncomeTaxRules } from "./domain/types";
3
+ export declare class UKIncomeTaxServiceImpl implements UKIncomeTaxService {
4
+ private _income;
5
+ private _rules;
6
+ constructor(income: number, rules: IncomeTaxRules);
7
+ calculateNetIncome(): ComputedIncomeTaxValues;
8
+ private computePersonalAllowance;
9
+ private computeTaxBrackets;
10
+ private computeNationalInsurance;
11
+ private round;
12
+ }
@@ -0,0 +1,27 @@
1
+ import { BracketAllocation, TaxBracket } from "../../domain/types";
2
+ export interface IncomeTaxRules {
3
+ taxBrackets: TaxBracket[];
4
+ personalAllowance: PersonalAllowanceRules;
5
+ nationalInsurance: NationalInsuranceRules;
6
+ }
7
+ export interface PersonalAllowanceRules {
8
+ amount: number;
9
+ taperThreshold: number;
10
+ taperRate: number;
11
+ }
12
+ export interface NationalInsuranceRules {
13
+ primaryThreshold: number;
14
+ upperEarningsLimit: number;
15
+ mainRate: number;
16
+ upperRate: number;
17
+ }
18
+ export interface ComputedIncomeTaxValues {
19
+ grossIncome: number;
20
+ incomeTax: number;
21
+ nationalInsurance: number;
22
+ personalAllowance: number;
23
+ totalDeductions: number;
24
+ netIncome: number;
25
+ effectiveTaxRate: number;
26
+ taxBracketBreakdown: BracketAllocation[];
27
+ }
@@ -23,5 +23,11 @@ export { AustraliaMortgageServiceImpl as AustraliaMortgageService } from './mort
23
23
  export { MortgageRules as AustraliaMortgageRules, MortgageInput as AustraliaMortgageInput, MortgageOutput as AustraliaMortgageOutput, } from './mortgage/australia/domain/types';
24
24
  export { AustraliaCorporateTaxServiceImpl as AustraliaCorporateTaxService } from './corporate/australia/AustraliaCorporateTaxServiceImpl';
25
25
  export { Input as AustraliaCorporateTaxInput, Rules as AustraliaCorporateTaxRules, Result as AustraliaCorporateTaxResult, } from './corporate/australia/domain/types';
26
+ export { UKIncomeTaxServiceImpl as UKIncomeTaxService } from './income-tax/uk/UKIncomeTaxServiceImpl';
27
+ export { ComputedIncomeTaxValues as UKComputedIncomeTaxValues, IncomeTaxRules as UKIncomeTaxRules, } from './income-tax/uk/domain/types';
28
+ export { UKCorporateTaxServiceImpl as UKCorporateTaxService } from './corporate/uk/UKCorporateTaxServiceImpl';
29
+ export { Input as UKCorporateTaxInput, Rules as UKCorporateTaxRules, Result as UKCorporateTaxResult, } from './corporate/uk/domain/types';
30
+ export { UKMortgageServiceImpl as UKMortgageService } from './mortgage/uk/UKMortgageServiceImpl';
31
+ export { MortgageRules as UKMortgageRules, MortgageInput as UKMortgageInput, MortgageOutput as UKMortgageOutput, } from './mortgage/uk/domain/types';
26
32
  export { IncomeTaxCalculatorSchema } from './income-tax/domain/types';
27
33
  export * from './income-tax/domain/types';
@@ -0,0 +1,4 @@
1
+ import { MortgageInput, MortgageOutput, MortgageRules } from "./domain/types";
2
+ export interface UKMortgageService {
3
+ calculate(input: MortgageInput, rules: MortgageRules): MortgageOutput;
4
+ }
@@ -0,0 +1,9 @@
1
+ import { UKMortgageService } from "./UKMortgageService";
2
+ import { MortgageInput, MortgageOutput, MortgageRules } from "./domain/types";
3
+ export declare class UKMortgageServiceImpl implements UKMortgageService {
4
+ calculate(input: MortgageInput, rules: MortgageRules): MortgageOutput;
5
+ private calculateStampDuty;
6
+ private applyStampDutyBrackets;
7
+ private calculateAmortizationSchedule;
8
+ private calculatePayment;
9
+ }
@@ -0,0 +1,49 @@
1
+ import { OtherFees } from "../../domain/types";
2
+ export interface MortgageRules {
3
+ loanConstraints: LoanConstraints;
4
+ interest: InterestRules;
5
+ stampDuty: StampDutyRules;
6
+ }
7
+ export interface LoanConstraints {
8
+ maxLtvPercent: number;
9
+ maxAmortizationYears: number;
10
+ }
11
+ export interface InterestRules {
12
+ compounding: 'MONTHLY';
13
+ }
14
+ export interface StampDutyBracket {
15
+ upTo?: number;
16
+ above?: number;
17
+ rate: number;
18
+ }
19
+ export interface StampDutyRules {
20
+ standardBrackets: StampDutyBracket[];
21
+ firstTimeBuyer: FirstTimeBuyerStampDuty;
22
+ }
23
+ export interface FirstTimeBuyerStampDuty {
24
+ brackets: StampDutyBracket[];
25
+ maxEligiblePropertyPrice: number;
26
+ }
27
+ export interface AmortizationScheduleItem {
28
+ year: number;
29
+ principal: number;
30
+ interest: number;
31
+ balance: number;
32
+ }
33
+ export interface MortgageInput {
34
+ propertyPrice: number;
35
+ downPayment: number;
36
+ annualInterestRate: number;
37
+ amortizationYears: number;
38
+ isFirstTimeBuyer: boolean;
39
+ }
40
+ export interface MortgageOutput {
41
+ loanAmount: number;
42
+ totalMortgage: number;
43
+ monthlyPayment: number;
44
+ totalInterestPaid: number;
45
+ totalPaid: number;
46
+ stampDuty: number;
47
+ amortizationSchedule: AmortizationScheduleItem[];
48
+ otherFees: OtherFees;
49
+ }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@novha/calc-engines",
3
- "version": "2.0.2",
3
+ "version": "3.0.0",
4
4
  "main": "dist/index.js",
5
5
  "types": "dist/types/index.d.ts",
6
6
  "scripts": {
@@ -0,0 +1,5 @@
1
+ import { Result } from "./domain/types";
2
+
3
+ export interface UKCorporateTaxService {
4
+ calculate(): Result;
5
+ }