我正在编写一个API,它接收一个JSON有效负载作为请求主体。
目前,我正在做这样的事情:
post '/doSomething' do
request.body.rewind
request_payload = JSON.parse request.body.read
#do something with request_payload
body request_payload['someKey']
end
有什么方法可以抽象出来,这样我就不需要为每条路线做这件事了?我的一些路由比这更复杂,因此request.body将通过这种方法重新读取和重新分析每个路由,我想避免。
有没有办法让request_payload神奇地可用于路由?像这样:
post '/doSomething' do
#do something with request_payload, it's already parsed and available
body request_payload['someKey']
end
答案 0 :(得分:53)
在处理程序之前使用sinatra:
before do
request.body.rewind
@request_payload = JSON.parse request.body.read
end
这会将它暴露给当前的请求处理程序。如果您希望它暴露给所有处理程序,请将它放在超类中并在处理程序中扩展该类。
答案 1 :(得分:7)
您也可以使用Rack Middleware来解析它。初始化Sinatra课程时,请参阅https://github.com/rack/rack-contrib仅use Rack::PostBodyContentTypeParser
。
答案 2 :(得分:4)
喜欢这个为sinatra 1.4.5工作
before do
if request.body.size > 0
request.body.rewind
@params = ActiveSupport::JSON.decode(request.body.read)
end
end
答案 3 :(得分:2)
您可以将https://github.com/rack/rack-contrib中的Hash
和Rack::PostBodyContentTypeParser
解析为JSON帖子正文:
require 'rack/contrib/post_body_content_type_parser'
class Api < Sinatra::Application
use Rack::PostBodyContentTypeParser
...
end
您甚至可以将自定义块传递给Rack::PostBodyContentTypeParser
,以将JSON解析为符号而不是字符串:
a_proc = proc { |body| JSON.parse(body, symbolize_names: true, create_additions: false) }
use Rack::PostBodyContentTypeParser, &a_proc
答案 4 :(得分:0)
before do
request.body.rewind
@request_payload = JSON.parse(request.body.read, symbolize_names: true)
end
因此,您还可以在解析JSON请求正文时使用symbolize_names,这将使您可以访问@request_payload[:user]
这样的嵌套参数