我有一个主要基于 REST 的 Rails 站点,我想添加 JSON API 支持。
对于干净的代码库,我应该在现有控制器中添加此支持还是创建仅处理此 API 方法的新控制器,然后将所有通用代码移至模型/帮助程序?
我有一个主要基于 REST 的 Rails 站点,我想添加 JSON API 支持。
对于干净的代码库,我应该在现有控制器中添加此支持还是创建仅处理此 API 方法的新控制器,然后将所有通用代码移至模型/帮助程序?
我使用了这两种技术:在同一个控制器中编写 API 逻辑,并为 API 请求制作单独的控制器。
如果它只是一个 API,一个仅供您使用的小应用程序,请使用 rails 提供的默认控制器模型关系。代码会很干净。您的路线文件也将是干净的。
如果您有一个网站并且您想构建一个 API,请单独进行。我已经在现有控制器旁边构建了一个,代码太乱了。代码我重构了好几次,但还是不喜欢(这也是个人口味的问题)。
另一种解决方案是制作带有前缀的控制器。例如:ApiUsersController
。这会使您的routes.rb
文件看起来很难看,因为您必须手动指定路由以匹配相应的控制器方法。
对我来说,可行的解决方案是将所有 API 逻辑移动到 API 命名空间下的单独控制器中。命名空间还允许您进行 API 版本控制。因此,例如,您的路线将是:
GET /api/v1/users.json
POST /api/v1/users.json
然后,将来您可以创建另一个 API 版本,比如说v2
,而不会破坏使用旧版本 API 的现有应用程序。
您可以在此处找到有关命名空间的更多信息:http: //guides.rubyonrails.org/routing.html#controller-namespaces-and-routing
关于带有版本控制的 REST-full API 的精彩教程:http ://railscasts.com/episodes/350-rest-api-versioning?view=asciicast
Rails 控制器生成器默认实现 JSON 响应。
例如,如果您有此方法:
class UsersController < ApplicationController
def index
@users = User.all
end
end
您可以像这样添加 JSON 响应
class UsersController < Application Controller
def index
respond_to do |format|
format.html
format.js { render :json => @users }
end
end
end
现在,你有两个回应/users
http://someapp.com/users
http://someapp.com/users.json
您可以很容易地添加另一个;例如,
format.xml { render :xml => @users }
现在您的应用将响应http://someapp.com/users.xml
您可能不想在 json 中输出表的所有字段。为此,请查看rails/jbuilder
. 它允许您使用构建器风格的 DSL 创建 JSON 结构。
jbuilder README 中的一个示例
Jbuilder.encode do |json|
json.content format_content(@message.content)
json.(@message, :created_at, :updated_at)
json.author do
json.name @message.creator.name.familiar
json.email_address @message.creator.email_address_with_name
json.url url_for(@message.creator, format: :json)
end
if current_user.admin?
json.visitors calculate_visitors(@message)
end
json.comments @message.comments, :content, :created_at
json.attachments @message.attachments do |attachment|
json.filename attachment.filename
json.url url_for(attachment)
end
end
产生以下输出:
{
"content": "<p>This is <i>serious</i> monkey business",
"created_at": "2011-10-29T20:45:28-05:00",
"updated_at": "2011-10-29T20:45:28-05:00",
"author": {
"name": "David H.",
"email_address": "'David Heinemeier Hansson' <david@heinemeierhansson.com>",
"url": "http://example.com/users/1-david.json"
},
"visitors": 15,
"comments": [
{ "content": "Hello everyone!", "created_at": "2011-10-29T20:45:28-05:00" },
{ "content": "To you my good sir!", "created_at": "2011-10-29T20:47:28-05:00" }
],
"attachments": [
{ "filename": "forecast.xls", "url": "http://example.com/downloads/forecast.xls" },
{ "filename": "presentation.pdf", "url": "http://example.com/downloads/presentation.pdf" }
]
}