lib/Classes/Public/TMBroker.ps1

#region Classes

class TMBroker {

    #region Non-Static Properties

    # The settings that define how the broker will operate
    [TMBrokerSetting]$Settings = [TMBrokerSetting]::new()

    # The Task in TM that created this class instance
    [TMTask]$Task = [TMTask]::new()

    # The Task in TM that holds the Action for initializing data for the broker
    [TMBrokerSubject]$Init

    # The TMSession that the broker will use as its connection to TM
    [TMSession]$TMSession = [TMSession]::new()

    # Data about the event and its Tasks
    [TMBrokerEventData]$EventData = [TMBrokerEventData]::new()

    # Object holding various values to indicate the status of the broker
    [TMBrokerStatus]$Status = [TMBrokerStatus]::new()

    # A cache that can be passed between Actions that are invoked by the broker
    [Object]$Cache

    # A list of the Tasks that the broker will automate
    [Collections.Generic.List[TMBrokerSubject]]$Subjects = [Collections.Generic.List[TMBrokerSubject]]::new()

    #endregion Non-Static Properties

    #region Constructors

    TMBroker() {}

    TMBroker([TMBrokerExecutionMode]$mode, [String]$taskProperty, [String[]]$matchingCriteria) {
        $this.Settings = [TMBrokerSetting]::new($mode, $taskProperty, $matchingCriteria)
    }

    TMBroker([TMBrokerExecutionMode]$mode, [ScriptBlock]$matchExpression) {
        $this.Settings = [TMBrokerSetting]::new($mode, $matchExpression)
    }

    TMBroker([TMBrokerExecutionMode]$mode, [TMBrokerTaskFilter]$taskFilter) {
        $this.Settings = [TMBrokerSetting]::new($mode, $taskFilter)
    }

    TMBroker(
        [TMBrokerExecutionMode]$mode,
        [String]$taskProperty,
        [String[]]$matchingCriteria,
        [Int32]$timeout,
        [Int32]$pauseSeconds
    ) {
        $this.Settings = [TMBrokerSetting]::new($mode, $taskProperty, $matchingCriteria, $timeout, $pauseSeconds)
        $this.Status = [TMBrokerStatus]::new($timeout)
    }

    TMBroker(
        [TMBrokerExecutionMode]$mode,
        [ScriptBlock]$matchExpression,
        [Int32]$timeout,
        [Int32]$pauseSeconds
    ) {
        $this.Settings = [TMBrokerSetting]::new($mode, $matchExpression, $timeout, $pauseSeconds)
        $this.Status = [TMBrokerStatus]::new($timeout)
    }

    TMBroker(
        [TMBrokerExecutionMode]$mode,
        [TMBrokerTaskFilter]$taskFilter,
        [Int32]$timeout,
        [Int32]$pauseSeconds
    ) {
        $this.Settings = [TMBrokerSetting]::new($mode, $taskFilter, $timeout, $pauseSeconds)
        $this.Status = [TMBrokerStatus]::new($timeout)
    }

    TMBroker(
        [TMBrokerExecutionMode]$mode,
        [String]$taskProperty,
        [String[]]$matchingCriteria,
        [Int32]$timeout,
        [Int32]$pauseSeconds,
        [Boolean]$parallel,
        [Int32]$throttle
    ) {
        $this.Settings = [TMBrokerSetting]::new($mode, $taskProperty, $matchingCriteria, $timeout, $pauseSeconds, $parallel, $throttle)
        $this.Status = [TMBrokerStatus]::new($timeout, $throttle)
    }

    TMBroker(
        [TMBrokerExecutionMode]$mode,
        [ScriptBlock]$matchExpression,
        [Int32]$timeout,
        [Int32]$pauseSeconds,
        [Boolean]$parallel,
        [Int32]$throttle
    ) {
        $this.Settings = [TMBrokerSetting]::new($mode, $matchExpression, $timeout, $pauseSeconds, $parallel, $throttle)
        $this.Status = [TMBrokerStatus]::new($timeout, $throttle)
    }

    TMBroker(
        [TMBrokerExecutionMode]$mode,
        [TMBrokerTaskFilter]$taskFilter,
        [Int32]$timeout,
        [Int32]$pauseSeconds,
        [Boolean]$parallel,
        [Int32]$throttle
    ) {
        $this.Settings = [TMBrokerSetting]::new($mode, $taskFilter, $timeout, $pauseSeconds, $parallel, $throttle)
        $this.Status = [TMBrokerStatus]::new($timeout, $throttle)
    }

    #endregion Constructors

    #region Non-Static Methods

    <#
        Summary:
            Loads the EventData property using this object's TMSession
        Params:
            None
        Outputs:
            None
    #>

    [void]GetEventData() {
        if (-not $this.TMSession) {
            [TMBrokerOutput]::Throw('A TM Session is required to invoke this method')
        }

        if ($this.Settings.SubjectScope.FilterType -eq [TMBrokerSubjectScopeFilterType]::TaskFilter) {
            [TMBrokerOutput]::Verbose('Getting Event data with Task filter')
            $this.EventData.GetEventData(
                $this.TMSession.UserContext.Project.Id,
                $this.TMSession.UserContext.Event.Name,
                $this.TMSession.Name,
                $this.Settings.SubjectScope.TaskFilter
                )
        } else {
            [TMBrokerOutput]::Verbose('Getting Event data')
            $this.EventData.GetEventData(
                $this.TMSession.UserContext.Project.Id,
                $this.TMSession.UserContext.Event.Name,
                $this.TMSession.Name
            )
        }
    }

    <#
        Summary:
            Loads all of the broker-related tasks
        Params:
            TaskId - The Id of broker task
        Outputs:
            None
    #>

    [void]GetTaskData($TaskId) {
        if (-not $this.EventData) {
            [TMBrokerOutput]::Throw('Event data must be loaded before invoking this method')
        }

        # Store this Broker Task's data
        [TMBrokerOutput]::Verbose("Loading Broker Task data for Task Id: $TaskId")
        $this.Task = ($this.EventData.Tasks | Where-Object { $_.Id -eq $TaskId })

        if (-not $this.Task) {
            $this.Task = Get-TMTask -Id $TaskId -TMSession $this.TMSession.Name
        }

        # Determine if there is an init cache Task
        $this.GetInitTask()

        # Get all of the subject tasks
        $this.GetSubjectTasks()
    }

    <#
        Summary:
            Gets the init task, if present, from the Event's task data
        Params:
            None
        Outputs:
            None
    #>

    [void]GetInitTask() {
        if ($this.Settings.ExecutionMode -eq [TMBrokerExecutionMode]::Inline) {
            if (-not $this.Task) {
                [TMBrokerOutput]::Throw('Broker Task data must be loaded before invoking this method')
            }
            $InitTask = (
                $this.EventData.Tasks |
                    Where-Object { $_.Id -in $this.Task.Successors.TaskId } |
                        Where-Object -FilterScript $this.Settings.SubjectScope.MatchExpression
            )
            if ($InitTask) {
                [TMBrokerOutput]::Verbose('Creating Init Task object')
                $this.Init = [TMBrokerSubject]::new($InitTask)
            }
        }
    }

    <#
        Summary:
            Gets all of the subject tasks that will be managed by the broker from the Event's task data
        Params:
            None
        Outputs:
            None
    #>

    [void]GetSubjectTasks() {
        switch ($this.Settings.ExecutionMode) {
            'Inline' {
                [TMBrokerOutput]::Verbose("Gathering Inline Subject Tasks")

                if (-not $this.Task -and -not $this.Init) {
                    [TMBrokerOutput]::Throw('Broker Task and Init Task data must be loaded before invoking this method')
                }

                # Initialize the subjects list
                $this.Subjects = [Collections.Generic.List[Collections.Generic.List[TMBrokerSubject]]]::new()

                foreach ($TaskId in ($this.Init.Task.Successors.TaskId ?? $this.Task.Successors.TaskId)) {

                    # Initialize a list to hold all of the subject tasks for a specific asset
                    $Workflow = [Collections.Generic.List[TMBrokerSubject]]::new()

                    # Find the first/direct successor subject task
                    $SubjectTask = $this.EventData.Tasks | Where-Object { $_.Id -eq $TaskId }

                    $i = 0
                    while ($SubjectTask) {
                        $i++

                        # Add the subject task data to the workflow
                        $Workflow.Add([TMBrokerSubject]::new($SubjectTask, $i))

                        # Look for the next subject task in the workflow
                        $SubjectTask = $this.EventData.Tasks | Where-Object { $_.Id -eq $SubjectTask.Successors.TaskId }
                    }

                    # Record how many tasks are in each asset's workflow
                    $this.Status.WorkflowTaskCount = $i

                    # Add this workflow to the list of subjects
                    $this.Subjects.Add($Workflow)
                }
            }

            'Service' {
                [TMBrokerOutput]::Verbose("Gathering Service Subject Tasks")

                # Initialize the subjects list
                $this.Subjects = [Collections.Generic.List[TMBrokerSubject]]::new()

                # Filter all Tasks down to the specified scope
                $ServiceSubjectTasks = $this.EventData.Tasks | Where-Object {
                    ($_.id -ne $Broker.task.id ) -and
                    ($_.Action.Id -ne 0) -and
                    ($_.Action.name -notlike '*broker*') -and
                    -not ($_.Action.MethodParams | Where-Object { $_.ParamName -match 'get_' })
                }

                # Apply the match expression to filter tasks further
                if ($this.Settings.SubjectScope.FilterType -eq 'MatchExpression') {
                    $ServiceSubjectTasks = $ServiceSubjectTasks | Where-Object -FilterScript $this.Settings.SubjectScope.MatchExpression
                }

                # Add the filtered Tasks to the list of subject tasks
                foreach ($Task in $ServiceSubjectTasks) {
                    $this.Subjects.Add([TMBrokerSubject]::new($Task))
                }
            }

            default { }
        }
    }

    <#
        Summary:
            Invokes the Init Task's Action to fill the cache
        Params:
            None
        Outputs:
            None
    #>

    [void]PopulateCache() {
        if (-not $this.Init) {
            [TMBrokerOutput]::Throw('Init Task data must be loaded before invoking this method')
        }

        $this.Init.Invoke($this.TMSession)
    }

    <#
        Summary:
            Updates each Task's status and Action settings using fresh data from TM
        Params:
            None
        Outputs:
            None
    #>

    [void]RefreshTaskData() {
        # Gather all Task Ids
        $TaskIds = [Array]@(
            $this.Subjects.Task.Id
            $this.Task.Id
            $this.Init.Task.Id
        ) | Where-Object { $_ -gt 0 }


        # Query TM for Task statuses and Action params
        [TMBrokerOutput]::Verbose("Requesting fresh Task data from TransitionManager")
        $Statement = "find Task by 'id' inList([$($TaskIds -join ', ')]) fetch 'id', 'status', 'lastUpdated', 'apiAction.methodParams'"
        $TaskData = Invoke-TMQLStatement -TMSession $this.TMSession.Name -Statement $Statement
        [TMBrokerOutput]::Verbose("Received data for $($TaskData.Count) Task(s)")

        # Update the broker and the init Task statuses
        $this.Task.Status = ($TaskData | Where-Object Id -eq $this.Task.Id).Status
        if ($this.Init.Task.Id) {
            $this.Init.Task.Status = ($TaskData | Where-Object Id -eq $this.Init.Id).Status
        }

        switch ($this.Settings.ExecutionMode) {
            'Inline' {
                [TMBrokerOutput]::Verbose("Updating Inline Task Data")

                # Update the status of each Task in each workflow
                foreach ($Workflow in $this.Subjects) {
                    foreach ($Subject in $Workflow) {
                        $Subject.Task.Status = ($TaskData | Where-Object Id -eq $Subject.Task.Id).Status
                    }
                }
            }

            'Service' {
                [TMBrokerOutput]::Verbose("Updating Service Task data")
                foreach ($Subject in $this.Subjects) {
                    $TaskFromTM = $TaskData | Where-Object Id -eq $Subject.Task.Id
                    $Subject.Task.Status = $TaskFromTM.Status
                    $Subject.Task.LastUpdated = $TaskFromTM.LastUpdated

                    # Update the Subject's Action settings if needed
                    $Subject.UpdateActionSettings($TaskFromTM.'apiAction.methodParams')

                    # Review Task states to update throttling settings
                    if ($this.Settings.Parallel) {
                        # Handle updating Subject Task data based on the status of the task
                        switch ($Subject.Task.Status) {
                            'Started' {
                                # Mark the Action as Started so the Broker ignores it for next time
                                $Subject.Action.ExecutionStatus = 'Started'

                                # Check if a timeout was defined in the Action params
                                if ($Subject.Action.ShouldTimeout) {
                                    # Attempt to place this Subject Task on hold with a timeout comment
                                    try {
                                        [TMBrokerOutput]::Info("Resetting Task #: $($Subject.Task.TaskNumber). Task has run longer than: $($Subject.Action.Settings.Timeout.Minutes) minutes", 'DarkYellow')
                                        $Subject.Timeout($this.TMSession.Name)
                                    } catch {
                                        [TMBrokerOutput]::Warning("Could not timeout Task # $($Subject.Task.TaskNumber): $($_.Exception.Message)")
                                    }
                                }
                            }

                            'Completed' {
                                # Check to ensure the broker does not believe it's running completed Tasks
                                if ($this.Status.ActiveSubjects -contains $Subject.Task.Id) {
                                    $this.Status.ActiveSubjects.Remove($Subject.Task.Id)
                                }
                                $Subject.Action.ExecutionStatus = 'Successful'
                            }

                            'Hold' {
                                # If the Task's status was changed either manually or due to failure,
                                # reset the execution status so that it can be re-run
                                if ($this.Status.ActiveSubjects -contains $Subject.Task.Id) {
                                    $this.Status.ActiveSubjects.Remove($Subject.Task.Id)
                                }
                                $Subject.Action.ExecutionStatus = 'Failed'

                                # Check if a retry was defined in the Action params
                                if ($Subject.Action.ShouldRetry) {
                                    # Attempt to reset the Task Action and Reset the Task to Ready
                                    try {
                                        [TMBrokerOutput]::Info("Resetting Task #: $($Subject.Task.TaskNumber). Retries left: $($Subject.Action.Settings.Retry.RemainingRetries)", 'DarkYellow')
                                        $Subject.QueueRetry($this.TMSession.Name)
                                        $this.Status.LastWebRequest = Get-Date
                                    } catch {
                                        [TMBrokerOutput]::Warning("Could not retry Task # $($Subject.Task.TaskNumber): $($_.Exception.Message)")
                                    }
                                }
                            }

                            { $_ -in 'Pending', 'Ready' } {
                                # If the Task's status was changed either manually or due to failure,
                                # reset the execution status so that it can be re-run
                                if ($this.Status.ActiveSubjects -contains $Subject.Task.Id) {
                                    $this.Status.ActiveSubjects.Remove($Subject.Task.Id)
                                }
                                $Subject.Action.ExecutionStatus = 'Pending'
                            }
                        }
                    }
                }
            }
        }
    }

    <#
        Summary:
            Updates the TMBrokerProgress properties on this Status object to be used for tracking and progress bars
        Params:
            None
        Outputs:
            None
    #>

    [void]RefreshBrokerProgress() {
        [TMBrokerOutput]::Verbose("Updating Progress data")
        $this.Status.CompletedTasks.Value = ($this.Subjects | Where-Object { $_.Task.Status -eq 'Completed' -and $_.Action.ExecutionStatus -eq 'Successful' }).Count
        $this.Status.ElapsedMinutes.Value = [Math]::Ceiling($this.Settings.Timing.Timer.Elapsed.TotalMinutes)
        if ($this.Settings.Parallel) {
            $this.Status.Throttle.Value = $this.Status.ActiveSubjects.Count
        }
    }

    <#
        Summary:
            Sends a lightweight request to TM to keep the web services jsession alive
        Params:
            None
        Outputs:
            None
    #>

    [void]KeepAlive() {
        if (((Get-Date) - $this.Status.LastWebRequest).TotalMinutes -gt 10) {
            try {
                [TMBrokerOutput]::Verbose("Making Keep Alive request to TransitionManager")
                $WebRequestSplat = @{
                    Uri = "https://$($this.TMSession.TMServer)/tdstm/ws/progress/demo"
                    Method = 'GET'
                    WebSession = $this.TMSession.TMWebSession
                    SkipCertificateCheck = $this.TMSession.AllowInsecureSSL
                }
                $Response = Invoke-WebRequest @WebRequestSplat

                if ($Response.StatusCode -notin 200, 204) {
                    throw "The status code $($Response.StatusCode) does not indicate success"
                }
                [TMBrokerOutput]::Verbose("Response Status Code: $($Response.StatusCode)")
            } catch {
                [TMBrokerOutput]::Warning("Keep alive request failed: $($_.Exception.Message)")
            }

            $this.Status.LastWebRequest = Get-Date
        }
    }

    <#
        Method: Run
        Description: Executes the scoped subject Tasks
        Parameters: None
    #>

    [void]Run() {
        [TMBrokerOutput]::Debug("Execution Mode: $($this.Settings.ExecutionMode)")
        [TMBrokerOutput]::Debug("Execution Order: $($this.Settings.ExecutionOrder)")
        [TMBrokerOutput]::Debug("Execution Sort Order: $($this.Settings.ExecutionSortOrder)")
        [TMBrokerOutput]::Debug("Parallel: $($this.Settings.Parallel)")
        [TMBrokerOutput]::Debug("Throttle: $($this.Settings.Throttle)")
        [TMBrokerOutput]::Debug("Timeout Minutes: $($this.Settings.Timing.TimeoutMinutes)")
        [TMBrokerOutput]::Debug("Pause Seconds: $($this.Settings.Timing.PauseSeconds)")
        [TMBrokerOutput]::Verbose("Starting Broker execution")


        # Initialize values for progress bars
        $this.Status.CompletedTasks.MaxValue = $this.Subjects.Count
        $this.Settings.Timing.Timer.Start()
        $this.RefreshTaskData()

        while (
            ($this.Settings.Timing.Timer.Elapsed.TotalMinutes -lt $this.Settings.Timing.TimeoutMinutes) -and
            ($this.Subjects | Where-Object { $_.Action.ExecutionStatus -eq 'Pending' })
        ) {

            # Force a refresh after a few tasks have been executed
            if ($this.Status.TasksExecutedSinceRefresh -ge 3) {
                $this.RefreshTaskData()
                $this.Status.TasksExecutedSinceRefresh = 0
            }

            # Saftey check the broker task status in TM, exit if the task status is not Started
            if ($this.Task.Status -ne 'Started') {
                [TMBrokerOutput]::Throw('The status of the Broker Task has changed outside of TMConsole')
            }

            # Refresh the progress properties to be output to the TMC UI
            $this.RefreshBrokerProgress()

            # If needed, make a small request to TM to keep the session alive
            $this.KeepAlive()

            # Update the TMC UI's progress bars
            $ProgressSplat = @{
                Id              = 1
                ParentId        = 0
                Activity        = 'Subject Tasks'
                Status          = "$($this.Status.CompletedTasks.Value) of $($this.Status.CompletedTasks.MaxValue) tasks completed"
                PercentComplete = $this.Status.CompletedTasks.PercentComplete
            }
            Write-Progress @ProgressSplat

            $ProgressSplat = @{
                Id              = 2
                ParentId        = 0
                Activity        = 'Timeout'
                Status          = "$([Math]::Ceiling($this.Settings.Timing.TimeoutMinutes - $this.Settings.Timing.Timer.Elapsed.TotalMinutes)) minutes left"
                PercentComplete = $this.Status.ElapsedMinutes.PercentComplete
            }
            Write-Progress @ProgressSplat

            if ($this.Settings.Parallel) {
                $ProgressSplat = @{
                    Id              = 3
                    ParentId        = 0
                    Activity        = 'Throttle'
                    Status          = "$($this.Status.ActiveSubjects.Count) of $($this.Settings.Throttle)"
                    PercentComplete = $this.Status.Throttle.PercentComplete
                }
                Write-Progress @ProgressSplat
            }

            # Execute Subject Tasks
            switch ($this.Settings.ExecutionMode) {

                # Inline Brokers run a workflow step worth of tasks at once
                'Inline' {
                    foreach ($Workflow in $this.Subjects) {
                        $NextWorkflowSubject = $Workflow |
                            Where-Object { $_.Task.Status -ne 'Completed' -and $_.Action.ExecutionStatus -eq 'Pending' -and $_.Task.Id -notin $this.Status.ActiveSubjects } |
                                Sort-Object Order | Select-Object -First 1

                        if ($NextWorkflowSubject) {
                            if ($this.Settings.Parallel) {
                                if ($this.Status.ActiveSubjects.Count -lt $this.Settings.Throttle) {
                                    # Record the Task ID as belonging to this broker for Throttling
                                    $this.Status.ActiveSubjects.Add($NextWorkflowSubject.Task.Id)

                                    # Invoke the next workflow Subject
                                    $NextWorkflowSubject.InvokeParallel($this.TMSession, $this.Cache)
                                }
                            } else {
                                $NextWorkflowSubject.Invoke($this.TMSession, $this.Cache)
                            }
                            $this.Status.TasksExecutedSinceRefresh++
                            $this.Status.LastWebRequest = Get-Date
                        }
                    }
                }

                # Service Brokers run one task at a time, when they become ready
                'Service' {
                    # Get the most preferred actionable subject
                    $PreferredActionableSubject = $this.Subjects |
                        Where-Object { $_.Task.Status -eq 'Ready' -and $_.Action.ExecutionStatus -eq 'Pending' } |
                            Sort-Object { $_.Task."$($this.Settings.ExecutionOrder)" } -Descending:$($this.Settings.ExecutionSortOrder -eq 'Descending') |
                                Select-Object -First 1


                    # Invoke the Most Preferred, Actionable Subject
                    if ($PreferredActionableSubject) {
                        # Update the local cache so this task won't run again until another refresh from TM
                        $PreferredActionableSubject.Task.Status = 'Started'

                        # Run a Subject in a normal invocation runspace, but track that task so it 'consumes' one runspace
                        if ($this.Settings.Parallel) {
                            # Honor Throttling settings
                            if ($this.Status.ActiveSubjects.Count -lt $this.Settings.Throttle) {
                                # Record the Task ID as belonging to this broker for Throttling
                                $this.Status.ActiveSubjects.Add($PreferredActionableSubject.Task.Id)

                                # Invoke the Subject
                                $PreferredActionableSubject.InvokeParallel($this.TMSession, $this.Cache)
                            }
                        } else {
                            # Invoke this ActionRequest directly, in this runspace
                            $PreferredActionableSubject.Invoke($this.TMSession, $this.Cache)
                        }
                        $this.Status.TasksExecutedSinceRefresh++
                        $this.Status.LastWebRequest = Get-Date
                    }
                }
            }

            # Sleep, unless there are more tasks ready
            if ($this.Subjects.Task.Status -notcontains 'Ready') {
                [TMBrokerOutput]::Verbose("Pausing for $($this.Settings.Timing.PauseSeconds) second(s)")
                Start-Sleep -Seconds $this.Settings.Timing.PauseSeconds

                # Refresh the Task statuses before
                $this.RefreshTaskData()
                $this.Status.TasksExecutedSinceRefresh = 0
            }
        }
    }

    #endregion Non-Static Methods

}


class TMBrokerEventData {

    #region Non-Static Properties

    [TMEvent]$Event
    [TMTask[]]$Tasks = [System.Collections.Generic.List[TMTask]]::new()

    #endregion Non-Static Properties

    #region Constructors

    TMBrokerEventData() {}

    TMBrokerEventData([Int32]$projectId, [String]$eventName, [String]$tmSession) {
        $this.GetEventData($projectId, $eventName, $tmSession)
    }

    [void]GetEventData([Int32]$projectId, [String]$eventName, [String]$tmSession) {

        if (-not $this.Event) {
            # Get the Event object
            $this.Event = Get-TMEvent -TMSession $tmSession -ProjectId $projectId -Name $eventName
        }

        # Get all of the broker-related Tasks in the Event
        $this.Tasks = Get-TMTask -TMSession $tmSession -ProjectId $projectId -EventName $this.Event.name
    }

    [void]GetEventData([Int32]$projectId, [String]$eventName, [String]$tmSession, [TMBrokerTaskFilter]$Filter) {

        if (-not $this.Event) {
            # Get the Event object
            $this.Event = Get-TMEvent -TMSession $tmSession -ProjectId $projectId -Name $eventName
        }

        # Get all of the broker-related Tasks in the Event
        $TaskSplat = $Filter.ToHashTable()
        $this.Tasks = Get-TMTask -TMSession $tmSession -ProjectId $projectId -EventName $this.Event.name @TaskSplat
    }

    #endregion Constructors

}


class TMBrokerSubjectScope {

    #region Non-Static Properties

    # If a Task filter or match expression is not defined, this is the Task property that will be evaluated with MatchingCriteria
    [TMBrokerSubjectScopeTaskProperty]$TaskProperty = [TMBrokerSubjectScopeTaskProperty]::Title

    # If a Task filter or match expression is not defined, this list of values will be matched against TaskProperty
    [String[]]$MatchingCriteria

    # A ScriptBlock containing an expression that will be used as the -FilterScript parameter on Where-Object
    # after all of the Event Tasks have been retrieved from TM
    [ScriptBlock]$MatchExpression

    # The type of filtering that will be used to determine the Broker's Subject Tasks
    [TMBrokerSubjectScopeFilterType]$FilterType = [TMBrokerSubjectScopeFilterType]::TaskFilter

    # The TaskFilter object that will be used as a splat with Get-TMTask
    [TMBrokerTaskFilter]$TaskFilter = [TMBrokerTaskFilter]::new()

    #endregion Non-Static Properties

    #region Constructors

    TMBrokerSubjectScope() {
        $this.TaskFilter.Title.Add('\[Subject\]')
    }

    TMBrokerSubjectScope([TMBrokerSubjectScopeTaskProperty]$taskProperty, [String[]]$matchingCriteria) {
        $this.TaskProperty = $taskProperty
        $this.MatchingCriteria = $matchingCriteria
        $matchingCriteria | ForEach-Object {
            $this.TaskFilter."$taskProperty".Add($_)
        }
    }

    TMBrokerSubjectScope([ScriptBlock]$matchExpression) {
        $this.MatchExpression = $matchExpression
        $this.FilterType = [TMBrokerSubjectScopeFilterType]::MatchExpression
    }

    TMBrokerSubjectScope([TMBrokerTaskFilter]$taskFilter) {
        $this.TaskFilter = $taskFilter
    }

    #endregion Constructors

    #region Non-Static Methods

    <#
        Summary:
            Sets the MatchExpression value to a ScriptBlock that can be used with Where-Object
        Params:
            None
        Outputs:
            None
    #>

    hidden [void]GetMatchExpression() {
        $this.MatchExpression = [ScriptBlock]::Create("`$_.$($this.TaskProperty) -match '$([TMBrokerSubjectScope]::GetMatchString($this.MatchingCriteria))'")
    }

    #endregion Non-Static Methods

    #region Static Methods

    <#
        Summary:
            Converts a list of values to a regular expression that can be used with -match
        Params:
            Criteria - The list of values to be converted to a match string
        Outputs:
            A String formatted as a regular expression
    #>

    static [String]GetMatchString([String[]]$Criteria) {
        return ('(' + ($Criteria -join ')|(') + ')')
    }

    #endregion Static Methods

}


class TMBrokerSetting {

    #region Non-Static Properties

    # The values that define how the Broker decides which Tasks it will manage and execute
    [TMBrokerSubjectScope]$SubjectScope = [TMBrokerSubjectScope]::new()

    # The mode that defines the way in which the Broker will execute its Subject Tasks
    [TMBrokerExecutionMode]$ExecutionMode = [TMBrokerExecutionMode]::Service

    # Values that define the Broker's timeout and refresh intervals
    [TMBrokerTiming]$Timing  = [TMBrokerTiming]::new()

    # When determining the next Subject Task to invoke, they will be sorted by this Task property
    [TMBrokerSettingExecutionOrder]$ExecutionOrder = [TMBrokerSettingExecutionOrder]::Score

    # When determining the next Subject Task to invoke, they will be sorted in this order
    [ValidateSet('Ascending', 'Descending')]
    [String]$ExecutionSortOrder = 'Descending'

    # Will the Subject Tasks be executed in parallel or one at a time?
    [Boolean]$Parallel = $true

    # The maximum number of Subject Tasks that can be runninng in parallel
    [Int32]$Throttle = 8

    #endregion Non-Static Properties

    #region Constructors

    TMBrokerSetting() {}

    TMBrokerSetting(
        [TMBrokerExecutionMode]$mode,
        [String]$taskProperty,
        [String[]]$matchingCriteria,
        [Int32]$timeout,
        [Int32]$pauseSeconds
    ) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($taskProperty, $matchingCriteria)
        $this.Timing = [TMBrokerTiming]::new($timeout, $pauseSeconds)
    }

    TMBrokerSetting(
        [TMBrokerExecutionMode]$mode,
        [ScriptBlock]$matchExpression,
        [Int32]$timeout,
        [Int32]$pauseSeconds
    ) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($matchExpression)
        $this.Timing = [TMBrokerTiming]::new($timeout, $pauseSeconds)
    }

    TMBrokerSetting(
        [TMBrokerExecutionMode]$mode,
        [TMBrokerTaskFilter]$taskFilter,
        [Int32]$timeout,
        [Int32]$pauseSeconds
    ) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($taskFilter)
        $this.Timing = [TMBrokerTiming]::new($timeout, $pauseSeconds)
    }

    TMBrokerSetting(
        [TMBrokerExecutionMode]$mode,
        [String]$taskProperty,
        [String[]]$matchingCriteria,
        [Int32]$timeout,
        [Int32]$pauseSeconds,
        [Boolean]$parallel,
        [Int32]$throttle
    ) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($taskProperty, $matchingCriteria)
        $this.Timing = [TMBrokerTiming]::new($timeout, $pauseSeconds)
        $this.Parallel = $parallel
        $this.Throttle = $throttle
    }

    TMBrokerSetting(
        [TMBrokerExecutionMode]$mode,
        [ScriptBlock]$matchExpression,
        [Int32]$timeout,
        [Int32]$pauseSeconds,
        [Boolean]$parallel,
        [Int32]$throttle
    ) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($matchExpression)
        $this.Timing = [TMBrokerTiming]::new($timeout, $pauseSeconds)
        $this.Parallel = $parallel
        $this.Throttle = $throttle
    }

    TMBrokerSetting(
        [TMBrokerExecutionMode]$mode,
        [TMBrokerTaskFilter]$taskFilter,
        [Int32]$timeout,
        [Int32]$pauseSeconds,
        [Boolean]$parallel,
        [Int32]$throttle
    ) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($taskFilter)
        $this.Timing = [TMBrokerTiming]::new($timeout, $pauseSeconds)
        $this.Parallel = $parallel
        $this.Throttle = $throttle
    }

    TMBrokerSetting([TMBrokerExecutionMode]$mode, [String]$taskProperty, [String[]]$matchingCriteria) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($taskProperty, $matchingCriteria)
    }

    TMBrokerSetting([TMBrokerExecutionMode]$mode, [ScriptBlock]$matchExpression) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($matchExpression)
    }

    TMBrokerSetting([TMBrokerExecutionMode]$mode, [TMBrokerTaskFilter]$taskFilter) {
        $this.ExecutionMode = $mode
        $this.SubjectScope = [TMBrokerSubjectScope]::new($taskFilter)
    }

    #endregion Constructors

}


class TMBrokerTiming {

    #region Non-Static Properties

    # How many minutes the Broker will run before ending its execution
    [Int64]$TimeoutMinutes = 120

    # If the Broker is idle, with no Tasks to invoke, the number of seconds to wait before
    # querying TM again for Task data
    [Int64]$PauseSeconds = 15

    # The timer that represents the Broker's execution time
    [Diagnostics.Stopwatch]$Timer = [Diagnostics.Stopwatch]::new()

    #endregion Non-Static Properties

    #region Constructors

    TMBrokerTiming () {}

    TMBrokerTiming ([Int64]$timeoutMinutes, [Int64]$pauseSeconds) {
        $this.TimeoutMinutes = $timeoutMinutes
        $this.PauseSeconds = $pauseSeconds
    }

    #endregion Constructors

}


class TMBrokerStatus {

    #region Non-Static Properties

    # A list of the Subject Task Ids that are being executed in parallel
    [Collections.Generic.List[Int64]]$ActiveSubjects = [Collections.Generic.List[Int64]]::new()

    # The number of Subject Tasks that have been executed since fresh Task data has been received from TM
    [Int64]$TasksExecutedSinceRefresh = 0

    # The last date/time that a request was made to one of TM's web services endpoints.
    # Used to determine if a keep alive ping needs to be made
    [DateTime]$LastWebRequest = (Get-Date)

    # The number of Subject Tasks that have been completed successfully. Used for TMC progress bars
    [TMBrokerProgress]$CompletedTasks = [TMBrokerProgress]::new()

    # The number of minutes that have elapsed since the Broker was started. Used for TMC progress bars
    [TMBrokerProgress]$ElapsedMinutes = [TMBrokerProgress]::new()

    # The number of Subject Tasks thatare currently running. Used for TMC progress bars
    [TMBrokerProgress]$Throttle = [TMBrokerProgress]::new()

    #endregion Non-Static Properties

    #region Constructors

    TMBrokerStatus () {}

    TMBrokerStatus ([Int32]$timeoutMinutes) {
        $this.ElapsedMinutes = [TMBrokerProgress]::new($timeoutMinutes)
    }

    TMBrokerStatus ([Int32]$timeoutMinutes, [Int32]$throttle) {
        $this.ElapsedMinutes = [TMBrokerProgress]::new($timeoutMinutes)
        $this.Throttle = [TMBrokerProgress]::new($throttle)
    }

    #endregion Constructors

}


class TMBrokerProgress {

    #region Non-Static Properties

    [Int32]$Value = 0
    [Int32]$MaxValue = 1

    #endregion Non-Static Properties

    #region Constructors

    TMBrokerProgress() {
        $this.addPublicMembers()
    }

    TMBrokerProgress([Int32]$maxValue) {
        $this.addPublicMembers()
        $this.MaxValue = $maxValue
    }

    TMBrokerProgress([Int32]$currentValue, [Int32]$maxValue) {
        $this.addPublicMembers()
        $this.Value = $currentValue
        $this.MaxValue = $maxValue
    }

    #endregion Constructors

    #region Private Methods

    <#
        Summary:
            Adds members with calculated get and/or set methods
        Params:
            None
        Outputs:
            None
    #>

    hidden [void]addPublicMembers() {
        # public readonly Int32 PercentComplete
        $this.PSObject.Properties.Add(
            [PSScriptProperty]::new(
                'PercentComplete',
                { # get
                    return [Int32][Math]::Ceiling(($this.Value / $this.MaxValue) * 100)
                }
            )
        )
    }

    #endregion Private Methods

}


class TMBrokerTaskFilter {

    #region Non-Static Properties

    [Collections.Generic.List[Int32]]$TaskNumber = [Collections.Generic.List[Int32]]::new()
    [Collections.Generic.List[Int32]]$TaskSpecId = [Collections.Generic.List[Int32]]::new()
    [Collections.Generic.List[String]]$Status = [Collections.Generic.List[String]]::new()
    [Collections.Generic.List[String]]$AssetName = [Collections.Generic.List[String]]::new()
    [Collections.Generic.List[String]]$AssetType = [Collections.Generic.List[String]]::new()
    [Collections.Generic.List[String]]$AssetClass = [Collections.Generic.List[String]]::new()
    [Collections.Generic.List[String]]$ActionName = [Collections.Generic.List[String]]::new()
    [Collections.Generic.List[String]]$Category = [Collections.Generic.List[String]]::new()
    [Collections.Generic.List[String]]$Title = [Collections.Generic.List[String]]::new()
    [Collections.Generic.List[String]]$Team = [Collections.Generic.List[String]]::new()

    #endregion Non-Static Properties

    #region Constructors

    TMBrokerTaskFilter() {}

    #endregion Constructors

    #region Non-Static Methods

    [Hashtable]ToHashTable() {
        $returnHashtable = @{}

        if ($this.TaskNumber) { $returnHashtable.TaskNumber = $this.TaskNumber }
        if ($this.TaskSpecId) { $returnHashtable.TaskSpecId = $this.TaskSpecId }
        if ($this.Status) { $returnHashtable.Status = $this.Status }
        if ($this.AssetName) { $returnHashtable.AssetName = $this.AssetName }
        if ($this.AssetType) { $returnHashtable.AssetType = $this.AssetType }
        if ($this.AssetClass) { $returnHashtable.AssetClass = $this.AssetClass }
        if ($this.ActionName) { $returnHashtable.ActionName = $this.ActionName }
        if ($this.Category) { $returnHashtable.Category = $this.Category }
        if ($this.Title) { $returnHashtable.Title = $this.Title }
        if ($this.Team) { $returnHashtable.Team = $this.Team }

        return $returnHashtable
    }

    [String]ToString() {
        return "{$($this.ToHashTable().Keys -join ', ')}"
    }

    #endregion Non-Static Methods

}

#endregion Classes


#region Enumerations

enum TMBrokerExecutionMode {
    Service
    Inline
}


enum TMBrokerSettingExecutionOrder {
    TaskNumber
    Score
    TaskSpecId
}


enum TMBrokerSubjectScopeTaskProperty {
    ActionName
    AssetClass
    AssetName
    AssetType
    Category
    Status
    TaskNumber
    TaskSpecId
    Team
    Title
}


enum TMBrokerSubjectScopeFilterType {
    TaskFilter
    MatchExpression
}

#endregion Enumerations