457 lines
14 KiB
C#
457 lines
14 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
|
|
// Filename: DateGenerator.cs
|
|
// Author:Sean Kessler
|
|
|
|
namespace MarketData.Utils
|
|
{
|
|
/// <summary>DateGenerator - Generate Historical Dates</summary>
|
|
/// "BusinessDay" is analagous to "MarketDay". To qualify as a business day the date must not be a weekend or a market holiday
|
|
[Serializable]
|
|
public class DateGenerator
|
|
{
|
|
private static readonly TimeSpan TIMESPAN_INCREASE_ONE_DAY=new TimeSpan(1,0,0,0);
|
|
private static readonly TimeSpan TIMESPAN_DECREASE_ONE_DAY=new TimeSpan(-1,0,0,0);
|
|
|
|
public DateGenerator()
|
|
{
|
|
}
|
|
/// <summary>GetCurrMonthStart - Finds the first business day of the previous month</summary>
|
|
/// <returns>DateTime</returns>
|
|
public DateTime GetPrevMonthStart(DateTime dateTime)
|
|
{
|
|
DateTime startDate = new DateTime(dateTime.Year, dateTime.Month, 1);
|
|
startDate = startDate.AddMonths(-1);
|
|
startDate = GetNextBusinessDay(startDate);
|
|
return startDate;
|
|
}
|
|
/// <summary>GetCurrMonthStart - Finds the first business day of the current month</summary>
|
|
/// <returns>DateTime</returns>
|
|
public DateTime GetCurrMonthStart(DateTime dateTime)
|
|
{
|
|
DateTime startDate = new DateTime(dateTime.Year, dateTime.Month, 1);
|
|
startDate = GetNextBusinessDay(startDate);
|
|
return startDate;
|
|
}
|
|
public static int GetPrevMonth(int month)
|
|
{
|
|
if(1==month)return 12;
|
|
return month-1;
|
|
}
|
|
/// <summary>FindPrevBusinessDay - Finds previous business day</summary>
|
|
/// <returns>DateTime</returns>
|
|
public DateTime FindPrevBusinessDay(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(1,0,0,0);
|
|
asOf=asOf.Subtract(oneDay);
|
|
while(IsWeekend(asOf)||IsHoliday(asOf))asOf=asOf.Subtract(oneDay);
|
|
return asOf.Date;
|
|
}
|
|
/// <summary>FindForwardBusinessDay - Finds following business day going daysForward days</summary>
|
|
/// <returns>DateTime</returns>
|
|
public DateTime FindForwardBusinessDay(DateTime asOf,int daysForward)
|
|
{
|
|
for (int index = 0; index < daysForward; index++)
|
|
{
|
|
asOf = FindNextBusinessDay(asOf);
|
|
}
|
|
return asOf.Date;
|
|
}
|
|
/// <summary>FindPastBusinessDay - Finds previous business day going daysPast days</summary>
|
|
/// <returns>DateTime</returns>
|
|
public DateTime FindPastBusinessDay(DateTime asOf,int daysPast)
|
|
{
|
|
for (int index = 0; index < daysPast; index++)
|
|
{
|
|
asOf = FindPrevBusinessDay(asOf);
|
|
}
|
|
return asOf.Date;
|
|
}
|
|
|
|
/// <summary>FindNextBusinessDay - Finds following nth business day</summary>
|
|
/// <returns>DateTime</returns>
|
|
public DateTime FindNextBusinessDay(DateTime asOf,int days)
|
|
{
|
|
DateTime nextDate=asOf;
|
|
for(int index=0;index<days;index++)
|
|
{
|
|
nextDate=FindNextBusinessDay(nextDate);
|
|
}
|
|
return nextDate;
|
|
}
|
|
|
|
/// <summary>FindNextBusinessDay - Finds following business day</summary>
|
|
/// <returns>DateTime</returns>
|
|
public DateTime FindNextBusinessDay(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(1,0,0,0);
|
|
asOf=asOf.Add(oneDay);
|
|
while(IsWeekend(asOf)||IsHoliday(asOf))asOf=asOf.Add(oneDay);
|
|
return asOf.Date;
|
|
}
|
|
|
|
/// <summary>GetPrevBusinessDay - Gets previous business day</summary>
|
|
/// <note>If the given date is a business day then this method will return the given date</note>
|
|
/// <returns>None</returns>
|
|
public DateTime GetPrevBusinessDay(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(1,0,0,0);
|
|
while(IsWeekend(asOf)||IsHoliday(asOf))asOf=asOf.Subtract(oneDay);
|
|
return asOf.Date;
|
|
}
|
|
|
|
/// <summary>GetNextBusinessDay - Gets next business day</summary>
|
|
/// <note>If the given date is a business day then this method will return the given date</note>
|
|
/// <returns>None</returns>
|
|
public DateTime GetNextBusinessDay(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(1,0,0,0);
|
|
while(IsWeekend(asOf)||IsHoliday(asOf))asOf=asOf.Add(oneDay);
|
|
return asOf.Date;
|
|
}
|
|
|
|
/// <summary>GetNextDay - Gets next business day</summary>
|
|
/// <note>If the given date is a business day then this method will return the given date</note>
|
|
/// <returns>None</returns>
|
|
public DateTime GetNextDay(DateTime asOf)
|
|
{
|
|
return (asOf.Add(Utility.OneDay)).Date;
|
|
}
|
|
|
|
/// <summary>GetPrevFriday - Gets date of prior friday</summary>
|
|
/// <note>Get the date of the previous friday - if previous Friday is holiday will seek previous business day</note>
|
|
/// <returns>None</returns>
|
|
public DateTime GetPrevFriday(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(1,0,0,0);
|
|
List<DateTime> historicalDates=null;
|
|
int daysToFetch=7;
|
|
|
|
if(DayOfWeek.Friday==asOf.DayOfWeek)
|
|
{
|
|
asOf=asOf.Subtract(oneDay);
|
|
daysToFetch--;
|
|
}
|
|
historicalDates=GenerateHistoricalDates(asOf,daysToFetch);
|
|
for(int index=0;index<historicalDates.Count;index++)
|
|
{
|
|
DateTime date=(DateTime)historicalDates[index];
|
|
if(DayOfWeek.Friday==date.DayOfWeek)return GetPrevBusinessDay(date);
|
|
}
|
|
throw new Exception("The week no longer contains Friday?");
|
|
}
|
|
|
|
public DateTime GetPrevQuarterStartDate(DateTime asOf)
|
|
{
|
|
int month=asOf.Month;
|
|
|
|
if(asOf.Month>=1 && asOf.Month<=3)month=1;
|
|
else if(asOf.Month>=4 && asOf.Month<=6)month=4;
|
|
else if(asOf.Month>=7 && asOf.Month<=9)month=7;
|
|
else month=10;
|
|
DateTime quarterStartDate=new DateTime(asOf.Year,month,1);
|
|
quarterStartDate=GetNextBusinessDay(quarterStartDate);
|
|
return quarterStartDate;
|
|
}
|
|
|
|
public DateTime GetNextQuarterStartDate(DateTime asOf)
|
|
{
|
|
DateTime prevQuarterStartDate=GetPrevQuarterStartDate(asOf);
|
|
int month=prevQuarterStartDate.Month;
|
|
int year=prevQuarterStartDate.Year;
|
|
if(prevQuarterStartDate.Month.Equals(10)){year++;}
|
|
month+=3;
|
|
if(month>12)month=1;
|
|
DateTime nextQuarterStartDate=new DateTime(year,month,1);
|
|
nextQuarterStartDate=GetNextBusinessDay(nextQuarterStartDate);
|
|
return nextQuarterStartDate;
|
|
}
|
|
|
|
public DateTime GetCurrentMonthEnd(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(-1,0,0,0);
|
|
DateTime date=new DateTime(asOf.Year,asOf.Month,asOf.Day);
|
|
date=new DateTime(date.Year,date.Month,DateTime.DaysInMonth(asOf.Year,asOf.Month));
|
|
while(IsWeekend(date)||IsHoliday(date))date=date.Add(oneDay);
|
|
return date;
|
|
}
|
|
|
|
public DateTime GetNextMonthEnd(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(-1,0,0,0);
|
|
DateTime date=new DateTime(asOf.Year,asOf.Month,asOf.Day);
|
|
date=date.AddMonths(1);
|
|
date=new DateTime(date.Year,date.Month,DateTime.DaysInMonth(date.Year,date.Month));
|
|
while(IsWeekend(date)||IsHoliday(date))date=date.Add(oneDay);
|
|
return date;
|
|
}
|
|
|
|
public DateTime GetPrevMonthEnd(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(-1,0,0,0);
|
|
DateTime date=new DateTime(asOf.Year,asOf.Month,asOf.Day);
|
|
date=date.AddMonths(-1);
|
|
date=new DateTime(date.Year,date.Month,DateTime.DaysInMonth(date.Year,date.Month));
|
|
while(IsWeekend(date)||IsHoliday(date))date=date.Add(oneDay);
|
|
return date;
|
|
}
|
|
|
|
public DateTime GetPrevMonthEnd(DateTime asOf,int count)
|
|
{
|
|
for(int index=0;index<count;index++)
|
|
{
|
|
asOf=GetPrevMonthEnd(asOf);
|
|
}
|
|
return asOf;
|
|
}
|
|
|
|
public DateTime EnsureWeekday(DateTime asOf)
|
|
{
|
|
TimeSpan oneDay=new TimeSpan(1,0,0,0);
|
|
if(!IsWeekend(asOf))return asOf;
|
|
while(IsWeekend(asOf=asOf.Add(oneDay)));
|
|
return asOf;
|
|
}
|
|
|
|
public DateTime GenerateHistoricalDate(DateTime startDate,int dayCount)
|
|
{
|
|
DateTime histDate;
|
|
int datedDates=0;
|
|
TimeSpan singleDay;
|
|
|
|
startDate = startDate.Date;
|
|
histDate=startDate;
|
|
if(dayCount<0)singleDay=TIMESPAN_DECREASE_ONE_DAY;
|
|
else singleDay=TIMESPAN_INCREASE_ONE_DAY;
|
|
dayCount=dayCount<0?-dayCount:dayCount;
|
|
while(datedDates<dayCount)
|
|
{
|
|
if(!(DayOfWeek.Sunday==histDate.DayOfWeek||DayOfWeek.Saturday==histDate.DayOfWeek))
|
|
{
|
|
if(!IsHoliday(histDate.Date))datedDates++;
|
|
}
|
|
histDate=histDate.Subtract(singleDay);
|
|
}
|
|
return histDate.Add(singleDay);
|
|
}
|
|
|
|
// Does not account for weekends etc.,
|
|
public DateTime GenerateFutureDate(DateTime startDate,int dayCount)
|
|
{
|
|
DateTime futureDate;
|
|
int datedDates=0;
|
|
TimeSpan singleDay;
|
|
|
|
startDate = startDate.Date;
|
|
futureDate=startDate;
|
|
singleDay=new TimeSpan(1,0,0,0);
|
|
while(datedDates<dayCount)
|
|
{
|
|
futureDate=futureDate.Add(singleDay);
|
|
datedDates++;
|
|
}
|
|
return futureDate.Add(singleDay);
|
|
}
|
|
|
|
// Accounts for weekends
|
|
public DateTime GenerateFutureBusinessDate(DateTime startDate,int dayCount)
|
|
{
|
|
DateTime futureDate;
|
|
int datedDates=0;
|
|
TimeSpan singleDay;
|
|
|
|
startDate = startDate.Date;
|
|
futureDate=startDate;
|
|
singleDay=new TimeSpan(1,0,0,0);
|
|
while(datedDates<dayCount)
|
|
{
|
|
if(IsMarketOpen(futureDate))datedDates++;
|
|
futureDate=futureDate.Add(singleDay);
|
|
}
|
|
return futureDate.Add(singleDay);
|
|
}
|
|
|
|
public int DaysBetweenActual(DateTime historicalDate)
|
|
{
|
|
DateTime today = DateTime.Now;
|
|
TimeSpan timeSpan = historicalDate.Date - today;
|
|
return (int)timeSpan.TotalDays;
|
|
}
|
|
|
|
public int DaysBetweenActual(DateTime historicalDate,DateTime startingDate)
|
|
{
|
|
TimeSpan timeSpan = historicalDate.Date-startingDate;
|
|
int totalDays = (int)timeSpan.TotalDays;
|
|
if (historicalDate < startingDate) totalDays = (Math.Abs(totalDays) * -1);
|
|
else totalDays = Math.Abs(totalDays);
|
|
return totalDays;
|
|
}
|
|
|
|
public DateTime DaysAddActual(DateTime date, int daysActual)
|
|
{
|
|
TimeSpan days = new TimeSpan(daysActual, 0, 0, 0);
|
|
return date + days;
|
|
}
|
|
|
|
public int DaysBetween(DateTime startDate, DateTime endDate)
|
|
{
|
|
if (startDate > endDate)
|
|
{
|
|
TimeSpan timeSpan = startDate.Date - endDate.Date;
|
|
return (int)timeSpan.TotalDays;
|
|
}
|
|
else
|
|
{
|
|
TimeSpan timeSpan = endDate.Date - startDate.Date;
|
|
return (int)timeSpan.TotalDays;
|
|
}
|
|
}
|
|
|
|
public int MonthsBetween(DateTime startDate,DateTime endDate)
|
|
{
|
|
return DaysBetween(startDate,endDate)/30;
|
|
}
|
|
|
|
public static List<int> GenerateHistoricalYear(int startYear,int years)
|
|
{
|
|
List<int> yearsList = new List<int>();
|
|
for (int index = 0; index < years; index++)
|
|
{
|
|
yearsList.Add(startYear);
|
|
startYear--;
|
|
}
|
|
return yearsList;
|
|
}
|
|
|
|
public List<DateTime> GenerateHistoricalDates(DateTime startDate, int dayCount)
|
|
{
|
|
List<DateTime> histDates=new List<DateTime>();
|
|
DateTime histDate;
|
|
TimeSpan singleDay;
|
|
|
|
startDate = startDate.Date;
|
|
histDate=startDate;
|
|
if(dayCount<0)singleDay=new TimeSpan(-1,0,0,0);
|
|
else singleDay=new TimeSpan(1,0,0,0);
|
|
dayCount=dayCount<0?-dayCount:dayCount;
|
|
while(histDates.Count<dayCount)
|
|
{
|
|
if(IsMarketOpen(histDate))histDates.Add(histDate);
|
|
histDate=histDate.Subtract(singleDay);
|
|
}
|
|
return histDates;
|
|
}
|
|
/// <summary>
|
|
/// Generates dates into the future respecting weekens and market holidays
|
|
/// </summary>
|
|
/// <param name="startDate"></param>
|
|
/// <param name="dayCount"></param>
|
|
/// <returns></returns>
|
|
public List<DateTime> GenerateFutureDates(DateTime startDate, int dayCount)
|
|
{
|
|
List<DateTime> futureDates=new List<DateTime>();
|
|
DateTime futureDate;
|
|
TimeSpan singleDay;
|
|
|
|
startDate = startDate.Date;
|
|
futureDate=startDate;
|
|
if(dayCount<0)singleDay=new TimeSpan(1,0,0,0);
|
|
else singleDay=new TimeSpan(1,0,0,0);
|
|
dayCount=dayCount<0?-dayCount:dayCount;
|
|
while(futureDates.Count<dayCount)
|
|
{
|
|
if(IsMarketOpen(futureDate))futureDates.Add(futureDate);
|
|
futureDate=futureDate.Add(singleDay);
|
|
}
|
|
return futureDates;
|
|
}
|
|
|
|
public int TradingDaysBetween(DateTime startDate,DateTime endDate)
|
|
{
|
|
List<DateTime> historicalDates=GenerateHistoricalDates(startDate,endDate);
|
|
return historicalDates.Count;
|
|
}
|
|
|
|
// The function will figure out which date is the most recent and which one is the historical date.
|
|
// Generally, make startDate the most recent and endDate the historical date.
|
|
// The returned series will contain an ascending date series with the earliest (most distant) date in the lowest numbered index.
|
|
public List<DateTime> GenerateHistoricalDates(DateTime startDate,DateTime endDate)
|
|
{
|
|
if (Utility.Epoch.Equals(startDate)||Utility.Epoch.Equals(endDate)) return null;
|
|
startDate = startDate.Date;
|
|
endDate = endDate.Date;
|
|
List<DateTime> histDates = new List<DateTime>();
|
|
DateTime histDate;
|
|
TimeSpan singleDay;
|
|
|
|
bool reverse = false;
|
|
if (endDate > startDate)
|
|
{
|
|
reverse = true;
|
|
DateTime swap = endDate;
|
|
endDate = startDate;
|
|
startDate = swap;
|
|
}
|
|
histDate=startDate;
|
|
singleDay=new TimeSpan(1,0,0,0);
|
|
while(histDate>=endDate)
|
|
{
|
|
if(IsMarketOpen(histDate))histDates.Add(histDate);
|
|
histDate=histDate.Subtract(singleDay);
|
|
}
|
|
if (reverse) histDates.Reverse();
|
|
return histDates;
|
|
}
|
|
|
|
// The function will figure out which date is the most recent and which one is the historical date.
|
|
// Generally, make startDate the most recent and endDate the historical date.
|
|
public List<DateTime> GenerateHistoricalDatesActual(DateTime startDate, DateTime endDate)
|
|
{
|
|
if (Utility.Epoch.Equals(startDate) || Utility.Epoch.Equals(endDate)) return null;
|
|
startDate = startDate.Date;
|
|
endDate = endDate.Date;
|
|
List<DateTime> histDates = new List<DateTime>();
|
|
DateTime histDate;
|
|
TimeSpan singleDay;
|
|
|
|
bool reverse = false;
|
|
if (endDate > startDate)
|
|
{
|
|
reverse = true;
|
|
DateTime swap = endDate;
|
|
endDate = startDate;
|
|
startDate = swap;
|
|
}
|
|
histDate = startDate;
|
|
singleDay = new TimeSpan(1, 0, 0, 0);
|
|
while (histDate >= endDate)
|
|
{
|
|
histDates.Add(histDate);
|
|
histDate = histDate.Subtract(singleDay);
|
|
}
|
|
if (reverse) histDates.Reverse();
|
|
return histDates;
|
|
}
|
|
|
|
public bool IsMarketOpen(DateTime dateTime)
|
|
{
|
|
if(IsWeekend(dateTime)||IsHoliday(dateTime))return false;
|
|
return true;
|
|
}
|
|
|
|
public bool IsWeekend(DateTime dateTime)
|
|
{
|
|
if(DayOfWeek.Sunday==dateTime.DayOfWeek || DayOfWeek.Saturday==dateTime.DayOfWeek)return true;
|
|
return false;
|
|
}
|
|
|
|
// This uses a singleton cache of holidays.
|
|
public bool IsHoliday(DateTime dateTime)
|
|
{
|
|
return HolidayCache.GetInstance().IsHoliday(dateTime);
|
|
}
|
|
}
|
|
}
|
|
|