mirror of
https://github.com/simstudioai/sim.git
synced 2026-01-22 21:38:05 -05:00
55 lines
1.4 KiB
TypeScript
55 lines
1.4 KiB
TypeScript
import { Edge } from 'reactflow'
|
|
|
|
/**
|
|
* Performs a depth-first search to detect all cycles in the graph
|
|
* @param edges - List of all edges in the graph
|
|
* @param startNode - Starting node for cycle detection
|
|
* @returns Array of all unique cycles found in the graph
|
|
*/
|
|
export function detectCycle(
|
|
edges: Edge[],
|
|
startNode: string
|
|
): { hasCycle: boolean; paths: string[][] } {
|
|
const visited = new Set<string>()
|
|
const recursionStack = new Set<string>()
|
|
const allCycles: string[][] = []
|
|
const currentPath: string[] = []
|
|
|
|
function dfs(node: string) {
|
|
visited.add(node)
|
|
recursionStack.add(node)
|
|
currentPath.push(node)
|
|
|
|
// Get all neighbors of current node
|
|
const neighbors = edges.filter((edge) => edge.source === node).map((edge) => edge.target)
|
|
|
|
for (const neighbor of neighbors) {
|
|
if (!recursionStack.has(neighbor)) {
|
|
if (!visited.has(neighbor)) {
|
|
dfs(neighbor)
|
|
}
|
|
} else {
|
|
// Found a cycle
|
|
const cycleStartIndex = currentPath.indexOf(neighbor)
|
|
if (cycleStartIndex !== -1) {
|
|
const cycle = currentPath.slice(cycleStartIndex)
|
|
// Only add cycles with length > 1
|
|
if (cycle.length > 1) {
|
|
allCycles.push([...cycle])
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
currentPath.pop()
|
|
recursionStack.delete(node)
|
|
}
|
|
|
|
dfs(startNode)
|
|
|
|
return {
|
|
hasCycle: allCycles.length > 0,
|
|
paths: allCycles,
|
|
}
|
|
}
|