POST 和 DELETE 在一个 http 请求中来自一个控制器 (rails)
POST and DELETE in one http request and from one controller (rails)
我正在用 rails 构建一个 api,目前正在尝试为 post 实现 heart/unheart 功能。此功能的工作方式是,如果用户对 post 点赞,则数据库中会显示一个新条目(一个用户只能对 post 点赞一次),如果用户取消关注它,则该条目将被删除。目前,我会检查前端是否用户愿意,并发送 POST 或 DELETE 的正确 HTTP 请求。但是,它不是很有效,我想知道是否有更好的方法可以在后端(控制器或模型)上自动完成它并且只有一个 http 请求(例如 POST)?
heart.rb Model:
class Heart < ApplicationRecord
belongs_to :user
belongs_to :post
validates :post_id, presence: true
validates :user_id, presence: true
validates :user_id, uniqueness: { scope: :post_id }
end
user.rb Model:
class User < ApplicationRecord
has_many :posts, dependent: :destroy
has_many :hearts, dependent: :destroy
def heart!(post)
self.hearts.create!(post_id: post.id)
end
def unheart!(post)
heart = self.hearts.find_by_post_id(post.id)
heart.destroy!
end
end
hearts_controller.rb
class HeartsController < ApplicationController
def heart
@user = current_user
@post = Post.find(params[:post_id])
if @user.heart!(@post)
end
end
def unheart
@user = current_user
@heart = @user.hearts.find_by_post_id(params[:post_id])
@post = Post.find(params[:post_id])
if @heart.destroy!
end
end
end
routes.rb
Rails.application.routes.draw do
match 'heart', to: 'hearts#heart', via: :post
match 'unheart', to: 'hearts#unheart', via: :delete
end
如您所见,我需要先检查它是否被接受,然后才需要发送正确的 POST 或 DELETE 请求。有没有一种更有效的方法来只发送一个请求,然后在 rails 内部检查它是否愿意,并从一个控制器检查 post/delete ?
我建议您只使用一个路由来监听 post 的 PATCH
方法。类似于:
patch '/posts/:id/heart' => 'posts#heart_post', as: :heart_post
而不是 HeartsController
,在 PostsController
中实施 heart_post
操作,带上你的 post 和当前用户并关注你的 post(无论目前状态,是否有心)。
def heart_post
@post = Post.find(params[:id])
@post.heart_action(@current_user.id)
# whatever you want to render
end
然后,您的 Post
模型将包含其自身的逻辑,并且它会知道它是否需要执行 heart 或 unheart 操作。
def heart_action (user_id)
heart = Heart.find_by(user_id: user_id, post_id: self.id)
if heart
heart.destroy
else
Heart.create(user_id: user_id, link_id: self.id)
end
end
此操作应该只有一条路线。
我正在用 rails 构建一个 api,目前正在尝试为 post 实现 heart/unheart 功能。此功能的工作方式是,如果用户对 post 点赞,则数据库中会显示一个新条目(一个用户只能对 post 点赞一次),如果用户取消关注它,则该条目将被删除。目前,我会检查前端是否用户愿意,并发送 POST 或 DELETE 的正确 HTTP 请求。但是,它不是很有效,我想知道是否有更好的方法可以在后端(控制器或模型)上自动完成它并且只有一个 http 请求(例如 POST)?
heart.rb Model:
class Heart < ApplicationRecord
belongs_to :user
belongs_to :post
validates :post_id, presence: true
validates :user_id, presence: true
validates :user_id, uniqueness: { scope: :post_id }
end
user.rb Model:
class User < ApplicationRecord
has_many :posts, dependent: :destroy
has_many :hearts, dependent: :destroy
def heart!(post)
self.hearts.create!(post_id: post.id)
end
def unheart!(post)
heart = self.hearts.find_by_post_id(post.id)
heart.destroy!
end
end
hearts_controller.rb
class HeartsController < ApplicationController
def heart
@user = current_user
@post = Post.find(params[:post_id])
if @user.heart!(@post)
end
end
def unheart
@user = current_user
@heart = @user.hearts.find_by_post_id(params[:post_id])
@post = Post.find(params[:post_id])
if @heart.destroy!
end
end
end
routes.rb
Rails.application.routes.draw do
match 'heart', to: 'hearts#heart', via: :post
match 'unheart', to: 'hearts#unheart', via: :delete
end
如您所见,我需要先检查它是否被接受,然后才需要发送正确的 POST 或 DELETE 请求。有没有一种更有效的方法来只发送一个请求,然后在 rails 内部检查它是否愿意,并从一个控制器检查 post/delete ?
我建议您只使用一个路由来监听 post 的 PATCH
方法。类似于:
patch '/posts/:id/heart' => 'posts#heart_post', as: :heart_post
而不是 HeartsController
,在 PostsController
中实施 heart_post
操作,带上你的 post 和当前用户并关注你的 post(无论目前状态,是否有心)。
def heart_post
@post = Post.find(params[:id])
@post.heart_action(@current_user.id)
# whatever you want to render
end
然后,您的 Post
模型将包含其自身的逻辑,并且它会知道它是否需要执行 heart 或 unheart 操作。
def heart_action (user_id)
heart = Heart.find_by(user_id: user_id, post_id: self.id)
if heart
heart.destroy
else
Heart.create(user_id: user_id, link_id: self.id)
end
end
此操作应该只有一条路线。