Winget.Update.System.psm1

$Script:WorkingDir = $PSScriptRoot
Get-ChildItem "$WorkingDir\functions" | ForEach-Object { . $_.FullName }

function Update-Winget-Apps {
    param(
        [Parameter(Mandatory = $False)] [ValidateRange(0, 99)] [int32] $MaxLogFiles = 3,
        [Parameter(Mandatory = $False)] [int64] $MaxLogSize = 1048576,
        [Parameter(Mandatory = $True)] [Alias('List')] [String] $ListPath,
        [Parameter(Mandatory = $True)] [Switch] $UseWhiteList = $True,
        [Parameter(Mandatory=$False)][string[]] $apps=@()
    )

    <# MAIN #>

    #Check if running account is system or interactive logon
    $Script:IsSystem = [System.Security.Principal.WindowsIdentity]::GetCurrent().IsSystem

    #Run log initialisation function
    Start-Init

    #Log running context and more...
    if ($IsSystem)
    {
        Write-ToLog "Running in System context"

        # Maximum number of log files to keep. Default is 3. Setting MaxLogFiles to 0 will keep all log files.
        $MaxLogFiles = $UPConfig.WAU_MaxLogFiles
        if ($null -eq $MaxLogFiles)
        {
            [int32]$MaxLogFiles = 3
        }
        else
        {
            [int32]$MaxLogFiles = $MaxLogFiles
        }

        # Maximum size of log file.
        $MaxLogSize = $UPConfig.WAU_MaxLogSize
        if (!$MaxLogSize)
        {
            [int64]$MaxLogSize = 1048576 # in bytes, default is 1048576 = 1 MB
        }
        else
        {
            [int64]$MaxLogSize = $MaxLogSize
        }

        #LogRotation if System
        $LogRotate = Invoke-LogRotation $LogFile $MaxLogFiles $MaxLogSize
        if ($LogRotate -eq $False)
        {
            Write-ToLog "An Exception occured during Log Rotation..."
        }

        #Run Scope Machine funtion if run as System
        $SettingsPath = "$Env:windir\system32\config\systemprofile\AppData\Local\Microsoft\WinGet\Settings\defaultState\settings.json"
        Add-ScopeMachine $SettingsPath
    }
    else
    {
        Write-ToLog "Running in User context"
    }

    #Check network connectivity
    if (Test-Network)
    {
        #Check if Winget is installed and get Winget cmd
        $TestWinget = Get-WingetCmd

        if ($TestWinget)
        {
            #Get Current Version
            $WAUCurrentVersion = $UPConfig.DisplayVersion
            Write-ToLog "WAU current version: $WAUCurrentVersion"
            if ($IsSystem)
            {
                #Delete previous list_/winget_error (if they exist) if run as System
                if (Test-Path "$WorkingDir\logs\error.txt")
                {
                    Remove-Item "$WorkingDir\logs\error.txt" -Force
                }
            }

            #Get External ListPath if run as System
            if ($UPConfig.WAU_ListPath) {
                $ListPathClean = $($UPConfig.WAU_ListPath.TrimEnd(" ", "\", "/"))
                Write-ToLog "WAU uses External Lists from: $ListPathClean"
                $NewList = Test-ListPath $ListPathClean $UPConfig.WAU_UseWhiteList "C:\Users\trmau\IdeaProjects\Winget-Updates\Winget-AutoUpdate\"
                if ($ReachNoPath) {
                    Write-ToLog "Couldn't reach/find/compare/copy from $ListPathClean..." "Red"
                    if ($ListPathClean -notlike "http*") {
                        if (Test-Path -Path "$ListPathClean" -PathType Leaf) {
                            Write-ToLog "PATH must end with a Directory, not a File..." "Red"
                        }
                    }
                    else {
                        if ($ListPathClean -match "apps.txt") {
                            Write-ToLog "PATH must end with a Directory, not a File..." "Red"
                        }
                    }
                    $Script:ReachNoPath = $False
                }
                if ($NewList) {
                    Write-ToLog "Newer List downloaded/copied to local path: C:\Users\trmau\IdeaProjects\Winget-Updates\Winget-AutoUpdate" "Yellow"
                }
            }


            #Get White or Black list
            if ($UPConfig.WAU_UseWhiteList -eq $True)
            {
                Write-ToLog "WAU uses White List config"
                $toUpdate = Get-IncludedApps
                $UseWhiteList = $true
            }
            else
            {
                Write-ToLog "WAU uses Black List config"
                $toSkip = Get-ExcludedApps
            }

            #Get outdated Winget packages
            Write-ToLog "Checking application updates on Winget Repository..." "yellow"
            $outdated = Get-WingetOutdatedApps

            #If something unusual happened
            if ($outdated -like "An unusual*")
            {
                Write-ToLog "$outdated" "cyan"
                $outdated = $False
            }

            #Run only if $outdated is populated!
            if ($outdated)
            {
                #Log list of app to update
                foreach ($app in $outdated)
                {
                    #List available updates
                    $Log = "-> Available update : $( $app.Name ). Current version : $( $app.Version ). Available version : $( $app.AvailableVersion ). App ID : $( $app.Id )"
                    $Log | Write-host
                    $Log | out-file -filepath $LogFile -Append
                }

                #Count good update installations
                $Script:InstallOK = 0

                #If White List
                if ($UseWhiteList)
                {
                    #For each app, update
                    foreach ($app in $outdated)
                    {
                        if (($toUpdate -contains $app.Id) -and $( $app.Version ) -ne "Unknown")
                        {
                            Update-App $app
                        }
                        #if current app version is unknown
                        elseif ($( $app.Version ) -eq "Unknown")
                        {
                            Write-ToLog "$( $app.Name ) : Skipped upgrade because current version is 'Unknown'" "Gray"
                        }
                        #if app is in "excluded list"
                        else
                        {
                            Write-ToLog "$( $app.Name ) : Skipped upgrade because it is not in the included app list" "Gray"
                        }
                    }
                }
                #If Black List or default
                else
                {
                    #For each app, notify and update
                    foreach ($app in $outdated)
                    {
                        if (-not($toSkip -contains $app.Id) -and $( $app.Version ) -ne "Unknown")
                        {
                            Update-App $app
                        }
                        #if current app version is unknown
                        elseif ($( $app.Version ) -eq "Unknown")
                        {
                            Write-ToLog "$( $app.Name ) : Skipped upgrade because current version is 'Unknown'" "Gray"
                        }
                        #if app is in "excluded list"
                        else
                        {
                            Write-ToLog "$( $app.Name ) : Skipped upgrade because it is in the excluded app list" "Gray"
                        }
                    }
                }

                if ($InstallOK -gt 0)
                {
                    Write-ToLog "$InstallOK apps updated ! No more update." "Green"
                }
            }

            if ($InstallOK -eq 0 -or !$InstallOK)
            {
                Write-ToLog "No new update." "Green"
            }
        }
        else
        {
            Write-ToLog "Critical: Winget not installed or detected, exiting..." "red"
            New-Item "$WorkingDir\logs\error.txt" -Value "Winget not installed or detected" -Force
            Write-ToLog "End of process!" "Cyan"
            Exit 1
        }
    }

    #End
    Write-ToLog "End of process!" "Cyan"
    Start-Sleep 3

}

function Install-Winget-Apps {
    Install-Prerequisites
    Install-Winget
}

function Install-Winget-Task {
    param (
        [Parameter(Mandatory = $true)] [ValidateSet("AtLogon", "Daily", "BiDaily", "Weekly", "BiWeekly", "Monthly")] [String] $Interval = "AtLogon",
        [Parameter(Mandatory = $true)] [String] $TaskName = "Metsys-Winget-AutoUpdate"
    )

    Write-Host "`n Setup Scheduled Task..." -ForegroundColor Yellow
    try {
        # Settings for the scheduled task for Updates
        $taskAction = New-ScheduledTaskAction -Execute "powershell.exe" -Argument "-NoProfile -ExecutionPolicy Bypass -File `"$($WingetUpdatePath)\winget-upgrade.ps1`""
        $taskTriggers = @()
        if ($Interval -eq "AtLogon") {
            $tasktriggers += New-ScheduledTaskTrigger -AtLogOn
        }
        elseif ($Interval -eq "Daily") {
            $tasktriggers += New-ScheduledTaskTrigger -Daily -At $UpdatesAtTime
        }
        elseif ($Interval -eq "BiDaily") {
            $tasktriggers += New-ScheduledTaskTrigger -Daily -At $UpdatesAtTime -DaysInterval 2
        }
        elseif ($Interval -eq "Weekly") {
            $tasktriggers += New-ScheduledTaskTrigger -Weekly -At $UpdatesAtTime -DaysOfWeek 2
        }
        elseif ($Interval -eq "BiWeekly") {
            $tasktriggers += New-ScheduledTaskTrigger -Weekly -At $UpdatesAtTime -DaysOfWeek 2 -WeeksInterval 2
        }
        elseif ($Interval -eq "Monthly") {
            $tasktriggers += New-ScheduledTaskTrigger -Weekly -At $UpdatesAtTime -DaysOfWeek 2 -WeeksInterval 4
        }
        $taskUserPrincipal = New-ScheduledTaskPrincipal -UserId S-1-5-18 -RunLevel Highest
        $taskSettings = New-ScheduledTaskSettingsSet -Compatibility Win8 -StartWhenAvailable -AllowStartIfOnBatteries -DontStopIfGoingOnBatteries -ExecutionTimeLimit 03:00:00

        # Set up the task, and register it
        $task = New-ScheduledTask -Action $taskAction -Principal $taskUserPrincipal -Settings $taskSettings -Trigger $taskTriggers
        Register-ScheduledTask -TaskName $TaskName -InputObject $task -Force | Out-Null
    }
    catch {
        Write-host "Setup Scheduled Task Installation failed! Run me with admin rights" -ForegroundColor Red
        Start-sleep 1
        return $False
    }
}