- Add full Telegram bot functionality with Z.AI API integration
- Implement 4 tools: Bash, FileEdit, WebSearch, Git
- Add 3 agents: Code Reviewer, Architect, DevOps Engineer
- Add 6 skills for common coding tasks
- Add systemd service file for 24/7 operation
- Add nginx configuration for HTTPS webhook
- Add comprehensive documentation
- Implement WebSocket server for real-time updates
- Add logging system with Winston
- Add environment validation
🤖 zCode CLI X - Agentic coder with Z.AI + Telegram integration
124 lines
5.1 KiB
JavaScript
124 lines
5.1 KiB
JavaScript
import { createInstrumentationScope, createResource, toAttributes, } from '../common/internal';
|
|
// Span flags constants matching the OTLP specification
|
|
const SPAN_FLAGS_CONTEXT_HAS_IS_REMOTE_MASK = 0x100;
|
|
const SPAN_FLAGS_CONTEXT_IS_REMOTE_MASK = 0x200;
|
|
/**
|
|
* Builds the 32-bit span flags value combining the low 8-bit W3C TraceFlags
|
|
* with the HAS_IS_REMOTE and IS_REMOTE bits according to the OTLP spec.
|
|
*/
|
|
function buildSpanFlagsFrom(traceFlags, isRemote) {
|
|
// low 8 bits are W3C TraceFlags (e.g., sampled)
|
|
let flags = (traceFlags & 0xff) | SPAN_FLAGS_CONTEXT_HAS_IS_REMOTE_MASK;
|
|
if (isRemote) {
|
|
flags |= SPAN_FLAGS_CONTEXT_IS_REMOTE_MASK;
|
|
}
|
|
return flags;
|
|
}
|
|
export function sdkSpanToOtlpSpan(span, encoder) {
|
|
const ctx = span.spanContext();
|
|
const status = span.status;
|
|
const parentSpanId = span.parentSpanContext?.spanId
|
|
? encoder.encodeSpanContext(span.parentSpanContext?.spanId)
|
|
: undefined;
|
|
return {
|
|
traceId: encoder.encodeSpanContext(ctx.traceId),
|
|
spanId: encoder.encodeSpanContext(ctx.spanId),
|
|
parentSpanId: parentSpanId,
|
|
traceState: ctx.traceState?.serialize(),
|
|
name: span.name,
|
|
// Span kind is offset by 1 because the API does not define a value for unset
|
|
kind: span.kind == null ? 0 : span.kind + 1,
|
|
startTimeUnixNano: encoder.encodeHrTime(span.startTime),
|
|
endTimeUnixNano: encoder.encodeHrTime(span.endTime),
|
|
attributes: toAttributes(span.attributes, encoder),
|
|
droppedAttributesCount: span.droppedAttributesCount,
|
|
events: span.events.map(event => toOtlpSpanEvent(event, encoder)),
|
|
droppedEventsCount: span.droppedEventsCount,
|
|
status: {
|
|
// API and proto enums share the same values
|
|
code: status.code,
|
|
message: status.message,
|
|
},
|
|
links: span.links.map(link => toOtlpLink(link, encoder)),
|
|
droppedLinksCount: span.droppedLinksCount,
|
|
flags: buildSpanFlagsFrom(ctx.traceFlags, span.parentSpanContext?.isRemote),
|
|
};
|
|
}
|
|
export function toOtlpLink(link, encoder) {
|
|
return {
|
|
attributes: link.attributes ? toAttributes(link.attributes, encoder) : [],
|
|
spanId: encoder.encodeSpanContext(link.context.spanId),
|
|
traceId: encoder.encodeSpanContext(link.context.traceId),
|
|
traceState: link.context.traceState?.serialize(),
|
|
droppedAttributesCount: link.droppedAttributesCount || 0,
|
|
flags: buildSpanFlagsFrom(link.context.traceFlags, link.context.isRemote),
|
|
};
|
|
}
|
|
export function toOtlpSpanEvent(timedEvent, encoder) {
|
|
return {
|
|
attributes: timedEvent.attributes
|
|
? toAttributes(timedEvent.attributes, encoder)
|
|
: [],
|
|
name: timedEvent.name,
|
|
timeUnixNano: encoder.encodeHrTime(timedEvent.time),
|
|
droppedAttributesCount: timedEvent.droppedAttributesCount || 0,
|
|
};
|
|
}
|
|
export function createExportTraceServiceRequest(spans, encoder) {
|
|
return {
|
|
resourceSpans: spanRecordsToResourceSpans(spans, encoder),
|
|
};
|
|
}
|
|
function createResourceMap(readableSpans) {
|
|
const resourceMap = new Map();
|
|
for (const record of readableSpans) {
|
|
let ilsMap = resourceMap.get(record.resource);
|
|
if (!ilsMap) {
|
|
ilsMap = new Map();
|
|
resourceMap.set(record.resource, ilsMap);
|
|
}
|
|
// TODO this is duplicated in basic tracer. Consolidate on a common helper in core
|
|
const instrumentationScopeKey = `${record.instrumentationScope.name}@${record.instrumentationScope.version || ''}:${record.instrumentationScope.schemaUrl || ''}`;
|
|
let records = ilsMap.get(instrumentationScopeKey);
|
|
if (!records) {
|
|
records = [];
|
|
ilsMap.set(instrumentationScopeKey, records);
|
|
}
|
|
records.push(record);
|
|
}
|
|
return resourceMap;
|
|
}
|
|
function spanRecordsToResourceSpans(readableSpans, encoder) {
|
|
const resourceMap = createResourceMap(readableSpans);
|
|
const out = [];
|
|
const entryIterator = resourceMap.entries();
|
|
let entry = entryIterator.next();
|
|
while (!entry.done) {
|
|
const [resource, ilmMap] = entry.value;
|
|
const scopeResourceSpans = [];
|
|
const ilmIterator = ilmMap.values();
|
|
let ilmEntry = ilmIterator.next();
|
|
while (!ilmEntry.done) {
|
|
const scopeSpans = ilmEntry.value;
|
|
if (scopeSpans.length > 0) {
|
|
const spans = scopeSpans.map(readableSpan => sdkSpanToOtlpSpan(readableSpan, encoder));
|
|
scopeResourceSpans.push({
|
|
scope: createInstrumentationScope(scopeSpans[0].instrumentationScope),
|
|
spans: spans,
|
|
schemaUrl: scopeSpans[0].instrumentationScope.schemaUrl,
|
|
});
|
|
}
|
|
ilmEntry = ilmIterator.next();
|
|
}
|
|
const processedResource = createResource(resource, encoder);
|
|
const transformedSpans = {
|
|
resource: processedResource,
|
|
scopeSpans: scopeResourceSpans,
|
|
schemaUrl: processedResource.schemaUrl,
|
|
};
|
|
out.push(transformedSpans);
|
|
entry = entryIterator.next();
|
|
}
|
|
return out;
|
|
}
|
|
//# sourceMappingURL=internal.js.map
|