19

我想MyMiddleware在我的 Rack 应用程序中运行,但仅限于某些路径。我希望使用Rack::Builderor 至少Rack::URLMap,但我不太清楚如何使用。

这是我认为可行的,但没有:

# in my rackup file or Rails environment.rb:
map '/foo' do
  use MyMiddleware, { :some => 'options' }
end

或者,更好的是,使用正则表达式:

map /^foo/ do
  use MyMiddleware, { :some => 'options' }
end

map似乎最后需要一个应用程序;它不会仅仅依靠将控制权传回其父级。undefined method 'each' for nil:NilClass(当 Rack 试图将该do...end块的末尾变成. 时,实际的错误是“ ” app。)

是否有一个中间件接受一组中间件和一个路径,并且只有在路径匹配时才运行它们?

4

2 回答 2

14

您可以让 MyMiddleware 检查路径,如果匹配,则不将控制权传递给下一个中间件。

class MyMiddleware
  def initialize app
    @app = app
  end
  def call env
    middlewary_stuff if env['PATH_INFO'] == '/foo'
    @app.call env
  end

  def middlewary_stuff
    #...
  end
end

或者,您可以使用不带 dslness 的 URLMap。它看起来像这样:

main_app = MainApp.new
Rack::URLMap.new '/'=>main_app, /^(foo|bar)/ => MyMiddleWare.new(main_app)

URLMap 实际上很容易理解

于 2009-07-28T16:55:07.973 回答
10

这不起作用,因为@app不存在于正确的范围内:

# in my_app.ru or any Rack::Builder context:
@app = self
map '/foo' do
  use MyMiddleware
  run lambda { |env| @app.call(env) }
end

但这将:

# in my_app.ru or any Rack::Builder context:
::MAIN_RACK_APP = self
map '/foo' do
  use MyMiddleware
  run lambda { |env| ::MAIN_RACK_APP.call(env) }
end

Rack::Builder将第一个参数map从路径的前面剥离,因此它不会无休止地递归。不幸的是,这意味着在去除该路径前缀之后,路径的其余部分不太可能正确匹配其他映射。

这是一个例子:

::MAIN_APP = self
use Rack::ShowExceptions
use Rack::Lint
use Rack::Reloader, 0
use Rack::ContentLength

map '/html' do
  use MyContentTypeSettingMiddleware, 'text/html'
  run lambda { |env| puts 'HTML!'; ::MAIN_APP.call(env) }
end

map '/xml' do
  use MyContentTypeSettingMiddleware, 'application/xml'
  run lambda { |env| puts 'XML!'; ::MAIN_APP.call(env) }
end

map '/' do
  use ContentType, 'text/plain'
  run lambda { |env| [ 200, {}, "<p>Hello!</p>" ] }
end

转到/html/xml会导致以下内容进入日志:

HTML!
XML!
127.0.0.1 - - [28/May/2009 17:41:42] "GET /html/xml HTTP/1.1" 200 13 0.3626

也就是说,映射到前缀'/html'条的应用程序现在与映射到的应用程序匹配。'/html''/xml'

于 2009-05-28T16:58:45.457 回答