functions/invoke-d365sdpinstallude.ps1
<# .SYNOPSIS Install a Software Deployable Package (SDP) in a unified development environment .DESCRIPTION A cmdlet that wraps some of the cumbersome work into a streamlined process. It first checks if the package is a zip file and extracts it if necessary. Then it checks if the package contains the necessary files and modules. Finally, it extracts the module zip files into the metadata directory. .PARAMETER Path Path to the package that you want to install into the environment The cmdlet supports a path to a zip-file or directory with the unpacked contents. .PARAMETER MetaDataDir The path to the meta data directory for the environment .PARAMETER LogPath The path where the log file(s) will be saved .PARAMETER Force Instruct the cmdlet to overwrite the "extracted" folder if it exists Used when the input is a zip file, that will auto extract to a folder named like the zip file. .EXAMPLE PS C:\> Invoke-D365SDPInstallUDE -Path "c:\temp\package.zip" -MetaDataDir "c:\MyRepository\Metadata" This will install the modules contained in the c:\temp\package.zip file into the c:\MyRepository\Metadata directory. .NOTES Author: Florian Hopfner (@FH-Inway) #> function Invoke-D365SDPInstallUDE { param ( [Parameter(Mandatory = $True, Position = 1 )] [Alias('Hotfix')] [Alias('File')] [string] $Path, [Parameter(Mandatory = $true, Position = 2 )] [string] $MetaDataDir, [Alias('LogDir')] [string] $LogPath = $(Join-Path -Path $Script:DefaultTempPath -ChildPath "Logs\SdpInstall"), [switch] $Force ) if ((Get-Process -Name "devenv" -ErrorAction SilentlyContinue).Count -gt 0) { Write-PSFMessage -Level Host -Message "It seems that you have a <c='em'>Visual Studio</c> running. Please ensure <c='em'>exit</c> Visual Studio and run the cmdlet again." Stop-PSFFunction -Message "Stopping because of running Visual Studio." return } Invoke-TimeSignal -Start #Test if input is a zipFile that needs to be extracted first if ($Path.EndsWith(".zip")) { Unblock-File -Path $Path $extractedPath = $path.Remove($path.Length - 4) if (-not $Force) { if (-not (Test-PathExists -Path $extractedPath -Type Container -ShouldNotExist)) { Write-PSFMessage -Level Host -Message "The directory at the <c='em'>$extractedPath</c> location already exists. If you want to override it - set the <c='em'>Force</c> parameter to clear the folder and extract the content into it." Stop-PSFFunction -Message "Stopping because output path was already present." return } } Get-ChildItem -Path $extractedPath -ErrorAction SilentlyContinue | Remove-Item -Recurse -Force -Confirm:$false # To allow the file system to flush the files # Allows the human to see the folder being wiped Start-Sleep -Seconds 2 Expand-Archive -Path $Path -DestinationPath $extractedPath -Force $Path = $extractedPath } # Input is a relative path which needs to be converted to an absolute path. # see https://powershellmagazine.com/2013/01/16/pstip-check-if-the-path-is-relative-or-absolute/ if (-not ([System.IO.Path]::IsPathRooted($Path) -or (Split-Path -Path $Path -IsAbsolute))) { $currentPath = Get-Location # https://stackoverflow.com/a/13847304/2720554 $absolutePath = Join-Path -Path $currentPath -ChildPath $Path $absolutePath = [System.IO.Path]::GetFullPath($absolutePath) Write-PSFMessage -Level Verbose "Updating path to '$absolutePath' as relative paths are not supported" $Path = $absolutePath } Get-ChildItem -Path $Path -Recurse | Unblock-File $packageDetails = Get-D365SDPDetails -Path $Path $packagesFolder = "$Path\AOSService\Packages" $filesFolder = Get-ChildItem -Path $packagesFolder -Directory -Filter "files" if ($filesFolder.Count -eq 0) { Write-PSFMessage -Level Host -Message "No /AOSService/Packages/files folder found in the package. Please ensure that the package is extracted correctly." Stop-PSFFunction -Message "Stopping because of missing files folder." return } $zipFiles = Get-ChildItem -Path $filesFolder.FullName -File -Filter "*.zip" if ($zipFiles.Count -eq 0) { Write-PSFMessage -Level Host -Message "No module zip files found in the package. Please ensure that the package is extracted correctly." Stop-PSFFunction -Message "Stopping because of missing zip files." return } $numberOfInstalledModules = 0 $packageDetails.Modules | ForEach-Object { $moduleZip = $zipFiles | Where-Object Name -eq "dynamicsax-$($_.Name).$($_.Version).zip" if (-not $moduleZip) { Write-PSFMessage -Level Host -Message "No module zip file found for module $($_.Name). Please ensure that the package is extracted correctly." Stop-PSFFunction -Message "Stopping because of missing module zip file." return } # Delete existing module folder if it exists $moduleFolderPath = Join-Path -Path $MetaDataDir -ChildPath $($_.Name) if (Test-Path -Path $moduleFolderPath) { Remove-Item -Path $moduleFolderPath -Recurse -Force Write-PSFMessage -Level Verbose -Message "Deleted existing module folder $moduleFolderPath" } # Unzip to $MetaDataDir $moduleZipPath = Join-Path -Path $MetaDataDir -ChildPath $($_.Name) Expand-Archive -Path $moduleZip.FullName -DestinationPath $moduleZipPath Write-PSFMessage -Level Verbose -Message "Unzipped module $($_.Name) to $moduleZipPath" $numberOfInstalledModules++ } Write-PSFMessage -Level Host -Message "Installed $numberOfInstalledModules module(s) into $MetaDataDir" Invoke-TimeSignal -End } |