3

我正在尝试创建一个外观 API,它通过 Sinatra 接收请求,然后在后端的 Github API 上启动 HTTP 请求。

在我的“hello world”脚本中,我有:

#!/usr/bin/ruby

require 'httpclient'
require 'sinatra'

get '/foo' do
    "hello world"
end

但是,它会遇到以下错误:

$ ./api.rb 
/usr/local/share/gems/gems/sinatra-1.4.3/lib/sinatra/base.rb:1408:in `run!': undefined method `run' for HTTP:Module (NoMethodError)
from /usr/local/share/gems/gems/sinatra-1.4.3/lib/sinatra/main.rb:25:in `block in <module:Sinatra>'

我不明白为什么会这样。如果我注释掉该require 'httpclient'行,它就可以正常工作:

#!/usr/bin/ruby

#require 'httpclient'
require 'sinatra'

get '/foo' do
  "hello world"
end

$ ./api.rb 
[2013-06-26 21:43:12] INFO  WEBrick 1.3.1
[2013-06-26 21:43:12] INFO  ruby 1.9.3 (2013-05-15) [x86_64-linux]
[2013-06-26 21:43:12] WARN  TCPServer Error: Cannot assign requested address - bind(2)
== Sinatra/1.4.3 has taken the stage on 4567 for development with backup from WEBrick
[2013-06-26 21:43:12] INFO  WEBrick::HTTPServer#start: pid=31272 port=4567

我的猜测是 Sinatra 自己使用 HTTPClient 并且正在发生某种命名空间冲突。有没有办法同时使用 HTTPClient 和 Sinatra?

好的,这是请求的信息:

$ gem list sinatra httpclient

*** LOCAL GEMS ***

sinatra (1.4.3)

$ gem out sinatra httpclient
bigdecimal (1.1.0 < 1.2.0)
io-console (0.3 < 0.4.2)
json (1.6.8 < 1.8.0)
rdoc (3.12 < 4.0.1)

我以这种方式计算出httpclient版本:

$ locate httpclient.rb
/usr/local/share/gems/gems/httpclient-2.3.3/lib/httpclient.rb
/usr/local/share/gems/gems/httpclient-2.3.3/test/test_httpclient.rb

我的操作系统是 Fedora 17,但不确定是否重要。

4

1 回答 1

13

我找到了原因。HTTPClient 定义了一个名为HTTP. 默认情况下,Sinatra 会按照命名空间名称HTTPWEBrick的顺序查找 Rack 处理程序。

由于HTTP命名空间已经定义,Sinatra 实际上认为它是一个 Rack 处理程序。我认为这是 Sinatra 中的一个错误。它应该在使用它之前检查处理程序是否响应run

无论如何,解决方法是使用 Thin,或者如果你想使用 WEBrick,然后明确告诉 Sinatra 跳过自动机架检测,方法是:

set :server, 'webrick'

这将阻止 Sinatra 认为 HTTPClientHTTP模块是 Rack 处理程序。

于 2013-06-27T06:17:27.573 回答