Skip to content

Commit

Permalink
fix: interface in serverless-app (#947)
Browse files Browse the repository at this point in the history
  • Loading branch information
czy88840616 authored Mar 27, 2021
1 parent 92e4ee6 commit 0b56648
Show file tree
Hide file tree
Showing 7 changed files with 346 additions and 343 deletions.
333 changes: 333 additions & 0 deletions packages-serverless/serverless-app/src/framework.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,333 @@
import {
IMidwayApplication,
IMidwayBootstrapOptions,
IMidwayContainer,
IMidwayContext,
IMidwayFramework,
MidwayFrameworkType,
} from '@midwayjs/core';
import { Application, IServerlessAppOptions } from './interface';
import { Server } from 'net';
import { StarterMap, TriggerMap } from './platform';
import * as express from 'express';
import { findNpmModule, output404 } from './utils';
import { start2 } from './start';
import * as bodyParser from 'body-parser';
import { getSpecFile, loadSpec } from '@midwayjs/serverless-spec-builder';
import { createExpressGateway } from '@midwayjs/gateway-common-http';

export class Framework
implements IMidwayFramework<Application, IServerlessAppOptions> {
app: Application;
configurationOptions: IServerlessAppOptions;
private innerApp: IMidwayApplication;
private innerFramework: IMidwayFramework<any, any>;
private runtime: any;
private server: Server;
private bootstrapOptions;
private spec;
configure(options: IServerlessAppOptions) {
this.configurationOptions = options;
return this;
}
async stop() {
if (this.server?.close) {
this.server.close();
}
}

getApplicationContext(): IMidwayContainer {
return this.innerApp.getApplicationContext();
}

getConfiguration(key?: string): any {
return this.innerApp.getConfig(key);
}

getCurrentEnvironment(): string {
return this.innerApp.getEnv();
}
getAppDir(): string {
return this.innerApp.getAppDir();
}

getLogger(name?: string): any {
return this.innerApp.getLogger(name);
}
getBaseDir(): string {
return this.innerApp.getBaseDir();
}
getCoreLogger() {
return (this.innerApp as any).coreLogger;
}
createLogger(name: string, options?: any): any {
return this.innerApp.createLogger(name, options);
}
getProjectName(): string {
return this.innerApp.getProjectName();
}
public getDefaultContextLoggerClass() {
return this.innerFramework.getDefaultContextLoggerClass();
}

async applicationInitialize(options: IMidwayBootstrapOptions) {}

public getFrameworkName() {
return 'midway:serverless:app';
}

public getFrameworkType(): MidwayFrameworkType {
return MidwayFrameworkType.SERVERLESS_APP;
}
public getApplication() {
return new Proxy(this.app, {
get: (target, key) => {
if (target[key]) {
return target[key];
}
if (this[key]) {
return this[key];
}
},
});
}

public getServer() {
return this.server;
}

private getStarterName() {
const starter = this.spec?.provider?.starterModule;
if (starter) {
return require.resolve(starter);
}
const platform = this.getPlatform();
const starterModList = StarterMap[platform];
if (!starterModList || !starterModList.length) {
throw new Error(`Current provider '${platform}' not support(no starter)`);
}
for (const mod of starterModList) {
try {
return require.resolve(mod);
} catch {
// continue
}
}
throw new Error(
`Platform starter '${
starterModList[starterModList.length - 1]
}' not found`
);
}

private getTriggerMap() {
const trigger = this.spec?.provider?.triggerModule;
if (trigger) {
return require(trigger);
}
const platform = this.getPlatform();
const triggerModList = TriggerMap[platform];
if (!triggerModList || !triggerModList.length) {
throw new Error(`Current provider '${platform}' not support(no trigger)`);
}
for (const mod of triggerModList) {
try {
return require(mod);
} catch {
// continue
}
}
throw new Error(
`Platform trigger '${
triggerModList[triggerModList.length - 1]
}' not found`
);
}

private async getServerlessInstance<T>(cls: any): Promise<T> {
// 如何传initializeContext
const context: IMidwayContext = await new Promise(resolve => {
this.runtime.asyncEvent(async ctx => {
resolve((this.innerFramework as any).getContext(ctx));
})({}, this.configurationOptions.initContext || {});
});

return context.requestContext.getAsync(cls);
}

private getPlatform() {
const provider = this.spec?.provider?.name;
if (provider) {
if (provider === 'fc' || provider === 'aliyun') {
return 'aliyun';
} else if (provider === 'scf' || provider === 'tencent') {
return 'tencent';
}
}
return provider;
}

async initialize(options: Partial<IMidwayBootstrapOptions>) {
process.env.MIDWAY_SERVER_ENV = process.env.MIDWAY_SERVER_ENV || 'local';
this.bootstrapOptions = options;
this.getFaaSSpec();
this.app = express() as any;
const { appDir, baseDir } = options;

const faasModule = '@midwayjs/faas';
const faasModulePath = findNpmModule(appDir, faasModule);
if (!faasModulePath) {
throw new Error(`Module '${faasModule}' not found`);
}
const starterName = this.getStarterName();
const usageFaaSModule = this.getFaaSModule();

let usageFaasModulePath = faasModulePath;
if (usageFaaSModule !== faasModule) {
usageFaasModulePath = findNpmModule(appDir, usageFaaSModule);
if (!usageFaasModulePath) {
throw new Error(`Module '${usageFaasModulePath}' not found`);
}
}

// 分析项目结构
const currentBaseDir = baseDir;

const triggerMap = this.getTriggerMap();
const layers = this.getLayers();
const { Framework } = require(usageFaasModulePath);
const startResult = await start2({
appDir,
baseDir: currentBaseDir,
framework: Framework,
layers: layers,
starter: require(starterName),
initializeContext: this.configurationOptions?.initContext,
});
this.innerFramework = startResult.framework;
this.runtime = startResult.runtime;
this.innerApp = startResult.framework.getApplication();
const invoke = startResult.invoke;
const httpFuncSpec = await startResult.getFunctionsFromDecorator();
if (!this.spec.functions) {
this.spec.functions = {};
}
Object.assign(this.spec.functions, httpFuncSpec);
this.app.getServerlessInstance = this.getServerlessInstance.bind(this);
this.app.use(bodyParser.urlencoded({ extended: false }));
this.app.use(bodyParser.json());
this.app.use((req, res, next) => {
const gateway = createExpressGateway({
functionDir: appDir,
});
gateway.transform(req, res, next, async () => {
return {
functionList: this.spec.functions,
invoke: async args => {
const trigger = [new triggerMap.http(...args.data)];
let newArgs = trigger;
let callBackTrigger;
if (newArgs?.[0] && typeof newArgs[0].toArgs === 'function') {
callBackTrigger = trigger[0];
newArgs = await trigger[0].toArgs();
}
const result = await new Promise((resolve, reject) => {
if (callBackTrigger?.useCallback) {
// 这个地方 callback 得调用 resolve
const cb = callBackTrigger.createCallback((err, result) => {
if (err) {
return reject(err);
}
return resolve(result);
});
newArgs.push(cb);
}
Promise.resolve(invoke(args.functionHandler, newArgs)).then(
resolve,
reject
);
});
if (callBackTrigger?.close) {
await callBackTrigger.close();
}
return result;
},
};
});
});

this.app.use((req, res) => {
res.statusCode = 404;
res.send(output404(req.path, this.spec.functions));
});

if (process.env.IN_CHILD_PROCESS) {
this.listenMessage();
}
}

protected getFaaSModule() {
return process.env.DEV_MIDWAY_FAAS_MODULE || '@midwayjs/faas';
}

protected getFaasStarterName() {
return 'FaaSStarter';
}

private getFaaSSpec() {
const { appDir } = this.bootstrapOptions;
const specFileInfo = getSpecFile(appDir);
this.spec = loadSpec(appDir, specFileInfo);
}

public async run() {
if (this.configurationOptions.port) {
this.server = require('http').createServer(this.app);
await new Promise<void>(resolve => {
this.server.listen(this.configurationOptions.port, () => {
resolve();
});
});
}
}

private listenMessage() {
process.on('message', async msg => {
if (!msg || !msg.type) {
return;
}
const type = msg.type;
let data;
switch (type) {
case 'functions':
data = this.spec.functions;
break;
}
process.send({ type: 'dev:' + type, data, id: msg.id });
});
}

private getLayers() {
const specLayers = [];
if (this.configurationOptions.layers) {
this.configurationOptions.layers.forEach(path => {
const layer = require(path);
specLayers.push(layer);
});
}
if (this.spec?.layers) {
Object.keys(this.spec.layers).forEach(layerName => {
const info = this.spec.layers[layerName];
if (!info?.path) {
return;
}
const [type, path] = info.path.split(':');
if (type === 'npm') {
const layer = require(path);
specLayers.push(layer);
}
});
}
return specLayers;
}
}
Loading

0 comments on commit 0b56648

Please sign in to comment.