forked from tchernicum/bcapps
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbc-cc-interest.pl
executable file
·252 lines (192 loc) · 6.62 KB
/
bc-cc-interest.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
#!/usr/bin/perl
# Usage: $0 --amount=a --months=m --rate=r --fee=f --minmonth=mm, where:
# a = amount borrowed in dollars
# m = months for which the interest rate 'r' is valid
# r = the interest rate (usually a special rate for cash advances)
# f = the cash advance fee as percentage of money borrowed
# mm = the minimum monthly payment, as a %age of the total balance
# if you borrow from your credit card and they have a cash advance fee
# and/or minimum monthly payment, this is the APR you'd need to break
# even, ie, the "true interest rate"
# This program is from my older work, and thus worse than usual
# Will eventually run at apr.barrycarter.info
# TODO: this doesn't work for certain values (I make certain
# assumptions that don't always hold). Example:
# http://apr.barrycarter.info/?amount=100&fee=.25&minmonth=.1&months=3&rate=0&submit=CALCULATE
# this might just be because I limit interest to 100%?
# TODO: the amount borrowed field is really superfluous
push(@INC,"/usr/local/lib");
require "bclib.pl";
# webifying
print "Content-type: text/html\n\n";
# cleansing
$ENV{QUERY_STRING}=~s/[^a-z0-9=&\.\-]//isg;
# read GET vars into globopts hash (not great, but...)
defaults($ENV{QUERY_STRING});
debug("ALPHA",%globopts);
# actual defaults (must come after string above to avoid overwriting)
defaults("amount=10000&fee=.03&minmonth=.02&months=16&rate=.04&profit=.05");
debug("BETA",%globopts);
# header for HTML page
print << "MARK";
<title>Hidden Interest</title>
When you borrow money from a credit card, what rate of return must you earn to breakeven? Find out below.<p>
Note: This site is experimental. Do not rely on this information.<p>
<form action="$ENV{REQUEST_URI}" method="GET">
<table border>
MARK
;
# create HTML table w/ headers and data (including text entry fields)
# world's most hideous way to order hash keys
%table = (
"0Amount borrowed:" => "amount",
"1Cash advance fee (.03=3%)" => "fee",
"2Minimum monthly payment (.02 = 2%)" => "minmonth",
"3Length of loan (months)" => "months",
"4Loan APR (.04 = 4%)" => "rate"
);
for $i (sort keys %table) {
# for safety, strip value of dangerous symbols
$table{$i}=~s/[^a-z0-9\.\-]//i;
# get rid of sort key for printing
$printi= $i;
$printi=~s/^\d+//isg;
debug("I: $i, tablei: $table{$i}, $globopts{$table{$i}}");
print << "MARK";
<tr><th>$printi</th><td>
<input type='text' name='$table{$i}' value='$globopts{$table{$i}}' size='10'>
</td></tr>
MARK
;
}
print << "MARK";
<tr><th colspan=2>
<input type="submit" name="submit" value="CALCULATE">
</th></tr>
</table></form>
MARK
;
$advance = $globopts{fee}*$globopts{amount};
$params = "amount=$globopts{amount}&months=$globopts{months}&rate=$globopts{rate}&fee=$advance&minmonth=$globopts{minmonth}";
# checks (not sure these are all necessary)
if ($globopts{amount}<1 || $globopts{amount}>1e+6) {
sorry("Amount borrowed must be between \$1 and \$999,999");
}
if ($globopts{months}<1 || $globopts{months}>480) {
sorry("Length of loan must be between 1 and 480 months");
}
if ($globopts{rate}<-.99 || $globopts{rate}>.99) {
sorry("Loan APR must be between -.99 and +.99");
}
if ($globopts{minmonth}<0 || $globopts{minmonth}>.99) {
sorry("Minimum monthly payment must be between 0 and +.99");
}
# trivial function for findroot (find interest rate so sinking table
# ends with 0 [ie, no profit/gain])
sub f {return sinking_table("$params&gainonly=1&investrate=$_[0]");}
$res = findroot(\&f,0,1,1e-6,100);
# this is redundant, since we calculate it as part of findroot
@res = sinking_table("$params&investrate=$res");
# print the table and true interest rate
printf("<b>True interest rate</b>: %0.3f%%<p>\n", $res*100);
print << "MARK";
<table border>
<tr>
<th>Month</th>
<th>Owed at start of month</th>
<th>Interest on money owed</th>
<th>Minimum payment to credit card</th>
<th>Owed at end of month</th>
<th> </th>
<th>Money in hand at start of month</th>
<th>Return on money in hand</th>
<th>Minimum payment to credit card</th>
<th>Money in hand at end of month</th>
<th> </th>
<th>Equity (money in hand minus money owed)</th>
</tr>
MARK
;
for $i (0..$#res) {
debug("HASH START: $i");
%hash = %{$res[$i]};
print "<tr>\n";
# month not printed in monetary format, so just do it here
print "<td>$i</td>\n";
for $j ("oldowed", "owedint", "-payment", "owed", "",
"oldhave", "haveprof", "-payment", "have", "", "gain") {
# print blanks
if ($j eq "") {print "<td> </td>\n"; next;}
# the amount to print
$key = $j;
if ($key=~s/^\-//) {$pr=-$hash{$key};} else {$pr=$hash{$key};}
# print it
print "<td>".money_print($pr)."</td>\n";
}
print "</tr>\n";
}
print "</table>\n";
=item sinking_table()
Returns sinking table (as list of hashes) for loan w/ following parameters:
amount - the amount in dollars (eg, 10000)
rate - the annual interest rate (eg, .05)
fee - upfront fee in dollars (eg, 300)
months - length of loan in months (eg, 60)
minmonth - percentage minimum payment per month (eg, .02)
investrate - the rate at which the borrowed money is invested (eg, .06)
gainonly - if set, return only final gain, nothing else
=cut
sub sinking_table {
my(%VALUE) = parse_form($_[0]);
# debug("alpha",unfold(%VALUE),"/alpha");
my($oldowed, $oldhave, $owedint, $payment, $haveprof, $gain);
my(@res);
# start out owing amount + fee
my($owed) = $VALUE{amount} + $VALUE{fee};
# start out having amount
my($have) = $VALUE{amount};
# iterate thru months
for $i (0..$VALUE{months}) {
# before values
$oldowed = $owed;
$oldhave = $have;
# interest on money you owe
$owedint = $owed*((1.+$VALUE{rate})**(1./12.)-1.);
# payment you make monthly
# HACK: TODO: if minmonth>1 assume it's a dollar amount
if (abs($VALUE{minmonth}) < 1) {
$payment = $owed*$VALUE{minmonth};
} else {
$payment = $VALUE{minmonth};
}
# profit from investment
$haveprof = $have*((1.+$VALUE{investrate})**(1./12.)-1.);
# subtract from owed/have
$owed += $owedint-$payment;
$have += $haveprof-$payment;
$gain = $have-$owed;
# really odd way of doing this
my(%hash) = parse_form("oldowed=$oldowed&oldhave=$oldhave&owedint=$owedint&payment=$payment&haveprof=$haveprof&owed=$owed&have=$have&gain=$gain");
push(@res,\%hash);
}
if ($VALUE{gainonly}) {return $gain;}
return @res;
}
# how to print monetary amounts for this prog
sub money_print {
my($val) = @_;
$val=sprintf("%0.2f",$val);
if ($val<0) {
$val=~s/^\-//;
$val="<font color='#ff0000'>-\$$val</font>";
} else {
$val="\$$val";
}
return $val;
}
# how to display errors
sub sorry {
my($err) = @_;
print "<b><font color='#ff0000'>Error: $err</font></b>\n";
exit(-1);
}