Let me state up front that I will only be discussing loans with no compounding of interest, more precisely, where interest is not earned on interest. In practice, this is how most of the consumer loans in the U.S. work. With this assumption, the interest (I) due after one payment interval (T) is the product of the interest rate (R) and the loan balance (A):
I = A * R
For example, if one has a loan of $10,000, and an annual interest rate of 12%, the interest due at the end of the first year is (10000) * (0.12) = $1200.
The interest rate is usually stated as the nominal annual interest rate, since it refers to the interest rate for a period of one year. For those few loans with annual payments (T = one year), this is all one needs. If the payment interval is something other than a year, then one needs to derive the effective interest rate (i) from the nominal rate.
For mortgage and many car loans, the payment interval is one month. Calculating i is then a simple matter of dividing I by 12. To continue our example, if the payment interval were the month instead of the year, the interest due at the end of the first month would be (10000) * (0.12 / 12) = $100.
For credit cards and money market deposits, however, the unit of time is the day, and i is often (but not always!) calculated by dividing I by 365. Why 365? Because there are supposed to be 365 days in a year. And yes, even in 2008, if you check the fine print in your credit card statement, you will probably see that the daily rate is 1/365th of the annual rate.
Oops. We no longer have the payment period equal to the interest interval. If (D) is the number of days between payments, our formula becomes
I = A * D * (R / 365)
Or, if we go 30 days in our example, I = (10000) * 30 * (0.12 / 365) = $98.63
Of course, if we went 31 days instead, I = (10000) * 31 * (0.12 / 365) = $101.92
Double oops. It looks as if the payee in the above exaple wants to make a monthly payment. Wouldn't it be nice if we could get the interest to be equal to our monthly example of $100? We could if we let the days in a month be 30 (always and everywhere) and changed the 365 to 360.
We finally can describe the wherefore and why of the days360 function. Days360 returns the number of days between two dates, and it tries to do it as if each year had exactly 360 days in it, and each month had exactly 30 days. Of course, eight months do not have 30 days, and there are several reasonable ways to treat those months. The European method (which is invoked by setting the third parameter to 'true') changes the last day of the month to the 30th of the month. The U.S. method (which is invoked by setting the third parameter to 'false') is more complicated, and the Excel help screen should be read if you really want all the details.
Anyway, here is a PL/SQL version of 'days360':
create or replace function days360(
p_start_date date,
p_end_date date,
p_rule_type char default 'F'
)
RETURN number
IS
v_mm1 pls_integer;
v_dd1 pls_integer;
v_yyyy1 pls_integer;
v_mm2 pls_integer;
v_dd2 pls_integer;
v_yyyy2 pls_integer;
BEGIN
v_yyyy1 := to_number(to_char(p_start_date,'yyyy'));
v_mm1 := to_number(to_char(p_start_date,'mm'));
v_dd1 := to_number(to_char(p_start_date,'dd'));
v_yyyy2 := to_number(to_char(p_end_date,'yyyy'));
v_mm2 := to_number(to_char(p_end_date,'mm'));
v_dd2 := to_number(to_char(p_end_date,'dd'));
IF p_rule_type = 'F' THEN
IF v_dd1 = 31 THEN v_dd1 := 30; END IF;
IF v_mm1 = 2 AND v_dd1 = to_number(to_char(last_day(p_start_date),'dd'))
THEN v_dd1 := 30; END IF;
IF v_dd2 = 31
THEN IF v_dd1 < 30
THEN v_dd2 := 1;
v_mm2 := v_mm2 + 1;
IF v_mm2 = 13 THEN v_mm2 := 1;
v_yyyy2 := v_yyyy2 +1;
END IF;
ELSE v_dd2 := 30;
END IF;
END IF;
IF v_mm2 = 2 AND v_dd2 = to_number(to_char(last_day(p_end_date),'dd'))
THEN v_dd2 := 30;
IF (v_dd1 < 30)
THEN v_dd2 := 1;
v_mm2 := 3;
END IF;
END IF;
IF v_mm2 IN (4, 6, 9, 11) AND v_dd2 = 30
AND v_dd1 < 30
THEN v_dd2 := 1;
v_mm2 := v_mm2 + 1;
END IF;
ELSIF p_rule_type = 'T' THEN
IF v_dd1 = 31 THEN v_dd1 := 30; END IF;
IF v_dd1 = 31 THEN v_dd1 := 30; END IF;
IF v_mm1 = 2 AND v_dd1 = to_number(to_char(last_day(p_start_date),'dd'))
THEN v_dd1 := 30; END IF;
IF v_dd2 = 31 THEN v_dd2 := 30; END IF;
IF v_mm2 = 2 AND v_dd2 = to_number(to_char(last_day(p_end_date),'dd'))
THEN v_dd2 := 30; END IF;
ELSE RAISE_APPLICATION_ERROR('-20002','3VL Not Allowed Here');
END IF;
RETURN (v_yyyy2 - v_yyyy1) * 360
+ (v_mm2 - v_mm1) * 30
+ (v_dd2 - v_dd1);
END;
/
In testing this, I set my NLS_DATE_FORMAT to YYYYMMDD, and I will use that format in this discussion.
Note that days360('20060730', '20070730','T') = days360('20060730', '20070730','T') = 360, as one would expect for a 360 day year. And the same holds true if you change either of the dates to the 31st of the month.
[And the reason I am not using any dates in 2008 is that I want to avoid the complication of leap year]
Now, suppose someone borrows some money on 20060731, and pays the loan back with payments on the last day of each month. In other words, the payment intervals are
Start Date Pay Date
---------- ---------
20060731 20060831
20060831 20060930
20060930 20061031
20061031 20061130
20061130 20061231
20061231 20070131
20070131 20070228
20070228 20070331
20070331 20070430
20070430 20070531
20070531 20070630
20070630 20070731
If you do the days360 calculation on those payment intervals, you will find that each interval has 30 days, and the 12 intervals in the year add up to 360 days. It seems that the days360 function is doing its job.
Finally, let's assume that someone else takes out an identical loan, but that this person makes some payments early. No payments are later than the last of the month, but some are made earlier. Let's see what happens:
Start Pay European U.S.
Date Date Days Days
------- --------- --------- -------
20060731 20060830 30 30
20060830 20060929 29 29
20060929 20061030 31 32
20061030 20061129 29 29
20061129 20061230 31 32
20061230 20070128 28 28
20070128 20070227 29 29
20070227 20070331 33 34
20070331 20070429 29 29
20070429 20070531 31 32
20070531 20070629 29 29
20070629 20070731 31 32
==== ===
Totals 360 365
That's right, you get dinged an extra 5 days interest from trying to pay your loan off early.
Now do you believe there are complications in day counting?
11 comments:
Hi Phil,
Well, there are a few problems in the code you posted:
>> THEN IF v_dd1 <>
>> THEN v_dd2 := 1;
>>IF v_dd1 = 31 THEN v_dd1 := 30; END IF;
>>IF v_dd1 = 31 THEN v_dd1 := 30; END IF;
Maybe there are more but this is what I spotted right away. Could you please post your final version?
Thanks.
Hans Mayr
Indeed, there were two problems: the omitted item Hans located, and some typos in the dates of my carefully crafted example.
Hi Phil,
Thanks for the corrections and thanks for the code itself :-)
Hans
Hi Phil,
today you saved me about 4 hours of work with your code.
I HAVE to thank you a lot!
Ciao from Italy,
--Fabio
I GET ERROR WITH :
28/01/2009 28/02/2009
This return 32
28/02/2009 28/03/2009
This return 28
It took quite a while (more research that I thought), but I am finally ready to respond to Rodrigo's comment. But, it makes more sense to make it a brand new post.
You can prove this
select sum(dias) from(
select
case when last_day(fecha)!=fecha then 1
else
case when extract(day from fecha)=30 then 1
when extract(day from fecha)=28 then 3
when extract(day from fecha)=29 then 2
else 0 end
end dias from (
SELECT to_date('2009-01-01','YYYY-MM-DD') + rownum fecha
FROM dual CONNECT BY LEVEL <= (to_date('2009-09-30','YYYY-MM-DD') - to_date('2009-01-01','YYYY-MM-DD') + 1)))
Oidorgonzalez, I'm going to have to look closer at your sample to understand exactly what is happening, but I am afraid that it isn't the final answer to this problem. Try changing the start and end dates to Jan. 28 and Feb. 27:
select sum(dias) from(
select
case when last_day(fecha)!=fecha then 1
else
case when extract(day from fecha)=30 then 1
when extract(day from fecha)=28 then 3
when extract(day from fecha)=29 then 2
else 0 end
end dias from (
SELECT to_date('2009-01-28','YYYY-MM-DD') + rownum fecha
FROM dual CONNECT BY LEVEL <= (to_date('2009-02-27','YYYY-MM-DD') - to_date('2009-01-28','YYYY-MM-DD') + 1)))
/
This query returns 32, but the correct answer is 29.
Hi Phil, you're right, i send the version that i use, this function gives 30 days for jan 28 feb 27, the days360 excel version deals 29 days, well in my program i need it so, you can dias360 - 1
CREATE FUNCTION DIAS360(ADESDE VARCHAR2, AHASTA VARCHAR2,AMETODO CHAR:='F') RETURN NUMBER
AS
VRESULTADO NUMBER;
VDESDE DATE:=TO_DATE(ADESDE,'DD/MM/YYYY');
VHASTA DATE:=TO_DATE(AHASTA,'DD/MM/YYYY');
VFACTORFEB28 NUMBER(1):=3;
VFACTORFEB29 NUMBER(1):=2;
BEGIN
IF VDESDE<>VHASTA THEN
IF AMETODO='V' THEN
VFACTORFEB28:=1;
VFACTORFEB29:=1;
END IF;
SELECT SUM(DIAS) INTO VRESULTADO FROM(
SELECT
CASE WHEN LAST_DAY(FECHA)!=FECHA THEN 1
ELSE
CASE WHEN EXTRACT(DAY FROM FECHA)=30 THEN 1
WHEN EXTRACT(DAY FROM FECHA)=28 THEN VFACTORFEB28
WHEN EXTRACT(DAY FROM FECHA)=29 THEN VFACTORFEB29
ELSE 0 END
END DIAS FROM (SELECT (VDESDE + LEVEL - 1) FECHA
FROM (SELECT VDESDE INICIO,VHASTA + 1 FIN
FROM DUAL)
CONNECT BY LEVEL <= (FIN -INICIO)));
ELSE VRESULTADO:=1;
END IF;
RETURN VRESULTADO;
END;
Well -- let's take a look at what is going on here. This bit of code:
SELECT to_date('StartDate','YYYY-MM-DD') + rownum fecha
FROM dual CONNECT BY LEVEL <= (to_date('EndDate','YYYY-MM-DD') - to_date('StartDate','YYYY-MM-DD') + 1)))
Does nothing more that produce a list of date between a suitably formatted StartDate and EndDate. Having this, one could do a count(*) against this as a subquery -- and count the number of calendar days between the two dates. On a 365 day calendar. But, we want a 360 day calendar.
The CASE statements transform those dates into a series of zeroes (for the 31st of any month), ones (for almost every other date) and a suitable correction for the last day in February. And when you sum up all the transformed numbers, one gets the number of days in a 30/360 calendar, providing the StartDate and EndDate are not the last day in February. Correcting for the last day in February makes the whole calculation quite unwieldy.
I am unlikely to ever use this trick (and it really is a nice trick)though, for the following reasons:
1) I would need evidence that generating that number series does not chew up computer resources.
2) The final code cannot be included in-line, but must be coded as a function. If it could be coded in-line in a SQL statement, it might be useful, but I don't think it is practical.
3) As a function, it is harder to read than the traditional formulae, which also happen to be what auditors are used to seeing.
This is, however, a trick I will remember.
This function gives better results:
FUNCTION days360 (p_start_date DATE, p_end_date DATE, p_method CHAR DEFAULT 'FALSE')
RETURN NUMBER DETERMINISTIC IS
v_yyyy1 PLS_INTEGER := EXTRACT (YEAR FROM p_start_date);
v_mm1 PLS_INTEGER := EXTRACT (MONTH FROM p_start_date);
v_dd1 PLS_INTEGER := EXTRACT (DAY FROM p_start_date);
v_yyyy2 PLS_INTEGER := EXTRACT (YEAR FROM p_end_date);
v_mm2 PLS_INTEGER := EXTRACT (MONTH FROM p_end_date);
v_dd2 PLS_INTEGER := EXTRACT (DAY FROM p_end_date);
BEGIN
IF p_method = 'FALSE' THEN
IF v_mm1 = 2
AND v_mm2 = 2
AND trunc (p_start_date) = last_day (p_start_date)
AND trunc (p_end_date) = last_day (p_end_date) THEN
v_dd2 := 30;
END IF;
IF v_dd1 = 31
OR ( v_mm1 = 2
AND trunc (p_start_date) = last_day (p_start_date)) THEN
v_dd1 := 30;
END IF;
IF v_dd1 = 30
AND v_dd2 = 31 THEN
v_dd2 := 30;
END IF;
ELSIF p_method = 'TRUE' THEN
IF v_dd1 = 31 THEN
v_dd1 := 30;
END IF;
IF v_dd2 = 31 THEN
v_dd2 := 30;
END IF;
END IF;
RETURN (v_yyyy2 - v_yyyy1) * 360 + (v_mm2 - v_mm1) * 30 + (v_dd2 - v_dd1);
END;
Post a Comment