Использование регулярного выражения в URI обработчика Mongrel

В настоящее время я использую Mongrel для разработки пользовательского проекта веб-приложения.

Я хотел бы, чтобы Mongrel использовал определенный обработчик Http на основе регулярного выражения. Например, каждый раз, когда кто-то вызывает URL вида http://test/bla1.js или http://test/bla2.js для управления запросом вызывается тот же обработчик Http.

Мой код пока выглядит так:

http_server = Mongrel::Configurator.new :host => config.get("http_host") do
  listener :port => config.get("http_port") do

    uri Regexp.escape("/[a-z0-9]+.js"), :handler => BLAH::CustomHandler.new
    uri '/ui/public', :handler => Mongrel::DirHandler.new("#{$d}/public/")
    uri '/favicon', :handler => Mongrel::Error404Handler.new('')

    trap("INT") { stop }
    run
  end
end

Как видите, здесь я пытаюсь использовать регулярное выражение вместо строки:

 uri Regexp.escape("/[a-z0-9]+.js"), :handler => BLAH::CustomHandler.new

но это не работает. Любое решение?

Спасибо за это.


person Benjamin    schedule 27.02.2010    source источник


Ответы (2)


Вы должны внедрить новый код в часть Mongrel URIClassifier, которая в противном случае счастливо не знает об URI регулярных выражений.

Ниже приведен один из способов сделать это:

#
# Must do the following BEFORE Mongrel::Configurator.new
#  Augment some of the key methods in Mongrel::URIClassifier
#  See lib/ruby/gems/XXX/gems/mongrel-1.1.5/lib/mongrel/uri_classifier.rb
#
Mongrel::URIClassifier.class_eval <<-EOS, __FILE__, __LINE__
  # Save original methods
  alias_method :register_without_regexp, :register
  alias_method :unregister_without_regexp, :unregister
  alias_method :resolve_without_regexp, :resolve

  def register(uri, handler)
    if uri.is_a?(Regexp)
      unless (@regexp_handlers ||= []).any? { |(re,h)| re==uri ? h.concat(handler) : false }
        @regexp_handlers << [ uri, handler ]
      end
    else
      # Original behaviour
      register_without_regexp(uri, handler)
    end
  end

  def unregister(uri)
    if uri.is_a?(Regexp)
      raise Mongrel::URIClassifier::RegistrationError, "\#{uri.inspect} was not registered" unless (@regexp_handlers ||= []).reject! { |(re,h)| re==uri }
    else
      # Original behaviour
      unregister_without_regexp(uri)
    end
  end

  def resolve(request_uri)
    # Try original behaviour FIRST
    result = resolve_without_regexp(request_uri)
    # If a match is not found with non-regexp URIs, try regexp
    if result[0].blank?
      (@regexp_handlers ||= []).any? { |(re,h)| (m = re.match(request_uri)) ? (result = [ m.pre_match + m.to_s, (m.to_s == Mongrel::Const::SLASH ? request_uri : m.post_match), h ]) : false }
    end
    result
  end
EOS

http_server = Mongrel::Configurator.new :host => config.get("http_host") do 
  listener :port => config.get("http_port") do 

    # Can pass a regular expression as URI
    #  (URI must be of type Regexp, no escaping please!)
    # Regular expression can match any part of an URL, start with "^/..." to
    #  anchor match at URI beginning.
    # The way this is implemented, regexp matches are only evaluated AFTER
    #  all non-regexp matches have failed (mostly for performance reasons.)
    # Also, for regexp URIs, the :in_front is ignored; adding multiple handlers
    #  to the same URI regexp behaves as if :in_front => false
    uri /^[a-z0-9]+.js/, :handler => BLAH::CustomHandler.new 

    uri '/ui/public', :handler => Mongrel::DirHandler.new("#{$d}/public/") 
    uri '/favicon', :handler => Mongrel::Error404Handler.new('') 

    trap("INT") { stop } 
    run 
  end 
end

Кажется, он отлично работает с Mongrel 1.1.5.

person vladr    schedule 05.03.2010
comment
Спасибо. Это как раз то, что мне было нужно. - person Benjamin; 06.03.2010

Вместо этого вам следует подумать о создании Rack-приложения. Стойка это:

  • стандарт для веб-приложений Ruby
  • используется всеми популярными веб-фреймворками Ruby (Rails, Мерб , Синатра, Кемпинг, Ramaze, ...)
  • гораздо проще продлить
  • готов к работе на любом сервере приложений (Mongrel, Webrick, Thin, Passenger, ...)

Rack имеет DSL-сопоставление URL, Rack::Builder, который позволяет вам сопоставлять различные приложения Rack с определенными префиксами URL. Обычно вы сохраняете его как config.ru и запускаете с rackup.

К сожалению, он также не позволяет использовать регулярные выражения. Но из-за простоты Rack очень легко написать «приложение» (фактически lambda), которое будет вызывать нужное приложение, если URL-адрес соответствует определенному регулярному выражению.

Исходя из вашего примера, ваш config.ru может выглядеть примерно так:

require "my_custom_rack_app" # Whatever provides your MyCustomRackApp.

js_handler = MyCustomRackApp.new

default_handlers = Rack::Builder.new do
  map "/public" do
    run Rack::Directory.new("my_dir/public")
  end

  # Uncomment this to replace Rack::Builder's 404 handler with your own:
  # map "/" do
  #   run lambda { |env|
  #     [404, {"Content-Type" => "text/plain"}, ["My 404 response"]]
  #   }
  # end
end

run lambda { |env|
  if env["PATH_INFO"] =~ %r{/[a-z0-9]+\.js}
    js_handler.call(env)
  else
    default_handlers.call(env)
  end
}

Затем запустите приложение Rack в командной строке:

% rackup

Если у вас установлен mongrel, он будет запущен на порту 9292. Готово!

person molf    schedule 05.03.2010