mirror of
https://github.com/simstudioai/sim.git
synced 2026-02-16 09:26:18 -05:00
* fix(billing): should allow restoring subscription (#1728) * fix(already-cancelled-sub): UI should allow restoring subscription * restore functionality fixed * fix * Add pause resume block * Add db schema * Initial test passes * Tests pass * Execution pauses * Snapshot serializer * Ui checkpoint * Works 1 * Pause resume simple v1 * Hitl block works in parallel branches without timing overlap * Pending status to logs * Pause resume ui link * Big context consolidation * HITL works in loops * Fix parallels * Reference blocks properly * Fix tag dropdown and start block resolution * Filter console logs for hitl block * Fix notifs * Fix logs page * Fix logs page again * Fix * Checkpoint * Cleanup v1 * Refactor v2 * Refactor v3 * Refactor v4 * Refactor v5 * Resume page * Fix variables in loops * Fix var res bugs * Ui changes * Approval block * Hitl works e2e v1 * Fix tets * Row level lock --------- Co-authored-by: Waleed <walif6@gmail.com> Co-authored-by: Vikhyath Mondreti <vikhyathvikku@gmail.com>
162 lines
4.7 KiB
TypeScript
162 lines
4.7 KiB
TypeScript
import { createLogger } from '@/lib/logs/console/logger'
|
|
import type { DAG } from '@/executor/dag/builder'
|
|
import type { ParallelScope } from '@/executor/execution/state'
|
|
import type { BlockStateWriter } from '@/executor/execution/types'
|
|
import type { ExecutionContext, NormalizedBlockOutput } from '@/executor/types'
|
|
import type { ParallelConfigWithNodes } from '@/executor/types/parallel'
|
|
import {
|
|
calculateBranchCount,
|
|
extractBaseBlockId,
|
|
extractBranchIndex,
|
|
parseDistributionItems,
|
|
} from '@/executor/utils/subflow-utils'
|
|
import type { SerializedParallel } from '@/serializer/types'
|
|
|
|
const logger = createLogger('ParallelOrchestrator')
|
|
|
|
export interface ParallelBranchMetadata {
|
|
branchIndex: number
|
|
branchTotal: number
|
|
distributionItem?: any
|
|
parallelId: string
|
|
}
|
|
|
|
export interface ParallelAggregationResult {
|
|
allBranchesComplete: boolean
|
|
results?: NormalizedBlockOutput[][]
|
|
completedBranches?: number
|
|
totalBranches?: number
|
|
}
|
|
|
|
export class ParallelOrchestrator {
|
|
constructor(
|
|
private dag: DAG,
|
|
private state: BlockStateWriter
|
|
) {}
|
|
|
|
initializeParallelScope(
|
|
ctx: ExecutionContext,
|
|
parallelId: string,
|
|
totalBranches: number,
|
|
terminalNodesCount = 1
|
|
): ParallelScope {
|
|
const scope: ParallelScope = {
|
|
parallelId,
|
|
totalBranches,
|
|
branchOutputs: new Map(),
|
|
completedCount: 0,
|
|
totalExpectedNodes: totalBranches * terminalNodesCount,
|
|
}
|
|
if (!ctx.parallelExecutions) {
|
|
ctx.parallelExecutions = new Map()
|
|
}
|
|
ctx.parallelExecutions.set(parallelId, scope)
|
|
return scope
|
|
}
|
|
|
|
handleParallelBranchCompletion(
|
|
ctx: ExecutionContext,
|
|
parallelId: string,
|
|
nodeId: string,
|
|
output: NormalizedBlockOutput
|
|
): boolean {
|
|
const scope = ctx.parallelExecutions?.get(parallelId)
|
|
if (!scope) {
|
|
logger.warn('Parallel scope not found for branch completion', { parallelId, nodeId })
|
|
return false
|
|
}
|
|
|
|
const branchIndex = extractBranchIndex(nodeId)
|
|
if (branchIndex === null) {
|
|
logger.warn('Could not extract branch index from node ID', { nodeId })
|
|
return false
|
|
}
|
|
|
|
if (!scope.branchOutputs.has(branchIndex)) {
|
|
scope.branchOutputs.set(branchIndex, [])
|
|
}
|
|
scope.branchOutputs.get(branchIndex)!.push(output)
|
|
scope.completedCount++
|
|
|
|
const allComplete = scope.completedCount >= scope.totalExpectedNodes
|
|
return allComplete
|
|
}
|
|
|
|
aggregateParallelResults(ctx: ExecutionContext, parallelId: string): ParallelAggregationResult {
|
|
const scope = ctx.parallelExecutions?.get(parallelId)
|
|
if (!scope) {
|
|
logger.error('Parallel scope not found for aggregation', { parallelId })
|
|
return { allBranchesComplete: false }
|
|
}
|
|
|
|
const results: NormalizedBlockOutput[][] = []
|
|
for (let i = 0; i < scope.totalBranches; i++) {
|
|
const branchOutputs = scope.branchOutputs.get(i) || []
|
|
results.push(branchOutputs)
|
|
}
|
|
this.state.setBlockOutput(parallelId, {
|
|
results,
|
|
})
|
|
return {
|
|
allBranchesComplete: true,
|
|
results,
|
|
completedBranches: scope.totalBranches,
|
|
totalBranches: scope.totalBranches,
|
|
}
|
|
}
|
|
extractBranchMetadata(nodeId: string): ParallelBranchMetadata | null {
|
|
const branchIndex = extractBranchIndex(nodeId)
|
|
if (branchIndex === null) {
|
|
return null
|
|
}
|
|
|
|
const baseId = extractBaseBlockId(nodeId)
|
|
const parallelId = this.findParallelIdForNode(baseId)
|
|
if (!parallelId) {
|
|
return null
|
|
}
|
|
const parallelConfig = this.dag.parallelConfigs.get(parallelId)
|
|
if (!parallelConfig) {
|
|
return null
|
|
}
|
|
const { totalBranches, distributionItem } = this.getParallelConfigInfo(
|
|
parallelConfig,
|
|
branchIndex
|
|
)
|
|
return {
|
|
branchIndex,
|
|
branchTotal: totalBranches,
|
|
distributionItem,
|
|
parallelId,
|
|
}
|
|
}
|
|
|
|
getParallelScope(ctx: ExecutionContext, parallelId: string): ParallelScope | undefined {
|
|
return ctx.parallelExecutions?.get(parallelId)
|
|
}
|
|
|
|
findParallelIdForNode(baseNodeId: string): string | undefined {
|
|
for (const [parallelId, config] of this.dag.parallelConfigs) {
|
|
const parallelConfig = config as ParallelConfigWithNodes
|
|
if (parallelConfig.nodes?.includes(baseNodeId)) {
|
|
return parallelId
|
|
}
|
|
}
|
|
return undefined
|
|
}
|
|
|
|
private getParallelConfigInfo(
|
|
parallelConfig: SerializedParallel,
|
|
branchIndex: number
|
|
): { totalBranches: number; distributionItem?: any } {
|
|
const distributionItems = parseDistributionItems(parallelConfig)
|
|
const totalBranches = calculateBranchCount(parallelConfig, distributionItems)
|
|
|
|
let distributionItem: any
|
|
if (Array.isArray(distributionItems) && branchIndex < distributionItems.length) {
|
|
distributionItem = distributionItems[branchIndex]
|
|
}
|
|
return { totalBranches, distributionItem }
|
|
}
|
|
}
|