69

我希望能够在同一个域下托管多个 NodeJS 应用程序,而不使用子域(例如 google.com/reader 而不是 images.google.com)。问题是我总是在 Express/NodeJS 中输入 URL 的第一部分,例如“/reader”。

如何设置 Express 应用程序以使基本 URL 为something.com/myapp

所以而不是:

app.get("/myapp", function (req, res) {
   // can be accessed from something.com/myapp
});

我可以:

// Some set-up
app.base = "/myapp"

app.get("/", function (req, res) {
   // can still be accessed from something.com/myapp
});

我还想将 Connect 的 staticProvider 配置为相同的行为方式(现在它默认为提供静态文件something.com/jssomething.com/css代替something.com/myapp/js

4

7 回答 7

148

express 路由器从 4.0 开始可以处理这个问题

http://expressjs.com/en/api.html#router

http://bulkan-evcimen.com/using_express_router_instead_of_express_namespace.html

var express = require('express');
var app = express();
var router = express.Router();

// simple logger for this router's requests
// all requests to this router will first hit this middleware
router.use(function(req, res, next) {
  console.log('%s %s %s', req.method, req.url, req.path);
  next();
});

// this will only be invoked if the path ends in /bar
router.use('/bar', function(req, res, next) {
  // ... maybe some additional /bar logging ...
  next();
});

// always invoked
router.use(function(req, res, next) {
  res.send('Hello World');
});

app.use('/foo', router);

app.listen(3000);

以前的答案(在表达 4.0 之前):

express-namespace 模块(现在已经死了)用来做这个把戏:

https://github.com/visionmedia/express-namespace

require('express-namespace');

app.namespace('/myapp', function() {
        app.get('/', function (req, res) {
           // can be accessed from something.com/myapp
        });
});
于 2011-05-13T15:40:35.020 回答
21

目前还不支持,自己添加也不容易。

整个路由的东西深埋在服务器代码中,作为奖励,它们自己的路由没有暴露。

我挖掘了源代码并检查了最新版本的 Express 和 Connect 中间件,但仍然不支持此类功能,您应该在ConnectExpress本身上打开一个问题。

同时...

自己打补丁,这是一种快速简便的方法,只需更改一行代码。

~/.local/lib/node/.npm/express/1.0.0/package/lib/express/servers.js中,搜索:

// Generate the route
this.routes[method](path, fn);

这应该在 line 周围357,将其替换为:

// Generate the route
this.routes[method](((self.settings.base || '') + path), fn);

现在只需添加设置:

app.set('base', '/myapp');

这适用于纯字符串路径,对于 RegEx 支持,您必须自己在路由器中间件中进行修改,在这种情况下最好提交问题。

就静态提供者而言,只需在/mypapp设置时添加即可。

更新

使它也可以与 RegExp 一起使用:

// replace
this.routes[method](baseRoute(self.settings.base || '', path), fn);

// helper
function baseRoute(base, path) {
    if (path instanceof RegExp) {
        var exp = RegExp(path).toString().slice(1, -1);
        return new RegExp(exp[0] === '^' ? '^' + base + exp.substring(1) : base + exp);

    } else {
        return (base || '') + path;
    }
}

我只用少数几个表达式对此进行了测试,所以这不是 100% 测试的,但理论上它应该可以工作。

更新 2

提交了补丁问题:
https ://github.com/visionmedia/express/issues/issue/478

于 2010-12-07T11:12:01.193 回答
12

只是为了更新线程,现在使用Express.js v4你可以不使用express-namespace

var express = require('express'),
    forumRouter = express.Router(),
    threadRouter = express.Router(),
    app = express();

forumRouter.get('/:id)', function(req, res){
  res.send('GET forum ' + req.params.id);
});

forumRouter.get('/:id/edit', function(req, res){
  res.send('GET forum ' + req.params.id + ' edit page');
});


forumRouter.delete('/:id', function(req, res){
  res.send('DELETE forum ' + req.params.id);
});

app.use('/forum', forumRouter);

threadRouter.get('/:id/thread/:tid', function(req, res){
  res.send('GET forum ' + req.params.id + ' thread ' + req.params.tid);
});

forumRouter.use('/', threadRouter);

app.listen(app.get("port") || 3000);

干杯!

于 2014-07-14T20:03:11.853 回答
7

我能够结合使用路由的 express-namespace 和下面的谷歌小组讨论中的静态资产修复来实现这一点。此代码段将对 /foo/javascripts/jquery.js 的请求视为对 /javascripts/jquery.js 的请求:

app.use('/foo', express.static(__dirname + '/public'));

来源: https ://groups.google.com/forum/#!msg/express-js/xlP6_DX6he0/6OTY4hwfV-0J

于 2012-09-18T21:32:44.320 回答
4

我知道这是一个非常古老的问题,但 Express 已经发生了很大变化,因为大多数这些答案都已发布,所以我想我会分享我的方法。

当然,您可以使用带有 Express 4 的路由器将特定路径后面的相关功能组合在一起。这是有据可查的,并且已经被其他答案所涵盖。

但是,也可以将整个应用程序安装在特定路径上。作为一个例子,假设我们的应用程序(我们想要托管的那个/myapp)在一个名为的文件中看起来像这样myapp.js

var express = require('express'),
    path = require('path'),
    app = express();

app.use(express.static(path.join(__dirname, 'public')));

app.get('/hello', function(req, res) {
    res.send('Hello');
});

// Lots of other stuff here

exports.app = app;

在我们的主 js 文件中,我们可以将整个应用程序挂载到路径/myapp

var express = require('express'),
    app = express(),
    myApp = require('./myapp').app;

app.use('/myapp', myApp);

app.listen(3000);

请注意,我们在这里创建了两个应用程序,一个安装在另一个上。主应用程序可以根据需要在不同的路径上安装更多的子应用程序。

中的代码myapp.js完全独立于它的安装位置。它类似于express-generator在这方面使用的结构。

可以在此处找到有关子应用程序的一些文档:

https://expressjs.com/en/4x/api.html#app.mountpath https://expressjs.com/en/4x/api.html#app.onmount

于 2017-09-13T20:59:19.597 回答
3

还有可靠性问题。如果可靠性很重要,一个常见的解决方案是使用前端反向 HTTP 代理,例如 nginx 或 HAProxy。它们都使用单线程事件架构,因此非常具有可扩展性。

然后,您可以为不同的子站点设置不同的节点进程,如果一个站点出现故障(未捕获的异常、内存泄漏、程序员错误等),其余的子站点继续工作。

于 2011-09-05T19:20:24.690 回答
1

我一直在寻找此功能,但寻找的是 API 路由,而不是静态文件。我所做的是,当我初始化路由器时,我添加了挂载路径。所以我的配置看起来像这样

//Default configuration
app.configure(function(){
    app.use(express.compress());
    app.use(express.logger('dev'));
    app.set('json spaces',0);
    app.use(express.limit('2mb'));
    app.use(express.bodyParser());

    app.use('/api', app.router);        // <---

    app.use(function(err, req, res, callback){
        res.json(err.code, {});
    });
});

调用路由器时注意'/api'

于 2013-04-19T13:10:44.593 回答