import-tasks.ps1

# <copyright file="import-tasks.ps1" company="Endjin Limited">
# Copyright (c) Endjin Limited. All rights reserved.
# </copyright>
[CmdletBinding()]
param (
    [Parameter(Mandatory)]
    [string] $ZfPath
)

# Functionality is provided by extensions that are specified in 2 ways:
# 1) Defining the '$zerofailedExtensions' variable early in the calling script (i.e. before calling 'ZeroFailed.tasks')
# 2) Via the 'ZF_EXTENSIONS' environment variable containing a JSON string with the same structure as #1. Note that this method will take precedence over #1.
if ($env:ZF_EXTENSIONS) {
    if ($zerofailedExtensions) {
        Write-Host "Overriding extensions configuration with environment variable 'ZF_EXTENSIONS'" -f Green
    }
    else {
        Write-Host "Loading extensions configuration from environment variable 'ZF_EXTENSIONS'" -f Green
    }

    try {
        [hashtable[]]$zerofailedExtensions = $env:ZF_EXTENSIONS | ConvertFrom-Json -AsHashtable
    }
    catch {
        Write-Warning "Failed to parse environment variable 'ZF_EXTENSIONS' as JSON: $($_.Exception.Message)"
        Write-Verbose "ZF_EXTENSIONS: $env:ZF_EXTENSIONS"
        [hashtable[]]$zerofailedExtensions = @()
    }
}

# By default, extensions are loaded from the PowerShell Gallery, but this can be overridden
# in a similar fashion to the 'ZF_EXTENSIONS' property.
$zerofailedExtensionsRepository ??= !$env:ZF_EXTENSIONS_PS_REPO ? "PSGallery" : $env:ZF_EXTENSIONS_PS_REPO

# Process the extensions configuration, obtaining them where necessary and
# filling-out the addtional metadata required by subsequent steps to load them.
if ($zerofailedExtensions.Count -gt 0) {
    Write-Host "*** Registering Extensions..." -f Green
    $registeredExtensions = Register-Extensions -Extensions $zerofailedExtensions `
                                                -DefaultPSRepository $zerofailedExtensionsRepository `
                                                -ZfPath $ZfPath `
                                                -Verbose:$VerbosePreference
}
else {
    Write-Host "No extensions specified" -f Yellow
}

# Validate whether extension dependencies are non-conflicting
# For the moment we'll just log a warning and remove duplicate references, with no regard for versioning - first one wins
($registeredExtensions | Group-Object -Property Name) |
    Where-Object { $_.Count -gt 1 } |
    ForEach-Object {
        Write-Warning "Multiple versions of extension '$($_.Name)' have been resolved - removing duplicates, will use the first one found: $($_.Group[0] | Select-Object Name,Version,Path | ConvertTo-Json)"
    }
$registeredExtensions = $registeredExtensions |
                            Group-Object -Property Name |
                                ForEach-Object {
                                    $_.Group |
                                    Select-Object -First 1
                                }

#
# Load the process definition
#
# The logical process needs to be defined before importing the tasks as they will likely
# reference other tasks that they depend on etc.

# First we decide where the core process is being defined:
# 1) Check whether an extension has been declared as providing it via the 'Process' property
# NOTE: For the moment, the first one found wins
# 2) If not, assume the calling process is defining their own process
[array]$processesFromExtension = $registeredExtensions |
                            Where-Object { $_.ContainsKey("Process") }
if ($processesFromExtension.Count -gt 1) {
    Write-Warning "Multiple extensions have declared a process definition, will use the first one found: $($processesFromExtension[0] | Select-Object Name,Version,Path,Process | ConvertTo-Json)"
}

$processFromExtension = $processesFromExtension | Select-Object -First 1
if ($processFromExtension) {
    $processPath = Join-Path $processFromExtension.Path $processFromExtension.Process
    Write-Host "Using process from extension '$($processFromExtension.Name)'" -f Green
    
    # Dot-source the file that defines the tasks representing the top-level process
    if (!(Test-Path $processPath)) {
        throw "Process definition not found: $processPath"
    }
    Write-Verbose "Importing process definition: $processPath"
    . $processPath
}
else {
    $localProcessDefinition = Get-ChildItem $ZfPath\*.process.ps1 | Select-Object -First 1
    if ($localProcessDefinition) {
        Write-Host "Found local process definition '$($localProcessDefinition.Name)'" -f Green
        . $localProcessDefinition.FullName
    }
    else {
        Write-Host "No process definition has been specified from an extension or found locally, assuming an in-line definition" -f Green
    }
}

#
# Load tasks & functions from extensions
#
if ($registeredExtensions.Count -gt 0) {
    Write-Host "Loading functions & tasks from extensions..." -f Green
    # We do this processing in reverse order (i.e. starting from the bottom of the dependency tree),
    # so that any overriding values that are set in the dependent extension(s) can take precedence.
    for ($i=$registeredExtensions.Count-1; $i -ge 0; $i--) {
        
        $extension = $registeredExtensions[$i]
        Write-Host $extension.Name -f Cyan
        $extensionName = $extension.Name
        if (!$extension.Enabled) {
            Write-Warning "Skipping disabled extension '$extensionName'"
            continue
        }
    
        # Import tasks
        Write-Host "- Importing tasks"
        $tasksDir = Join-Path $extension.Path "tasks"
        $tasksToImport = Get-TasksFileListFromExtension -TasksPath $tasksDir
        if (!($tasksToImport)) {
            Write-Warning "No tasks found in '$extensionName'"
        }
        else {
            $tasksToImport | ForEach-Object {
                Write-Verbose "Importing task '$($_.FullName)'"
                . $_
            }
        }
    
        # Import functions
        Write-Host "- Importing functions"
        $functionsDir = Join-Path $extension.Path "functions"
        $functionsToImport = Get-FunctionsFileListFromExtension -FunctionsPath $functionsDir
        $functionsToImport | ForEach-Object {
            Write-Verbose "Importing function '$($_.FullName)'"
            . $_
        }
    
    }
}
Write-Host "*** Extensions registration complete`n" -f Green