一个基于 Node.js vm 实现的个人 serverless 服务
发布于 5 年前 作者 lqs469 6269 次浏览 来自 分享

Github

基于 Node.js vm 实现个人 serverless 服务。

使用 MidwayTS 搭建。跟其他大型 FaaS 服务一样,你可以在上面运行、开发和管理你的 serverless 函数,而无需考虑构建和部署基础框架,也不用写任何框架相关的代码,只专注于业务。

动机

我想搭建一些简单的个人助理服务,例如天气提示,新闻推送或单纯提醒我不要错过比赛直播。 而这些小需求并没有必要用完整的框架来搭建几个复杂完整的应用程序来解决。 而 serverless 显然很合适。 所以,我做了这个能满足我需求且简易,灵活的 serverless 服务。

Quick Start

$ npm i
$ npm run dev
$ open http://localhost:7001/

Example

你可以在 /src/fns/ 下找到所有例子。每个函数都使用 main() 作为统一的入口,当你请求接口 /vm/[functionName] 时,会触发对应函数中的 main() 方法,同时接口将返回 main()return 的内容。

1. Hello world

// function_a
async function main() {
  return 'Hello world!';
}

GET //127.0.0.1:7001/vm/function_a

Hello world

2. overtime

函数默认请求超时时间为 1.5s,如果超时,系统会自动抛出异常中止请求,而服务并不会受到影响。

// function_b
function main(ctx) {
  while (1) {}

  return 'Hello world!!';
}

GET //127.0.0.1:7001/vm/function_b

{
  "error": "Error: Script execution timed out after 1500ms at Script..."
}

3. 异步

main 函数可以是一个异步函数,通过使用 anycawait 实现。在函数中会暴露一个全局变量 ctx,这个全局变量就是 egg.js 中的ctx,通过它你就可以和平时一样调用 egg.js 提供的方法来快速实现逻辑了,比如下面例子中的 ctx.curl。更多文档可以查看egg.js contextegg.jskoa.js.

Fetch an URL:

// function_c
async function main() {
  const { location } = ctx.query;

  const URL = `https://bing.com`;
  const data = await ctx.curl(URL, {
    dataType: 'json',
  });

  return { data } || {};
}

Sleep 1s:

// function_d
async function main() {
  const start = new Date().toLocaleTimeString();

  const end = await new Promise(resolve => {
    setTimeout(() => {
      resolve(new Date().toLocaleTimeString());
    }, 1000 * 1);
  });

  return {
    start,
    end,
  };
}

4. 做一个 Github 趋势 服务

// github_trending
async function main() {
  const url = 'https://github-trending-api.now.sh/repositories';

  const res = await ctx.curl(
    'https://github-trending-api.now.sh/repositories',
    { dataType: 'json' },
  );

  return res.data.map(item => ({
    title: `${item.name} | 👨‍💻${item.author} | ⭐️${item.stars} | ${item.language}`,
    url: item.url,
    desc: item.description,
  }));
}

GET //127.0.0.1:7001/vm/github_trending

[
  {
    "title": "...",
    "url": "...",
    "desc": "..."
  },
  ...
]

5. 做一个实时 Hacker news 服务

async function main() {
  const tops =
    (await ctx.curl(
      'https://hacker-news.firebaseio.com/v0/topstories.json?print=pretty',
      {
        dataType: 'json',
      },
    )) || [];

  let hnList = [];

  await Promise.all(
    tops.data
      .slice(0, 10)
      .map(item =>
        ctx.curl(
          `https://hacker-news.firebaseio.com/v0/item/${item}.json?print=pretty`,
          { dataType: 'json' },
        ),
      ),
  ).then(data => {
    hnList = data.map(item => ({
      title: item.data.title,
      url:
        item.data.url || `https://news.ycombinator.com/item?id=${item.data.id}`,
      info:
        `${item.data.score} points By ${item.data.by}` +
        `${new Date(item.data.time * 1000).toLocaleString()} | ` +
        `${item.data.descendants} comments`,
    }));
  });

  return hnList;
}

6. 通过地理位置查询今日天气

你可以给函数加入参数,方法时通过请求 URL 的 query,然后在函数中通过 ctx.query 取到。比下面的例子可以请求://127.0.0.1:7001/vm/weather?location=Tokyo

// weather.js

async function main() {
  const { location = 'New York' } = ctx.query;
  const url = `http://api.weatherstack.com/current?access_key=95f5ee664befefc1c49fa0dac0da19c7&query=${location}`;

  const res = await ctx.curl(url, { dataType: 'json' });

  return res.data;
}

GET //127.0.0.1:7001/vm/weather?location=Tokyo

FAQ

这个应用能用来干嘛 ?

它用来运行一些个人服务会非常方便,例如查看天气,推送最新新闻和待办事项提醒。您可以轻松,迅速地在上面建立好自己的服务,而无需考虑基础结构应用程序,只需要一个函数即可。

该项目可以用于生产吗?

我并不建议您这样做,有两个原因:首先是安全性,尽管 vm 保证了安全的沙箱,但仍然存在风险。如果您想了解更多有关信息,请查看 API 文档vm。 另一个原因是,使用 vm 来搭建 serverless 服务的初衷是在个人设备上运行简单的小型服务,它几乎无法自动根据工作负载的大小进行扩展,而这整是大型 serverless 服务必备的。因此,我对性能没有考虑太多(实际上,框架 egg.js 在性能方面做了很多工作,并且天生就是“为构建企业应用程序”而设计,但是我并不知道 vm 的性能究竟如何,这需要更多的实验来证明)。

License

MIT License (MIT)

8 回复

是个不错的想法,不过没法在生成环境应用的。vm2模块安全要好很多,可以看一下

@i5ting 谢谢,只是想到了用vm的思路来实现了一下玩玩,生产肯定不会这样解决方案,所以也就没用vm2,只用Node API了。

vm可以做开发环境的代码热更新?

有没有用vm2结合woker_threads 来做await vm.run()的?

@jerikchan vm可以在一个沙盒环境里执行目标代码。不过你说的的开发环境或许直接重启应用就足够,也不必考虑vm了。

@HobaiRiku 可以实现的,不过vm或者vm2这类方案只适合在toy级别的项目里实验玩玩,生产环境还是算了,所以也没深究API了

@lqs469 哈哈,我们已经准备上线了,不过后续对于python和java,可能还是用阿里的函数计算省心。

const Worker = require('worker_threads').Worker;
const worker_code = `
  (async () => {
    const { workerData, parentPort } = require('worker_threads');
    const http = require('http');
    const https = require('https');
    const { VM } = require('vm2');
    const t = Date.now();
    const vm = new VM({
      sandbox: {
        setTimeout,
        https,
        http,
    }});
    try {
      const ret = await vm.run(workerData.code);
      console.log('运行"await vm.run() 用时" :', Date.now() - t,'ms');
      parentPort.postMessage({
        data: typeof ret === 'string' ? ret : JSON.stringify(ret),
      });
    } catch (e) {
      parentPort.postMessage({
        err: e.toString(),
      });
    }
  })();
  `;
const script = `
  function Fibonacci (n) {
      if ( n <= 1 ) {return 1};
      return Fibonacci(n - 1) + Fibonacci(n - 2);
    };
  async function main() {
    const result = await new Promise((ok, err) => {
        const req = http.get('http://127.0.0.1:3333', res => {
          let data = '';
          res.on('data', c => {
            data += c;
          });
          res.on('end', () => {
            Fibonacci(35);
            ok(data);
          });
          res.on('error', error => err(error));
        });
        req.on('error',error => err(error));
        req.end();
    });
    return result;
  }
  main();
  `;
async function runJSAsync(params, code) {
  return new Promise((ok, err) => {
    const worker = new Worker(worker_code, {
      workerData: { params, code },
	  eval: true
    });
    worker.on('message', msg => {
      ok(msg);
    });
    worker.on('error', err);
    worker.on('exit', code => {
      if (code !== 0) {
        err(new Error(`工作线程终止,code:${code}`));
      }
    });
  });
}

setImmediate(async () => {
  const http = require('http');

  const server = http.createServer((req, res) => {
    res.end('{"message":"ok"}');
  });
  server.on('clientError', (err, socket) => {
    socket.end('HTTP/1.1 400 Bad Request\r\n\r\n');
  });
  server.listen(3333);
  try {
    const ret = await runJSAsync(null, script);
    console.log('异步运行VM2结束,运行结果:', ret);
  } catch (error) {
    console.error(error);
  }
});
let i = 1;
setInterval(() => {
  console.log('主进程其他代码--------', i++, 's');
}, 1000);

回到顶部