functions/Get-ExtensionFromGitRepository.ps1

# <copyright file="Get-ExtensionFromGitRepository.ps1" company="Endjin Limited">
# Copyright (c) Endjin Limited. All rights reserved.
# </copyright>
function Get-ExtensionFromGitRepository {
    <#
        .SYNOPSIS
        Retrieves an extension from a Git repository using the git CLI.
         
        .DESCRIPTION
        If not already available as a locally-installed module, this function installs the extension from the specified git repository.
        It also derives the additional metadata about the extension required by the tooling.
         
        .PARAMETER Name
        Specifies the name of the extension being retrieved.
         
        .PARAMETER RepositoryUri
        Specifies the Git repository URI from which to retrieve the extension.
 
        .PARAMETER RepositoryFolderPath
        Specifies the folder path within the repository where the extension is located. Defaults to standard ZF convention of 'module'.
         
        .PARAMETER TargetPath
        Specifies the path where the extension should be installed.
 
        .PARAMETER GitRef
        Specifies the version of the extension to retrieve. If not specified, the 'main' branch will be retrieved.
         
        .INPUTS
        None. You can't pipe objects to Get-ExtensionFromGitRepository.
 
        .OUTPUTS
        Hashtable.
         
        Returns a hashtable containing completed set of metadata for the extension. This consists of the originally supplied metadata
        plus these additional properties:
        - Path: The path to the installed extension.
        - Enabled: Indicates whether the extension is enabled.
         
        .EXAMPLE
        PS:> Get-ExtensionFromGitRepository -Name "MyExtension" -TargetPath "C:/MyProject/.zf" -RepositoryUri "https://github.com/myorg/MyExtension.git"
        Retrieves the 'main' branch version of the "MyExtension" extension from a git repository that uses the default ZF extension folder structure.
 
        .EXAMPLE
        PS:> Get-ExtensionFromGitRepository -Name "MyExtension" -GitRef "refs/tags/1.0" -TargetPath "C:/MyProject/.zf" -RepositoryUri "https://github.com/myorg/MyExtension.git"
        Retrieves '1.0' tagged version of the "MyExtension" extension from a git repository that uses the default ZF extension folder structure.
 
        .EXAMPLE
        PS:> Get-ExtensionFromGitRepository -Name "MyExtension" -TargetPath "C:/MyProject/.zf" -RepositoryUri "https://github.com/myorg/MyExtension.git" -RepositoryFolderPath 'modules/MyExtension'
        Retrieves the 'main' branch version of the "MyExtension" extension from a git repository that uses a custom folder structure.
    #>

    
    [CmdletBinding()]
    param(
        [Parameter(Mandatory)]
        [string] $Name,

        [Parameter(Mandatory)]
        [uri] $RepositoryUri,

        [Parameter()]
        [string] $RepositoryFolderPath = 'module',

        [Parameter(Mandatory)]
        [string] $TargetPath,

        [Parameter(Mandatory)]
        [string] $GitRef
    )

    # Potential approaches:
    # - clone the repo into the target path, checkout the required version - would need to handle pulls/updates
    # - would sub-module be any easier?
    # - only really interested in the module folder, not the whole repo
    # - clone into a temporary folder, checkout the required version, copy the module into the target path
    # - how to handle updates since the ref could be the same (i.e. branch name), but the content could have changed
    # - have an 'always pull' option?

    # Check whether module is already installed
    # TODO: Should we retain the same module-based folder structure as with PowerShell modules?
    # If so, will it even work given that the equivalent of module version will be the git ref?
    $safeGitRef = $GitRef.Replace('/', '-')
    $existingExtensionPath,$existingExtensionVersion = Get-InstalledExtensionDetails -Name $Name -TargetPath $TargetPath -GitRefAsFolderName $safeGitRef

    # Handle getting the module from the repository
    if (!$existingExtensionPath -or $existingExtensionVersion -ne $safeGitRef) {
        if (!$existingExtensionPath) {
            Write-Verbose "Extension '$Name' not found locally."
        }
        elseif ($existingExtensionVersion -ne $safeGitRef) {
            Write-Verbose "Extension '$Name' found locally but version mismatch detected. Found: '$existingExtensionVersion'; Required: '$safeGitRef' [$GitRef]"
        }
        
        Write-Host "Installing extension $Name from $RepositoryUri" -f Cyan
        
        Copy-FolderFromGitRepo `
                -RepoUrl $RepositoryUri `
                -DestinationPath (Join-Path $TargetPath $Name $safeGitRef) `
                -RepoFolderPath $RepositoryFolderPath `
                -GitRef $gitRef `
                -ErrorAction Continue       # Log the errors but we'll use the logic below to handle them

        $existingExtensionPath,$existingExtensionVersion = Get-InstalledExtensionDetails -Name $Name -TargetPath $TargetPath -GitRefAsFolderName $safeGitRef
        if (!$existingExtensionPath) {
            throw "Failed to install extension $Name ($GitRef) from $RepositoryUri repository"
        }
        Write-Host "INSTALLED MODULE: $Name ($existingExtensionVersion)" -f Cyan
    }
    else {
        Write-Host "FOUND MODULE: $Name ($existingExtensionVersion)" -f Cyan
    }

    # Return the additional extension metadata that this function has populated
    $additionalMetadata = @{
        Path = $existingExtensionPath
        Enabled = $true
    }

    return $additionalMetadata
}