refactor(server): simplify metrics creation and usage (#5115)

This commit is contained in:
liuyi 2023-11-29 08:05:08 +00:00
parent 7a7cbc45d7
commit 89f267a3fe
No known key found for this signature in database
GPG Key ID: 56709255DC7EC728
13 changed files with 341 additions and 90 deletions

View File

@ -48,6 +48,7 @@
"@opentelemetry/instrumentation-ioredis": "^0.35.3", "@opentelemetry/instrumentation-ioredis": "^0.35.3",
"@opentelemetry/instrumentation-nestjs-core": "^0.33.3", "@opentelemetry/instrumentation-nestjs-core": "^0.33.3",
"@opentelemetry/instrumentation-socket.io": "^0.34.3", "@opentelemetry/instrumentation-socket.io": "^0.34.3",
"@opentelemetry/resources": "^1.18.1",
"@opentelemetry/sdk-metrics": "^1.18.1", "@opentelemetry/sdk-metrics": "^1.18.1",
"@opentelemetry/sdk-node": "^0.45.1", "@opentelemetry/sdk-node": "^0.45.1",
"@opentelemetry/sdk-trace-node": "^1.18.1", "@opentelemetry/sdk-trace-node": "^1.18.1",

View File

@ -20,7 +20,7 @@ export class GQLLoggerPlugin implements ApolloServerPlugin {
const res = reqContext.contextValue.req.res as Response; const res = reqContext.contextValue.req.res as Response;
const operation = reqContext.request.operationName; const operation = reqContext.request.operationName;
metrics().gqlRequest.add(1, { operation }); metrics.gql.counter('query_counter').add(1, { operation });
const start = Date.now(); const start = Date.now();
return Promise.resolve({ return Promise.resolve({
@ -30,7 +30,9 @@ export class GQLLoggerPlugin implements ApolloServerPlugin {
'Server-Timing', 'Server-Timing',
`gql;dur=${costInMilliseconds};desc="GraphQL"` `gql;dur=${costInMilliseconds};desc="GraphQL"`
); );
metrics().gqlTimer.record(costInMilliseconds, { operation }); metrics.gql
.histogram('query_duration')
.record(costInMilliseconds, { operation });
return Promise.resolve(); return Promise.resolve();
}, },
didEncounterErrors: () => { didEncounterErrors: () => {
@ -39,7 +41,9 @@ export class GQLLoggerPlugin implements ApolloServerPlugin {
'Server-Timing', 'Server-Timing',
`gql;dur=${costInMilliseconds};desc="GraphQL ${operation}"` `gql;dur=${costInMilliseconds};desc="GraphQL ${operation}"`
); );
metrics().gqlTimer.record(costInMilliseconds, { operation }); metrics.gql
.histogram('query_duration')
.record(costInMilliseconds, { operation });
return Promise.resolve(); return Promise.resolve();
}, },
}); });

View File

@ -1,76 +1,129 @@
import opentelemetry, { Attributes, Observable } from '@opentelemetry/api'; import {
Attributes,
Counter,
Histogram,
Meter,
MetricOptions,
} from '@opentelemetry/api';
interface AsyncMetric { import { getMeter } from './opentelemetry';
ob: Observable;
get value(): any;
get attrs(): Attributes | undefined;
}
let _metrics: ReturnType<typeof createBusinessMetrics> | undefined = undefined; type MetricType = 'counter' | 'gauge' | 'histogram';
type Metric<T extends MetricType> = T extends 'counter'
? Counter
: T extends 'gauge'
? Histogram
: T extends 'histogram'
? Histogram
: never;
export function getMeter(name = 'business') { export type ScopedMetrics = {
return opentelemetry.metrics.getMeter(name); [T in MetricType]: (name: string, opts?: MetricOptions) => Metric<T>;
} };
type MetricCreators = {
[T in MetricType]: (
meter: Meter,
name: string,
opts?: MetricOptions
) => Metric<T>;
};
function createBusinessMetrics() { export type KnownMetricScopes =
const meter = getMeter(); | 'socketio'
const asyncMetrics: AsyncMetric[] = []; | 'gql'
| 'jwst'
| 'auth'
| 'controllers'
| 'doc';
function createGauge(name: string) { const metricCreators: MetricCreators = {
counter(meter: Meter, name: string, opts?: MetricOptions) {
return meter.createCounter(name, opts);
},
gauge(meter: Meter, name: string, opts?: MetricOptions) {
let value: any; let value: any;
let attrs: Attributes | undefined; let attrs: Attributes | undefined;
const ob = meter.createObservableGauge(name); const ob = meter.createObservableGauge(name, opts);
asyncMetrics.push({
ob, ob.addCallback(result => {
get value() { result.observe(value, attrs);
return value;
},
get attrs() {
return attrs;
},
}); });
return (newValue: any, newAttrs?: Attributes) => { return {
value = newValue; record: (newValue, newAttrs) => {
attrs = newAttrs; value = newValue;
}; attrs = newAttrs;
},
} satisfies Histogram;
},
histogram(meter: Meter, name: string, opts?: MetricOptions) {
return meter.createHistogram(name, opts);
},
};
const scopes = new Map<string, ScopedMetrics>();
function make(scope: string) {
const meter = getMeter();
const metrics = new Map<string, { type: MetricType; metric: any }>();
const prefix = scope + '/';
function getOrCreate<T extends MetricType>(
type: T,
name: string,
opts?: MetricOptions
): Metric<T> {
name = prefix + name;
const metric = metrics.get(name);
if (metric) {
if (type !== metric.type) {
throw new Error(
`Metric ${name} has already been registered as ${metric.type} mode, but get as ${type} again.`
);
}
return metric.metric;
} else {
const metric = metricCreators[type](meter, name, opts);
metrics.set(name, { type, metric });
return metric;
}
} }
const metrics = { return {
socketIOConnectionGauge: createGauge('socket_io_connection'), counter(name, opts) {
return getOrCreate('counter', name, opts);
gqlRequest: meter.createCounter('gql_request'),
gqlError: meter.createCounter('gql_error'),
gqlTimer: meter.createHistogram('gql_timer'),
jwstCodecMerge: meter.createCounter('jwst_codec_merge'),
jwstCodecDidnotMatch: meter.createCounter('jwst_codec_didnot_match'),
jwstCodecFail: meter.createCounter('jwst_codec_fail'),
authCounter: meter.createCounter('auth'),
authFailCounter: meter.createCounter('auth_fail'),
docHistoryCounter: meter.createCounter('doc_history_created'),
docRecoverCounter: meter.createCounter('doc_history_recovered'),
};
meter.addBatchObservableCallback(
result => {
asyncMetrics.forEach(metric => {
result.observe(metric.ob, metric.value, metric.attrs);
});
}, },
asyncMetrics.map(({ ob }) => ob) gauge(name, opts) {
); return getOrCreate('gauge', name, opts);
},
return metrics; histogram(name, opts) {
return getOrCreate('histogram', name, opts);
},
} satisfies ScopedMetrics;
} }
export function registerBusinessMetrics() { /**
if (!_metrics) { * @example
_metrics = createBusinessMetrics(); *
* ```
* metrics.scope.counter('example_count').add(1, {
* attr1: 'example-event'
* })
* ```
*/
export const metrics = new Proxy<Record<KnownMetricScopes, ScopedMetrics>>(
// @ts-expect-error proxied
{},
{
get(_, scopeName: string) {
let scope = scopes.get(scopeName);
if (!scope) {
scope = make(scopeName);
scopes.set(scopeName, scope);
}
return scope;
},
} }
);
return _metrics;
}
export const metrics = registerBusinessMetrics;

View File

@ -1,5 +1,6 @@
import { MetricExporter } from '@google-cloud/opentelemetry-cloud-monitoring-exporter'; import { MetricExporter } from '@google-cloud/opentelemetry-cloud-monitoring-exporter';
import { TraceExporter } from '@google-cloud/opentelemetry-cloud-trace-exporter'; import { TraceExporter } from '@google-cloud/opentelemetry-cloud-trace-exporter';
import { metrics } from '@opentelemetry/api';
import { import {
CompositePropagator, CompositePropagator,
W3CBaggagePropagator, W3CBaggagePropagator,
@ -16,6 +17,8 @@ import { NestInstrumentation } from '@opentelemetry/instrumentation-nestjs-core'
import { SocketIoInstrumentation } from '@opentelemetry/instrumentation-socket.io'; import { SocketIoInstrumentation } from '@opentelemetry/instrumentation-socket.io';
import { import {
ConsoleMetricExporter, ConsoleMetricExporter,
type MeterProvider,
MetricProducer,
MetricReader, MetricReader,
PeriodicExportingMetricReader, PeriodicExportingMetricReader,
} from '@opentelemetry/sdk-metrics'; } from '@opentelemetry/sdk-metrics';
@ -24,10 +27,11 @@ import {
BatchSpanProcessor, BatchSpanProcessor,
ConsoleSpanExporter, ConsoleSpanExporter,
SpanExporter, SpanExporter,
TraceIdRatioBasedSampler,
} from '@opentelemetry/sdk-trace-node'; } from '@opentelemetry/sdk-trace-node';
import { PrismaInstrumentation } from '@prisma/instrumentation'; import { PrismaInstrumentation } from '@prisma/instrumentation';
import { registerBusinessMetrics } from './metrics'; import { PrismaMetricProducer } from './prisma';
abstract class OpentelemetryFactor { abstract class OpentelemetryFactor {
abstract getMetricReader(): MetricReader; abstract getMetricReader(): MetricReader;
@ -44,9 +48,14 @@ abstract class OpentelemetryFactor {
]; ];
} }
getMetricsProducers(): MetricProducer[] {
return [new PrismaMetricProducer()];
}
create() { create() {
const traceExporter = this.getSpanExporter(); const traceExporter = this.getSpanExporter();
return new NodeSDK({ return new NodeSDK({
sampler: new TraceIdRatioBasedSampler(0.1),
traceExporter, traceExporter,
metricReader: this.getMetricReader(), metricReader: this.getMetricReader(),
spanProcessor: new BatchSpanProcessor(traceExporter), spanProcessor: new BatchSpanProcessor(traceExporter),
@ -67,7 +76,10 @@ class GCloudOpentelemetryFactor extends OpentelemetryFactor {
return new PeriodicExportingMetricReader({ return new PeriodicExportingMetricReader({
exportIntervalMillis: 30000, exportIntervalMillis: 30000,
exportTimeoutMillis: 10000, exportTimeoutMillis: 10000,
exporter: new MetricExporter(), exporter: new MetricExporter({
prefix: 'custom.googleapis.com',
}),
metricProducers: this.getMetricsProducers(),
}); });
} }
@ -78,7 +90,9 @@ class GCloudOpentelemetryFactor extends OpentelemetryFactor {
class LocalOpentelemetryFactor extends OpentelemetryFactor { class LocalOpentelemetryFactor extends OpentelemetryFactor {
override getMetricReader(): MetricReader { override getMetricReader(): MetricReader {
return new PrometheusExporter(); return new PrometheusExporter({
metricProducers: this.getMetricsProducers(),
});
} }
override getSpanExporter(): SpanExporter { override getSpanExporter(): SpanExporter {
@ -90,6 +104,7 @@ class DebugOpentelemetryFactor extends OpentelemetryFactor {
override getMetricReader(): MetricReader { override getMetricReader(): MetricReader {
return new PeriodicExportingMetricReader({ return new PeriodicExportingMetricReader({
exporter: new ConsoleMetricExporter(), exporter: new ConsoleMetricExporter(),
metricProducers: this.getMetricsProducers(),
}); });
} }
@ -111,9 +126,30 @@ function createSDK() {
return factor?.create(); return factor?.create();
} }
let OPENTELEMETRY_STARTED = false;
function ensureStarted() {
if (!OPENTELEMETRY_STARTED) {
OPENTELEMETRY_STARTED = true;
start();
}
}
function getMeterProvider() {
ensureStarted();
return metrics.getMeterProvider();
}
function registerCustomMetrics() { function registerCustomMetrics() {
const host = new HostMetrics({ name: 'instance-host-metrics' }); const hostMetricsMonitoring = new HostMetrics({
host.start(); name: 'instance-host-metrics',
meterProvider: getMeterProvider() as MeterProvider,
});
hostMetricsMonitoring.start();
}
export function getMeter(name = 'business') {
return getMeterProvider().getMeter(name);
} }
export function start() { export function start() {
@ -122,6 +158,5 @@ export function start() {
if (sdk) { if (sdk) {
sdk.start(); sdk.start();
registerCustomMetrics(); registerCustomMetrics();
registerBusinessMetrics();
} }
} }

View File

@ -0,0 +1,132 @@
import { HrTime, ValueType } from '@opentelemetry/api';
import { hrTime } from '@opentelemetry/core';
import { Resource } from '@opentelemetry/resources';
import {
AggregationTemporality,
CollectionResult,
DataPointType,
InstrumentType,
MetricProducer,
ScopeMetrics,
} from '@opentelemetry/sdk-metrics';
import { PrismaService } from '../prisma';
function transformPrismaKey(key: string) {
// replace first '_' to '/' as a scope prefix
// example: prisma_client_query_duration_seconds_sum -> prisma/client_query_duration_seconds_sum
return key.replace(/_/, '/');
}
export class PrismaMetricProducer implements MetricProducer {
private readonly startTime: HrTime = hrTime();
async collect(): Promise<CollectionResult> {
const result: CollectionResult = {
resourceMetrics: {
resource: Resource.EMPTY,
scopeMetrics: [],
},
errors: [],
};
if (!PrismaService.INSTANCE) {
return result;
}
const prisma = PrismaService.INSTANCE;
const endTime = hrTime();
const metrics = await prisma.$metrics.json();
const scopeMetrics: ScopeMetrics = {
scope: {
name: '',
},
metrics: [],
};
for (const counter of metrics.counters) {
scopeMetrics.metrics.push({
descriptor: {
name: transformPrismaKey(counter.key),
description: counter.description,
unit: '1',
type: InstrumentType.COUNTER,
valueType: ValueType.INT,
},
dataPointType: DataPointType.SUM,
aggregationTemporality: AggregationTemporality.CUMULATIVE,
dataPoints: [
{
startTime: this.startTime,
endTime: endTime,
value: counter.value,
attributes: counter.labels,
},
],
isMonotonic: true,
});
}
for (const gauge of metrics.gauges) {
scopeMetrics.metrics.push({
descriptor: {
name: transformPrismaKey(gauge.key),
description: gauge.description,
unit: '1',
type: InstrumentType.UP_DOWN_COUNTER,
valueType: ValueType.INT,
},
dataPointType: DataPointType.GAUGE,
aggregationTemporality: AggregationTemporality.CUMULATIVE,
dataPoints: [
{
startTime: this.startTime,
endTime: endTime,
value: gauge.value,
attributes: gauge.labels,
},
],
});
}
for (const histogram of metrics.histograms) {
const boundaries = [];
const counts = [];
for (const [boundary, count] of histogram.value.buckets) {
boundaries.push(boundary);
counts.push(count);
}
scopeMetrics.metrics.push({
descriptor: {
name: transformPrismaKey(histogram.key),
description: histogram.description,
unit: 'ms',
type: InstrumentType.HISTOGRAM,
valueType: ValueType.DOUBLE,
},
dataPointType: DataPointType.HISTOGRAM,
aggregationTemporality: AggregationTemporality.CUMULATIVE,
dataPoints: [
{
startTime: this.startTime,
endTime: endTime,
value: {
buckets: {
boundaries,
counts,
},
count: histogram.value.count,
sum: histogram.value.sum,
},
attributes: histogram.labels,
},
],
});
}
result.resourceMetrics.scopeMetrics.push(scopeMetrics);
return result;
}
}

View File

@ -1,8 +1,9 @@
import { Attributes } from '@opentelemetry/api'; import { Attributes } from '@opentelemetry/api';
import { getMeter } from './metrics'; import { KnownMetricScopes, metrics } from './metrics';
export const CallTimer = ( export const CallTimer = (
scope: KnownMetricScopes,
name: string, name: string,
attrs?: Attributes attrs?: Attributes
): MethodDecorator => { ): MethodDecorator => {
@ -18,9 +19,11 @@ export const CallTimer = (
} }
desc.value = function (...args: any[]) { desc.value = function (...args: any[]) {
const timer = getMeter().createHistogram(name, { const timer = metrics[scope].histogram(name, {
description: `function call time costs of ${name}`, description: `function call time costs of ${name}`,
unit: 'ms',
}); });
const start = Date.now(); const start = Date.now();
const end = () => { const end = () => {
@ -48,6 +51,7 @@ export const CallTimer = (
}; };
export const CallCounter = ( export const CallCounter = (
scope: KnownMetricScopes,
name: string, name: string,
attrs?: Attributes attrs?: Attributes
): MethodDecorator => { ): MethodDecorator => {
@ -63,7 +67,7 @@ export const CallCounter = (
} }
desc.value = function (...args: any[]) { desc.value = function (...args: any[]) {
const count = getMeter().createCounter(name, { const count = metrics[scope].counter(name, {
description: `function call counter of ${name}`, description: `function call counter of ${name}`,
}); });

View File

@ -89,12 +89,13 @@ export class NextAuthController {
res.redirect(`/signin${query}`); res.redirect(`/signin${query}`);
return; return;
} }
metrics().authCounter.add(1);
const [action, providerId] = req.url // start with request url const [action, providerId] = req.url // start with request url
.slice(BASE_URL.length) // make relative to baseUrl .slice(BASE_URL.length) // make relative to baseUrl
.replace(/\?.*/, '') // remove query part, use only path part .replace(/\?.*/, '') // remove query part, use only path part
.split('/') as [AuthAction, string]; // as array of strings; .split('/') as [AuthAction, string]; // as array of strings;
metrics.auth.counter('call_counter').add(1, { action, providerId });
const credentialsSignIn = const credentialsSignIn =
req.method === 'POST' && providerId === 'credentials'; req.method === 'POST' && providerId === 'credentials';
let userId: string | undefined; let userId: string | undefined;
@ -126,7 +127,9 @@ export class NextAuthController {
const options = this.nextAuthOptions; const options = this.nextAuthOptions;
if (req.method === 'POST' && action === 'session') { if (req.method === 'POST' && action === 'session') {
if (typeof req.body !== 'object' || typeof req.body.data !== 'object') { if (typeof req.body !== 'object' || typeof req.body.data !== 'object') {
metrics().authFailCounter.add(1, { reason: 'invalid_session_data' }); metrics.auth
.counter('call_fails_counter')
.add(1, { reason: 'invalid_session_data' });
throw new BadRequestException(`Invalid new session data`); throw new BadRequestException(`Invalid new session data`);
} }
const user = await this.updateSession(req, req.body.data); const user = await this.updateSession(req, req.body.data);
@ -209,9 +212,10 @@ export class NextAuthController {
if (redirect?.endsWith('api/auth/error?error=AccessDenied')) { if (redirect?.endsWith('api/auth/error?error=AccessDenied')) {
this.logger.log(`Early access redirect headers: ${req.headers}`); this.logger.log(`Early access redirect headers: ${req.headers}`);
metrics().authFailCounter.add(1, { metrics.auth
reason: 'no_early_access_permission', .counter('call_fails_counter')
}); .add(1, { reason: 'no_early_access_permission' });
if ( if (
!req.headers?.referer || !req.headers?.referer ||
checkUrlOrigin(req.headers.referer, 'https://accounts.google.com') checkUrlOrigin(req.headers.referer, 'https://accounts.google.com')

View File

@ -68,7 +68,11 @@ export class DocHistoryManager {
// safe to ignore // safe to ignore
// only happens when duplicated history record created in multi processes // only happens when duplicated history record created in multi processes
}); });
metrics().docHistoryCounter.add(1, {}); metrics.doc
.counter('history_created_counter', {
description: 'How many times the snapshot history created',
})
.add(1);
this.logger.log( this.logger.log(
`History created for ${snapshot.id} in workspace ${snapshot.workspaceId}.` `History created for ${snapshot.id} in workspace ${snapshot.workspaceId}.`
); );
@ -182,7 +186,11 @@ export class DocHistoryManager {
// which is not the solution in CRDT. // which is not the solution in CRDT.
// let user revert in client and update the data in sync system // let user revert in client and update the data in sync system
// `await this.db.snapshot.update();` // `await this.db.snapshot.update();`
metrics().docRecoverCounter.add(1, {}); metrics.doc
.counter('history_recovered_counter', {
description: 'How many times history recovered request happened',
})
.add(1);
return history.timestamp; return history.timestamp;
} }

View File

@ -125,13 +125,13 @@ export class DocManager implements OnModuleInit, OnModuleDestroy {
this.config.doc.manager.experimentalMergeWithJwstCodec && this.config.doc.manager.experimentalMergeWithJwstCodec &&
updates.length < 100 /* avoid overloading */ updates.length < 100 /* avoid overloading */
) { ) {
metrics().jwstCodecMerge.add(1); metrics.jwst.counter('codec_merge_counter').add(1);
const yjsResult = Buffer.from(encodeStateAsUpdate(doc)); const yjsResult = Buffer.from(encodeStateAsUpdate(doc));
let log = false; let log = false;
try { try {
const jwstResult = jwstMergeUpdates(updates); const jwstResult = jwstMergeUpdates(updates);
if (!compare(yjsResult, jwstResult)) { if (!compare(yjsResult, jwstResult)) {
metrics().jwstCodecDidnotMatch.add(1); metrics.jwst.counter('codec_not_match').add(1);
this.logger.warn( this.logger.warn(
`jwst codec result doesn't match yjs codec result for: ${guid}` `jwst codec result doesn't match yjs codec result for: ${guid}`
); );
@ -142,7 +142,7 @@ export class DocManager implements OnModuleInit, OnModuleDestroy {
} }
} }
} catch (e) { } catch (e) {
metrics().jwstCodecFail.add(1); metrics.jwst.counter('codec_fails_counter').add(1);
this.logger.warn(`jwst apply update failed for ${guid}: ${e}`); this.logger.warn(`jwst apply update failed for ${guid}: ${e}`);
log = true; log = true;
} finally { } finally {

View File

@ -45,6 +45,7 @@ export const GatewayErrorWrapper = (): MethodDecorator => {
try { try {
result = originalMethod.apply(this, args); result = originalMethod.apply(this, args);
} catch (e) { } catch (e) {
metrics.socketio.counter('unhandled_errors').add(1);
return { return {
error: new InternalError(e as Error), error: new InternalError(e as Error),
}; };
@ -52,6 +53,7 @@ export const GatewayErrorWrapper = (): MethodDecorator => {
if (result instanceof Promise) { if (result instanceof Promise) {
return result.catch(e => { return result.catch(e => {
metrics.socketio.counter('unhandled_errors').add(1);
return { return {
error: new InternalError(e), error: new InternalError(e),
}; };
@ -68,7 +70,7 @@ export const GatewayErrorWrapper = (): MethodDecorator => {
const SubscribeMessage = (event: string) => const SubscribeMessage = (event: string) =>
applyDecorators( applyDecorators(
GatewayErrorWrapper(), GatewayErrorWrapper(),
CallTimer('socket_io_event_duration', { event }), CallTimer('socketio', 'event_duration', { event }),
RawSubscribeMessage(event) RawSubscribeMessage(event)
); );
@ -104,12 +106,12 @@ export class EventsGateway implements OnGatewayConnection, OnGatewayDisconnect {
handleConnection() { handleConnection() {
this.connectionCount++; this.connectionCount++;
metrics().socketIOConnectionGauge(this.connectionCount); metrics.socketio.gauge('realtime_connections').record(this.connectionCount);
} }
handleDisconnect() { handleDisconnect() {
this.connectionCount--; this.connectionCount--;
metrics().socketIOConnectionGauge(this.connectionCount); metrics.socketio.gauge('realtime_connections').record(this.connectionCount);
} }
@Auth() @Auth()

View File

@ -34,7 +34,7 @@ export class WorkspacesController {
// //
// NOTE: because graphql can't represent a File, so we have to use REST API to get blob // NOTE: because graphql can't represent a File, so we have to use REST API to get blob
@Get('/:id/blobs/:name') @Get('/:id/blobs/:name')
@CallTimer('doc_controller', { method: 'get_blob' }) @CallTimer('controllers', 'workspace_get_blob')
async blob( async blob(
@Param('id') workspaceId: string, @Param('id') workspaceId: string,
@Param('name') name: string, @Param('name') name: string,
@ -59,7 +59,7 @@ export class WorkspacesController {
@Get('/:id/docs/:guid') @Get('/:id/docs/:guid')
@Auth() @Auth()
@Publicable() @Publicable()
@CallTimer('doc_controller', { method: 'get_doc' }) @CallTimer('controllers', 'workspace_get_doc')
async doc( async doc(
@CurrentUser() user: UserType | undefined, @CurrentUser() user: UserType | undefined,
@Param('id') ws: string, @Param('id') ws: string,
@ -106,7 +106,7 @@ export class WorkspacesController {
@Get('/:id/docs/:guid/histories/:timestamp') @Get('/:id/docs/:guid/histories/:timestamp')
@Auth() @Auth()
@CallTimer('doc_controller', { method: 'get_history' }) @CallTimer('controllers', 'workspace_get_history')
async history( async history(
@CurrentUser() user: UserType, @CurrentUser() user: UserType,
@Param('id') ws: string, @Param('id') ws: string,

View File

@ -7,6 +7,13 @@ export class PrismaService
extends PrismaClient extends PrismaClient
implements OnModuleInit, OnModuleDestroy implements OnModuleInit, OnModuleDestroy
{ {
static INSTANCE: PrismaService | null = null;
constructor() {
super();
PrismaService.INSTANCE = this;
}
async onModuleInit() { async onModuleInit() {
await this.$connect(); await this.$connect();
} }

View File

@ -734,6 +734,7 @@ __metadata:
"@opentelemetry/instrumentation-ioredis": "npm:^0.35.3" "@opentelemetry/instrumentation-ioredis": "npm:^0.35.3"
"@opentelemetry/instrumentation-nestjs-core": "npm:^0.33.3" "@opentelemetry/instrumentation-nestjs-core": "npm:^0.33.3"
"@opentelemetry/instrumentation-socket.io": "npm:^0.34.3" "@opentelemetry/instrumentation-socket.io": "npm:^0.34.3"
"@opentelemetry/resources": "npm:^1.18.1"
"@opentelemetry/sdk-metrics": "npm:^1.18.1" "@opentelemetry/sdk-metrics": "npm:^1.18.1"
"@opentelemetry/sdk-node": "npm:^0.45.1" "@opentelemetry/sdk-node": "npm:^0.45.1"
"@opentelemetry/sdk-trace-node": "npm:^1.18.1" "@opentelemetry/sdk-trace-node": "npm:^1.18.1"
@ -9017,7 +9018,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@opentelemetry/resources@npm:1.18.1": "@opentelemetry/resources@npm:1.18.1, @opentelemetry/resources@npm:^1.18.1":
version: 1.18.1 version: 1.18.1
resolution: "@opentelemetry/resources@npm:1.18.1" resolution: "@opentelemetry/resources@npm:1.18.1"
dependencies: dependencies: