@effect-ts/system
Version:
Effect-TS is a zero dependency set of libraries to write highly productive, purely functional TypeScript at scale.
119 lines (105 loc) • 3.62 kB
text/typescript
// ets_tracing: off
import type { Chunk } from "../Collections/Immutable/Chunk/index.js"
import { constant, tuple } from "../Function/index.js"
import * as IT from "../Iterable/index.js"
import { fold_ } from "../Option/index.js"
import { parseMs } from "../Utils/parse-ms.js"
import type { UIO } from "./_internal/effect.js"
import * as T from "./_internal/effect-api.js"
import type { Runtime } from "./core.js"
import { FiberDump } from "./dump.js"
import { fiberName } from "./fiberName.js"
import type { Status } from "./status.js"
export function dump<E, A>(fiber: Runtime<E, A>): T.UIO<FiberDump> {
return T.map_(
T.zipPar_(fiber.getRef(fiberName), fiber.status),
({ tuple: [name, status] }) => FiberDump(fiber.id, name, status)
)
}
export function dumpFibers(fibers: Iterable<Runtime<any, any>>): UIO<Chunk<FiberDump>> {
return T.forEach_(fibers, dump)
}
export function dumpStr(
fibers: Iterable<Runtime<any, any>>,
withTrace: false
): UIO<string> {
const du = T.forEach_(fibers, dump)
const now = T.succeedWith(() => new Date().getTime())
return T.map_(T.zipWith_(du, now, tuple), ([dumps, now]) => {
const tree = renderHierarchy(dumps)
const dumpStrings = withTrace ? collectTraces(dumps, now) : []
return IT.reduce_(dumpStrings, tree, (acc, v) => acc + "\n" + v)
})
}
export function prettyPrintM(dump: FiberDump): UIO<string> {
return T.succeed(prettyPrint(dump, new Date().getTime()))
}
/**
* @internal
*/
export function prettyPrint(dump: FiberDump, now: number): string {
const { days, hours, milliseconds, minutes, seconds } = parseMs(
now - dump.fiberId.startTimeMillis
)
const name = fold_(dump.fiberName, constant(""), (n) => `"${n}" `)
const lifeMsg =
(days === 0 ? "" : `${days}d`) +
(days === 0 && hours === 0 ? "" : `${hours}h`) +
(days === 0 && hours === 0 && minutes === 0 ? "" : `${minutes}m`) +
(days === 0 && hours === 0 && minutes === 0 && seconds === 0 ? "" : `${seconds}s`) +
`${milliseconds}ms`
const waitMsg = (function (status: Status) {
switch (status._tag) {
case "Suspended":
return status.blockingOn.length > 0
? `waiting on ` + status.blockingOn.map((id) => `${id.seqNumber}`).join(", ")
: ""
default:
return ""
}
})(dump.status)
const statMsg = renderStatus(dump.status)
return [
`${name}#${dump.fiberId.seqNumber} (${lifeMsg}) ${waitMsg}`,
` Status: ${statMsg}`
].join("\n")
}
/**
* @internal
*/
export function renderOne(tree: FiberDump): string {
const prefix = ""
const name = fold_(tree.fiberName, constant(""), (n) => '"' + n + '" ')
const statusMsg = renderStatus(tree.status)
return `${prefix}+---${name}#${tree.fiberId.seqNumber} Status: ${statusMsg}\n`
}
/**
* @internal
*/
export function renderStatus(status: Status): string {
switch (status._tag) {
case "Done":
return "Done"
case "Finishing":
return `Finishing(${status.interrupting ? "interrupting" : ""})`
case "Running":
return `Running(${status.interrupting ? "interrupting" : ""})`
case "Suspended": {
const inter = status.interruptible ? "interruptible" : "uninterruptible"
const ep = `${status.epoch} asyncs`
return `Suspended(${inter}, ${ep})`
}
}
}
/**
* @internal
*/
export function renderHierarchy(trees: Iterable<FiberDump>): string {
return IT.reduce_(IT.map_(trees, renderOne), "", (acc, str) => acc + str)
}
export function collectTraces(
dumps: Iterable<FiberDump>,
now: number
): Iterable<string> {
return IT.map_(dumps, (d) => prettyPrint(d, now))
}