PowerShell.BTDF.psm1
#Requires -RunAsAdministrator #Requires -PSEdition Desktop if (-not [System.Reflection.Assembly]::LoadWithPartialName("Microsoft.BizTalk.ExplorerOM")) { throw [System.ApplicationException]::new("Failed to load Microsoft.BizTalk.ExplorerOM") } if (-not [System.Reflection.Assembly]::LoadWithPartialName("Microsoft.BizTalk.Operations")) { throw [System.ApplicationException]::new("Failed to load Microsoft.BizTalk.Operations") } if (-not [System.Reflection.Assembly]::LoadWithPartialName("SSOSettingsFileReader")) { throw [System.ApplicationException]::new("Failed to load SSOSettingsFileReader") } #region Variables $btdfTargets = @{ "BounceBizTalk" = @{ "Message" = "Boucing BizTalk" "Target" = "BounceBizTalk" } "Deploy" = @{ "Message" = "Deploying application" "Target" = "Deploy" } "DeployBAM" = @{ "Message" = "Deploying BAM" "Target" = "GetSoftwarePaths;InitializeAppName;DeployBAM" } "DeployBRE" = @{ "Message" = "Deploying BRE" "Target" = "DeployVocabAndRules" } "DeploySSO" = @{ "Message" = "Deploying SSO" "Target" = "DeploySSO" } "ImportBindings" = @{ "Message" = "Importing bindings" "Target" = "ImportBindings" } "Installer" = @{ "Message" = "Packaging" "Target" = "Installer" } "PreProcessBindings" = @{ "Message" = "Pre-Processing bindings" "Target" = "PreprocessBindings" } "QuickDeploy" = @{ "Message" = "Quick deploying application" "Target" = "UpdateOrchestration" } "Undeploy" = @{ "Message" = "Undeploying application" "Target" = "Undeploy" } "UndeployBAM" = @{ "Message" = "Undeploying BAM" "Target" = "GetSoftwarePaths;InitializeAppName;UndeployBAM" } "UndeployBRE" = @{ "Message" = "Undeploying BRE" "Target" = "UndeployVocabAndRules" } } #endregion #region BTDF <# .SYNOPSIS Helper function to clean BizTalk environment using BTDF .DESCRIPTION Uses BizTalk ExplorerOM to build a list of BizTalk applications currently installed. Each application is undeployed using "Deploy-BTDFApplication" to cleanly undeploy applications as well as handling dependencies. An exemption list is also included to leave specified, base applications .EXAMPLE PS C:\> Clean-BTDFEnvironment Removes all BTDF BizTalk applications using the Debug configuration, the default for local development .EXAMPLE PS C:\> Clean-BTDFEnvironment -Configuration Release Removes all BTDF BizTalk applications using the Release configuration .EXAMPLE PS C:\> Clean-BTDFEnvironment -Exemptions Common Removes all BTDF BizTalk applications using the Debug configuration except for the "Common" application .PARAMETER Configuration Deploys the BizTalk artifacts from the specified bin location. Debug and Release are default build configuration in VS. Server is used to (un)deploy BizTalk artifacts from an MSI installation .PARAMETER TerminateInstances Terminates BizTalk service instances as part of the undeployment process .PARAMETER Exemptions A dynamic list of applications which can be exempt from the cleaning process .NOTES General notes #> function Clean-BTDFEnvironment { [CmdletBinding()] Param ( [Parameter(HelpMessage = "Valid parameters are Debug, Release and Server. Defaults to Debug")] [ValidateSet("Debug", "Release", "Server")] [string]$Configuration = "Debug", [Parameter(HelpMessage = "Terminate instances related to deployed app")] [switch]$TerminateInstances ) DynamicParam { $btsCatalog.Refresh() $btsApps = $btsCatalog.Applications | Select-Object -Property Name, $projectPathColumn | Where-Object { $_.ProjectPath } $paramDict = [System.Management.Automation.RuntimeDefinedParameterDictionary]::new() #region DeployBTMgmtDB if ($Configuration -eq "Server") { $attributeCollection = [System.Collections.ObjectModel.Collection[System.Attribute]]::new() $parameter = [System.Management.Automation.ParameterAttribute]::new() $parameter.HelpMessage = "Register artifacts in BizTalk DB" $attributeCollection.Add($parameter) $deployBTMgmtDBParameter = [System.Management.Automation.RuntimeDefinedParameter]::new("DeployBTMgmtDB", [switch], $attributeCollection) $paramDict.Add("DeployBTMgmtDB", $deployBTMgmtDBParameter) } #endregion #region Exemptions $attributeCollection = [System.Collections.ObjectModel.Collection[System.Attribute]]::new() $parameter = [System.Management.Automation.ParameterAttribute]::new() $attributeCollection.Add($parameter) $validateSet = [System.Management.Automation.ValidateSetAttribute]::new(($btsApps | Select-Object -ExpandProperty Name)) $attributeCollection.Add($validateSet) $exemptionsParameter = [System.Management.Automation.RuntimeDefinedParameter]::new("Exemptions", [System.Collections.Generic.List[string]], $attributeCollection) $exemptionsParameter.Value = @() $paramDict.Add("Exemptions", $exemptionsParameter) $PSBoundParameters["Exemptions"] = [System.Collections.Generic.List[string]]::new() #endregion return $paramDict } Process { $cleanParams = @{ } if ($PSBoundParameters.ContainsKey("DeployBTMgmtDB")) { $cleanParams["DeployBTMgmtDB"] = $PSBoundParameters["DeployBTMgmtDB"] } if ($PSBoundParameters.ContainsKey("TerminateInstances")) { $cleanParams["TerminateInstances"] = $TerminateInstances } foreach ($a in $btsApps) { Write-Debug "App: $($a.Name)" $btsCatalog.Refresh() if (-not $btsCatalog.Applications[$a.Name]) { Write-Warning "$($a.Name) already removed" } elseif ($PSBoundParameters["Exemptions"].Contains($a.Name)) { Write-Verbose "$($a.Name) exempt" } else { Write-Information "Removing $($a.Name)" Deploy-BTDFApplication -ProjectPath $a.ProjectPath ` -DeploymentType Undeploy ` -Configuration $Configuration ` @cleanParams } } } } <# .SYNOPSIS Deploys a BTDF based BizTalk application .DESCRIPTION Deploys a BTDF based BizTalk application and allows configuration to support multi-server setups (Deploying resources to secondary nodes and registering application on primary node). This also uses BizTalk types to check if an upgraded application is a dependency and will strip and restore this application recursively and in order. Supports both local (Debug/Release) deployments as well as server deployments. The parameters are validated and may need adjusting should your environments be named differently .PARAMETER ProjectPath The path which the BTDF application is stored in (Eqivalent to the installation path/targetdir). Usually one level above the deployment folder for an application .PARAMETER DeploymentType Choose whether to "Deploy", "Undeploy" or "Quick Deploy (UpdateOrchestration)" the BizTalk application (Default is "Deploy") .PARAMETER Configuration Deploys the BizTalk artifacts from the specified bin location. Debug and Release are default build configuration in VS. Server is used to (un)deploy BizTalk artifacts from an MSI installation .PARAMETER Environment Select the environment to deploy. Must match what is configured in BTDF environmentsettings spreadsheet. Defaults to Local (Only applicable to Server deployments) .PARAMETER DeployBTMgmtDB Registers application in BizTalk. Assemblies/components are GACed otherwise (Only applicable to Server deployments) .PARAMETER SkipUnDeploy Skip undeploying application prior to deployment (Only applicable to Server deployments) .PARAMETER SkipBizTalkRestart If supported by deployment project, skip restarting BizTalk Host instances post-deployment .PARAMETER SkipIISRestart If supported by deployment project, skip IISReset post-deployment .PARAMETER SkipApplicationStart If supported by deployment project, don't start application after deployment .PARAMETER TerminateInstances If supported by deployment project, terminate running/suspended instances .PARAMETER SkipRestore Skips restoring dependant applciations, if any .EXAMPLE Deploy application components/assemblies C:\PS> Deploy-BTDFApplication -ProjectPath "C:\Program Files (x86)\Scratchpad" .EXAMPLE Deploy application and register in BizTalk C:\PS> Deploy-BTDFApplication -ProjectPath "C:\Program Files (x86)\Scratchpad" -Configuration Server -DeployBTMgmtDB .EXAMPLE Deploy application and register in BizTalk with live environment settings C:\PS> Deploy-BTDFApplication -ProjectPath "C:\Program Files (x86)\Scratchpad" -Configuration Server -Environment Prod -DeployBTMgmtDB .EXAMPLE Undeploys application and removes from BizTalk C:\PS> Deploy-BTDFApplication -ProjectPath "C:\Program Files (x86)\Scratchpad" -DeploymentType Undeploy -Configuration Server -DeployBTMgmtDB .NOTES General notes #> function Deploy-BTDFApplication { [CmdletBinding(SupportsShouldProcess = $true)] [OutputType([hashtable])] Param ( [Parameter(Position = 0, Mandatory = $true, ValueFromPipeline = $true, HelpMessage = "Path to the folder of the BizTalk app")] [System.IO.DirectoryInfo[]]$ProjectPath, [Parameter(HelpMessage = "Defaults to Deploy")] [ValidateSet("BounceBizTalk", "Deploy", "DeployBAM", "DeployBRE", "DeploySSO", "ImportBindings", "Installer", "PreProcessBindings", "QuickDeploy", "Undeploy", "UndeployBAM", "UndeployBRE")] [string]$DeploymentType = "Deploy", [Parameter(HelpMessage = "Valid parameters are Debug, Release and Server. Defaults to Debug")] [ValidateSet("Debug", "Release", "Server")] [string]$Configuration = "Debug", [Parameter(HelpMessage = "Valid parameters are Local, Dev , Int, UAT and Prod. Defaults to Local")] [ValidateSet("Local", "Dev", "Int", "UAT", "Train", "Prod")] [string]$Environment = "Local", [Parameter(HelpMessage = "Register artifacts in BizTalk DB")] [switch]$DeployBTMgmtDB, [Parameter(HelpMessage = "Only deploy instead of redeploy")] [switch]$SkipUnDeploy, [Parameter(HelpMessage = "Skip starting BizTalk hosts back up post deployment")] [switch]$SkipBizTalkRestart, [Parameter(HelpMessage = "Skip IISReset post deployment")] [switch]$SkipIISRestart, [Parameter(HelpMessage = "Skipping starting BizTalk app post deployment")] [switch]$SkipApplicationStart, [Parameter(HelpMessage = "Terminate instances related to deployed app")] [switch]$TerminateInstances, [Parameter(HelpMessage = "Skip restoring dependant applications")] [switch]$SkipRestore ) Process { #region Parameter checks if ($Configuration -ne "Server") { if ($PSBoundParameters.ContainsKey("Environment")) { throw [System.ArgumentException]::new("Environment must be used with the `"Server`" configuration") } if ($PSBoundParameters.ContainsKey("DeployBTMgmtDB")) { throw [System.ArgumentException]::new("DeployBTMgmtDB must be used with the `"Server`" configuration") } if ($PSBoundParameters.ContainsKey("SkipUnDeploy")) { throw [System.ArgumentException]::new("SkipUnDeploy must be used with the `"Server`" configuration") } } #endregion foreach ($pp in $ProjectPath) { Write-Debug "ProjectPath = $pp" #region Get properties from BTDF project $btdfProject = Join-Path -Path $pp -ChildPath "Deployment\Deployment.btdfproj" $btdfProjectXml = [xml](Get-Content -Path $btdfProject) $manufacturer = $btdfProjectXml.GetElementsByTagName("Manufacturer")."#text" Write-Debug "Manufacturer = $manufacturer" $projectName = $btdfProjectXml.GetElementsByTagName("ProjectName")."#text" Write-Debug "ProjectName = $projectName" #endregion #region Get back referenced applications $btsCatalog.Refresh() $btsApp = $btsCatalog.Applications["$projectName"] Write-Verbose "Checking back references for: $projectName" try { if ((-not ($Configuration -eq "Server" -and -not $DeployBTMgmtDB)) ` -and $btsApp ` -and ($DeploymentType -in "Deploy", "UnDeploy")) { if (-not $backRefs) { Write-Verbose "Creating new back refs stack" $backRefs = [System.Collections.Generic.Stack[System.Object]]::new() } $btsApps = $btsCatalog.Applications["$projectName"].BackReferences | Select-Object -Property Name, $projectPathColumn if ($btsApps.Count -gt 0) { Write-Verbose "Back refs found" } foreach ($a in $btsApps) { $btsCatalog.Refresh() if ($btsCatalog.Applications[$a.Name]) { Write-Verbose "Removing back reference: $($a.Name)" Write-Debug "BizTalk App: $($a.Name) = $($a.ProjectPath)" $undeployParams = $PSBoundParameters $undeployParams["ProjectPath"] = $a.ProjectPath $undeployParams["DeploymentType"] = "Undeploy" try { if ($PSCmdlet.ShouldProcess($a.Name, "Removing BizTalk Application")) { Deploy-BTDFApplication @undeployParams } } finally { $backRefs.Push($a) } Write-Verbose "Removed back reference: $($a.Name)" } else { Write-Warning "$($a.Name) already removed" } } if (($backRefs.Count -gt 0) -and ($btsApps.Count -gt 0)) { Write-Debug ($backRefs.GetEnumerator() | Out-String) } } else { Write-Verbose "No back references" } $backRefsRemoved = $true } catch { Write-Error -Message "Undeploying back references failed" -ErrorAction Continue Write-Error -Message "$_" -ErrorAction Continue } #endregion #region Calculate deployment variables #Calculate required properties $deployment = (Get-ChildItem -Path $pp -Filter *Deployment)[0] Write-Debug "Deployment = $($deployment.FullName)" $resultsPath = Join-Path -Path $pp -ChildPath "DeployResults\DeployResults.txt" if ($PSCmdlet.ShouldProcess($resultsPath, "Create empty MSBuild log file")) { New-Item -Path $resultsPath -ItemType File -Force | Out-Null } Write-Debug "Results = $resultsPath" #Get/Create project registry keys $manufacturerReg = if ($softwareReg32 | Join-Path -ChildPath $manufacturer | Test-Path) { $softwareReg32 | Join-Path -ChildPath $manufacturer | Get-Item } else { New-Item -Path $softwareReg32.PSPath -Name $manufacturer Write-Verbose "Created manufacturer key: $manufacturer" } $projectReg = if ($manufacturerReg | Join-Path -ChildPath $projectName | Test-Path) { $manufacturerReg | Join-Path -ChildPath $projectName | Get-Item } else { New-Item -Path $manufacturerReg.PSPath -Name $projectName Write-Verbose "Created project key: $projectName" } #Get properties from Application BTDF registry try { $version = $projectReg.GetValue("InstalledVersion") if (-not $version) { throw } Write-Debug "Version = $version" } catch { Write-Warning "Unable to find version" } #endregion #region Run EnvironmentSettingsExporter if (($DeploymentType -eq "Deploy" -or $DeploymentType -eq "PreProcessBindings") -and ($Configuration -eq "Server")) { $envSettingsDir = $deployment | Join-Path -ChildPath "EnvironmentSettings" | Get-Item Write-Debug "Environment Settings Dir = $($envSettingsDir.FullName)" $settingsExporter = Join-Path -Path $deployment.FullName -ChildPath "Framework\DeployTools\EnvironmentSettingsExporter.exe" Write-Debug "Settings Exporter = $settingsExporter" if ($PSCmdlet.ShouldProcess($settingsExporter, "Exporting BTDF settings")) { Invoke-Process -FilePath $settingsExporter ` -ArgumentList "`"$(Join-Path -Path $envSettingsDir.FullName -ChildPath \SettingsFileGenerator.xml)`"", "`"$($envSettingsDir.FullName)`"" | Out-Null } $envSettings = Join-Path -Path $envSettingsDir -ChildPath "Exported_$Environment`Settings.xml" | Get-Item Write-Debug "Environment Settings = $($envSettings.FullName)" } #endregion #region Run MSBuild and deploy if ($backRefsRemoved) { $projectReg | New-ItemProperty -Name "Configuration" -Value $Configuration -Force | Out-Null $projectReg | New-ItemProperty -Name "LastDeploySettingsFilePath" -Value $envSettings.FullName -Force | Out-Null $projectReg | New-ItemProperty -Name "Version" -Value $version -Force | Out-Null Write-Verbose "$($btdfTargets[$DeploymentType]["Message"]): $projectName" $msbuildArgs = "/t:$($btdfTargets[$DeploymentType]["Target"])", $(if ($version) { "/p:ProductVersion=$version" }), $(if (($DeploymentType -eq "Deploy") -and ($Configuration -eq "Server")) { "/p:ENV_SETTINGS=`"$($envSettings)`"" }), $(if ($PSBoundParameters.ContainsKey("TerminateInstances")) { "/p:AutoTerminateInstances=$TerminateInstances" }), $(if ($Configuration) { "/p:Configuration=$Configuration" }), $(if ($Configuration -eq "Server") { "/p:DeployBizTalkMgmtDB=$DeployBTMgmtDB" }), $(if ($PSBoundParameters.ContainsKey("SkipBizTalkRestart")) { "/p:SkipHostInstancesRestart=$SkipBizTalkRestart" }), $(if ($PSBoundParameters.ContainsKey("SkipIISRestart")) { "/p:SkipIISReset=$SkipIISRestart" }), $(if ($PSBoundParameters.ContainsKey("SkipUnDeploy")) { "/p:SkipUndeploy=$SkipUnDeploy" }), $(if ($PSBoundParameters.ContainsKey("SkipApplicationStart")) { "/p:StartApplicationOnDeploy=$(-not $SkipApplicationStart)" "/p:StartReferencedApplicationsOnDeploy=$(-not $SkipApplicationStart)" }), "/l:FileLogger,Microsoft.Build.Engine;logfile=`"$resultsPath`"" if ($PSCmdlet.ShouldProcess($btdfProject, "Deploy BTDF packaged application")) { Invoke-MSBuild -Project $btdfProject ` -ArgumentList $msbuildArgs ` -Version 4.0 ` -Run32Bit ` -ErrorAction Stop } } #endregion #region Restore back referenced applications in reverse if ($SkipRestore -and ($backRefs.Count -gt 0)) { Write-Verbose "Skipping restore" } if (($backRefs.Count -gt 0) -and ($DeploymentType -eq "Deploy") -and (-not $SkipRestore)) { Write-Debug "Back References Count = $($backRefs.Count)" while (($backRefs.Count -gt 0) -and ($DeploymentType -eq "Deploy")) { $app = $backRefs.Pop() Write-Verbose "Restoring: $($app.Name)" $deployParams = $PSBoundParameters $deployParams["ProjectPath"] = $app.ProjectPath $deployParams["DeploymentType"] = $DeploymentType $deployParams["Configuration"] = $Configuration if ($PSCmdlet.ShouldProcess($app.Name, "Restoring BizTalk Application")) { Deploy-BTDFApplication @deployParams } } } #endregion } } } <# .SYNOPSIS Installs a BTDF packaged BizTalk application to the specified directory and deploys .DESCRIPTION Requires Admin rights locally and for BizTalk. Locally installs a BTDF packaged BizTalk application to the specified directory and deploys. Deployment can be configured to support multi-server setups (install and gac assemblies on secondary nodes and register to BizTalk on primary node). This also takes into account applications which references an application to be upgraded and will strip and redeploy these. Supports configuration of multiple environments which are validated (cmdlet will need to be modified to support additional environments) .PARAMETER MsiFile Path to the MsiFile .PARAMETER TargetDir Path for the directory to install the application. The application will be installed WITHIN the specified folder rather than autocreating a folder .PARAMETER DeploymentType Choose whether to DEPLOY and UNDEPLOY the BizTalk application (Default is DEPLOY) .PARAMETER Environment Select the environment to deploy. Must match what is configured in BTDF environmentsettings spreadsheet(Defaults to Local)(Alter validated set if needed) .PARAMETER DeployBTMgmtDB Registers application in BizTalk (Without this only the assemblies/components are GACed) .PARAMETER SkipUnDeploy Skip undeploying application prior to deployment (deploy only instead of redeploy) .PARAMETER SkipBizTalkRestart If supported by deployment project, skip restarting BizTalk Host instances post-deployment .PARAMETER SkipIISRestart If supported by deployment project, skip IISReset post-deployment .PARAMETER SkipApplicationStart If supported by deployment project, don't start application after deployment .PARAMETER TerminateInstances If supported by deployment project, terminate running/suspended instances .PARAMETER SkipRestore Skips restoring dependant applciations, if any .EXAMPLE Install application components/assemblies C:\PS> Install-BTDFApplication -MsiFile C:\Deployments\Scratchpad.msi -TargetDir "C:\Program Files (x86)\Scratchpad" .EXAMPLE Install application and register in BizTalk C:\PS> Install-BTDFApplication -MsiFile C:\Deployments\Scratchpad.msi -TargetDir "C:\Program Files (x86)\Scratchpad" -DeployBTMgmtDB .EXAMPLE Install application and register in BizTalk with live environment settings C:\PS> Install-BTDFApplication -MsiFile C:\Deployments\Scratchpad.msi -TargetDir "C:\Program Files (x86)\Scratchpad" -Environment Prod -DeployBTMgmtDB .EXAMPLE Undeploys application and removes from BizTalk C:\PS> Install-BTDFApplication -MsiFile C:\Deployments\Scratchpad.msi -TargetDir "C:\Program Files (x86)\Scratchpad" -DeploymentType UnDeploy -DeployBTMgmtDB #> function Install-BTDFApplication { [CmdletBinding(SupportsShouldProcess = $true)] Param ( [Parameter(Position = 0, Mandatory = $true, HelpMessage = "Path to existing BTDF MSI")] [ValidateScript( { Test-Path -Path $_.Fullname -PathType Leaf -Include *.msi })] [System.IO.FileInfo]$MsiFile, [Parameter(Position = 1, Mandatory = $true, HelpMessage = "Path wherein the resource files will be installed")] [System.IO.DirectoryInfo]$TargetDir, [Parameter(HelpMessage = "Valid parameters as Deploy, Undeploy, UpdateOrchestration. Defaults to Deploy")] [ValidateSet("Deploy", "UnDeploy", "UpdateOrchestration")] [string]$DeploymentType = "Deploy", [Parameter(HelpMessage = "Valid parameters are Local, Dev, Int, UAT, Train and Prod. Defaults to Local")] [ValidateSet("Local", "Dev", "Int", "UAT", "Train", "Prod")] [string]$Environment = "Local", [Parameter(HelpMessage = "Register artifacts in BizTalk DB")] [switch]$DeployBTMgmtDB, [Parameter(HelpMessage = "Only deploy instead of redeploy")] [switch]$SkipUnDeploy, [Parameter(HelpMessage = "Skip starting BizTalk hosts back up post deployment")] [switch]$SkipBizTalkRestart, [Parameter(HelpMessage = "Skip IISReset post deployment")] [switch]$SkipIISRestart, [Parameter(HelpMessage = "Skipping starting BizTalk app post deployment")] [switch]$SkipApplicationStart, [Parameter(HelpMessage = "Terminate instances related to deployed app")] [switch]$TerminateInstances, [Parameter(HelpMessage = "Skip restoring dependant applications")] [switch]$SkipRestore ) Process { $TargetDir = Install-MsiFile -MsiFile $MsiFile -TargetDir $TargetDir #region Build splat params $splatParams = @{ } if ($PSBoundParameters.ContainsKey("DeployBTMgmtDB")) { $splatParams.Add("DeployBTMgmtDB", $DeployBTMgmtDB) } if ($PSBoundParameters.ContainsKey("SkipUndeploy")) { $splatParams.Add("SkipUndeploy", $SkipUndeploy) } if ($PSBoundParameters.ContainsKey("SkipBizTalkRestart")) { $splatParams.Add("SkipBizTalkRestart", $SkipBizTalkRestart) } if ($PSBoundParameters.ContainsKey("SkipIISRestart")) { $splatParams.Add("SkipIISRestart", $SkipIISRestart) } if ($PSBoundParameters.ContainsKey("SkipApplicationStart")) { $splatParams.Add("SkipApplicationStart", $SkipApplicationStart) } if ($PSBoundParameters.ContainsKey("TerminateInstances")) { $splatParams.Add("TerminateInstances", $TerminateInstances) } if ($PSBoundParameters.ContainsKey("SkipRestore")) { $splatParams.Add("SkipRestore", $SkipRestore) } #endregion Deploy-BTDFApplication -ProjectPath $TargetDir ` -Configuration "Server" ` -DeploymentType $DeploymentType ` -Environment $Environment ` @splatParams } } #endregion #region SSO <# .SYNOPSIS Read BTDF SSO settings for deployed BizTalk applications .DESCRIPTION Makes use of BTDF SSOSettingsFileReader helper assembly to read all SSO settings for a deployed application .PARAMETER Application Name of the BTDF deployed application to read SSO settings .EXAMPLE PS C:\> Read-SSO -Application Scratchpad Returns a hashtable of SSO settings for the Scratchpad application .OUTPUTS Hashtable #> function Read-SSO { [CmdletBinding()] [OutputType([hashtable])] Param ( [Parameter(Mandatory = $true, Position = 0, ValueFromPipeline = $true)] [string]$Application ) Process { return [SSOSettingsFileReader]::Read($Application) } } <# .SYNOPSIS Read BTDF SSO settings for deployed BizTalk applications .DESCRIPTION Makes use of BTDF SSOSettingsFileReader helper assembly to read all SSO settings for a deployed application .PARAMETER Application Name of the BTDF deployed application to read SSO settings .PARAMETER Value Name of the SSO setting to get the value of .EXAMPLE PS C:\> Read-SSO -Application Scratchpad -Value NumberSetting Returns a int typed value from SSO for the Scatchpad appliction for the setting "NumberSetting" .OUTPUTS int #> function Read-SSOInt32 { [CmdletBinding()] [OutputType([int])] Param ( [Parameter(Mandatory = $true, ValueFromPipeline = $true, Position = 0)] [string]$Application, [Parameter(Mandatory = $true, Position = 1)] [string]$Value ) Process { return [SSOSettingsFileReader]::ReadInt32($Application, $Value) } } <# .SYNOPSIS Read BTDF SSO settings for deployed BizTalk applications .DESCRIPTION Makes use of BTDF SSOSettingsFileReader helper assembly to read all SSO settings for a deployed application .PARAMETER Application Name of the BTDF deployed application to read SSO settings .PARAMETER Value Name of the SSO setting to get the value of .EXAMPLE PS C:\> Read-SSO -Application Scratchpad -Value StringSetting Returns a string typed value from SSO for the Scatchpad appliction for the setting "StringSetting" .OUTPUTS string #> function Read-SSOString { [CmdletBinding()] [OutputType([string])] Param ( [Parameter(Mandatory = $true, ValueFromPipeline = $true, Position = 0)] [string]$Application, [Parameter(Mandatory = $true, Position = 1)] [string]$Value ) Process { return [SSOSettingsFileReader]::ReadString($Application, $Value) } } <# .SYNOPSIS Write settings to BTDF SSO for deployed BizTalk applications .DESCRIPTION Makes use of BTDF SSOSettingsFileReader helper assembly to read all SSO settings for a deployed application .PARAMETER Application Name of the BTDF deployed application to read SSO settings .PARAMETER Setting Name of the setting to write to. Will create the setting if it doesn't already exist .PARAMETER Value Name of the SSO setting to get the value of .EXAMPLE PS C:\> Write-SSOSetting -Application Scratchpad -Setting StringSetting -Value StringValue Returns a string typed value from SSO for the Scatchpad appliction for the setting "StringSetting" .OUTPUTS string #> function Write-SSOSetting { [CmdletBinding()] Param ( [Parameter(Mandatory = $true, ValueFromPipeline = $true, Position = 0)] [string]$Application, [Parameter(Mandatory = $true, Position = 1)] [string]$Setting, [Parameter(Mandatory = $true, Position = 2)] [string]$Value ) Process { [SSOSettingsFileManager.SSOSettingsManager]::WriteSetting($Application, $Setting, $Value) } } #endregion #region Private function Connect-BTSCatalog { [CmdletBinding()] [OutputType([Microsoft.BizTalk.ExplorerOM.BtsCatalogExplorer])] Param ( [Parameter()] [string]$Server = $env:COMPUTERNAME, [Parameter()] [string]$SQLInstance, [Parameter()] [string]$ManagementDB ) Process { #Calculate BizTalk connection string $wmi = Get-WmiObject -Class MSBTS_GroupSetting -Namespace root\MicrosoftBizTalkServer -ComputerName $server -ErrorAction Stop if (-not $SQLInstance) { $SQLInstance = $wmi.MgmtDbServerName } if (-not $ManagementDB) { $ManagementDB = $wmi.MgmtDbName } $btsCatalog = [Microsoft.BizTalk.ExplorerOM.BtsCatalogExplorer]::new() $btsCatalog.ConnectionString = "SERVER=$SQLInstance;DATABASE=$ManagementDB;Integrated Security=SSPI" return $btsCatalog } } Function Install-MsiFile { [CmdletBinding()] Param ( [Parameter(Mandatory = $true, ValueFromPipeline = $true, ValueFromPipelinebyPropertyName = $true)] [ValidateScript( { Test-Path -Path $_ -PathType Leaf -Include *.msi })] [string]$MsiFile, [Parameter()] [ValidateNotNullorEmpty()] [string]$TargetDir, [Parameter()] [ValidateNotNullorEmpty()] [System.Collections.Generic.List[string]]$ArgumentList ) Process { $ArgumentList += "/i", "`"$msiFile`"", $(if ($targetDir) { "INSTALLDIR=`"$targetDir`"" }), "ADDLOCAL=ALL", "/qn", "/norestart" Write-Verbose "Installing MSI File..." $process = Invoke-Process -FilePath "$env:windir\System32\msiexec.exe" -ArgumentList $ArgumentList switch ($process.ExitCode) { 0 { Write-Verbose "MSI been successfully installed" } Default { Write-Error "Installing $MsiFile failed!, Exit Code: $($process.ExitCode)" } } return $TargetDir } } function Invoke-MSBuild { [CmdletBinding()] Param ( [Parameter(Mandatory = $true, ValueFromPipeline = $true, Position = 0)] [ValidateScript( { Test-Path -Path $_ -PathType Leaf -Include *.*proj })] [string]$Project, [Parameter(Position = 1)] [System.Collections.Generic.List[string]]$ArgumentList, [Parameter()] [switch]$Run32Bit # [Parameter()] # [ValidateNotNullOrEmpty()] # [ValidateScript({[version]::Parse($_)})] # [string]$Version ) DynamicParam { $msbuildReg = if ($Run32Bit -and ($env:PROCESSOR_ARCHITECTURE -eq "AMD64")) { Get-Item -Path "HKLM:\SOFTWARE\Wow6432Node\Microsoft\MSBuild\ToolsVersions" } else { Get-Item -Path "HKLM:\SOFTWARE\Microsoft\MSBuild\ToolsVersions" } $versions = $msbuildReg | Get-ChildItem | Get-ItemProperty | Sort-Object -Property @{Expression = { [version]::Parse($_.PSChildName) } } -Descending $paramDict = New-Object System.Management.Automation.RuntimeDefinedParameterDictionary #region Version $attributeCollection = New-Object System.Collections.ObjectModel.Collection[System.Attribute] $parameter = New-Object System.Management.Automation.ParameterAttribute $attributeCollection.Add($parameter) $validateSet = New-Object System.Management.Automation.ValidateSetAttribute($versions | Select-Object -ExpandProperty PSChildName) $attributeCollection.Add($validateSet) $versionParameter = New-Object System.Management.Automation.RuntimeDefinedParameter("Version", [string], $attributeCollection) $paramDict.Add("Version", $versionParameter) #endregion return $paramDict } Process { $versionsHash = @{ } $versions | ForEach-Object { $versionsHash[$($_.PSChildName)] = $_.MSBuildToolsPath } $version = if (-not $PSBoundParameters.ContainsKey("Version")) { $versions[0].PSChildName } else { $PSBoundParameters.Version } Write-Debug "Version = $version" $msbuildFolder = $versionsHash[$version] Write-Debug "MSBuildFolder = $msbuildFolder" $msbuild = Join-Path -Path $msbuildFolder -ChildPath "msbuild.exe" $ArgumentList = "`"$Project`"", $ArgumentList, "/p:UseSharedCompilation=false", "/noLogo", "/maxcpucount:1", "/nodeReuse:false" Invoke-Process -FilePath $msbuild -ArgumentList $ArgumentList | Out-Null } } function Invoke-Process { [CmdletBinding()] Param ( [Parameter(Mandatory = $true, ValueFromPipeline = $true)] [ValidateNotNullOrEmpty()] [ValidateScript( { Test-Path -Path $_ -PathType Leaf -Include *.* })] [System.IO.FileInfo]$FilePath, [Parameter()] [ValidateNotNullOrEmpty()] [string[]]$ArgumentList ) Process { $stdOutTempFile = "$env:TEMP\$((New-Guid).Guid)" $stdErrTempFile = "$env:TEMP\$((New-Guid).Guid)" Write-Verbose "Invoking $($FilePath.Name)" $processParams = @{ "FilePath" = $FilePath.FullName "Wait" = $true "PassThru" = $true "NoNewWindow" = $true } Write-Debug "FilePath = $($FilePath.FullName)" if ($ArgumentList) { $processParams.Add("ArgumentList", $ArgumentList) Write-Debug "ArgumentList = $ArgumentList" } #Redirect output if executed remotely if ((Get-Host).Name -eq "ServerRemoteHost") { $processParams["RedirectStandardOutput"] = $stdOutTempFile $processParams["RedirectStandardError"] = $stdErrTempFile Write-Debug "TempStdOutput = $stdOutTempFile" Write-Debug "TempStdError = $stdErrTempFile" } try { $process = Start-Process @processParams switch ($process.ExitCode) { 0 { Write-Verbose "Process has been successfully run" return $process } Default { throw "Process failed!, Exit Code: $($process.ExitCode)" } } } catch { throw $_ } finally { if ((Get-Host).Name -eq "ServerRemoteHost") { $processOutput = (Get-Content -Path $stdOutTempFile -Raw) $processError = Get-Content -Path $stdErrTempFile -Raw if ($processOutput) { Write-Information $processOutput } if ($processError) { Write-Error $processError } } Remove-Item -Path $stdOutTempFile, $stdErrTempFile ` -Force ` -Verbose:$false ` -ErrorAction Ignore ` } } } #endregion #region Variables $btsCatalog = Connect-BTSCatalog $projectPathColumn = @{Name = "ProjectPath"; Expression = { Read-SSOString -Application $_.Name -Value "ProjectPath" } } $softwareReg32 = if ($env:PROCESSOR_ARCHITECTURE -eq "AMD64") { Get-Item -Path HKLM:\SOFTWARE\Wow6432Node } else { Get-Item -Path HKLM:\SOFTWARE } #endregion |