Public/Set-IBObject.ps1

function Set-IBObject
{
    [CmdletBinding(SupportsShouldProcess)]
    param(
        [Parameter(ParameterSetName='ObjectOnly',Mandatory,Position=0,ValueFromPipeline)]
        [PSObject]$IBObject,

        [Parameter(ParameterSetName='RefAndTemplate',Mandatory,Position=0,ValueFromPipeline,ValueFromPipelineByPropertyName)]
        [Alias('_ref','ref')]
        [string]$ObjectRef,

        [Parameter(ParameterSetName='RefAndTemplate',Mandatory,Position=1)]
        [PSObject]$TemplateObject,

        [Alias('fields','ReturnFields')]
        [string[]]$ReturnField,
        [Alias('base','ReturnBaseFields')]
        [switch]$ReturnBase,
        [switch]$BatchMode,
        [ValidateRange(1,2147483647)]
        [int]$BatchGroupSize = 1000,

        [ValidateScript({Test-ValidProfile $_ -ThrowOnFail})]
        [string]$ProfileName,
        [ValidateScript({Test-NonEmptyString $_ -ThrowOnFail})]
        [Alias('host')]
        [string]$WAPIHost,
        [ValidateScript({Test-VersionString $_ -ThrowOnFail})]
        [Alias('version')]
        [string]$WAPIVersion,
        [PSCredential]$Credential,
        [switch]$SkipCertificateCheck
    )

    Begin {
        # grab the variables we'll be using for our REST calls
        try { $opts = Initialize-CallVars @PSBoundParameters } catch { $PsCmdlet.ThrowTerminatingError($_) }

        $querystring = [String]::Empty

        # process the return fields
        if ($ReturnField.Count -gt 0) {
            if ($ReturnBase) {
                $querystring = "?_return_fields%2B=$($ReturnField -join ',')"
            }
            else {
                $querystring = "?_return_fields=$($ReturnField -join ',')"
            }
        }
        elseif ($ReturnBase) {
            $querystring = "?_return_fields%2B"
        }

        if ($BatchMode) {
            # create a list to save the objects in
            $deferredObjects = [Collections.Generic.List[PSObject]]::new()
        }
    }

    Process {

        if ($BatchMode) {
            # add the appropriate object to the list for processing during End{}
            if ('ObjectOnly' -eq $PsCmdlet.ParameterSetName) {
                $deferredObjects.Add($IBObject)
            } else {
                $deferredObjects.Add($ObjectRef)
            }
            return
        }

        if ('ObjectOnly' -eq $PsCmdlet.ParameterSetName) {

            # get the ObjectRef from the input object
            if (-not $IBObject._ref) {
                $PSCmdlet.WriteError([Management.Automation.ErrorRecord]::new(
                    "IBObject is missing '_ref' field.", $null, [Management.Automation.ErrorCategory]::InvalidArgument, $null
                ))
                return
            }
            $ObjectRef = $IBObject._ref
            $IBObject.PSObject.Properties.Remove('_ref')

            $TemplateObject = $IBObject
        }

        $query = '{0}{1}' -f $ObjectRef,$querystring
        if ($PsCmdlet.ShouldProcess($opts.WAPIHost, 'PUT')) {
            try {
                Invoke-IBWAPI -Query $query -Method 'PUT' -Body $TemplateObject @opts -EA Stop
            } catch { $PsCmdlet.WriteError($_) }
        }
    }

    End {
        if (-not $BatchMode -or $deferredObjects.Count -eq 0) { return }
        Write-Verbose "BatchMode deferred objects: $($deferredObjects.Count), group size $($BatchGroupSize)"

        # build the 'args' value for each object
        $retArgs = @{}
        if ($ReturnField.Count -gt 0) {
            if ($ReturnBase) {
                $retArgs.'_return_fields+' = $ReturnField -join ','
            } else {
                $retArgs.'_return_fields'  = $ReturnField -join ','
            }
        } else {
            $retArgs.'_return_fields+' = ''
        }

        # make calls based on the group size
        for ($i=0; $i -lt $deferredObjects.Count; $i += $BatchGroupSize) {
            $groupEnd = [Math]::Min($deferredObjects.Count, ($i+$BatchGroupSize-1))

            # build the json for this group's objects
            $body = $deferredObjects[$i..$groupEnd] | ForEach-Object {

                if ('ObjectOnly' -eq $PsCmdlet.ParameterSetName) {

                    # get the ObjectRef from the input object
                    if (-not $_._ref) {
                        $PSCmdlet.WriteError([Management.Automation.ErrorRecord]::new(
                            "IBObject is missing '_ref' field.", $null, [Management.Automation.ErrorCategory]::InvalidArgument, $null
                        ))
                        return
                    }
                    $ObjectRef = $_._ref
                    $_.PSObject.Properties.Remove('_ref')

                    $TemplateObject = $_
                } else {
                    $ObjectRef = $_
                }

                @{
                    method = 'PUT'
                    object = $ObjectRef
                    data = $TemplateObject
                    args = $retArgs
                }
            }

            if (-not $body) {
                Write-Warning "No batched objects to update. WAPI call cancelled."
                return
            }

            if ($PSCmdlet.ShouldProcess($opts.WAPIHost, 'POST')) {
                try {
                    Invoke-IBWAPI -Query 'request' -Method 'POST' -Body $body @opts -EA Stop
                } catch { $PsCmdlet.WriteError($_) }
            }
        }

    }
}