如何使用hapi.js在同一台服务器上支持多个网站?

时间:2016-02-29 11:54:04

标签: javascript node.js virtualhost hapijs

假设我想在同一台服务器上使用单个IP地址(即使用虚拟主机)托管我的2个网站(cats.com和dogs.com)。我想用hapi.js编写它们并将它们作为单个进程运行。

网站可能有重叠路径,例如它们可能都有/about页。

我怎么能用hapi实现这个?

2 个答案:

答案 0 :(得分:4)

使用hapi实现这一目标的一个好方法是在加载插件时将不同的网站分开plugins并使用vhost修饰符,理想情况下使用Glue

以下是一个例子:

站点/ dogs.js

exports.register = function (server, options, next) {

    // Put all your routes for the site in here

    server.route({
        method: 'GET',
        path: '/',
        handler: function (request, reply) {

            reply('Dogs homepage');
        }
    });

    next();
};

exports.register.attributes = { name: 'dogs' };

站点/ cats.js

exports.register = function (server, options, next) {

    // Put all your routes for the site in here

    server.route({
        method: 'GET',
        path: '/',
        handler: function (request, reply) {

            reply('Cats homepage');
        }
    });

    next();
};

exports.register.attributes = { name: 'cats' };

<强> index.js

const Glue = require('glue');
const Hoek = require('hoek');

const manifest = {
    connections: [{
        port: 4000,
    }],
    registrations: [
        {
            plugin: {
                register: './sites/cats'
            },
            options: {
                routes: {
                    vhost: 'cats.com'
                }
            }
        },
        {
            plugin: {
                register: './sites/dogs'
            },
            options: {
                routes: {
                    vhost: 'dogs.com'
                }
            }
        }
    ]
};

const options = {
    relativeTo: __dirname
};

Glue.compose(manifest, options, (err, server) => {

    Hoek.assert(!err, err);
    server.start((err) => {

        Hoek.assert(!err, err);
        console.log('server started');
    });
});

然后,您可以使用几个cURL命令确认路由是否正常工作:

$ curl -H "Host: cats.com" localhost:4000/
Cats homepage

$ curl -H "Host: dogs.com" localhost:4000/
Dogs homepage

浏览器会为您设置主机标题,因此当您浏览到http://cats.comhttp://dogs.com hapi将为您提供正确的内容(假设您的DNS配置正确)。

答案 1 :(得分:0)

如果你使用的是Hapi版本&gt; = 17,虽然想法是相同的,但细节略有改变。

我们希望每个网站都有一个插件。然后,一旦我们将每个网站都提取到插件中(catsdogs下方),我们就可以使用glue撰写单独的配置,并使用hapi为网站提供服务。< / p>

在下面的示例中,插件不会知道或关心他们在哪个域上投放。

代码

Cats Plugin

这是与cats.com一起使用的“服务器”。它返回根路径Hello Cats!处的文本/。在现实生活中它会做一些更有用的事情,你可能会在一个真实的项目中有更多的路线和处理程序,但这个想法仍然是相同的。

// ./sites/cats.js
exports.plugin = {
  name: 'cats',
  version: '1.0.0',
  register: async function(server, options) {
    server.route({
      method: 'GET',
      path: '/',
      handler: (request, h) => {
        return 'Hello Cats!'
      }
    })
  }
}

狗插件

这是显示在dogs.com的内容的服务器。它与cats插件完全相同,只是它返回文本Hello Dogs!。同样,这不是一个有用的插件,仅供参考。

// ./sites/dogs.js
exports.plugin = {
  name: 'dogs',
  version: '1.0.0',
  register: async function(server, options) {
    server.route({
      method: 'GET',
      path: '/',
      handler: (request, h) => {
        return 'Hello Dogs!'
      }
    })
  }
}

主服务器

这是指定vhost的位置,将插件分配给主机cats.comdogs.com

// ./server.js
const Hapi = require('hapi')
const Glue = require('glue')

const manifest = {
  server: {
    host: process.env.HOST || '0.0.0.0',
    port: process.env.PORT || 8080
  },
  register: {
    plugins: [
      {
        plugin: './sites/cats',
        routes: {
          vhost: 'cats.com'
        }
      },
      {
        plugin: './sites/dogs',
        routes: {
          vhost: 'dogs.com'
        }
      }
    ]
  }
}

const options = {
  relativeTo: __dirname
}

const startServer = async function () {
  try {
    const server = await Glue.compose(manifest, options)
    await server.start()
    console.log('Hapi days for Cats and for Dogs!')
  }
  catch (err) {
    console.error(err)
    process.exit(1)
  }
}

startServer()

结果

访问不同版本

启动服务器

$ node server.js
Hapi days for Cats and for Dogs!

采样服务器输出

$ curl -H "Host: cats.com" localhost:8080/
Hello Cats!

$ curl -H "Host: dogs.com" localhost:8080/
Hello Dogs!

$ curl localhost:8080/
{"statusCode":404,"error":"Not Found","message":"Not Found"}

$ curl -H "Host: platypus.com" localhost:8080/
{"statusCode":404,"error":"Not Found","message":"Not Found"}

请注意,默认主机没有路由,因此不指定主机将导致返回404