2020-09-30 16:53:55 +02:00

1280 lines
35 KiB
C++

//+----------------------------------------------------------------------------
//
// Job Scheduler service
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1996.
//
// File: sch_at.cxx
//
// Contents: scheduler class object methods to support the NetSchedule
// (AT) APIs.
//
// Classes: CSchedule
//
// History: 30-Jan-96 EricB created
//
//-----------------------------------------------------------------------------
#include "..\pch\headers.hxx"
#pragma hdrstop
#include "Sched.hxx"
//
// Forward references
//
VOID
SetDomTrigger2Days(
DWORD dwDaysOfMonth,
WORD wFirstDayToCheck,
WORD wLastDayToCheck,
SYSTEMTIME *pstStart2,
SYSTEMTIME *pstEnd2);
HRESULT
CSchedule::AddAtJobCommon(
const AT_INFO &At,
DWORD *pID,
CJob **ppJob,
WCHAR wszName[],
size_t cchBuff,
WCHAR wszID[]
)
{
HRESULT hr = S_OK;
//
// If the next at id is > 1 but there aren't any at jobs, the id can be
// reset to 1.
//
if (m_dwNextID > 1 && S_FALSE == _AtTaskExists())
{
ResetAtID();
}
//
// Compose a name for the new AT job.
//
StringCchCopy(wszName, cchBuff, m_ptszFolderPath);
StringCchCat(wszName, cchBuff, L"\\" TSZ_AT_JOB_PREFIX);
_itow(m_dwNextID, wszID, 10);
StringCchCat(wszName, cchBuff, wszID);
StringCchCat(wszName, cchBuff, L"." TSZ_JOB);
//
// Create a new job
//
CJob * pJob = CJob::Create();
if (pJob == NULL)
{
ERR_OUT("CSchedule::AddAtJob: CJob::Create", E_OUTOFMEMORY);
return E_OUTOFMEMORY;
}
//
// Convert the AT command line.
//
WCHAR * pwszApp, * pwszParams, wszCommand[MAX_PATH];
//
// net\svcdlls\atcmd\atcmd.c defines MAX_COMMAND_LEN to be 128. This
// should at some point be changed to MAX_PATH.
//
StringCchCopy(wszCommand, MAX_PATH, At.Command);
pwszApp = wszCommand;
//
// The app name and any command line params are all passed in one string,
// At.Command, so separate the app name from the params. Any path to the
// app plus the app name may be quoted. Otherwise, the parameters are
// separated from the app name by white space.
//
if (*pwszApp == L'"')
{
//
// Initial quote found, scan for end quote. The app name passed to
// SetApplicationName should not be quoted.
//
pwszApp++;
pwszParams = pwszApp + 1;
while (TRUE)
{
if (*pwszParams == L'\0')
{
//
// End of string found, no params.
//
pwszParams = NULL;
break;
}
if (*pwszParams == L'"')
{
//
// End quote found.
//
break;
}
pwszParams++;
}
}
else
{
//
// App path/name not quoted, scan for first white space for parameters.
//
pwszParams = wcspbrk(pwszApp, L" \t");
}
if (pwszParams != NULL)
{
// Null terminate app name string.
//
*pwszParams = L'\0';
//
// Move to first char of the parameters.
//
pwszParams++;
//
// Skip any leading white space.
//
while (*pwszParams != L'\0')
{
if (*pwszParams != L' ' && *pwszParams != L'\t')
{
break;
}
pwszParams++;
}
if (*pwszParams == L'\0')
{
//
// No params.
//
pwszParams = NULL;
}
}
hr = pJob->SetApplicationName(pwszApp);
if (FAILED(hr))
{
ERR_OUT("AddAtJob: SetApplicationName", hr);
pJob->Release();
return hr;
}
if (pwszParams != NULL)
{
hr = pJob->SetParameters(pwszParams);
if (FAILED(hr))
{
ERR_OUT("AddAtJob: SetParameters", hr);
pJob->Release();
return hr;
}
}
pJob->m_rgFlags |= JOB_I_FLAG_NET_SCHEDULE | TASK_FLAG_DELETE_WHEN_DONE;
if (!(At.Flags & JOB_NONINTERACTIVE))
{
pJob->m_rgFlags |= TASK_FLAG_INTERACTIVE;
}
pJob->m_hrStatus = SCHED_S_TASK_READY;
WCHAR szComment[SCH_BUF_LEN + 1];
if (LoadString(g_hInstance,
IDS_NETSCHED_COMMENT,
szComment,
SCH_BUF_LEN) > 0)
{
pJob->SetComment(szComment);
}
//
// Convert from NetSchedule representation to Job Scheduler representation
// of the run dates and times.
//
SYSTEMTIME stNow, stStart;
SYSTEMTIME stDomStart1, stDomEnd1, stDomStart2, stDomEnd2;
SYSTEMTIME stDowStart, stDowEnd;
stDomStart2.wDay = 0; // this serves as a flag
GetLocalTime(&stNow);
stStart = stNow;
//
// JobTime is expressed as milliseconds after midnight, so convert to
// minutes.
//
DWORD dwMins = (DWORD)(At.JobTime / JOB_MILLISECONDS_PER_MINUTE);
stStart.wHour = (WORD)(dwMins / JOB_MINS_PER_HOUR);
stStart.wMinute = (WORD)(dwMins % JOB_MINS_PER_HOUR);
stStart.wSecond = stStart.wMilliseconds = 0;
DWORD DaysOfMonth = At.DaysOfMonth;
WORD wFirstDowRunOffset = 0, wFirstDomRunOffset = 0;
TASK_TRIGGER Trigger;
if (At.Flags & JOB_ADD_CURRENT_DATE)
{
//
// The flag is set, so add today as the first run date.
//
DaysOfMonth |= 1 << (stStart.wDay - 1);
}
else
{
if (DaysOfMonth == 0 && At.DaysOfWeek == 0)
{
//
// Neither bitmask is set, so run at the next opportunity.
//
Trigger.TriggerType = TASK_TIME_TRIGGER_ONCE;
if (! IsFirstTimeEarlier(&stNow, &stStart))
{
// Job runs tomorrow
IncrementDay(&stStart);
}
}
}
//
// Set the trigger values and save the new trigger(s).
//
// Initialize the start and end dates in case this is a periodic trigger.
// If it is not periodic, then new start and end dates will overwrite
// these initialization values.
//
Trigger.cbTriggerSize = sizeof(TASK_TRIGGER);
Trigger.Reserved1 = pJob->m_Triggers.GetCount();
Trigger.wBeginYear = stStart.wYear;
Trigger.wBeginMonth = stStart.wMonth;
Trigger.wBeginDay = stStart.wDay;
Trigger.wEndYear = 0;
Trigger.wEndMonth = 0;
Trigger.wEndDay = 0;
Trigger.wStartHour = stStart.wHour;
Trigger.wStartMinute = stStart.wMinute;
Trigger.Reserved2 = 0;
Trigger.wRandomMinutesInterval = 0;
Trigger.rgFlags = (At.Flags & JOB_RUN_PERIODICALLY)
? 0 : TASK_TRIGGER_FLAG_HAS_END_DATE;
Trigger.MinutesInterval = Trigger.MinutesDuration = 0;
if (DaysOfMonth == 0 && At.DaysOfWeek == 0)
{
// First, zero out the end date flag
Trigger.rgFlags &= ~JOB_RUN_PERIODICALLY;
// This is a TASK_TIME_TRIGGER_ONCE job, and we are ready to commit
hr = pJob->m_Triggers.Add(Trigger);
if (FAILED(hr))
{
ERR_OUT("AddAtJob: m_Triggers.Add Once", hr);
pJob->Release();
return hr;
}
}
if (DaysOfMonth > 0)
{
Trigger.TriggerType = TASK_TIME_TRIGGER_MONTHLYDATE;
Trigger.Type.MonthlyDate.rgfDays = DaysOfMonth;
Trigger.Type.MonthlyDate.rgfMonths = JOB_RGFMONTHS_MAX;
if (!(At.Flags & JOB_RUN_PERIODICALLY))
{
CalcDomTriggerDates(DaysOfMonth,
stNow,
stStart,
&stDomStart1,
&stDomEnd1,
&stDomStart2,
&stDomEnd2);
Trigger.wBeginYear = stDomStart1.wYear;
Trigger.wBeginMonth = stDomStart1.wMonth;
Trigger.wBeginDay = stDomStart1.wDay;
Trigger.wEndYear = stDomEnd1.wYear;
Trigger.wEndMonth = stDomEnd1.wMonth;
Trigger.wEndDay = stDomEnd1.wDay;
}
hr = pJob->m_Triggers.Add(Trigger);
if (FAILED(hr))
{
ERR_OUT("AddAtJob: m_Triggers.Add Dom1", hr);
pJob->Release();
return hr;
}
if (stDomStart2.wDay != 0)
{
Trigger.wBeginYear = stDomStart2.wYear;
Trigger.wBeginMonth = stDomStart2.wMonth;
Trigger.wBeginDay = stDomStart2.wDay;
Trigger.wEndYear = stDomEnd2.wYear;
Trigger.wEndMonth = stDomEnd2.wMonth;
Trigger.wEndDay = stDomEnd2.wDay;
Trigger.Reserved1 = pJob->m_Triggers.GetCount();
hr = pJob->m_Triggers.Add(Trigger);
if (FAILED(hr))
{
ERR_OUT("AddAtJob: m_Triggers.Add Dom2", hr);
pJob->Release();
return hr;
}
}
}
if (At.DaysOfWeek > 0)
{
Trigger.Reserved1 = pJob->m_Triggers.GetCount();
Trigger.TriggerType = TASK_TIME_TRIGGER_WEEKLY;
Trigger.Type.Weekly.WeeksInterval = 1;
//
// Convert AT_INFO DOW to Scheduler DOW:
// Scheduler rgfDaysOfTheWeek: Sunday = bit 0, Monday = bit 1.
// AT_INFO DaysOfWeek: Monday = bit 0, Sunday = bit 6.
//
Trigger.Type.Weekly.rgfDaysOfTheWeek = At.DaysOfWeek << 1;
if (Trigger.Type.Weekly.rgfDaysOfTheWeek & 0x0080)
{
Trigger.Type.Weekly.rgfDaysOfTheWeek &= ~0x0080;
Trigger.Type.Weekly.rgfDaysOfTheWeek |= 1;
}
if (!(At.Flags & JOB_RUN_PERIODICALLY))
{
CalcDowTriggerDate(stNow,
stStart,
&stDowStart,
&stDowEnd);
Trigger.wBeginYear = stDowStart.wYear;
Trigger.wBeginMonth = stDowStart.wMonth;
Trigger.wBeginDay = stDowStart.wDay;
Trigger.wEndYear = stDowEnd.wYear;
Trigger.wEndMonth = stDowEnd.wMonth;
Trigger.wEndDay = stDowEnd.wDay;
}
hr = pJob->m_Triggers.Add(Trigger);
if (FAILED(hr))
{
ERR_OUT("AddAtJob: m_Triggers.Add", hr);
pJob->Release();
return hr;
}
}
// get the AT task maximum run time from the registry
// if the call fails, the default of 72 will be used
DWORD dwMaxRunTime = 0;
if (SUCCEEDED(GetAtTaskMaxHours(&dwMaxRunTime)))
pJob->SetMaxRunTime(dwMaxRunTime);
//
// Set the same flags as in CJob::CreateTrigger. (We should really call
// CreateTrigger in this function instead of creating the trigger ourselves.)
//
pJob->SetFlag(JOB_I_FLAG_PROPERTIES_DIRTY);
pJob->SetTriggersDirty();
//
// Save the new job. Obviously this is one place we definitely don't want
// the AT job flag cleared on save!
//
hr = pJob->SaveWithRetry(wszName,
TRUE,
SAVEP_VARIABLE_LENGTH_DATA |
SAVEP_PRESERVE_NET_SCHEDULE);
if (FAILED(hr))
{
if (hr == HRESULT_FROM_WIN32(ERROR_FILE_EXISTS))
{
//
// Name collision; someone has renamed a task to match the next
// AT job. Recalc the max AT job ID.
//
GetNextAtID(&m_dwNextID);
StringCchCopy(wszName, cchBuff, m_ptszFolderPath);
StringCchCat(wszName, cchBuff, L"\\" TSZ_AT_JOB_PREFIX);
_itow(m_dwNextID, wszID, 10);
StringCchCat(wszName, cchBuff, wszID);
StringCchCat(wszName, cchBuff, TSZ_DOTJOB);
//
// Now, retry the save.
//
hr = pJob->SaveWithRetry(wszName,
TRUE,
SAVEP_VARIABLE_LENGTH_DATA |
SAVEP_PRESERVE_NET_SCHEDULE);
if (FAILED(hr))
{
ERR_OUT("CSchedule::AddAtJob: Save", hr);
pJob->Release();
return hr;
}
}
else
{
ERR_OUT("CSchedule::AddAtJob: Save", hr);
pJob->Release();
return hr;
}
}
*ppJob = pJob;
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Member: CSchedule::GetAtTaskMaxHours
//
// Synopsis: Check a registry setting to see what max run time value a user
// has specified for AT tasks. If the key is not present or can't
// be opened then interpret as the normal task default of 72.
//
// Arguments: none
//
// Returns: bool
//
// Notes: This method is not exposed to external clients, thus it is not
// part of a public interface.
//-----------------------------------------------------------------------------
HRESULT CSchedule::GetAtTaskMaxHours(DWORD* pdwMaxHours)
{
//
// Open the schedule service key
//
long lErr;
HKEY hSchedKey;
lErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE, SCH_SVC_KEY, 0, KEY_READ | KEY_WRITE, &hSchedKey);
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("RegOpenKeyEx of Scheduler key", lErr);
return(HRESULT_FROM_WIN32(lErr));
}
//
// Get the AtTaskMaxHours setting
//
bool bNeedToUpdate = false;
DWORD dwMaxHours = 0;
DWORD cb = sizeof(DWORD);
lErr = RegQueryValueEx(hSchedKey, SCH_ATTASKMAXHOURS_VALUE, NULL, NULL, (LPBYTE)&dwMaxHours, &cb);
if (lErr != ERROR_SUCCESS)
{
if (lErr != ERROR_FILE_NOT_FOUND)
{
ERR_OUT("Read of AtTaskMaxHours registry value", lErr);
RegCloseKey(hSchedKey);
return(HRESULT_FROM_WIN32(lErr));
}
//
// Need to create the missing registry entry
//
dwMaxHours = DEFAULT_MAXRUNTIME_HOURS;
bNeedToUpdate = true;
}
//
// Correct out-of-bounds stored registry values
//
if (dwMaxHours > 999)
{
dwMaxHours = 999;
bNeedToUpdate = true;
}
if (bNeedToUpdate)
{
lErr = RegSetValueEx(hSchedKey, SCH_ATTASKMAXHOURS_VALUE, 0, REG_DWORD,(CONST BYTE *)&dwMaxHours, sizeof(DWORD));
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("Update of AtTaskMaxHours registry value", lErr);
RegCloseKey(hSchedKey);
return(HRESULT_FROM_WIN32(lErr));
}
}
//
// Convert the stored value to a value that has meaning to the scheduler
//
if (!dwMaxHours)
dwMaxHours = INFINITE; // If value is zero, return infinite (-1) for max run time
else
dwMaxHours *= 3600000; // The stored value is in hours, but the value passed back
// needs to be in milliseconds, so convert
RegCloseKey(hSchedKey);
// Set the value to be passed back
*pdwMaxHours = dwMaxHours;
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Member: CSchedule::AddAtJob
//
// Synopsis: create a downlevel job
//
// Arguments: [At] - reference to an AT_INFO struct
// [pID] - returns the new ID (optional, can be NULL)
//
// Returns: HRESULTS
//
// Notes: This method is not exposed to external clients, thus it is not
// part of a public interface.
//-----------------------------------------------------------------------------
STDMETHODIMP
CSchedule::AddAtJob(const AT_INFO &At, DWORD * pID)
{
TRACE(CSchedule, AddAtJob);
HRESULT hr = S_OK;
CJob *pJob;
WCHAR wszName[MAX_PATH + 1];
WCHAR wszID[SCH_SMBUF_LEN];
hr = AddAtJobCommon(At, pID, &pJob, wszName, MAX_PATH + 1, wszID);
if (FAILED(hr))
{
ERR_OUT("AddAtJob: AddAtJobCommon", hr);
return hr;
}
//
// Free the job object.
//
pJob->Release();
//
// Return the new job's ID and increment the ID counter
//
if (pID != NULL)
{
*pID = m_dwNextID;
}
hr = IncrementAndSaveID();
return hr;
}
//+---------------------------------------------------------------------------
//
// Function: IsMonthBitSet
//
// Synopsis: Returns nonzero if 1-based bit [wDay] in [dwDaysOfMonth] is
// set.
//
// History: 09-26-96 DavidMun Created
//
//----------------------------------------------------------------------------
inline BOOL
IsMonthBitSet(DWORD dwDaysOfMonth, WORD wDay)
{
return dwDaysOfMonth & (1 << (wDay - 1));
}
//+---------------------------------------------------------------------------
//
// Function: CalcDomTriggerDates
//
// Synopsis: Calculate the dates for the start and end of the Day Of Month
// trigger(s).
//
// Arguments: [dwDaysOfMonth] - bit array, bit 0=day 1, etc. At least one
// bit must be set!
// [stNow] - current time
// [stStart] - same as [stNow] but has hour & minute
// values of actual job run time
// [pstStart1] - filled with start date of first trigger
// [pstEnd1] - filled with end date of first trigger
// [pstStart2] - filled with start date of second trigger;
// wDay is 0 if second trigger not needed
// [pstEnd2] - filled with end date of second trigger,
// wDay is 0 if second trigger not needed
//
// Modifies: All output arguments.
//
// History: 09-26-96 DavidMun Created
//
// Notes: Only the month, day, and year values in the output structs
// are used.
//
//----------------------------------------------------------------------------
VOID
CalcDomTriggerDates(
DWORD dwDaysOfMonth,
const SYSTEMTIME &stNow,
const SYSTEMTIME &stStart,
SYSTEMTIME *pstStart1,
SYSTEMTIME *pstEnd1,
SYSTEMTIME *pstStart2,
SYSTEMTIME *pstEnd2)
{
BOOL fDone = FALSE;
WORD wStart1MonthDays;
// assert not all bits below 32 are zero
Win4Assert(dwDaysOfMonth & JOB_RGFDAYS_MAX);
//
// Find the start date for the first DOM trigger.
//
*pstStart1 = stNow;
if (IsFirstTimeEarlier(&stStart, &stNow))
{
// already past run time today
IncrementDay(pstStart1);
}
HRESULT hr = MonthDays(pstStart1->wMonth, pstStart1->wYear, &wStart1MonthDays);
if (FAILED(hr))
{
schAssert(!"Bad systemtime");
return;
}
do
{
while (!IsMonthBitSet(dwDaysOfMonth, pstStart1->wDay) &&
pstStart1->wDay <= wStart1MonthDays)
{
pstStart1->wDay++;
}
//
// now either:
// start1.wDay > wStart1MonthDays or
// bit at start1.wDay is 1
//
if (pstStart1->wDay > wStart1MonthDays)
{
// have to go on to next month to get the first start date
pstStart1->wDay = 1;
IncrementMonth(pstStart1);
MonthDays(pstStart1->wMonth, pstStart1->wYear, &wStart1MonthDays);
}
else
{
fDone = TRUE;
}
} while (!fDone);
//
// Now bit at pstStart1->wDay is on, and pstStart1->wDay is a valid day in
// pstStart1->wMonth, and wStart1MonthDays is the number of days in the
// month pstStart1->wMonth. Next we need to find end1.
//
// end1 is initialized to start1.
//
// If there are any days set before the start day, then end1.wMonth will
// be start1.wMonth + 1, and end1.wDay will be the last of the days that
// is set before start1.wDay, with the restriction that end1.wDay is not
// greater than the number of days in end1.wMonth.
//
*pstEnd1 = *pstStart1;
WORD wDay;
if (pstStart1->wDay > 1)
{
WORD wEnd1Month;
WORD wEnd1MonthDays;
wEnd1Month = pstEnd1->wMonth + 1;
if (wEnd1Month > 12)
{
wEnd1Month = 1;
}
MonthDays(wEnd1Month, pstEnd1->wYear, &wEnd1MonthDays);
WORD wMaxDay = min(pstStart1->wDay - 1, wEnd1MonthDays);
for (wDay = 1; wDay <= wMaxDay; wDay++)
{
if (IsMonthBitSet(dwDaysOfMonth, wDay))
{
pstEnd1->wDay = wDay;
}
}
}
//
// If any day bits were set before start1.wDay then end1.wDay will no
// longer == start1.wDay, and end1 will be referring to the next month.
//
// Otherwise, End1 will remain in the same month as Start1, but will
// need to be set to the last day bit set in the Start1 month.
//
if (pstEnd1->wDay < pstStart1->wDay)
{
IncrementMonth(pstEnd1);
}
else
{
for (wDay = pstStart1->wDay + 1; wDay <= wStart1MonthDays; wDay++)
{
if (IsMonthBitSet(dwDaysOfMonth, wDay))
{
pstEnd1->wDay = wDay;
}
}
}
//
// Now start1 and end1 are set. next, check if there's a need for the
// second trigger. There are two cases where a second trigger is
// required.
//
// Case a: second trigger must fill time between end of first trigger
// and start of first trigger. for example, job is to run on next
// 1, 30, 31 and start1 is 1/31. then end1 will be 2/1, and a second
// trigger must go from 3/30 to 3/30. Note this case can only occur
// if End1.wMonth == February.
//
// Case b: second trigger must fill time somewhere in the 29-31 day range.
// For example, job is to run on next 1-31, and current day is 4/1. so
// start1 is 4/1, end1 is 4/30, then start2 must be 5/31 to 5/31.
//
// As another example, job is to run on next 27, 28, 30, current day is
// 2/28. then start1 is 2/28, end1 is 3/27, start2 is 3/30, end2 is 3/30.
//
// Case b only occurs when there are bits set for days beyond the last day
// of pstStart1->wmonth.
//
//
//
// test if we need case a.
//
if (pstEnd1->wMonth == 2 &&
pstStart1->wDay > 29 &&
pstEnd1->wDay < pstStart1->wDay - 1)
{
//
// There's a gap between end1 and start1. we need the second trigger
// if there are any day bits set in that gap.
//
Win4Assert(pstStart1->wMonth == 1);
Win4Assert(pstEnd1->wDay + 1 <= 30);
*pstStart2 = *pstEnd1;
pstStart2->wMonth = 3;
*pstEnd2 = *pstStart2;
SetDomTrigger2Days(dwDaysOfMonth,
pstEnd1->wDay + 1,
pstStart1->wDay - 1,
pstStart2,
pstEnd2);
}
else if (wStart1MonthDays < 31)
{
//
// we have case b if any bits after the last day in pstStart1->wMonth
// are set.
//
*pstStart2 = *pstEnd1;
if (pstEnd1->wMonth == pstStart1->wMonth)
{
pstStart2->wMonth++;
}
*pstEnd2 = *pstStart2;
SetDomTrigger2Days(dwDaysOfMonth,
wStart1MonthDays + 1,
31,
pstStart2,
pstEnd2);
}
else
{
// no second trigger
pstStart2->wDay = 0;
pstEnd2->wDay = 0;
}
}
//+---------------------------------------------------------------------------
//
// Function: SetDomTrigger2Days
//
// Synopsis: Set the start and end dates for the second DOM trigger.
//
// Arguments: [dwDaysOfMonth] - bit array, bit 0=day 1, etc. At least
// one bit must be set!
// [wFirstDayToCheck] - 1 based
// [wLastDayToCheck] - 1 based, must be >= [wFirstDayToCheck]
// [pstStart2] - filled with start date of second
// trigger; wDay is 0 if no second
// trigger is required.
// [pstEnd2] - filled with end date of second trigger;
// wDay is 0 if no second trigger is
// required.
//
// Modifies: All out args.
//
// History: 09-26-96 DavidMun Created
//
// Notes: This is a helper function called only by
// CalcDomTriggerDates.
//
//----------------------------------------------------------------------------
VOID
SetDomTrigger2Days(
DWORD dwDaysOfMonth,
WORD wFirstDayToCheck,
WORD wLastDayToCheck,
SYSTEMTIME *pstStart2,
SYSTEMTIME *pstEnd2)
{
WORD wDay;
pstStart2->wDay = 0;
pstEnd2->wDay = 0;
for (wDay = wFirstDayToCheck; wDay <= wLastDayToCheck; wDay++)
{
if (IsMonthBitSet(dwDaysOfMonth, wDay))
{
//
// if the start of the second trigger hasn't been assigned
// yet, assign it. otherwise update the end of the second
// trigger to the current day.
//
if (!pstStart2->wDay)
{
pstStart2->wDay = wDay;
}
else
{
pstEnd2->wDay = wDay;
}
}
}
//
// there may have been only one day on in the gap, so the start and
// end of trigger 2 are the same day.
//
if (pstStart2->wDay && !pstEnd2->wDay)
{
pstEnd2->wDay = pstStart2->wDay;
}
}
//+---------------------------------------------------------------------------
//
// Function: CalcDowTriggerDate
//
// Synopsis: Set the start and end dates for the Day of Week trigger.
//
// Arguments: [stNow] - Current time
// [stStart] - same as [stNow] but with hour and minute of
// actual run time
// [pstStart] - filled with start date
// [pstEnd] - filled with end date
//
// Modifies: *[pstStart], *[pstEnd]
//
// History: 09-26-96 DavidMun Created
//
//----------------------------------------------------------------------------
VOID
CalcDowTriggerDate(
const SYSTEMTIME &stNow,
const SYSTEMTIME &stStart,
SYSTEMTIME *pstStart,
SYSTEMTIME *pstEnd)
{
*pstStart = stNow;
//
// If it's too late for the job to run today, make the start date
// tomorrow.
//
if (IsFirstTimeEarlier(&stStart, &stNow))
{
IncrementDay(pstStart);
}
//
// Make the end date 6 days later than the start date, that way we cover a
// full week and all runs will happen.
//
*pstEnd = *pstStart;
pstEnd->wDay += 6;
WORD wLastDay;
HRESULT hr = MonthDays(pstEnd->wMonth, pstEnd->wYear, &wLastDay);
if (FAILED(hr))
{
schAssert(!"Bad systemtime");
}
else
{
if (pstEnd->wDay > wLastDay)
{
//
// Wrap to the next month.
//
pstEnd->wDay -= wLastDay;
IncrementMonth(pstEnd);
}
}
}
//+----------------------------------------------------------------------------
//
// Member: CSchedule::GetAtJob, private
//
// Synopsis: retrieves a downlevel job's AT info
//
// Arguments: [pwszFileName] - job object's file name
// [pAt] - pointer to an AT_INFO struct
// [pwszCommand] - buffer for the command string
// [pcchCommand] - on input, size of supplied buffer, on output,
// size needed if supplied buffer is too small.
//
// Returns: HRESULTS - ERROR_INSUFFICIENT_BUFFER if too small
// - SCHED_E_NOT_AN_AT_JOB if not an AT job
//
// Notes: This method is not exposed to external clients, thus it is not
// part of a public interface.
//-----------------------------------------------------------------------------
STDMETHODIMP
CSchedule::GetAtJob(LPCTSTR pwszFileName, AT_INFO * pAt, LPWSTR pwszCommand,
DWORD * pcchCommand)
{
TRACE(CSchedule, GetAtJob);
CJob * pJob = CJob::Create();
if (pJob == NULL)
{
return E_OUTOFMEMORY;
}
HRESULT hr = pJob->LoadP(pwszFileName, 0, TRUE, TRUE);
if (FAILED(hr))
{
ERR_OUT("GetAtJob: LoadP", hr);
pJob->Release();
return hr;
}
hr = pJob->GetAtInfo(pAt, pwszCommand, pcchCommand);
if (FAILED(hr))
{
ERR_OUT("GetAtJob: GetAtInfo", hr);
pJob->Release();
return hr;
}
pJob->Release();
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Member: CSchedule::IncrementAndSaveID
//
// Synopsis: Increment the NextJobID value and save it to the registry.
//
//-----------------------------------------------------------------------------
HRESULT
CSchedule::IncrementAndSaveID(void)
{
EnterCriticalSection(&m_CriticalSection);
long lErr;
HKEY hSchedKey;
lErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE, SCH_SVC_KEY, 0, KEY_SET_VALUE,
&hSchedKey);
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("RegOpenKeyEx of Scheduler key", lErr);
LeaveCriticalSection(&m_CriticalSection);
return(HRESULT_FROM_WIN32(lErr));
}
m_dwNextID++;
//
// update the registry entry
//
lErr = RegSetValueEx(hSchedKey, SCH_NEXTATJOBID_VALUE, 0, REG_DWORD,
(CONST BYTE *)&m_dwNextID, sizeof(DWORD));
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("Create of NextAtJobId registry value", lErr);
m_dwNextID--;
RegCloseKey(hSchedKey);
LeaveCriticalSection(&m_CriticalSection);
return(HRESULT_FROM_WIN32(lErr));
}
RegCloseKey(hSchedKey);
LeaveCriticalSection(&m_CriticalSection);
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Member: CSchedule::ResetAtID
//
// Synopsis: Set the next at id value in the registry to 1
//
//-----------------------------------------------------------------------------
HRESULT
CSchedule::ResetAtID(void)
{
HRESULT hr = S_OK;
HKEY hSchedKey = NULL;
EnterCriticalSection(&m_CriticalSection);
m_dwNextID = 1;
do
{
LONG lErr;
//
// Open the schedule service key
//
lErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
SCH_SVC_KEY,
0,
KEY_READ | KEY_WRITE,
&hSchedKey);
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("RegOpenKeyEx of Scheduler key", lErr);
hr = HRESULT_FROM_WIN32(lErr);
break;
}
//
// Set the next At Job ID value to 1. If the value is not present,
// it will be created.
//
lErr = RegSetValueEx(hSchedKey,
SCH_NEXTATJOBID_VALUE,
0,
REG_DWORD,
(CONST BYTE *) &m_dwNextID,
sizeof(m_dwNextID));
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("Create of NextAtJobId registry value", lErr);
hr = HRESULT_FROM_WIN32(lErr);
}
} while (0);
if (hSchedKey)
{
RegCloseKey(hSchedKey);
}
LeaveCriticalSection(&m_CriticalSection);
return hr;
}
//+----------------------------------------------------------------------------
//
// Member: CSchedule::_AtTaskExists, private
//
// Synopsis: Check for existing AT tasks
//
// Returns: S_OK - an AT task was found
// S_FALSE - no AT tasks were found
// E_*
//
//-----------------------------------------------------------------------------
HRESULT
CSchedule::_AtTaskExists(void)
{
WIN32_FIND_DATA fd;
HANDLE hFileFindContext;
hFileFindContext = FindFirstFile(g_wszAtJobSearchPath, &fd);
if (hFileFindContext == INVALID_HANDLE_VALUE)
{
return S_FALSE;
}
CJob * pJob = CJob::Create();
if (pJob == NULL)
{
return E_OUTOFMEMORY;
}
HRESULT hr = S_FALSE;
DWORD rgFlags;
do
{
if (!IsValidAtFilename(fd.cFileName))
{
continue;
}
HRESULT hrLoad = LoadAtJob(pJob, fd.cFileName);
if (FAILED(hrLoad))
{
hr = hrLoad;
break;
}
pJob->GetAllFlags(&rgFlags);
if (rgFlags & JOB_I_FLAG_NET_SCHEDULE)
{
hr = S_OK;
break;
}
} while (FindNextFile(hFileFindContext, &fd));
FindClose(hFileFindContext);
pJob->Release();
return hr;
}
//+----------------------------------------------------------------------------
//
// Member: CSchedule::InitAtID
//
// Synopsis: Obtains the current AT task ID from the registry.
//
//-----------------------------------------------------------------------------
HRESULT
CSchedule::InitAtID(void)
{
//
// Open the schedule service key
//
long lErr;
HKEY hSchedKey;
lErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE, SCH_SVC_KEY, 0,
KEY_READ | KEY_WRITE, &hSchedKey);
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("RegOpenKeyEx of Scheduler key", lErr);
return(HRESULT_FROM_WIN32(lErr));
}
//
// Get the next At Job ID
//
DWORD cb = sizeof(DWORD);
lErr = RegQueryValueEx(hSchedKey, SCH_NEXTATJOBID_VALUE, NULL, NULL,
(LPBYTE)&m_dwNextID, &cb);
if (lErr != ERROR_SUCCESS)
{
if (lErr != ERROR_FILE_NOT_FOUND)
{
ERR_OUT("Read of NextAtJobId registry value", lErr);
RegCloseKey(hSchedKey);
return(HRESULT_FROM_WIN32(lErr));
}
//
// Scan AT jobs for value if registry entry absent
//
GetNextAtID(&m_dwNextID);
//
// Create registry entry
//
lErr = RegSetValueEx(hSchedKey, SCH_NEXTATJOBID_VALUE, 0, REG_DWORD,
(CONST BYTE *)&m_dwNextID, sizeof(DWORD));
if (lErr != ERROR_SUCCESS)
{
ERR_OUT("Create of NextAtJobId registry value", lErr);
RegCloseKey(hSchedKey);
return(HRESULT_FROM_WIN32(lErr));
}
}
RegCloseKey(hSchedKey);
return S_OK;
}