]>
Commit | Line | Data |
---|---|---|
1 | "use strict"; | |
2 | ||
3 | function near(a, b, epsilon = 1e-6) { | |
4 | return Math.abs(a - b) < epsilon; | |
5 | } | |
6 | ||
7 | function less_than_or_near(a, b, epsilon = 1e-6) { | |
8 | return a < b || near(a, b, epsilon); | |
9 | } | |
10 | ||
11 | function parse_table(as_text) { | |
12 | function parse_line(line) { | |
13 | return line.trim().split(' ').filter(x => x !== '').map(x => parseFloat(x)); | |
14 | } | |
15 | return as_text.trim().split('\n').map(parse_line); | |
16 | } | |
17 | ||
18 | function tax_table_from_table(table) { | |
19 | return table.map(([start, rate], i, table) => | |
20 | [start, i < table.length - 1 ? table[i+1][0] : Infinity, rate / 100.0]); | |
21 | } | |
22 | ||
23 | function parse_tax_table(as_text) { | |
24 | return tax_table_from_table(parse_table(as_text)); | |
25 | } | |
26 | ||
27 | function sum(nums) { | |
28 | return nums.reduce((total, num) => total + num, 0); | |
29 | } | |
30 | ||
31 | function tax(table, income) { | |
32 | return sum(table.map(([start, end, rate]) => | |
33 | Math.max(0, Math.min(income, end) - start) * rate)); | |
34 | } | |
35 | ||
36 | function apply_deductible(table, deductible) { | |
37 | return table.map(([start, end, rate]) => [start + deductible, end + deductible, rate]); | |
38 | } | |
39 | ||
40 | function merge_tax_tables(t1, t2) { | |
41 | if (t1.length == 0) return t2; | |
42 | if (t2.length == 0) return t1; | |
43 | const [start1, end1, rate1] = t1[0]; | |
44 | const [start2, end2, rate2] = t2[0]; | |
45 | if (start1 == start2) { | |
46 | if (end1 == end2) { | |
47 | return [[start1, end1, rate1 + rate2]].concat(merge_tax_tables(t1.slice(1), t2.slice(1))); | |
48 | } | |
49 | if (end1 < end2) { | |
50 | return [[start1, end1, rate1 + rate2]].concat(merge_tax_tables(t1.slice(1), [[end1, end2, rate2]].concat(t2.slice(1)))); | |
51 | } | |
52 | return merge_tax_tables(t2, t1); | |
53 | } | |
54 | if (start1 < start2) { | |
55 | if (end1 <= start2) { | |
56 | return [t1[0]].concat(merge_tax_tables(t1.slice(1), t2)); | |
57 | } | |
58 | return [[start1, start2, rate1]].concat(merge_tax_tables([[start2, end1, rate1]].concat(t1.slice(1)), t2)); | |
59 | } | |
60 | return merge_tax_tables(t2, t1); | |
61 | } | |
62 | ||
63 | function invert(table) { | |
64 | // Here we solve | |
65 | // net = m * gross + b | |
66 | // for gross: | |
67 | // net - b = m * gross | |
68 | // (net - b) / m = gross | |
69 | ||
70 | const ms = table.map(([start, end, rate]) => 1 - rate); | |
71 | const full_brackets = [0].concat(table.map(([start, end, rate]) => (end - start) * rate)).slice(0, table.length); | |
72 | function sum_lower_brackets(remaining_brackets, acc = 0) { | |
73 | if (remaining_brackets.length == 0) return []; | |
74 | return [acc + remaining_brackets[0]].concat(sum_lower_brackets(remaining_brackets.slice(1), acc + remaining_brackets[0])); | |
75 | } | |
76 | const bs = sum_lower_brackets(full_brackets).map((lower_brackets, i) => { | |
77 | const [start, end, rate] = table[i]; | |
78 | // Finding b: | |
79 | // net = gross - lower_brackets - rate * (gross - start) | |
80 | // net = gross - lower_brackets - rate * gross + rate * start | |
81 | // net = gross - rate * gross - lower_brackets + rate * start | |
82 | // net = (1 - rate) * gross - lower_brackets + rate * start | |
83 | // net = m * gross - lower_brackets + rate * start | |
84 | // \_____________________________/ - here is b | |
85 | return rate * start - lower_brackets; | |
86 | }); | |
87 | const inverse_table = table.map(([start, end, rate], i) => { | |
88 | const m = ms[i]; | |
89 | const b = bs[i]; | |
90 | return [m * start + b, m * end + b, m, b]; | |
91 | }); | |
92 | return function(net) { | |
93 | for (const [start, end, m, b] of inverse_table) { | |
94 | if (less_than_or_near(start, net) && less_than_or_near(net, end)) { | |
95 | return (net - b) / m; | |
96 | } | |
97 | } | |
98 | return net; | |
99 | }; | |
100 | } |