3847 lines
177 KiB
XML
3847 lines
177 KiB
XML
<job id='wi_compile'>
|
|
<!-- includes for constants definitions -->
|
|
<script language='VBScript' src='.\vbsconst.inc'></script>
|
|
<script language='VBScript' src='.\xmlconst.inc'></script>
|
|
<script language='VBScript' src='.\witables.inc'></script>
|
|
<script language='VBScript' src='.\wival.inc'></script>
|
|
<script language='VBScript' src='.\wiconst.inc'></script>
|
|
|
|
<script language='VBScript' src='.\ritables.inc'></script>
|
|
|
|
<!-- includes for function declarations -->
|
|
<script language='VBScript' src='.\wixerror.inc'></script>
|
|
<script language='VBScript' src='.\wixload.inc'></script>
|
|
|
|
<!-- main -->
|
|
<script Language='VBScript'>
|
|
|
|
' Compiles Windows Installer XML manifest into a Windows Installer database
|
|
Option Explicit
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' main
|
|
Public xmldoc 'As XMLDOMDocument
|
|
Public installer 'As Installer
|
|
Public database 'As Database
|
|
Public sumInfoArray(19)
|
|
Public lastDiskId 'As integer
|
|
Public featureDisplay 'As integer
|
|
Public dictView 'As Dictionary
|
|
Public dictVars 'As Dictionary
|
|
Public dictStandardProperties 'As Dictionary
|
|
Public dictFiles 'As Dictionary
|
|
Public dictModules 'As Dictionary
|
|
Public fso 'As FileSystemObject
|
|
Public tempDir 'As String
|
|
Public fCreate, fStyle, fTransform, fVerbose, fOpenModel, fHelp, fLogOps, fNoOnError, fNoModularize, fNoAddValidation
|
|
Public fNoBinary, fNoUI, fNoSumInfo, fNoSeqTables, fNoLinkerInfo
|
|
Public fModule ' if creating Merge Module
|
|
Public fPatch ' if creating Patch
|
|
Public moduleId 'As String
|
|
|
|
|
|
Public fileSeq : fileSeq = 1 ' used to make file sequencing count each file
|
|
Public regCount : regCount = 0 ' used to make registry keys unique
|
|
Public osCount : osCount = 0 ' used for dummy primary key in redist os elements
|
|
Public patchOrder : patchOrder = 1 ' used to make patch order count
|
|
Public externalOrder : externalOrder = 1 ' used to make external file order count
|
|
|
|
Public manifestPath, stylePath, databasePath, outputPath
|
|
Public g_sBaseDir ' base directory used for update and CAB'ing
|
|
|
|
Dim openMode, sumInfo, index
|
|
|
|
' Connect to Windows Installer, create dictionaries views, variables
|
|
Set installer = Nothing
|
|
Set installer = WScript.CreateObject("WindowsInstaller.Installer") : CheckError
|
|
Set fso = WScript.CreateObject("Scripting.FileSystemObject") : CheckError
|
|
Set dictView = WScript.CreateObject("Scripting.Dictionary") : CheckError
|
|
Set dictVars = WScript.CreateObject("Scripting.Dictionary") : CheckError
|
|
Set dictFiles = WScript.CreateObject("Scripting.Dictionary") : CheckError
|
|
Set dictModules = WScript.CreateObject("Scripting.Dictionary") : CheckError
|
|
|
|
' properties standard in the Windows Installer that are not to be Modularized
|
|
Set dictStandardProperties = WScript.CreateObject("Scripting.Dictionary") : CheckError
|
|
dictStandardProperties.Add "TARGETDIR", ""
|
|
dictStandardProperties.Add "Manufacturer", ""
|
|
dictStandardProperties.Add "Privileged", ""
|
|
dictStandardProperties.Add "ProductCode", ""
|
|
dictStandardProperties.Add "ProductID", ""
|
|
dictStandardProperties.Add "ProductLanguage", ""
|
|
dictStandardProperties.Add "ProductName", ""
|
|
dictStandardProperties.Add "ProductVersion", ""
|
|
|
|
tempDir = installer.Environment("TMP")
|
|
If Len(tempDir) = 0 Then tempDir = installer.Environment("TEMP")
|
|
|
|
ParseCommandLine
|
|
If Not fNoOnError Then On Error Resume Next
|
|
|
|
If fHelp Or IsEmpty(manifestPath) Then
|
|
ShowHelp
|
|
WScript.Quit 1
|
|
End If
|
|
|
|
' load manifest and apply any style sheets
|
|
Dim rootElement : Set rootElement = LoadDocument(manifestPath, stylePath, dictVars)
|
|
|
|
' mark if this is a module or not
|
|
If "Module" = rootElement.nodeName Then fModule = True Else fModule = False
|
|
If "Patch" = rootElement.nodeName Then fPatch = True Else fPatch = False
|
|
|
|
' open or create new database according to defined schema
|
|
If IsEmpty(outputPath) Then
|
|
' if a database wasn't specified use the base of the XML file and add the appropriate extension
|
|
If IsEmpty(databasePath) Then
|
|
Dim offset : offset = InStrRev(manifestPath, ".", -1, vbTextCompare)
|
|
databasePath = Left(manifestPath, offset - 1)
|
|
If fModule Then
|
|
databasePath = databasePath & ".wixobj"
|
|
ElseIf fPatch Then
|
|
databasePath = databasePath & ".pcp"
|
|
Else
|
|
databasePath = databasePath & ".wixobj"
|
|
End If
|
|
fCreate = True
|
|
End If
|
|
|
|
If fTransform Then Fail "Must supply an output name for transform"
|
|
If fCreate Then openMode = msiOpenDatabaseModeCreate Else openMode = msiOpenDatabaseModeTransact
|
|
ElseIf fTransform Then
|
|
openMode = "temptran.msi" ' temporary until we implement a better way
|
|
Else
|
|
openMode = outputPath
|
|
End If
|
|
Set database = installer.OpenDatabase(databasePath, openMode) : CheckError
|
|
|
|
' do the processing
|
|
If fPatch Then
|
|
ProcessPatchElement rootElement
|
|
Else
|
|
ProcessProductElement rootElement
|
|
End If
|
|
|
|
' if this is a transform create the transform
|
|
If fTransform Then
|
|
Dim databaseRef : Set databaseRef = installer.OpenDatabase(databasePath, msiOpenDatabaseModeReadOnly) : CheckError
|
|
database.GenerateTransform databaseRef, outputPath : CheckError
|
|
database.CreateTransformSummaryInfo databaseRef, outputPath, 0, 0 : CheckError
|
|
' !! need to provide validation options in XML package element
|
|
Set database = Nothing
|
|
Else ' not creating a transform
|
|
' write the linker info
|
|
If Not fNoLinkerInfo And Not fPatch Then WriteLinkerInfo
|
|
|
|
If Not IsEmpty(sumInfoArray) Then
|
|
If IsEmpty(outputPath) Then
|
|
Set sumInfo = database.SummaryInformation(20) : CheckError
|
|
Else
|
|
database.Commit : CheckError
|
|
Set database = Nothing
|
|
Set sumInfo = installer.SummaryInformation(outputPath, 20) : CheckError
|
|
End If
|
|
|
|
' write the summary information into the database
|
|
For index = 1 To UBound(sumInfoArray)
|
|
If Not IsEmpty(sumInfoArray(index)) Then sumInfo.Property(index) = sumInfoArray(index) : CheckError
|
|
Next
|
|
sumInfo.Persist : CheckError
|
|
End If
|
|
|
|
If IsEmpty(outputPath) Then
|
|
database.Commit : CheckError
|
|
Set database = Nothing
|
|
End If
|
|
End If
|
|
Set dictVars = Nothing
|
|
|
|
WScript.Quit 0
|
|
|
|
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' Error handling and command-line parsing routines
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''' robmen ''
|
|
' ParseCommandLine
|
|
Function ParseCommandLine()
|
|
Dim arg, argIndex
|
|
Dim chFlag
|
|
|
|
If 0 = WScript.Arguments.Count Then fHelp = True : Exit Function
|
|
|
|
For argIndex = 0 To WScript.Arguments.Count - 1
|
|
arg = WScript.Arguments(argIndex)
|
|
chFlag = AscW(arg)
|
|
|
|
' if this a variable
|
|
If InStr(arg, "=") Then
|
|
Dim expr : expr = Split(arg, "=")
|
|
If IsNumeric(expr(1)) Then expr(1) = CLng(expr(1))
|
|
dictVars.Item(expr(0)) = expr(1)
|
|
' command line parameter
|
|
ElseIf (chFlag = AscW("/")) Or (chFlag = AscW("-")) Then
|
|
chFlag = LCase(Mid(arg, 2))
|
|
Select Case chFlag
|
|
Case "a" ' stylesheet to apply
|
|
argIndex = argIndex + 1
|
|
stylePath = WScript.Arguments(argIndex)
|
|
fStyle = True
|
|
Case "b" ' base directory
|
|
argIndex = argIndex + 1
|
|
g_sBaseDir = WScript.Arguments(argIndex)
|
|
Case "c" ' database to create
|
|
argIndex = argIndex + 1
|
|
databasePath = WScript.Arguments(argIndex)
|
|
If fso.FileExists(databasePath) Then WScript.Echo "Warning, overwriting database: " & databasePath
|
|
fCreate = True
|
|
Case "d" ' database to update
|
|
argIndex = argIndex + 1
|
|
databasePath = WScript.Arguments(argIndex)
|
|
If Not fso.FileExists(databasePath) Then Fail "Cannot find database: " & databasePath
|
|
Case "t" ' transform
|
|
argIndex = argIndex + 1
|
|
outputPath = WScript.Arguments(argIndex)
|
|
fTransform= True
|
|
Case "sb" : fNoBinary = True
|
|
Case "su" : fNoUI = True
|
|
Case "ss" : fNoSumInfo = True
|
|
Case "sq" : fNoSeqTables = True
|
|
Case "sl" : fNoLinkerInfo = True
|
|
Case "l" : fLogOps = True
|
|
Case "o" : fOpenModel = True
|
|
Case "v" : fVerbose = True
|
|
Case "e" : fNoOnError = True
|
|
Case "sm" : fNoModularize = True
|
|
Case "sv" : fNoAddValidation = True
|
|
Case "sf" : WScript.Echo "-sf has been deprecated"
|
|
Case "sc" : WScript.Echo "-sc has been deprecated"
|
|
Case "?" : fHelp = True
|
|
Case Else : Fail "Invalid option flag: " & arg
|
|
End Select
|
|
' must be the xml file
|
|
Else
|
|
If Not IsEmpty(manifestPath) Then Fail "Cannot specify two input xml documents"
|
|
manifestPath = arg
|
|
End If
|
|
Next
|
|
End Function ' ParseCommandLine
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''' robmen ''
|
|
' ShowHelp
|
|
Sub ShowHelp()
|
|
Dim sHelp
|
|
sHelp = "candle - compiles Windows Installer Xml into a Windows Installer Database" & vbCrLf & _
|
|
vbCrLf & _
|
|
"candle.wsf [-?] [-sb] [-sl] [-sm] [-sq] [-ss] [-su] [-sv] [-i FilePaths]" & vbCrLf & _
|
|
" [-t foo.mst] [-b basedir] [-a foo.wxs] [-c destfile.wixobj]" & vbCrLf & _
|
|
" [-d foo.msi] [-l] [-o] [-v] [-e] foo.wxm" & vbCrLf & _
|
|
vbCrLf & _
|
|
" -a apply Windows installer Xml Stylesheet [default extension .wxs]" & vbCrLf & _
|
|
" -b base directory for 'src' attributes" & vbCrLf & _
|
|
" -c database / module to create from Windows installer Xml [will overwrite]" & vbCrLf & _
|
|
" -d database to open and apply Windows installer Xml to [will not overwrite]" & vbCrLf & _
|
|
" -e errors crash compiler, useful for debugging compiler" & vbCrLf & _
|
|
" -i include paths to search (not yet implemented!)" & vbCrLf & _
|
|
" -l log all operations, useful for debugging" & vbCrLf & _
|
|
" -o open document model, ignores unexpected elements and attributes" & vbCrLf & _
|
|
" -sb suppress processing of Binary-encoded data" & vbCrLf & _
|
|
" -sc [DEPRECATED] suppress CAB'ing [valid only for Merge Modules]" & vbCrLf & _
|
|
" CAB'ing done by linker [see light.wsf]" & vbCrLf & _
|
|
" -sf [DEPRECATED] suppress updating the size, version, and language of files" & vbCrLf & _
|
|
" updating file info done by linker [see light.wsf]" & vbCrLf & _
|
|
" -sl suppress writing information for linker" & vbCrLf & _
|
|
" -sm suppress modularization for merge modules" & vbCrLf & _
|
|
" -sq suppress processing of Sequence elements" & vbCrLf & _
|
|
" -ss suppress processing of Summary Information" & vbCrLf & _
|
|
" -su suppress processing of UI elements" & vbCrLf & _
|
|
" -sv suppress automatic creation of Validation table" & vbCrLf & _
|
|
" -t transform to create [default extension .mst]" & vbCrLf & _
|
|
" -v verbose output, useful for debugging" & vbCrLf & _
|
|
" -? this help information" & vbCrLf & _
|
|
vbCrLf & _
|
|
"Common extensions:" & vbCrLf & _
|
|
" .wxf - Windows installer Xml Fragment" & vbCrLf & _
|
|
" .wxm - Windows installer Xml Module" & vbCrLf & _
|
|
" .wxp - Windows installer Xml Product" & vbCrLf & _
|
|
" .wxa - Windows installer Xml Patch" & vbCrLf & _
|
|
" .wixobj - Windows installer Xml Object File (in MSI format)" & vbCrLf & _
|
|
vbCrLf & _
|
|
" .msm - Windows installer Merge Module" & vbCrLf & _
|
|
" .msi - Windows installer Product Database" & vbCrLf & _
|
|
" .mst - Windows installer Transform" & vbCrLf & _
|
|
" .pcp - Windows installer Patch Creation Package" & vbCrLf & _
|
|
vbCrLf & _
|
|
"For more information see: http://compcat/wix"
|
|
WScript.Echo sHelp
|
|
End Sub ' ShowHelp
|
|
|
|
Sub CheckError
|
|
Dim message, errRec
|
|
If Err = 0 Then Exit Sub
|
|
message = Err.Source & " " & Hex(Err) & ": " & Err.Description
|
|
If Not installer Is Nothing Then
|
|
Set errRec = installer.LastErrorRecord
|
|
If Not errRec Is Nothing Then message = message & vbNewLine & errRec.FormatText
|
|
End If
|
|
Fail message
|
|
End Sub
|
|
|
|
Sub Unexpected(child, parent)
|
|
If Not fOpenModel Then
|
|
Fail "Unexpected " & child.nodeTypeString & " node: " & child.nodeName & ", parent = " & parent.nodeName
|
|
End If
|
|
End Sub
|
|
|
|
Function DosDate(convertdate)
|
|
DosDate = 0 ' !!! TODO: do the conversion to Dos times
|
|
End Function ' DosDate
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' BaseDir
|
|
Function BaseDir(sPath)
|
|
If IsEmpty(g_sBaseDir) Then g_sBaseDir = "."
|
|
|
|
If "sourcedir\" = LCase(Left(sPath, 10)) Then
|
|
BaseDir = g_sBaseDir & Mid(sPath, 10)
|
|
Else
|
|
BaseDir = sPath
|
|
End If
|
|
End Function ' BaseDir
|
|
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' Database processing routines
|
|
|
|
Sub CreateTable(table)
|
|
Dim primaryKeys, index, query, name
|
|
For index = 1 To UBound(table)
|
|
If Instr(table(index), " PRIMARY KEY") <> 0 Then
|
|
If Not IsEmpty(primaryKeys) Then primaryKeys = primaryKeys & ","
|
|
primaryKeys = primaryKeys & Split(table(index))(0)
|
|
End If
|
|
Next
|
|
query = "CREATE TABLE " & Replace(Join(table, ","), ",", "(", 1, 1)
|
|
query = Replace(query, " PRIMARY KEY", "")
|
|
query = query & " PRIMARY KEY " & primaryKeys & ")"
|
|
If fVerbose Then Wscript.Echo query
|
|
database.OpenView(query).Execute : CheckError
|
|
If Not fPatch And Not fNoAddValidation Then AddValidation installer, database, Replace(table(0), "`", ""), fVerbose
|
|
|
|
' some tables must exist to fulfill the Windows Installer's whims
|
|
If table(0) = "`AppSearch`" And database.TablePersistent("Signature") = 2 Then CreateTable SignatureTable
|
|
If table(0) = "`Dialog`" And database.TablePersistent("ListBox") = 2 Then CreateTable ListBoxTable
|
|
If table(0) = "`Extension`" And database.TablePersistent("Verb") = 2 Then CreateTable VerbTable
|
|
If table(0) = "`ProgId`" And database.TablePersistent("Extension") = 2 Then CreateTable ExtensionTable
|
|
|
|
If table(0) = "`File`" Then CreateTable MsiFileHashTable
|
|
End Sub
|
|
|
|
Function CreateView(table)
|
|
If Not fNoOnError Then On Error Resume Next
|
|
If database.TablePersistent(Replace(table(0),"`","")) = 2 Then CreateTable(table)
|
|
Set CreateView = database.OpenView("SELECT * FROM "& table(0)) : CheckError
|
|
CreateView.Execute : CheckError
|
|
End Function
|
|
|
|
Sub CloseView(table)
|
|
dictView.Remove(Replace(table(0),"`",""))
|
|
End Sub
|
|
|
|
Sub DoAction(table, op, row)
|
|
If Not fNoOnError Then On Error Resume Next
|
|
If IsEmpty(op) Then op = "merge" ' default supplied here rather than in the schema
|
|
|
|
If fLogOps Then
|
|
Dim format, index, name, delim
|
|
delim = ": "
|
|
For index = 1 To row.FieldCount
|
|
format = format & "{" & delim & Split(table(index),"`")(1) & "=[" & index & "]}"
|
|
delim = ", "
|
|
Next
|
|
row.StringData(0) = format
|
|
Wscript.Echo Replace(table(0),"`"," ") & op & row.FormatText
|
|
row.StringData(0) = ""
|
|
End If
|
|
|
|
' Get existing view for table, else create new one and add to view dictionary
|
|
Dim tableName : tableName = Replace(table(0),"`","")
|
|
Dim view
|
|
If dictView.Exists(tableName) Then
|
|
Set view = dictView.Item(tableName)
|
|
Else
|
|
Set view = CreateView(table)
|
|
dictView.Add tableName, view
|
|
End If
|
|
' Select the update mode for processing the row with the view
|
|
Dim modifyMode
|
|
Select Case(op)
|
|
Case "insert"
|
|
modifyMode = msiViewModifyInsert
|
|
Case "merge"
|
|
modifyMode = msiViewModifyMerge
|
|
Case "replace"
|
|
modifyMode = msiViewModifyAssign
|
|
Case "delete"
|
|
view.Modify msiViewModifySeek, row
|
|
If Err <> 0 Then Fail "'delete' item '" & row.StringData(1) & "' not present in table " & table(0)
|
|
modifyMode = msiViewModifyDelete
|
|
Case "exist"
|
|
modifyMode = msiViewModifySeek
|
|
Case "ensure"
|
|
view.Modify msiViewModifySeek, row
|
|
If Err <> 0 Then modifyMode = msiViewModifyInsert
|
|
Case Else : Fail "Invalid op attribute value: " & op
|
|
End Select
|
|
view.Modify modifyMode, row
|
|
If Err <> 0 Then Fail "Operation '" & op & "' failed for item '" & row.StringData(1) & "' in table " & table(0)
|
|
End Sub
|
|
|
|
' append module.guid (note: guid is stored in moduleId) if this is a module and not a standard property
|
|
Function ModularizeX(s)
|
|
If fModule And Len(s) > 0 And Not dictStandardProperties.Exists(s) Then ModularizeX = s & "." & moduleId Else ModularizeX = s
|
|
End Function
|
|
|
|
Function Modularize(s)
|
|
If fNoModularize Then Modularize = s Else Modularize = ModularizeX(s)
|
|
End Function
|
|
|
|
' append module.guid if this is a non-standard property and it is a module
|
|
Function ModularizeProperty(s)
|
|
Dim nStart, nPropStart, nPropEnd, sProp
|
|
|
|
If fModule And Not fNoModularize And Len(s) > 0 Then
|
|
nStart = 1
|
|
Do
|
|
nPropStart = InStr(nStart, s, "[")
|
|
nPropEnd = InStr(nStart, s, "]")
|
|
|
|
If nPropEnd > nPropStart Then
|
|
sProp = Mid(s, nPropStart + 1, nPropEnd - nPropStart - 1)
|
|
If Not dictStandardProperties.Exists(sProp) Then
|
|
s = Left(s, nPropEnd - 1) & "." & moduleId & Mid(s, nPropEnd)
|
|
nPropEnd = nPropEnd + 37 ' slide past the .GUID
|
|
End If
|
|
Else
|
|
Exit Do
|
|
End If
|
|
|
|
nStart = nPropEnd + 1' move past the close bracket
|
|
Loop
|
|
End If
|
|
ModularizeProperty = s
|
|
End Function ' ModularizeProperty
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' Linker information routines
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' WriteLinkerInfo
|
|
Sub WriteLinkerInfo
|
|
Dim vw, vw2, rec, rec2
|
|
Dim op
|
|
|
|
If 2 = database.TablePersistent("candle_Info") Then
|
|
Set vw = database.OpenView("CREATE TABLE `candle_Info` (`LinkProperty` CHAR(0) NOT NULL, `Value` CHAR(0) PRIMARY KEY `LinkProperty`)")
|
|
vw.Execute
|
|
End If
|
|
|
|
Set vw = database.OpenView("SELECT `Value` FROM `candle_Info` WHERE `LinkProperty`=?")
|
|
Set vw2 = database.OpenView("SELECT `LinkProperty`, `Value` FROM `candle_Info`")
|
|
|
|
Set rec = installer.CreateRecord(1)
|
|
rec.StringData(1) = "SourceFile"
|
|
vw.Execute rec
|
|
Set rec2 = vw.Fetch
|
|
If rec2 Is Nothing Then
|
|
Set rec2 = installer.CreateRecord(2)
|
|
rec2.StringData(1) = "SourceFile"
|
|
rec2.StringData(2) = manifestPath
|
|
vw2.Modify msiViewModifyInsert, rec2
|
|
Else
|
|
rec2.StringData(1) = manifestPath
|
|
vw.Modify msiViewModifyUpdate, rec2
|
|
End If
|
|
|
|
|
|
rec.StringData(1) = "IsModule"
|
|
vw.Execute rec
|
|
Set rec2 = vw.Fetch
|
|
If rec2 Is Nothing Then
|
|
Set rec2 = installer.CreateRecord(2)
|
|
rec2.StringData(1) = "IsModule"
|
|
If fModule Then rec2.StringData(2) = "1" Else rec2.StringData(2) = "0"
|
|
vw2.Modify msiViewModifyInsert, rec2
|
|
Else
|
|
If fModule Then rec2.StringData(1) = "1" Else rec2.StringData(1) = "0"
|
|
vw.Modify msiViewModifyUpdate, rec2
|
|
End If
|
|
|
|
WriteFileInfo
|
|
If Not fModule Then WriteModuleInfo
|
|
End Sub ' WriteLinkerInfo
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' WriteFileInfo
|
|
Sub WriteFileInfo
|
|
Dim vwDisk, vwFile, rec
|
|
Dim sKey, aData
|
|
|
|
If 0 = dictFiles.Count Then Exit Sub
|
|
|
|
If 2 = database.TablePersistent("candle_Files") Then
|
|
Set vwFile = database.OpenView("CREATE TABLE `candle_Files` (`File_` CHAR(72) NOT NULL, `Path` CHAR(0) NOT NULL PRIMARY KEY `File_`)")
|
|
vwFile.Execute
|
|
End If
|
|
Set vwFile = database.OpenView("SELECT `File_`, `Path` FROM `candle_Files`")
|
|
vwFile.Execute
|
|
|
|
If 2 = database.TablePersistent("candle_DiskInfo") Then
|
|
Set vwDisk = database.OpenView("CREATE TABLE `candle_DiskInfo` (`Identifier` CHAR(72) NOT NULL, `DiskId` INTEGER NOT NULL, `IsModule` INTEGER PRIMARY KEY `Identifier`)")
|
|
vwDisk.Execute
|
|
End If
|
|
Set vwDisk = database.OpenView("SELECT `Identifier`, `DiskId`, `IsModule` FROM `candle_DiskInfo`")
|
|
vwDisk.Execute
|
|
|
|
Set rec = installer.CreateRecord(3)
|
|
For Each sKey In dictFiles
|
|
aData = dictFiles.Item(sKey)
|
|
|
|
rec.StringData(1) = sKey
|
|
rec.StringData(2) = aData(0)
|
|
vwFile.Modify msiViewModifyInsert, rec
|
|
|
|
rec.StringData(1) = sKey
|
|
rec.IntegerData(2) = CInt(aData(1))
|
|
rec.IntegerData(3) = 0
|
|
vwDisk.Modify msiViewModifyInsert, rec
|
|
Next
|
|
|
|
End Sub ' WriteFileInfo
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' WriteModuleInfo
|
|
Sub WriteModuleInfo
|
|
Dim vwDisk, vwModule, rec
|
|
Dim sKey, aData
|
|
Dim sPrimaryFeature, aSecondaryFeatures, i, sConnectFeatures
|
|
|
|
If 0 = dictModules.Count Then Exit Sub
|
|
|
|
If 2 = database.TablePersistent("candle_Modules") Then
|
|
Set vwModule = database.OpenView("CREATE TABLE `candle_Modules` (`Module` CHAR(72) NOT NULL, `Path` CHAR(0) NOT NULL, `Language` INTEGER NOT NULL, `PrimaryFeature_` CHAR(38) NOT NULL, `ConnectFeatures_` CHAR(0), `RedirectDirectory_` CHAR(72) PRIMARY KEY `Module`)")
|
|
vwModule.Execute
|
|
End If
|
|
Set vwModule = database.OpenView("SELECT `Module`, `Path`, `Language`, `PrimaryFeature_`, `ConnectFeatures_`, `RedirectDirectory_` FROM `candle_Modules`")
|
|
vwModule.Execute
|
|
|
|
If 2 = database.TablePersistent("candle_DiskInfo") Then
|
|
Set vwDisk = database.OpenView("CREATE TABLE `candle_DiskInfo` (`Identifier` CHAR(72) NOT NULL, `DiskId` INTEGER NOT NULL, `IsModule` INTEGER PRIMARY KEY `Identifier`)")
|
|
vwDisk.Execute
|
|
End If
|
|
Set vwDisk = database.OpenView("SELECT `Identifier`, `DiskId`, `IsModule` FROM `candle_DiskInfo`")
|
|
vwDisk.Execute
|
|
|
|
Set rec = installer.CreateRecord(6)
|
|
For Each sKey In dictModules
|
|
aData = dictModules.Item(sKey)
|
|
|
|
If IsEmpty(aData(4)) Then
|
|
aSecondaryFeatures = Split(aData(5), ":")
|
|
If -1 = UBound(aSecondaryFeatures) Then Fail "Error, Module: " & sKey & " not part of any Features"
|
|
If 0 < UBound(aSecondaryFeatures) Then Fail "Error, Module: " & sKey & " is part of many Features, but no Feature is marked primary"
|
|
|
|
sPrimaryFeature = aSecondaryFeatures(0)
|
|
For i = 1 To UBound(aSecondaryFeatures)
|
|
If 1 < i Then sConnectFeatures = sConnectFeatures & ":"
|
|
sConnectFeatures = sConnectFeatures & aSecondaryFeatures(i)
|
|
Next
|
|
Else
|
|
sPrimaryFeature = aData(4)
|
|
sConnectFeatures = aData(5)
|
|
End If
|
|
|
|
rec.StringData(1) = sKey
|
|
rec.StringData(2) = aData(0)
|
|
rec.IntegerData(3) = CInt(aData(2))
|
|
rec.StringData(4) = sPrimaryFeature
|
|
rec.StringData(5) = sConnectFeatures
|
|
rec.StringData(6) = aData(3)
|
|
'WScript.Echo "X Directory: " & aData(3)
|
|
|
|
vwModule.Modify msiViewModifyInsert, rec
|
|
|
|
rec.StringData(1) = sKey
|
|
rec.IntegerData(2) = CInt(aData(1))
|
|
rec.IntegerData(3) = 1
|
|
vwDisk.Modify msiViewModifyInsert, rec
|
|
|
|
Next
|
|
End Sub ' WriteModuleInfo
|
|
|
|
|
|
'---------------------------------------------------------------------------------'
|
|
' XML parsing routines and conditional execution logic
|
|
'---------------------------------------------------------------------------------'
|
|
Function ElementHasText(node)
|
|
ElementHasText = Not node.selectSingleNode("text()") Is Nothing
|
|
End Function
|
|
|
|
Function ElementText(node)
|
|
If node Is Nothing Then Fail "passed dead node to ElementText"
|
|
Dim child : Set child = node.selectSingleNode("text()")
|
|
If child Is Nothing Then Fail "Missing text value for element: " & node.nodeName
|
|
ElementText = child.text
|
|
End Function
|
|
|
|
Function LoadDocument(path, stylePath, dictVars)
|
|
Dim xmlDoc : Set xmlDoc = WixLoad(path, Empty, dictVars, True)
|
|
|
|
If Not IsEmpty(stylePath) Then
|
|
WixApplyStyleSheet xmlDoc, stylePath
|
|
|
|
If fVerbose Then
|
|
WScript.Echo "--------------------"
|
|
WScript.Echo "Transformed manifest:"
|
|
WScript.Echo xmlDoc.xml
|
|
WScript.Echo
|
|
End If
|
|
End If
|
|
|
|
' return the root of the document
|
|
Set LoadDocument = xmlDoc.documentElement
|
|
End Function
|
|
|
|
Function GetEncoding(node)
|
|
Dim xmldecl : Set xmldecl = node.ownerDocument.selectSingleNode("pi('xml')")
|
|
If (Not xmldecl Is Nothing) Then
|
|
Dim encattr 'As IXMLDOMNode
|
|
Set encattr = xmldecl.Attributes.getNamedItem("encoding")
|
|
If Not encattr Is Nothing Then GetEncoding = encattr.Text
|
|
End If
|
|
End Function
|
|
|
|
Function NameToBit(names, name, value)
|
|
Dim index, bit
|
|
bit = 1
|
|
For index = 0 To UBound(names)
|
|
If names(index) = name Then
|
|
If value = "yes" Then NameToBit = bit Else NameToBit = 0
|
|
Exit Function
|
|
End If
|
|
If bit = &h40000000 Then bit = &h80000000 Else bit = bit + bit
|
|
Next
|
|
End Function
|
|
|
|
Function GetElementName(node)
|
|
GetElementName = Empty
|
|
If node.nodeType = NODE_ELEMENT Then GetElementName = node.nodeName
|
|
End Function
|
|
|
|
'---------------------------------------------------------------------------------'
|
|
' Non-UI element handlers
|
|
'---------------------------------------------------------------------------------'
|
|
Public productCode, productName, productLanguage, productAuthor ' product properties used as defaults for suminfo
|
|
|
|
Sub ProcessProductElement(node)
|
|
If Not fNoOnError Then On Error Resume Next
|
|
Dim child, attribute, value, op, version, sumInfo, index
|
|
|
|
' Walk XML nodes and populate tables
|
|
lastDiskId = 0
|
|
featureDisplay = 0
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : productName = value : If Not fModule Then ProcessProperty "ProductName", value, "replace"
|
|
Case "Id" : productCode = "{"&value&"}" : If Not fModule And Not IsEmpty(value) And value <> "" Then ProcessProperty "ProductCode", productCode, "replace" Else moduleId = Replace(value, "-", "_")
|
|
Case "UpgradeCode" : value = "{"&value&"}" : If Not fModule Then ProcessProperty "UpgradeCode", value, "replace"
|
|
Case "Manufacturer": productAuthor = value : If Not fModule Then ProcessProperty "Manufacturer", value, "replace"
|
|
Case "Language" : productLanguage = value : If Not fModule Then ProcessProperty "ProductLanguage", value, "replace"
|
|
Case "Version" : version = value : If Not fModule Then ProcessProperty "ProductVersion", value, "replace"
|
|
Case "xmlns" : ' ProcessProperty "XMLSchema", value, "replace"
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If openMode = fCreate And IsEmpty(productCode) Then Fail "Id attribute required for created database"
|
|
|
|
If fModule Then
|
|
Dim row : Set row = installer.CreateRecord(UBound(ModuleSignatureTable))
|
|
row.StringData (ModuleSignature_ModuleID) = ModularizeX(productName)
|
|
row.StringData (ModuleSignature_Language) = productLanguage
|
|
row.StringData (ModuleSignature_Version) = version
|
|
DoAction ModuleSignatureTable, op, row
|
|
' if there is no FeatureComponents table add it
|
|
If database.TablePersistent(Replace(FeatureComponentsTable(0),"`","")) = 2 Then CreateTable(FeatureComponentsTable)
|
|
End If
|
|
|
|
ProcessProductChildElements(node)
|
|
Set dictView = Nothing ' close all views, could also use RemoveAll method of Dictionary object
|
|
End Sub ' ProcessProductElement
|
|
|
|
|
|
Sub ProcessSummaryInformation(node)
|
|
If Not fNoOnError Then On Error Resume Next
|
|
Dim sumInfo
|
|
Dim attribute, op, value, sourceBits
|
|
Dim packageCode, packageLanguage, packageAuthor, packageName, msiVersion, keywords, comments, codepage, platform
|
|
If fCreate Then ' default unspecified package properties to product properties if creating a new package
|
|
sumInfoArray(2) = "Installation Database"
|
|
sumInfoArray(12) = Now
|
|
sumInfoArray(18) = "Windows Installer XML (candle/light)"
|
|
sumInfoArray(19) = 1 'Read-only recommended
|
|
packageCode = productCode
|
|
packageName = productName
|
|
packageAuthor = productAuthor
|
|
packageLanguage = productLanguage
|
|
msiVersion = 100 ' lowest released version, really should be specified
|
|
codepage = 0 ' neutral, really should be specified
|
|
keywords = "Installer"
|
|
sourceBits = 0
|
|
Else
|
|
Set sumInfo = database.SummaryInformation(0) : CheckError
|
|
sumInfoArray(2) = sumInfo.Property(2)
|
|
sumInfoArray(12) = sumInfo.Property(12)
|
|
sumInfoArray(18) = sumInfo.Property(18)
|
|
sumInfoArray(19) = sumInfo.Property(19)
|
|
codepage = sumInfo.Property(1)
|
|
packageName = sumInfo.Property(3)
|
|
packageAuthor = sumInfo.Property(4)
|
|
packageCode = sumInfo.Property(9)
|
|
msiVersion = sumInfo.Property(14)
|
|
sourceBits = sumInfo.Property(15)
|
|
value = Split(sumInfo.Property(7), ";")
|
|
platform = value(0)
|
|
If UBound(value) = 1 Then packageLanguage = value(1)
|
|
End If
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Value" : value = value
|
|
Case "Id" : packageCode = "{"&value&"}"
|
|
Case "Description" : packageName = value
|
|
Case "Manufacturer" : packageAuthor = value
|
|
Case "Languages" : packageLanguage = value
|
|
Case "Platforms" : platform = value
|
|
Case "InstallerVersion": msiVersion = CInt(value)
|
|
Case "Keywords" : keywords = value
|
|
Case "Comments" : comments = value
|
|
Case "SummaryCodepage" : codepage = CInt(value)
|
|
Case "ShortNames": If value="yes" Then sourceBits=sourceBits Or 1 Else If value="no" Then sourceBits=sourceBits And Not 1
|
|
Case "Compressed": If value="yes" Then sourceBits=sourceBits Or 2 Else If value="no" Then sourceBits=sourceBits And Not 2
|
|
Case "AdminImage": If value="yes" Then sourceBits=sourceBits Or 4 Else If value="no" Then sourceBits=sourceBits And Not 4
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
sumInfoArray(1) = codepage
|
|
sumInfoArray(3) = packageName
|
|
sumInfoArray(4) = packageAuthor
|
|
sumInfoArray(5) = keywords
|
|
sumInfoArray(6) = comments
|
|
sumInfoArray(7) = platform & ";" & packageLanguage
|
|
sumInfoArray(9) = packageCode
|
|
sumInfoArray(13)= Now
|
|
sumInfoArray(14)= msiVersion
|
|
sumInfoArray(15)= sourceBits
|
|
End Sub
|
|
|
|
Sub ProcessProductChildElements(node)
|
|
Dim child
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Redist" : ProcessRedistElement child
|
|
Case "Condition" : ProcessLaunchCondition child
|
|
Case "Property" : ProcessPropertyElement child
|
|
Case "Directory" : ProcessDirectoryElement child, Empty, Empty
|
|
Case "Component" : ProcessComponentElement child, Empty, Empty ' !!! passing Empty for sPath is not the right thing to do
|
|
Case "Feature" : ProcessFeatureElement child, Empty, featureDisplay
|
|
Case "Media" : ProcessMediaElement child, lastDiskId
|
|
Case "AppId" : ProcessAppIdElement child
|
|
Case "CustomAction" : ProcessCustomActionElement child
|
|
Case "CustomTable" : ProcessCustomTableElement child
|
|
Case "UI": If Not fNoUI Then ProcessUIElement child
|
|
Case "Package": If Not fNoSumInfo Then ProcessSummaryInformation child
|
|
Case "InstallExecuteSequence" : ProcessSequence InstallExecuteSequenceTable, child
|
|
Case "InstallUISequence" : ProcessSequence InstallUISequenceTable, child
|
|
Case "AdminExecuteSequence" : ProcessSequence AdminExecuteSequenceTable, child
|
|
Case "AdminUISequence" : ProcessSequence AdminUISequenceTable, child
|
|
Case "AdvtExecuteSequence" : ProcessSequence AdvtExecuteSequenceTable, child
|
|
Case "AdvertiseExecuteSequence" : ProcessSequence AdvtExecuteSequenceTable, child
|
|
Case "AdvtUISequence" : ProcessSequence AdvtUISequenceTable, child
|
|
Case "AdvertiseUISequence" : ProcessSequence AdvtUISequenceTable, child
|
|
Case "Binary" : ProcessBinaryElement child, BinaryTable
|
|
Case "Icon" : ProcessBinaryElement child, IconTable
|
|
Case "Dependency" : ProcessDependencyElement child
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' redist elements
|
|
Sub ProcessRedistElement(node)
|
|
Dim attribute, child, row, op, sDescription, sDistribution, sType
|
|
Set row = installer.CreateRecord(UBound(RedistInfoTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Distribution" : sDistribution = attribute.value
|
|
Case "Type" : sType = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(sDistribution) Or sDistribution = "internal" Then
|
|
row.IntegerData(RedistInfo_Distribution) = 0
|
|
ElseIf sDistribution = "external" Then
|
|
row.IntegerData(RedistInfo_Distribution) = 1
|
|
Else
|
|
Fail "Unexpected Redist.Distribution: " & sDistribution
|
|
End If
|
|
|
|
If IsEmpty(sType) Or sType = "retail" Then
|
|
row.IntegerData(RedistInfo_Type) = 1
|
|
ElseIf sType = "debug" Then
|
|
row.IntegerData(RedistInfo_Type) = 0
|
|
Else
|
|
Fail "Unexpected Redist.Type: " & sType
|
|
End If
|
|
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Description" : If IsEmpty(sDescription) Then sDescription = ElementText(child) Else Fail "Cannot have two Redist.Description elements"
|
|
' Case "Keyword" : ProcessRedistKeywordElement child
|
|
Case "Contact" : ProcessRedistContactElement child
|
|
Case "AllowUser" : ProcessRedistAllowUserElement child
|
|
Case "Windows9x" : ProcessOsElement child
|
|
Case "Windows32" : ProcessOsElement child
|
|
Case "Windows64" : ProcessOsElement child
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
row.StringData(RedistInfo_DummyPk) = "RedistPack"
|
|
row.StringData(RedistInfo_Description) = sDescription
|
|
DoAction RedistInfoTable, op, row
|
|
End Sub
|
|
|
|
' Sub ProcessRedistKeywordElement(node)
|
|
' Dim attribute, row, op
|
|
' Set row = installer.CreateRecord(UBound(RedistKeywordsTable))
|
|
' For Each attribute In node.Attributes
|
|
' Select Case(attribute.name)
|
|
' Case "op" : op = attribute.value
|
|
' Case Else : Unexpected attribute, node
|
|
' End Select
|
|
' Next
|
|
' row.StringData(RedistKeywords_Keyword) = ElementText(node)
|
|
' DoAction RedistKeywordsTable, op, row
|
|
' End Sub
|
|
|
|
Sub ProcessRedistContactElement(node)
|
|
Dim attribute, row, op
|
|
Set row = installer.CreateRecord(UBound(RedistContactsTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(RedistContacts_Contact) = ElementText(node)
|
|
DoAction RedistContactsTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessRedistAllowUserElement(node)
|
|
Dim attribute, row, op, sDomain, sAlias
|
|
Set row = installer.CreateRecord(UBound(RedistPermissionsTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Domain" : sDomain = attribute.value
|
|
Case "Alias" : sAlias = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(sDomain) Then sDomain = "REDMOND"
|
|
If IsEmpty(sAlias) Then Fail "Must specify an alias for Permission elements"
|
|
|
|
row.StringData(RedistPermissions_Domain) = sDomain
|
|
row.StringData(RedistPermissions_Alias) = sAlias
|
|
DoAction RedistPermissionsTable, op, row
|
|
|
|
End Sub
|
|
|
|
Sub ProcessOsElement(node)
|
|
Dim attribute, row, op
|
|
Set row = installer.CreateRecord(UBound(RedistOsTable))
|
|
|
|
row.IntegerData(RedistOs_DummyPk) = osCount
|
|
row.StringData(RedistOs_Type) = GetElementName(node)
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Flavor" : row.StringData(RedistOs_Flavor) = attribute.value
|
|
Case "Language" : row.StringData(RedistOs_Language) = attribute.value
|
|
Case "MinVersion" : row.StringData(RedistOs_MinVersion) = attribute.value
|
|
Case "MaxVersion" : row.StringData(RedistOs_MaxVersion) = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
osCount = osCount + 1
|
|
|
|
DoAction RedistOsTable, op, row
|
|
End Sub
|
|
|
|
' end redist elements
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
|
|
Sub ProcessProperty(property, value, op)
|
|
Dim row : Set row = installer.CreateRecord(UBound(PropertyTable))
|
|
row.StringData (Property_Property) = Modularize(property)
|
|
row.StringData (Property_Value) = value
|
|
DoAction PropertyTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessPropertyElement(node)
|
|
Dim attribute, op, child, control, table, property, order, value, fAppSearch, signature
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Value" : value = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
property = ElementText(node)
|
|
' see if this property is used for AppSearch
|
|
fAppSearch = False
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "IniFileSearch" : If fAppSearch Then Fail "Only one search type can appear under Property: " & property Else fAppSearch = True : signature = ProcessIniFileSearchElement (child)
|
|
Case "RegistrySearch" : If fAppSearch Then Fail "Only one search type can appear under Property: " & property Else fAppSearch = True : signature = ProcessRegistrySearchElement (child)
|
|
Case "ComponentSearch": If fAppSearch Then Fail "Only one search type can appear under Property: " & property Else fAppSearch = True : signature = ProcessComponentSearchElement(child)
|
|
Case "DirectorySearch": If fAppSearch Then Fail "Only one search type can appear under Property: " & property Else fAppSearch = True : signature = ProcessDirectorySearchElement(child, "")
|
|
Case "FileSearch" : If fAppSearch Then Fail "Only one search type can appear under Property: " & property Else fAppSearch = True : signature = ProcessFileSearchElement (child, "")
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
If fAppSearch Then
|
|
If Not IsEmpty(value) Then Fail "Cannot specify a Value for search Property: " & property
|
|
ProcessAppSearch property, signature, op
|
|
Else
|
|
ProcessProperty property, value, op
|
|
End If
|
|
End Sub
|
|
|
|
Sub ProcessLaunchCondition(node)
|
|
Dim attribute, row, op
|
|
Set row = installer.CreateRecord(UBound(LaunchConditionTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Message" : row.StringData(LaunchCondition_Description) = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(LaunchCondition_Condition) = ElementText(node)
|
|
DoAction LaunchConditionTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessDirectoryElement(node, parent, sPath)
|
|
Dim child, directory, value, attribute, row, op, name, longName, sTarget, sourceName, longSource, sSource, sDefaultDir
|
|
|
|
Dim bStdDir, bStdParent
|
|
bStdDir = False
|
|
bStdParent = False
|
|
|
|
directory = ElementText(node)
|
|
Set row = installer.CreateRecord(UBound(DirectoryTable))
|
|
' don't modularize TARGETDIR
|
|
If "TARGETDIR" = directory Or bStdDir Then row.StringData(Directory_Directory) = directory Else row.StringData(Directory_Directory) = Modularize(directory)
|
|
If "TARGETDIR" = parent Or bStdParent Then row.StringData(Directory_Directory_Parent) = parent Else row.StringData(Directory_Directory_Parent) = Modularize(parent)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : name = value
|
|
Case "LongName" : longName = value
|
|
Case "SourceName" : sourceName = value
|
|
Case "LongSource" : longSource = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If Len(name) = 0 Then Fail "Must specify a Name for Directory: " & directory
|
|
If IsEmpty(sourceName) And Not IsEmpty(longSource) Then Fail "Must specify a SourceName for Directory: " & directory
|
|
sTarget = name
|
|
If Not IsEmpty(longName) Then sTarget = sTarget & "|" & longName
|
|
sSource = sourceName
|
|
If Not IsEmpty(longSource) Then sSource = sSource & "|" & longSource
|
|
sDefaultDir = sTarget
|
|
If Not IsEmpty(sSource) Then sDefaultDir = sDefaultDir & ":" & sSource
|
|
row.StringData(Directory_DefaultDir) = sDefaultDir
|
|
REM DefaultDir required if "insert" or "replace" or "merge"
|
|
REM If IsEmpty(parent) And op <> "exist" Then Fail "Root Directory element op must be 'exist'"
|
|
DoAction DirectoryTable, op, row
|
|
|
|
' build up the path
|
|
If Not IsEmpty(sourceName) Then sPath = sPath & sourceName & "\" Else sPath = sPath & name & "\"
|
|
|
|
If fModule Then
|
|
If dictStdDirs.Exists(directory) Then bStdDir = True Else bStdDir = False
|
|
' if adding one of the standard Windows Installer directories to module
|
|
If bStdDir Then
|
|
If database.TablePersistent("CustomAction") = 2 Then CreateTable CustomActionTable
|
|
If database.TablePersistent("InstallExecuteSequence") = 2 Then CreateTable InstallExecuteSequenceTable
|
|
If database.TablePersistent("InstallUISequence") = 2 Then CreateTable InstallUISequenceTable
|
|
If database.TablePersistent("AdminExecuteSequence") = 2 Then CreateTable AdminExecuteSequenceTable
|
|
If database.TablePersistent("AdminUISequence") = 2 Then CreateTable AdminUISequenceTable
|
|
If database.TablePersistent("AdvtExecuteSequence") = 2 Then CreateTable AdvtExecuteSequenceTable
|
|
If database.TablePersistent("AdvtUISequence") = 2 Then CreateTable AdvtUISequenceTable
|
|
End If
|
|
End If
|
|
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Directory" : ProcessDirectoryElement child, (directory), (sPath)
|
|
Case "Module" : ProcessDirectoryModuleElement child, (directory)
|
|
Case "Component" : ProcessComponentElement child, (directory), (sPath)
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub
|
|
|
|
Sub ProcessCreateFolderElement(node, component, directory)
|
|
Dim op, row, attribute, child
|
|
Set row = installer.CreateRecord(UBound(CreateFolderTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Directory" : directory = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (CreateFolder_Directory_) = Modularize(directory)
|
|
row.StringData (CreateFolder_Component_) = Modularize(component)
|
|
DoAction CreateFolderTable, op, row
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Shortcut" : ProcessShortcutElement child, component, "[" & directory & "]"
|
|
Case "Permission" : ProcessPermissionElement child, directory, "CreateFolder"
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub
|
|
|
|
Sub ProcessCopyFileElement(node, component, fileId)
|
|
Dim op, row, attribute, value
|
|
Dim table, destFile, destDir, destFileColumn, destDirColumn, sourceFile, sourceDir, bits
|
|
bits = 0
|
|
If IsEmpty(fileId) Then
|
|
table = MoveFileTable
|
|
destFileColumn = MoveFile_DestName
|
|
destDirColumn = MoveFile_DestFolder
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "SourceFolder" : sourceDir = value
|
|
Case "SourceName" : sourceFile = value
|
|
Case "DestinationFolder" : destDir = value
|
|
Case "DestinationName" : destFile = value
|
|
Case "Delete" : If value = "yes" Then bits = 1
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
Else
|
|
table = DuplicateFileTable
|
|
destFileColumn = DuplicateFile_DestName
|
|
destDirColumn = DuplicateFile_DestFolder
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "DestinationFolder" : destDir = value
|
|
Case "DestinationName" : destFile = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
End If
|
|
Set row = installer.CreateRecord(UBound(table))
|
|
row.StringData(MoveFile_FileKey) = Modularize(ElementText(node))
|
|
row.StringData(MoveFile_Component_) = Modularize(component)
|
|
row.StringData(destFileColumn) = destFile
|
|
row.StringData(destDirColumn) = Modularize(destDir)
|
|
If IsEmpty(fileId) Then
|
|
row.StringData (MoveFile_SourceName) = sourceFile
|
|
row.StringData (MoveFile_SourceFolder) = Modularize(sourceDir)
|
|
row.IntegerData(MoveFile_Options) = bits
|
|
Else
|
|
row.StringData(DuplicateFile_File_) = Modularize(fileId)
|
|
End If
|
|
DoAction table, op, row
|
|
End Sub
|
|
|
|
Sub ProcessReserveCostElement(node, component, directory)
|
|
Dim op, row, attribute, value
|
|
Set row = installer.CreateRecord(UBound(ReserveCostTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Directory" : directory = value
|
|
Case "RunLocal" : row.StringData(ReserveCost_ReserveLocal) = CLng(value)
|
|
Case "RunFromSource" : row.StringData(ReserveCost_ReserveSource) = CLng(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (ReserveCost_ReserveKey) = Modularize(ElementText(node)) ' !! need to auto-generate
|
|
row.StringData (ReserveCost_Component_) = Modularize(component)
|
|
row.StringData (ReserveCost_ReserveFolder) = Modularize(directory)
|
|
DoAction ReserveCostTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessIsolateComponentElement(node, component)
|
|
Dim op, row, attribute
|
|
Set row = installer.CreateRecord(UBound(IsolatedComponentTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (IsolatedComponent_Component_Shared) = Modularize(ElementText(node))
|
|
row.StringData (IsolatedComponent_Component_Application) = Modularize(component)
|
|
DoAction IsolatedComponentTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessComponentElement(node, directory, sPath)
|
|
Dim op, row, attribute, value, child
|
|
Dim component, keyPath, keyFound, keyPossible, bits, keyBit, keyBits, comPlusBits, condition, nDiskId
|
|
Set row = installer.CreateRecord(UBound(ComponentTable))
|
|
bits = 0
|
|
nDiskId = 0
|
|
component = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Id" : if value<>"" Then row.StringData(Component_ComponentId) = "{" & value & "}" Else row.StringData(Component_ComponentId) = ""
|
|
Case "SharedDllRefCount" : If value="yes" Then bits = bits Or msidbComponentAttributesSharedDllRefCount
|
|
Case "Permanent" : If value="yes" Then bits = bits Or msidbComponentAttributesPermanent
|
|
Case "Transitive" : If value="yes" Then bits = bits Or msidbComponentAttributesTransitive
|
|
Case "NeverOverwrite" : If value="yes" Then bits = bits Or msidbComponentAttributesNeverOverwrite
|
|
Case "KeyPath" : If value="yes" Then ' Directory_ is KeyPath
|
|
keyFound = "yes"
|
|
keyPath = directory
|
|
keyBits = keyBit
|
|
End If
|
|
Case "ComPlusFlags" : comPlusBits = CInt(value)
|
|
Case "Location"
|
|
If value="source" Then
|
|
bits = bits Or msidbComponentAttributesSourceOnly
|
|
ElseIf value="either" Then
|
|
bits = bits Or msidbComponentAttributesOptional
|
|
End If
|
|
Case "DiskId" : nDiskId = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(directory) And op <> "exist" Then Fail "Root Component element op must be 'exist'"
|
|
For Each child In node.childNodes
|
|
keyPossible = "no"
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "File" : keyPossible = ProcessFileElement (child, component, (nDiskId), sPath) : keyBit = 0
|
|
Case "Registry" : keyPossible = ProcessRegistryElement(child, component) : keyBit = msidbComponentAttributesRegistryKeyPath
|
|
Case "ODBCDataSource" : keyPossible = ProcessODBCDataSource (child, component, Empty) : keyBit = msidbComponentAttributesODBCDataSource
|
|
Case "ODBCDriver" : ProcessODBCDriver child, component, Empty, ODBCDriverTable
|
|
Case "ODBCTranslator" : ProcessODBCDriver child, component, Empty, ODBCTranslatorTable
|
|
Case "TypeLib" : Call ProcessTypeLibElement (child, component, Empty)
|
|
Case "Shortcut" : ProcessShortcutElement child, component, "[" & directory & "]"
|
|
Case "IniFile" : ProcessIniElement child, component
|
|
Case "CreateFolder" : ProcessCreateFolderElement child, component, (directory)
|
|
Case "CopyFile" : ProcessCopyFileElement child, component, Empty
|
|
Case "IsolateComponent" : ProcessIsolateComponentElement child, component
|
|
Case "ReserveCost" : ProcessReserveCostElement child, component, (directory)
|
|
Case "RemoveFile" : ProcessRemoveFileElement child, component, (directory)
|
|
Case "Environment" : ProcessEnvironmentElement child, component
|
|
Case "ServiceControl" : ProcessServiceControlElement child, component
|
|
Case "ServiceInstall" : ProcessServiceInstallElement child, component
|
|
Case "Class" : ProcessClassElement child, component, Empty ' no feature, no advertise
|
|
Case "Condition"
|
|
If Not IsEmpty(condition) Then Fail "Can only have one Condition for a component"
|
|
condition = ElementText(child)
|
|
'!! need to make sure no attributes
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
|
|
If keyPossible = "yes" And keyFound = "yes" Then
|
|
Fail "Component has more than one KeyPath: " & component
|
|
ElseIf keyPossible = "yes" Or (keyFound = Empty And keyPossible <> "no") Then
|
|
keyFound = keyPossible
|
|
keyPath = ElementText(child)
|
|
keyBits = keyBit
|
|
End If
|
|
Next
|
|
|
|
If keyFound = "no" Or keyFound = "noreg" Then Fail "Component has no KeyPath element and there is more than one choice: " & component
|
|
row.StringData (Component_Component) = Modularize(component)
|
|
row.StringData (Component_Directory_) = Modularize(directory)
|
|
row.StringData (Component_Condition) = condition
|
|
row.IntegerData(Component_Attributes) = bits Or keyBits
|
|
row.StringData (Component_KeyPath) = Modularize(keyPath)
|
|
DoAction ComponentTable, op, row
|
|
If Not IsEmpty(comPlusBits) Then
|
|
row.ClearData
|
|
row.StringData (Complus_Component_) = component
|
|
row.IntegerData(Complus_ExpType) = comPlusBits
|
|
DoAction ComplusTable, op, row
|
|
End If
|
|
If fModule Then
|
|
Set row = installer.CreateRecord(UBound(ModuleComponentsTable))
|
|
row.StringData (ModuleComponents_Component)= Modularize(component)
|
|
row.StringData (ModuleComponents_ModuleID) = ModularizeX(productName)
|
|
row.IntegerData(ModuleComponents_Language) = CInt(productLanguage)
|
|
DoAction ModuleComponentsTable, op, row
|
|
End If
|
|
End Sub
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' ProcessDirectoryModuleElement
|
|
Sub ProcessDirectoryModuleElement(node, directory)
|
|
Dim attribute, value
|
|
Dim sName, nLang, src, nDiskId
|
|
Dim aModuleData(5)
|
|
|
|
' not valid in schema, but double check anyway
|
|
If fModule Then Fail "Cannot specify a Module inside a Module"
|
|
If IsEmpty(directory) Then Fail "Module must be found under a Directory element"
|
|
|
|
sName = ElementText(node)
|
|
nDiskId = 0
|
|
nLang = 0 ' default language is "neutral"
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : Fail "robmen - ProcessDirectoryModuleElement() - op attribute currently unsupported in this context"
|
|
Case "Language" : nLang = CInt(value)
|
|
Case "DiskId" : nDiskId = value
|
|
Case "src" : src = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If Not fModule And 0 = nDiskId Then Fail "Must specify a DiskId for Module: " & sName
|
|
If IsEmpty(src) Then Fail "Must specify a 'src' for every Module"
|
|
|
|
' if the Module was already defined in Feature tree
|
|
If dictModules.Exists(sName) Then
|
|
Dim aData
|
|
aData = dictModules.Item(sName)
|
|
If IsEmpty(aData(0)) And IsEmpty(aData(2)) And IsEmpty(aData(3)) Then
|
|
aData(0) = src
|
|
aData(1) = nDiskId
|
|
aData(2) = nLang
|
|
If "targetdir" = LCase(directory) Then aData(3) = Empty Else aData(3) = directory
|
|
'WScript.Echo "Y Directory: " & aData(3)
|
|
|
|
dictModules.Item(sName) = aData
|
|
Else
|
|
Fail "Cannot merge same Module twice: " & sName
|
|
End If
|
|
Else
|
|
aModuleData(0) = src
|
|
aModuleData(1) = nDiskId
|
|
aModuleData(2) = nLang
|
|
If "targetdir" = LCase(directory) Then aModuleData(3) = Empty Else aModuleData(3) = directory
|
|
aModuleData(4) = Empty ' no primary Feature yet
|
|
aModuleData(5) = Empty ' no secondary Features yet
|
|
'WScript.Echo "Z Directory: " & aModuleData(3)
|
|
|
|
dictModules.Add sName, aModuleData
|
|
End If
|
|
End Sub ' ProcessDirectoryModuleElement
|
|
|
|
Sub ProcessFeatureElement(node, parent, lastDisplay)
|
|
Dim child, value, attribute, row, bits, op, feature, display, childDisplay
|
|
Set row = installer.CreateRecord(UBound(FeatureTable))
|
|
bits = 0
|
|
If fModule Then
|
|
feature = "{00000000-0000-0000-0000-000000000000}"
|
|
Else
|
|
feature = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Display" : display = value
|
|
Case "Title" : row.StringData (Feature_Title) = value
|
|
Case "Description" : row.StringData (Feature_Description) = value
|
|
Case "Level" : row.IntegerData(Feature_Level) = CInt(value)
|
|
Case "ConfigurableDirectory" : row.StringData (Feature_Directory_) = value
|
|
Case "InstallDefault" : If value="source" Then bits = bits Or msidbFeatureAttributesFavorSource
|
|
Case "TypicalDefault" : If value="advertise" Then bits = bits Or msidbFeatureAttributesFavorAdvertise
|
|
Case "FollowParent" : If value="yes" Then bits = bits Or msidbFeatureAttributesFollowParent
|
|
Case "Absent" : If value="disallow" Then bits = bits Or msidbFeatureAttributesUIDisallowAbsent
|
|
Case "AllowAdvertise"
|
|
If value="no" Then
|
|
bits = bits Or msidbFeatureAttributesDisallowAdvertise
|
|
ElseIf value="system" Then
|
|
bits = bits Or msidbFeatureAttributesNoUnsupportedAdvertise
|
|
End If
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(display) Then display = "collapse"
|
|
Select Case(display)
|
|
Case "hidden" : display = 0
|
|
Case "expand" : lastDisplay = (lastDisplay + 1) Or 1 : display = lastDisplay
|
|
Case "collapse" : lastDisplay = (lastDisplay Or 1) + 1 : display = lastDisplay
|
|
Case Else : Fail "Unexpected Feature Display value: " & display
|
|
End Select
|
|
|
|
row.StringData (Feature_Feature) = feature
|
|
row.StringData (Feature_Feature_Parent) = parent
|
|
row.IntegerData(Feature_Attributes) = bits
|
|
row.IntegerData(Feature_Display) = display
|
|
DoAction FeatureTable, op, row
|
|
End If
|
|
childDisplay = 0
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Feature" : ProcessFeatureElement child, feature, childDisplay
|
|
Case "Condition" : ProcessFeatureCondition child, feature
|
|
Case "Module" : ProcessFeatureModule child, feature
|
|
Case "Component" : ProcessFeatureComponent child, feature
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' ProcessFeatureCondition
|
|
Sub ProcessFeatureCondition(node, feature)
|
|
Dim attribute, child, row, op, level
|
|
Set row = installer.CreateRecord(UBound(ConditionTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Level" : level = CInt(attribute.value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (Condition_Feature_) = feature
|
|
row.IntegerData(Condition_Level) = level
|
|
row.StringData (Condition_Condition) = ElementText(node)
|
|
DoAction ConditionTable, op, row
|
|
End Sub ' ProcessFeatureCondition
|
|
|
|
Sub ProcessFeatureComponent(node, feature)
|
|
Dim child, component, attribute, row, op
|
|
Set row = installer.CreateRecord(UBound(FeatureComponentsTable))
|
|
component = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Location" ' ignore default value passed by XML parser
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Category" : ProcessCategoryElement child, component, feature
|
|
Case "Class" : ProcessClassElement child, component, feature
|
|
Case "Shortcut" : ProcessShortcutElement child, component, (feature)
|
|
Case "Extension" : ProcessExtensionElement child, component, feature, Empty
|
|
Case "ProgId" : ProcessProgIdElement child, component, feature, Empty, Empty, Empty
|
|
Case "Assembly" : ProcessAssemblyElement child, component, feature
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(component) Then Fail "Missing Component key name"
|
|
If Not fModule Then
|
|
row.StringData (FeatureComponents_Feature_) = feature
|
|
row.StringData (FeatureComponents_Component_) = component
|
|
DoAction FeatureComponentsTable, op, row
|
|
End If
|
|
End Sub
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' ProcessFeatureModule
|
|
Sub ProcessFeatureModule(node, feature)
|
|
Dim attribute
|
|
Dim sName, fPrimary
|
|
Dim aModuleData(5)
|
|
|
|
' not valid in schema, but double check anyway
|
|
If fModule Then Fail "Cannot specify a Module inside a Module"
|
|
|
|
sName = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : Fail "robmen - ProcessFeatureModule() - op attribute currently unsupported in this context"
|
|
Case "Primary" : If "yes" = attribute.value Then fPrimary = True
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
' if the Module was already defined in Directory or Feature tree
|
|
If dictModules.Exists(sName) Then
|
|
Dim aData
|
|
aData = dictModules.Item(sName)
|
|
If fPrimary Then
|
|
If Not IsEmpty(aData(4)) Then Fail "Cannot specify two 'primary' Features for Module: " & sName
|
|
aData(4) = feature
|
|
If 0 = Len(aData(5)) Then aData(5) = Empty
|
|
Else ' not the primary feature
|
|
If 0 = Len(aData(5)) Then
|
|
aData(5) = feature
|
|
Else
|
|
aData(5) = aData(5) & ":" & feature
|
|
End If
|
|
End If
|
|
'WScript.Echo "A Directory: " & aData(3)
|
|
|
|
dictModules.Item(sName) = aData
|
|
Else
|
|
aModuleData(0) = Empty
|
|
aModuleData(1) = 0 ' invalid DiskId
|
|
aModuleData(2) = Empty
|
|
aModuleData(3) = Empty
|
|
If fPrimary Then
|
|
aModuleData(4) = feature
|
|
If 0 = Len(aModuleData(5)) Then aModuleData(5) = Empty
|
|
Else
|
|
aModuleData(4) = Empty
|
|
If 0 = Len(aModuleData(5)) Then
|
|
aModuleData(5) = feature
|
|
Else
|
|
aModuleData(5) = aModuleData(5) & ":" & feature
|
|
End If
|
|
End If
|
|
'WScript.Echo "B Directory: " & aModuleData(3)
|
|
|
|
dictModules.Add sName, aModuleData
|
|
End If
|
|
End Sub ' ProcessFeatureModule
|
|
|
|
Function ProcessFileElement(node, component, nDiskId, sPath)
|
|
Dim op, value, attribute, child, row, bits, fileId, bindPath, selfRegCost, shortName, longName, nFileSize, fontTitle, nSequence, src
|
|
Dim aData(1)
|
|
|
|
Set row = installer.CreateRecord(UBound(FileTable))
|
|
nFileSize = 0
|
|
bits = 0
|
|
nSequence = Empty
|
|
ProcessFileElement = "file"
|
|
fileId = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : shortName = value
|
|
Case "LongName" : longName = value
|
|
Case "FileSize" : nFileSize = CLng(value)
|
|
Case "Version" : row.StringData (File_Version) = value
|
|
Case "Language" : row.StringData (File_Language) = value
|
|
Case "Sequence" : nSequence = CInt(value)
|
|
Case "ReadOnly" : If value="yes" Then bits = bits Or msidbFileAttributesReadOnly
|
|
Case "Hidden" : If value="yes" Then bits = bits Or msidbFileAttributesHidden
|
|
Case "System" : If value="yes" Then bits = bits Or msidbFileAttributesSystem
|
|
Case "Vital" : If value="yes" Then bits = bits Or msidbFileAttributesVital
|
|
Case "Checksum" : If value="yes" Then bits = bits Or msidbFileAttributesChecksum
|
|
Case "PatchAdded" : If value="yes" Then bits = bits Or msidbFileAttributesPatchAdded
|
|
Case "Noncompressed" : If value="yes" Then bits = bits Or msidbFileAttributesNoncompressed
|
|
Case "Compressed" : If value="yes" Then bits = bits Or msidbFileAttributesCompressed
|
|
Case "KeyPath" : ProcessFileElement = value
|
|
Case "BindPath" : bindPath = value
|
|
Case "SelfRegCost" : selfRegCost = value
|
|
Case "TrueType" : If value="yes" Then fontTitle = ""
|
|
Case "FontTitle" : fontTitle = value
|
|
Case "DiskId" : nDiskId = value
|
|
Case "src" : src = value ' BaseDir() is called later, not now
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(nSequence) Then nSequence = fileSeq : fileSeq = fileSeq + 1 Else fileSeq = nSequence + 1
|
|
If fModule And 0 <> nDiskId Then Fail "Cannot specify a DiskId when compiling a Module" ' not allowed by schema but check anyway
|
|
If Not fModule And 0 = nDiskId Then Fail "Must specify a DiskId for File: " & fileId
|
|
|
|
If Not IsEmpty(bindPath) Then ProcessBindImage fileId, bindPath, op
|
|
If Not IsEmpty(selfRegCost) Then ProcessSelfReg fileId, selfregCost, op
|
|
If Not IsEmpty(fontTitle) Then ProcessFont fileId, fontTitle, op
|
|
|
|
row.StringData(File_File) = Modularize(fileId)
|
|
row.StringData(File_Component_) = Modularize(component)
|
|
If IsEmpty(longName) Then row.StringData(File_FileName) = shortName Else row.StringData(File_FileName) = shortName & "|" & longName
|
|
row.IntegerData(File_FileSize) = nFileSize
|
|
row.IntegerData(File_Attributes) = bits
|
|
row.IntegerData(File_Sequence) = nSequence
|
|
DoAction FileTable, op, row
|
|
|
|
' add to build dictionary
|
|
If 0 < Len(src) Then
|
|
If "\" = Right(src, 1) Then
|
|
If IsEmpty(longName) Then src = src & shortName Else src = src & longName
|
|
End If
|
|
aData(0) = src
|
|
Else
|
|
If IsEmpty(longName) Then aData(0) = sPath & shortName Else aData(0) = sPath & longName
|
|
End If
|
|
aData(1) = nDiskId
|
|
dictFiles.Add Modularize(fileId), aData
|
|
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Shortcut" : ProcessShortcutElement child, component, "[#" & fileId & "]"
|
|
Case "CopyFile" : ProcessCopyFileElement child, component, Modularize(fileId)
|
|
Case "ODBCDriver" : ProcessODBCDriver child, component, Modularize(fileId), ODBCDriverTable
|
|
Case "ODBCTranslator" : ProcessODBCDriver child, component, Modularize(fileId), ODBCTranslatorTable
|
|
Case "Permission" : ProcessPermissionElement child, Modularize(fileId), "File"
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Function
|
|
|
|
Sub ProcessBindImage(file, path, op)
|
|
Dim row : Set row = installer.CreateRecord(UBound(BindImageTable))
|
|
row.StringData (BindImage_File_) = Modularize(file)
|
|
row.StringData (BindImage_Path) = ModularizeProperty(path)
|
|
DoAction BindImageTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessSelfReg(file, cost, op)
|
|
Dim row : Set row = installer.CreateRecord(UBound(SelfRegTable))
|
|
row.StringData (SelfReg_File_) = Modularize(file)
|
|
row.StringData (SelfReg_Cost) = cost
|
|
DoAction SelfRegTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessFont(file, fontTitle, op)
|
|
Dim row : Set row = installer.CreateRecord(UBound(FontTable))
|
|
row.StringData (Font_File_) = Modularize(file)
|
|
row.StringData (Font_FontTitle) = fontTitle
|
|
DoAction FontTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessRemoveFileElement(node, component, directory)
|
|
Dim op, row, attribute, value
|
|
Set row = installer.CreateRecord(UBound(RemoveFileTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Directory" : directory = value
|
|
Case "Name" : row.StringData(RemoveFile_FileName) = value
|
|
Case "On"
|
|
Select Case(value)
|
|
Case "install" : value = 1
|
|
Case "uninstall" : value = 2
|
|
Case "both" : value = 3
|
|
Case Else : Fail "Unexpected value for RemoveFile 'On' attribute: " & value
|
|
End Select
|
|
row.IntegerData(RemoveFile_InstallMode) = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (RemoveFile_FileKey) = Modularize(ElementText(node)) ' !! need to auto-generate
|
|
row.StringData (RemoveFile_Component_) = Modularize(component)
|
|
row.StringData (RemoveFile_DirProperty) = Modularize(directory)
|
|
DoAction RemoveFileTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessPermissionElement(node, tableKey, tableName)
|
|
Dim value, attribute, row, op, source, target, bit, bits, specialPermissions
|
|
Set row = installer.CreateRecord(UBound(LockPermissionsTable))
|
|
bits = CLng(0)
|
|
Select Case(tableName)
|
|
Case "File" : specialPermissions = filePermissions
|
|
Case "CreateFolder" : specialPermissions = folderPermissions
|
|
Case "Registry" : specialPermissions = registryPermissions
|
|
Case Else : Fail "Invalid parent element type for Permission: " & tableName
|
|
End Select
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Domain" : row.StringData(LockPermissions_Domain) = value
|
|
Case "User" : row.StringData(LockPermissions_User) = value
|
|
Case Else
|
|
bit = NameToBit(standardPermissions, attribute.name, value)
|
|
If Not IsEmpty(bit) Then
|
|
bit = bit * 65536
|
|
Else
|
|
bit = NameToBit(genericPermissions, attribute.name, value)
|
|
If Not IsEmpty(bit) Then
|
|
If bit = 8 Then bit = &h80000000 Else bit = bit * &h10000000
|
|
Else
|
|
bit = NameToBit(specialPermissions, attribute.name, value)
|
|
If IsEmpty(bit) Then Unexpected attribute, node
|
|
End If
|
|
End If
|
|
bits = bits Or bit
|
|
End Select
|
|
Next
|
|
row.StringData (LockPermissions_LockObject) = Modularize(tableKey)
|
|
row.StringData (LockPermissions_Table) = tableName
|
|
row.IntegerData(LockPermissions_Permission) = bits
|
|
DoAction LockPermissionsTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessCategoryElement(node, component, feature)
|
|
Dim value, attribute, child, row, op
|
|
Set row = installer.CreateRecord(UBound(PublishComponentTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Qualifier" : row.StringData (PublishComponent_Qualifier) = value
|
|
Case "AppData" : row.StringData (PublishComponent_AppData) = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(PublishComponent_ComponentId) = ElementText(node)
|
|
row.StringData(PublishComponent_Component_) = Modularize(component)
|
|
row.StringData(PublishComponent_Feature_) = feature
|
|
DoAction PublishComponentTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessShortcutElement(node, component, target)
|
|
Dim value, attribute, child, row, op, shortName, longName
|
|
Set row = installer.CreateRecord(UBound(ShortcutTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Target" : target = value
|
|
Case "Name" : shortName = value
|
|
Case "LongName" : longName = value
|
|
Case "Directory" : row.StringData (Shortcut_Directory_) = Modularize(value)
|
|
Case "Description" : row.StringData (Shortcut_Description) = value
|
|
Case "Arguments" : row.StringData (Shortcut_Arguments) = value
|
|
Case "Hotkey" : row.IntegerData(Shortcut_Hotkey) = CInt(value)
|
|
Case "Icon" : row.StringData (Shortcut_Icon_) = value
|
|
Case "IconIndex" : row.IntegerData(Shortcut_IconIndex) = CInt(value)
|
|
Case "Show" : If value = "normal" Then row.IntegerData(Shortcut_ShowCmd) = 1 Else If value = "maximized" Then row.IntegerData(Shortcut_ShowCmd) = 3 Else If value = "minimized" Then row.IntegerData(Shortcut_ShowCmd) = 7
|
|
Case "WorkingDirectory" : row.StringData (Shortcut_WkDir) = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(Shortcut_Shortcut) = Modularize(ElementText(node))
|
|
row.StringData(Shortcut_Component_) = Modularize(component)
|
|
row.StringData(Shortcut_Target) = ModularizeProperty(target)
|
|
If Not IsEmpty(longName) Then shortName = shortName & "|" & longName
|
|
row.StringData(Shortcut_Name) = shortName
|
|
DoAction ShortcutTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessIniElement(node, component)
|
|
Dim value, attribute, row, op, table, action
|
|
Set row = installer.CreateRecord(UBound(IniFileTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Action" : action = value
|
|
Case "Name" : row.StringData (IniFile_FileName) = value
|
|
Case "Directory" : row.StringData (IniFile_DirProperty) = Modularize(value)
|
|
Case "Section" : row.StringData (IniFile_Section) = value
|
|
Case "Key" : row.StringData (IniFile_Key) = value
|
|
Case "Value" : row.StringData (IniFile_Value) = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
Select Case(action)
|
|
Case "addLine" : action = msidbIniFileActionAddLine
|
|
Case "createLine" : action = msidbIniFileActionCreateLine
|
|
Case "addTag" : action = msidbIniFileActionAddTag
|
|
Case "removeLine" : action = msidbIniFileActionRemoveLine
|
|
Case "removeTag" : action = msidbIniFileActionRemoveTag
|
|
Case Else : Fail "Unexpected IniFile action: " & action
|
|
End Select
|
|
row.StringData (IniFile_IniFile) = Modularize(ElementText(node)) '!!! auto-generate?
|
|
If action = msidbIniFileActionRemoveLine Or action = msidbIniFileActionRemoveTag Then
|
|
table = RemoveIniFileTable
|
|
Else
|
|
table = IniFileTable
|
|
End If
|
|
DoAction table, op, row
|
|
End Sub
|
|
|
|
Sub ProcessEnvironmentElement(node, component)
|
|
Dim op, row, attribute, value, text, system, name, action, uninstall, separator, part
|
|
Set row = installer.CreateRecord(UBound(EnvironmentTable))
|
|
uninstall = "-" ' default to remove at uninstall
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : name = value
|
|
Case "Value" : text = value
|
|
Case "Separator" : separator = value
|
|
Case "Part" : part = value
|
|
Case "System" : If value = "yes" Then system = "*"
|
|
Case "Permanent" : If value = "yes" Then uninstall = Empty
|
|
Case "Action"
|
|
Select Case(value)
|
|
Case "create" : action = "="
|
|
Case "set" : action = "+"
|
|
Case "remove" : action = "!"
|
|
Case Else : Fail "Unexpected Environment Action value: " & value
|
|
End Select
|
|
End Select
|
|
Next
|
|
Select Case(part)
|
|
Case Empty
|
|
Case "all"
|
|
Case "first" : text = text & delim & "[~]"
|
|
Case "last" : text = "[~]" & delim & text
|
|
Case Else : Fail "Unexpected Environment Part value: " & part
|
|
End Select
|
|
row.StringData (Environment_Environment) = Modularize(ElementText(node)) ' !! need to auto-generate
|
|
row.StringData (Environment_Component_) = Modularize(component)
|
|
row.StringData (Environment_Name) = action & uninstall & system & name
|
|
row.StringData (Environment_Value) = text
|
|
DoAction EnvironmentTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessServiceControlElement(node, component)
|
|
Dim child, op, row, attribute, value, name, events, wait, arguments
|
|
Set row = installer.CreateRecord(UBound(ServiceControlTable))
|
|
events = 0 ' default do nothing
|
|
wait = ""
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : name = value
|
|
Case "Start"
|
|
Select Case(value)
|
|
Case "install" : events = events Or msidbServiceControlEventStart
|
|
Case "uninstall" : events = events Or msidbServiceControlEventUninstallStart
|
|
Case "both" : events = events Or msidbServiceControlEventStart Or msidbServiceControlEventUninstallStart
|
|
Case Else : Fail "Unknown Service start type: " & value
|
|
End Select
|
|
Case "Stop"
|
|
Select Case(value)
|
|
Case "install" : events = events Or msidbServiceControlEventStop
|
|
Case "uninstall" : events = events Or msidbServiceControlEventUninstallStop
|
|
Case "both" : events = events Or msidbServiceControlEventStop Or msidbServiceControlEventUninstallStop
|
|
Case Else : Fail "Unknown Service stop type: " & value
|
|
End Select
|
|
Case "Remove"
|
|
Select Case(value)
|
|
Case "install" : events = events Or msidbServiceControlEventRemove
|
|
Case "uninstall" : events = events Or msidbServiceControlEventUninstallRemove
|
|
Case "both" : events = events Or msidbServiceControlEventRemove Or msidbServiceControlEventUninstallRemove
|
|
Case Else : Fail "Unknown Service remove type: " & value
|
|
End Select
|
|
Case "Wait"
|
|
Select Case(value)
|
|
Case "yes" : wait = "1" ' strings used since integer column is nullable
|
|
Case "no" : wait = "0"
|
|
Case Else : Fail "Unknown Wait value: " & value
|
|
End Select
|
|
End Select
|
|
Next
|
|
' get the ServiceControl arguments
|
|
arguments = ""
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "ServiceArgument"
|
|
If Len(arguments) > 0 Then arguments = arguments & "[~]"
|
|
arguments = arguments & ElementText(child)
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
row.StringData (ServiceControl_ServiceControl) = Modularize(ElementText(node)) ' !! need to auto-generate
|
|
row.StringData (ServiceControl_Name) = name
|
|
row.IntegerData(ServiceControl_Event) = events
|
|
row.StringData (ServiceControl_Arguments) = arguments
|
|
row.StringData (ServiceControl_Wait) = wait
|
|
row.StringData (ServiceControl_Component_) = Modularize(component)
|
|
DoAction ServiceControlTable, op, row
|
|
End Sub ' ProcessServiceControlElement
|
|
|
|
Sub ProcessServiceInstallElement(node, component)
|
|
Dim child, op, row, attribute, value, typebits, errorbits, erasedesc, dependencies
|
|
Set row = installer.CreateRecord(UBound(ServiceInstallTable))
|
|
typebits = 0
|
|
errorbits = 0
|
|
erasedesc = False ' don't erase the description
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : row.StringData (ServiceInstall_Name) = value
|
|
Case "DisplayName" : row.StringData (ServiceInstall_DisplayName) = value
|
|
Case "Type"
|
|
Select Case(value)
|
|
Case "ownProcess" : typebits = typebits Or msidbServiceInstallOwnProcess
|
|
Case "shareProcess" : typebits = typebits Or msidbServiceInstallShareProcess
|
|
Case "kernelDriver" : Fail "Service type not currently supported by the Windows Installer: " & value ' = 1
|
|
Case "systemDriver" : Fail "Service type not currently supported by the Windows Installer: " & value ' = 2
|
|
Case Else : Fail "Unknown Service type: " & value
|
|
End Select
|
|
Case "Interactive" : If "yes" = value Then typebits = typebits Or msidbServiceInstallInteractive
|
|
Case "Start"
|
|
Select Case(value)
|
|
Case "auto" : row.IntegerData(ServiceInstall_StartType) = msidbServiceInstallAutoStart
|
|
Case "demand" : row.IntegerData(ServiceInstall_StartType) = msidbServiceInstallDemandStart
|
|
Case "disabled" : row.IntegerData(ServiceInstall_StartType) = msidbServiceInstallDisabled
|
|
Case "boot" : Fail "Service start type not currently supported by the Windows Installer: " & value ' = 0
|
|
Case "system" : Fail "Service start type not currently supported by the Windows Installer: " & value ' = 1
|
|
Case Else : Fail "Unknown Service start type: " & value
|
|
End Select
|
|
Case "ErrorControl"
|
|
Select Case(value)
|
|
Case "ignore" : errorbits = errorbits Or msidbServiceInstallErrorIgnore
|
|
Case "normal" : errorbits = errorbits Or msidbServiceInstallErrorNormal
|
|
Case "critical" : errorbits = errorbits Or msidbServiceInstallErrorCritical
|
|
Case Else : Fail "Unknown Service error control type: " & value
|
|
End Select
|
|
Case "Vital" : If "yes" = value Then errorbits = errorbits Or msidbServiceInstallErrorControlVital
|
|
Case "LocalGroup" : row.StringData (ServiceInstall_LoadOrderGroup) = value
|
|
Case "Account" : row.StringData (ServiceInstall_StartName) = value
|
|
Case "Password" : row.StringData (ServiceInstall_Password) = value
|
|
Case "Arguments" : row.StringData (ServiceInstall_Arguments) = value
|
|
Case "Description" : row.StringData (ServiceInstall_Description) = value
|
|
Case "EraseDescription": If "yes"=value Then erasedesc = True
|
|
End Select
|
|
Next
|
|
If erasedesc Then row.StringData (ServiceInstall_Description) = "[~]"
|
|
|
|
' get the ServiceInstall dependencies
|
|
dependencies = ""
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "ServiceDependency"
|
|
' If Len(dependencies) > 0 Then dependencies = dependencies & "[~]" ' !!! ??? If two [~] are not necessary on the end of Dependencies
|
|
' dependencies = dependencies & ProcessServiceDependency(child) ' uncomment this code and clean up the lines below
|
|
dependencies = dependencies & ProcessServiceDependency(child) & "[~]"
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
If Len(dependencies) > 0 Then dependencies = dependencies & "[~]"
|
|
row.StringData (ServiceInstall_ServiceInstall) = Modularize(ElementText(node)) ' !! need to auto-generate
|
|
row.IntegerData(ServiceInstall_ServiceType) = typebits
|
|
row.IntegerData(ServiceInstall_ErrorControl) = errorbits
|
|
row.StringData (ServiceInstall_Dependencies) = dependencies
|
|
row.StringData (ServiceInstall_Component_) = Modularize(component)
|
|
DoAction ServiceInstallTable, op, row
|
|
End Sub ' ProcessServiceInstallElement
|
|
|
|
Function ProcessServiceDependency(node)
|
|
Dim attribute, value
|
|
|
|
ProcessServiceDependency = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "Group" If "yes"=value Then ProcessServiceDependency = "+" & ProcessServiceDependency
|
|
End Select
|
|
Next
|
|
End Function ' ProcessServiceDependency
|
|
|
|
Sub ProcessRegistry(id, root, key, name, data, component, op)
|
|
' if no id was provided, create one
|
|
If IsEmpty(id) Then id = "r" & regCount : regCount = regCount + 1
|
|
Dim row : Set row = installer.CreateRecord(UBound(RegistryTable))
|
|
row.StringData (Registry_Registry) = Modularize(id)
|
|
row.StringData (Registry_Component_) = Modularize(component)
|
|
row.IntegerData(Registry_Root) = root
|
|
row.StringData (Registry_Key) = ModularizeProperty(key)
|
|
row.StringData (Registry_Name) = ModularizeProperty(name)
|
|
row.StringData (Registry_Value) = ModularizeProperty(data)
|
|
DoAction RegistryTable, op, row
|
|
End Sub
|
|
|
|
Function ProcessRegistryElement(node, component)
|
|
Dim value, attribute, row, op, child, root, key, name, data, action
|
|
ProcessRegistryElement = "reg"
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Key" : key = value
|
|
Case "Name" : name = value
|
|
Case "Value" : data = value
|
|
Case "Action" : action = value
|
|
Case "KeyPath" : ProcessRegistryElement = value
|
|
Case "Root"
|
|
Select Case(value)
|
|
Case "HKMU" : root = -1
|
|
Case "HKCR" : root = 0
|
|
Case "HKCU" : root = 1
|
|
Case "HKLM" : root = 2
|
|
Case "HKU" : root = 3
|
|
Case Else : Fail "Unknown Registry root type: " & value
|
|
End Select
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
Select Case(action)
|
|
Case Empty
|
|
Case "write" : action = Empty
|
|
Case "remove" : action = name
|
|
Case "removeKey" : action = "-"
|
|
Case Else : Fail "Unexpected Registry 'Action' value: " & value
|
|
End Select
|
|
If IsEmpty(action) Then
|
|
ProcessRegistry ElementText(node), root, key, name, data, component, op
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Permission" : ProcessPermissionElement child, ElementText(node), "Registry"
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
Else
|
|
ProcessRegistryElement = Empty
|
|
Set row = installer.CreateRecord(UBound(RemoveRegistryTable))
|
|
row.StringData (RemoveRegistry_RemoveRegistry) = Modularize(ElementText(node)) '!!! auto-generate?
|
|
row.StringData (RemoveRegistry_Component_) = Modularize(component)
|
|
row.IntegerData(RemoveRegistry_Root) = root
|
|
row.StringData (RemoveRegistry_Key) = key
|
|
row.StringData (RemoveRegistry_Name) = action
|
|
DoAction RemoveRegistryTable, op, row
|
|
End If
|
|
End Function
|
|
|
|
Sub ProcessClassElement(node, component, feature)
|
|
Dim child, value, attribute, row, op
|
|
Dim bits, classId, context, defaultProgId, description, libId, threadingModel, insertable, version, programmable
|
|
Dim icon, iconIndex, server
|
|
Set row = installer.CreateRecord(UBound(ClassTable))
|
|
bits = 0
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Description" : description = value
|
|
Case "Context" : context = value
|
|
Case "AppId" : row.StringData (Class_AppId_) = "{" & value & "}"
|
|
Case "FileTypeMask" : row.StringData (Class_FileTypeMask) = value
|
|
Case "Icon" : icon = value
|
|
Case "IconIndex" : iconIndex = CInt(value)
|
|
Case "Handler" : row.StringData (Class_DefInprocHandler) = value
|
|
Case "Argument" : row.StringData (Class_Argument) = value
|
|
Case "RelativePath" : If value="yes" Then bits = bits Or msidbClassAttributesRelativePath
|
|
' The following attributes result in rows added to the Registry table rather than the class table
|
|
Case "ThreadingModel" : threadingModel = value
|
|
Case "Version" : version = value
|
|
Case "Programmable" : If value="yes" Then programmable = "Programmable"
|
|
Case "Insertable"
|
|
Select Case(value)
|
|
Case "yes" : insertable = "Insertable"
|
|
Case "no" : insertable = "NotInsertable"
|
|
Case Else : Fail "Unexpected Class Insertable option: " & value
|
|
End Select
|
|
Case "Server" : server = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
classId = "{" & ElementText(node) & "}"
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "ProgId"
|
|
ProcessProgIdElement child, component, feature, classId, (description), Empty
|
|
If IsEmpty(defaultProgId) Then defaultProgId = ElementText(child)
|
|
Case "TypeLib"
|
|
libId = ProcessTypeLibElement(child, component, feature)
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\TypeLib",Empty,"{" & libId & "}",component,op
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
If Not IsEmpty(threadingModel) Then
|
|
threadingModel = UCase(Left(threadingModel,1)) & Right(threadingModel, Len(threadingModel)-1)
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\" & context,"ThreadingModel",threadingModel,component,op
|
|
End If
|
|
If Not IsEmpty(version) Then ProcessRegistry Empty, 0, "CLSID\" & classId & "\Version", Empty, version,component,op
|
|
If Not IsEmpty(insertable) Then ProcessRegistry Empty, 0, "CLSID\" & classId & "\" & insertable, Empty, Empty,component,op
|
|
If Not IsEmpty(programmable) Then ProcessRegistry Empty, 0, "CLSID\" & classId & "\" & programmable, Empty, Empty,component,op
|
|
|
|
' if this is being advertised under a feature
|
|
If Not IsEmpty(feature) Then
|
|
If Not IsEmpty(server) Then Fail "Cannot specify a Server for an advertised Class Id"
|
|
|
|
row.StringData (Class_CLSID) = classId
|
|
row.StringData (Class_Context) = context
|
|
row.StringData (Class_Component_) = Modularize(component)
|
|
row.StringData (Class_ProgId_Default) = defaultProgId
|
|
row.StringData (Class_Description) = description
|
|
row.StringData (Class_Icon_) = icon
|
|
row.StringData (Class_IconIndex) = iconIndex
|
|
row.StringData (Class_Feature_) = feature
|
|
If bits <> 0 Then row.IntegerData(Class_Attributes) = bits
|
|
DoAction ClassTable, op, row
|
|
Else
|
|
If IsEmpty(server) Then Fail "Must specify a Server for a non-advertised Class Id"
|
|
|
|
' ClassId's Context
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\" & context,"", server, component, op
|
|
' ClassId's Description
|
|
If 0 < Len(description) Then ProcessRegistry Empty, 0, "CLSID\" & classId, "", description, component, op
|
|
' ClassId's AppId
|
|
If 0 < Len(row.StringData(Class_AppId_)) Then ProcessRegistry Empty, 0, "CLSID\" & classId & "\" & context,"AppID", row.StringData(Class_AppId_), component,op
|
|
' ClassId's FileTypeMask
|
|
If 0 < Len(row.StringData(Class_FileTypeMask)) Then Fail "Don't know how to convert FileTypeMask into Registry elements - robmen"
|
|
' ClassId's Default Icon
|
|
If 0 < Len(icon) Then
|
|
If 0 < Len(iconIndex) Then icon = icon & "," & iconIndex
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\" & context & "\DefaultIcon","", icon, component,op
|
|
End If
|
|
' ClassId's Handler
|
|
If 0 < Len(row.StringData(Class_DefInprocHandler)) Then
|
|
Select Case row.StringData(Class_DefInprocHandler)
|
|
Case "1" : ProcessRegistry Empty, 0, "CLSID\" & classId & "\InprocHandler","", "ole.dll", component,op
|
|
Case "2" : ProcessRegistry Empty, 0, "CLSID\" & classId & "\InprocHandler32","", "ole32.dll", component,op
|
|
Case "3"
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\InprocHandler","", "ole.dll", component,op
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\InprocHandler32","", "ole32.dll", component,op
|
|
Case Else : ProcessRegistry regId, 0, "CLSID\" & classId & "\InprocHandler32","", row.StringData(Class_DefInprocHandler), component,op
|
|
End Select
|
|
End If
|
|
' ClassId's Argument
|
|
If 0 < Len(row.StringData(Class_Argument)) Then Fail "Don't know how to convert Arguments into Registry elements - robmen"
|
|
' ClassId's RelativePath
|
|
If 0 < Len(row.StringData(Class_Argument)) Then Fail "Don't know how to convert RelativePath into Registry elements - robmen"
|
|
End If
|
|
End Sub
|
|
|
|
Sub ProcessProgIdElement(node, component, feature, classId, description, parent)
|
|
Dim child, value, attribute, row, op, progId, icon, iconIndex
|
|
Set row = installer.CreateRecord(UBound(ProgIdTable))
|
|
progId = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Description" : description = value
|
|
Case "Icon" : icon = Modularize(value)
|
|
Case "IconIndex" : iconIndex = CInt(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Extension" : ProcessExtensionElement child, component, feature, progId
|
|
Case "ProgId"
|
|
If IsEmpty(feature) Then
|
|
ProcessProgIdElement child, component, Empty, (classId), (description), progId
|
|
Else
|
|
ProcessProgIdElement child, component, feature, Empty, (description), progId
|
|
End If
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(feature) Then
|
|
' ProgId
|
|
ProcessRegistry Empty, 0, progId, "", description, component, op
|
|
' ProgId's ClassId
|
|
If 0 < Len(classId) Then
|
|
ProcessRegistry Empty, 0, progId & "\CLSID", "", classId, component, op
|
|
' if this is a version independent ProgId
|
|
If 0 < Len(parent) Then
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\VersionIndependentProgID", "", progId, component, op
|
|
Else
|
|
ProcessRegistry Empty, 0, "CLSID\" & classId & "\ProgID", "", progId, component, op
|
|
End If
|
|
End If
|
|
' ProgId's Default Icon
|
|
If 0 < Len(icon) Then
|
|
If 0 < Len(iconIndex) Then icon = icon & "," & iconIndex
|
|
ProcessRegistry Empty, 0, progId & "\DefaultIcon","", icon, component,op
|
|
End If
|
|
Else
|
|
row.StringData (ProgId_ProgId) = progId
|
|
row.StringData (ProgId_ProgId_Parent) = parent
|
|
row.StringData (ProgId_Class_) = classId
|
|
row.StringData (ProgId_Description) = description
|
|
row.StringData (ProgId_Icon_) = icon
|
|
If Not IsEmpty(iconIndex) Then row.IntegerData(ProgId_IconIndex) = iconIndex
|
|
DoAction ProgIdTable, op, row
|
|
End If
|
|
End Sub
|
|
|
|
Sub ProcessExtensionElement(node, component, feature, progId)
|
|
Dim child, value, attribute, row, op, extension, mime
|
|
Set row = installer.CreateRecord(UBound(ExtensionTable))
|
|
extension = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "ContentType" : mime = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Verb" : ProcessVerbElement child, extension, progId, feature
|
|
Case "MIME"
|
|
value = ProcessMIMEElement(child, extension, feature)
|
|
If value <> "" And IsEmpty(mime) Then mime = value
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(feature) Then
|
|
' Extension
|
|
ProcessRegistry Empty, 0, "." & extension, "", progId, component, op
|
|
' Extension's MIME ContentType
|
|
If 0 < Len(mime) Then ProcessRegistry Empty, 0, "." & extension, "Content Type", mime, component, op
|
|
Else
|
|
row.StringData (Extension_Extension) = extension
|
|
row.StringData (Extension_Component_) = Modularize(component)
|
|
row.StringData (Extension_ProgId_) = progId
|
|
row.StringData (Extension_MIME_) = mime
|
|
row.StringData (Extension_Feature_) = feature
|
|
DoAction ExtensionTable, op, row
|
|
End If
|
|
End Sub
|
|
|
|
Sub ProcessVerbElement(node, extension, progId, feature)
|
|
Dim attribute, value, row, op, target, command, argument, sequence
|
|
Set row = installer.CreateRecord(UBound(VerbTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Target" : target = value
|
|
Case "Command" : command = value
|
|
Case "Argument" : argument = value
|
|
case "Sequence" : sequence = CLng(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(feature) Then
|
|
If Not IsEmpty(target) Then Fail "Must specify a Target for a non-advertised Verb"
|
|
' handle arguments
|
|
If 0 < Len(argument) Then target = target & " " & argument
|
|
' handle if verb is under progId or under extension
|
|
If 0 < Len(progId) Then
|
|
ProcessRegistry Empty, 0, progId & "\shell\" & command & "\command", "", target, component, op
|
|
Else
|
|
ProcessRegistry Empty, 0, "." & extension & "\shell\" & command & "\command", "", target, component, op
|
|
End If
|
|
Else
|
|
If Not IsEmpty(target) Then Fail "Cannot specify a Target for an advertised Verb"
|
|
|
|
row.StringData (Verb_Extension_) = extension
|
|
row.StringData (Verb_Verb) = ElementText(node)
|
|
If Not IsEmpty(sequence) Then row.IntegerData(Verb_Sequence) = sequence
|
|
row.StringData (Verb_Command) = command
|
|
row.StringData (Verb_Argument) = argument
|
|
DoAction VerbTable, op, row
|
|
End If
|
|
End Sub
|
|
|
|
Function ProcessMIMEElement(node, extension, feature)
|
|
Dim attribute, value, row, op, contentType, classId
|
|
Set row = installer.CreateRecord(UBound(MIMETable))
|
|
contentType = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Class" : classId = value
|
|
Case "Default" : If value = "yes" Then ProcessMIMEElement = contentType
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(feature) Then
|
|
ProcessRegistry Empty, 0, "MIME\Content Type\" & contentType & "", "Extension", extension, component, op
|
|
If 0 < Len(classId) Then ProcessRegistry Empty, 0, "MIME\Content Type\" & contentType & "", "CLSID", classId, component, op
|
|
Else
|
|
row.StringData (MIME_ContentType) = contentType
|
|
row.StringData (MIME_Extension_) = extension
|
|
row.StringData (MIME_CLSID) = classId
|
|
DoAction MIMETable, op, row
|
|
End If
|
|
End Function
|
|
|
|
Function ProcessTypeLibElement(node, component, feature)
|
|
Dim value, attribute, row, op, version
|
|
Set row = installer.CreateRecord(UBound(TypeLibTable))
|
|
version = 0
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "MajorVersion" : version = CInt(value) * 256 + version
|
|
Case "MinorVersion" : version = CInt(value) + version
|
|
Case "Language" : row.IntegerData(TypeLib_Language) = CInt(value)
|
|
Case "HelpDirectory" : row.StringData (TypeLib_Directory_) = Modularize(value)
|
|
Case "Description" : row.StringData (TypeLib_Description) = value
|
|
Case "dt" ' bug in IE5 msxml
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If fModule Then feature = "{00000000-0000-0000-0000-000000000000}"
|
|
ProcessTypeLibElement = ElementText(node)
|
|
row.StringData (TypeLib_LibID) = "{" & ProcessTypeLibElement & "}"
|
|
row.StringData (TypeLib_Component_) = Modularize(component)
|
|
row.StringData (TypeLib_Feature_) = feature
|
|
row.IntegerData(TypeLib_Version) = version
|
|
DoAction TypeLibTable, op, row
|
|
End Function
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' ProcessMediaElement
|
|
Sub ProcessMediaElement(node, lastId)
|
|
Dim value, attribute, child, row, op, diskId
|
|
Dim nLastSequence, sCabinet, fEmbed
|
|
|
|
nLastSequence = 0
|
|
fEmbed = False
|
|
Set row = installer.CreateRecord(UBound(MediaTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "DiskId" : diskId = value
|
|
Case "LastSequence" : nLastSequence = CInt(value)
|
|
Case "DiskPrompt" : row.StringData (Media_DiskPrompt) = value
|
|
Case "Cabinet" : sCabinet = value
|
|
Case "EmbedCab" : If "yes" = value Then fEmbed = True
|
|
Case "VolumeLabel" : row.StringData (Media_VolumeLabel) = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(diskId) Then diskId = lastId + 1
|
|
lastId = diskId
|
|
If fEmbed Then
|
|
If 0 = Len(sCabinet) Then Fail "Must specify a 'Cabinet' when embedding"
|
|
If "#" <> Left(sCabinet, 1) Then sCabinet = "#" & sCabinet
|
|
End If
|
|
|
|
row.IntegerData(Media_DiskId) = CInt(diskId)
|
|
row.IntegerData(Media_LastSequence) = nLastSequence
|
|
row.StringData (Media_Cabinet) = sCabinet
|
|
DoAction MediaTable, op, row
|
|
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "DigitalSignature" : ProcessDigitalSignatureElement child, "Media", diskId
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub ' ProcessMediaElement
|
|
|
|
Sub ProcessAppIdElement(node)
|
|
Dim value, attribute, row, op
|
|
Set row = installer.CreateRecord(UBound(AppIdTable))
|
|
version = 0
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "RemoteServerName" : row.StringData(AppId_RemoteServerName) = value
|
|
Case "LocalService" : row.StringData(AppId_LocalService) = value
|
|
Case "ServiceParameters" : row.StringData(AppId_ServiceParameters) = value
|
|
Case "DllSurrogate" : row.StringData(AppId_DllSurrogate) = value
|
|
Case "ActivateAtStorage" : If value = "yes" Then row.IntegerData(AppId_ActivateAtStorage) = 1
|
|
Case "RunAsInteractiveUser" : If value = "yes" Then row.IntegerData(AppId_RunAsInteractiveUser) = 1
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (AppId_AppId) = "{" & ElementText(node) & "}"
|
|
DoAction AppIdTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessCustomActionElement(node)
|
|
Dim value, attribute, row, op, source, target, bits, sourceBits, targetBits
|
|
Set row = installer.CreateRecord(UBound(CustomActionTable))
|
|
bits = 0
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "BinaryKey" : source = value : sourceBits = msidbCustomActionTypeBinaryData
|
|
Case "FileKey" : source = value : sourceBits = msidbCustomActionTypeSourceFile
|
|
Case "Property" : source = value : sourceBits = msidbCustomActionTypeProperty
|
|
Case "Directory" : source = value : sourceBits = msidbCustomActionTypeDirectory
|
|
Case "DllEntry" : target = value : targetBits = msidbCustomActionTypeDll
|
|
Case "ExeCommand" : target = value : targetBits = msidbCustomActionTypeExe
|
|
Case "JScriptCall" : target = value : targetBits = msidbCustomActionTypeJScript
|
|
Case "VBScriptCall" : target = value : targetBits = msidbCustomActionTypeVBScript
|
|
Case "Value" : target = value : targetBits = msidbCustomActionTypeTextData
|
|
Case "InstallProperties" : target = value : targetBits = msidbCustomActionTypeInstall
|
|
Case "Impersonate" : If value="no" Then bits = bits Or msidbCustomActionTypeNoImpersonate
|
|
Case "Return"
|
|
Select Case(value)
|
|
Case "check"
|
|
Case "ignore" : bits = bits Or msidbCustomActionTypeContinue
|
|
Case "asyncWait" : bits = bits Or msidbCustomActionTypeAsync
|
|
Case "asyncNoWait" : bits = bits Or msidbCustomActionTypeAsync Or msidbCustomActionTypeContinue
|
|
Case Else : Fail "Unknown CustomAction Return type: " & value
|
|
End Select
|
|
Case "Execute"
|
|
Select Case(value)
|
|
Case "immediate"
|
|
Case "deferred" : bits = bits Or msidbCustomActionTypeInScript
|
|
Case "rollback" : bits = bits Or msidbCustomActionTypeInScript Or msidbCustomActionTypeRollback
|
|
Case "commit" : bits = bits Or msidbCustomActionTypeInScript Or msidbCustomActionTypeCommit
|
|
Case "oncePerProcess" : bits = bits Or msidbCustomActionTypeOncePerProcess
|
|
Case "firstSequence" : bits = bits Or msidbCustomActionTypeFirstSequence
|
|
Case "secondSequence" : bits = bits Or msidbCustomActionTypeClientRepeat
|
|
Case Else : Fail "Unknown CustomAction Execute type: " & value
|
|
End Select
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (CustomAction_Action) = Modularize(ElementText(node))
|
|
row.IntegerData(CustomAction_Type) = bits Or sourceBits Or targetBits
|
|
row.StringData (CustomAction_Source) = Modularize(source)
|
|
row.StringData (CustomAction_Target) = target
|
|
DoAction CustomActionTable, op, row
|
|
End Sub
|
|
|
|
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' ProcessCustomTableElement
|
|
Sub ProcessCustomTableElement(node)
|
|
Dim tableName, columnName, columnCount, rowCount, customTable(), columnNames(32), columnTypes(32), columnDef
|
|
Dim value, attribute, child, item, state, row, op, primaryKey, width, nullable, localizable, typeName, index
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
tableName = ElementText(node)
|
|
ReDim customTable(32)
|
|
customTable(0) = tableName
|
|
columnCount = 0
|
|
For Each child In node.childNodes
|
|
Next
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Column"
|
|
columnCount = columnCount + 1
|
|
columnName = ElementText(child)
|
|
columnNames(columnCount) = columnName
|
|
primaryKey = False
|
|
nullable = False
|
|
localizable = False
|
|
width = 0
|
|
For Each attribute In child.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "PrimaryKey" : If value = "yes" Then primaryKey = True
|
|
Case "Nullable" : If value = "yes" Then nullable = True
|
|
Case "Localizable": If value = "yes" Then localizable= True
|
|
Case "Width" : width = CInt(value)
|
|
Case "Type"
|
|
Select Case(value)
|
|
Case "int" : typeName = "SHORT"
|
|
Case "string" : typeName = "CHAR"
|
|
Case "binary" : typeName = "OBJECT"
|
|
Case Else : Fail "Unknown CustomTable data type: " & value
|
|
End Select
|
|
columnTypes(columnCount) = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If typeName = "SHORT" Then
|
|
If width = 4 Then typeName = "LONG" Else If width <> 2 Then Fail "Invalid integer width: " & width
|
|
End If
|
|
If typeName = "CHAR" Then
|
|
If width = 0 Then typeName = "LONGCHAR" Else typeName = typeName & "(" & width & ")"
|
|
End If
|
|
columnDef = "`" & columnName & "` " & typeName
|
|
If Not nullable Then columnDef = columnDef & " NOT NULL"
|
|
If primaryKey Then columnDef = columnDef & " PRIMARY KEY"
|
|
If localizable Then columnDef = columnDef & " LOCALIZABLE"
|
|
customTable(columnCount) = columnDef
|
|
Case "Row" ' processed on second pass
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
ReDim Preserve customTable(columnCount)
|
|
CreateView(customTable) 'CreateView will call CreateTable if table doesn't already exist
|
|
|
|
Set row = installer.CreateRecord(columnCount)
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Column" ' columns already processed
|
|
Case "Row"
|
|
row.ClearData
|
|
For Each attribute In child.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
For Each item In child.childNodes
|
|
Select Case (GetElementName(item))
|
|
Case Empty
|
|
Case "Data"
|
|
columnName = Empty
|
|
For Each attribute In item.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Column" : columnName = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(columnName) Then Fail "Missing column name for Data"
|
|
For index = columnCount To 0 Step -1
|
|
If columnNames(index) = columnName Then Exit For
|
|
Next
|
|
If index = 0 Then Fail "Undefined column for Data: " & columnName
|
|
value = ElementText(item)
|
|
If columnTypes(index) = "string" Then
|
|
row.StringData(index) = value
|
|
Else
|
|
row.IntegerData(index) = CLng(value)
|
|
End If
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
DoAction customTable, op, row
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub ' ProcessCustomTableElement
|
|
|
|
|
|
Sub ProcessBinaryElement(node, table)
|
|
Dim attribute, op, child, value, row, name, length, fileName, outPath, binStream, line, index, char, nxtc, state
|
|
If fNoBinary Then Exit Sub
|
|
Set row = installer.CreateRecord(UBound(table))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : name = value
|
|
Case "src" : outPath = BaseDir(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
' if a file wasn't specified process the bin encoded data
|
|
If IsEmpty(outPath) Then
|
|
fileName = name
|
|
outPath = tempDir & "\wi.tmp"
|
|
Set binStream = fso.CreateTextFile(outPath, OverwriteIfExist, OpenAsASCII) : CheckError
|
|
If fVerbose Then Wscript.echo "Binary file: " & name & " --> " & outPath
|
|
For Each child In node.childNodes
|
|
If child.nodeType = NODE_TEXT Then
|
|
value = child.text
|
|
length = Len(value)
|
|
line = Empty
|
|
For index = 1 To length
|
|
char = Asc(Mid(value, index, 1))
|
|
If char < 48 Then '0
|
|
If char = 47 Then '/
|
|
char = 63
|
|
ElseIf char = 43 Then '+
|
|
char = 62
|
|
Else
|
|
Fail "Illegal bin.base64 char: " & char
|
|
End If
|
|
ElseIf char = 61 Then '=
|
|
state = 4
|
|
ElseIf char <= 57 Then
|
|
char = char + 4 '9
|
|
ElseIf char <= 90 Then
|
|
char = char - 65 'Z
|
|
ElseIf char <= 122 Then
|
|
char = char - 71 'z
|
|
Else
|
|
Fail "Illegal bin.base64 char: " & char
|
|
End If
|
|
Select Case (state)
|
|
Case 0: state = 1 : nxtc = char * 4
|
|
Case 1: state = 2 : line = line & Chr((char \ 16) + nxtc) : nxtc = (char Mod 16) * 16
|
|
Case 2: state = 3 : line = line & Chr((char \ 4) + nxtc) : nxtc = (char Mod 4) * 64
|
|
Case 3: state = 0 : line = line & Chr( char + nxtc)
|
|
Case Else: state = 0
|
|
End Select
|
|
Next
|
|
binStream.Write line
|
|
End If
|
|
Next
|
|
binStream.Close
|
|
Set binStream = Nothing ' release to allow reading
|
|
End If
|
|
|
|
' On Error Resume Next
|
|
row.StringData(Binary_Name) = Modularize(name)
|
|
row.SetStream Binary_Data, outPath : CheckError
|
|
DoAction table, op, row
|
|
CloseView table ' force table out of memory to release file
|
|
End Sub
|
|
|
|
Sub ProcessSequence(table, node)
|
|
Dim child, attribute, row, op, action, sequence, condition, lastSequence, defaultOp
|
|
If fNoSeqTables Then Exit Sub
|
|
Set row = installer.CreateRecord(UBound(table))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : defaultOp = attribute.value
|
|
Case "xmlns" ' ignore, processed by XML engine
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
' due to a bug in the mergemod.dll add all sequences to a MSI when ever the first one is needed
|
|
If database.TablePersistent("InstallExecuteSequence") = 2 Then CreateTable InstallExecuteSequenceTable
|
|
If database.TablePersistent("InstallUISequence") = 2 Then CreateTable InstallUISequenceTable
|
|
If database.TablePersistent("AdminExecuteSequence") = 2 Then CreateTable AdminExecuteSequenceTable
|
|
If database.TablePersistent("AdminUISequence") = 2 Then CreateTable AdminUISequenceTable
|
|
If database.TablePersistent("AdvtExecuteSequence") = 2 Then CreateTable AdvtExecuteSequenceTable
|
|
If database.TablePersistent("AdvtUISequence") = 2 Then CreateTable AdvtUISequenceTable
|
|
|
|
For Each child In node.childNodes
|
|
action = GetElementName(child)
|
|
If Not IsEmpty(action) Then
|
|
sequence = Empty
|
|
condition = Empty
|
|
op = Empty
|
|
For Each attribute In child.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "xmlns" ' ignore, processed by XML engine
|
|
Case "Action" : If action = "Custom" Then action = attribute.value Else Unexpected attribute, child
|
|
Case "Dialog" : If action = "Show" Then action = attribute.value Else Unexpected attribute, child
|
|
Case "Sequence" : sequence = CInt(attribute.value)
|
|
Case "OnExit"
|
|
If Not IsEmpty(sequence) Then Fail "Can't specify both Sequence and OnExit"
|
|
Select Case(attribute.value)
|
|
Case "success" : sequence = -1
|
|
Case "cancel" : sequence = -2
|
|
Case "error" : sequence = -3
|
|
Case "suspend" : sequence = -4
|
|
Case Else : Fail "Unexpected OnExit value: " & attribute.value
|
|
End Select
|
|
Case Else : Unexpected attribute, child
|
|
End Select
|
|
Next
|
|
If action = "Custom" Then Fail "Missing Action attribute for Custom action"
|
|
If action = "Show" Then Fail "Missing Dialog attribute for Show action"
|
|
If ElementHasText(child) Then condition = ElementText(child)
|
|
If IsEmpty(sequence) Or sequence = 0 Then sequence = lastSequence + 1
|
|
If IsEmpty(op) Then op = defaultOp
|
|
lastSequence = sequence
|
|
row.StringData (InstallExecuteSequence_Action) = action
|
|
row.IntegerData(InstallExecuteSequence_Sequence) = sequence
|
|
row.StringData (InstallExecuteSequence_Condition) = condition
|
|
DoAction table, op, row
|
|
End If
|
|
Next
|
|
End Sub
|
|
|
|
''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
|
|
' ODBC handlers
|
|
|
|
Sub ProcessODBCDriver(node, component, file, table) ' also handles ODBCTranslator
|
|
Dim child, value, attribute, row, op, driver, setup, name, driverKey, childOp
|
|
Set row = installer.CreateRecord(UBound(table))
|
|
driver = file
|
|
setup = file
|
|
driverKey = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : name = value
|
|
Case "File" : driver = value
|
|
Case "SetupFile" : setup = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(ODBCDriver_Driver) = driverKey
|
|
row.StringData(ODBCDriver_Component_) = Modularize(component)
|
|
row.StringData(ODBCDriver_Description) = name
|
|
row.StringData(ODBCDriver_File_) = Modularize(driver)
|
|
row.StringData(ODBCDriver_File_Setup) = Modularize(setup)
|
|
DoAction table, op, row
|
|
If table(0) <> "`ODBCDriver`" Then Exit Sub ' translators have no attributes or data sources
|
|
For Each child In node.childNodes
|
|
childOp = op
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "ODBCDataSource" : Call ProcessODBCDataSource(child, component, name)
|
|
Case "Property"
|
|
row.ClearData
|
|
For Each attribute In child.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : childOp = attribute.value
|
|
Case "Value" : value = attribute.value
|
|
Case Else : Unexpected attribute, child
|
|
End Select
|
|
Next
|
|
row.StringData(ODBCAttribute_Driver_) = driverKey
|
|
row.StringData(ODBCAttribute_Attribute) = ElementText(child)
|
|
row.StringData(ODBCAttribute_Value) = value
|
|
DoAction ODBCAttributeTable, childOp, row
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub
|
|
|
|
Function ProcessODBCDataSource(node, component, driverName)
|
|
Dim child, value, attribute, row, op, name, sourceKey, childOp, registration
|
|
Set row = installer.CreateRecord(UBound(ODBCDataSourceTable))
|
|
sourceKey = ElementText(node)
|
|
ProcessODBCDataSource = "reg"
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : name = value
|
|
Case "DriverName" : driverName = value
|
|
Case "KeyPath" : ProcessODBCDataSource = value
|
|
Case "Registration"
|
|
Select Case(value)
|
|
Case "machine" : registration = 0
|
|
Case "user" : registration = 1
|
|
Case Else : Fail "Unexpected ODBCDataSource Registration: " & value
|
|
End Select
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (ODBCDataSource_DataSource) = sourceKey
|
|
row.StringData (ODBCDataSource_Component_) = Modularize(component)
|
|
row.StringData (ODBCDataSource_Description) = name
|
|
row.StringData (ODBCDataSource_DriverDescription) = driverName
|
|
row.IntegerData(ODBCDataSource_Registration) = regitration
|
|
DoAction ODBCDataSourceTable, op, row
|
|
For Each child In node.childNodes
|
|
childOp = op
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Property"
|
|
row.ClearData
|
|
For Each attribute In child.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : childOp = attribute.value
|
|
Case "Value" : value = attribute.value
|
|
Case Else : Unexpected attribute, child
|
|
End Select
|
|
Next
|
|
row.StringData(ODBCSourceAttribute_DataSource_) = sourceKey
|
|
row.StringData(ODBCSourceAttribute_Attribute) = ElementText(child)
|
|
row.StringData(ODBCSourceAttribute_Value) = value
|
|
DoAction ODBCSourceAttributeTable, childOp, row
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Function
|
|
|
|
'---------------------------------------------------------------------------------'
|
|
' AppSearch and CCP handlers
|
|
'---------------------------------------------------------------------------------'
|
|
|
|
Sub ProcessAppSearch(property, signature, op)
|
|
If property <> UCase(property) Then Fail "Must uppercase search Property: " & property
|
|
|
|
Dim row : Set row = installer.CreateRecord(UBound(AppSearchTable))
|
|
row.StringData (AppSearch_Property) = Modularize(property)
|
|
row.StringData (AppSearch_Signature_) = Modularize(signature)
|
|
DoAction AppSearchTable, op, row
|
|
End Sub
|
|
|
|
Function ProcessIniFileSearchElement(node)
|
|
Dim child, op, row, attribute, value, signature, fOneChild
|
|
Set row = installer.CreateRecord(UBound(IniLocatorTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "File" : row.StringData (IniLocator_FileName) = value
|
|
Case "Section" : row.StringData (IniLocator_Section) = value
|
|
Case "Key" : row.StringData (IniLocator_Key) = value
|
|
Case "Field" : row.StringData (IniLocator_Field) = value
|
|
Case "Type"
|
|
Select Case(value)
|
|
Case "directory" : row.IntegerData(IniLocator_Type) = 0
|
|
Case "file" : row.IntegerData(IniLocator_Type) = 1
|
|
Case "registry" : row.IntegerData(IniLocator_Type) = 2
|
|
Case Else : Fail "Unknown Ini search type: " & value
|
|
End Select
|
|
End Select
|
|
Next
|
|
signature = Modularize(ElementText(node)) ' !! maybe auto-generate?
|
|
row.StringData (IniLocator_Signature_) = Modularize(signature)
|
|
DoAction IniLocatorTable, op, row
|
|
|
|
fOneChild = False
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "DirectorySearch": If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessDirectorySearchElement(child, signature)
|
|
Case "FileSearch" : If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessFileSearchElement (child, signature)
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
ProcessIniFileSearchElement = signature
|
|
End Function ' ProcessIniFileSearchElement
|
|
|
|
Function ProcessRegistrySearchElement(node)
|
|
Dim child, op, row, attribute, value, signature, fOneChild
|
|
Set row = installer.CreateRecord(UBound(RegLocatorTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Root"
|
|
Select Case(value)
|
|
Case "HKCR" : row.IntegerData(RegLocator_Root) = 0
|
|
Case "HKCU" : row.IntegerData(RegLocator_Root) = 1
|
|
Case "HKLM" : row.IntegerData(RegLocator_Root) = 2
|
|
Case "HKU" : row.IntegerData(RegLocator_Root) = 3
|
|
Case Else : Fail "Unknown Registry search type: " & value
|
|
End Select
|
|
Case "Key" : row.StringData (RegLocator_Key) = value
|
|
Case "Name" : row.StringData (RegLocator_Name) = value
|
|
Case "Type"
|
|
Select Case(value)
|
|
Case "directory" : row.IntegerData(RegLocator_Type) = 0
|
|
Case "file" : row.IntegerData(RegLocator_Type) = 1
|
|
Case "registry" : row.IntegerData(RegLocator_Type) = 2
|
|
Case Else : Fail "Unknown Registry search type: " & value
|
|
End Select
|
|
End Select
|
|
Next
|
|
signature = ElementText(node) ' !! maybe auto-generate?
|
|
row.StringData (RegLocator_Signature_) = Modularize(signature)
|
|
DoAction RegLocatorTable, op, row
|
|
|
|
fOneChild = False
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "DirectorySearch": If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessDirectorySearchElement(child, signature)
|
|
Case "FileSearch" : If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessFileSearchElement (child, signature)
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
ProcessRegistrySearchElement = signature
|
|
End Function ' ProcessRegistrySearchElement
|
|
|
|
Function ProcessComponentSearchElement(node)
|
|
Dim child, op, row, attribute, value, signature, fOneChild
|
|
Set row = installer.CreateRecord(UBound(CompLocatorTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Id" : row.StringData (CompLocator_ComponentId) = "{" & value & "}" ' add curly braces on GUID
|
|
Case "Type"
|
|
Select Case(value)
|
|
Case "directory" : row.IntegerData(CompLocator_Type) = 0
|
|
Case "file" : row.IntegerData(CompLocator_Type) = 1
|
|
Case Else : Fail "Unknown Component search type: " & value
|
|
End Select
|
|
End Select
|
|
Next
|
|
signature = ElementText(node) ' !! maybe auto-generate?
|
|
row.StringData (CompLocator_Signature_) = Modularize(signature)
|
|
DoAction CompLocatorTable, op, row
|
|
|
|
fOneChild = False
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "DirectorySearch": If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessDirectorySearchElement(child, signature)
|
|
Case "FileSearch" : If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessFileSearchElement (child, signature)
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
ProcessComponentSearchElement = signature
|
|
End Function ' ProcessComponentSearchElement
|
|
|
|
Function ProcessDirectorySearchElement(node, parent)
|
|
Dim child, op, row, attribute, value, signature, fOneChild
|
|
Set row = installer.CreateRecord(UBound(DrLocatorTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Path" : row.StringData (DrLocator_Path) = value
|
|
Case "Depth" : row.IntegerData(DrLocator_Depth) = CInt(value)
|
|
End Select
|
|
Next
|
|
signature = ElementText(node) ' !! maybe auto-generate?
|
|
row.StringData (DrLocator_Signature_) = Modularize(signature)
|
|
row.StringData (DrLocator_Parent) = Modularize(parent)
|
|
DoAction DrLocatorTable, op, row
|
|
|
|
fOneChild = False
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "DirectorySearch": If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessDirectorySearchElement(child, signature)
|
|
Case "FileSearch" : If fOneChild Then Fail "Only one search type can appear under Property: " & property Else fOneChild = True : signature = ProcessFileSearchElement (child, signature)
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
ProcessDirectorySearchElement = signature
|
|
End Function ' ProcessDirectorySearchElement
|
|
|
|
Function ProcessFileSearchElement(node, parent)
|
|
Dim child, op, row, attribute, value, signature, fOneChild
|
|
Set row = installer.CreateRecord(UBound(SignatureTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : row.StringData (Signature_FileName) = value
|
|
Case "MinVersion" : row.StringData (Signature_MinVersion) = value
|
|
Case "MaxVersion" : row.StringData (Signature_MaxVersion) = value
|
|
Case "MinSize" : row.IntegerData(Signature_MinSize) = CLng(value)
|
|
Case "MaxSize" : row.IntegerData(Signature_MaxSize) = CLng(value)
|
|
Case "MinDate" : row.IntegerData(Signature_MinDate) = DosDate(value)
|
|
Case "MaxDate" : row.IntegerData(Signature_MaxDate) = DosDate(value)
|
|
Case "Languages" : row.StringData (Signature_Languages) = value
|
|
End Select
|
|
Next
|
|
If ElementHasText(node) Then
|
|
signature = ElementText(node)
|
|
Else
|
|
If Len(parent) = 0 Then Fail "Missing identifier for File search." Else signature = parent
|
|
End If
|
|
row.StringData (Signature_Signature) = Modularize(signature)
|
|
DoAction SignatureTable, op, row
|
|
|
|
ProcessFileSearchElement = signature
|
|
End Function ' ProcessFileSearchElement
|
|
|
|
'---------------------------------------------------------------------------------'
|
|
' Module handlers
|
|
'---------------------------------------------------------------------------------'
|
|
|
|
Sub ProcessDependencyElement(node)
|
|
Dim op, row, attribute, value, required, requiredId, language, version
|
|
Set row = installer.CreateRecord(UBound(ModuleDependencyTable))
|
|
required = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Id" : requiredId = Replace(value, "-", "_")
|
|
Case "Version" : version = value
|
|
Case "Language" : language = CInt(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(language) Then language = 0
|
|
If Len(requiredId) <> 36 Then Fail "An Id must be specified for required Module"
|
|
row.StringData (ModuleDependency_ModuleID) = Modularize(productName)
|
|
row.IntegerData(ModuleDependency_ModuleLanguage) = CInt(productLanguage)
|
|
row.StringData (ModuleDependency_RequiredID) = required & "." & requiredId
|
|
row.IntegerData(ModuleDependency_RequiredLanguage) = language
|
|
row.StringData (ModuleDependency_RequiredVersion) = version
|
|
DoAction ModuleDependencyTable, op, row
|
|
End Sub
|
|
|
|
|
|
|
|
'---------------------------------------------------------------------------------'
|
|
' Windows Installer 1.5 elements
|
|
'---------------------------------------------------------------------------------'
|
|
Sub ProcessAssemblyElement(node, sComponent, sFeature)
|
|
Dim op, row, child, attribute, value, sManifest, sApplication, nAttributes
|
|
Set row = installer.CreateRecord(UBound(MsiAssemblyTable))
|
|
|
|
For Each attribute in node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Manifest" : sManifest = value
|
|
Case "Application" : sApplication = value
|
|
Case "Type"
|
|
Select Case(value)
|
|
Case ".net" : nAttributes = 0
|
|
Case "win32" : nAttributes = 1
|
|
Case Else : Fail "Unknown Assembly.Type: " & value
|
|
End Select
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(nAttributes) Then Fail "Must specify a 'Type' for <Assembly/>"
|
|
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Property" : ProcessAssemblyPropertyElement child, sComponent
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
row.StringData(MsiAssembly_Component_) = Modularize(sComponent)
|
|
row.StringData(MsiAssembly_Feature_) = sFeature
|
|
row.StringData(MsiAssembly_File_Manifest) = Modularize(sManifest)
|
|
row.StringData(MsiAssembly_File_Application) = sApplication
|
|
row.IntegerData(MsiAssembly_Attributes) = nAttributes
|
|
DoAction MsiAssemblyTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessAssemblyPropertyElement(node, sComponent)
|
|
Dim op, row, child, attribute, value, sName, sValue
|
|
Set row = installer.CreateRecord(UBound(MsiAssemblyNameTable))
|
|
|
|
sName = ElementText(node)
|
|
For Each attribute in node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Value" : sValue = value
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(sName) Then Fail "Must specify a 'Name' for <Assembly><Property/></Assembly>"
|
|
|
|
row.StringData(MsiAssemblyName_Component_) = Modularize(sComponent)
|
|
row.StringData(MsiAssemblyName_Name) = sName
|
|
row.StringData(MsiAssemblyName_Value) = sValue
|
|
DoAction MsiAssemblyNameTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessDigitalSignatureElement(node, sTable, sResource)
|
|
Dim op, row, child, attribute, value, src, sCertificate
|
|
Set row = installer.CreateRecord(UBound(MsiDigitalSignatureTable))
|
|
|
|
For Each attribute in node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "src" : src = BaseDir(value)
|
|
End Select
|
|
Next
|
|
|
|
If ElementHasText(node) Then Fail "hex-encoded <DigitalSignature/> not currently supported"
|
|
If IsEmpty(src) Then Fail "Must specify a source file for DigitalSignature hash"
|
|
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "DigitalCertificate" : If IsEmpty(sCertificate) Then sCertificate = ProcessDigitalCertificateElement(child) Else Fail "Only one <DigitalCertificate/> per <DigitalSignature/>"
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
If IsEmpty(sCertificate) Then Fail "Must have a single <DigitalCertificate/> for a <DigitalSignature/>"
|
|
|
|
row.StringData(MsiDigitalSignature_Table) = sTable
|
|
row.StringData(MsiDigitalSignature_SignObject) = sResource
|
|
row.StringData(MsiDigitalSignature_DigitalCertificate_) = sCertificate
|
|
row.SetStream MsiDigitalSignature_Hash, src
|
|
DoAction MsiDigitalSignatureTable, op, row
|
|
End Sub
|
|
|
|
Function ProcessDigitalCertificateElement(node)
|
|
Dim op, row, child, attribute, value, sName, src
|
|
Set row = installer.CreateRecord(UBound(MsiDigitalCertificateTable))
|
|
|
|
For Each attribute in node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Name" : sName = value
|
|
Case "src" : src = BaseDir(value)
|
|
End Select
|
|
Next
|
|
|
|
If ElementHasText(node) Then Fail "hex-encoded <DigitalCertificate/> not currently supported"
|
|
If IsEmpty(sName) Then Fail "Must specify a 'Name' for <DigitalCertificate/>"
|
|
If IsEmpty(src) Then Fail "Must specify a 'src' file for <DigitalCertificate/>"
|
|
|
|
row.StringData(MsiDigitalCertificate_DigitalCertificate) = sName
|
|
row.SetStream MsiDigitalCertificate_CertData, src
|
|
DoAction MsiDigitalCertificateTable, op, row
|
|
|
|
ProcessDigitalCertificateElement = sName
|
|
End Function
|
|
|
|
|
|
'---------------------------------------------------------------------------------'
|
|
' Patch element handlers
|
|
'---------------------------------------------------------------------------------'
|
|
Sub ProcessPatchElement(node)
|
|
If Not fNoOnError Then On Error Resume Next
|
|
Dim attribute, value
|
|
Dim sWholeFiles, sProductMismatches, sVersionMismatches, sClean
|
|
|
|
sWholeFiles = "0"
|
|
sProductMismatches = "0"
|
|
sVersionMismatches = "0"
|
|
sClean = "1"
|
|
sReplaceGUIDs = Empty
|
|
|
|
' Walk XML nodes and populate .pcp tables
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "Id" : value = "{"& value &"}" : ProcessProperties "PatchGUID", value, "replace"
|
|
Case "OutputPath" : ProcessProperties "PatchOutputPath", value, "replace"
|
|
Case "WholeFilesOnly" : If "yes" = value Then sWholeFiles = "1" Else sWholeFiles = "0"
|
|
Case "SourceList" : ProcessProperties "PatchSourceList", value, "replace"
|
|
Case "AllowProductCodeMismatches" : If "yes" = value Then sProductMismatches = "1" Else sProductMismatches = "0"
|
|
Case "AllowMajorVersionMismatches": If "yes" = value Then sVersionMismatches = "1" Else sVersionMismatches = "0"
|
|
Case "CleanWorkingFolder" : If "yes" = value Then sClean = "0" Else sClean = "1"
|
|
Case "OptionFlags" : ProcessProperties "ApiPatchingOptionFlags", value, "replace"
|
|
Case "SymbolFlags" : ProcessProperties "ApiPatchingSymbolFlags", value, "replace"
|
|
Case "xmlns" : ' ProcessProperties "XMLSchema", value, "replace"
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
ProcessProperties "IncludeWholeFilesOnly", sWholeFiles, "replace"
|
|
ProcessProperties "AllowProductCodeMismatches", sProductMismatches, "replace"
|
|
ProcessProperties "AllowProductVersionMajorMismatches", sVersionMismatches, "replace"
|
|
ProcessProperties "DontRemoveTempFolderWhenFinished", sClean, "replace"
|
|
|
|
Dim child, sReplaceGUIDs, sTargetProducts
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Property" : ProcessPropertiesElement child
|
|
Case "Family" : ProcessFamilyElement child
|
|
Case "ReplacePatch" : sReplaceGUIDs = sReplaceGUIDs & "{" & ElementText(child) & "}"
|
|
Case "TargetProductCode"
|
|
value = ElementText(child)
|
|
If "*" <> value And 38 <> Len(value) Then Fail "Invalid TargetProductCode value: " & value
|
|
If Not IsEmpty(sTargetProducts) Then sTargetProducts = sTargetProducts & ";"
|
|
If "*" <> value Then value = "{" & value & "}"
|
|
sTargetProducts = sTargetProducts & value
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
If Not IsEmpty(sReplaceGUIDs) Then ProcessProperties "ListOfPatchGUIDsToReplace", sReplaceGUIDs, "replace"
|
|
ProcessProperties "ListOfTargetProductCodes", sTargetProducts, "replace"
|
|
Set dictView = Nothing ' close all views, could also use RemoveAll method of Dictionary object
|
|
End Sub ' ProcessPatchElement
|
|
|
|
|
|
Sub ProcessProperties(sProperty, value, op)
|
|
Dim row : Set row = installer.CreateRecord(UBound(PropertiesTable))
|
|
row.StringData (Properties_Name) = sProperty
|
|
row.StringData (Properties_Value) = value
|
|
DoAction PropertiesTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessPropertiesElement(node)
|
|
Dim attribute, op, child, sProperty, value
|
|
|
|
sProperty = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Value" : value = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
ProcessProperties sProperty, value, op
|
|
End Sub
|
|
|
|
Sub ProcessFamilyElement(node)
|
|
Dim op, row, attribute, value
|
|
Dim sFamily
|
|
|
|
Set row = installer.CreateRecord(UBound(ImageFamiliesTable))
|
|
|
|
sFamily = ElementText(node) ' !! maybe auto-generate?
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "MediaSrcProp" : row.StringData(ImageFamilies_MediaSrcPropName) = value
|
|
Case "DiskId" : row.IntegerData(ImageFamilies_MediaDiskId) = CLng(value)
|
|
Case "SequenceStart": row.IntegerData(ImageFamilies_FileSequenceStart) = CLng(value)
|
|
Case "DiskPrompt" : row.StringData(ImageFamilies_DiskPrompt) = value
|
|
Case "VolumeLabel" : row.StringData(ImageFamilies_VolumeLabel) = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(ImageFamilies_Family) = sFamily
|
|
DoAction ImageFamiliesTable, op, row
|
|
|
|
Dim child
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "ExternalFile": ProcessExternalFileElement child, sFamily
|
|
Case "ProtectFile" : ProcessProtectFileElement child, sFamily
|
|
Case "UpgradeImage": ProcessUpgradeImageElement child, sFamily
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub ' ProcessFamilyElement
|
|
|
|
|
|
Sub ProcessExternalFileElement(node, sFamily)
|
|
Dim op, row, attribute, value
|
|
Dim sFile, nOrder
|
|
|
|
Set row = installer.CreateRecord(UBound(ExternalFilesTable))
|
|
|
|
sFile = ElementText(node) ' !! maybe auto-generate?
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "src" : row.StringData(ExternalFiles_FilePath) = BaseDir(value)
|
|
Case "Order" : nOrder = CLng(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(nOrder) Then nOrder = externalOrder : externalOrder = externalOrder + 1 Else externalOrder = nOrder + 1
|
|
row.StringData(ExternalFiles_Family) = sFamily
|
|
row.StringData(ExternalFiles_FTK) = sFile
|
|
row.IntegerData(ExternalFiles_Order) = nOrder
|
|
|
|
Dim child, sSymbols
|
|
Dim sProtectOffsets, sProtectLengths
|
|
Dim sIgnoreOffsets, sIgnoreLengths
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "SymbolPaths" : If Not IsEmpty(sSymbols) Then sSymbols = sSymbols & ";" : sSymbols = sSymbols & BaseDir(ElementText(child))
|
|
Case "ProtectRange": ProcessRangeElement child, sProtectOffsets, sProtectLengths
|
|
Case "IgnoreRange" : ProcessRangeElement child, sIgnoreOffsets, sIgnoreLengths
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
row.StringData(ExternalFiles_SymbolPaths) = sSymbols
|
|
row.StringData(ExternalFiles_IgnoreOffsets) = sIgnoreOffsets
|
|
row.StringData(ExternalFiles_IgnoreLengths) = sIgnoreLengths
|
|
|
|
If Not IsEmpty(sProtectOffsets) Then
|
|
row.StringData(ExternalFiles_RetainOffsets) = sProtectOffsets
|
|
|
|
Dim row2 : Set row2 = installer.CreateRecord(UBound(FamilyFileRangesTable))
|
|
row2.StringData(FamilyFileRanges_Family) = sFamily
|
|
row2.StringData(FamilyFileRanges_FTK) = sFile
|
|
row2.StringData(FamilyFileRanges_RetainOffsets) = sProtectOffsets
|
|
row2.StringData(FamilyFileRanges_RetainLengths) = sProtectLengths
|
|
|
|
DoAction FamilyFileRangesTable, "merge", row2
|
|
End If
|
|
|
|
DoAction ExternalFilesTable, op, row
|
|
End Sub ' ProcessExternalFileElement
|
|
|
|
|
|
Sub ProcessProtectFileElement(node, sFamily)
|
|
Dim op, row, attribute, value
|
|
Dim sFile
|
|
|
|
Set row = installer.CreateRecord(UBound(FamilyFileRangesTable))
|
|
|
|
sFile = ElementText(node) ' !! maybe auto-generate?
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(FamilyFileRanges_Family) = sFamily
|
|
row.StringData(FamilyFileRanges_FTK) = sFile
|
|
|
|
Dim child
|
|
Dim sProtectOffsets, sProtectLengths
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "ProtectRange" : ProcessRangeElement child, sProtectOffsets, sProtectLengths
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(sProtectOffsets) Then Fail "Must specify Offsets for ProtectRange"
|
|
If IsEmpty(sProtectLengths) Then Fail "Must specify Lengths for ProtectRange"
|
|
row.StringData(FamilyFileRanges_RetainOffsets) = sProtectOffsets
|
|
row.StringData(FamilyFileRanges_RetainLengths) = sProtectLengths
|
|
|
|
DoAction FamilyFileRangesTable, op, row
|
|
End Sub ' ProcessProtectFileElement
|
|
|
|
|
|
Sub ProcessUpgradeImageElement(node, sFamily)
|
|
Dim op, row, attribute, value
|
|
Dim sUpgraded
|
|
|
|
Set row = installer.CreateRecord(UBound(UpgradedImagesTable))
|
|
|
|
sUpgraded = ElementText(node) ' !! maybe auto-generate?
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "src" : row.StringData(UpgradedImages_MsiPath) = BaseDir(value)
|
|
Case "srcPatch": row.StringData(UpgradedImages_PatchMsiPath) = BaseDir(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(UpgradedImages_Upgraded) = sUpgraded
|
|
row.StringData(UpgradedImages_Family) = sFamily
|
|
|
|
Dim child, sSymbols
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "SymbolPaths": If Not IsEmpty(sSymbols) Then sSymbols = sSymbols & ";" : sSymbols = sSymbols & BaseDir(ElementText(child))
|
|
Case "UpgradeFile": ProcessUpgradeFileElement child, sUpgraded
|
|
Case "TargetImage": ProcessTargetImageElement child, sUpgraded, sFamily
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
row.StringData(UpgradedImages_SymbolPaths) = sSymbols
|
|
|
|
DoAction UpgradedImagesTable, op, row
|
|
End Sub ' ProcessUpgradeImageElement
|
|
|
|
|
|
Sub ProcessUpgradeFileElement(node, sUpgraded)
|
|
Dim op, row, attribute, value
|
|
Dim sFile, fIgnore, nAllowIgnore, nWholeFile
|
|
|
|
nAllowIgnore = 0
|
|
nWholeFile = 0
|
|
|
|
sFile = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Ignore" : If "yes" = value Then fIgnore = True Else fIgnore = False
|
|
Case "AllowIgnoreOnError": If "yes" = value Then nAllowIgnore = 1 Else nAllowIgnore = 0
|
|
Case "WholeFile" : If "yes" = value Then nWholeFile = 1 Else nWholeFile = 0
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
|
|
If fIgnore Then
|
|
Set row = installer.CreateRecord(UBound(UpgradedFilesToIgnoreTable))
|
|
row.StringData(UpgradedFilesToIgnore_Upgraded) = sUpgraded
|
|
row.StringData(UpgradedFilesToIgnore_Upgraded) = sFile
|
|
DoAction UpgradedFilesToIgnoreTable, op, row
|
|
Else
|
|
Set row = installer.CreateRecord(UBound(UpgradedFiles_OptionalDataTable))
|
|
row.StringData(UpgradedFiles_OptionalDataTable_Upgraded) = sUpgraded
|
|
row.StringData(UpgradedFiles_OptionalDataTable_File) = sFile
|
|
row.IntegerData(UpgradedFiles_OptionalDataTable_AllowIgnoreOnPatchError) = nAllowIgnore
|
|
row.IntegerData(UpgradedFiles_OptionalDataTable_IncludeWholeFile) = nWholeFile
|
|
|
|
Dim child, sSymbols
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "SymbolPaths": If Not IsEmpty(sSymbols) Then sSymbols = sSymbols & ";" : sSymbols = sSymbols & BaseDir(ElementText(child))
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
row.StringData(UpgradedFiles_OptionalDataTable_SymbolPaths) = sSymbols
|
|
|
|
DoAction UpgradedFiles_OptionalDataTable, op, row
|
|
End If
|
|
End Sub ' ProcessUpgradeFileElement
|
|
|
|
|
|
Sub ProcessTargetImageElement(node, sUpgraded, sFamily)
|
|
Dim op, row, attribute, value
|
|
Dim sTarget, nIgnore, nOrder
|
|
|
|
nIgnore = 0
|
|
Set row = installer.CreateRecord(UBound(TargetImagesTable))
|
|
|
|
sTarget = ElementText(node) ' !! maybe auto-generate?
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "src" : row.StringData(TargetImages_MsiPath) = BaseDir(value)
|
|
Case "Order" : nOrder = CLng(value)
|
|
Case "Validation": row.StringData(TargetImages_ProductValidateFlags) = value
|
|
Case "IgnoreMissingFiles": If "yes" = value Then nIgnore = 1 Else nIgnore = 0
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(nOrder) Then nOrder = patchOrder : patchOrder = patchOrder + 1 Else patchOrder = nOrder + 1
|
|
row.StringData(TargetImages_Target) = sTarget
|
|
row.StringData(TargetImages_Upgraded) = sUpgraded
|
|
row.IntegerData(TargetImages_IgnoreMissingSrcFiles) = nIgnore
|
|
row.IntegerData(TargetImages_Order) = nOrder
|
|
|
|
Dim child, sSymbols
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "SymbolPaths": If Not IsEmpty(sSymbols) Then sSymbols = sSymbols & ";" : sSymbols = sSymbols & BaseDir(ElementText(child))
|
|
Case "TargetFile": ProcessTargetFileElement child, sTarget, sFamily
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
row.StringData(TargetImages_SymbolPaths) = sSymbols
|
|
|
|
DoAction TargetImagesTable, op, row
|
|
End Sub ' ProcessTargetImageElement
|
|
|
|
|
|
Sub ProcessTargetFileElement(node, sTarget, sFamily)
|
|
Dim op, row, attribute, value
|
|
Dim sFile, nIgnore
|
|
|
|
nIgnore = 0
|
|
Set row = installer.CreateRecord(UBound(TargetImagesTable))
|
|
|
|
sTarget = ElementText(node) ' !! maybe auto-generate?
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData(TargetFiles_OptionalData_Target) = sTarget
|
|
row.StringData(TargetFiles_OptionalData_FTK) = sFile
|
|
|
|
Dim child, sSymbols
|
|
Dim sProtectOffsets, sProtectLengths
|
|
Dim sIgnoreOffsets, sIgnoreLengths
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "SymbolPaths" : If Not IsEmpty(sSymbols) Then sSymbols = sSymbols & ";" : sSymbols = sSymbols & BaseDir(ElementText(child))
|
|
Case "ProtectRange": ProcessRangeElement child, sProtectOffsets, sProtectLengths
|
|
Case "IgnoreRange" : ProcessRangeElement child, sIgnoreOffsets, sIgnoreLengths
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
row.StringData(TargetFiles_OptionalData_SymbolPaths) = sSymbols
|
|
row.StringData(TargetFiles_OptionalData_IgnoreOffsets) = sIgnoreOffsets
|
|
row.StringData(TargetFiles_OptionalData_IgnoreLengths) = sIgnoreLengths
|
|
|
|
If Not IsEmpty(sProtectOffsets) Then
|
|
row.StringData(TargetFiles_OptionalData_RetainOffsets) = sProtectOffsets
|
|
|
|
Dim row2 : Set row2 = installer.CreateRecord(UBound(FamilyFileRangesTable))
|
|
row2.StringData(FamilyFileRanges_Family) = sFamily
|
|
row2.StringData(FamilyFileRanges_FTK) = sFile
|
|
row2.StringData(FamilyFileRanges_RetainOffsets) = sProtectOffsets
|
|
row2.StringData(FamilyFileRanges_RetainLengths) = sProtectLengths
|
|
|
|
DoAction FamilyFileRangesTable, "insert", row2
|
|
End If
|
|
|
|
DoAction TargetImagesTable, op, row
|
|
End Sub ' ProcessTargetImageElement
|
|
|
|
Sub ProcessRangeElement(node, ByRef sOffsets, ByRef sLengths)
|
|
Dim op, row, attribute, value, sOffset, sLength
|
|
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "Offset" : sOffset = value
|
|
Case "Length" : sLength = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(sOffset) Then Fail "Range missing Offset"
|
|
If IsEmpty(sLength) Then Fail "Range missing Length"
|
|
If Not IsEmpty(sOffsets) Then sOffsets = sOffsets & ","
|
|
If Not IsEmpty(sLengths) Then sLengths = sLengths & ","
|
|
sOffsets = sOffsets & sOffset
|
|
sLengths = sLengths & sLength
|
|
End Sub
|
|
|
|
'---------------------------------------------------------------------------------'
|
|
' UI element handlers
|
|
'---------------------------------------------------------------------------------'
|
|
Sub ProcessUIElement(node)
|
|
Dim child
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Error" : ProcessErrorElement child
|
|
Case "ProgressText" : ProcessActionTextElement child
|
|
Case "Dialog" : ProcessDialogElement child
|
|
Case "TextStyle" : ProcessTextStyleElement child
|
|
Case "UIText" : ProcessUITextElement child
|
|
Case "BillboardAction" : ProcessBillboardActionElement child
|
|
Case "ListBox" : ProcessControlGroupElement child, ListBoxTable, "ListItem"
|
|
Case "ComboBox" : ProcessControlGroupElement child, ComboBoxTable, "ListItem"
|
|
Case "ListView" : ProcessControlGroupElement child, ListViewTable, "ListItem"
|
|
Case "RadioGroup" : ProcessControlGroupElement child, RadioButtonTable, "RadioButton"
|
|
' the following are available indentically under the UI and Programs tabs for document organization use only
|
|
Case "Property" : ProcessPropertyElement child
|
|
Case "InstallUISequence" : ProcessSequence InstallUISequenceTable, child
|
|
Case "AdminUISequence" : ProcessSequence AdminUISequenceTable, child
|
|
Case "AdvertiseUISequence" : ProcessSequence AdvtUISequenceTable, child
|
|
Case "Binary" : ProcessBinaryElement child, BinaryTable
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
End Sub
|
|
|
|
Sub ProcessListItemElement(node, table, property, op, order)
|
|
Dim attribute, row, text, icon
|
|
order = order + 1
|
|
Set row = installer.CreateRecord(UBound(table))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Text" : text = attribute.value
|
|
Case "Icon" : If row.FieldCount = ListView_Binary_ Then icon = attribute.value Else Unexpected attribute, node
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (ListView_Property) = Modularize(property)
|
|
row.IntegerData(ListView_Order) = order
|
|
row.StringData (ListView_Value) = ElementText(node)
|
|
row.StringData (ListView_Text) = text
|
|
If Len(icon) <> 0 Then row.StringData (ListView_Binary_) = Modularize(icon)
|
|
DoAction table, op, row
|
|
End Sub
|
|
|
|
Sub ProcessRadioButtonElement(node, property, op, order)
|
|
Dim attribute, value, row, tooltip, help
|
|
order = order + 1
|
|
Set row = installer.CreateRecord(UBound(RadioButtonTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "X" : row.IntegerData(RadioButton_X) = CInt(value)
|
|
Case "Y" : row.IntegerData(RadioButton_Y) = CInt(value)
|
|
Case "Width" : row.IntegerData(RadioButton_Width) = CInt(value)
|
|
Case "Height" : row.IntegerData(RadioButton_Height) = CInt(value)
|
|
Case "Text" : row.StringData (RadioButton_Text) = value
|
|
Case "Icon" : row.StringData (RadioButton_Text) = value
|
|
Case "ToolTip" : tooltip = value
|
|
Case "Help" : help = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (RadioButton_Property) = Modularize(property)
|
|
row.IntegerData(RadioButton_Order) = order
|
|
If Len(tooltip) + Len(help) Then
|
|
row.StringData (RadioButton_Help) = tooltip & "|" & help
|
|
End If
|
|
row.StringData (RadioButton_Value) = ElementText(node)
|
|
DoAction RadioButtonTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessBillboardGroupElement(node)
|
|
Dim attribute, child, op, row, action, feature, order, billboard, grandchild
|
|
Set row = installer.CreateRecord(UBound(BillboardTable))
|
|
action = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
Unexpected attribute, node ' grouping element only, no attributes
|
|
Next
|
|
For Each child In node.childNodes
|
|
If GetElementName(child) <> "Billboard" Then Unexpected child, node
|
|
order = order + 1
|
|
billboard = ElementText(child)
|
|
For Each attribute In child.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Feature" : feature = attribute.value
|
|
Case Else Unexpected attribute, child
|
|
End Select
|
|
Next
|
|
row.StringData (Billboard_Billboard) = billboard
|
|
row.StringData (Billboard_Feature_) = feature
|
|
row.StringData (Billboard_Action) = action
|
|
row.IntegerData(Billboard_Ordering) = order
|
|
DoAction BillboardTable, op, row
|
|
For Each grandchild in child.childNodes
|
|
If GetElementName(grandchild) <> "Control" Then Unexpected grandchild, child
|
|
ProcessControlElement grandchild, billboard, BBControlTable, Empty, Empty, Empty, Empty
|
|
Next
|
|
Next
|
|
End Sub
|
|
|
|
Sub ProcessControlGroupElement(node, table, childTag)
|
|
Dim attribute, op, child, property, order, value, childName
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Property" : property = attribute.value
|
|
End Select
|
|
Next
|
|
For Each child In node.childNodes
|
|
childName = GetElementName(child)
|
|
If Not IsEmpty(childName) Then
|
|
If childName <> childTag Then Unexpected child, node
|
|
Select Case (childName)
|
|
Case "ListItem" : ProcessListItemElement child, table, property, op, order
|
|
Case "RadioButton" : ProcessRadioButtonElement child, property, op, order
|
|
End Select
|
|
End If
|
|
Next
|
|
End Sub
|
|
|
|
Sub ProcessErrorElement(node)
|
|
Dim attribute, op, row, id
|
|
Set row = installer.CreateRecord(UBound(ErrorTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Id" : id = CInt(attribute.value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.IntegerData(Error_Error) = id
|
|
row.StringData (Error_Message) = ElementText(node)
|
|
DoAction ErrorTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessActionTextElement(node)
|
|
Dim attribute, op, row, id
|
|
Set row = installer.CreateRecord(UBound(ActionTextTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Action" : row.StringData (ActionText_Action) = attribute.value
|
|
Case "Template" : row.StringData (ActionText_Template) = attribute.value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
REM Dim child : Set child = node.selectSingleNode("text()")
|
|
REM If Not child Is Nothing Then row.StringData (ActionText_Description) = child.text
|
|
row.StringData (ActionText_Description) = ElementText(node)
|
|
DoAction ActionTextTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessTableElement(node, table, attributes)
|
|
Dim attribute, op, row, id, index
|
|
Set row = installer.CreateRecord(UBound(table))
|
|
For Each attribute In node.Attributes
|
|
If attribute.name = "op" Then
|
|
op = attribute.value
|
|
Else
|
|
For index = 1 To UBound(table)
|
|
If attributes(index) = attribute.name Then
|
|
If InStr(table(index), "CHAR") > 0 Then
|
|
row.StringData(index) = attribute.value
|
|
Else
|
|
row.IntegerData(index) = CLng(attribute.value)
|
|
End If
|
|
End If
|
|
Next
|
|
If index > UBound(table) Then Unexpected attribute, node
|
|
End If
|
|
Next
|
|
For index = 1 To UBound(table)
|
|
If IsEmpty(attributes(index)) Then row.StringData(index) = ElementText(node)
|
|
Next
|
|
DoAction table, op, row
|
|
End Sub
|
|
|
|
Sub ProcessUITextElement(node)
|
|
Dim attribute, op, row, child, text
|
|
Set row = installer.CreateRecord(UBound(UITextTable))
|
|
For Each attribute In node.Attributes
|
|
Select Case(attribute.name)
|
|
Case "op" : op = attribute.value
|
|
Case "Text" : text = attribute.value
|
|
Case Else : Unexpected attribute, child
|
|
End Select
|
|
Next
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Text" : text = ElementText(child)
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
row.StringData (UIText_Key) = Modularize(ElementText(node))
|
|
row.StringData (UIText_Text) = text
|
|
DoAction UITextTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessTextStyleElement(node)
|
|
Dim attribute, value, op, row, bits, color
|
|
Set row = installer.CreateRecord(UBound(TextStyleTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Red" : color = color + CInt(attribute.value)
|
|
Case "Green" : color = color + CInt(attribute.value) * 256
|
|
Case "Blue" : color = color + CInt(attribute.value) * 65536
|
|
Case "Bold" : If value = "yes" Then bits = bits Or 1
|
|
Case "Italic" : If value = "yes" Then bits = bits Or 2
|
|
Case "Underline" : If value = "yes" Then bits = bits Or 4
|
|
Case "Strike" : If value = "yes" Then bits = bits Or 8
|
|
Case "FaceName" : row.StringData (TextStyle_FaceName) = value
|
|
Case "Size" : row.IntegerData(TextStyle_Size) = CInt(value)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (TextStyle_TextStyle) = ElementText(node) '!!! BUG: Must end in _UL so can you modularize?
|
|
If Not IsEmpty(color) Then row.IntegerData(TextStyle_Color) = color
|
|
If Not IsEmpty(bits) Then row.IntegerData(TextStyle_StyleBits) = bits
|
|
DoAction TextStyleTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessDialogElement(node)
|
|
Dim child, value, attribute, row, bits, op
|
|
Dim dialog, control, firstControl, defaultControl, cancelControl
|
|
Dim x,y : x = 50 : y = 50
|
|
Set row = installer.CreateRecord(UBound(DialogTable))
|
|
bits = msidbDialogAttributesVisible + msidbDialogAttributesModal + msidbDialogAttributesMinimize
|
|
dialog = ElementText(node)
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "X" : x = CInt(value)
|
|
Case "Y" : y = CInt(value)
|
|
Case "Width" : row.IntegerData(Dialog_Width) = CInt(value)
|
|
Case "Height" : row.IntegerData(Dialog_Height) = CInt(value)
|
|
Case "Title" : row.StringData (Dialog_Title) = value
|
|
Case "Hidden" : If value="yes" Then bits = bits Xor msidbDialogAttributesVisible
|
|
Case "Modeless" : If value="yes" Then bits = bits Xor msidbDialogAttributesModal
|
|
Case "NoMinimize" : If value="yes" Then bits = bits Xor msidbDialogAttributesMinimize
|
|
Case "SystemModal" : If value="yes" Then bits = bits Xor msidbDialogAttributesSysModal
|
|
Case "KeepModeless" : If value="yes" Then bits = bits Xor msidbDialogAttributesKeepModeless
|
|
Case "TrackDiskSpace" : If value="yes" Then bits = bits Xor msidbDialogAttributesTrackDiskSpace
|
|
Case "CustomPalette" : If value="yes" Then bits = bits Xor msidbDialogAttributesUseCustomPalette
|
|
Case "RightToLeft" : If value="yes" Then bits = bits Xor msidbDialogAttributesRTLRO
|
|
Case "RightAligned" : If value="yes" Then bits = bits Xor msidbDialogAttributesRightAligned
|
|
Case "LeftScroll" : If value="yes" Then bits = bits Xor msidbDialogAttributesLeftScroll
|
|
Case "ErrorDialog" : If value="yes" Then bits = bits Xor msidbDialogAttributesError
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
Dim lastTabRow : Set lastTabRow = Nothing
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Control" ProcessControlElement child, dialog, ControlTable, lastTabRow, firstControl, defaultControl, cancelControl
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
Set child = Nothing
|
|
ProcessControlElement child, dialog, ControlTable, lastTabRow, firstControl, defaultControl, cancelControl
|
|
|
|
row.StringData (Dialog_Dialog) = Modularize(dialog)
|
|
row.IntegerData(Dialog_HCentering) = x
|
|
row.IntegerData(Dialog_VCentering) = y
|
|
row.IntegerData(Dialog_Attributes) = bits
|
|
row.StringData (Dialog_Control_First) = Modularize(firstControl)
|
|
row.StringData (Dialog_Control_Default) = Modularize(defaultControl)
|
|
row.StringData (Dialog_Control_Cancel) = Modularize(cancelControl)
|
|
DoAction DialogTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessControlElement(node, dialog, table, lastTabRow, firstControl, defaultControl, cancelControl)
|
|
Dim child, value, attribute, row, op
|
|
Dim control, noTab, controlType, specialAttributes, bit, bits, publishOrder, text, property, help, disabled
|
|
Dim checkboxValue, checkboxRow
|
|
Dim x,y : x = 50 : y = 50
|
|
If node Is Nothing Then ' called at exit of Dialog child processing loop to force out cached row
|
|
If Not lastTabRow Is Nothing Then
|
|
If lastTabRow.StringData(Control_Control) <> firstControl Then
|
|
lastTabRow.StringData(Control_Control_Next) = Modularize(firstControl)
|
|
End If
|
|
DoAction ControlTable, op, lastTabRow
|
|
End If
|
|
Exit Sub ' last time through
|
|
End If
|
|
control = ElementText(node)
|
|
Set row = installer.CreateRecord(UBound(table))
|
|
controlType = node.Attributes.GetNamedItem("Type").value ' need to get first
|
|
Select Case(controlType)
|
|
Case "Text" : specialAttributes = textControlAttributes : If Not IsEmpty(firstControl) Then noTab = True
|
|
Case "Edit" : specialAttributes = editControlAttributes
|
|
Case "MaskedEdit" : specialAttributes = editControlAttributes
|
|
Case "PathEdit" : specialAttributes = editControlAttributes
|
|
Case "Icon" : specialAttributes = iconControlAttributes : noTab = True : disabled = True
|
|
Case "Bitmap" : specialAttributes = bitmapControlAttributes : noTab = True : disabled = True
|
|
Case "ProgressBar" : specialAttributes = progressControlAttributes : noTab = True : disabled = True
|
|
Case "DirectoryCombo" : specialAttributes = volumeControlAttributes
|
|
Case "VolumeSelectCombo" : specialAttributes = volumeControlAttributes
|
|
Case "VolumeCostList" : specialAttributes = volumeControlAttributes : noTab = True
|
|
Case "ListBox" : specialAttributes = listboxControlAttributes
|
|
Case "ListView" : specialAttributes = listviewControlAttributes
|
|
Case "ComboBox" : specialAttributes = comboboxControlAttributes
|
|
Case "PushButton" : specialAttributes = buttonControlAttributes
|
|
Case "CheckBox" : specialAttributes = checkboxControlAttributes
|
|
Case "RadioButtonGroup" : specialAttributes = radioControlAttributes
|
|
Case "ScrollableText" : specialAttributes = Array()
|
|
Case "SelectionTree" : specialAttributes = Array()
|
|
Case "DirectoryList" : specialAttributes = Array()
|
|
Case "GroupBox" : specialAttributes = Array() : noTab = True : disabled = True
|
|
Case "Line" : specialAttributes = Array() : noTab = True : disabled = True
|
|
Case "Billboard" : specialAttributes = Array() : noTab = True : disabled = True
|
|
Case Else : specialAttributes = Array() : noTab = True
|
|
End Select
|
|
If disabled Then bits = msidbControlAttributesEnabled ' bit will be inverted when stored
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "Type" ' already processed
|
|
Case "op" : op = value
|
|
Case "Default" : If value = "yes" Then defaultControl = control
|
|
Case "Cancel" : If value = "yes" Then cancelControl = control
|
|
Case "TabSkip" : If value = "yes" Then noTab = True Else If value = "no" Then notab = False
|
|
Case "X" : x = CInt(value)
|
|
Case "Y" : y = CInt(value)
|
|
Case "Text" : text = value
|
|
Case "Property" : property = value
|
|
Case "Help" : help = value
|
|
Case "Width" : row.IntegerData(Control_Width) = CInt(value)
|
|
Case "Height" : row.IntegerData(Control_Height) = CInt(value)
|
|
Case "CheckBoxValue" : checkboxValue = value
|
|
Case "IconSize"
|
|
Select Case(value)
|
|
Case "16" : bit = NameToBit(specialAttributes, "Icon16", "yes")
|
|
Case "32" : bit = NameToBit(specialAttributes, "Icon32", "yes")
|
|
Case "48" : bit = (NameToBit(specialAttributes, "Icon16", "yes") Or NameToBit(specialAttributes, "Icon32", "yes"))
|
|
Case Else : Fail "Invalid IconSize: " & value
|
|
End Select
|
|
If IsEmpty(bit) Then Unexpected attribute, node
|
|
bits = bits Xor (bit * 65536)
|
|
Case Else
|
|
bit = NameToBit(commonControlAttributes, attribute.name, value)
|
|
If IsEmpty(bit) Then bit = NameToBit(specialAttributes, attribute.name, value) * 65536
|
|
If IsEmpty(bit) Then Unexpected attribute, node
|
|
bits = bits Xor bit
|
|
End Select
|
|
Next
|
|
For Each child In node.childNodes
|
|
Select Case (GetElementName(child))
|
|
Case Empty
|
|
Case "Text" : text = ElementText(child)
|
|
Case "Condition" : ProcessControlConditionElement child, dialog, control
|
|
Case "Publish" : ProcessPublishElement child, dialog, control, publishOrder
|
|
Case "Subscribe" : ProcessSubscribeElement child, dialog, control
|
|
Case Else : Unexpected child, node
|
|
End Select
|
|
Next
|
|
|
|
row.StringData (Control_Dialog_) = Modularize(dialog)
|
|
row.StringData (Control_Control) = Modularize(control)
|
|
row.StringData (Control_Type) = controlType
|
|
row.IntegerData(Control_X) = x
|
|
row.IntegerData(Control_Y) = y
|
|
row.IntegerData(Control_Attributes) = bits Xor (msidbControlAttributesVisible Or msidbControlAttributesEnabled)
|
|
If IsEmpty(lastTabRow) Then ' Billboard control
|
|
row.StringData (BBControl_Text) = text
|
|
Else
|
|
row.StringData (Control_Text) = text
|
|
row.StringData (Control_Property) = property
|
|
row.StringData (Control_Help) = help
|
|
End If
|
|
If noTab Then
|
|
DoAction table, op, row
|
|
Else
|
|
If IsEmpty(lastTabRow) Then Fail "Tabbable Control not allowed in Billboard: " & controlType
|
|
If IsEmpty(firstControl) Then firstControl = control
|
|
If Not lastTabRow Is Nothing Then
|
|
lastTabRow.StringData(Control_Control_Next) = control
|
|
DoAction ControlTable, op, lastTabRow
|
|
End If
|
|
Set lastTabRow = row
|
|
End If
|
|
If Not IsEmpty(checkBoxValue) Then
|
|
If controlType <> "CheckBox" Then Fail "CheckBoxValue attribute valid only with CheckBox"
|
|
Set checkboxRow = installer.CreateRecord(UBound(CheckBoxTable))
|
|
checkboxRow.StringData(CheckBox_Property) = Modularize(property)
|
|
checkboxRow.StringData(CheckBox_Value) = checkboxValue
|
|
DoAction CheckBoxTable, op, checkboxRow
|
|
End If
|
|
End Sub
|
|
|
|
Sub ProcessControlConditionElement(node, dialog, control)
|
|
Dim attribute, value, op, row, id, action
|
|
Set row = installer.CreateRecord(UBound(ControlConditionTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Action" : action = UCase(Left(value,1)) & Right(value, Len(value)-1)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (ControlCondition_Dialog_) = Modularize(dialog)
|
|
row.StringData (ControlCondition_Control_) = Modularize(control)
|
|
row.StringData (ControlCondition_Action) = action
|
|
row.StringData (ControlCondition_Condition) = ElementText(node)
|
|
DoAction ControlConditionTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessPublishElement(node, dialog, control, order)
|
|
Dim attribute, value, op, row, id, event_, argument
|
|
Set row = installer.CreateRecord(UBound(ControlEventTable))
|
|
order = order + 1
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Event" : event_ = UCase(Left(value,1)) & Right(value, Len(value)-1)
|
|
Case "Property" : event_ = "[" & value & "]"
|
|
Case "Value" : argument = value
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
If IsEmpty(event_) Then event_ = "{}"
|
|
row.StringData (ControlEvent_Dialog_) = Modularize(dialog)
|
|
row.StringData (ControlEvent_Control_) = Modularize(control)
|
|
row.StringData (ControlEvent_Event) = Modularize(event_)
|
|
row.StringData (ControlEvent_Argument) = argument
|
|
row.IntegerData(ControlEvent_Ordering) = order
|
|
If ElementHasText(node) Then row.StringData (ControlEvent_Condition) = ElementText(node)
|
|
DoAction ControlEventTable, op, row
|
|
End Sub
|
|
|
|
Sub ProcessSubscribeElement(node, dialog, control)
|
|
Dim attribute, value, op, row, id, event_, controlAttribute
|
|
Set row = installer.CreateRecord(UBound(EventMappingTable))
|
|
For Each attribute In node.Attributes
|
|
value = attribute.value
|
|
Select Case(attribute.name)
|
|
Case "op" : op = value
|
|
Case "Event" : event_ = UCase(Left(value,1)) & Right(value, Len(value)-1)
|
|
Case "Attribute" : controlAttribute = UCase(Left(value,1)) & Right(value, Len(value)-1)
|
|
Case Else : Unexpected attribute, node
|
|
End Select
|
|
Next
|
|
row.StringData (EventMapping_Dialog_) = Modularize(dialog)
|
|
row.StringData (EventMapping_Control_) = Modularize(control)
|
|
row.StringData (EventMapping_Event) = Modularize(event_)
|
|
row.StringData (EventMapping_Attribute) = controlAttribute
|
|
DoAction EventMappingTable, op, row
|
|
End Sub
|
|
</script>
|
|
</job>
|