161 lines
4.4 KiB
TypeScript

/**
* This file is part of the NocoBase (R) project.
* Copyright (c) 2020-2024 NocoBase Co., Ltd.
* Authors: NocoBase Team.
*
* This project is dual-licensed under AGPL-3.0 and NocoBase Commercial License.
* For more information, please refer to: https://www.nocobase.com/agreement.
*/
import cors from '@koa/cors';
import { requestLogger } from '@nocobase/logger';
import { Resourcer } from '@nocobase/resourcer';
import { uid } from '@nocobase/utils';
import { Command } from 'commander';
import { randomUUID } from 'crypto';
import fs from 'fs';
import i18next from 'i18next';
import bodyParser from 'koa-bodyparser';
import { createHistogram, RecordableHistogram } from 'perf_hooks';
import Application, { ApplicationOptions } from './application';
import { dataWrapping } from './middlewares/data-wrapping';
import { i18n } from './middlewares/i18n';
export function createI18n(options: ApplicationOptions) {
const instance = i18next.createInstance();
instance.init({
lng: process.env.INIT_LANG || 'en-US',
resources: {},
keySeparator: false,
nsSeparator: false,
...options.i18n,
});
return instance;
}
export function createResourcer(options: ApplicationOptions) {
return new Resourcer({ ...options.resourcer });
}
export function registerMiddlewares(app: Application, options: ApplicationOptions) {
app.use(
async function generateReqId(ctx, next) {
app.context.reqId = randomUUID();
await next();
},
{ tag: 'generateReqId' },
);
app.use(requestLogger(app.name, app.requestLogger, options.logger?.request), { tag: 'logger' });
app.use(
cors({
exposeHeaders: ['content-disposition'],
...options.cors,
}),
{
tag: 'cors',
after: 'bodyParser',
},
);
if (options.bodyParser !== false) {
const bodyLimit = '10mb';
app.use(
bodyParser({
jsonLimit: bodyLimit,
formLimit: bodyLimit,
textLimit: bodyLimit,
...options.bodyParser,
}),
{
tag: 'bodyParser',
after: 'logger',
},
);
}
app.use(async (ctx, next) => {
ctx.getBearerToken = () => {
const token = ctx.get('Authorization').replace(/^Bearer\s+/gi, '');
return token || ctx.query.token;
};
await next();
});
app.use(i18n, { tag: 'i18n', before: 'cors' });
if (options.dataWrapping !== false) {
app.use(dataWrapping(), { tag: 'dataWrapping', after: 'cors' });
}
app.use(app.dataSourceManager.middleware(), { tag: 'dataSource', after: 'dataWrapping' });
}
export const createAppProxy = (app: Application) => {
return new Proxy(app, {
get(target, prop, ...args) {
if (typeof prop === 'string' && ['on', 'once', 'addListener'].includes(prop)) {
return (eventName: string, listener: any) => {
listener['_reinitializable'] = true;
return target[prop](eventName, listener);
};
}
return Reflect.get(target, prop, ...args);
},
});
};
export const getCommandFullName = (command: Command) => {
const names = [];
names.push(command.name());
let parent = command?.parent;
while (parent) {
if (!parent?.parent) {
break;
}
names.unshift(parent.name());
parent = parent.parent;
}
return names.join('.');
};
/* istanbul ignore next -- @preserve */
export const tsxRerunning = async () => {
await fs.promises.writeFile(process.env.WATCH_FILE, `export const watchId = '${uid()}';`, 'utf-8');
};
/* istanbul ignore next -- @preserve */
export const enablePerfHooks = (app: Application) => {
app.context.getPerfHistogram = (name: string) => {
if (!app.perfHistograms.has(name)) {
app.perfHistograms.set(name, createHistogram());
}
return app.perfHistograms.get(name);
};
app.resourcer.define({
name: 'perf',
actions: {
view: async (ctx, next) => {
const result = {};
const histograms = ctx.app.perfHistograms as Map<string, RecordableHistogram>;
const sortedHistograms = [...histograms.entries()].sort(([i, a], [j, b]) => b.mean - a.mean);
sortedHistograms.forEach(([name, histogram]) => {
result[name] = histogram;
});
ctx.body = result;
await next();
},
reset: async (ctx, next) => {
const histograms = ctx.app.perfHistograms as Map<string, RecordableHistogram>;
histograms.forEach((histogram: RecordableHistogram) => histogram.reset());
await next();
},
},
});
app.acl.allow('perf', '*', 'public');
};