public/integrations/sync-ProjectItemsBetweenProjects.ps1


<#
.SYNOPSIS
    Syncs project items fields between two projects.
.DESCRIPTION
    This function syncs project items fields between two projects.
    It will download both projects and compare the items fields by name using slug prefix in destination used. (See FieldSlug parameter).
    The function will only update the fields that are defined in the FieldsList parameter.
    Changes will be commited to the module project staging area. Use `Show-ProjectItemStaged`to see the changes. Use `Sync-ProjectItemStaged` to commit the changes to the destination project.

.PARAMETER SourceOwner
    The owner of the source project.
.PARAMETER SourceProjectNumber
    The project number of the source project.
.PARAMETER DestinationOwner
    The owner of the destination project.
.PARAMETER DestinationProjectNumber
    The project number of the destination project.
.PARAMETER FieldsList
    The list of fields to sync between the source and destination projects.
.PARAMETER FieldSlug
    The slug to use for the fields in the destination project.
    Slug is the prefix of the field name in the destination project.
.EXAMPLE
    Sync-ProjectItemsbetweenProjects -SourceOwner github -DestinationOwner github -SourceProjectNumber $oaProject -DestinationProjectNumber $rlProject -FieldsList @("Focus","Country") -FieldSlug "oa_"
    #>

function Update-ProjectItemsBetweenProjects {
    [CmdletBinding()]
    param (
        [Parameter(Position = 0)][string]$SourceOwner,
        [Parameter(Position = 1)][string]$SourceProjectNumber,
        [Parameter(Position = 2)][string]$DestinationOwner,
        [Parameter(Position = 3)][string]$DestinationProjectNumber,
        [Parameter()][string]$FieldSlug,
        [Parameter()][switch]$NotDone
    )

    # Get destination project for error handling and caching
    ($DestinationOwner,$DestinationProjectNumber) = Get-OwnerAndProjectNumber -Owner $DestinationOwner -ProjectNumber $DestinationProjectNumber
    if([string]::IsNullOrWhiteSpace($DestinationOwner) -or [string]::IsNullOrWhiteSpace($DestinationProjectNumber)){ "Owner and ProjectNumber are required" | Write-MyError; return $null}
    $destinationProject = Get-Project -Owner $DestinationOwner -ProjectNumber $DestinationProjectNumber -Force

    # Get source project for error handling and caching
    ($SourceOwner,$SourceProjectNumber) = Get-OwnerAndProjectNumber -Owner $SourceOwner -ProjectNumber $SourceProjectNumber
    if([string]::IsNullOrWhiteSpace($SourceOwner) -or [string]::IsNullOrWhiteSpace($SourceProjectNumber)){ "Source Owner and ProjectNumber are required" | Write-MyError; return $null}
    $sourceProject = Get-Project -Owner $SourceOwner -ProjectNumber $SourceProjectNumber -Force

    # check if any of the projects are null
    if($null -eq $sourceProject -or $null -eq $destinationProject){
        "Source or Destination project not found" | Write-MyError
        return $null
    }

    # Check all the fields on the source project
    $FieldsList = $sourceProject.fields.Values.name

    # Get source project items
    # Filter items based on the NotDone parameter
    $sourceItems = $NotDone ? $($sourceProject.items | Select-ProjectItemsNotDone) : $sourceProject.items

    # Process each item in the source project
    foreach($sourceItem in $sourceItems.Values){
        # Find matchin item destination project
        # Use URL
        # By the moment we are not going to sync Drafts as they belong to single project and therefore no matching is posible
        if($null -eq $sourceItem.url){
            "Item with no URL probably a draft. Skipping." | Write-MyVerbose
            continue
        }

        $destinationItem = $destinationProject.items.Values | Where-Object { $_.url -eq $sourceItem.url }
        if($null -eq $destinationItem){
            "Item with URL $($sourceItem.url) not found in destination project" | Write-MyVerbose
            continue
        }

        # Create hashtable with the values of the fields defined in $fieldlist
        $values = @{}
        foreach($field in $FieldsList){
            $values[$field] = $sourceItem.$field
        }

        # Check if values is empty or null
        $param = @{
            Owner = $destinationOwner
            ProjectNumber = $DestinationProjectNumber
            ItemId = $destinationItem.id
            Values = $values
            FieldSlug = $FieldSlug
        }
        Edit-ProjectItemWithValues  @param

    }
} Export-ModuleMember -Function Update-ProjectItemsBetweenProjects