Public/New-APFConfigDeployment.ps1
function New-APFConfigDeployment { param( [CmdletBinding(SupportsShouldProcess, HelpUri = 'https://pwsh.dev.tatux.co.uk/tatux.intune.packaging/docs/New-APFConfigDeployment.html')] [OutputType([string])] [Parameter(ParameterSetName = "config")] [Alias("ApplicationName", "AppName")] [Parameter(HelpMessage = "The name of the application. `nThis is written in to the exported configuration files. `nIf you do not provide a name, the script will attempt to generate it.")] [string]$Name, [Parameter(ParameterSetName = "config")] [Alias("ApplicationVersion", "AppVersion")] [Parameter(HelpMessage = "The version of the application. `nThis is written in to the exported configuration files. `nThis should be in the format of x.x.x.x. `nIf you do not provide a version, the script will attempt to extract it from the installer file.")] [version]$Version, [Parameter(ParameterSetName = "config")] [Parameter(HelpMessage = "The target for the deployment, User context or System context. Default is 'system'.")] [ValidateSet("system", "user")] [string]$Target = "system", [Parameter(ParameterSetName = "config")] [Parameter(ParameterSetName = "registry")] [Parameter(ParameterSetName = "files")] [Parameter(ParameterSetName = "script-os")] [Parameter(ParameterSetName = "script-app")] [Parameter(ParameterSetName = "script-user")] [Parameter(ParameterSetName = "custom")] [Parameter(HelpMessage = "The type of configuration this command is for.")] [ValidateSet("Registry", "Files", "Script-OS", "Script-App", "Script-User", "Custom")] [string]$ConfigurationType, [Parameter(ParameterSetName = "registry")] [Parameter(HelpMessage = "The registry key details to create or modify. \ `nThis should be in the format of fullPath: HKEY_LOCAL_MACHINE\Software\, KeyName: MyKey, KeyType: DWORD, KeyData: Setting1, State: ADD|REMOVE|MODIFY\ `nSo you would enter `"HKEY_LOCAL_MACHINE\Software\,MySoftware,MyName,DWORD,Setting1,ADD`" as the value.\ `nThere will also be a .csv file created where you can add as many registy items as you like.\ `nBare in mind that any failure of any registry item will cause the whole configuration to fail.")] [string]$RegistryValue, [Parameter(ParameterSetName = "config")] [Parameter(HelpMessage = "Destination path for the files on the target system.")] [string]$DestinationPath = $($PWD.Path), [Parameter(Mandatory = $false)] [Alias("IncludedFile", "SourceFile")] [ValidateScript({ foreach ($i in $_) { if (-not (Test-Path $_)) { throw "Path $($i) is invalid, double check and try again." } } return $true })] [string[]]$Path, [Parameter(HelpMessage = "The folder where the output will be created. Default is the current directory.")] [string]$DestinationFolder = $($PWD.Path), [Parameter(HelpMessage = "Create a Intune package for the application. Default is false.")] [switch]$CreateIntuneWinPackage ) begin { # Generate execution ID $ExecutionID = [System.Guid]::NewGuid().ToString() try { $CurrentConfig = Get-ModuleConfig -CommandPath $PSCommandPath $TelmetryArgs = @{ ModuleName = $CurrentConfig.ModuleName ModulePath = $CurrentConfig.ModulePath ModuleVersion = $MyInvocation.MyCommand.Module.Version ExecutionID = $ExecutionID CommandName = $MyInvocation.MyCommand.Name URI = 'https://telemetry.tatux.in/api/telemetry' } if ($CurrentConfig.BasicTelemetry -eq 'True') { $TelmetryArgs.Add('Minimal', $true) } Invoke-TelemetryCollection @TelmetryArgs -Stage start -ClearTimer } catch { Write-Verbose "Failed to load telemetry" } # Invoke-TelemetryCollection @TelmetryArgs -Stage End -ClearTimer -Failed $true -Exception $_ # Invoke-TelemetryCollection @TelmetryArgs -Stage End -ClearTimer } Process { # Create Switch on ConfigurationType switch ($ConfigurationType) { "Registry" { # Create Registry CSV file and add any commandline provided registry keys $DestinationFolder = Join-Path -Path $DestinationFolder -ChildPath $Name if (-not (Test-Path $DestinationFolder)) { New-Item -Path $DestinationFolder -ItemType Directory | Out-Null } else { if ($PSCmdlet.ShouldContinue("Overwrite existing folder for the deployment? Warning: This will recursively delete all files in the folder.", "Confirm Overwrite")) { Remove-Item -Path $DestinationFolder -Recurse -Force | Out-Null New-Item -Path $DestinationFolder -ItemType Directory | Out-Null } } $RegistryFile = Join-Path -Path $DestinationFolder -ChildPath $("$Name" + "_Registry.csv") Write-Verbose "Creating registry file at $RegistryFile" if (-not (Test-Path $RegistryFile)) { New-Item -Path $RegistryFile -ItemType File | Out-Null Copy-Item -Path "$PSScriptRoot\Templates\Registry\registry_entries.config.csv" -Destination $RegistryFile -Force } else { if ($PSCmdlet.ShouldContinue("Overwrite existing registry file? Warning: This will delete the existing file.", "Confirm Overwrite")) { Remove-Item -Path $RegistryFile -Force Copy-Item -Path "$PSScriptRoot\Templates\Registry\registry_entries.config.csv" -Destination $RegistryFile } } if ($RegistryValue) { # add registry keys to the csv file # Import current file, loop through and check if supplied key is already in the file try { $RegistryKeys = Import-Csv -Path $RegistryFile -ErrorAction Stop $RegistryKeys | ForEach-Object { $exists = $false if ($_ -eq $RegistryValue) { Write-Warning "The registry values supplied '$($RegistryValue)' already exists in the file." $exists = $true return } if ($exists -eq $false) { Export-Csv -Path $RegistryFile -InputObject $RegistryValue -Append -NoTypeInformation } } } catch { Write-Error "Failed to import the supplied entry to the registry file.`nError: $_" break } } else { Write-Verbose "No registry keys were supplied." } $IncludeFolder = Join-Path -Path $DestinationFolder -ChildPath $Name if (-not (Test-Path $IncludeFolder)) { New-Item -Path $IncludeFolder -ItemType Directory | Out-Null } else { if ($PSCmdlet.ShouldContinue("Overwrite existing subfolder for the deployment? Warning: This will recursively delete all files in the folder.", "Confirm Overwrite")) { Remove-Item -Path $IncludeFolder -Recurse -Force | Out-Null New-Item -Path $IncludeFolder -ItemType Directory | Out-Null } } if ($IncludedFiles) { foreach ($file in $IncludedFiles) { Copy-Item -Path $file -Destination $IncludeFolder } } # Copy the template files to the main folder try { Copy-Item -Path "$PSScriptRoot\Templates\Registry\*" -Destination $DestinationFolder -Recurse -Exclude *.md, *.config.csv } catch { Write-Error "Failed to copy the template files to the main folder.`nError: $_" break } # Update the template files with the deployment name and version $MainConfig = Get-Content -Path "$DestinationFolder\config.installer.json" | ConvertFrom-Json $MainConfig.name = $Name $MainConfig.version = $Version.ToString() $MainConfig.target = $Target $MainConfig.registryfile = "$($Name)_Registry.csv" $MainConfig.packagedby = $env:USERNAME $MainConfig | ConvertTo-Json -Depth 10 | Set-Content -Path "$DestinationFolder\$($Name)_config.installer.json" $DetectionScript = Get-Content -Path "$DestinationFolder\Intune-D-RegistryDetection.ps1" $DetectionScript = $DetectionScript -replace "##NAME_TEMPLATE", $Name $DetectionScript = $DetectionScript -replace "##VERSION_TEMPLATE", $Version.ToString() $DetectionScript = $DetectionScript -replace "##FILENAME_TEMPLATE", (Get-Item -Path $DestinationFolder).Name $DetectionScript | Set-Content -Path "$DestinationFolder\Intune-D-RegistryDetection.ps1" } "Files" { } "Script-OS" { } "Script-App" { # Create subfolder for the application $AppFolder = Join-Path -Path $DestinationFolder -ChildPath $Name if (-not (Test-Path $AppFolder)) { New-Item -Path $AppFolder -ItemType Directory | Out-Null } else { if ($PSCmdlet.ShouldContinue("Overwrite existing subfolder for the application? Warning: This will recursively delete all files in the folder.", "Confirm Overwrite")) { Remove-Item -Path $AppFolder -Recurse -Force | Out-Null New-Item -Path $AppFolder -ItemType Directory | Out-Null } } if ($Path) { # Copy the installer file and any additional files to the application folder Copy-Item -Path $Path -Destination $AppFolder } if ($IncludedFiles) { foreach ($file in $IncludedFiles) { Copy-Item -Path $file -Destination $AppFolder } } # Copy the template files to the application folder # Copy-Item -Path "$PSScriptRoot\Templates\Application\*" -Destination $AppFolder -Recurse # Update the template files with the application name and version $InstallerFileName = (Get-ChildItem -Path $Path).BaseName $MainConfig = Get-Content -Path "$AppFolder\config.installer.json" | ConvertFrom-Json $MainConfig.name = $Name $MainConfig.version = $Version.ToString() $MainConfig.filename = (Get-ChildItem -Path $Path).Name $MainConfig.target = $Target $MainConfig.installSwitches = $InstallSwitches $MainConfig.uninstallSwitches = $UninstallSwitches $MainConfig.uninstallPath = $UninstallPath $MainConfig | ConvertTo-Json -Depth 10 | Set-Content -Path "$AppFolder\config.installer.json" $DetectionScript = Get-Content -Path "$AppFolder\Intune-D-AppDetection.ps1" $DetectionScript = $DetectionScript -replace "##NAME_TEMPLATE", $Name $DetectionScript = $DetectionScript -replace "##VERSION_TEMPLATE", $Version.ToString() $DetectionScript = $DetectionScript -replace "##FILENAME_TEMPLATE", (Get-ChildItem -Path $Path).Name $DetectionScript | Set-Content -Path "$AppFolder\Intune-D-AppDetection.ps1" } "Script-User" { } "Custom" { } } # Create IntuneWin Package if ($CreateIntuneWinPackage) { try { # Get module directory path $ModulePath = Split-Path -Path $MyInvocation.MyCommand.Module.Path # Test if the IntuneWinAppUtil application exists in module directory if (-not (Test-Path "$ModulePath\IntuneWinAppUtil.exe")) { if ($PSCmdlet.ShouldContinue("The IntuneWinAppUtil.exe application was not found in the module directory. Would you like to download it now?", "Download Now?")) { Get-IntunePackagingTool -Path $ModulePath } else { Write-Warning "The IntuneWinAppUtil.exe application is required to create IntuneWin packages. Please download it manually." return } } # Create IntuneWin package $IntunewinFullPath = Join-Path -Path $DestinationFolder -ChildPath "$InstallerFileName.intunewin" $MainInstallerFilePath = Join-Path -Path $AppFolder -ChildPath (Get-Item -Path $Path).Name if (Test-Path $IntunewinFullPath) { if ($PSCmdlet.ShouldContinue("Overwrite existing IntuneWin package? Warning: This will delete the existing package.", "Confirm Overwrite")) { Remove-Item -Path $IntunewinFullPath -Force } else { Write-Warning "The IntuneWin package already exists. Please delete it manually or choose a different destination folder." return } } Start-Process -FilePath "$ModulePath\IntuneWinAppUtil.exe" -ArgumentList "-c `"$AppFolder`"", "-s `"$MainInstallerFilePath`"", "-o `"$DestinationFolder`"" -Wait -NoNewWindow -ErrorAction Stop if (-not (Test-Path $IntunewinFullPath)) { throw "$IntuneWinFullPath was not created" } else { Write-Output "The application '$Name' has been successfully packaged.`nThis can be found in the folder '$AppFolder'." Write-Output "The application '$Name' was also packaged to an intunewin file.`nThis can be found in the folder '$DestinationFolder'." } } catch { Write-Error "Failed to create IntuneWin package: $_" } } else { Write-Output "The application '$Name' has been successfully packaged.`nThis can be found in the folder '$DestinationFolder'." } Write-Output "When publishing the application to Intune, use`n'powershell.exe -ExecutionPolicy Bypass -File Intune-I-MainInstaller.ps1' for the install Command and`n'powershell.exe -ExecutionPolicy Bypass -File Intune-I-MainInstaller.ps1 -Uninstall' for the Uninstall Command." } } |