mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2083 lines
94 KiB
2083 lines
94 KiB
'********************************************************************
|
|
'*
|
|
'* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
'*
|
|
'* Module Name: EVENTQUERY.vbs
|
|
'*
|
|
'* Abstract: Enables an administrator to query/view all existing
|
|
'* events in a given event log(s).
|
|
'*
|
|
'*
|
|
'********************************************************************
|
|
' Global declaration
|
|
OPTION EXPLICIT
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
'----------------------------------------------------------------
|
|
' Start of localization Content
|
|
'----------------------------------------------------------------
|
|
|
|
' the filter operators specified by the user
|
|
CONST L_OperatorEq_Text = "eq"
|
|
CONST L_OperatorNe_Text = "ne"
|
|
CONST L_OperatorGe_Text = "ge"
|
|
CONST L_OperatorLe_Text = "le"
|
|
CONST L_OperatorGt_Text = "gt"
|
|
CONST L_OperatorLt_Text = "lt"
|
|
|
|
' the filters as given by the user
|
|
CONST L_UserFilterDateTime_Text = "datetime"
|
|
CONST L_UserFilterType_Text = "type"
|
|
CONST L_UserFilterUser_Text = "user"
|
|
CONST L_UserFilterComputer_Text = "computer"
|
|
CONST L_UserFilterSource_Text = "source"
|
|
CONST L_UserFilterDateCategory_Text = "category"
|
|
CONST L_UserFilterId_Text = "id"
|
|
|
|
' Define default values
|
|
CONST L_ConstDefaultFormat_Text = "TABLE"
|
|
|
|
' Define other format values
|
|
CONST L_Const_List_Format_Text = "LIST"
|
|
CONST L_Const_Csv_Format_Text = "CSV"
|
|
|
|
' the text displayed in columns when no output is obtained for display
|
|
CONST L_TextNa_Text = "N/A"
|
|
CONST L_TextNone_Text = "None"
|
|
|
|
' the following texts are used while parsing the command-line arguments
|
|
' (passed as input to the function component.getArguments)
|
|
CONST L_MachineName_Text = "Server Name"
|
|
CONST L_UserName_Text = "User Name"
|
|
CONST L_UserPassword_Text = "User Password"
|
|
CONST L_Format_Text = "Format"
|
|
CONST L_Range_Text = "Range"
|
|
CONST L_Filter_Text = "Filter"
|
|
CONST L_Log_Text = "Logname"
|
|
|
|
' the column headers used in the output display
|
|
CONST L_ColHeaderType_Text = "Type"
|
|
CONST L_ColHeaderDateTime_Text = "Date Time"
|
|
CONST L_ColHeaderSource_Text = "Source"
|
|
CONST L_ColHeaderCategory_Text = "Category"
|
|
CONST L_ColHeaderEventcode_Text = "Event"
|
|
CONST L_ColHeaderUser_Text = "User"
|
|
CONST L_ColHeaderComputerName_Text = "ComputerName"
|
|
CONST L_ColHeaderDesription_Text = "Description"
|
|
|
|
' variable use to concatenate the Localization Strings.
|
|
' Error Messages
|
|
Dim UseCscriptErrorMessage
|
|
Dim InvalidParameterErrorMessage
|
|
Dim InvalidFormatErrorMessage
|
|
Dim InvalidCredentialsForServerErrorMessage
|
|
Dim InvalidCredentialsForUserErrorMessage
|
|
Dim InvalidSyntaxErrorMessage
|
|
Dim InvalidInputErrorMessage
|
|
Dim InvalidORSyntaxInFilterErrorMessage
|
|
Dim InvalidSyntaxMoreNoRepeatedErrorMessage
|
|
|
|
UseCscriptErrorMessage = L_UseCscript1_ErrorMessage & vbCRLF & _
|
|
L_UseCscript2_ErrorMessage & vbCRLF & vbCRLF & _
|
|
L_UseCscript3_ErrorMessage & vbCRLF & _
|
|
L_UseCscript4_ErrorMessage & vbCRLF & vbCRLF & _
|
|
L_UseCscript5_ErrorMessage
|
|
|
|
CONST L_HelpSyntax1_Message = "Type ""%1 /?"" for usage."
|
|
CONST L_HelpSyntax2_Message = "Type ""%2 /?"" for usage."
|
|
|
|
CONST L_InvalidParameter1_ErrorMessage = "ERROR: Invalid Argument/Option - '%1'."
|
|
InvalidParameterErrorMessage = L_InvalidParameter1_ErrorMessage & vbCRLF & L_HelpSyntax2_Message
|
|
|
|
CONST L_InvalidFormat1_ErrorMessage = "ERROR: Invalid 'FORMAT' '%1' specified."
|
|
InvalidFormatErrorMessage = L_InvalidFormat1_ErrorMessage & vbCRLF & L_HelpSyntax2_Message
|
|
|
|
CONST L_InvalidRange_ErrorMessage = "ERROR: Invalid 'RANGE' '%1' specified."
|
|
CONST L_Invalid_ErrorMessage = "ERROR: Invalid '%1'."
|
|
CONST L_InvalidType_ErrorMessage = "ERROR: Invalid 'TYPE' '%1' specified for the 'FILTER' '%2'."
|
|
CONST L_InvalidUser_ErrorMessage = "ERROR: Invalid 'USER' '%1' specified for the 'FILTER '%2'."
|
|
CONST L_InvalidId_ErrorMessage = "ERROR: Invalid 'ID' '%1' specified for the 'FILTER' '%2'."
|
|
CONST L_InvalidFilter_ErrorMessage = "ERROR: Invalid 'FILTER' '%1' specified for the 'FILTER' '%2'."
|
|
CONST L_InvalidFilterFormat_ErrorMessage = "ERROR: The FILTER '%1' is not in the required format."
|
|
CONST L_InvalidFilterOperation_ErrorMessage = "ERROR: Invalid FILTER operator '%1' specified for the filter '%2'."
|
|
|
|
CONST L_InvalidCredentialsForServer1_ErrorMessage = "ERROR: Invalid Syntax. /U can be specified only when /S is specified."
|
|
InvalidCredentialsForServerErrorMessage = L_InvalidCredentialsForServer1_ErrorMessage & vbCRLF & L_HelpSyntax1_Message
|
|
|
|
CONST L_InvalidCredentialsForUser1_ErrorMessage = "ERROR: Invalid Syntax. /P can be specified only when /U is specified."
|
|
InvalidCredentialsForUserErrorMessage = L_InvalidCredentialsForUser1_ErrorMessage & vbCRLF & L_HelpSyntax1_Message
|
|
|
|
CONST L_InvalidOperator_ErrorMessage = "ERROR: Invalid operator specified for the range of dates in the 'DATETIME' filter."
|
|
CONST L_InvalidDateTimeFormat_ErrorMessage = "ERROR: Invalid 'DATETIME' format specified. Format:mm/dd/yy(yyyy),hh:mm:ssAM(/PM)"
|
|
|
|
CONST L_ExecuteQuery_ErrorMessage = "ERROR: Unable to execute the query for the '%1' log."
|
|
CONST L_LogDoesNotExist_ErrorMessage = "ERROR: The log file '%1' does not exist."
|
|
CONST L_InstancesFailed_ErrorMessage = "ERROR: Unable to get the log details from the system."
|
|
|
|
CONST L_InvalidSyntax1_ErrorMessage = "ERROR: Invalid Syntax."
|
|
InvalidSyntaxErrorMessage = L_InvalidSyntax1_ErrorMessage & vbCRLF & L_HelpSyntax1_Message
|
|
|
|
CONST L_InvalidInput1_ErrorMessage = "ERROR: Invalid input. Please check the input Values."
|
|
InvalidInputErrorMessage = L_InvalidInput1_ErrorMessage & vbCRLF & L_HelpSyntax1_Message
|
|
|
|
CONST L_ObjCreationFail_ErrorMessage = "ERROR: Unexpected Error , Query failed. "
|
|
CONST L_InfoUnableToInclude_ErrorMessage = "ERROR: Unable to include the common module""CmdLib.Wsc""."
|
|
CONST L_NoHeaderaNotApplicable_ErrorMessage = "ERROR: /NH option is allowed only for ""TABLE"" and ""CSV"" formats."
|
|
CONST L_InValidServerName_ErrorMessage = "ERROR: Invalid Syntax. System name cannot be empty."
|
|
CONST L_InValidUserName_ErrorMessage = "ERROR: Invalid Syntax. User name cannot be empty. "
|
|
|
|
CONST L_InvalidORSyntaxInFilter1_ErrorMessage = "ERROR: Invalid 'OR' operation is specified for the filter."
|
|
CONST L_InvalidORSyntaxInFilter2_ErrorMessage = "'OR' operation valid only for filters TYPE and ID."
|
|
InvalidORSyntaxInFilterErrorMessage = L_InvalidORSyntaxInFilter1_ErrorMessage & vbCRLF & L_InvalidORSyntaxInFilter2_ErrorMessage
|
|
|
|
CONST L_InvalidSyntaxMoreNoRepeated1_ErrorMessage = "ERROR: Invalid Syntax. '%1' option is not allowed more than 1 time(s)."
|
|
InvalidSyntaxMoreNoRepeatedErrorMessage = L_InvalidSyntaxMoreNoRepeated1_ErrorMessage & vbCRLF & L_HelpSyntax2_Message
|
|
|
|
' Hints given in case of errors
|
|
CONST L_HintCheckConnection_Message = "ERROR: Please check the system name, credentials and WBEM Core."
|
|
|
|
' Informational messages
|
|
CONST L_InfoNoRecordsInFilter_Message = "INFO: No records available for the '%1' log with the specified criteria."
|
|
CONST L_InfoNoRecords_Message = "INFO: No records available for the '%1' log."
|
|
CONST L_InfoNoLogsPresent_Message = "INFO: No logs are available in the system."
|
|
CONST L_InfoDisplayLog_Message = "Listing the events in '%1' log of host '%2'"
|
|
|
|
' Cscript usage strings
|
|
CONST L_UseCscript1_ErrorMessage = "This script should be executed from the Command Prompt using CSCRIPT.EXE."
|
|
CONST L_UseCscript2_ErrorMessage = "For example: CSCRIPT EVENTQUERY.vbs <arguments>"
|
|
CONST L_UseCscript3_ErrorMessage = "To set CScript as the default application to run .vbs files run the following"
|
|
CONST L_UseCscript4_ErrorMessage = " CSCRIPT //H:CSCRIPT //S"
|
|
CONST L_UseCscript5_ErrorMessage = "You can then run ""EVENTQUERY.vbs <arguments>"" without preceding the script with CSCRIPT."
|
|
|
|
' Contents for showing help for Usage
|
|
CONST L_ShowUsageLine00_Text = "No logs are available on this system for query."
|
|
CONST L_ShowUsageLine01_Text = "EVENTQUERY.vbs [/S system [/U username [/P password]]] [/FI filter]"
|
|
CONST L_ShowUsageLine02_Text = " [/FO format] [/R range] [/NH] [/V] [/L logname | *]"
|
|
CONST L_ShowUsageLine03_Text = "Description:"
|
|
CONST L_ShowUsageLine04_Text = " The EVENTQUERY.vbs script enables an administrator to list"
|
|
CONST L_ShowUsageLine05_Text = " the events and event properties from one or more event logs."
|
|
CONST L_ShowUsageLine06_Text = "Parameter List:"
|
|
CONST L_ShowUsageLine07_Text = " /S system Specifies the remote system to connect to."
|
|
CONST L_ShowUsageLine08_Text = " /U [domain\]user Specifies the user context under which the"
|
|
CONST L_ShowUsageLine09_Text = " command should execute."
|
|
CONST L_ShowUsageLine10_Text = " /P password Specifies the password for the given"
|
|
CONST L_ShowUsageLine11_Text = " user context."
|
|
CONST L_ShowUsageLine12_Text = " /V Specifies that the detailed information"
|
|
CONST L_ShowUsageLine13_Text = " should be displayed in the output."
|
|
CONST L_ShowUsageLine14_Text = " /FI filter Specifies the types of events to"
|
|
CONST L_ShowUsageLine15_Text = " filter in or out of the query."
|
|
CONST L_ShowUsageLine16_Text = " /FO format Specifies the format in which the output"
|
|
CONST L_ShowUsageLine17_Text = " is to be displayed."
|
|
CONST L_ShowUsageLine18_Text = " Valid formats are ""TABLE"", ""LIST"", ""CSV""."
|
|
CONST L_ShowUsageLine19_Text = " /R range Specifies the range of events to list."
|
|
CONST L_ShowUsageLine20_Text = " Valid Values are:"
|
|
CONST L_ShowUsageLine21_Text = " 'N' - Lists 'N' most recent events."
|
|
CONST L_ShowUsageLine22_Text = " '-N' - Lists 'N' oldest events."
|
|
CONST L_ShowUsageLine23_Text = " 'N1-N2' - Lists the events N1 to N2."
|
|
CONST L_ShowUsageLine24_Text = " /NH Specifies that the ""Column Header"" should"
|
|
CONST L_ShowUsageLine25_Text = " not be displayed in the output."
|
|
CONST L_ShowUsageLine26_Text = " Valid only for ""TABLE"" and ""CSV"" formats."
|
|
CONST L_ShowUsageLine27_Text = " /L logname Specifies the log(s) to query."
|
|
CONST L_ShowUsageLine28_Text = " /? Displays this help/usage."
|
|
CONST L_ShowUsageLine29_Text = " Valid Filters Operators allowed Valid Values"
|
|
CONST L_ShowUsageLine30_Text = " ------------- ------------------ ------------"
|
|
CONST L_ShowUsageLine31_Text = " DATETIME eq,ne,ge,le,gt,lt mm/dd/yy(yyyy),hh:mm:ssAM(/PM)"
|
|
CONST L_ShowUsageLine32_Text = " TYPE eq,ne ERROR, INFORMATION, WARNING,"
|
|
CONST L_ShowUsageLine33_Text = " SUCCESSAUDIT, FAILUREAUDIT"
|
|
CONST L_ShowUsageLine34_Text = " ID eq,ne,ge,le,gt,lt non-negative integer"
|
|
CONST L_ShowUsageLine35_Text = " USER eq,ne string"
|
|
CONST L_ShowUsageLine36_Text = " COMPUTER eq,ne string"
|
|
CONST L_ShowUsageLine37_Text = " SOURCE eq,ne string"
|
|
CONST L_ShowUsageLine38_Text = " CATEGORY eq,ne string"
|
|
CONST L_ShowUsageLine39_Text = "NOTE: Filter ""DATETIME"" can be specified as ""FromDate-ToDate"""
|
|
CONST L_ShowUsageLine40_Text = " Only ""eq"" operator can be used for this format."
|
|
CONST L_ShowUsageLine41_Text = "Examples:"
|
|
CONST L_ShowUsageLine42_Text = " EVENTQUERY.vbs "
|
|
CONST L_ShowUsageLine43_Text = " EVENTQUERY.vbs /L system "
|
|
CONST L_ShowUsageLine44_Text = " EVENTQUERY.vbs /S system /U user /P password /V /L *"
|
|
CONST L_ShowUsageLine45_Text = " EVENTQUERY.vbs /R 10 /L Application /NH"
|
|
CONST L_ShowUsageLine46_Text = " EVENTQUERY.vbs /R -10 /FO LIST /L Security"
|
|
CONST L_ShowUsageLine47_Text = " EVENTQUERY.vbs /R 5-10 /L ""DNS Server"""
|
|
CONST L_ShowUsageLine48_Text = " EVENTQUERY.vbs /FI ""Type eq Error"" /L Application"
|
|
CONST L_ShowUsageLine49_Text = " EVENTQUERY.vbs /L Application"
|
|
CONST L_ShowUsageLine50_Text = " /FI ""Datetime eq 06/25/00,03:15:00AM-06/25/00,03:15:00PM"""
|
|
CONST L_ShowUsageLine51_Text = " EVENTQUERY.vbs /FI ""Datetime gt 08/03/00,06:20:00PM"" "
|
|
CONST L_ShowUsageLine52_Text = " /FI ""Id gt 700"" /FI ""Type eq warning"" /L System"
|
|
CONST L_ShowUsageLine53_Text = " EVENTQUERY.vbs /FI ""Type eq error OR Id gt 1000 """
|
|
'-------------------------------------------------------------------------
|
|
' END of localization content
|
|
'-------------------------------------------------------------------------
|
|
|
|
' Define constants
|
|
CONST CONST_ERROR = 0
|
|
CONST CONST_CSCRIPT = 2
|
|
CONST CONST_SHOW_USAGE = 3
|
|
CONST CONST_PROCEED = 4
|
|
CONST CONST_ERROR_USAGE = 5
|
|
CONST CONST_NO_MATCHES_FOUND = 0
|
|
|
|
' Define the Exit Values
|
|
CONST EXIT_SUCCESS = 0
|
|
CONST EXIT_UNEXPECTED = 255
|
|
CONST EXIT_INVALID_INPUT = 254
|
|
CONST EXIT_METHOD_FAIL = 250
|
|
CONST EXIT_INVALID_PARAM = 999
|
|
CONST EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED = 777
|
|
|
|
' Define default values
|
|
CONST CONST_ARRAYBOUND_NUMBER = 10
|
|
CONST CONST_ID_NUMBER = 65535
|
|
|
|
' Define namespace and class names of wmi
|
|
CONST CONST_NAMESPACE_CIMV2 = "root\cimv2"
|
|
CONST CLASS_EVENTLOG_FILE = "Win32_NTEventlogFile"
|
|
|
|
' for blank line in help usage
|
|
CONST EmptyLine_Text = " "
|
|
|
|
' Define the various strings used in the script
|
|
'=============================================
|
|
' the valid options supported by the script
|
|
CONST OPTION_SERVER = "s"
|
|
CONST OPTION_USER = "u"
|
|
CONST OPTION_PASSWORD = "p"
|
|
CONST OPTION_FORMAT = "fo"
|
|
CONST OPTION_RANGE = "r"
|
|
CONST OPTION_NOHEADER = "nh"
|
|
CONST OPTION_VERBOSE = "v"
|
|
CONST OPTION_FILTER = "fi"
|
|
CONST OPTION_HELP = "?"
|
|
CONST OPTION_LOGNAME = "l"
|
|
|
|
' the property names on which the user given filters are applied
|
|
CONST FLD_FILTER_DATETIME = "TimeGenerated"
|
|
CONST FLD_FILTER_TYPE = "Type"
|
|
CONST FLD_FILTER_USER = "User"
|
|
CONST FLD_FILTER_COMPUTER = "ComputerName"
|
|
CONST FLD_FILTER_SOURCE = "SourceName"
|
|
CONST FLD_FILTER_CATEGORY = "CategoryString"
|
|
CONST FLD_FILTER_ID = "EventCode"
|
|
CONST FLD_FILTER_EVENTTYPE = "EventType"
|
|
|
|
' Define matching patterns used in validations
|
|
CONST PATTERNFORMAT = "^(table|list|csv)$"
|
|
CONST PATTERNTYPE = "^(ERROR|INFORMATION|WARNING|SUCCESSAUDIT|FAILUREAUDIT)$"
|
|
|
|
' Property values on which the user is given for the filter TYPE is applied
|
|
CONST PATTERNTYPE_ERROR = "ERROR"
|
|
CONST PATTERNTYPE_WARNING = "WARNING"
|
|
CONST PATTERNTYPE_INFORMATION = "INFORMATION"
|
|
CONST PATTERNTYPE_SUCCESSAUDIT = "SUCCESSAUDIT"
|
|
CONST PATTERNTYPE_FAILUREAUDIT = "FAILUREAUDIT"
|
|
|
|
CONST FLDFILTERTYPE_SUCCESSAUDIT = "audit success"
|
|
CONST FLDFILTERTYPE_FAILUREAUDIT = "audit failure"
|
|
|
|
' Define EventType
|
|
CONST EVENTTYPE_ERROR = "1"
|
|
CONST EVENTTYPE_WARNING = "2"
|
|
CONST EVENTTYPE_INFORMATION = "3"
|
|
CONST EVENTTYPE_SUCCESSAUDIT = "4"
|
|
CONST EVENTTYPE_FAILUREAUDIT = "5"
|
|
|
|
' the operator symbols
|
|
CONST SYMBOL_OPERATOR_EQ = "="
|
|
CONST SYMBOL_OPERATOR_NE = "<>"
|
|
CONST SYMBOL_OPERATOR_GE = ">="
|
|
CONST SYMBOL_OPERATOR_LE = "<="
|
|
CONST SYMBOL_OPERATOR_GT = ">"
|
|
CONST SYMBOL_OPERATOR_LT = "<"
|
|
|
|
' Define matching patterns used in validations
|
|
CONST PATTERN_RANGE = "^\d*-?\d+$"
|
|
CONST PATTERN_FILTER = "^([a-z]+)([\s]+)([a-z]+)([\s]+)([\w+]|[\W+]|\\)"
|
|
CONST PATTERN_DATETIME = "^\d{1,2}\/\d{1,2}\/\d{2,4},\d{1,2}:\d{1,2}:\d{1,2}(A|P)M$"
|
|
CONST PATTERN_INVALID_USER = "\|\[|\]|\:|\||\<|\>|\+|\=|\;|\,|\?|\*"
|
|
CONST PATTERN_ID = "^(\d+)$"
|
|
CONST PATTERN_DATETIME_RANGE = "^\d{1,2}\/\d{1,2}\/\d{2,4},\d{1,2}:\d{1,2}:\d{1,2}(A|P)M\-\d{1,2}\/\d{1,2}\/\d{2,4},\d{1,2}:\d{1,2}:\d{1,2}(A|P)M$"
|
|
|
|
' Define UNC format for server name
|
|
CONST UNC_Format_Servername = "\\"
|
|
|
|
' Define const for filter separation when OR is specified in filter
|
|
CONST L_OperatorOR_Text = " OR "
|
|
|
|
' Variable to trap local if already connection in wmiconnect function
|
|
Dim blnLocalConnection
|
|
|
|
blnLocalConnection = False 'defalut value
|
|
|
|
' to include the common module
|
|
Dim component ' object to store common module
|
|
|
|
Set component = CreateObject( "Microsoft.CmdLib" )
|
|
|
|
If Err.Number Then
|
|
WScript.Echo(L_InfoUnableToInclude_ErrorMessage)
|
|
WScript.Quit(EXIT_METHOD_FAIL)
|
|
End If
|
|
|
|
' referring the script host to common module
|
|
Set component.ScriptingHost = WScript.Application
|
|
|
|
' Check whether the script is run using CScript
|
|
If CInt( component.checkScript() ) <> CONST_CSCRIPT Then
|
|
WScript.Echo (UseCscriptErrorMessage)
|
|
WScript.Quit(EXIT_UNEXPECTED)
|
|
End If
|
|
|
|
' Calling the Main function
|
|
Call VBMain()
|
|
|
|
' end of the Main
|
|
Wscript.Quit(EXIT_SUCCESS)
|
|
|
|
|
|
'********************************************************************
|
|
'* Sub: VBMain
|
|
'*
|
|
'* Purpose: This is main function to starts execution
|
|
'*
|
|
'*
|
|
'* Input/ Output: None
|
|
'********************************************************************
|
|
Sub VBMain()
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.clear
|
|
|
|
' Declare variables
|
|
Dim intOpCode ' to check the operation asked for, Eg:Help etc
|
|
Dim strMachine ' the machine to query the events from
|
|
Dim strUserName ' the user name to use to query the machine
|
|
Dim strPassword ' the password for the user to query the machine
|
|
Dim strFormat ' format of display, default is table
|
|
Dim strRange ' to store the range of records specified
|
|
Dim blnNoHeader ' flag to store if header is not required
|
|
Dim blnVerboseDisplay ' flag to verify if verbose display is needed
|
|
ReDim arrFilters(5) ' to store all the given filters
|
|
Dim objLogs ' a object to store all the given logfles
|
|
|
|
' Initialize variables
|
|
intOpCode = 0
|
|
strFormat = L_ConstDefaultFormat_Text
|
|
strRange = ""
|
|
blnNoHeader = FALSE
|
|
blnVerboseDisplay = FALSE
|
|
|
|
Set objLogs = CreateObject("Scripting.Dictionary")
|
|
|
|
If Err.Number Then
|
|
WScript.Echo (L_ObjCreationFail_ErrorMessage)
|
|
WScript.Quit(EXIT_METHOD_FAIL)
|
|
End If
|
|
|
|
' setting Dictionary object compare mode to VBBinaryCompare
|
|
objLogs.CompareMode = VBBinaryCompare
|
|
|
|
' Parse the command line
|
|
intOpCode = intParseCmdLine(strMachine, _
|
|
strUserName, _
|
|
strPassword, _
|
|
arrFilters, _
|
|
strFormat, _
|
|
strRange, _
|
|
blnVerboseDisplay, _
|
|
blnNoHeader, _
|
|
objLogs)
|
|
|
|
If Err.number then
|
|
' error in parsing the Command line
|
|
component.vbPrintf InvalidInputErrorMessage ,Array(Ucase(Wscript.ScriptName))
|
|
WScript.Quit(EXIT_UNEXPECTED)
|
|
End If
|
|
|
|
' check the operation specified by the user
|
|
Select Case intOpCode
|
|
|
|
Case CONST_SHOW_USAGE
|
|
' help asked for
|
|
Call ShowUsage()
|
|
|
|
Case CONST_PROCEED
|
|
Call ShowEvents(strMachine, strUserName, strPassword, _
|
|
arrFilters, strFormat, strRange, _
|
|
blnVerboseDisplay, blnNoHeader, objLogs)
|
|
' completed successfully
|
|
WScript.Quit(EXIT_SUCCESS)
|
|
|
|
Case CONST_ERROR
|
|
' print common help message.
|
|
component.vbPrintf L_HelpSyntax1_Message, Array(Ucase(Wscript.ScriptName))
|
|
Wscript.Quit(EXIT_INVALID_INPUT)
|
|
|
|
Case CONST_ERROR_USAGE
|
|
' help is asked help with some other parameters
|
|
component.vbPrintf InvalidSyntaxErrorMessage, Array(Ucase(Wscript.ScriptName))
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
|
|
Case EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
'More no of times input values specified.message is captured at parser level so exit only with code.
|
|
Wscript.Quit(EXIT_INVALID_PARAM)
|
|
|
|
Case Else
|
|
'Invalid input values specified.
|
|
component.vbPrintf InvalidSyntaxErrorMessage, Array(Ucase(Wscript.ScriptName))
|
|
Wscript.Quit(EXIT_INVALID_PARAM)
|
|
|
|
End Select
|
|
|
|
End Sub
|
|
'*************************** End of Main **************************
|
|
|
|
'********************************************************************
|
|
'* Function: intParseCmdLine
|
|
'*
|
|
'* Purpose: Parses the command line arguments to the variables
|
|
'*
|
|
'* Input:
|
|
'* [out] strMachine machine to query events from
|
|
'* [out] strUserName user name to connect to the machine
|
|
'* [out] strPassword password for the user
|
|
'* [out] arrFilters the array containing the filters
|
|
'* [out] strFormat the display format
|
|
'* [out] strRange the range of records required
|
|
'* [out] blnVerboseDisplay flag to verify if verbose display is needed
|
|
'* [out] blnNoHeader flag to verify if noheader display is needed
|
|
'* [out] objLogs to store all the given logfles
|
|
'* Output: Returns CONST_PROCEED, CONST_SHOW_USAGE or CONST_ERROR
|
|
'* Displays error message and quits if invalid option is asked
|
|
'*
|
|
'********************************************************************
|
|
Private Function intParseCmdLine( ByRef strMachine, _
|
|
ByRef strUserName, _
|
|
ByRef strPassword, _
|
|
ByRef arrFilters, _
|
|
ByRef strFormat, _
|
|
ByRef strRange, _
|
|
ByRef blnVerboseDisplay, _
|
|
ByRef blnNoHeader,_
|
|
ByRef objLogs)
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Dim strUserGivenArg ' to temporarily store the user given arguments to script
|
|
Dim strTemp ' to store temporary values
|
|
Dim intArgIter ' to count the number of arguments given by user
|
|
Dim intArgLogType ' to count number of log files specified - Used in ReDim
|
|
Dim intFilterCount ' to count number of filters specified - Used in ReDim
|
|
|
|
Dim blnHelp ' to check if already Help is specified
|
|
Dim blnFormat ' to check if already Format is specified
|
|
Dim blnRange ' to check if already Range is specified
|
|
Dim blnServer ' to check if already Server is specified
|
|
Dim blnPassword ' to check if already Password is specified
|
|
Dim blnUser ' to check if already User is specified
|
|
|
|
strUserGivenArg = ""
|
|
intArgLogType = 0
|
|
intFilterCount = 0
|
|
intArgIter = 0
|
|
|
|
'default values
|
|
blnHelp = False
|
|
blnPassword = False
|
|
blnUser = False
|
|
blnServer = False
|
|
blnFormat = False
|
|
|
|
|
|
' Retrieve the command line and set appropriate variables
|
|
Do While intArgIter <= Wscript.arguments.Count - 1
|
|
strUserGivenArg = Wscript.arguments.Item(intArgIter)
|
|
|
|
IF Left( strUserGivenArg,1) = "/" OR Left( strUserGivenArg,1) = "-" Then
|
|
strUserGivenArg = Right( strUserGivenArg,Len(strUserGivenArg) -1 )
|
|
|
|
Select Case LCase(strUserGivenArg)
|
|
Case LCase(OPTION_SERVER)
|
|
|
|
'If more than 1 time(s) is spcecified
|
|
If blnServer =True Then
|
|
component.vbPrintf InvalidSyntaxMoreNoRepeatedErrorMessage, Array(Wscript.arguments.Item(intArgIter), Ucase(Wscript.ScriptName))
|
|
intParseCmdLine = EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
Exit Function
|
|
End If
|
|
|
|
If Not component.getArguments(L_MachineName_Text, strMachine, intArgIter, FALSE) Then
|
|
intParseCmdLine = CONST_ERROR
|
|
Exit Function
|
|
End If
|
|
|
|
blnServer =True
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_USER)
|
|
|
|
'If more than 1 time(s) is spcecified
|
|
If blnUser =True Then
|
|
component.vbPrintf InvalidSyntaxMoreNoRepeatedErrorMessage, Array(Wscript.arguments.Item(intArgIter), Ucase(Wscript.ScriptName))
|
|
intParseCmdLine = EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
Exit Function
|
|
End If
|
|
|
|
If Not component.getArguments(L_UserName_Text, strUserName, intArgIter, FALSE) Then
|
|
intParseCmdLine = CONST_ERROR
|
|
Exit Function
|
|
End If
|
|
|
|
blnUser =True
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_PASSWORD)
|
|
|
|
'If more than 1 time(s) is spcecified
|
|
If blnPassword =True Then
|
|
component.vbPrintf InvalidSyntaxMoreNoRepeatedErrorMessage, Array(Wscript.arguments.Item(intArgIter), Ucase(Wscript.ScriptName))
|
|
intParseCmdLine = EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
Exit Function
|
|
End If
|
|
|
|
If Not component.getArguments(L_UserPassword_Text, strPassword, intArgIter, FALSE) Then
|
|
intParseCmdLine = CONST_ERROR
|
|
Exit Function
|
|
End If
|
|
|
|
blnPassword =True
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_FORMAT)
|
|
|
|
'If more than 1 time(s) is spcecified
|
|
If blnFormat =True Then
|
|
component.vbPrintf InvalidSyntaxMoreNoRepeatedErrorMessage, Array(Wscript.arguments.Item(intArgIter), Ucase(Wscript.ScriptName))
|
|
intParseCmdLine = EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
Exit Function
|
|
End If
|
|
|
|
If Not component.getArguments(L_Format_Text,strFormat, intArgIter, FALSE) Then
|
|
intParseCmdLine = CONST_ERROR
|
|
Exit Function
|
|
End If
|
|
|
|
|
|
blnFormat =True
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_RANGE)
|
|
|
|
'If more than 1 time(s) is spcecified
|
|
If blnRange =True Then
|
|
component.vbPrintf InvalidSyntaxMoreNoRepeatedErrorMessage, Array(Wscript.arguments.Item(intArgIter), Ucase(Wscript.ScriptName))
|
|
intParseCmdLine = EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
Exit Function
|
|
End If
|
|
|
|
If Not component.getArguments(L_Range_Text,strRange, intArgIter,TRUE) Then
|
|
intParseCmdLine = CONST_ERROR
|
|
Exit Function
|
|
End If
|
|
|
|
blnRange =True
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_NOHEADER)
|
|
|
|
'If more than 1 time(s) is spcecified
|
|
If blnNoHeader =True Then
|
|
component.vbPrintf InvalidSyntaxMoreNoRepeatedErrorMessage, Array(Wscript.arguments.Item(intArgIter), Ucase(Wscript.ScriptName))
|
|
intParseCmdLine = EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
Exit Function
|
|
End If
|
|
|
|
blnNoHeader = TRUE
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_VERBOSE)
|
|
|
|
'If more than 1 time(s) is spcecified
|
|
If blnVerboseDisplay =True Then
|
|
component.vbPrintf InvalidSyntaxMoreNoRepeatedErrorMessage, Array(Wscript.arguments.Item(intArgIter), Ucase(Wscript.ScriptName))
|
|
intParseCmdLine = EXIT_INVALID_PARAM_DEFAULT_OPTION_REPEATED
|
|
Exit Function
|
|
End If
|
|
|
|
blnVerboseDisplay = TRUE
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_FILTER)
|
|
|
|
If Not component.getArguments(L_Filter_Text, strTemp, intArgIter, FALSE) Then
|
|
intParseCmdLine = CONST_ERROR
|
|
Exit Function
|
|
End If
|
|
|
|
arrFilters(intFilterCount) = strTemp
|
|
intFilterCount = intFilterCount + 1
|
|
intArgIter = intArgIter + 1
|
|
|
|
If ((intFilterCount MOD 5) = 0) Then
|
|
ReDim PRESERVE arrFilters(intFilterCount + 5)
|
|
End If
|
|
|
|
Case LCase(OPTION_HELP)
|
|
|
|
If blnHelp =True then
|
|
intParseCmdLine = EXIT_INVALID_PARAM
|
|
Exit Function
|
|
End If
|
|
|
|
blnHelp =True
|
|
intParseCmdLine = CONST_SHOW_USAGE
|
|
intArgIter = intArgIter + 1
|
|
|
|
Case LCase(OPTION_LOGNAME)
|
|
If Not component.getArguments(L_Log_Text, strTemp, intArgIter, FALSE) Then
|
|
intParseCmdLine = CONST_ERROR
|
|
Exit Function
|
|
Else
|
|
If NOT objLogs.Exists(LCase(strTemp)) Then
|
|
objLogs.Add LCase(strTemp), -1
|
|
End If
|
|
intArgIter = intArgIter + 1
|
|
End if
|
|
|
|
Case Else
|
|
' invalid switch specified
|
|
component.vbPrintf InvalidParameterErrorMessage, Array(Wscript.arguments.Item(intArgIter),Ucase(Wscript.ScriptName))
|
|
Wscript.Quit(EXIT_INVALID_INPUT)
|
|
|
|
End Select
|
|
Else
|
|
' invalid argument specified
|
|
component.vbPrintf InvalidParameterErrorMessage, Array(Wscript.arguments.Item(intArgIter),Ucase(Wscript.ScriptName))
|
|
Wscript.Quit(EXIT_INVALID_INPUT)
|
|
End IF
|
|
|
|
Loop '** intArgIter <= Wscript.arguments.Count - 1
|
|
|
|
' preserving the array with current dimension
|
|
ReDim PRESERVE arrFilters(intFilterCount-1)
|
|
|
|
' if no logs specified for query
|
|
If (ObjLogs.Count = 0 ) Then
|
|
ObjLogs.Add "*", -1
|
|
End If
|
|
|
|
' check for invalid usage of help
|
|
If blnHelp and intArgIter > 1 Then
|
|
intParseCmdLine = CONST_ERROR_USAGE
|
|
Exit Function
|
|
End If
|
|
|
|
'check with default case : no arguments specified
|
|
If IsEmpty(intParseCmdLine) Then
|
|
intParseCmdLine = CONST_PROCEED
|
|
End If
|
|
|
|
End Function
|
|
|
|
'********************************************************************
|
|
'* Function: ValidateArguments
|
|
'*
|
|
'* Purpose: Validates the command line arguments given by the user
|
|
'*
|
|
'* Input:
|
|
'* [in] strMachine machine to query events from
|
|
'* [in] strUserName user name to connect to the machine
|
|
'* [in] strPassword password for the user
|
|
'* [in] strFormat the display format
|
|
'* [in] strRange the range of records required
|
|
'* [in] blnNoHeader flag to verify if noheader display is needed
|
|
'* [out] arrFilters the array containing the filters
|
|
'*
|
|
'* Output: Returns true if all valid else displays error message and quits
|
|
'* Gets the password from the user if not specified along with User.
|
|
'*
|
|
'********************************************************************
|
|
Private Function ValidateArguments (ByVal strMachine, _
|
|
ByVal strUserName, _
|
|
ByVal strPassword, _
|
|
ByRef arrFilters, _
|
|
ByVal strFormat, _
|
|
ByVal strRange,_
|
|
ByVal blnNoHeader)
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Dim arrTemp ' to store temporary array values
|
|
|
|
' Check if invalid Server name is given
|
|
If NOT ISEMPTY(strMachine) THEN
|
|
If Trim(strMachine) = vbNullString Then
|
|
WScript.Echo (L_InValidServerName_ErrorMessage)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
End If
|
|
|
|
'Check if invalid User name is given
|
|
If NOT ISEMPTY(strUserName) THEN
|
|
If Trim(strUserName) = vbNullString Then
|
|
WScript.Echo (L_InValidUserName_ErrorMessage )
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
End If
|
|
|
|
' ERROR if user is given without machine OR
|
|
' password is given without user
|
|
If ((strUserName <> VBEmpty) AND (strMachine = VBEmpty)) Then
|
|
component.vbPrintf InvalidCredentialsForServerErrorMessage, Array(Ucase(Wscript.ScriptName))
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
ElseIf ((strPassword <> VBEmpty) AND (strUserName = VBEmpty))Then
|
|
component.vbPrintf InvalidCredentialsForUserErrorMessage, Array(Ucase(Wscript.ScriptName))
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
' only table, list and csv display formats allowed
|
|
' PATTERNFORMAT '"^(table|list|csv)$"
|
|
|
|
If CInt(component.matchPattern(PATTERNFORMAT,strFormat)) = CONST_NO_MATCHES_FOUND Then
|
|
component.vbPrintf InvalidFormatErrormessage, Array(strFormat ,Ucase(Wscript.ScriptName))
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
' check : -n header is specified for format of 'LIST' option
|
|
If blnNoHeader =True and Lcase(strFormat) = Lcase(L_Const_List_Format_Text) then
|
|
WScript.Echo (L_NoHeaderaNotApplicable_ErrorMessage)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
If Len(Trim(strRange)) > 0 Then
|
|
' range is specified, valid formats are N, -N or N1-N2
|
|
' PATTERN_RANGE '"^(\d+|\-\d+|\d+\-\d+)$"
|
|
If CInt(component.matchPattern(PATTERN_RANGE, strRange)) = CONST_NO_MATCHES_FOUND Then
|
|
component.vbPrintf L_InvalidRange_ErrorMessage, Array(strRange)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
Else
|
|
|
|
strRange = CLng(Abs(strRange))
|
|
|
|
'this err an be trappped when N1-N2 option is given
|
|
If Err.Number Then
|
|
arrTemp = split(strRange, "-", 2, VBBinaryCompare)
|
|
If CLng(arrTemp(0)) => CLng(arrTemp(1)) Then
|
|
' invalid range
|
|
component.vbPrintf L_InvalidRange_ErrorMessage, Array(strRange)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
Err.Clear 'if no invalid range N1-N2 clear the error
|
|
Else
|
|
If Abs(strRange) = 0 Then
|
|
component.vbPrintf L_InvalidRange_ErrorMessage, Array(strRange)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
End If
|
|
End If
|
|
End If
|
|
|
|
ValidateArguments = TRUE
|
|
End Function
|
|
|
|
'********************************************************************
|
|
'* Function: ValidateFilters
|
|
'*
|
|
'* Purpose: Validates the filters given by the user.
|
|
'*
|
|
'* Input: [in] Objservice the service object
|
|
'* Input: [out] arrFilters the array containing the filters
|
|
'*
|
|
'* Output: If filter is invalid, displays error message and quits
|
|
'* If valid, filter is prepared for the query and returns true
|
|
'*
|
|
'********************************************************************
|
|
Private Function ValidateFilters(ByRef arrFilters ,ByVal ObjService)
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Dim j ' to use in the loop
|
|
Dim strFilter ' to store the user given filter (Eg:"Type eq Error")
|
|
Dim arrTempProp ' to store the temporary array filterproperty
|
|
Dim arrTempOperAndVal ' to store the temporary array filteroperator and filtervalue
|
|
Dim strTemp ' to store temporary values
|
|
Dim arrTemp ' to store temporary values of datetime when Range is given (Date1-Date2)
|
|
Dim strFilterProperty ' the filter criteria that is specified (Eg:Type, ID)
|
|
Dim strFilterOperation ' the operation specified (Eg: eq, gt)
|
|
Dim strFilterValue ' the filter value specified
|
|
|
|
Dim objInstance ' to refer to the instances of the objEnumerator
|
|
Dim objEnumerator ' to store the results of the query is executed
|
|
Dim strTempQuery ' string to make query
|
|
Dim strTimeZone ' to store the TimeZone of the Queried system
|
|
Dim strSign ' to store "+|-" sign value of TimeZone
|
|
|
|
' validate each filter stored in the array
|
|
For j = 0 to UBound(arrFilters)
|
|
strFilter = arrFilters(j)
|
|
|
|
'check eigther "OR" is pesent inthe filter value
|
|
'Example : "type eq warning " OR " type eq error" [to support ORing in Filter Switch]
|
|
'Make a flag in this case "blnOR" present/not
|
|
'split it by "OR" SEND as No. of Array elements
|
|
Dim blnOR 'boolean to refer 'OR' operation is specified
|
|
Dim strArrFilter 'string to store array of filters if OR is specified
|
|
|
|
blnOR=False 'Initialise to False
|
|
|
|
|
|
If UBOUND(Split(LCase(strFilter),LCase(L_OperatorOR_Text)) ) > 0 Then
|
|
'setting the flag if " OR " specified in filter
|
|
blnOR =TRUE
|
|
|
|
'split with "OR"
|
|
strArrFilter = Split(LCase(strFilter),LCase(L_OperatorOR_Text))
|
|
Else
|
|
'make single dimention array UBOUND = 0
|
|
strArrFilter = Array(strFilter)
|
|
End If
|
|
|
|
Dim k ' to use in the loop
|
|
Dim strTempFilter ' used to format Query string
|
|
|
|
'process the array for validatation
|
|
'UBOUND = 0 say normal filter specified
|
|
For k = 0 to UBound(strArrFilter)
|
|
|
|
If UBound(strArrFilter) > 0 then
|
|
strFilter =strArrFilter(k)
|
|
Else
|
|
'this is the first element allways
|
|
strFilter =strArrFilter(0)
|
|
End If
|
|
|
|
' check if 3 parameters are passed as input to filter
|
|
' PATTERN_FILTER "^([a-z]+)([\s]+)([a-z]+)([\s]+)(\w+)"
|
|
|
|
strFilter = Trim( strFilter ) ' trim the value
|
|
If CInt(component.matchPattern(PATTERN_FILTER, strFilter)) <= 0 Then
|
|
component.vbPrintf L_InvalidFilterFormat_ErrorMessage, Array(strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
|
|
' This to eliminate any no.of blank Char(s) between three valid input values
|
|
' i.e..filter "property ---operation ----value"
|
|
' first SPLIT the space delimiter string into array size of 2.
|
|
' and get the property value
|
|
arrTempProp = split(Trim(strFilter)," ",2,VBBinaryCompare)
|
|
strFilterProperty = arrTempProp(0)
|
|
|
|
' now trim it and again SPLIT the second element of arrTempProp into an array of size 2.
|
|
' and get the operation and value
|
|
arrTempOperAndVal = split(Trim(arrTempProp(1))," ",2,VBBinaryCompare)
|
|
strFilterOperation = arrTempOperAndVal(0)
|
|
strFilterValue = Ltrim(arrTempOperAndVal(1))
|
|
|
|
If LCase(strFilterProperty) = LCase(L_UserFilterDateTime_Text) OR _
|
|
LCase(strFilterProperty) = LCase(L_UserFilterId_Text) Then
|
|
' the following are valid operators
|
|
If LCase(strFilterOperation) = LCase(L_OperatorEq_Text) OR _
|
|
LCase(strFilterOperation) = LCase(L_OperatorNe_Text) OR _
|
|
LCase(strFilterOperation) = LCase(L_OperatorGe_Text) OR _
|
|
LCase(strFilterOperation) = LCase(L_OperatorLe_Text) OR _
|
|
LCase(strFilterOperation) = LCase(L_OperatorGt_Text) OR _
|
|
LCase(strFilterOperation) = LCase(L_OperatorLt_Text) Then
|
|
|
|
strTemp = ReplaceOperators(strFilterOperation)
|
|
strFilterOperation = strTemp
|
|
Else
|
|
component.vbPrintf L_InvalidFilterOperation_ErrorMessage, Array(strFilterOperation, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
ElseIf LCase(strFilterProperty) = LCase(L_UserFilterType_Text) OR _
|
|
LCase(strFilterProperty) = LCase(L_UserFilterUser_Text) OR _
|
|
LCase(strFilterProperty) = LCase(L_UserFilterComputer_Text) OR _
|
|
LCase(strFilterProperty) = LCase(L_UserFilterSource_Text) OR _
|
|
LCase(strFilterProperty) = LCase(L_UserFilterDateCategory_Text) Then
|
|
' for others, only these two operators are valid
|
|
If LCase(strFilterOperation) = LCase(L_OperatorEq_Text) OR _
|
|
LCase(strFilterOperation) = LCase(L_OperatorNe_Text) Then
|
|
|
|
strTemp = ReplaceOperators(strFilterOperation)
|
|
strFilterOperation = strTemp
|
|
Else
|
|
component.vbPrintf L_InvalidFilterOperation_ErrorMessage, _
|
|
Array(strFilterOperation, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
Else
|
|
component.vbPrintf L_InvalidFilterOperation_ErrorMessage, _
|
|
Array(strFilterProperty, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
' validate the filter asked for
|
|
Select Case LCase(strFilterProperty)
|
|
|
|
Case L_UserFilterDateTime_Text
|
|
|
|
'Checking " OR " is only supported property EQ "TYPE OR ID" only
|
|
If blnOR = True then
|
|
WScript.Echo InvalidORSyntaxInFilterErrorMessage
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
' Here To find Time Zone of system from CLASS_TIMEZONE_FILE
|
|
strTempQuery = "SELECT * FROM Win32_OperatingSystem "
|
|
|
|
Set objEnumerator = objService.ExecQuery(strTempQuery,,0)
|
|
|
|
' getting the Time Zone
|
|
For each objInstance in objEnumerator
|
|
strTimeZone = objInstance.CurrentTimeZone
|
|
Next
|
|
|
|
'here to format timeZome value as '+/-' UUU
|
|
|
|
If Isnull(strTimeZone) or IsEmpty(strTimeZone)then
|
|
strTimeZone =0
|
|
End If
|
|
|
|
'default sign
|
|
strSign ="+"
|
|
|
|
IF strTimeZone < 0 THEN
|
|
strSign ="-"
|
|
End If
|
|
|
|
If Len(strTimeZone) < 4 then
|
|
If Len(strTimeZone) = 3 then
|
|
If strTimeZone < 0 then
|
|
strTimeZone = Replace(strTimeZone,"-","0")
|
|
End If
|
|
ElseIf Len(strTimeZone) = 2 then
|
|
If strTimeZone < 0 then
|
|
strTimeZone = Replace(strTimeZone,"-","00")
|
|
Else
|
|
strTimeZone = "0" & strTimeZone
|
|
End If
|
|
ElseIf Len(strTimeZone) = 1 then
|
|
IF strTimeZone >= 0 Then
|
|
strTimeZone = "00" & strTimeZone
|
|
End if
|
|
End If
|
|
'return to a format as "+|-" & UUU
|
|
strTimeZone= strSign & strTimeZone
|
|
End If
|
|
|
|
' check for the valid format - mm/dd/yy,hh:mm:ssPM
|
|
' PATTERN_DATETIME
|
|
If CInt(component.matchPattern(PATTERN_DATETIME, strFilterValue)) > 0 Then
|
|
If component.validateDateTime(strFilterValue) Then
|
|
' a valid datetime filter. Prepare for query
|
|
strFilterProperty = FLD_FILTER_DATETIME
|
|
strTemp = component.changeToWMIDateTime(strFilterValue,strTimeZone)
|
|
' Format the input
|
|
' TimeGenerated > "07/25/2000 10:12:00 PM"
|
|
strFilterValue = Chr(34) & strTemp & Chr(34)
|
|
End If
|
|
Else
|
|
' match for range of dates in the format
|
|
' mm/dd/yy,hh:mm:ssPM - mm/dd/yy,hh:mm:ssAM
|
|
' PATTERN_DATETIME_RANGE
|
|
|
|
If CInt(component.matchPattern(PATTERN_DATETIME_RANGE, strFilterValue)) > 0 Then
|
|
strFilterProperty = FLD_FILTER_DATETIME
|
|
' Only = operation supported in this format
|
|
If strFilterOperation <> "=" Then
|
|
WScript.Echo (L_InvalidOperator_ErrorMessage)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
arrTemp = split(strFilterValue,"-",2,VBBinaryCompare)
|
|
|
|
If component.validateDateTime(arrTemp(0)) Then
|
|
' a valid datetime filter. Prepare for query
|
|
strTemp = component.changeToWMIDateTime(arrTemp(0),strTimeZone)
|
|
' Format the input
|
|
' TimeGenerated > "07/25/2000 10:12:00 PM"
|
|
strFilterOperation = ">="
|
|
strFilterValue = Chr(34) & strTemp & Chr(34)
|
|
|
|
If component.validateDateTime(arrTemp(1)) Then
|
|
' a valid datetime filter. Prepare for query
|
|
strTemp = component.changeToWMIDateTime(arrTemp(1),strTimeZone)
|
|
' Format the input
|
|
' TimeGenerated > "07/25/2000 10:12:00 PM"
|
|
strFilterValue = strFilterValue & _
|
|
" AND " & strFilterProperty & "<="& Chr(34)_
|
|
& strTemp & Chr(34)
|
|
End If
|
|
End If
|
|
Else
|
|
component.vbPrintf L_InvalidDateTimeFormat_ErrorMessage, Array(strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
End If
|
|
|
|
Case L_UserFilterType_Text
|
|
|
|
' the following values are only valid for the "Type" filter
|
|
' Valid: ERROR|INFORMATION|WARNING|SUCCESSAUDIT|FAILUREAUDIT
|
|
' PATTERNTYPE
|
|
|
|
If CInt(component.matchPattern(PATTERNTYPE, strFilterValue)) = _
|
|
CONST_NO_MATCHES_FOUND Then
|
|
component.vbPrintf L_InvalidType_ErrorMessage, Array(strFilterValue, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
Else
|
|
' here i need to check WINXP or not
|
|
If ( IsWinXP ( ObjService) = TRUE ) Then
|
|
|
|
' a valid type filter. Prepare for query
|
|
If LCase(strFilterValue) =LCase(PATTERNTYPE_ERROR) Then
|
|
strFilterValue = EVENTTYPE_ERROR
|
|
ElseIf LCase(strFilterValue) =LCase(PATTERNTYPE_WARNING) Then
|
|
strFilterValue = EVENTTYPE_WARNING
|
|
ElseIf LCase(strFilterValue) =LCase(PATTERNTYPE_INFORMATION) Then
|
|
strFilterValue = EVENTTYPE_INFORMATION
|
|
ElseIf LCase(strFilterValue) =LCase(PATTERNTYPE_SUCCESSAUDIT) Then
|
|
strFilterValue = EVENTTYPE_SUCCESSAUDIT
|
|
ElseIf LCase(strFilterValue) =LCase(PATTERNTYPE_FAILUREAUDIT) Then
|
|
strFilterValue = EVENTTYPE_FAILUREAUDIT
|
|
End If
|
|
|
|
' a valid type filter. Prepare for query
|
|
strFilterProperty = FLD_FILTER_EVENTTYPE
|
|
Else
|
|
' a valid type filter. Prepare for query
|
|
If LCase(strFilterValue) =LCase(PATTERNTYPE_SUCCESSAUDIT) Then
|
|
strFilterValue = FLDFILTERTYPE_SUCCESSAUDIT
|
|
ElseIf LCase(strFilterValue) =LCase(PATTERNTYPE_FAILUREAUDIT) Then
|
|
strFilterValue = FLDFILTERTYPE_FAILUREAUDIT
|
|
End If
|
|
|
|
' a valid type filter. Prepare for query
|
|
strFilterProperty = FLD_FILTER_TYPE
|
|
|
|
End If
|
|
|
|
End If
|
|
|
|
Case L_UserFilterUser_Text
|
|
|
|
'Checking " OR " is only supported property EQ "TYPE OR ID" only
|
|
If blnOR = True then
|
|
WScript.Echo InvalidORSyntaxInFilterErrorMessage
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
' these are invalid characters for a user name
|
|
' PATTERN_INVALID_USER
|
|
|
|
If CInt(component.matchPattern(PATTERN_INVALID_USER, strFilterValue)) > 0 Then
|
|
component.vbPrintf L_InvalidUser_ErrorMessage , Array(strFilterValue, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
Else
|
|
|
|
' a valid user filter. Prepare for query
|
|
If InStr(1, strFilterValue, "\", VBBinaryCompare) Then
|
|
strFilterValue = Replace(strFilterValue, "\","\\")
|
|
End If
|
|
|
|
If LCase(strFilterValue) =LCase(L_TextNa_Text) Then
|
|
strFilterValue = Null
|
|
End If
|
|
|
|
End If
|
|
strFilterProperty = FLD_FILTER_USER
|
|
|
|
Case L_UserFilterComputer_Text
|
|
' a valid computer filter. Prepare for query
|
|
strFilterProperty = FLD_FILTER_COMPUTER
|
|
|
|
'Checking " OR " is only supported property EQ "TYPE OR ID" only
|
|
If blnOR = True then
|
|
WScript.Echo InvalidORSyntaxInFilterErrorMessage
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
Case L_UserFilterSource_Text
|
|
' a valid Source filter. Prepare for query
|
|
strFilterProperty = FLD_FILTER_SOURCE
|
|
|
|
'Checking " OR " is only supported property EQ "TYPE OR ID" only
|
|
If blnOR = True then
|
|
WScript.Echo InvalidORSyntaxInFilterErrorMessage
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
Case L_UserFilterDateCategory_Text
|
|
|
|
'Checking " OR " is only supported property EQ "TYPE OR ID" only
|
|
If blnOR = True then
|
|
WScript.Echo InvalidORSyntaxInFilterErrorMessage
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
' a valid Category filter. Prepare for query
|
|
If LCase(strFilterValue) =LCase(L_TextNone_Text) Then
|
|
strFilterValue = Null
|
|
End If
|
|
|
|
strFilterProperty = FLD_FILTER_CATEGORY
|
|
|
|
Case L_UserFilterId_Text
|
|
' check if the given id is a number
|
|
' PATTERN_ID '"^(\d+)$"
|
|
If CInt(component.matchPattern(PATTERN_ID, strFilterValue)) = CONST_NO_MATCHES_FOUND Then
|
|
component.vbPrintf L_InvalidId_ErrorMessage, Array(strFilterValue, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
Else
|
|
' Invalid ID Number validation
|
|
If ( Clng(strFilterValue) > CONST_ID_NUMBER )Then
|
|
component.vbPrintf L_InvalidId_ErrorMessage, Array(strFilterValue, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
|
|
' a valid id filter. Prepare for query
|
|
strFilterProperty = FLD_FILTER_ID
|
|
End If
|
|
|
|
Case Else
|
|
' invalid filter specified
|
|
component.vbPrintf L_InvalidFilter_ErrorMessage, Array(strFilterProperty, strFilter)
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End Select
|
|
|
|
If LCase(strFilterProperty) = LCase(FLD_FILTER_DATETIME) OR IsNull(strFilterValue) Then
|
|
' This is to handle NULL Property values i.e for category ,type
|
|
If IsNull(strFilterValue) Then
|
|
strFilter = strFilterProperty & strFilterOperation & strFilterValue & "Null"
|
|
Else
|
|
strFilter = strFilterProperty & strFilterOperation & strFilterValue
|
|
End If
|
|
|
|
Else
|
|
strFilter = strFilterProperty & _
|
|
strFilterOperation & Chr(34) & strFilterValue & Chr(34)
|
|
End If
|
|
|
|
'Binding the string with "OR" to Prepare for query if blnOR is true
|
|
If blnOR =TRUE Then
|
|
|
|
If k = 0 then
|
|
strTempFilter = strFilter
|
|
Else
|
|
strTempFilter = strTempFilter & " OR " & strFilter
|
|
End If
|
|
|
|
End If
|
|
|
|
Next
|
|
|
|
'Set again making single filter string element if blnOR is TRUE
|
|
If blnOR =TRUE Then
|
|
'this "()" Add the order of precedence of operation is SQL
|
|
strFilter = "( " & strTempFilter & ")"
|
|
End If
|
|
|
|
'Here setting filter to main array
|
|
arrFilters(j) = strFilter
|
|
|
|
|
|
Next
|
|
|
|
ValidateFilters = TRUE
|
|
|
|
End Function
|
|
|
|
'********************************************************************
|
|
'* Function: ReplaceOperators
|
|
'*
|
|
'* Purpose: Replaces the operator in string form with its symbol
|
|
'*
|
|
'* Input:
|
|
'* [in] strFilterOperation the operation
|
|
'*
|
|
'* Output: Returns the symbolic operator
|
|
'* If invalid operator, displays error message and quits
|
|
'*
|
|
'********************************************************************
|
|
Private Function ReplaceOperators(ByVal strFilterOperation)
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Select Case LCase(strFilterOperation)
|
|
|
|
Case L_OperatorEq_Text
|
|
ReplaceOperators = SYMBOL_OPERATOR_EQ
|
|
|
|
Case L_OperatorNe_Text
|
|
ReplaceOperators = SYMBOL_OPERATOR_NE
|
|
|
|
Case L_OperatorGe_Text
|
|
ReplaceOperators = SYMBOL_OPERATOR_GE
|
|
|
|
Case L_OperatorLe_Text
|
|
ReplaceOperators = SYMBOL_OPERATOR_LE
|
|
|
|
Case L_OperatorGt_Text
|
|
ReplaceOperators = SYMBOL_OPERATOR_GT
|
|
|
|
Case L_OperatorLt_Text
|
|
ReplaceOperators = SYMBOL_OPERATOR_LT
|
|
|
|
Case Else
|
|
' not a valid operator
|
|
component.vbPrintf L_Invalid_ErrorMessage, Array(strFilterOperation)
|
|
WScript.Quit(EXIT_INVALID_PARAM)
|
|
End Select
|
|
End Function
|
|
'********************************************************************
|
|
'* Sub : VerifyLogAndGetMaxRecords
|
|
'*
|
|
'* Purpose: populates the output array with count of records in given input array
|
|
'*
|
|
'* Input: [in] objService the service object
|
|
'* [out] objLogs the object containing the logs & max count of records corresponding log
|
|
'*
|
|
'* Output: array's are populates with logfile names and its count of max records
|
|
'*
|
|
'********************************************************************
|
|
Private Sub VerifyLogAndGetMaxRecords(ByVal objService, _
|
|
ByRef objLogs)
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Dim strTempQuery ' string to make query
|
|
Dim objEnumerator ' to get the collection object after query
|
|
Dim objInstance ' to refer to each instance of the results got
|
|
Dim i ' for initialing loop
|
|
Dim strLogFile ' used to store log file inside loop
|
|
Dim arrKeyName ' used to store key value of Dictionary object for processing loop
|
|
|
|
arrKeyName = objLogs.Keys
|
|
|
|
For i = 0 to objLogs.Count -1
|
|
strLogFile = arrKeyName(i)
|
|
If Not strLogFile = "*" Then
|
|
' Check if log file exists, by querying
|
|
strTempQuery = "SELECT NumberOfRecords FROM Win32_NTEventlogFile " &_
|
|
"WHERE LogfileName=" & Chr(34) & strLogFile & Chr(34)
|
|
|
|
Set objEnumerator = objService.ExecQuery(strTempQuery,,0)
|
|
|
|
If Err.Number Then
|
|
component.vbPrintf L_ExecuteQuery_ErrorMessage, Array(strLogFile)
|
|
WScript.Quit(EXIT_METHOD_FAIL)
|
|
End If
|
|
|
|
' check if given log is present
|
|
If ObjEnumerator.Count <> 1 Then
|
|
component.vbPrintf L_LogDoesNotExist_ErrorMessage, Array(strLogFile)
|
|
'If Count of Logs = 1 Quit Here
|
|
If objLogs.Count= 1 Then
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
'If more proceed ..
|
|
objLogs.Remove(strLogFile)
|
|
Else
|
|
' get maximum number of records in that log(used if range specified)
|
|
For each objInstance in objEnumerator
|
|
If objInstance.NumberOfRecords <> "" Then
|
|
objLogs.Item(strLogFile) = objInstance.NumberOfRecords
|
|
Else
|
|
objLogs.Item(strLogFile) = 0
|
|
End If
|
|
Next
|
|
End If
|
|
Set ObjEnumerator = Nothing
|
|
End If
|
|
Next
|
|
|
|
If objLogs.Exists("*") Then
|
|
' if the * is specified, populate array with elements
|
|
objLogs.Remove("*")
|
|
' get the instances of the logs present in the system
|
|
Set objEnumerator = objService.InstancesOf(CLASS_EVENTLOG_FILE)
|
|
|
|
If Err.number Then
|
|
Wscript.Echo (L_InstancesFailed_ErrorMessage)
|
|
WScript.Quit(EXIT_METHOD_FAIL)
|
|
End If
|
|
|
|
' if no logs present
|
|
If objEnumerator.Count <= 0 Then
|
|
WScript.Echo (L_InfoNoLogsPresent_Message)
|
|
WScript.Quit(EXIT_UNEXPECTED)
|
|
Else
|
|
For Each objInstance In objEnumerator
|
|
If Not IsEmpty(objInstance.LogfileName) Then
|
|
If NOT objLogs.Exists(LCase(objInstance.LogfileName)) Then
|
|
If objInstance.NumberOfRecords Then
|
|
objLogs.Add LCase(objInstance.LogfileName), objInstance.NumberOfRecords
|
|
Else
|
|
objLogs.Add LCase(objInstance.LogfileName), 0
|
|
End If
|
|
End If
|
|
End If
|
|
Next
|
|
End If
|
|
End If
|
|
|
|
End Sub
|
|
|
|
'********************************************************************
|
|
'* Function: BuildFiltersForQuery
|
|
'*
|
|
'* Purpose: Builds the query with the filter arguments
|
|
'*
|
|
'* Input: [in] arrFilters the array containing the filter conditions
|
|
'*
|
|
'* Output: Returns the string to be concatenated to the main query
|
|
'*
|
|
'********************************************************************
|
|
Function BuildFiltersForQuery(ByVal arrFilters)
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Dim strTempFilter ' to store the return string
|
|
Dim i ' used in loop
|
|
|
|
strTempFilter = ""
|
|
For i = 0 to UBound(arrFilters)
|
|
strTempFilter = strTempFilter & " AND "
|
|
strTempFilter = strTempFilter & arrFilters(i)
|
|
Next
|
|
|
|
BuildFiltersForQuery = strTempFilter
|
|
|
|
End Function
|
|
|
|
'********************************************************************
|
|
'* Function : BuildRangeForQuery
|
|
'*
|
|
'* Purpose: Builds the range boundaries to display the records.
|
|
'*
|
|
'* Input: [in] strRange ' the range specified by the user
|
|
'* Will be in the format N, -N or N-N
|
|
'* [in] intFiltersSpecified ' array containing the filters number
|
|
'* [in] objService ' the service object
|
|
'* [out] intRecordRangeFrom ' where do we start the display of records?
|
|
'* [out] intRecordRangeTo ' where do we stop displaying records
|
|
'* [out] strFilterLog ' log file to build query
|
|
'* [out] strQuery ' to build query according to given Range Type
|
|
'* Output: Sets the value for the start and end of display boundaries.
|
|
'*
|
|
'********************************************************************
|
|
Private Function BuildRangeForQuery(ByVal strRange, _
|
|
ByRef intRecordRangeFrom, _
|
|
ByRef intRecordRangeTo,_
|
|
ByVal intFiltersSpecified,_
|
|
ByRef strQuery,_
|
|
ByVal ObjService,_
|
|
ByVal strFilterLog )
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Dim intMaxEventRecordsPresent ' to store the max recods in the log
|
|
Dim arrRangeValues ' to store the split values if range is of the type N-N
|
|
Dim objInstance ' to refer to the instances of the objEnumerator
|
|
Dim objEnumerator ' to store the results of the query is executed
|
|
Dim FilterRecordCount ' to store the count of records if filter with +N specified
|
|
|
|
FilterRecordCount = 0
|
|
|
|
BuildRangeForQuery = strquery 'intialize
|
|
|
|
Dim currentMaxRecordnumber 'curentMaxrecord number
|
|
Dim currentMinRecordnumber 'curentMinrecord number
|
|
|
|
currentMaxRecordnumber = 0
|
|
currentMinRecordnumber = 0
|
|
|
|
' save the max. no. of records available in the current log
|
|
intMaxEventRecordsPresent = intRecordRangeTo
|
|
|
|
|
|
' find the count of events / logfile if Filter is specified .
|
|
If intFiltersSpecified >= 0 Then
|
|
Set objEnumerator = objService.ExecQuery(strQuery,"WQL",0,null)
|
|
If Err.number Then
|
|
component.vbPrintf L_ExecuteQuery_ErrorMessage, Array(strFilterLog)
|
|
Exit Function
|
|
End if
|
|
|
|
FilterRecordCount= objEnumerator.count
|
|
|
|
Set objEnumerator= Nothing 'releases the memory
|
|
End If
|
|
|
|
' check the type of range specified ( first N / last N / N1 - N2 )
|
|
If ( IsNumeric(strRange) ) Then
|
|
|
|
' range is first N or last N
|
|
' now check whether it is first N or last N
|
|
If strRange < 0 Then
|
|
If intFiltersSpecified >= 0 Then
|
|
' first N records
|
|
' initial the counter so that all the out is displayed
|
|
|
|
If FilterRecordCount > CLng(Abs(strRange)) then
|
|
intRecordRangeFrom = FilterRecordCount - CLng(Abs(strRange)) + 1
|
|
intRecordRangeTo = FilterRecordCount
|
|
Else
|
|
intRecordRangeFrom = 0
|
|
intRecordRangeTo = FilterRecordCount
|
|
End If
|
|
|
|
Else
|
|
|
|
Set objEnumerator = objService.ExecQuery(strQuery,"WQL",48,null)
|
|
For Each objInstance In objEnumerator
|
|
currentMaxRecordnumber= objInstance.RecordNumber
|
|
Exit for
|
|
Next
|
|
|
|
If currentMaxRecordnumber > intMaxEventRecordsPresent then
|
|
currentMinRecordnumber = currentMaxRecordnumber - intMaxEventRecordsPresent
|
|
intMaxEventRecordsPresent = currentMaxRecordnumber
|
|
End If
|
|
Set objEnumerator= Nothing 'releases the memory
|
|
|
|
' N means record number <= N
|
|
' initial the counter s+o that all the out is displayed
|
|
' build the query
|
|
BuildRangeForQuery = strQuery & " AND RecordNumber <= "& CLng(Abs(strRange)) + currentMinRecordnumber
|
|
|
|
End If
|
|
Else
|
|
' *** range is last N (i.e -N)
|
|
If intFiltersSpecified >= 0 Then
|
|
|
|
If FilterRecordCount > CLng(Abs(strRange)) then
|
|
intRecordRangeFrom =0
|
|
intRecordRangeTo = CLng(Abs(strRange))
|
|
Else
|
|
intRecordRangeFrom =0
|
|
intRecordRangeTo = FilterRecordCount
|
|
End If
|
|
|
|
Else
|
|
|
|
Set objEnumerator = objService.ExecQuery(strQuery,"WQL",48,null)
|
|
'getting current max recordnumber
|
|
For Each objInstance In objEnumerator
|
|
currentMaxRecordnumber= objInstance.RecordNumber
|
|
Exit for
|
|
Next
|
|
|
|
If currentMaxRecordnumber > intMaxEventRecordsPresent then
|
|
currentMinRecordnumber = currentMaxRecordnumber - intMaxEventRecordsPresent
|
|
intMaxEventRecordsPresent = currentMaxRecordnumber
|
|
End If
|
|
|
|
Set objEnumerator= Nothing 'releases the memory
|
|
|
|
' -N means record number > (maxNumber - N )
|
|
' initial the counter so that all the out is displayed
|
|
' build the query
|
|
If CLng(Abs(strRange)) > intMaxEventRecordsPresent Then
|
|
'Show all records
|
|
BuildRangeForQuery =strQuery & " AND RecordNumber > 0 "
|
|
Else
|
|
BuildRangeForQuery =strQuery & " AND RecordNumber > " & intMaxEventRecordsPresent - CLng(Abs(strRange))
|
|
End If
|
|
End If
|
|
End If
|
|
Else
|
|
' range of records asked for N-N case
|
|
arrRangeValues = split(strRange,"-", 2, VBBinaryCompare)
|
|
|
|
If intFiltersSpecified >= 0 Then
|
|
If CLng(arrRangeValues(0)) < FilterRecordCount then
|
|
|
|
' initial the counter so that all the out is displayed
|
|
intRecordRangeFrom = CLng(arrRangeValues(0))
|
|
intRecordRangeTo = CLng(arrRangeValues(1))
|
|
Else
|
|
'forcebly putting the invaid query
|
|
'when N1 > FilterRecordCount to avoid unnessaray looping between intRecordRangeFrom TO intRecordRangeTo
|
|
BuildRangeForQuery =strQuery & " AND RecordNumber = 0 "
|
|
End If
|
|
Else
|
|
Set objEnumerator = objService.ExecQuery(strQuery,"WQL",48,null)
|
|
For Each objInstance In objEnumerator
|
|
currentMaxRecordnumber= objInstance.RecordNumber
|
|
Exit for
|
|
Next
|
|
|
|
If currentMaxRecordnumber > intMaxEventRecordsPresent then
|
|
currentMinRecordnumber = currentMaxRecordnumber - intMaxEventRecordsPresent
|
|
intMaxEventRecordsPresent = currentMaxRecordnumber
|
|
End If
|
|
Set objEnumerator= Nothing 'releases the memory
|
|
|
|
' build the query
|
|
BuildRangeForQuery =strQuery & " AND RecordNumber >= "& CLng(arrRangeValues(0))+ currentMinRecordnumber & " AND RecordNumber <= " & CLng(arrRangeValues(1)) + currentMinRecordnumber
|
|
|
|
End If
|
|
End If
|
|
|
|
End Function
|
|
|
|
'********************************************************************
|
|
'* Sub: ShowEvents
|
|
'*
|
|
'* Purpose: Displays the EventLog details
|
|
'*
|
|
'* Input:
|
|
'* [in] strMachine machine to query events from
|
|
'* [in] strUserName user name to connect to the machine
|
|
'* [in] strPassword password for the user
|
|
'* [in] arrFilters the array containing the filters
|
|
'* [in] strFormat the display format
|
|
'* [in] strRange the range of records required
|
|
'* [in] blnVerboseDisplay flag to verify if verbose display is needed
|
|
'* [in] blnNoHeader flag to verify if noheader display is needed
|
|
'* [in] objLogs to store all the given logfles
|
|
'* Output: Displays error message and quits if connection fails
|
|
'* Calls component.showResults() to display the event records
|
|
'*
|
|
'********************************************************************
|
|
Private Sub ShowEvents(ByVal strMachine, _
|
|
ByVal strUserName, _
|
|
ByVal strPassword, _
|
|
ByRef arrFilters, _
|
|
ByVal strFormat, _
|
|
ByVal strRange, _
|
|
ByVal blnVerboseDisplay, _
|
|
ByVal blnNoHeader,_
|
|
ByRef objLogs)
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
Dim objService ' the service object
|
|
Dim objEnumerator ' to store the results of the query is executed
|
|
Dim objInstance ' to refer to the instances of the objEnumerator
|
|
Dim strFilterLog ' to refer to each log specified by the user
|
|
Dim strTemp ' to store the temporary variables
|
|
Dim strQuery ' to store the query obtained for given conditions
|
|
Dim arrResults ' to store the columns of each filter
|
|
Dim arrHeader ' to store the array header values
|
|
Dim arrMaxLength ' to store the maximum length for each column
|
|
Dim arrFinalResults ' used to send the arrResults to component.showResults()
|
|
Dim arrTemp ' to store temporary array values
|
|
Dim intLoopCount ' used in the loop
|
|
Dim intElementCount ' used as array subscript
|
|
Dim strFilterQuery ' to store the query for the given filters
|
|
Dim intResultCount ' used to count no of records that are fetched in the query
|
|
Dim blnPrintHeader ' used to check header is printed or not in resulted Query
|
|
|
|
' the following are used for implementing the range option
|
|
Dim intRecordRangeFrom ' to store the display record beginning number
|
|
Dim intRecordRangeTo ' to store the display record ending number
|
|
Dim arrKeyName ' to store then key value of dictionary object
|
|
Dim strTempQuery ' to store a string for -N range values
|
|
Dim arrblnDisplay ' array to show the status of display of verbose mode for showresults function
|
|
Dim intDataCount ' used in looping to get value of Insertion string for the field "Description column"
|
|
Dim i 'used for looping to enable All special privileges
|
|
|
|
' flag to set condition specific locale & default value setting
|
|
Dim bLocaleChanged
|
|
bLocaleChanged =FALSE
|
|
|
|
'Validating the arguments which is passed from commandline
|
|
If NOT (ValidateArguments(strMachine, strUserName, strPassword, _
|
|
arrFilters, strFormat, strRange , blnNoHeader)) Then
|
|
WScript.Quit(EXIT_UNEXPECTED)
|
|
End If
|
|
|
|
|
|
' checking for UNC format for the system name
|
|
If Left(strMachine,2) = UNC_Format_Servername Then
|
|
If Len(strMachine) = 2 Then
|
|
component.vbPrintf InvalidInputErrorMessage ,Array(Wscript.ScriptName)
|
|
WScript.Quit(EXIT_UNEXPECTED)
|
|
End if
|
|
strMachine = Mid(strMachine,3,Len(strMachine))
|
|
End If
|
|
|
|
'getting the password ....
|
|
If ((strUserName <> VBEmpty) AND (strPassword = VBEmpty)) Then
|
|
strPassword = component.getPassword()
|
|
End If
|
|
|
|
' To set GetSupportedUserLocale for Some Diff locales
|
|
bLocaleChanged =GetSupportedUserLocale()
|
|
|
|
'Establish a connection with the server.
|
|
If NOT component.wmiConnect(CONST_NAMESPACE_CIMV2 , _
|
|
strUserName , _
|
|
strPassword , _
|
|
strMachine , _
|
|
blnLocalConnection , _
|
|
objService ) Then
|
|
|
|
Wscript.Echo(L_HintCheckConnection_Message)
|
|
WScript.Quit(EXIT_METHOD_FAIL)
|
|
|
|
End If
|
|
|
|
' set the previlige's To query all event's in eventlog's .
|
|
objService.Security_.Privileges.AddAsString("SeSecurityPrivilege")
|
|
|
|
'Enable all privileges as some DC's were requiring special privileges
|
|
For i = 1 to 26
|
|
objService.Security_.Privileges.Add(i)
|
|
Next
|
|
|
|
' get the HostName from the function
|
|
strMachine = component.getHostName( objService)
|
|
|
|
' Validating the Filters which is passed from commandline
|
|
If UBound(arrFilters) >= 0 Then
|
|
' filters are specified. Validate them
|
|
If Not ValidateFilters(arrFilters,objService ) Then
|
|
WScript.Quit(EXIT_INVALID_INPUT)
|
|
End If
|
|
End If
|
|
|
|
|
|
blnPrintHeader = TRUE
|
|
|
|
If blnNoHeader Then
|
|
blnPrintHeader = FALSE
|
|
End If
|
|
|
|
' Initialize - header to display, the maximum length of each column and
|
|
' number of columns present
|
|
arrHeader = Array(L_ColHeaderType_Text,L_ColHeaderEventcode_Text, L_ColHeaderDateTime_Text,_
|
|
L_ColHeaderSource_Text,L_ColHeaderComputerName_Text)
|
|
' first initialize the array with N/A
|
|
arrResults = Array(L_TextNa_Text,L_TextNa_Text,L_TextNa_Text,L_TextNa_Text,L_TextNa_Text,L_TextNa_Text,_
|
|
L_TextNa_Text,L_TextNa_Text)
|
|
|
|
arrMaxLength = Array(13,6, 24, 17, 14, 15, 20,750)
|
|
arrblnDisplay = Array(0, 0, 0, 0, 0, 1, 1, 1)
|
|
|
|
If blnVerboseDisplay Then
|
|
arrblnDisplay = Array(0, 0, 0, 0, 0, 0, 0,0)
|
|
arrHeader = Array( L_ColHeaderType_Text,L_ColHeaderEventcode_Text, L_ColHeaderDateTime_Text, _
|
|
L_ColHeaderSource_Text,L_ColHeaderComputerName_Text,L_ColHeaderCategory_Text,_
|
|
L_ColHeaderUser_Text, L_ColHeaderDesription_Text)
|
|
End IF
|
|
|
|
If UBound(arrFilters) >=0 Then
|
|
strFilterQuery = BuildFiltersForQuery(arrFilters)
|
|
End If
|
|
|
|
' call function to verify given log and also get records count in log
|
|
Call VerifyLogAndGetMaxRecords(objService, objLogs)
|
|
|
|
arrKeyName = objLogs.Keys
|
|
|
|
intResultCount = 0
|
|
intLoopCount = 0
|
|
|
|
'blank line before first data is displayed on console
|
|
WScript.Echo EmptyLine_Text
|
|
|
|
Do While (intLoopCount < objLogs.Count)
|
|
|
|
'setting Header to print every Log file explicilty
|
|
If blnNoHeader Then
|
|
blnPrintHeader = FALSE
|
|
Else
|
|
blnPrintHeader = TRUE
|
|
End If
|
|
|
|
|
|
If CInt(objLogs.Item(arrKeyName(intLoopCount))) > 0 Then
|
|
strFilterLog = arrKeyName(intLoopCount)
|
|
intRecordRangeFrom = 0
|
|
intRecordRangeTo = CInt(objLogs.Item(arrKeyName(intLoopCount)))
|
|
|
|
|
|
' build the query
|
|
strQuery = "Select * FROM Win32_NTLogEvent WHERE Logfile=" &_
|
|
Chr(34) & strFilterLog & Chr(34)
|
|
|
|
If UBound(arrFilters) >=0 Then
|
|
strQuery = strQuery & strFilterQuery
|
|
End If
|
|
|
|
|
|
If Len(Trim(CStr(strRange))) > 0 Then
|
|
' building again query for -N condition in range switch
|
|
strQuery = BuildRangeForQuery(strRange,intRecordRangeFrom, _
|
|
intRecordRangeTo, UBound(arrFilters),strQuery,objService,strFilterLog)
|
|
|
|
End If
|
|
|
|
' process the results, else go for next log
|
|
Set objEnumerator = objService.ExecQuery(strQuery,"WQL",48,null)
|
|
|
|
If Err.Number Then
|
|
component.vbPrintf L_ExecuteQuery_ErrorMessage, Array(strFilterLog)
|
|
' if error occurred in the query, go for next log
|
|
intLoopCount = intLoopCount + 1
|
|
Err.clear ' for next loop if more logs present
|
|
Else
|
|
|
|
intElementCount = 0
|
|
|
|
ReDim arrFinalResults(CONST_ARRAYBOUND_NUMBER)
|
|
|
|
For each objInstance in objEnumerator
|
|
|
|
' inside error trapping for most unexpected case...
|
|
If Err.number then Exit For
|
|
|
|
intResultCount = intResultCount + 1
|
|
|
|
' print the header for each log file along with Host Name
|
|
'imp:: if and only if have Data
|
|
If intResultCount = 1 Then
|
|
WScript.Echo(String(78,"-"))
|
|
component.vbPrintf L_InfoDisplayLog_Message ,Array(strFilterLog,strMachine)
|
|
WScript.Echo(String(78,"-"))
|
|
End If
|
|
|
|
' check whether the current record is fitting in
|
|
' the required range
|
|
If ( intResultCount >= intRecordRangeFrom ) And _
|
|
( intResultCount <= intRecordRangeTo ) Then
|
|
' record fitting the range ... this has to be displayed
|
|
|
|
If objInstance.Type <> "" Then
|
|
arrResults(0) = objInstance.Type
|
|
Else
|
|
arrResults(0) = L_TextNa_Text
|
|
End If
|
|
|
|
If objInstance.EventCode <> "" Then
|
|
arrResults(1) = objInstance.EventCode
|
|
Else
|
|
arrResults(1) = L_TextNa_Text
|
|
End If
|
|
|
|
If (NOT IsEmpty(objInstance.TimeGenerated)) Then
|
|
|
|
strTemp = objInstance.TimeGenerated
|
|
|
|
'is LOCALE CHANGED
|
|
If bLocaleChanged <> TRUE Then
|
|
'format DatTime as DATE & "Space" & TIME
|
|
arrResults(2)= Formatdatetime( Mid(strTemp,5,2) & "/" & Mid(strTemp,7,2) & "/" &_
|
|
Mid(strTemp,1,4)) & " " & formatdatetime( Mid(strTemp,9,2) & ":" &_
|
|
Mid(strTemp,11,2) & ":" & Mid(strTemp,13,2))
|
|
Else
|
|
arrResults(2) = Mid(strTemp,5,2) & "/" & Mid(strTemp,7,2) & "/" &_
|
|
Mid(strTemp,1,4) & " " & Mid(strTemp,9,2) & ":" &_
|
|
Mid(strTemp,11,2) & ":" & Mid(strTemp,13,2)
|
|
End If
|
|
|
|
Else
|
|
arrResults(2) = L_TextNa_Text
|
|
End If
|
|
|
|
If objInstance.SourceName <> "" Then
|
|
arrResults(3) = objInstance.SourceName
|
|
Else
|
|
arrResults(3) = L_TextNa_Text
|
|
End If
|
|
|
|
If objInstance.ComputerName <> "" Then
|
|
arrResults(4) =objInstance.ComputerName
|
|
Else
|
|
arrResults(4) = L_TextNa_Text
|
|
End If
|
|
|
|
If blnVerboseDisplay Then
|
|
If objInstance.CategoryString <> "" Then
|
|
arrResults(5) = Replace(objInstance.CategoryString, VbCrLf, "")
|
|
Else
|
|
arrResults(5) = L_TextNone_Text ' None display
|
|
End If
|
|
|
|
If (NOT IsNull(objInstance.User)) Then
|
|
arrResults(6) = objInstance.User
|
|
Else
|
|
arrResults(6) = L_TextNa_Text
|
|
End If
|
|
|
|
If objInstance.Message <> "" Then
|
|
arrResults(7) = Trim(Replace(objInstance.Message, VbCrLf, " "))
|
|
Else
|
|
'Check here eighter value in presenet "InsertionStrings" column .
|
|
If (NOT IsNull(objInstance.InsertionStrings)) Then
|
|
arrTemp = objInstance.InsertionStrings
|
|
'removing default value "N/A"
|
|
arrResults(7)= ""
|
|
For intDataCount = 0 to UBound(arrTemp)
|
|
arrResults(7) = arrResults(7) & " " & arrTemp(intDataCount)
|
|
Next
|
|
|
|
arrResults(7) = Trim(arrResults(7))
|
|
Else
|
|
arrResults(7) = L_TextNa_Text
|
|
End If
|
|
|
|
End If
|
|
|
|
End If
|
|
|
|
' add the record to the queue of records that has to be displayed
|
|
arrFinalResults( intElementCount ) = arrResults
|
|
intElementCount = intElementCount + 1 ' increment the buffer
|
|
' check whether the output buffer is filled and ready for display
|
|
' onto the screen or not
|
|
If intElementCount = CONST_ARRAYBOUND_NUMBER +1 Then
|
|
' Call the display function with required parameters
|
|
Call component.showResults(arrHeader, arrFinalResults, arrMaxLength, _
|
|
strFormat, blnPrintHeader, arrblnDisplay)
|
|
blnPrintHeader = FALSE
|
|
|
|
Redim arrFinalResults(CONST_ARRAYBOUND_NUMBER) ' clear the existing buffer contents
|
|
intElementCount = 0 ' reset the buffer start
|
|
End If
|
|
End If
|
|
|
|
' check whether the last record number that has to be displayed is
|
|
' crossed or not ... if crossed exit the loop without proceeding further
|
|
If ( intResultCount >= intRecordRangeTo ) Then
|
|
' max. TO range is crossed/reached ... no need of further looping
|
|
Exit For
|
|
End If
|
|
Next
|
|
|
|
' check whether there any pending in the output buffer that has to be
|
|
' displayed
|
|
If intElementCount > 0 Then
|
|
' resize the array so that the buffer is shrinked to its content size
|
|
ReDim Preserve arrFinalResults( intElementCount - 1 )
|
|
|
|
' Call the display function with required parameters
|
|
Call component.showResults(arrHeader, arrFinalResults, arrMaxLength, _
|
|
strFormat, blnPrintHeader, arrblnDisplay)
|
|
Else ' array bounds checking
|
|
If intResultCount = 0 Then
|
|
'ie no records found
|
|
If UBound(arrFilters) >= 0 OR Len(Trim(CStr(strRange))) > 0 Then
|
|
' message no records present if filter specified
|
|
component.vbPrintf L_InfoNoRecordsInFilter_Message, Array(strFilterLog)
|
|
Else
|
|
'message no records present if filter not specified
|
|
component.vbPrintf L_InfoNoRecords_Message, Array(strFilterLog)
|
|
End If
|
|
|
|
End If ' intResultCount = 0
|
|
|
|
End If ' array bounds checking
|
|
End If
|
|
Else
|
|
'message no records present
|
|
component.vbPrintf L_InfoNoRecords_Message, Array(arrKeyName(intLoopCount))
|
|
End If
|
|
|
|
' re-initialize all the needed variables
|
|
intResultCount = 0
|
|
Set objEnumerator = Nothing
|
|
intLoopCount = intLoopCount + 1
|
|
|
|
'blank line before end of the Next Each Log file details
|
|
WScript.Echo EmptyLine_Text
|
|
|
|
Loop ' do-while
|
|
|
|
End Sub
|
|
|
|
'********************************************************************
|
|
'* Function: GetSupportedUserLocale
|
|
'*
|
|
'* Purpose:This function checks if the current locale is supported or not.
|
|
'*
|
|
'* Output: Returns TRUE or FALSE
|
|
'*
|
|
'********************************************************************
|
|
Private Function GetSupportedUserLocale()
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
GetSupportedUserLocale =FALSE
|
|
|
|
CONST LANG_ARABIC = &H01
|
|
CONST LANG_HEBREW = &H0d
|
|
CONST LANG_HINDI = &H39
|
|
CONST LANG_TAMIL = &H49
|
|
CONST LANG_THAI = &H1e
|
|
CONST LANG_VIETNAMESE = &H2a
|
|
|
|
Dim Lcid 'to store LocaleId
|
|
|
|
' get the current locale
|
|
Lcid=GetLocale()
|
|
|
|
CONST PRIMARYLANGID = 1023 '0x3ff
|
|
|
|
Dim LANGID 'to store LangID
|
|
|
|
' Convert LCID >>>>>>>>>>>>> LANGID
|
|
' BIT Wise And Operation
|
|
'formating to compare HEX Value's
|
|
LANGID = Hex ( Lcid AND PRIMARYLANGID )
|
|
|
|
' check whether the current locale is supported by our tool or not
|
|
' if not change the locale to the English which is our default locale
|
|
Select Case LANGID
|
|
|
|
'here to chaeck the values
|
|
Case Hex(LANG_ARABIC),Hex(LANG_HEBREW),Hex(LANG_THAI) ,Hex(LANG_HINDI ),Hex(LANG_TAMIL) ,Hex(LANG_VIETNAMESE)
|
|
|
|
GetSupportedUserLocale =TRUE
|
|
Exit Function
|
|
End Select
|
|
|
|
End Function
|
|
|
|
|
|
' ****************************************************************************************
|
|
'* Function : IsWinXP
|
|
'*
|
|
'* Purpose:This function checks if the OS is XP or Above.
|
|
'*
|
|
'* Input: [in] Objservice the service object
|
|
'* Output: Returns TRUE or FALSE
|
|
'*
|
|
' ****************************************************************************************
|
|
|
|
Private Function IsWinXP ( ByVal objService)
|
|
|
|
ON ERROR RESUME NEXT
|
|
Err.Clear
|
|
|
|
CONST WIN2K_MAJOR_VERSION = 5000
|
|
CONST WINXP_MAJOR_VERSION = 5001
|
|
|
|
Dim strQuery ' to store the query to be executed
|
|
Dim objEnum ' collection object
|
|
Dim objInstance ' instance object
|
|
Dim strVersion ' to store the OS version
|
|
Dim arrVersionElements ' to store the OS version elements
|
|
Dim CurrentMajorVersion ' the major version number
|
|
|
|
ISWinXP= FALSE
|
|
|
|
strQuery = "Select * From Win32_operatingsystem"
|
|
|
|
Set objEnum = objService.ExecQuery(strQuery,"WQL",0,NULL)
|
|
|
|
For each objInstance in objEnum
|
|
strVersion= objInstance.Version
|
|
Next
|
|
|
|
' OS Version : 5.1.xxxx(Whistler), 5.0.xxxx(Win2K)
|
|
arrVersionElements = split(strVersion,".")
|
|
' converting to major version
|
|
CurrentMajorVersion = arrVersionElements(0) * 1000 + arrVersionElements(1)
|
|
|
|
' Determine the OS Type
|
|
' WinXP > Win2K
|
|
If CInt(CurrentMajorVersion) >= CInt(WINXP_MAJOR_VERSION) Then
|
|
IsWinXP= TRUE
|
|
End If
|
|
|
|
End Function
|
|
|
|
'********************************************************************
|
|
'* Sub: ShowUsage
|
|
'*
|
|
'* Purpose: Shows the correct usage to the user.
|
|
'*
|
|
'* Output: Help messages are displayed on screen.
|
|
'*
|
|
'********************************************************************
|
|
|
|
Private Sub ShowUsage ()
|
|
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine01_Text
|
|
WScript.Echo L_ShowUsageLine02_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine03_Text
|
|
WScript.Echo L_ShowUsageLine04_Text
|
|
WScript.Echo L_ShowUsageLine05_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine06_Text
|
|
WScript.Echo L_ShowUsageLine07_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine08_Text
|
|
WScript.Echo L_ShowUsageLine09_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine10_Text
|
|
WScript.Echo L_ShowUsageLine11_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine12_Text
|
|
WScript.Echo L_ShowUsageLine13_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine14_Text
|
|
WScript.Echo L_ShowUsageLine15_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine16_Text
|
|
WScript.Echo L_ShowUsageLine17_Text
|
|
WScript.Echo L_ShowUsageLine18_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine19_Text
|
|
WScript.Echo L_ShowUsageLine20_Text
|
|
WScript.Echo L_ShowUsageLine21_Text
|
|
WScript.Echo L_ShowUsageLine22_Text
|
|
WScript.Echo L_ShowUsageLine23_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine24_Text
|
|
WScript.Echo L_ShowUsageLine25_Text
|
|
WScript.Echo L_ShowUsageLine26_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine27_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine28_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine29_Text
|
|
WScript.Echo L_ShowUsageLine30_Text
|
|
WScript.Echo L_ShowUsageLine31_Text
|
|
WScript.Echo L_ShowUsageLine32_Text
|
|
WScript.Echo L_ShowUsageLine33_Text
|
|
WScript.Echo L_ShowUsageLine34_Text
|
|
WScript.Echo L_ShowUsageLine35_Text
|
|
WScript.Echo L_ShowUsageLine36_Text
|
|
WScript.Echo L_ShowUsageLine37_Text
|
|
WScript.Echo L_ShowUsageLine38_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine39_Text
|
|
WScript.Echo L_ShowUsageLine40_Text
|
|
WScript.Echo EmptyLine_Text
|
|
WScript.Echo L_ShowUsageLine41_Text
|
|
WScript.Echo L_ShowUsageLine42_Text
|
|
WScript.Echo L_ShowUsageLine43_Text
|
|
WScript.Echo L_ShowUsageLine44_Text
|
|
WScript.Echo L_ShowUsageLine45_Text
|
|
WScript.Echo L_ShowUsageLine46_Text
|
|
WScript.Echo L_ShowUsageLine47_Text
|
|
WScript.Echo L_ShowUsageLine48_Text
|
|
WScript.Echo L_ShowUsageLine49_Text
|
|
WScript.Echo L_ShowUsageLine50_Text
|
|
WScript.Echo L_ShowUsageLine51_Text
|
|
WScript.Echo L_ShowUsageLine52_Text
|
|
WScript.Echo L_ShowUsageLine53_Text
|
|
|
|
End Sub
|
|
|
|
'-----------------------------------------------------------------------------
|
|
' End of the Script
|
|
'-----------------------------------------------------------------------------
|
|
|