欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Node.Js中实现端口重用原理详解

程序员文章站 2022-07-06 12:14:40
本文介绍了node.js中实现端口重用原理详解,分享给大家,具体如下: 起源,从官方实例中看多进程共用端口 const cluster = require('c...

本文介绍了node.js中实现端口重用原理详解,分享给大家,具体如下:

起源,从官方实例中看多进程共用端口

const cluster = require('cluster');
const http = require('http');
const numcpus = require('os').cpus().length;

if (cluster.ismaster) {
 console.log(`master ${process.pid} is running`);

 for (let i = 0; i < numcpus; i++) {
  cluster.fork();
 }

 cluster.on('exit', (worker, code, signal) => {
  console.log(`worker ${worker.process.pid} died`);
 });
} else {
 http.createserver((req, res) => {
  res.writehead(200);
  res.end('hello world\n');
 }).listen(8000);

 console.log(`worker ${process.pid} started`);
}

执行结果:

$ node server.js
master 3596 is running
worker 4324 started
worker 4520 started
worker 6056 started
worker 5644 started

了解http.js模块:

我们都只有要创建一个http服务,必须引用http模块,http模块最终会调用net.js实现网络服务

// lib/net.js
'use strict';

 ...
server.prototype.listen = function(...args) {
  ...
 if (options instanceof tcp) {
   this._handle = options;
   this[async_id_symbol] = this._handle.getasyncid();
   listenincluster(this, null, -1, -1, backlogfromargs); // 注意这个方法调用了cluster模式下的处理办法
   return this;
  }
  ...
};

function listenincluster(server, address, port, addresstype,backlog, fd, exclusive) {
// 如果是master 进程或者没有开启cluster模式直接启动listen
if (cluster.ismaster || exclusive) {
  //_listen2,细心的人一定会发现为什么是listen2而不直接使用listen
 // _listen2 包裹了listen方法,如果是worker进程,会调用被hack后的listen方法,从而避免出错端口被占用的错误
  server._listen2(address, port, addresstype, backlog, fd);
  return;
 }
 const serverquery = {
  address: address,
  port: port,
  addresstype: addresstype,
  fd: fd,
  flags: 0
 };

// 是fork 出来的进程,获取master上的handel,并且监听,
// 现在是不是很好奇_getserver方法做了什么
 cluster._getserver(server, serverquery, listenonmasterhandle);
}
 ...

答案很快就可以通过cluster._getserver 这个函数找到

  1. 代理了server._listen2 这个方法在work进程的执行操作
  2. 向master发送queryserver消息,向master注册一个内部tcp服务器
// lib/internal/cluster/child.js
cluster._getserver = function(obj, options, cb) {
 // ...
 const message = util._extend({
  act: 'queryserver',  // 关键点:构建一个queryserver的消息
  index: indexes[indexeskey],
  data: null
 }, options);

 message.address = address;

// 发送queryserver消息给master进程,master 在收到这个消息后,会创建一个开始一个server,并且listen
 send(message, (reply, handle) => {
   rr(reply, indexeskey, cb);       // round-robin.
 });

 obj.once('listening', () => {
  cluster.worker.state = 'listening';
  const address = obj.address();
  message.act = 'listening';
  message.port = address && address.port || options.port;
  send(message);
 });
};
 //...
 // round-robin. master distributes handles across workers.
function rr(message, indexeskey, cb) {
  if (message.errno) return cb(message.errno, null);
  var key = message.key;
  // 这里hack 了listen方法
  // 子进程调用的listen方法,就是这个,直接返回0,所以不会报端口被占用的错误
  function listen(backlog) {
    return 0;
  }
  // ...
  const handle = { close, listen, ref: noop, unref: noop };
  handles[key] = handle;
  // 这个cb 函数是net.js 中的listenonmasterhandle 方法
  cb(0, handle);
}
// lib/net.js
/*
function listenonmasterhandle(err, handle) {
  err = checkbinderror(err, port, handle);
  server._handle = handle;
  // _listen2 函数中,调用的handle.listen方法,也就是上面被hack的listen
  server._listen2(address, port, addresstype, backlog, fd);
 }
*/

master进程收到queryserver消息后进行启动服务

  1. 如果地址没被监听过,通过roundrobinhandle监听开启服务
  2. 如果地址已经被监听,直接绑定handel到已经监听到服务上,去消费请求
// lib/internal/cluster/master.js
function queryserver(worker, message) {

  const args = [
    message.address,
    message.port,
    message.addresstype,
    message.fd,
    message.index
  ];

  const key = args.join(':');
  var handle = handles[key];

  // 如果地址没被监听过,通过roundrobinhandle监听开启服务
  if (handle === undefined) {
    var constructor = roundrobinhandle;
    if (schedulingpolicy !== sched_rr ||
      message.addresstype === 'udp4' ||
      message.addresstype === 'udp6') {
      constructor = sharedhandle;
    }

    handles[key] = handle = new constructor(key,
      address,
      message.port,
      message.addresstype,
      message.fd,
      message.flags);
  }

  // 如果地址已经被监听,直接绑定handel到已经监听到服务上,去消费请求
  // set custom server data
  handle.add(worker, (errno, reply, handle) => {
    reply = util._extend({
      errno: errno,
      key: key,
      ack: message.seq,
      data: handles[key].data
    }, reply);

    if (errno)
      delete handles[key]; // gives other workers a chance to retry.

    send(worker, reply, handle);
  });
}

看到这一步,已经很明显,我们知道了多进行端口共享的实现原理

  1. 其实端口仅由master进程中的内部tcp服务器监听了一次
  2. 因为net.js 模块中会判断当前的进程是master还是worker进程
  3. 如果是worker进程调用cluster._getserver 去hack原生的listen 方法
  4. 所以在child调用的listen方法,是一个return 0 的空方法,所以不会报端口占用错误

那现在问题来了,既然worker进程是如何获取到master进程监听服务接收到的connect呢?

  1. 监听master进程启动的tcp服务器的connection事件
  2. 通过轮询挑选出一个worker
  3. 向其发送newconn内部消息,消息体中包含了客户端句柄
  4. 有了句柄,谁都知道要怎么处理了哈哈
// lib/internal/cluster/round_robin_handle.js

function roundrobinhandle(key, address, port, addresstype, fd) {

  this.server = net.createserver(assert.fail);

  if (fd >= 0)
    this.server.listen({ fd });
  else if (port >= 0)
    this.server.listen(port, address);
  else
    this.server.listen(address); // unix socket path.

  this.server.once('listening', () => {
    this.handle = this.server._handle;
    // 监听onconnection方法
    this.handle.onconnection = (err, handle) => this.distribute(err, handle);
    this.server._handle = null;
    this.server = null;
  });
}

roundrobinhandle.prototype.add = function (worker, send) {
  // ...
};

roundrobinhandle.prototype.remove = function (worker) {
  // ...
};

roundrobinhandle.prototype.distribute = function (err, handle) {
  // 负载均衡地挑选出一个worker
  this.handles.push(handle);
  const worker = this.free.shift();
  if (worker) this.handoff(worker);
};

roundrobinhandle.prototype.handoff = function (worker) {
  const handle = this.handles.shift();
  const message = { act: 'newconn', key: this.key };
  // 向work进程其发送newconn内部消息和客户端的句柄handle
  sendhelper(worker.process, message, handle, (reply) => {
  // ...
    this.handoff(worker);
  });
};

下面让我们看看worker进程接收到newconn消息后进行了哪些操作

// lib/child.js
function onmessage(message, handle) {
  if (message.act === 'newconn')
   onconnection(message, handle);
  else if (message.act === 'disconnect')
   _disconnect.call(worker, true);
 }

// round-robin connection.
// 接收连接,并且处理
function onconnection(message, handle) {
 const key = message.key;
 const server = handles[key];
 const accepted = server !== undefined;

 send({ ack: message.seq, accepted });

 if (accepted) server.onconnection(0, handle);
}

总结

  1. net模块会对进程进行判断,是worker 还是master, 是worker的话进行hack net.server实例的listen方法
  2. worker 调用的listen 方法是hack掉的,直接return 0,不过会向master注册一个connection接手的事件
  3. master 收到客户端connection事件后,会轮询向worker发送connection上来的客户端句柄
  4. worker收到master发送过来客户端的句柄,这时候就可以处理客户端请求了

分享出于共享学习的目的,如有错误,欢迎大家留言指导,不喜勿喷。也希望大家多多支持。