@langchain/core
Version:
Core LangChain.js abstractions and schemas
121 lines (120 loc) • 4.1 kB
JavaScript
import { getDefaultProjectName, } from "langsmith";
import { RunTree } from "langsmith/run_trees";
import { getCurrentRunTree } from "langsmith/singletons/traceable";
import { BaseTracer } from "./base.js";
import { getDefaultLangChainClientSingleton } from "../singletons/tracer.js";
export class LangChainTracer extends BaseTracer {
constructor(fields = {}) {
super(fields);
Object.defineProperty(this, "name", {
enumerable: true,
configurable: true,
writable: true,
value: "langchain_tracer"
});
Object.defineProperty(this, "projectName", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "exampleId", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "client", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "replicas", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "usesRunTreeMap", {
enumerable: true,
configurable: true,
writable: true,
value: true
});
const { exampleId, projectName, client, replicas } = fields;
this.projectName = projectName ?? getDefaultProjectName();
this.replicas = replicas;
this.exampleId = exampleId;
this.client = client ?? getDefaultLangChainClientSingleton();
const traceableTree = LangChainTracer.getTraceableRunTree();
if (traceableTree) {
this.updateFromRunTree(traceableTree);
}
}
async persistRun(_run) { }
async onRunCreate(run) {
const runTree = this.getRunTreeWithTracingConfig(run.id);
await runTree?.postRun();
}
async onRunUpdate(run) {
const runTree = this.getRunTreeWithTracingConfig(run.id);
await runTree?.patchRun();
}
getRun(id) {
return this.runTreeMap.get(id);
}
updateFromRunTree(runTree) {
this.runTreeMap.set(runTree.id, runTree);
let rootRun = runTree;
const visited = new Set();
while (rootRun.parent_run) {
if (visited.has(rootRun.id))
break;
visited.add(rootRun.id);
if (!rootRun.parent_run)
break;
rootRun = rootRun.parent_run;
}
visited.clear();
const queue = [rootRun];
while (queue.length > 0) {
const current = queue.shift();
if (!current || visited.has(current.id))
continue;
visited.add(current.id);
this.runTreeMap.set(current.id, current);
if (current.child_runs) {
queue.push(...current.child_runs);
}
}
this.client = runTree.client ?? this.client;
this.replicas = runTree.replicas ?? this.replicas;
this.projectName = runTree.project_name ?? this.projectName;
this.exampleId = runTree.reference_example_id ?? this.exampleId;
}
getRunTreeWithTracingConfig(id) {
const runTree = this.runTreeMap.get(id);
if (!runTree)
return undefined;
return new RunTree({
...runTree,
client: this.client,
project_name: this.projectName,
replicas: this.replicas,
reference_example_id: this.exampleId,
tracingEnabled: true,
});
}
static getTraceableRunTree() {
try {
return (
// The type cast here provides forward compatibility. Old versions of LangSmith will just
// ignore the permitAbsentRunTree arg.
getCurrentRunTree(true));
}
catch {
return undefined;
}
}
}