Rails -使用JBuilder构建自定义JSON



我目前正在处理一个个人Rails 4项目,在其中一个控制器中,我有一个包含三个不同对象的数组,我试图将其转换为JSON。我使用find_by_sql方法用自定义MySQL查询查询DB。代码如下:

控制器:

class ActivitiesController < ApplicationController
  def index
    user = current_user.id # using devise
    limit = 100
    likes = Like.find_by_sql(...)
    shares = Share.find_by_sql(...)
    comments = Comment.find_by_sql(...)
    @activities = [likes, shares, comments]
  end
end

三个对象中的每一个都包含不同的字段。例如,likes对象包含如下字段:post_idfirst_namelast_name等。shares对象包含如下字段:post_idshare_countshared_time等。

这是我要输出的JSON:

{
    "likes": [
      { "post_id": "7", "first_name": "Yss", "last_name": "Salas", "profile_pic": "345943754_o.png" },
      { "post_id": "34", "first_name": "Jessica", "last_name": "Nigri", "profile_pic": "pikachu.png" }
    ],
    "shares": [
      { "post_id": "43", "share_count": 54, "shared_time": "2014-05-04 15:14:45" },
      { "post_id": "54", "share_count": 17, "shared_time": "2014-05-24 03:43:45" }
    ],
    "comments": [
      { "post_id": "34", "first_name": "Yss", "last_name": "Salas", "comment": "¡Me gustas mucho!" },
      { "post_id": "12", "first_name": "Jenna", "last_name": "Marbles", "comment": "Blah blah blah look at my ugly dog awwwww!" }
    ]
}

index.json.jbuilder:

json.array! @activities do |subactivities|
  json.array! subactivities do |activity|
    if activity.class == Like
      json.likes.post_id activity.post_id #exception raised in this line
      json.likes.title activity.title
      json.likes.first_name activity.first_name
      json.likes.last_name activity.last_name
      json.likes.profile_pic activity.profile_pic
    elsif activity.class == Share
      json.shares.post_id activity.post_id
      json.shares.share_count activity.share_count
      json.shares.shared_time activity.shared_time
    else
      json.comments.post_id activity.post_id
      json.comments.first_name activity.first_name
      json.comments.last_name activity.last_name
      json.comments.comment activity.comment
    end
  end
end

上面会引发一个异常NoMethodError in Activities#index, undefined method 'post_id' for #<Object:0x00000003064250>。我是Rails的新手,我猜这是因为没有json.likes属性来分配值。如何使用JBuilder构建上述JSON结构?

你可以不使用JBuilder,更简单,更快捷。

def index
  user = current_user.id # using devise
  limit = 100
  likes = Like.find_by_sql(...).map(&:attributes) # this will return an array of JSON objects
  shares = Share.find_by_sql(...).map(&:attributes)
  comments = Comment.find_by_sql(...).map(&:attributes)
  @activities = {"likes" => likes, "shares" => shares, "comments" => comments}
end
像这样,你就有了一个包含所有所需信息的散列。将寻找喜欢、分享和评论的逻辑提取到单独的类中也可能更好,这样您就可以调用InformationCollector.collect,它将为您提供所需的响应。

从Jbuilder的GitHub页面,解决方案是相当简单的。所有需要做的就是将实例变量传递到一个块中,并手动分配自定义键/值对。

下面的示例将为@likes实例变量生成一个键/值对数组:
# activities/index.json.jbuilder
json.likes @likes do |like|
  json.post_id like.post_id
  json.first_name like.user.first_name
  json.last_name like.user.last_name
  json.profile_pic like.user.profile_pic
end

上面将输出:

{
  "likes": [
    { "post_id": 7, "first_name": "Yss", "last_name": "Salas", "profile_pic": "345943754_o.png" },
    { "post_id": 34, "first_name": "Jessica", "last_name": "Nigri", "profile_pic": "pikachu.png" }
  ],
}

最新更新