mirror of
https://gitee.com/nocobase/nocobase.git
synced 2025-05-05 21:49:25 +08:00
* chore: upgrade koa and koa/cors * fix: cors breaking change * fix: header * fix: origin header
169 lines
4.7 KiB
TypeScript
169 lines
4.7 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 { extractClientIp } from './middlewares/extract-client-ip';
|
|
|
|
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(app.auditManager.middleware(), { tag: 'audit', after: 'generateReqId' });
|
|
|
|
app.use(requestLogger(app.name, app.requestLogger, options.logger?.request), { tag: 'logger' });
|
|
|
|
app.use(
|
|
cors({
|
|
exposeHeaders: ['content-disposition'],
|
|
origin(ctx) {
|
|
return ctx.get('origin');
|
|
},
|
|
...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 function getBearerToken(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' });
|
|
|
|
app.use(extractClientIp(), { tag: 'extractClientIp', before: 'cors' });
|
|
}
|
|
|
|
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');
|
|
};
|