winamp/Src/Plugins/Library/ml_wire/RFCDate.cpp
2024-09-24 14:54:57 +02:00

216 lines
5.3 KiB
C++

#include "main.h"
#include "RFCDate.h"
#include <strsafe.h>
void MakeDateString(__time64_t convertTime, wchar_t *date_str, size_t len)
{
SYSTEMTIME sysTime = {0};
tm *newtime = _localtime64(&convertTime);
sysTime.wYear = newtime->tm_year + 1900;
sysTime.wMonth = newtime->tm_mon + 1;
sysTime.wDayOfWeek = newtime->tm_wday;
sysTime.wDay = newtime->tm_mday;
sysTime.wHour = newtime->tm_hour;
sysTime.wMinute = newtime->tm_min;
sysTime.wSecond = newtime->tm_sec;
GetDateFormat(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &sysTime, NULL, date_str, (int)len);
}
void MakeRFCDateString(__time64_t convertTime, wchar_t *data_str, size_t len)
{
SYSTEMTIME sysTime = {0};
tm *newtime = _gmtime64(&convertTime);
sysTime.wYear = newtime->tm_year + 1900;
sysTime.wMonth = newtime->tm_mon + 1;
sysTime.wDayOfWeek = newtime->tm_wday;
sysTime.wDay = newtime->tm_mday;
sysTime.wHour = newtime->tm_hour;
sysTime.wMinute = newtime->tm_min;
sysTime.wSecond = newtime->tm_sec;
wchar_t rfcDate[64] = {0}, rfcTime[64] = {0};
GetDateFormat(MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT), 0, &sysTime, L"ddd',' d MMM yyyy ", rfcDate, 64);
GetTimeFormat(MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT), 0, &sysTime, L"HH':'mm':'ss 'GMT'", rfcTime, 64);
StringCchPrintf(data_str,len,L"%s%s",rfcDate,rfcTime);
}
static int getMonthOfYear(const wchar_t *str);
static int validateTime(struct tm *tmloc, int gmoffset);
static const wchar_t *getNextField(const wchar_t *pos);
static int getGMOffset(const wchar_t *str);
enum
{
DAY_OF_MONTH = 0,
MONTH_OF_YEAR,
YEAR,
TIME,
TIMEZONE,
DATE_END
};
__time64_t MakeRFCDate(const wchar_t *date)
{
__time64_t result = 0;
const wchar_t *pos = date;
tm tmloc = {0};
const wchar_t *strmin;
const wchar_t *strsec;
int gmoffset = 1;
int state = DAY_OF_MONTH;
tzset();
/* skip weekday if present */
while (pos && *pos && !iswdigit(*pos)) pos++;
do
{
switch (state)
{
case DAY_OF_MONTH:
tmloc.tm_mday = _wtoi(pos);
break;
case MONTH_OF_YEAR:
tmloc.tm_mon = getMonthOfYear(pos);
break;
case YEAR:
{
/* TODO: we're only accepting 4-digit dates...*/
const wchar_t *test = pos; int numDigits = 0;
while (iswdigit(*test) && *test) { test++; numDigits++; }
if (numDigits == 2) // let's hope we never have 2 digit years!
tmloc.tm_year = _wtoi(pos) + 100; // assume 2 digit years are 20xx
else
tmloc.tm_year = _wtoi(pos) - 1900;
}
break;
case TIME:
strmin = wcschr(pos, L':');
strsec = strmin ? wcschr(strmin + 1, L':') : 0;
tmloc.tm_hour = _wtoi(pos);
if (!strmin) return _time64(0);
tmloc.tm_min = _wtoi(strmin + 1);
if (!strsec) return _time64(0);
tmloc.tm_sec = _wtoi(strsec + 1);
break;
case TIMEZONE:
gmoffset = getGMOffset(pos);
break;
case DATE_END:
pos = 0;
break;
}
state++;
}
while ((pos = getNextField(pos)));
tmloc.tm_isdst = 0; //_daylight;
if (validateTime(&tmloc, gmoffset))
{
result = _mktime64(&tmloc) - _timezone;
//if (_daylight)
}
return result;
}
const wchar_t *getNextField(const wchar_t *pos)
{
if (!pos)
return 0;
while (pos && *pos && !iswspace(*pos)) pos++;
while (pos && *pos && iswspace(*pos)) pos++;
return ((pos && *pos) ? pos : 0);
}
int validateTime(struct tm *tmloc, int gmoffset)
{
int result = 1;
if (tmloc->tm_mday < 1 || tmloc->tm_mday > 31 ||
tmloc->tm_mon < 0 || tmloc->tm_mon > 11 ||
tmloc->tm_year < 0 || tmloc->tm_year > 2000 ||
tmloc->tm_hour < 0 || tmloc->tm_hour > 23 ||
tmloc->tm_min < 0 || tmloc->tm_min > 59 ||
tmloc->tm_sec < 0 || tmloc->tm_sec > 59 ||
gmoffset == 1)
result = 0;
return result;
}
int getMonthOfYear(const wchar_t *str)
{
int mon = -1;
/* This is not the most efficient way to determine
the month (we could use integer comparisons, for instance)
but I don't think this will be a performance bottleneck :)
*/
if (!wcsnicmp(str, L"Jan", 3))
mon = 0;
else if (!wcsnicmp(str, L"Feb", 3))
mon = 1;
else if (!wcsnicmp(str, L"Mar", 3))
mon = 2;
else if (!wcsnicmp(str, L"Apr", 3))
mon = 3;
else if (!wcsnicmp(str, L"May", 3))
mon = 4;
else if (!wcsnicmp(str, L"Jun", 3))
mon = 5;
else if (!wcsnicmp(str, L"Jul", 3))
mon = 6;
else if (!wcsnicmp(str, L"Aug", 3))
mon = 7;
else if (!wcsnicmp(str, L"Sep", 3))
mon = 8;
else if (!wcsnicmp(str, L"Oct", 3))
mon = 9;
else if (!wcsnicmp(str, L"Nov", 3))
mon = 10;
else if (!wcsnicmp(str, L"Dec", 3))
mon = 11;
return mon;
}
int getGMOffset(const wchar_t *str)
{
int secs = 0;
/* See note in getMonthOfYear() */
if (!wcsnicmp(str, L"UT", 2) || !wcsnicmp(str, L"GMT", 3))
secs = 0;
else if (!wcsnicmp(str, L"EDT", 3))
secs = -4 * 3600;
else if (!wcsnicmp(str, L"PST", 3))
secs = -8 * 3600;
else if (!wcsnicmp(str, L"EST", 3) || !wcsnicmp(str, L"CDT", 3))
secs = -5 * 3600;
else if (!wcsnicmp(str, L"MST", 3) || !wcsnicmp(str, L"PDT", 3))
secs = -7 * 3600;
else if (!wcsnicmp(str, L"CST", 3) || !wcsnicmp(str, L"MDT", 3))
secs = -6 * 3600;
else if ( (str[0] == L'+' || str[0] == L'-') &&
iswdigit(str[1]) && iswdigit(str[2]) &&
iswdigit(str[3]) && iswdigit(str[4]))
{
secs = 3600 * (10 * (str[1] - 48) + str[2] - 48);
secs += 60 * (10 * (str[3] - 48) + str[4] - 48);
if (str[0] == L'-')
secs = -secs;
}
else
secs = 1;
return secs;
}