private/Get-ManifestDetails.ps1

<#
    .SYNOPSIS
        Retrieves details from an MSIX or APPV package manifest.
 
    .DESCRIPTION
        The Get-ManifestDetail function retrieves details from an MSIX or APPV package manifest.
 
    .PARAMETER Filename
        Specifies the path to the package.
 
    .NOTES
        Function : Get-ManifestDetail
        Author : John Billekens
        Copyright: Copyright (c) AppVentiX
        Version : 1.3
#>

function Get-ManifestDetail {
    [CmdletBinding()]
    Param (
        [ValidateScript({
                try {
                    if ((Test-Path -Path "$_") -and ("$_" -like '*.msix' -or "$_" -like '*.appv')) {
                        $true
                    } else {
                        Throw [System.Management.Automation.ItemNotFoundException] "The parameter must be a valid and accessable *.msix or *.appv file!"
                    }
                } catch {
                    Throw [System.Management.Automation.ItemNotFoundException] "The parameter must be a valid and accessable *.msix or *.appv file!"
                }

            })]
        [System.IO.FileInfo]$Filename
    )
    $packageType = ($Filename.Name.Split('.')[-1]).ToUpper()
    Write-Verbose "Retrieve the manifest from the package"
    [xml]$manifestXml = Get-AppXManifest -Filename $Filename
    Write-Verbose "Gather the package data"
    $appManifest = @{}
    $appManifest.Name = $manifestXml.Package.Identity.Name
    $appManifest.Version = $manifestXml.Package.Identity.Version
    $appManifest.Publisher = $manifestXml.Package.Identity.Publisher
    $appManifest.PublisherID = Get-PublisherId -Publisher $appManifest.Publisher
    switch ($packageType) {
        "MSIX" {
            Write-Verbose "Gather the MSIX package data"
            $appManifest.Architecture = $manifestXml.Package.Identity.ProcessorArchitecture
            $appManifest.PackageFullName = Get-PackageFullName @appManifest
        }
        "APPV" {
            Write-Verbose "Gather the APPV package data"
            $appManifest.PackageId = $manifestXml.Package.Identity.PackageId
            $appManifest.VersionId = $manifestXml.Package.Identity.VersionId
        }
        Default {
            Write-Warning "Unknown package type: $packageType"
        }
    }
    Write-Verbose "Gather the package data"
    $appManifest.Type = $packageType
    $appManifest.FamilyName = '{0}_{1}' -f $manifestXml.Package.Identity.Name, $appManifest.PublisherID
    $appManifest.DisplayName = $manifestXml.Package.Properties.DisplayName
    $appManifest.PublisherDisplayName = $manifestXml.Package.Properties.PublisherDisplayName
    $appManifest.LastUpdated = ([DateTime]$Filename.LastWriteTime).ToString("o").Substring(0, 27)
    Write-Verbose "Gather the application data"
    $appManifest.Applications = @()
    Write-Verbose "Get the package files, treat the package as zip file to extract file entries"
    $packageFiles = [IO.Compression.ZipFile]::OpenRead($Filename)
    $packageFilesEntries = $packageFiles.Entries
    Write-Verbose "Release the package as soon as possible to remove the lock on the file"
    $packageFiles.Dispose()
    $appvNumber = 0
    ForEach ($application in $manifestXml.Package.Applications.Application) {
        Write-Verbose "Gather info for application: $($application.Id)"
        $app = @{}
        switch ($packageType) {
            "MSIX" {
                $app.Id = $application.Id
                $app.AppUserModelID = '{0}!{1}' -f $appManifest.FamilyName, $application.Id
                $app.Executable = $application.Executable
                $app.Description = $application.VisualElements.Description
                $app.FriendlyName = $application.VisualElements.DisplayName
                $app.AppVentiXExecutable = '{0}#{1}' -f $application.Id, $application.Executable
                Write-Verbose "Select the icon image"
                #Select the icon image, this is a bit tricky because the icon image can be in different locations
                if (-Not [String]::IsNullOrEmpty($($application.VisualElements.Square44x44Logo))) {
                    $imageFilename = Split-Path -Path $application.VisualElements.Square44x44Logo -Leaf
                    $imageExtension = $imageFilename.Split(".")[-1]
                    $imageBase = $imageFilename.Replace(".$($imageExtension)", $null)
                    $imagePath = Split-Path -Path $application.VisualElements.Square44x44Logo -Parent
                    $iconImageNamePath = $packageFilesEntries | Where-Object { $_.Fullname -like "*$($imagePath)/$($imageBase)*.$($imageExtension)" } | Select-Object -ExpandProperty FullName
                    if ([String]::IsNullOrEmpty($iconImageNamePath) -or $iconImageNamePath.Count -gt 1) {
                        $iconImageNamePath = $packageFilesEntries | Where-Object { $_.Fullname -like "*$($imagePath)/$($imageBase)*scale-100*.$($imageExtension)" } | Select-Object -ExpandProperty FullName
                    }
                }
                if ([String]::IsNullOrEmpty($iconImageNamePath) -or $iconImageNamePath.Count -gt 1) {
                    $iconImageNamePath = $packageFilesEntries | Where-Object { $_.FullName -like "*Assets*$($application.Id)Square44x44*" } | Select-Object -ExpandProperty FullName
                }
                if ($iconImageNamePath.Count -gt 1) {
                    $iconImageNamePath = $iconImageNamePath | Where-Object { $_ -like "*scale-100*" }
                }
                if ([String]::IsNullOrEmpty($iconImageNamePath)) {
                    $iconImageNamePath = $packageFilesEntries | Where-Object { $_.FullName -like '*Square44x44Logo.*' -or $_.FullName -like '*Square44x44Logo.scale-100.*' } | Select-Object -ExpandProperty FullName | Select-Object -First 1 | Select-Object -ExpandProperty FullName
                    Write-Warning "Selected a random icon: $($iconImageNamePath)"
                }
                if ([String]::IsNullOrEmpty($iconImageNamePath)) {
                    Write-Warning "Cannot set an application image (path), none found"
                }
                Write-Verbose "Icon image: $iconImageNamePath"
                $app.Iconpath = $iconImageNamePath.Replace('/', '\')
                $app.IconWindowsPath = [io.path]::combine('C:\Program Files\WindowsApps', $appManifest.PackageFullName, $iconImageNamePath.Replace('/', '\'))
                $app.IconPackagePath = [io.path]::combine($appManifest.PackageFullName, $iconImageNamePath.Replace('/', '\'))
            }
            "APPV" {
                $app.Number = $appvNumber
                $app.Executable = $application.Target
                $app.Id = $application.Id
                $app.Description = $application.VisualElements.Description
                $app.FriendlyName = $application.VisualElements.Name
                $app.Version = $application.VisualElements.Version
                $app.Iconpath = $application.Target.Replace('[{', $null).Replace('}]', $null)
                $app.IconWindowsPath = [io.path]::combine('C:\ProgramData\App-V', $appManifest.PackageId, $appManifest.VersionId, 'Root\VFS', $app.Iconpath)
                $app.AppVentiXExecutable = $application.Target
                $appvNumber++
            }
            Default {
                Write-Warning "Unknown package type: $packageType"
            }
        }
        $appManifest.Applications += [PSCustomObject]$app.Clone()
    }
    Write-Verbose "Clear the XML data"
    $manifestXml.RemoveAll()
    Write-Verbose "Return all the gathered data"
    return [PSCustomObject]$appManifest
}

# SIG # Begin signature block
# MIITYgYJKoZIhvcNAQcCoIITUzCCE08CAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCAi/feORqvAVUtZ
# M2UW0XEE63I5UmqLo9Uu5wX8HJsi4aCCEHUwggTzMIID26ADAgECAhAsJ03zZBC0
# i/247uUvWN5TMA0GCSqGSIb3DQEBCwUAMHwxCzAJBgNVBAYTAkdCMRswGQYDVQQI
# ExJHcmVhdGVyIE1hbmNoZXN0ZXIxEDAOBgNVBAcTB1NhbGZvcmQxGDAWBgNVBAoT
# D1NlY3RpZ28gTGltaXRlZDEkMCIGA1UEAxMbU2VjdGlnbyBSU0EgQ29kZSBTaWdu
# aW5nIENBMB4XDTIxMDUwNTAwMDAwMFoXDTI0MDUwNDIzNTk1OVowWzELMAkGA1UE
# BhMCTkwxEjAQBgNVBAcMCVZlbGRob3ZlbjEbMBkGA1UECgwSSm9oYW5uZXMgQmls
# bGVrZW5zMRswGQYDVQQDDBJKb2hhbm5lcyBCaWxsZWtlbnMwggEiMA0GCSqGSIb3
# DQEBAQUAA4IBDwAwggEKAoIBAQCsfgRG81keOHalHfCUgxOa1Qy4VNOnGxB8SL8e
# rjP9SfcF13McP7F1HGka5Be495pTZ+duGbaQMNozwg/5Dg9IRJEeBabeSSJJCbZo
# SNpmUu7NNRRfidQxlPC81LxTVHxJ7In0MEfCVm7rWcri28MRCAuafqOfSE+hyb1Z
# /tKyCyQ5RUq3kjs/CF+VfMHsJn6ZT63YqewRkwHuc7UogTTZKjhPJ9prGLTer8UX
# UgvsGRbvhYZXIEuy+bmx/iJ1yRl1kX4nj6gUYzlhemOnlSDD66YOrkLDhXPMXLym
# AN7h0/W5Bo//R5itgvdGBkXkWCKRASnq/9PTcoxW6mwtgU8xAgMBAAGjggGQMIIB
# jDAfBgNVHSMEGDAWgBQO4TqoUzox1Yq+wbutZxoDha00DjAdBgNVHQ4EFgQUZWMy
# gC0i1u2NZ1msk2Mm5nJm5AswDgYDVR0PAQH/BAQDAgeAMAwGA1UdEwEB/wQCMAAw
# EwYDVR0lBAwwCgYIKwYBBQUHAwMwEQYJYIZIAYb4QgEBBAQDAgQQMEoGA1UdIARD
# MEEwNQYMKwYBBAGyMQECAQMCMCUwIwYIKwYBBQUHAgEWF2h0dHBzOi8vc2VjdGln
# by5jb20vQ1BTMAgGBmeBDAEEATBDBgNVHR8EPDA6MDigNqA0hjJodHRwOi8vY3Js
# LnNlY3RpZ28uY29tL1NlY3RpZ29SU0FDb2RlU2lnbmluZ0NBLmNybDBzBggrBgEF
# BQcBAQRnMGUwPgYIKwYBBQUHMAKGMmh0dHA6Ly9jcnQuc2VjdGlnby5jb20vU2Vj
# dGlnb1JTQUNvZGVTaWduaW5nQ0EuY3J0MCMGCCsGAQUFBzABhhdodHRwOi8vb2Nz
# cC5zZWN0aWdvLmNvbTANBgkqhkiG9w0BAQsFAAOCAQEARjv9ieRocb1DXRWm3XtY
# jjuSRjlvkoPd9wS6DNfsGlSU42BFd9LCKSyRREZVu8FDq7dN0PhD4bBTT+k6AgrY
# KG6f/8yUponOdxskv850SjN2S2FeVuR20pqActMrpd1+GCylG8mj8RGjdrLQ3QuX
# qYKS68WJ39WWYdVB/8Ftajir5p6sAfwHErLhbJS6WwmYjGI/9SekossvU8mZjZwo
# Gbu+fjZhPc4PhjbEh0ABSsPMfGjQQsg5zLFjg/P+cS6hgYI7qctToo0TexGe32DY
# fFWHrHuBErW2qXEJvzSqM5OtLRD06a4lH5ZkhojhMOX9S8xDs/ArDKgX1j1Xm4Tu
# DjCCBYEwggRpoAMCAQICEDlyRDr5IrdR19NsEN0xNZUwDQYJKoZIhvcNAQEMBQAw
# ezELMAkGA1UEBhMCR0IxGzAZBgNVBAgMEkdyZWF0ZXIgTWFuY2hlc3RlcjEQMA4G
# A1UEBwwHU2FsZm9yZDEaMBgGA1UECgwRQ29tb2RvIENBIExpbWl0ZWQxITAfBgNV
# BAMMGEFBQSBDZXJ0aWZpY2F0ZSBTZXJ2aWNlczAeFw0xOTAzMTIwMDAwMDBaFw0y
# ODEyMzEyMzU5NTlaMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMKTmV3IEplcnNl
# eTEUMBIGA1UEBxMLSmVyc2V5IENpdHkxHjAcBgNVBAoTFVRoZSBVU0VSVFJVU1Qg
# TmV0d29yazEuMCwGA1UEAxMlVVNFUlRydXN0IFJTQSBDZXJ0aWZpY2F0aW9uIEF1
# dGhvcml0eTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAIASZRc2DsPb
# CLPQrFcNdu3NJ9NMrVCDYeKqIE0JLWQJ3M6Jn8w9qez2z8Hc8dOx1ns3KBErR9o5
# xrw6GbRfpr19naNjQrZ28qk7K5H44m/Q7BYgkAk+4uh0yRi0kdRiZNt/owbxiBhq
# kCI8vP4T8IcUe/bkH47U5FHGEWdGCFHLhhRUP7wz/n5snP8WnRi9UY41pqdmyHJn
# 2yFmsdSbeAPAUDrozPDcvJ5M/q8FljUfV1q3/875PbcstvZU3cjnEjpNrkyKt1ya
# tLcgPcp/IjSufjtoZgFE5wFORlObM2D3lL5TN5BzQ/Myw1Pv26r+dE5px2uMYJPe
# xMcM3+EyrsyTO1F4lWeL7j1W/gzQaQ8bD/MlJmszbfduR/pzQ+V+DqVmsSl8MoRj
# VYnEDcGTVDAZE6zTfTen6106bDVc20HXEtqpSQvf2ICKCZNijrVmzyWIzYS4sT+k
# OQ/ZAp7rEkyVfPNrBaleFoPMuGfi6BOdzFuC00yz7Vv/3uVzrCM7LQC/NVV0CUnY
# SVgaf5I25lGSDvMmfRxNF7zJ7EMm0L9BX0CpRET0medXh55QH1dUqD79dGMvsVBl
# CeZYQi5DGky08CVHWfoEHpPUJkZKUIGy3r54t/xnFeHJV4QeD2PW6WK61l9VLupc
# xigIBCU5uA4rqfJMlxwHPw1S9e3vL4IPAgMBAAGjgfIwge8wHwYDVR0jBBgwFoAU
# oBEKIz6W8Qfs4q8p74Klf9AwpLQwHQYDVR0OBBYEFFN5v1qqK0rPVIDh2JvAnfKy
# A2bLMA4GA1UdDwEB/wQEAwIBhjAPBgNVHRMBAf8EBTADAQH/MBEGA1UdIAQKMAgw
# BgYEVR0gADBDBgNVHR8EPDA6MDigNqA0hjJodHRwOi8vY3JsLmNvbW9kb2NhLmNv
# bS9BQUFDZXJ0aWZpY2F0ZVNlcnZpY2VzLmNybDA0BggrBgEFBQcBAQQoMCYwJAYI
# KwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmNvbW9kb2NhLmNvbTANBgkqhkiG9w0BAQwF
# AAOCAQEAGIdR3HQhPZyK4Ce3M9AuzOzw5steEd4ib5t1jp5y/uTW/qofnJYt7wNK
# fq70jW9yPEM7wD/ruN9cqqnGrvL82O6je0P2hjZ8FODN9Pc//t64tIrwkZb+/UNk
# fv3M0gGhfX34GRnJQisTv1iLuqSiZgR2iJFODIkUzqJNyTKzuugUGrxx8VvwQQuY
# AAoiAxDlDLH5zZI3Ge078eQ6tvlFEyZ1r7uq7z97dzvSxAKRPRkA0xdcOds/exgN
# Rc2ThZYvXd9ZFk8/Ub3VRRg/7UqO6AZhdCMWtQ1QcydER38QXYkqa4UxFMToqWpM
# gLxqeM+4f452cpkMnf7XkQgWoaNflTCCBfUwggPdoAMCAQICEB2iSDBvmyYY0ILg
# ln0z02owDQYJKoZIhvcNAQEMBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpO
# ZXcgSmVyc2V5MRQwEgYDVQQHEwtKZXJzZXkgQ2l0eTEeMBwGA1UEChMVVGhlIFVT
# RVJUUlVTVCBOZXR3b3JrMS4wLAYDVQQDEyVVU0VSVHJ1c3QgUlNBIENlcnRpZmlj
# YXRpb24gQXV0aG9yaXR5MB4XDTE4MTEwMjAwMDAwMFoXDTMwMTIzMTIzNTk1OVow
# fDELMAkGA1UEBhMCR0IxGzAZBgNVBAgTEkdyZWF0ZXIgTWFuY2hlc3RlcjEQMA4G
# A1UEBxMHU2FsZm9yZDEYMBYGA1UEChMPU2VjdGlnbyBMaW1pdGVkMSQwIgYDVQQD
# ExtTZWN0aWdvIFJTQSBDb2RlIFNpZ25pbmcgQ0EwggEiMA0GCSqGSIb3DQEBAQUA
# A4IBDwAwggEKAoIBAQCGIo0yhXoYn0nwli9jCB4t3HyfFM/jJrYlZilAhlRGdDFi
# xRDtsocnppnLlTDAVvWkdcapDlBipVGREGrgS2Ku/fD4GKyn/+4uMyD6DBmJqGx7
# rQDDYaHcaWVtH24nlteXUYam9CflfGqLlR5bYNV+1xaSnAAvaPeX7Wpyvjg7Y96P
# v25MQV0SIAhZ6DnNj9LWzwa0VwW2TqE+V2sfmLzEYtYbC43HZhtKn52BxHJAteJf
# 7wtF/6POF6YtVbC3sLxUap28jVZTxvC6eVBJLPcDuf4vZTXyIuosB69G2flGHNyM
# fHEo8/6nxhTdVZFuihEN3wYklX0Pp6F8OtqGNWHTAgMBAAGjggFkMIIBYDAfBgNV
# HSMEGDAWgBRTeb9aqitKz1SA4dibwJ3ysgNmyzAdBgNVHQ4EFgQUDuE6qFM6MdWK
# vsG7rWcaA4WtNA4wDgYDVR0PAQH/BAQDAgGGMBIGA1UdEwEB/wQIMAYBAf8CAQAw
# HQYDVR0lBBYwFAYIKwYBBQUHAwMGCCsGAQUFBwMIMBEGA1UdIAQKMAgwBgYEVR0g
# ADBQBgNVHR8ESTBHMEWgQ6BBhj9odHRwOi8vY3JsLnVzZXJ0cnVzdC5jb20vVVNF
# UlRydXN0UlNBQ2VydGlmaWNhdGlvbkF1dGhvcml0eS5jcmwwdgYIKwYBBQUHAQEE
# ajBoMD8GCCsGAQUFBzAChjNodHRwOi8vY3J0LnVzZXJ0cnVzdC5jb20vVVNFUlRy
# dXN0UlNBQWRkVHJ1c3RDQS5jcnQwJQYIKwYBBQUHMAGGGWh0dHA6Ly9vY3NwLnVz
# ZXJ0cnVzdC5jb20wDQYJKoZIhvcNAQEMBQADggIBAE1jUO1HNEphpNveaiqMm/EA
# AB4dYns61zLC9rPgY7P7YQCImhttEAcET7646ol4IusPRuzzRl5ARokS9At3Wpwq
# QTr81vTr5/cVlTPDoYMot94v5JT3hTODLUpASL+awk9KsY8k9LOBN9O3ZLCmI2pZ
# aFJCX/8E6+F0ZXkI9amT3mtxQJmWunjxucjiwwgWsatjWsgVgG10Xkp1fqW4w2y1
# z99KeYdcx0BNYzX2MNPPtQoOCwR/oEuuu6Ol0IQAkz5TXTSlADVpbL6fICUQDRn7
# UJBhvjmPeo5N9p8OHv4HURJmgyYZSJXOSsnBf/M6BZv5b9+If8AjntIeQ3pFMcGc
# TanwWbJZGehqjSkEAnd8S0vNcL46slVaeD68u28DECV3FTSK+TbMQ5Lkuk/xYpMo
# JVcp+1EZx6ElQGqEV8aynbG8HArafGd+fS7pKEwYfsR7MUFxmksp7As9V1DSyt39
# ngVR5UR43QHesXWYDVQk/fBO4+L4g71yuss9Ou7wXheSaG3IYfmm8SoKC6W59J7u
# mDIFhZ7r+YMp08Ysfb06dy6LN0KgaoLtO0qqlBCk4Q34F8W2WnkzGJLjtXX4oemO
# CiUe5B7xn1qHI/+fpFGe+zmAEc3btcSnqIBv5VPU4OOiwtJbGvoyJi1qV3AcPKRY
# LqPzW0sH3DJZ84enGm1YMYICQzCCAj8CAQEwgZAwfDELMAkGA1UEBhMCR0IxGzAZ
# BgNVBAgTEkdyZWF0ZXIgTWFuY2hlc3RlcjEQMA4GA1UEBxMHU2FsZm9yZDEYMBYG
# A1UEChMPU2VjdGlnbyBMaW1pdGVkMSQwIgYDVQQDExtTZWN0aWdvIFJTQSBDb2Rl
# IFNpZ25pbmcgQ0ECECwnTfNkELSL/bju5S9Y3lMwDQYJYIZIAWUDBAIBBQCggYQw
# GAYKKwYBBAGCNwIBDDEKMAigAoAAoQKAADAZBgkqhkiG9w0BCQMxDAYKKwYBBAGC
# NwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQx
# IgQg1vQAlZ2qMho+/EH3lWVgEmVnot7zKFfvSfnsqDToePkwDQYJKoZIhvcNAQEB
# BQAEggEADZpP0XT8FiROIPdPZz35QXiCQ9BiwwbQobr3AaLK8An74giG0SpAITre
# UTtz74Q2Tjz8pYZo4/S54Nn4p0OnoIQ8Ik/F2PYLnVjaZn5NLpEx7mCQxsImuwhg
# swJl2y5bIRIFdEBMaRH/ZmV2xH+SS+pe8GTXYYgG12ciG2P4blLqqC6/jM0tEQnm
# xO2B4SMAfwLBwhH7WHkfkUnDTaFyDfmv51VMToS8zSGdJkXehyv4EOCII0zyBLTt
# NbpFIZnMKIOwph/t5c3CzeAwOqJsfxICSvXEx80o/XFQIam+qLgqqjeB2p0N6JlU
# EdD16wPz6Ek3batPGJoZlr4jchgSgA==
# SIG # End signature block