mirror of
https://github.com/SickGear/SickGear.git
synced 2024-11-16 18:05:04 +00:00
167 lines
7.8 KiB
Python
167 lines
7.8 KiB
Python
"""
|
|
ms task/job file parser
|
|
|
|
Author: Jeff Bryner
|
|
Creation date: 2010-11
|
|
References:
|
|
http://msdn.microsoft.com/en-us/library/cc248286%28v=PROT.13%29.aspx
|
|
http://msdn.microsoft.com/en-us/library/cc248287%28v=PROT.13%29.aspx
|
|
http://technet.microsoft.com/en-us/library/bb490996.aspx
|
|
"""
|
|
|
|
|
|
from hachoir.parser import Parser
|
|
from hachoir.field import (FieldSet, RootSeekableFieldSet,
|
|
UInt32, UInt16,
|
|
Bit, RawBits, RawBytes, Enum)
|
|
from hachoir.core.endian import LITTLE_ENDIAN
|
|
from hachoir.parser.common.win32 import PascalStringWin16, GUID
|
|
|
|
|
|
class TaskTrigger(FieldSet):
|
|
TRIGGER_TYPE = {
|
|
0x00000000: "ONCE",
|
|
0x00000001: "DAILY",
|
|
0x00000002: "WEEKLY",
|
|
0x00000003: "MONTHLYDATE",
|
|
0x00000004: "MONTHLYDOW",
|
|
0x00000005: "EVENT_ON_IDLE",
|
|
0x00000006: "EVENT_AT_SYSTEMSTART",
|
|
0x00000007: "EVENT_AT_LOGON"
|
|
}
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
FieldSet.__init__(self, *args, **kwargs)
|
|
self._size = self["TriggerSize"].value * 8
|
|
|
|
def createFields(self):
|
|
yield UInt16(self, "TriggerSize")
|
|
yield UInt16(self, "Reserved[]")
|
|
yield UInt16(self, "BeginYear")
|
|
yield UInt16(self, "BeginMonth")
|
|
yield UInt16(self, "BeginDay")
|
|
yield UInt16(self, "EndYear")
|
|
yield UInt16(self, "EndMonth")
|
|
yield UInt16(self, "EndDay")
|
|
yield UInt16(self, "StartHour")
|
|
yield UInt16(self, "StartMinute")
|
|
yield UInt32(self, "MinutesDuration")
|
|
yield UInt32(self, "MinutesInterval", "Time period between repeated trigger firings.")
|
|
yield Bit(self, "HasEndDate", "Can task stop at some point in time?")
|
|
yield Bit(self, "KillAtDurationEnd", "Can task be stopped at the end of the repetition period?")
|
|
yield Bit(self, "TriggerDisabled", "Is this trigger disabled?")
|
|
yield RawBits(self, "Unused[]", 29)
|
|
yield Enum(UInt32(self, "TriggerType"), self.TRIGGER_TYPE)
|
|
yield UInt16(self, "TriggerSpecific0")
|
|
yield UInt16(self, "TriggerSpecific1")
|
|
yield UInt16(self, "TriggerSpecific2")
|
|
yield UInt16(self, "Padding")
|
|
yield UInt16(self, "Reserved[]")
|
|
yield UInt16(self, "Reserved[]")
|
|
|
|
|
|
class MSTaskFile(Parser, RootSeekableFieldSet):
|
|
PARSER_TAGS = {
|
|
"id": "mstask",
|
|
"category": "misc", # "archive", "audio", "container", ...
|
|
# TODO: Example ("bmp",) to parse the file "image.bmp"
|
|
"file_ext": ("job",),
|
|
# TODO: Minimum file size (x bits, or x*8 in bytes)
|
|
"min_size": 100,
|
|
# TODO: Example: "A bitmap picture",
|
|
"description": ".job 'at' file parser from ms windows",
|
|
}
|
|
|
|
endian = LITTLE_ENDIAN
|
|
|
|
PRODUCT_VERSION = {
|
|
0x0400: "Windows NT 4.0",
|
|
0x0500: "Windows 2000",
|
|
0x0501: "Windows XP",
|
|
0x0600: "Windows Vista",
|
|
0x0601: "Windows 7"
|
|
}
|
|
|
|
TASK_STATUS = {
|
|
0x00041300: "Task Ready",
|
|
0x00041301: "Task running",
|
|
0x00041302: "Task disabled",
|
|
0x00041303: "Task has not run",
|
|
0x00041304: "Task has no more runs",
|
|
0x00041305: "Task not scheduled",
|
|
0x00041306: "Task terminated",
|
|
0x00041307: "Task has no valid triggers",
|
|
0x00041308: "Task contains only event triggers that do not have set run times",
|
|
0x00041309: "Task trigger not found",
|
|
0x0004130A: "One or more of the properties that are required to run this task have not been set.",
|
|
0x0004130B: "There is no running instance of the task",
|
|
0x0004130C: "Task Schedule Remoting Protocol service is not installed",
|
|
0x0004130D: "Task object cannot be opened",
|
|
0x0004130E: "Task object is invalid",
|
|
0x0004130F: "No Account information could be found in Task Scheduler Remoting Protocol security database for the task indicated."
|
|
}
|
|
|
|
def validate(self):
|
|
# The MAGIC for a task file is the windows version that created it
|
|
# http://msdn.microsoft.com/en-us/library/2d1fbbab-fe6c-4ae5-bdf5-41dc526b2439%28v=PROT.13%29#id11
|
|
if self['WindowsVersion'].value not in self.PRODUCT_VERSION:
|
|
return "Invalid Product Version Field"
|
|
return True
|
|
|
|
def createFields(self):
|
|
yield Enum(UInt16(self, "WindowsVersion"), self.PRODUCT_VERSION)
|
|
yield UInt16(self, "FileVersion")
|
|
yield GUID(self, "JobUUID")
|
|
yield UInt16(self, "AppNameOffset", "App Name Length Offset")
|
|
yield UInt16(self, "TriggerOffset", "Contains the offset in bytes within the .JOB file where the task triggers are located.")
|
|
yield UInt16(self, "ErrorRetryCount", "Contains the number of execute attempts that are attempted for the task if the task fails to start.")
|
|
yield UInt16(self, "ErrorRetryInterval", "Contains the interval, in minutes, between successive retries")
|
|
yield UInt16(self, "IdleDeadline", "Contains a maximum time in minutes to wait for the machine to become idle for Idle Wait minutes.")
|
|
yield UInt16(self, "IdleWait", "Contains a value in minutes. The machine remains idle for this many minutes before it runs the task")
|
|
yield UInt32(self, "Priority")
|
|
yield UInt32(self, "MaxRunTime", "Maximum run time in milliseconds")
|
|
yield UInt32(self, "ExitCode", "This contains the exit code of the executed task upon the completion of that task.")
|
|
yield Enum(UInt32(self, "Status"), self.TASK_STATUS)
|
|
yield Bit(self, "Interactive", "Can Task interact with user?")
|
|
yield Bit(self, "DeleteWhenDone", "Remove the task file when done?")
|
|
yield Bit(self, "Disabled", "Is Task disabled?")
|
|
yield Bit(self, "StartOnlyIfIdle", "Task begins only if computer is not in use at the scheduled time")
|
|
yield Bit(self, "KillOnIdleEnd", "Kill task if user input is detected, terminating idle state?")
|
|
yield Bit(self, "DontStartIfOnBatteries")
|
|
yield Bit(self, "KillIfGoingOnBatteries")
|
|
yield Bit(self, "RunOnlyIfDocked")
|
|
yield Bit(self, "HiddenTask")
|
|
yield Bit(self, "RunIfConnectedToInternet")
|
|
yield Bit(self, "RestartOnIdleResume")
|
|
yield Bit(self, "SystemRequired", "Can task cause system to resume or awaken if system is sleeping?")
|
|
yield Bit(self, "OnlyIfUserLoggedOn")
|
|
yield Bit(self, "ApplicationNameExists", "Does task have an application name defined?")
|
|
yield Bit(self, "Unused[]")
|
|
yield Bit(self, "Unused[]")
|
|
yield RawBytes(self, "flags", 2)
|
|
yield UInt16(self, "LastRunYear")
|
|
yield UInt16(self, "LastRunMonth")
|
|
yield UInt16(self, "LastRunWeekday", "Sunday=0,Saturday=6")
|
|
yield UInt16(self, "LastRunDay")
|
|
yield UInt16(self, "LastRunHour")
|
|
yield UInt16(self, "LastRunMinute")
|
|
yield UInt16(self, "LastRunSecond")
|
|
yield UInt16(self, "LastRunMillisecond")
|
|
yield UInt16(self, "RunningInstanceCount")
|
|
yield PascalStringWin16(self, "AppNameLength", strip='\0')
|
|
yield PascalStringWin16(self, "Parameters", strip='\0')
|
|
yield PascalStringWin16(self, "WorkingDirectory", strip='\0')
|
|
yield PascalStringWin16(self, "Author", strip='\0')
|
|
yield PascalStringWin16(self, "Comment", strip='\0')
|
|
|
|
yield UInt16(self, "UserDataSize")
|
|
# todo: read optional userdata
|
|
yield UInt16(self, "ReservedDataSize")
|
|
if self["ReservedDataSize"].value == 8:
|
|
yield Enum(UInt32(self, "StartError", "contains the HRESULT error from the most recent attempt to start the task"), self.TASK_STATUS)
|
|
yield UInt32(self, "TaskFlags")
|
|
elif self["ReservedDataSize"].value:
|
|
yield RawBytes(self, "Reserved", self["ReservedDataSize"].value)
|
|
yield UInt16(self, "TriggerCount", "size of the array of triggers")
|
|
for i in range(self["TriggerCount"].value):
|
|
yield TaskTrigger(self, "Trigger[]")
|