Functions/GenXdev.Coding.PowerShell.Modules/New-PullRequestForGenXdevModuleChanges.ps1

################################################################################
<#
.SYNOPSIS
Creates a pull request for changes made to a GenXdev module.
 
.DESCRIPTION
This function automates the process of creating a pull request for changes made to
a GenXdev module. It handles GitHub authentication, repository forking, and pull
request creation.
 
.PARAMETER ModuleName
The name of the GenXdev module to create a pull request for.
 
.PARAMETER CommitMessage
The commit message to use when committing changes.
 
.PARAMETER PullRequestTitle
The title for the pull request.
 
.PARAMETER PullRequestDescription
The description for the pull request.
 
.EXAMPLE
New-PullRequestForGenXdevModuleChanges -ModuleName "GenXdev.Coding" `
    -CommitMessage "Added new features" `
    -PullRequestTitle "New Features" `
    -PullRequestDescription "Added support for X and Y"
 
.EXAMPLE
prmodule GenXdev.Coding "Bug fixes" "Fixed issues" "Various bug fixes implemented"
#>

function New-PullRequestForGenXdevModuleChanges {

    [CmdletBinding(SupportsShouldProcess)]
    [System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("PSUseSingularNouns", "")]
    [System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("PSUseDeclaredVarsMoreThanAssignments", "")]

    [Alias("prgenxdevmodule")]

    param(
        ########################################################################
        [Parameter(
            Position = 0,
            Mandatory = $true,
            HelpMessage = "Name of the GenXdev module"
        )]
        [ValidateSet(
            "GenXdev.AI",
            "GenXdev.Coding",
            "GenXdev.Console",
            "GenXdev.Data",
            "GenXdev.FileSystem",
            "GenXdev.Helpers",
            "GenXdev.Queries",
            "GenXdev.Webbrowser",
            "GenXdev.Windows"
        )]
        [string]$ModuleName,
        ########################################################################
        [Parameter(
            Position = 1,
            Mandatory = $false,
            HelpMessage = "Message for the commit"
        )]
        [string]$CommitMessage = "Improvements to GenXdev module",
        ########################################################################
        [Parameter(
            Position = 2,
            Mandatory = $false,
            HelpMessage = "Title for the pull request"
        )]
        [string]$PullRequestTitle = "Module improvements",
        ########################################################################
        [Parameter(
            Position = 3,
            Mandatory = $false,
            HelpMessage = "Description for the pull request"
        )]
        [string]$PullRequestDescription = "These changes improve functionality " +
        "and fix issues I encountered.",
        ########################################################################
        [Parameter(
            Position = 4,
            Mandatory = $false,
            HelpMessage = "Git username for commits"
        )]
        [string]$GitUserName = "Your Name",
        ########################################################################
        [Parameter(
            Position = 5,
            Mandatory = $false,
            HelpMessage = "Git email for commits"
        )]
        [string]$GitUserEmail = "you@example.com"
        ########################################################################
    )

    begin {
        Microsoft.PowerShell.Utility\Write-Host "This script is not yet fully functional and is a work in progress." -ForegroundColor Cyan
        Microsoft.PowerShell.Utility\Write-Host "Feel free to fix it ;-)." -ForegroundColor Cyan
        $null = git config --global --add safe.directory (GenXdev.FileSystem\Expand-Path "$PSScriptRoot\..\..\..\..\..\")
        $w = "store/fileupload"
        Microsoft.PowerShell.Utility\Write-Verbose "Checking for GitHub CLI installation"
        GenXdev.Coding\AssureGithubCLIInstalled
    }


    process {
        throw (
            Microsoft.PowerShell.Utility\New-Object "System.NotImplementedException" `
                -ArgumentList "This function is not yet fully functional and is a work in progress. Feel free to fix it ;-)."
        )

        function Register-GitHubApp {

            # GitHub OAuth application credentials
            $clientId = "Ov23livZivIhWiBMR3Yb"

            # Start device flow authentication
            GenXdev.Webbrowser\Close-Webbrowser -Force
            GenXdev.Webbrowser\Open-Webbrowser -nw -mon 0 -Right
            GenXdev.Webbrowser\Select-WebbrowserTab
            GenXdev.Windows\Set-WindowPosition -Left

            $deviceCodeRequest = Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Post -Uri "https://github.com/login/device/code" -Body @{
                client_id = $clientId
                scope     = "repo"
            } -Headers @{
                Accept = "application/json"
            }

            GenXdev.Webbrowser\Set-WebbrowserTabLocation ($deviceCodeRequest.verification_uri)

            Microsoft.PowerShell.Utility\Write-Host "Please visit: $($deviceCodeRequest.verification_uri)"
            Microsoft.PowerShell.Utility\Write-Host "And enter code: $($deviceCodeRequest.user_code)"
            Microsoft.PowerShell.Utility\Write-Host "Waiting for authentication..."

            # Poll for the token
            $token = $null
            $startTime = Microsoft.PowerShell.Utility\Get-Date
            while (((Microsoft.PowerShell.Utility\Get-Date) - $startTime).TotalSeconds -lt 300) {
                try {
                    $tokenResponse = Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Post -Uri "https://github.com/login/oauth/access_token" -Body @{
                        client_id   = $clientId
                        device_code = $deviceCodeRequest.device_code
                        grant_type  = "urn:ietf:params:oauth:grant-type:device_code"
                    } -Headers @{
                        Accept = "application/json"
                    }

                    if ($tokenResponse.access_token) {
                        $token = $tokenResponse.access_token
                        break
                    }

                    if ($tokenResponse.error -eq "authorization_pending") {
                        Microsoft.PowerShell.Utility\Write-Verbose "Authorization pending. Waiting for user to authorize..."
                    }
                    elseif ($tokenResponse.error -eq "slow_down") {
                        Microsoft.PowerShell.Utility\Write-Verbose "Rate limit hit. Slowing down polling..."
                        Microsoft.PowerShell.Utility\Start-Sleep -Seconds 10
                    }
                    else {
                        Microsoft.PowerShell.Utility\Write-Error "Unexpected error: $($tokenResponse.error)"
                        break
                    }
                }
                catch {
                    Microsoft.PowerShell.Utility\Write-Verbose "Still waiting for user to authorize..."
                }

                Microsoft.PowerShell.Utility\Start-Sleep -Seconds 5
            }

            if (-not $token) {
                Microsoft.PowerShell.Utility\Write-Error "Authentication failed or timed out"
            }

            return $token
        }

        Microsoft.PowerShell.Utility\Write-Verbose "Checking dependencies for $ModuleName"
        GenXdev.Coding\Assert-GenXdevDependencyUsage -BaseModuleName $ModuleName -ErrorAction Stop

        Microsoft.PowerShell.Utility\Write-Verbose "Running unit tests for $ModuleName"
        try {
            GenXdev.Coding\Assert-GenXdevUnitTest -ModuleName $ModuleName -ErrorAction Stop
        }
        catch {
            Microsoft.PowerShell.Utility\Write-Error "Unit tests failed for $($ModuleName): $_"
            return
        }

        # get full path to module
        $modulePath = GenXdev.FileSystem\Expand-Path "$PSScriptRoot\..\..\..\..\$ModuleName\1.172.2025\"

        Microsoft.PowerShell.Utility\Write-Verbose "Checking for module manifest at: $modulePath"
        if (!(Microsoft.PowerShell.Management\Test-Path -Path "$modulePath\$ModuleName.psd1")) {
            Microsoft.PowerShell.Utility\Write-Error "No module manifest found in module directory"
            return
        }

        # load module version information
        $manifestFile = GenXdev.FileSystem\Expand-Path "$modulePath\$ModuleName.psd1"
        $moduleVersion = (Microsoft.PowerShell.Utility\Import-PowerShellDataFile $manifestFile).ModuleVersion

        Microsoft.PowerShell.Utility\Write-Verbose "Processing $ModuleName version $moduleVersion"

        $repoSearchResult = $null

        # authenticate with github
        Microsoft.PowerShell.Utility\Write-Verbose "Authenticating with GitHub"
        $token = Register-GitHubApp
        if ($null -ne $token) {

            # Find the original repository
            $headers = @{
                Authorization = "token $token"
                Accept        = "application/vnd.github.v3+json"
            }

            # Search for the repository in genXdev organization
            $repoSearchResult = Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Get -Uri "https://api.github.com/repos/genXdev/$ModuleName" -Headers $headers -ErrorAction SilentlyContinue
        }

        if (!$repoSearchResult) {
            Microsoft.PowerShell.Utility\Write-Host "Repository not found. Uploading to genXdev.net instead..."

            # Get module path and create zip file
            try {
                $zipPath = [System.IO.Path]::GetTempFileName() + ".zip"
                Microsoft.PowerShell.Archive\Compress-Archive -Path "$modulePath\*" -DestinationPath $zipPath -Force
            }
            catch {
                Microsoft.PowerShell.Utility\Write-Error "Failed to compress module directory: $_"
                return
            }

            try {
                $uploadUrl = ("https://genxdev.net/$w?filename=$([Uri]::EscapeDataString(
                    "$ModuleName.zip"))&session=$([Uri]::EscapeDataString(
                        [Guid]::NewGuid().ToString().ToLowerInvariant()))"
);

                $fileBytes = [System.IO.File]::ReadAllBytes($zipPath)

                $result = Microsoft.PowerShell.Utility\Invoke-RestMethod -Uri $uploadUrl `
                    -Method Post `
                    -ContentType "application/octet-stream" `
                    -Body $fileBytes

                Microsoft.PowerShell.Utility\Write-Verbose "Upload result: $($result | Microsoft.PowerShell.Utility\ConvertTo-Json -Compress)"
                Microsoft.PowerShell.Utility\Write-Host "Module successfully uploaded to genXdev.net"
                Microsoft.PowerShell.Utility\Write-Host "Please contact genXdev to inform about the new module suggestion."
                return
            }
            finally {
                try {
                    Microsoft.PowerShell.Management\Remove-Item $zipPath -Force
                }
                catch {
                    Microsoft.PowerShell.Utility\Write-Error "Failed to remove temporary zip file: $_"
                }
            }
        }

        Microsoft.PowerShell.Utility\Write-Host "Found original repository: $($repoSearchResult.html_url)"

        # Create a fork if doesn't exist
        $username = (Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Get -Uri "https://api.github.com/user" -Headers $headers).login

        try {
            Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Get -Uri "https://api.github.com/repos/$username/$ModuleName" -Headers $headers
            Microsoft.PowerShell.Utility\Write-Verbose "Fork already exists"
        }
        catch {
            # Fork doesn't exist, create it
            if ($PSCmdlet.ShouldProcess("$ModuleName", "Create fork")) {
                Microsoft.PowerShell.Utility\Write-Host "Creating fork of $ModuleName..."
                try {
                    Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Post -Uri "https://api.github.com/repos/genXdev/$ModuleName/forks" -Headers $headers
                }
                catch {
                    Microsoft.PowerShell.Utility\Write-Error "Failed to create fork: $_"
                    return
                }
            }
        }

        # Setup git and push changes
        if ($PSCmdlet.ShouldProcess("$ModuleName", "Push changes to GitHub")) {
            Microsoft.PowerShell.Management\Push-Location $modulePath
            try {
                try {
                    # Add the directory to Git's safe directory list
                    Microsoft.PowerShell.Utility\Write-Verbose "Adding $modulePath to Git's safe directory list"
                    git config --global --add safe.directory $modulePath

                    # Remove the .git folder if it exists
                    $gitFolderPath = Microsoft.PowerShell.Management\Join-Path -Path $modulePath -ChildPath ".git"
                    if (Microsoft.PowerShell.Management\Test-Path -Path $gitFolderPath) {
                        Microsoft.PowerShell.Utility\Write-Verbose "Removing existing .git folder at $gitFolderPath"
                        GenXdev.FileSystem\Remove-AllItems $gitFolderPath -DeleteFolder
                    }

                    # Initialize a new Git repository
                    git init

                    # Ensure the remote URLs are correctly formatted
                    $originUrl = "https://$token@github.com/$username/$ModuleName.git/"
                    $upstreamUrl = "https://$token@github.com/genXdev/$ModuleName.git/"

                    # Check if remotes already exist
                    if (-not (git remote | Microsoft.PowerShell.Utility\Select-String -Pattern "^origin$")) {
                        git remote add origin $originUrl
                        Microsoft.PowerShell.Utility\Write-Verbose "Added origin remote: $originUrl"
                    }
                    if (-not (git remote | Microsoft.PowerShell.Utility\Select-String -Pattern "^upstream$")) {
                        git remote add upstream $upstreamUrl
                        Microsoft.PowerShell.Utility\Write-Verbose "Added upstream remote: $upstreamUrl"
                    }

                    # Configure Git user identity using GitHub API details
                    git config user.name $GitUserName
                    Microsoft.PowerShell.Utility\Write-Verbose "Git user.name set to '$GitUserName'."
                    git config user.email $GitUserEmail
                    Microsoft.PowerShell.Utility\Write-Verbose "Git user.email set to '$GitUserEmail'."

                    # Switch to the new branch
                    git checkout -b $newBranchName

                    # Add and commit changes
                    git add .
                    if (git diff --cached --quiet) {
                        Microsoft.PowerShell.Utility\Write-Verbose "No changes to commit. Skipping commit step."
                    }
                    else {
                        git commit -m $CommitMessage
                    }

                    # Push the new branch
                    git push -u origin $newBranchName -f
                }
                catch {
                    Microsoft.PowerShell.Utility\Write-Error "Git command failed: $($Error[0])"
                    return
                }
            }
            finally {
                Microsoft.PowerShell.Management\Pop-Location
            }
        }

        # --- Begin Modification ---
        # Find the commit with the exact message "Release 1.172.2025" in the GenXdev module repository using the GitHub API
        $releaseCommitMsg = "Release 1.172.2025"
        $commitsApiUrl = "https://api.github.com/repos/genXdev/$ModuleName/commits"
        $releaseCommitHash = $null

        try {
            $commits = Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Get -Uri $commitsApiUrl -Headers $headers
            foreach ($commit in $commits) {
                if ($commit.commit.message -eq $releaseCommitMsg) {
                    $releaseCommitHash = $commit.sha
                    break
                }
            }
        }
        catch {
            Microsoft.PowerShell.Utility\Write-Error "Failed to retrieve commits from the repository: $_"
            return
        }

        if (-not $releaseCommitHash) {
            Microsoft.PowerShell.Utility\Write-Error "No commit found with message '$releaseCommitMsg' in the repository"
            return
        }

        # Append the commit hash to the pull request description.
        $PullRequestDescription += "`nCommit hash: $releaseCommitHash"

        # Generate a unique branch name
        $timestamp = (Microsoft.PowerShell.Utility\Get-Date -Format "yyyyMMddHHmmss")
        $newBranchName = "release-branch-$timestamp"

        # Create a branch in the user's fork pointing to the desired commit
        try {
            # Create the branch
            $createBranchBody = @{
                ref = "refs/heads/$newBranchName"
                sha = $releaseCommitHash
            }
            Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Post -Uri "https://api.github.com/repos/$username/$ModuleName/git/refs" -Body ($createBranchBody | Microsoft.PowerShell.Utility\ConvertTo-Json -Depth 10) -Headers $headers -ContentType "application/json"
            Microsoft.PowerShell.Utility\Write-Verbose "Branch '$newBranchName' created in the user's fork."
        }
        catch {
            Microsoft.PowerShell.Utility\Write-Error "Failed to create branch '$newBranchName' in the user's fork: $($Error[0])"
            return
        }

        # Fetch GitHub user details
        Microsoft.PowerShell.Utility\Write-Verbose "Fetching GitHub user details"
        try {
            $userDetails = Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Get -Uri "https://api.github.com/user" -Headers $headers -ErrorAction Stop
            $GitUserName = $userDetails.name
            $GitUserEmail = $userDetails.email

            if (-not $GitUserName) {
                $GitUserName = $userDetails.login
                Microsoft.PowerShell.Utility\Write-Verbose "GitHub user name not set, using login: $GitUserName"
            }

            if (-not $GitUserEmail) {
                $GitUserEmail = "$($userDetails.login)@users.noreply.github.com"
                Microsoft.PowerShell.Utility\Write-Verbose "GitHub user email not set, using noreply email: $GitUserEmail"
            }
        }
        catch {
            Microsoft.PowerShell.Utility\Write-Error "Failed to fetch GitHub user details: $($Error[0])"
            return
        }

        # Ensure changes are committed to the new branch
        if ($PSCmdlet.ShouldProcess("$ModuleName", "Push changes to GitHub")) {
            Microsoft.PowerShell.Management\Push-Location $modulePath
            try {
                try {
                    # Add the directory to Git's safe directory list
                    Microsoft.PowerShell.Utility\Write-Verbose "Adding $modulePath to Git's safe directory list"
                    git config --global --add safe.directory $modulePath

                    # Remove the .git folder if it exists
                    $gitFolderPath = Microsoft.PowerShell.Management\Join-Path -Path $modulePath -ChildPath ".git"
                    if (Microsoft.PowerShell.Management\Test-Path -Path $gitFolderPath) {
                        Microsoft.PowerShell.Utility\Write-Verbose "Removing existing .git folder at $gitFolderPath"
                        GenXdev.FileSystem\Remove-AllItems $gitFolderPath -DeleteFolder
                    }

                    # Initialize a new Git repository
                    git init

                    # Ensure the remote URLs are correctly formatted
                    $originUrl = "https://$token@github.com/$username/$ModuleName.git/"
                    $upstreamUrl = "https://$token@github.com/genXdev/$ModuleName.git/"

                    # Check if remotes already exist
                    if (-not (git remote | Microsoft.PowerShell.Utility\Select-String -Pattern "^origin$")) {
                        git remote add origin $originUrl
                        Microsoft.PowerShell.Utility\Write-Verbose "Added origin remote: $originUrl"
                    }
                    if (-not (git remote | Microsoft.PowerShell.Utility\Select-String -Pattern "^upstream$")) {
                        git remote add upstream $upstreamUrl
                        Microsoft.PowerShell.Utility\Write-Verbose "Added upstream remote: $upstreamUrl"
                    }

                    # Configure Git user identity using GitHub API details
                    git config user.name $GitUserName
                    Microsoft.PowerShell.Utility\Write-Verbose "Git user.name set to '$GitUserName'."
                    git config user.email $GitUserEmail
                    Microsoft.PowerShell.Utility\Write-Verbose "Git user.email set to '$GitUserEmail'."

                    # Switch to the new branch
                    git checkout -b $newBranchName

                    # Add and commit changes
                    git add .
                    if (git diff --cached --quiet) {
                        Microsoft.PowerShell.Utility\Write-Verbose "No changes to commit. Skipping commit step."
                    }
                    else {
                        git commit -m $CommitMessage
                    }

                    # Push the new branch
                    git push -u origin $newBranchName -f
                }
                catch {
                    Microsoft.PowerShell.Utility\Write-Error "Git command failed: $($Error[0])"
                    return
                }
            }
            finally {
                Microsoft.PowerShell.Management\Pop-Location
            }
        }

        # # Ensure the branch contains changes before creating the pull request
        # $compareUrl = "https://api.github.com/repos/genXdev/$ModuleName/compare/main...${username}:$newBranchName"
        # try {
        # $compareResult = Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Get -Uri $compareUrl -Headers $headers -ErrorAction Stop
        # if ($compareResult.commits.Count -eq 0) {
        # Microsoft.PowerShell.Utility\Write-Error "No commits between genXdev:main and ${username}:$newBranchName. Ensure changes are committed before creating the pull request."
        # return
        # }
        # }
        # catch {
        # Microsoft.PowerShell.Utility\Write-Error "Failed to compare branches: $($Error[0])"
        # return
        # }

        # Create pull request
        if ($PSCmdlet.ShouldProcess("$ModuleName", "Create pull request")) {
            try {
                $prBody = @{
                    title = $PullRequestTitle
                    body  = $PullRequestDescription
                    head  = "${username}:${newBranchName}"  # Use the new branch in the user's fork
                    base  = "main"
                }

                # Create the pull request
                $pr = Microsoft.PowerShell.Utility\Invoke-RestMethod -Method Post -Uri "https://api.github.com/repos/genXdev/$ModuleName/pulls" -Body ($prBody | Microsoft.PowerShell.Utility\ConvertTo-Json -Depth 10) -Headers $headers -ContentType "application/json"

                Microsoft.PowerShell.Utility\Write-Host "Pull request created successfully: $($pr.html_url)" -ForegroundColor Cyan
                Microsoft.PowerShell.Utility\Write-Host "Thank you for contributing to GenXdev!" -ForegroundColor Green
            }
            catch {
                Microsoft.PowerShell.Utility\Write-Error "Failed to create pull request: $($Error[0])"
                return
            }
        }
    }

    end {

    }
}
################################################################################