AppHandling/Compile-AppInNavContainer.ps1
<#
.Synopsis Use NAV/BC Container to Compile App .Description .Parameter containerName Name of the container in which you want to use to compile the app .Parameter tenant tenant to use if container is multitenant .Parameter credential Credentials of the SUPER user if using NavUserPassword authentication .Parameter appProjectFolder Location of the project. This folder (or any of its parents) needs to be shared with the container. .Parameter appOutputFolder Folder in which the output will be placed. This folder (or any of its parents) needs to be shared with the container. Default is $appProjectFolder\output. .Parameter appSymbolsFolder Folder in which the symbols of dependent apps will be placed. This folder (or any of its parents) needs to be shared with the container. Default is $appProjectFolder\symbols. .Parameter appName File name of the app. Default is to compose the file name from publisher_appname_version from app.json. .Parameter basePath Base Path of the files in the ALC output, to convert file paths to relative paths. This folder (or any of its parents) needs to be shared with the container. .Parameter UpdateSymbols Add this switch to indicate that you want to force the download of symbols for all dependent apps. .Parameter UpdateDependencies Update the dependency version numbers to the actual version number used during compilation .Parameter CopySymbolsFromContainer Add this switch to copy system and base application symbols from container to speed up symbol download. .Parameter CopyAppToSymbolsFolder Add this switch to copy the compiled app to the appSymbolsFolder. .Parameter GenerateReportLayout Add this switch to invoke report layout generation during compile. Default is default alc.exe behavior, which is to generate report layout .Parameter AzureDevOps Add this switch to convert the output to Azure DevOps Build Pipeline compatible output .Parameter gitHubActions Include this switch to convert the output to GitHub Actions compatible output .Parameter EnableCodeCop Add this switch to Enable CodeCop to run .Parameter EnableAppSourceCop Add this switch to Enable AppSourceCop to run .Parameter EnablePerTenantExtensionCop Add this switch to Enable PerTenantExtensionCop to run .Parameter EnableUICop Add this switch to Enable UICop to run .Parameter RulesetFile Specify a ruleset file for the compiler .Parameter enableExternalRulesets Add this switch to Enable External Rulesets .Parameter CustomCodeCops Add custom AL code Cops when compiling apps. .Parameter Failon Specify if you want Compilation to fail on Error or Warning .Parameter nowarn Specify a nowarn parameter for the compiler .Parameter generateErrorLog Switch parameter on whether to generate an alerts log file. Default is false. The file will be placed in the same folder as the app file, and will have the same name as the app file, but with the extension .errorLog.json. .Parameter preProcessorSymbols PreProcessorSymbols to set when compiling the app. .Parameter generatecrossreferences Include this flag to generate cross references when compiling .Parameter reportSuppressedDiagnostics Set reportSuppressedDiagnostics flag on ALC when compiling to ignore pragma warning disables .Parameter bcAuthContext Authorization Context created by New-BcAuthContext. By specifying BcAuthContext and environment, the compile function will use the online Business Central Environment as target for the compilation .Parameter environment Environment to use for the compilation. .Parameter assemblyProbingPaths Specify a comma separated list of paths to include in the search for dotnet assemblies for the compiler .Parameter SourceRepositoryUrl Repository holding the source code for the app. Will be stamped into the app manifest. .Parameter SourceCommit The commit identifier for the source code for the app. Will be stamped into the app manifest. .Parameter BuildBy Information about which product built the app. Will be stamped into the app manifest. .Parameter BuildUrl The URL for the build job, which built the app. Will be stamped into the app manifest. .Parameter OutputTo Compiler output is sent to this scriptblock for output. Default value for the scriptblock is: { Param($line) Write-Host $line } .Example Compile-AppInBcContainer -containerName test -credential $credential -appProjectFolder "C:\Users\freddyk\Documents\AL\Test" .Example Compile-AppInBcContainer -containerName test -appProjectFolder "C:\Users\freddyk\Documents\AL\Test" .Example Compile-AppInBcContainer -containerName test -appProjectFolder "C:\Users\freddyk\Documents\AL\Test" -outputTo { Param($line) if ($line -notlike "*sourcepath=C:\Users\freddyk\Documents\AL\Test\Org\*") { Write-Host $line } } #> function Compile-AppInBcContainer { Param ( [string] $containerName = $bcContainerHelperConfig.defaultContainerName, [Parameter(Mandatory=$false)] [string] $tenant = "default", [Parameter(Mandatory=$false)] [PSCredential] $credential = $null, [Parameter(Mandatory=$true)] [string] $appProjectFolder, [Parameter(Mandatory=$false)] [string] $appOutputFolder = (Join-Path $appProjectFolder "output"), [Parameter(Mandatory=$false)] [string] $appSymbolsFolder = (Join-Path $appProjectFolder ".alpackages"), [Parameter(Mandatory=$false)] [string] $appName = "", [string] $basePath = "", [switch] $UpdateSymbols, [switch] $UpdateDependencies, [switch] $CopySymbolsFromContainer, [switch] $CopyAppToSymbolsFolder, [ValidateSet('Yes','No','NotSpecified')] [string] $GenerateReportLayout = 'NotSpecified', [switch] $AzureDevOps, [switch] $gitHubActions, [switch] $EnableCodeCop, [switch] $EnableAppSourceCop, [switch] $EnablePerTenantExtensionCop, [switch] $EnableUICop, [ValidateSet('none','error','warning')] [string] $FailOn = 'none', [Parameter(Mandatory=$false)] [string] $rulesetFile, [switch] $enableExternalRulesets, [string[]] $CustomCodeCops = @(), [Parameter(Mandatory=$false)] [string] $nowarn, [switch] $generateErrorLog, [string[]] $preProcessorSymbols = @(), [switch] $GenerateCrossReferences, [switch] $ReportSuppressedDiagnostics, [Parameter(Mandatory=$false)] [string] $assemblyProbingPaths, [Parameter(Mandatory=$false)] [ValidateSet('ExcludeGeneratedTranslations','GenerateCaptions','GenerateLockedTranslations','NoImplicitWith','TranslationFile','LcgTranslationFile')] [string[]] $features = @(), [Hashtable] $bcAuthContext, [string] $environment, [string[]] $treatWarningsAsErrors = $bcContainerHelperConfig.TreatWarningsAsErrors, [string] $sourceRepositoryUrl = '', [string] $sourceCommit = '', [string] $buildBy = "BcContainerHelper,$BcContainerHelperVersion", [string] $buildUrl = '', [scriptblock] $outputTo = { Param($line) Write-Host $line } ) $telemetryScope = InitTelemetryScope -name $MyInvocation.InvocationName -parameterValues $PSBoundParameters -includeParameters @() try { $startTime = [DateTime]::Now $platform = Get-BcContainerPlatformversion -containerOrImageName $containerName if ("$platform" -eq "") { $platform = (Get-BcContainerNavVersion -containerOrImageName $containerName).Split('-')[0] } [System.Version]$platformversion = $platform $containerProjectFolder = Get-BcContainerPath -containerName $containerName -path $appProjectFolder if ("$containerProjectFolder" -eq "") { throw "The appProjectFolder ($appProjectFolder) is not shared with the container." } $containerFolder = Get-BcContainerPath -containerName $containerName -path (Join-Path $bcContainerHelperConfig.hostHelperFolder "Extensions\$containerName") if (!$PSBoundParameters.ContainsKey("assemblyProbingPaths")) { if ($platformversion.Major -ge 13) { $assemblyProbingPaths = Invoke-ScriptInBcContainer -containerName $containerName -ScriptBlock { Param($containerFolder, $appProjectFolder, $platformVersion) $assemblyProbingPaths = "" $netpackagesPath = Join-Path $appProjectFolder ".netpackages" if (Test-Path $netpackagesPath) { $assemblyProbingPaths += """$netpackagesPath""," } $roleTailoredClientFolder = "C:\Program Files (x86)\Microsoft Dynamics NAV\*\RoleTailored Client" if (Test-Path $roleTailoredClientFolder) { $assemblyProbingPaths += """$((Get-Item $roleTailoredClientFolder).FullName)""," } $serviceTierFolder = (Get-Item "C:\Program Files\Microsoft Dynamics NAV\*\Service").FullName if ($platformversion.Major -ge 22) { $dotnetAssembliesFolder = Join-Path $containerFolder ".netPackages\Service" if (!(Test-Path $dotnetAssembliesFolder)) { New-Item $dotnetAssembliesFolder -ItemType Directory -Force | Out-Null Write-Host "Copying DLLs from $serviceTierFolder to assemblyProbingPath" Copy-Item -Path $serviceTierFolder -filter '*.dll' -Destination $dotnetAssembliesFolder -Recurse -Force -ErrorAction SilentlyContinue $mockAssembliesPath = "C:\Test Assemblies\Mock Assemblies" Copy-Item -Path $mockAssembliesPath -filter '*.dll' -Destination $dotnetAssembliesFolder -Recurse -Force -ErrorAction SilentlyContinue Write-Host "Removing dotnet Framework Assemblies" $dotnetServiceFolder = Join-Path $dotnetAssembliesFolder "Service" Remove-Item -Path (Join-Path $dotnetserviceFolder 'Management') -Recurse -Force -ErrorAction SilentlyContinue Remove-Item -Path (Join-Path $dotnetserviceFolder 'SideServices') -Recurse -Force -ErrorAction SilentlyContinue Remove-Item -Path (Join-Path $dotnetserviceFolder 'WindowsServiceInstaller') -Recurse -Force -ErrorAction SilentlyContinue } $assemblyProbingPaths += """$dotnetAssembliesFolder""" $assemblyProbingPaths = """C:\Program Files\dotnet\shared"",$assemblyProbingPaths" } else { $assemblyProbingPaths += """$serviceTierFolder"",""C:\Program Files (x86)\Open XML SDK\V2.5\lib""" $assemblyProbingPaths += ',"c:\Windows\Microsoft.NET\Assembly"' $mockAssembliesPath = "C:\Test Assemblies\Mock Assemblies" if (Test-Path $mockAssembliesPath -PathType Container) { $assemblyProbingPaths += ",""$mockAssembliesPath""" } } $assemblyProbingPaths } -ArgumentList $containerFolder, $containerProjectFolder, $platformversion } } $containerOutputFolder = Get-BcContainerPath -containerName $containerName -path $appOutputFolder if ("$containerOutputFolder" -eq "") { throw "The appOutputFolder ($appOutputFolder) is not shared with the container." } $containerSymbolsFolder = Get-BcContainerPath -containerName $containerName -path $appSymbolsFolder if ("$containerSymbolsFolder" -eq "") { throw "The appSymbolsFolder ($appSymbolsFolder) is not shared with the container." } $containerRulesetFile = "" if ($rulesetFile) { $containerRulesetFile = Get-BcContainerPath -containerName $containerName -path $rulesetFile if ("$containerRulesetFile" -eq "") { throw "The rulesetFile ($rulesetFile) is not shared with the container." } } $CustomCodeCopFiles = @() if ($CustomCodeCops.Count -gt 0) { $CustomCodeCops | ForEach-Object { $customCopPath = Get-BcContainerPath -containerName $containerName -path $_ if ("$customCopPath" -eq "") { throw "The custom code cop ($_) is not shared with the container." } $CustomCodeCopFiles += $customCopPath } } if (!(Test-Path $appOutputFolder -PathType Container)) { New-Item $appOutputFolder -ItemType Directory | Out-Null } $appJsonFile = Join-Path $appProjectFolder 'app.json' $appJsonObject = [System.IO.File]::ReadAllLines($appJsonFile) | ConvertFrom-Json if ("$appName" -eq "") { $appName = "$($appJsonObject.Publisher)_$($appJsonObject.Name)_$($appJsonObject.Version).app".Split([System.IO.Path]::GetInvalidFileNameChars()) -join '' } if ([bool]($appJsonObject.PSobject.Properties.name -eq "id")) { AddTelemetryProperty -telemetryScope $telemetryScope -key "id" -value $appJsonObject.id } elseif ([bool]($appJsonObject.PSobject.Properties.name -eq "appid")) { AddTelemetryProperty -telemetryScope $telemetryScope -key "id" -value $appJsonObject.appid } AddTelemetryProperty -telemetryScope $telemetryScope -key "publisher" -value $appJsonObject.Publisher AddTelemetryProperty -telemetryScope $telemetryScope -key "name" -value $appJsonObject.Name AddTelemetryProperty -telemetryScope $telemetryScope -key "version" -value $appJsonObject.Version AddTelemetryProperty -telemetryScope $telemetryScope -key "appname" -value $appName Write-Host "Using Symbols Folder: $appSymbolsFolder" if (!(Test-Path -Path $appSymbolsFolder -PathType Container)) { New-Item -Path $appSymbolsFolder -ItemType Directory | Out-Null } if ($CopySymbolsFromContainer) { Invoke-ScriptInBcContainer -containerName $containerName -scriptblock { Param($appSymbolsFolder) if (Test-Path "C:\Extensions\*.app") { $paths = @( "C:\Program Files\Microsoft Dynamics NAV\*\AL Development Environment\System.app" "C:\Extensions\*.app" ) } else { $paths = @( "C:\Program Files\Microsoft Dynamics NAV\*\AL Development Environment\System.app" "C:\Applications.*\Microsoft_Application_*.app,C:\Applications\Application\Source\Microsoft_Application.app" "C:\Applications.*\Microsoft_Base Application_*.app,C:\Applications\BaseApp\Source\Microsoft_Base Application.app" "C:\Applications.*\Microsoft_System Application_*.app,C:\Applications\System Application\source\Microsoft_System Application.app" ) } $paths | ForEach-Object { $appFiles = $_.Split(',') $appFile = "" if (Test-Path -Path $appFiles[0]) { $appFile = $appFiles[0] } elseif (Test-Path -path $appFiles[1]) { $appFile = $appFiles[1] } if ($appFile) { Get-Item -Path $appFile | ForEach-Object { Write-Host "Copying $([System.IO.Path]::GetFileName($_.FullName)) from Container" Copy-Item -Path $_.FullName -Destination $appSymbolsFolder -Force } } } } -argumentList $containerSymbolsFolder } $GenerateReportLayoutParam = "" if (($GenerateReportLayout -ne "NotSpecified") -and ($platformversion.Major -ge 14)) { if ($GenerateReportLayout -eq "Yes") { $GenerateReportLayoutParam = "/GenerateReportLayout+" } else { $GenerateReportLayoutParam = "/GenerateReportLayout-" } } # unpack compiler Invoke-ScriptInBcContainer -containerName $containerName -ScriptBlock { if (!(Test-Path "c:\build" -PathType Container)) { $tempZip = Join-Path ([System.IO.Path]::GetTempPath()) "alc.zip" Copy-item -Path (Get-Item -Path "c:\run\*.vsix").FullName -Destination $tempZip Expand-Archive -Path $tempZip -DestinationPath "c:\build\vsix" } } $customConfig = Get-BcContainerServerConfiguration -ContainerName $containerName $dependencies = @() if (([bool]($appJsonObject.PSobject.Properties.name -eq "application")) -and $appJsonObject.application) { AddTelemetryProperty -telemetryScope $telemetryScope -key "application" -value $appJsonObject.application $dependencies += @{"publisher" = "Microsoft"; "name" = "Application"; "appId" = ''; "version" = $appJsonObject.application } } if (([bool]($appJsonObject.PSobject.Properties.name -eq "platform")) -and $appJsonObject.platform) { AddTelemetryProperty -telemetryScope $telemetryScope -key "platform" -value $appJsonObject.platform $dependencies += @{"publisher" = "Microsoft"; "name" = "System"; "appId" = ''; "version" = $appJsonObject.platform } } if (([bool]($appJsonObject.PSobject.Properties.name -eq "test")) -and $appJsonObject.test) { $dependencies += @{"publisher" = "Microsoft"; "name" = "Test"; "appId" = ''; "version" = $appJsonObject.test } if (([bool]($customConfig.PSobject.Properties.name -eq "EnableSymbolLoadingAtServerStartup")) -and ($customConfig.EnableSymbolLoadingAtServerStartup -eq "true")) { throw "app.json should NOT have a test dependency when running hybrid development (EnableSymbolLoading)" } } if (([bool]($appJsonObject.PSobject.Properties.name -eq "dependencies")) -and $appJsonObject.dependencies) { $appJsonObject.dependencies | ForEach-Object { $dep = $_ try { $appId = $dep.id } catch { $appId = $dep.appId } $dependencies += @{ "publisher" = $dep.publisher; "name" = $dep.name; "appId" = $appId; "version" = $dep.version } } } $existingApps = @() if (!$updateSymbols) { $existingApps = Invoke-ScriptInBcContainer -containerName $containerName -ScriptBlock { Param($appSymbolsFolder) Get-ChildItem -Path (Join-Path $appSymbolsFolder '*.app') | ForEach-Object { $appInfo = Get-NavAppInfo -Path $_.FullName $appInfo } } -ArgumentList $containerSymbolsFolder } $publishedApps = @() if ($customConfig.ServerInstance) { $publishedApps = Invoke-ScriptInBcContainer -containerName $containerName -ScriptBlock { Param($tenant) Get-NavAppInfo -ServerInstance $ServerInstance -tenant $tenant Get-NavAppInfo -ServerInstance $ServerInstance -symbolsOnly } -ArgumentList $tenant | Where-Object { $_ -isnot [System.String] } } $applicationApp = $publishedApps | Where-Object { $_.publisher -eq "Microsoft" -and $_.name -eq "Application" } if (-not $applicationApp) { # locate application version number in database if using SQLEXPRESS try { if (($customConfig.DatabaseServer -eq "localhost") -and ($customConfig.DatabaseInstance -eq "SQLEXPRESS")) { $appVersion = Invoke-ScriptInBcContainer -containerName $containerName -scriptblock { Param($databaseName) (invoke-sqlcmd -ServerInstance 'localhost\SQLEXPRESS' -ErrorAction Stop -Query "SELECT [applicationversion] FROM [$databaseName].[dbo].[`$ndo`$dbproperty]").applicationVersion } -argumentList $customConfig.DatabaseName $publishedApps += @{ "Name" = "Application"; "Publisher" = "Microsoft"; "Version" = $appversion } } } catch { # ignore errors - use version number in app.json } } $sslVerificationDisabled = $false $serverInstance = $customConfig.ServerInstance $headers = @{} $useDefaultCredentials = $false $timeout = 100 if ($bcAuthContext -and $environment) { $bcAuthContext = Renew-BcAuthContext -bcAuthContext $bcAuthContext $bcEnvironment = Get-BcEnvironments -bcAuthContext $bcAuthContext | Where-Object { $_.Name -eq $environment -and $_.Type -eq "Sandbox" } if (!$bcEnvironment) { throw "Environment $environment doesn't exist in the current context or it is not a Sandbox environment." } $publishedApps = Get-BcPublishedApps -bcAuthContext $bcAuthContext -environment $environment | Where-Object { $_.state -eq "installed" } $devServerUrl = "$($bcContainerHelperConfig.apiBaseUrl.TrimEnd('/'))/v2.0/$environment" $bearerAuthValue = "Bearer $($bcAuthContext.AccessToken)" $headers."Authorization" = $bearerAuthValue } elseif ($serverInstance -eq "") { if ($updateSymbols) { Write-Host -ForegroundColor Yellow "INFO: You have to specify AuthContext and Environment if you are compiling in a filesOnly container in order to download dependencies" } $devServerUrl = "" } else { if ($customConfig.DeveloperServicesSSLEnabled -eq "true") { $protocol = "https://" } else { $protocol = "http://" } $ip = Get-BcContainerIpAddress -containerName $containerName if ($ip) { $devServerUrl = "$($protocol)$($ip):$($customConfig.DeveloperServicesPort)/$ServerInstance" } else { $devServerUrl = "$($protocol)$($containerName):$($customConfig.DeveloperServicesPort)/$ServerInstance" } $timeout = 300000 $sslVerificationDisabled = ($protocol -eq "https://") if ($customConfig.ClientServicesCredentialType -eq "Windows") { $useDefaultCredentials = $true } else { if (!($credential)) { throw "You need to specify credentials when you are not using Windows Authentication" } $pair = ("$($Credential.UserName):"+[System.Runtime.InteropServices.Marshal]::PtrToStringAuto([System.Runtime.InteropServices.Marshal]::SecureStringToBSTR($credential.Password))) $bytes = [System.Text.Encoding]::ASCII.GetBytes($pair) $base64 = [System.Convert]::ToBase64String($bytes) $basicAuthValue = "Basic $base64" $headers."Authorization" = $basicAuthValue } } $depidx = 0 while ($depidx -lt $dependencies.Count) { $dependency = $dependencies[$depidx] Write-Host "Processing dependency $($dependency.Publisher)_$($dependency.Name)_$($dependency.Version) ($($dependency.AppId))" $existingApp = $existingApps | Where-Object { if ($platformversion -ge [System.Version]"19.0.0.0") { ((($dependency.appId -ne '' -and $_.AppId.ToString() -eq $dependency.appId) -or ($dependency.appId -eq '' -and $_.Name -eq $dependency.Name)) -and ([System.Version]$_.Version -ge [System.Version]$dependency.version)) } else { (($_.Name -eq $dependency.name) -and ($_.Name -eq "Application" -or (($_.Publisher -eq $dependency.publisher) -and ([System.Version]$_.Version -ge [System.Version]$dependency.version)))) } } if ($existingApp) { Write-Host "Dependency App exists" } if ($updateSymbols -or !$existingApp) { $publisher = $dependency.publisher $name = $dependency.name $appId = $dependency.appId $version = $dependency.version $symbolsName = "$($publisher)_$($name)_$($version).app".Split([System.IO.Path]::GetInvalidFileNameChars()) -join '' $publishedApps | Where-Object { $_.publisher -eq $publisher -and $_.name -eq $name } | ForEach-Object { $symbolsName = "$($publisher)_$($name)_$($_.version).app".Split([System.IO.Path]::GetInvalidFileNameChars()) -join '' } if (($devServerUrl -eq "") -or ($headers -eq @{} -and !$useDefaultCredentials)) { Write-Host -ForegroundColor Yellow "WARNING: Unable to download symbols for $symbolsName" } else { $symbolsFile = Join-Path $appSymbolsFolder $symbolsName Write-Host "Downloading symbols: $symbolsName" $publisher = [uri]::EscapeDataString($publisher) $name = [uri]::EscapeDataString($name) if ($appId -and $platformversion -ge [System.Version]"20.0.0.0") { $url = "$devServerUrl/dev/packages?appId=$($appId)&versionText=$($version)&tenant=$tenant" } else { $url = "$devServerUrl/dev/packages?publisher=$($publisher)&appName=$($name)&versionText=$($version)&tenant=$tenant" } Write-Host "Url : $Url" try { DownloadFileLow -sourceUrl $url -destinationFile $symbolsFile -timeout $timeout -useDefaultCredentials:$useDefaultCredentials -Headers $headers -skipCertificateCheck:$sslVerificationDisabled } catch { $throw = $true if ($customConfig.ServerInstance -ne '' -and $customConfig.ClientServicesCredentialType -eq "Windows") { try { Invoke-ScriptInBcContainer -containerName $containerName -scriptblock { Param($url, $symbolsFile) $webClient = [System.Net.WebClient]::new() $webClient.UseDefaultCredentials = $true $webClient.DownloadFile($url, $symbolsFile) } -argumentList $url, (Get-BcContainerPath -containerName $containerName -path $symbolsFile) $throw = $false } catch { } } if ($throw) { Write-Host "ERROR $($_.Exception.Message)" throw (GetExtendedErrorMessage $_) } } if (Test-Path -Path $symbolsFile) { $addDependencies = Invoke-ScriptInBcContainer -containerName $containerName -ScriptBlock { Param($symbolsFile, $platformversion) # Wait for file to be accessible in container While (-not (Test-Path $symbolsFile)) { Start-Sleep -Seconds 1 } if ($platformversion.Major -ge 15) { Add-Type -AssemblyName System.IO.Compression.FileSystem Add-Type -AssemblyName System.Text.Encoding try { # Import types needed to invoke the compiler $alcPath = 'C:\build\vsix\extension\bin' $alToolExe = Join-Path $alcPath 'win32\altool.exe' $alToolExists = Test-Path -Path $alToolExe -PathType Leaf if ($alToolExists) { $manifest = & "$alToolExe" GetPackageManifest "$symbolsFile" | ConvertFrom-Json if ($manifest.PSObject.Properties.Name -eq 'application' -and $manifest.application) { @{ "publisher" = "Microsoft"; "name" = "Application"; "appId" = ''; "version" = $manifest.Application } } if ($manifest.PSObject.Properties.Name -eq 'dependencies') { foreach ($dependency in $manifest.dependencies) { @{ "publisher" = $dependency.Publisher; "name" = $dependency.name; "appId" = $dependency.id; "Version" = $dependency.Version } } } } else { Add-Type -Path (Join-Path $alcPath Newtonsoft.Json.dll) Add-Type -Path (Join-Path $alcPath System.Collections.Immutable.dll) if (Test-Path (Join-Path $alcPath System.IO.Packaging.dll)) { Add-Type -Path (Join-Path $alcPath System.IO.Packaging.dll) } Add-Type -Path (Join-Path $alcPath Microsoft.Dynamics.Nav.CodeAnalysis.dll) $packageStream = [System.IO.File]::OpenRead($symbolsFile) $package = [Microsoft.Dynamics.Nav.CodeAnalysis.Packaging.NavAppPackageReader]::Create($PackageStream, $true) $manifest = $package.ReadNavAppManifest() if ($manifest.application) { @{ "publisher" = "Microsoft"; "name" = "Application"; "appId" = ''; "version" = $manifest.Application } } foreach ($dependency in $manifest.dependencies) { $appId = '' if ($dependency.psobject.Properties.name -eq 'appid') { $appId = $dependency.appid } elseif ($dependency.psobject.Properties.name -eq 'id') { $appId = $dependency.id } @{ "publisher" = $dependency.Publisher; "name" = $dependency.name; "appId" = $appId; "Version" = $dependency.Version } } } } catch [System.Reflection.ReflectionTypeLoadException] { if ($_.Exception.LoaderExceptions) { $_.Exception.LoaderExceptions | ForEach-Object { Write-Host "LoaderException: $($_.Message)" } } throw } finally { if ($package) { $package.Dispose() } if ($packageStream) { $packageStream.Dispose() } } } } -ArgumentList (Get-BcContainerPath -containerName $containerName -path $symbolsFile), $platformversion $addDependencies | ForEach-Object { $addDependency = $_ $found = $false $dependencies | ForEach-Object { if ((($_.appId) -and ($_.appId -eq $addDependency.appId)) -or ($_.Publisher -eq $addDependency.Publisher -and $_.Name -eq $addDependency.Name)) { $found = $true } } if (!$found) { Write-Host "Adding dependency to $($addDependency.Name) from $($addDependency.Publisher)" $dependencies += $addDependency } } } } } $depidx++ } $errorLogFilePath = "" if($generateErrorLog) { $errorLogFilePath = (Join-Path $containerOutputFolder $($appName -replace '.app$','.errorLog.json')) } $result = Invoke-ScriptInBcContainer -containerName $containerName -ScriptBlock { Param($appProjectFolder, $appSymbolsFolder, $appOutputFile, $EnableCodeCop, $EnableAppSourceCop, $EnablePerTenantExtensionCop, $EnableUICop, $CustomCodeCops, $rulesetFile, $enableExternalRulesets, $assemblyProbingPaths, $nowarn, $errorLogFilePath, $GenerateCrossReferences, $ReportSuppressedDiagnostics, $generateReportLayoutParam, $features, $preProcessorSymbols, $platformversion, $updateDependencies, $sourceRepositoryUrl, $sourceCommit, $buildBy, $buildUrl ) if ($updateDependencies) { $appJsonFile = Join-Path $appProjectFolder 'app.json' $appJsonObject = [System.IO.File]::ReadAllLines($appJsonFile) | ConvertFrom-Json $changes = $false Write-Host "Enumerating Existing Apps" $existingApps = Get-ChildItem -Path (Join-Path $appSymbolsFolder '*.app') | ForEach-Object { $appInfo = Get-NavAppInfo -Path $_.FullName Write-Host "- FileName=$($_.Name), Id=$($appInfo.AppId), Publisher=$($appInfo.Publisher), Name=$($appInfo.Name), Version=$($appInfo.Version)" $appInfo } Write-Host "Modifying Dependencies" if (([bool]($appJsonObject.PSobject.Properties.name -eq "dependencies")) -and $appJsonObject.dependencies) { $appJsonObject.dependencies = @($appJsonObject.dependencies | ForEach-Object { $dependency = $_ $dependencyAppId = "$(if ($dependency.PSObject.Properties.name -eq 'AppId') { $dependency.AppId } else { $dependency.Id })" Write-Host "Dependency: Id=$dependencyAppId, Publisher=$($dependency.Publisher), Name=$($dependency.Name), Version=$($dependency.Version)" $existingApps | Where-Object { "$($_.AppId)" -eq $dependencyAppId -and $_.Version -gt [System.Version]$dependency.Version } | ForEach-Object { $dependency.Version = "$($_.Version)" Write-Host "- Set dependency version to $($_.Version)" $changes = $true } $dependency }) } if (([bool]($appJsonObject.PSobject.Properties.name -eq "application")) -and $appJsonObject.application) { Write-Host "Application Dependency $($appJsonObject.application)" $existingApps | Where-Object { $_.Name -eq "Application" -and $_.Version -gt [System.Version]$appJsonObject.application } | ForEach-Object { $appJsonObject.Application = "$($_.Version)" Write-Host "- Set Application dependency to $($_.Version)" $changes = $true } } if (([bool]($appJsonObject.PSobject.Properties.name -eq "platform")) -and $appJsonObject.platform) { Write-Host "Platform Dependency $($appJsonObject.platform)" $existingApps | Where-Object { $_.Name -eq "System" -and $_.Version -gt [System.Version]$appJsonObject.platform } | ForEach-Object { $appJsonObject.platform = "$($_.Version)" Write-Host "- Set Platform dependency to $($_.Version)" $changes = $true } } if ($changes) { Write-Host "Updating app.json" $appJsonObject | ConvertTo-Json -depth 99 | Set-Content $appJsonFile -encoding UTF8 } } $binPath = 'C:\build\vsix\extension\bin' $alcPath = Join-Path $binPath 'win32' if (-not (Test-Path $alcPath)) { $alcPath = $binPath } if (Test-Path -Path $appOutputFile -PathType Leaf) { Remove-Item -Path $appOutputFile -Force } Write-Host "Compiling..." Set-Location -Path $alcPath $alcItem = Get-Item -Path (Join-Path $alcPath 'alc.exe') [System.Version]$alcVersion = $alcItem.VersionInfo.FileVersion $alcParameters = @("/project:""$($appProjectFolder.TrimEnd('/\'))""", "/packagecachepath:""$($appSymbolsFolder.TrimEnd('/\'))""", "/out:""$appOutputFile""") if ($GenerateReportLayoutParam) { $alcParameters += @($GenerateReportLayoutParam) } if ($EnableCodeCop) { $alcParameters += @("/analyzer:$(Join-Path $binPath 'Analyzers\Microsoft.Dynamics.Nav.CodeCop.dll')") } if ($EnableAppSourceCop) { $alcParameters += @("/analyzer:$(Join-Path $binPath 'Analyzers\Microsoft.Dynamics.Nav.AppSourceCop.dll')") } if ($EnablePerTenantExtensionCop) { $alcParameters += @("/analyzer:$(Join-Path $binPath 'Analyzers\Microsoft.Dynamics.Nav.PerTenantExtensionCop.dll')") } if ($EnableUICop) { $alcParameters += @("/analyzer:$(Join-Path $binPath 'Analyzers\Microsoft.Dynamics.Nav.UICop.dll')") } if ($CustomCodeCops.Count -gt 0) { $CustomCodeCops | ForEach-Object { $alcParameters += @("/analyzer:$_") } } if ($rulesetFile) { $alcParameters += @("/ruleset:$rulesetfile") } if ($enableExternalRulesets) { $alcParameters += @("/enableexternalrulesets") } if ($nowarn) { $alcParameters += @("/nowarn:$nowarn") } if ($errorLogFilePath) { $alcParameters += @("/errorLog:""$errorLogFilePath""") } if ($GenerateCrossReferences -and $platformversion.Major -ge 18) { $alcParameters += @("/generatecrossreferences") } if ($ReportSuppressedDiagnostics) { if ($alcVersion -ge [System.Version]"9.1.0.0") { $alcParameters += @("/reportsuppresseddiagnostics") } else { Write-Host -ForegroundColor Yellow "ReportSuppressedDiagnostics was specified, but the version of the AL Language Extension does not support this. Get-LatestAlLanguageExtensionUrl returns a location for the latest AL Language Extension" } } if ($alcVersion -ge [System.Version]"12.0.12.41479") { if ($sourceRepositoryUrl) { $alcParameters += @("/SourceRepositoryUrl:""$sourceRepositoryUrl""") } if ($sourceCommit) { $alcParameters += @("/SourceCommit:""$sourceCommit""") } if ($buildBy) { $alcParameters += @("/BuildBy:""$buildBy""") } if ($buildUrl) { $alcParameters += @("/BuildUrl:""$buildUrl""") } } if ($assemblyProbingPaths) { $alcParameters += @("/assemblyprobingpaths:$assemblyProbingPaths") } if ($features) { $alcParameters +=@("/features:$($features -join ',')") } $preprocessorSymbols | where-Object { $_ } | ForEach-Object { $alcParameters += @("/D:$_") } Write-Host ".\alc.exe $([string]::Join(' ', $alcParameters))" & .\alc.exe $alcParameters if ($lastexitcode -ne 0 -and $lastexitcode -ne -1073740791) { "App generation failed with exit code $lastexitcode" } } -ArgumentList $containerProjectFolder, $containerSymbolsFolder, (Join-Path $containerOutputFolder $appName), $EnableCodeCop, $EnableAppSourceCop, $EnablePerTenantExtensionCop, $EnableUICop, $CustomCodeCopFiles, $containerRulesetFile, $enableExternalRulesets, $assemblyProbingPaths, $nowarn, $errorLogFilePath, $GenerateCrossReferences, $ReportSuppressedDiagnostics, $GenerateReportLayoutParam, $features, $preProcessorSymbols, $platformversion, $updateDependencies, $sourceRepositoryUrl, $sourceCommit, $buildBy, $buildUrl if ($treatWarningsAsErrors) { $regexp = ($treatWarningsAsErrors | ForEach-Object { if ($_ -eq '*') { ".*" } else { $_ } }) -join '|' $result = $result | ForEach-Object { $_ -replace "^(.*)warning ($regexp):(.*)`$", '$1error $2:$3' } } $devOpsResult = "" if ($result) { $Parameters = @{ "FailOn" = $FailOn "AlcOutput" = $result "DoNotWriteToHost" = $true } if ($gitHubActions) { $Parameters += @{ "gitHubActions" = $true } if (-not $basePath) { $basePath = $ENV:GITHUB_WORKSPACE } } if ($basePath) { $Parameters += @{ "basePath" = (Get-BcContainerPath -containerName $containerName -path $basePath) } } $devOpsResult = Convert-ALCOutputToAzureDevOps @Parameters } if ($AzureDevOps -or $gitHubActions) { $devOpsResult | ForEach-Object { $outputTo.Invoke($_) } } else { $result | ForEach-Object { $outputTo.Invoke($_) } if ($devOpsResult -like "*task.complete result=Failed*") { throw "App generation failed" } } $result | Where-Object { $_ -like "App generation failed*" } | ForEach-Object { throw $_ } $timespend = [Math]::Round([DateTime]::Now.Subtract($startTime).Totalseconds) $appFile = Join-Path $appOutputFolder $appName if (Test-Path -Path $appFile) { Write-Host "$appFile successfully created in $timespend seconds" if ($CopyAppToSymbolsFolder) { Copy-Item -Path $appFile -Destination $appSymbolsFolder -ErrorAction SilentlyContinue if (Test-Path -Path (Join-Path -Path $appSymbolsFolder -ChildPath $appName)) { Write-Host "$($appName) copied to $($appSymbolsFolder)" Invoke-ScriptInBcContainer -containerName $containerName -ScriptBlock { Param($appSymbolsFolder, $appName) $appFile = Join-Path -Path $appSymbolsFolder -ChildPath $appName while (-not (Test-Path -Path $appFile)) { Start-Sleep -Seconds 1 } } -ArgumentList $containerSymbolsFolder,"$($appName)" } } } else { throw "App generation failed" } $appFile } catch { TrackException -telemetryScope $telemetryScope -errorRecord $_ throw } finally { TrackTrace -telemetryScope $telemetryScope } } Set-Alias -Name Compile-AppInNavContainer -Value Compile-AppInBcContainer Export-ModuleMember -Function Compile-AppInBcContainer -Alias Compile-AppInNavContainer # SIG # Begin signature block # MIIr3wYJKoZIhvcNAQcCoIIr0DCCK8wCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCcusDmZ1Umrdes # rBGbgyZyLS3LAwcSF6bZi4+oW3p2/KCCJPcwggVvMIIEV6ADAgECAhBI/JO0YFWU # jTanyYqJ1pQWMA0GCSqGSIb3DQEBDAUAMHsxCzAJBgNVBAYTAkdCMRswGQYDVQQI # DBJHcmVhdGVyIE1hbmNoZXN0ZXIxEDAOBgNVBAcMB1NhbGZvcmQxGjAYBgNVBAoM # EUNvbW9kbyBDQSBMaW1pdGVkMSEwHwYDVQQDDBhBQUEgQ2VydGlmaWNhdGUgU2Vy # dmljZXMwHhcNMjEwNTI1MDAwMDAwWhcNMjgxMjMxMjM1OTU5WjBWMQswCQYDVQQG # EwJHQjEYMBYGA1UEChMPU2VjdGlnbyBMaW1pdGVkMS0wKwYDVQQDEyRTZWN0aWdv # IFB1YmxpYyBDb2RlIFNpZ25pbmcgUm9vdCBSNDYwggIiMA0GCSqGSIb3DQEBAQUA # A4ICDwAwggIKAoICAQCN55QSIgQkdC7/FiMCkoq2rjaFrEfUI5ErPtx94jGgUW+s # hJHjUoq14pbe0IdjJImK/+8Skzt9u7aKvb0Ffyeba2XTpQxpsbxJOZrxbW6q5KCD # J9qaDStQ6Utbs7hkNqR+Sj2pcaths3OzPAsM79szV+W+NDfjlxtd/R8SPYIDdub7 # P2bSlDFp+m2zNKzBenjcklDyZMeqLQSrw2rq4C+np9xu1+j/2iGrQL+57g2extme # me/G3h+pDHazJyCh1rr9gOcB0u/rgimVcI3/uxXP/tEPNqIuTzKQdEZrRzUTdwUz # T2MuuC3hv2WnBGsY2HH6zAjybYmZELGt2z4s5KoYsMYHAXVn3m3pY2MeNn9pib6q # RT5uWl+PoVvLnTCGMOgDs0DGDQ84zWeoU4j6uDBl+m/H5x2xg3RpPqzEaDux5mcz # mrYI4IAFSEDu9oJkRqj1c7AGlfJsZZ+/VVscnFcax3hGfHCqlBuCF6yH6bbJDoEc # QNYWFyn8XJwYK+pF9e+91WdPKF4F7pBMeufG9ND8+s0+MkYTIDaKBOq3qgdGnA2T # OglmmVhcKaO5DKYwODzQRjY1fJy67sPV+Qp2+n4FG0DKkjXp1XrRtX8ArqmQqsV/ # AZwQsRb8zG4Y3G9i/qZQp7h7uJ0VP/4gDHXIIloTlRmQAOka1cKG8eOO7F/05QID # AQABo4IBEjCCAQ4wHwYDVR0jBBgwFoAUoBEKIz6W8Qfs4q8p74Klf9AwpLQwHQYD # VR0OBBYEFDLrkpr/NZZILyhAQnAgNpFcF4XmMA4GA1UdDwEB/wQEAwIBhjAPBgNV # HRMBAf8EBTADAQH/MBMGA1UdJQQMMAoGCCsGAQUFBwMDMBsGA1UdIAQUMBIwBgYE # VR0gADAIBgZngQwBBAEwQwYDVR0fBDwwOjA4oDagNIYyaHR0cDovL2NybC5jb21v # ZG9jYS5jb20vQUFBQ2VydGlmaWNhdGVTZXJ2aWNlcy5jcmwwNAYIKwYBBQUHAQEE # KDAmMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5jb21vZG9jYS5jb20wDQYJKoZI # hvcNAQEMBQADggEBABK/oe+LdJqYRLhpRrWrJAoMpIpnuDqBv0WKfVIHqI0fTiGF # OaNrXi0ghr8QuK55O1PNtPvYRL4G2VxjZ9RAFodEhnIq1jIV9RKDwvnhXRFAZ/ZC # J3LFI+ICOBpMIOLbAffNRk8monxmwFE2tokCVMf8WPtsAO7+mKYulaEMUykfb9gZ # pk+e96wJ6l2CxouvgKe9gUhShDHaMuwV5KZMPWw5c9QLhTkg4IUaaOGnSDip0TYl # d8GNGRbFiExmfS9jzpjoad+sPKhdnckcW67Y8y90z7h+9teDnRGWYpquRRPaf9xH # +9/DUp/mBlXpnYzyOmJRvOwkDynUWICE5EV7WtgwggWNMIIEdaADAgECAhAOmxiO # +dAt5+/bUOIIQBhaMA0GCSqGSIb3DQEBDAUAMGUxCzAJBgNVBAYTAlVTMRUwEwYD # VQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xJDAi # BgNVBAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9vdCBDQTAeFw0yMjA4MDEwMDAw # MDBaFw0zMTExMDkyMzU5NTlaMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdp # Q2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERp # Z2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCC # AgoCggIBAL/mkHNo3rvkXUo8MCIwaTPswqclLskhPfKK2FnC4SmnPVirdprNrnsb # hA3EMB/zG6Q4FutWxpdtHauyefLKEdLkX9YFPFIPUh/GnhWlfr6fqVcWWVVyr2iT # cMKyunWZanMylNEQRBAu34LzB4TmdDttceItDBvuINXJIB1jKS3O7F5OyJP4IWGb # NOsFxl7sWxq868nPzaw0QF+xembud8hIqGZXV59UWI4MK7dPpzDZVu7Ke13jrclP # XuU15zHL2pNe3I6PgNq2kZhAkHnDeMe2scS1ahg4AxCN2NQ3pC4FfYj1gj4QkXCr # VYJBMtfbBHMqbpEBfCFM1LyuGwN1XXhm2ToxRJozQL8I11pJpMLmqaBn3aQnvKFP # ObURWBf3JFxGj2T3wWmIdph2PVldQnaHiZdpekjw4KISG2aadMreSx7nDmOu5tTv # kpI6nj3cAORFJYm2mkQZK37AlLTSYW3rM9nF30sEAMx9HJXDj/chsrIRt7t/8tWM # cCxBYKqxYxhElRp2Yn72gLD76GSmM9GJB+G9t+ZDpBi4pncB4Q+UDCEdslQpJYls # 5Q5SUUd0viastkF13nqsX40/ybzTQRESW+UQUOsxxcpyFiIJ33xMdT9j7CFfxCBR # a2+xq4aLT8LWRV+dIPyhHsXAj6KxfgommfXkaS+YHS312amyHeUbAgMBAAGjggE6 # MIIBNjAPBgNVHRMBAf8EBTADAQH/MB0GA1UdDgQWBBTs1+OC0nFdZEzfLmc/57qY # rhwPTzAfBgNVHSMEGDAWgBRF66Kv9JLLgjEtUYunpyGd823IDzAOBgNVHQ8BAf8E # BAMCAYYweQYIKwYBBQUHAQEEbTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5k # aWdpY2VydC5jb20wQwYIKwYBBQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0 # LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcnQwRQYDVR0fBD4wPDA6oDig # NoY0aHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9v # dENBLmNybDARBgNVHSAECjAIMAYGBFUdIAAwDQYJKoZIhvcNAQEMBQADggEBAHCg # v0NcVec4X6CjdBs9thbX979XB72arKGHLOyFXqkauyL4hxppVCLtpIh3bb0aFPQT # SnovLbc47/T/gLn4offyct4kvFIDyE7QKt76LVbP+fT3rDB6mouyXtTP0UNEm0Mh # 65ZyoUi0mcudT6cGAxN3J0TU53/oWajwvy8LpunyNDzs9wPHh6jSTEAZNUZqaVSw # uKFWjuyk1T3osdz9HNj0d1pcVIxv76FQPfx2CWiEn2/K2yCNNWAcAgPLILCsWKAO # QGPFmCLBsln1VWvPJ6tsds5vIy30fnFqI2si/xK4VC0nftg62fC2h5b9W9FcrBjD # TZ9ztwGpn1eqXijiuZQwggYaMIIEAqADAgECAhBiHW0MUgGeO5B5FSCJIRwKMA0G # CSqGSIb3DQEBDAUAMFYxCzAJBgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExp # bWl0ZWQxLTArBgNVBAMTJFNlY3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBSb290 # IFI0NjAeFw0yMTAzMjIwMDAwMDBaFw0zNjAzMjEyMzU5NTlaMFQxCzAJBgNVBAYT # AkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0ZWQxKzApBgNVBAMTIlNlY3RpZ28g # UHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYwggGiMA0GCSqGSIb3DQEBAQUAA4IB # jwAwggGKAoIBgQCbK51T+jU/jmAGQ2rAz/V/9shTUxjIztNsfvxYB5UXeWUzCxEe # AEZGbEN4QMgCsJLZUKhWThj/yPqy0iSZhXkZ6Pg2A2NVDgFigOMYzB2OKhdqfWGV # oYW3haT29PSTahYkwmMv0b/83nbeECbiMXhSOtbam+/36F09fy1tsB8je/RV0mIk # 8XL/tfCK6cPuYHE215wzrK0h1SWHTxPbPuYkRdkP05ZwmRmTnAO5/arnY83jeNzh # P06ShdnRqtZlV59+8yv+KIhE5ILMqgOZYAENHNX9SJDm+qxp4VqpB3MV/h53yl41 # aHU5pledi9lCBbH9JeIkNFICiVHNkRmq4TpxtwfvjsUedyz8rNyfQJy/aOs5b4s+ # ac7IH60B+Ja7TVM+EKv1WuTGwcLmoU3FpOFMbmPj8pz44MPZ1f9+YEQIQty/NQd/ # 2yGgW+ufflcZ/ZE9o1M7a5Jnqf2i2/uMSWymR8r2oQBMdlyh2n5HirY4jKnFH/9g # Rvd+QOfdRrJZb1sCAwEAAaOCAWQwggFgMB8GA1UdIwQYMBaAFDLrkpr/NZZILyhA # QnAgNpFcF4XmMB0GA1UdDgQWBBQPKssghyi47G9IritUpimqF6TNDDAOBgNVHQ8B # Af8EBAMCAYYwEgYDVR0TAQH/BAgwBgEB/wIBADATBgNVHSUEDDAKBggrBgEFBQcD # AzAbBgNVHSAEFDASMAYGBFUdIAAwCAYGZ4EMAQQBMEsGA1UdHwREMEIwQKA+oDyG # Omh0dHA6Ly9jcmwuc2VjdGlnby5jb20vU2VjdGlnb1B1YmxpY0NvZGVTaWduaW5n # Um9vdFI0Ni5jcmwwewYIKwYBBQUHAQEEbzBtMEYGCCsGAQUFBzAChjpodHRwOi8v # Y3J0LnNlY3RpZ28uY29tL1NlY3RpZ29QdWJsaWNDb2RlU2lnbmluZ1Jvb3RSNDYu # cDdjMCMGCCsGAQUFBzABhhdodHRwOi8vb2NzcC5zZWN0aWdvLmNvbTANBgkqhkiG # 9w0BAQwFAAOCAgEABv+C4XdjNm57oRUgmxP/BP6YdURhw1aVcdGRP4Wh60BAscjW # 4HL9hcpkOTz5jUug2oeunbYAowbFC2AKK+cMcXIBD0ZdOaWTsyNyBBsMLHqafvIh # rCymlaS98+QpoBCyKppP0OcxYEdU0hpsaqBBIZOtBajjcw5+w/KeFvPYfLF/ldYp # mlG+vd0xqlqd099iChnyIMvY5HexjO2AmtsbpVn0OhNcWbWDRF/3sBp6fWXhz7Dc # ML4iTAWS+MVXeNLj1lJziVKEoroGs9Mlizg0bUMbOalOhOfCipnx8CaLZeVme5yE # Lg09Jlo8BMe80jO37PU8ejfkP9/uPak7VLwELKxAMcJszkyeiaerlphwoKx1uHRz # NyE6bxuSKcutisqmKL5OTunAvtONEoteSiabkPVSZ2z76mKnzAfZxCl/3dq3dUNw # 4rg3sTCggkHSRqTqlLMS7gjrhTqBmzu1L90Y1KWN/Y5JKdGvspbOrTfOXyXvmPL6 # E52z1NZJ6ctuMFBQZH3pwWvqURR8AgQdULUvrxjUYbHHj95Ejza63zdrEcxWLDX6 # xWls/GDnVNueKjWUH3fTv1Y8Wdho698YADR7TNx8X8z2Bev6SivBBOHY+uqiirZt # g0y9ShQoPzmCcn63Syatatvx157YK9hlcPmVoa1oDE5/L9Uo2bC5a4CH2RwwggZZ # MIIEwaADAgECAhANIM3qwHRbWKHw+Zq6JhzlMA0GCSqGSIb3DQEBDAUAMFQxCzAJ # BgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0ZWQxKzApBgNVBAMTIlNl # Y3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYwHhcNMjExMDIyMDAwMDAw # WhcNMjQxMDIxMjM1OTU5WjBdMQswCQYDVQQGEwJESzEUMBIGA1UECAwLSG92ZWRz # dGFkZW4xGzAZBgNVBAoMEkZyZWRkeSBLcmlzdGlhbnNlbjEbMBkGA1UEAwwSRnJl # ZGR5IEtyaXN0aWFuc2VuMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA # gYC5tlg+VRktRRkahxxaV8+DAd6vHoDpcO6w7yT24lnSoMuA6nR7kgy90Y/sHIwK # E9Wwt/px/GAY8eBePWjJrFpG8fBtJbXadRTVd/470Hs/q9t+kh6A/0ELj7wYsKSN # OyuFPoy4rtClOv9ZmrRpoDVnh8Epwg2DpklX2BNzykzBQxIbkpp+xVo2mhPNWDIe # sntc4/BnSebLGw1Vkxmu2acKkIjYrne/7lsuyL9ue0vk8TGk9JBPNPbGKJvHu9sz # P9oGoH36fU1sEZ+AacXrp+onsyPf/hkkpAMHAhzQHl+5Ikvcus/cDm06twm7Vywm # Zcas2rFAV5MyE6WMEaYAolwAHiPz9WAs2GDhFtZZg1tzbRjJIIgPpR+doTIcpcDB # cHnNdSdgWKrTkr2f339oT5bnJfo7oVzc/2HGWvb8Fom6LQAqSC11vWmznHYsCm72 # g+foTKqW8lLDfLF0+aFvToLosrtW9l6Z+l+RQ8MtJ9EHOm2Ny8cFLzZCDZYw32By # dwcLV5rKdy4Ica9on5xZvyMOLiFwuL4v2V4pjEgKJaGSS/IVSMEGjrM9DHT6YS4/ # oq9q20rQUmMZZQmGmEyyKQ8t11si8VHtScN5m0Li8peoWfCU9mRFxSESwTWow8d4 # 62+o9/SzmDxCACdFwzvfKx4JqDMm55cL+beunIvc0NsCAwEAAaOCAZwwggGYMB8G # A1UdIwQYMBaAFA8qyyCHKLjsb0iuK1SmKaoXpM0MMB0GA1UdDgQWBBTZD6uy9ZWI # IqQh3srYu1FlUhdM0TAOBgNVHQ8BAf8EBAMCB4AwDAYDVR0TAQH/BAIwADATBgNV # HSUEDDAKBggrBgEFBQcDAzARBglghkgBhvhCAQEEBAMCBBAwSgYDVR0gBEMwQTA1 # BgwrBgEEAbIxAQIBAwIwJTAjBggrBgEFBQcCARYXaHR0cHM6Ly9zZWN0aWdvLmNv # bS9DUFMwCAYGZ4EMAQQBMEkGA1UdHwRCMEAwPqA8oDqGOGh0dHA6Ly9jcmwuc2Vj # dGlnby5jb20vU2VjdGlnb1B1YmxpY0NvZGVTaWduaW5nQ0FSMzYuY3JsMHkGCCsG # AQUFBwEBBG0wazBEBggrBgEFBQcwAoY4aHR0cDovL2NydC5zZWN0aWdvLmNvbS9T # ZWN0aWdvUHVibGljQ29kZVNpZ25pbmdDQVIzNi5jcnQwIwYIKwYBBQUHMAGGF2h0 # dHA6Ly9vY3NwLnNlY3RpZ28uY29tMA0GCSqGSIb3DQEBDAUAA4IBgQASEbZACurQ # eQN8WDTR+YyNpoQ29YAbbdBRhhzHkT/1ao7LE0QIOgGR4GwKRzufCAwu8pCBiMOU # TDHTezkh0rQrG6khxBX2nSTBL5i4LwKMR08HgZBsbECciABy15yexYWoB/D0H8Wu # Ge63PhGWueR4IFPbIz+jEVxfW0Nyyr7bXTecpKd1iprm+TOmzc2E6ab95dkcXdJV # x6Zys++QrrOfQ+a57qEXkS/wnjjbN9hukL0zg+g8L4DHLKTodzfiQOampvV8Qzbn # B7Y8YjNcxR9s/nptnlQH3jorNFhktiBXvD62jc8pAIg6wyH6NxSMjtTsn7QhkIp2 # kuswIQwD8hN/fZ/m6gkXZhRJWFr2WRZOz+edZ62Jf25C/NYWscwfBwn2hzRZf1Hg # yxkXAl88dvvUA3kw1T6uo8aAB9IcL6Owiy7q4T+RLRF7oqx0vcw0193Yhq/gPOaU # FlqzExP6TQ5TR9XWVPQk+a1B1ATKMLi1JShO6KWTmNkFkgkgpkW69BEwggauMIIE # lqADAgECAhAHNje3JFR82Ees/ShmKl5bMA0GCSqGSIb3DQEBCwUAMGIxCzAJBgNV # BAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdp # Y2VydC5jb20xITAfBgNVBAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDAeFw0y # MjAzMjMwMDAwMDBaFw0zNzAzMjIyMzU5NTlaMGMxCzAJBgNVBAYTAlVTMRcwFQYD # VQQKEw5EaWdpQ2VydCwgSW5jLjE7MDkGA1UEAxMyRGlnaUNlcnQgVHJ1c3RlZCBH # NCBSU0E0MDk2IFNIQTI1NiBUaW1lU3RhbXBpbmcgQ0EwggIiMA0GCSqGSIb3DQEB # AQUAA4ICDwAwggIKAoICAQDGhjUGSbPBPXJJUVXHJQPE8pE3qZdRodbSg9GeTKJt # oLDMg/la9hGhRBVCX6SI82j6ffOciQt/nR+eDzMfUBMLJnOWbfhXqAJ9/UO0hNoR # 8XOxs+4rgISKIhjf69o9xBd/qxkrPkLcZ47qUT3w1lbU5ygt69OxtXXnHwZljZQp # 09nsad/ZkIdGAHvbREGJ3HxqV3rwN3mfXazL6IRktFLydkf3YYMZ3V+0VAshaG43 # IbtArF+y3kp9zvU5EmfvDqVjbOSmxR3NNg1c1eYbqMFkdECnwHLFuk4fsbVYTXn+ # 149zk6wsOeKlSNbwsDETqVcplicu9Yemj052FVUmcJgmf6AaRyBD40NjgHt1bicl # kJg6OBGz9vae5jtb7IHeIhTZgirHkr+g3uM+onP65x9abJTyUpURK1h0QCirc0PO # 30qhHGs4xSnzyqqWc0Jon7ZGs506o9UD4L/wojzKQtwYSH8UNM/STKvvmz3+Drhk # Kvp1KCRB7UK/BZxmSVJQ9FHzNklNiyDSLFc1eSuo80VgvCONWPfcYd6T/jnA+bIw # pUzX6ZhKWD7TA4j+s4/TXkt2ElGTyYwMO1uKIqjBJgj5FBASA31fI7tk42PgpuE+ # 9sJ0sj8eCXbsq11GdeJgo1gJASgADoRU7s7pXcheMBK9Rp6103a50g5rmQzSM7TN # sQIDAQABo4IBXTCCAVkwEgYDVR0TAQH/BAgwBgEB/wIBADAdBgNVHQ4EFgQUuhbZ # bU2FL3MpdpovdYxqII+eyG8wHwYDVR0jBBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4c # D08wDgYDVR0PAQH/BAQDAgGGMBMGA1UdJQQMMAoGCCsGAQUFBwMIMHcGCCsGAQUF # BwEBBGswaTAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEEG # CCsGAQUFBzAChjVodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRU # cnVzdGVkUm9vdEc0LmNydDBDBgNVHR8EPDA6MDigNqA0hjJodHRwOi8vY3JsMy5k # aWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNybDAgBgNVHSAEGTAX # MAgGBmeBDAEEAjALBglghkgBhv1sBwEwDQYJKoZIhvcNAQELBQADggIBAH1ZjsCT # tm+YqUQiAX5m1tghQuGwGC4QTRPPMFPOvxj7x1Bd4ksp+3CKDaopafxpwc8dB+k+ # YMjYC+VcW9dth/qEICU0MWfNthKWb8RQTGIdDAiCqBa9qVbPFXONASIlzpVpP0d3 # +3J0FNf/q0+KLHqrhc1DX+1gtqpPkWaeLJ7giqzl/Yy8ZCaHbJK9nXzQcAp876i8 # dU+6WvepELJd6f8oVInw1YpxdmXazPByoyP6wCeCRK6ZJxurJB4mwbfeKuv2nrF5 # mYGjVoarCkXJ38SNoOeY+/umnXKvxMfBwWpx2cYTgAnEtp/Nh4cku0+jSbl3ZpHx # cpzpSwJSpzd+k1OsOx0ISQ+UzTl63f8lY5knLD0/a6fxZsNBzU+2QJshIUDQtxMk # zdwdeDrknq3lNHGS1yZr5Dhzq6YBT70/O3itTK37xJV77QpfMzmHQXh6OOmc4d0j # /R0o08f56PGYX/sr2H7yRp11LB4nLCbbbxV7HhmLNriT1ObyF5lZynDwN7+YAN8g # Fk8n+2BnFqFmut1VwDophrCYoCvtlUG3OtUVmDG0YgkPCr2B2RP+v6TR81fZvAT6 # gt4y3wSJ8ADNXcL50CN/AAvkdgIm2fBldkKmKYcJRyvmfxqkhQ/8mJb2VVQrH4D6 # wPIOK+XW+6kvRBVK5xMOHds3OBqhK/bt1nz8MIIGwjCCBKqgAwIBAgIQBUSv85Sd # CDmmv9s/X+VhFjANBgkqhkiG9w0BAQsFADBjMQswCQYDVQQGEwJVUzEXMBUGA1UE # ChMORGlnaUNlcnQsIEluYy4xOzA5BgNVBAMTMkRpZ2lDZXJ0IFRydXN0ZWQgRzQg # UlNBNDA5NiBTSEEyNTYgVGltZVN0YW1waW5nIENBMB4XDTIzMDcxNDAwMDAwMFoX # DTM0MTAxMzIzNTk1OVowSDELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0 # LCBJbmMuMSAwHgYDVQQDExdEaWdpQ2VydCBUaW1lc3RhbXAgMjAyMzCCAiIwDQYJ # KoZIhvcNAQEBBQADggIPADCCAgoCggIBAKNTRYcdg45brD5UsyPgz5/X5dLnXaEO # CdwvSKOXejsqnGfcYhVYwamTEafNqrJq3RApih5iY2nTWJw1cb86l+uUUI8cIOrH # mjsvlmbjaedp/lvD1isgHMGXlLSlUIHyz8sHpjBoyoNC2vx/CSSUpIIa2mq62DvK # Xd4ZGIX7ReoNYWyd/nFexAaaPPDFLnkPG2ZS48jWPl/aQ9OE9dDH9kgtXkV1lnX+ # 3RChG4PBuOZSlbVH13gpOWvgeFmX40QrStWVzu8IF+qCZE3/I+PKhu60pCFkcOvV # 5aDaY7Mu6QXuqvYk9R28mxyyt1/f8O52fTGZZUdVnUokL6wrl76f5P17cz4y7lI0 # +9S769SgLDSb495uZBkHNwGRDxy1Uc2qTGaDiGhiu7xBG3gZbeTZD+BYQfvYsSzh # Ua+0rRUGFOpiCBPTaR58ZE2dD9/O0V6MqqtQFcmzyrzXxDtoRKOlO0L9c33u3Qr/ # eTQQfqZcClhMAD6FaXXHg2TWdc2PEnZWpST618RrIbroHzSYLzrqawGw9/sqhux7 # UjipmAmhcbJsca8+uG+W1eEQE/5hRwqM/vC2x9XH3mwk8L9CgsqgcT2ckpMEtGlw # Jw1Pt7U20clfCKRwo+wK8REuZODLIivK8SgTIUlRfgZm0zu++uuRONhRB8qUt+JQ # ofM604qDy0B7AgMBAAGjggGLMIIBhzAOBgNVHQ8BAf8EBAMCB4AwDAYDVR0TAQH/ # BAIwADAWBgNVHSUBAf8EDDAKBggrBgEFBQcDCDAgBgNVHSAEGTAXMAgGBmeBDAEE # AjALBglghkgBhv1sBwEwHwYDVR0jBBgwFoAUuhbZbU2FL3MpdpovdYxqII+eyG8w # HQYDVR0OBBYEFKW27xPn783QZKHVVqllMaPe1eNJMFoGA1UdHwRTMFEwT6BNoEuG # SWh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRHNFJTQTQw # OTZTSEEyNTZUaW1lU3RhbXBpbmdDQS5jcmwwgZAGCCsGAQUFBwEBBIGDMIGAMCQG # CCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wWAYIKwYBBQUHMAKG # TGh0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRHNFJT # QTQwOTZTSEEyNTZUaW1lU3RhbXBpbmdDQS5jcnQwDQYJKoZIhvcNAQELBQADggIB # AIEa1t6gqbWYF7xwjU+KPGic2CX/yyzkzepdIpLsjCICqbjPgKjZ5+PF7SaCinEv # GN1Ott5s1+FgnCvt7T1IjrhrunxdvcJhN2hJd6PrkKoS1yeF844ektrCQDifXcig # LiV4JZ0qBXqEKZi2V3mP2yZWK7Dzp703DNiYdk9WuVLCtp04qYHnbUFcjGnRuSvE # xnvPnPp44pMadqJpddNQ5EQSviANnqlE0PjlSXcIWiHFtM+YlRpUurm8wWkZus8W # 8oM3NG6wQSbd3lqXTzON1I13fXVFoaVYJmoDRd7ZULVQjK9WvUzF4UbFKNOt50MA # cN7MmJ4ZiQPq1JE3701S88lgIcRWR+3aEUuMMsOI5ljitts++V+wQtaP4xeR0arA # VeOGv6wnLEHQmjNKqDbUuXKWfpd5OEhfysLcPTLfddY2Z1qJ+Panx+VPNTwAvb6c # Kmx5AdzaROY63jg7B145WPR8czFVoIARyxQMfq68/qTreWWqaNYiyjvrmoI1VygW # y2nyMpqy0tg6uLFGhmu6F/3Ed2wVbK6rr3M66ElGt9V/zLY4wNjsHPW2obhDLN9O # TH0eaHDAdwrUAuBcYLso/zjlUlrWrBciI0707NMX+1Br/wd3H3GXREHJuEbTbDJ8 # WC9nR2XlG3O2mflrLAZG70Ee8PBf4NvZrZCARK+AEEGKMYIGPjCCBjoCAQEwaDBU # MQswCQYDVQQGEwJHQjEYMBYGA1UEChMPU2VjdGlnbyBMaW1pdGVkMSswKQYDVQQD # EyJTZWN0aWdvIFB1YmxpYyBDb2RlIFNpZ25pbmcgQ0EgUjM2AhANIM3qwHRbWKHw # +Zq6JhzlMA0GCWCGSAFlAwQCAQUAoIGEMBgGCisGAQQBgjcCAQwxCjAIoAKAAKEC # gAAwGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEOMAwG # CisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIFfF6xaCBDGayTZMsjTBVJxuBQih # m6U2ZKABIScKnCCCMA0GCSqGSIb3DQEBAQUABIICAENk5L/Btwg8HxsVCPSoFW6f # zaZlHTM7oC5WtrmSIFtf9dHIfnEZpbJQDzdxDw6rVWsyKEQkyzDCAE+1D0RgrvUu # vUkLVlqB488wrx4r7FgjRnUwAXi4GLneE+gBhiydVjkNC0BQKuAZatQV8jxM40af # 5O9HALpIRKTxgJwrj3XnnxMRFKv7Ccg21VfIP++e+cMRtcGbwNmqYvvm5ZXDPxcy # iGiygcBpCKieXufsKexPztmE5EgQ1PVmWHrBtMlBOWtL6Cr8lLVlc2P9TON7REvR # oNSSvYmz7a8R6LkHeyFlU/lZAyspu/agVTZshnE9oxlyv4687RTuWMBM7hdmRGY5 # y0kxwjg6iIuvJFMleCLdufG0dNR6CxFZZ9EXq/8+NHyNmYELllc23ZvwI1bfBgKc # baHgXL/Fr6WKlzZgTa1SA75lv0/fE7dgh0rJTxUoBwcJFVWUkT6BET+O3lNWPc0Q # 5vb7oFmBAZF/j3dZ5HLvGhsKouOZkCsVDe47wspRuKXtxLrrd54/cKdeOYJeyplA # Qj6YUdSAB0F6sQnzPY1EAyCsRRogNaDiGAsgK2BICntdnjSelEW+MbUuclOMepnx # MPRCWCOC1CZKMDWmaUpxhoelgw8B1VrB83anmeaJqv32IlZ9GP5k7b5WMZTpAZdb # x+1DRmncBjnedjIwio8joYIDIDCCAxwGCSqGSIb3DQEJBjGCAw0wggMJAgEBMHcw # YzELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMTswOQYDVQQD # EzJEaWdpQ2VydCBUcnVzdGVkIEc0IFJTQTQwOTYgU0hBMjU2IFRpbWVTdGFtcGlu # ZyBDQQIQBUSv85SdCDmmv9s/X+VhFjANBglghkgBZQMEAgEFAKBpMBgGCSqGSIb3 # DQEJAzELBgkqhkiG9w0BBwEwHAYJKoZIhvcNAQkFMQ8XDTI0MDMxMzEyNTMzMFow # LwYJKoZIhvcNAQkEMSIEINUuHn6M/r99JnBMFwE0Uu3DwzjishA3TVTL5+RZzbz6 # MA0GCSqGSIb3DQEBAQUABIICAEvhiOkiNOxHOMn9c6CRcrANPYAd4u91gejFMaLb # l9aAEEZoi5TbJoZuFrCa+3plac3If0OdDDdfXTb7SpimBTmSWvp5qOxh8IBVu5On # WBHft08m2fTOW7O/3fOBpqdtae24M+PgO6IoYHC7ng5JhxPtIINtNsjG5DflwLmE # qLV6UiYo61vFvKDttERyp8VkG0DTpF1xI6lY8huvRbJiJwCVpigTgfikuYhNqr8Y # kiovbQOzP67KzlufUOT3CRik3JKRWaJevXtm67iuWv7PtbehANZHSeSUL6qHLr3/ # zlMyG7U9miNwyBoZPAgg8JdJtke3ugxyrKsXZFw8H42406016++JMgvEi53IqPbm # vS55RHQOliiBJSA7ez3t0cxdzhatmdK/l0VWqFYVfI78DotqQwZbcAfNBGWDAQS2 # McgM440m7NpgmkOaNwwQJEYb73SQzQEMMcdBMOoe9f4qotS7IxVJnqHq54PTzeEh # yaBWFtl93ccXvX27s7fMj03o3itl6BYVv0JopmesJ8g8/cpLvk29aZo2aS7GUvkf # ExnWDBs3R5uhKnhvTfsTIgl1EL0u5zA0AD7SPqp0MX8eMzBfIV8efSnsWfW5JD+3 # 2vusCvEc0RrL2sjPsS0soVmU9cpWrFNxKxmh5RrniBXWG7193zL3FpK7CVm8qOlb # i2rT # SIG # End signature block |