Rails:查找尚未通过连接表连接的所有资源



我正在尝试重写将产品与项目连接和断开连接的操作。目前,我的select_to_project视图显示所有产品,但我希望它仅显示尚未连接到给定项目的产品。

产品和项目通过联接表连接

class Product < ActiveRecord::Base
has_and_belongs_to_many :projects, :join_table => "projects_products"  
end
class Project < ActiveRecord::Base
has_and_belongs_to_many :products, :join_table => "projects_products" 
end
class ProjectsProduct < ActiveRecord::Base
attr_accessible :project_id, :product_id
belongs_to :project
belongs_to :product
end

在我的产品控制器中,我目前有:

def select_to_project
@project = Project.find(params[:id])
@products = Product.find(:all)
end
def select_from_project
@project = Project.find(params[:id])
end

显然,select_to_project视图当前显示所有可能的产品,甚至包括那些已经通过连接表连接的产品。

我认为select_to_project操作应该更改为以下内容:

def select_to_project
@project = Project.find(params[:id])
@products = Product.joins(:projects => :products).where('products_projects_join.product_id IS NOT ?', @product)
end

但是当我尝试加载相对视图时,我目前收到MySQL错误:

Mysql::Error: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '1)' at line 1: SELECT `products`.* FROM `products` INNER JOIN `projects_products` ON `projects_products`.`product_id` = `products`.`id` INNER JOIN `projects` ON `projects`.`id` = `projects_products`.`project_id` INNER JOIN `projects_products` `products_projects_join` ON `products_projects_join`.`project_id` = `projects`.`id` INNER JOIN `products` `products_projects` ON `products_projects`.`id` = `products_projects_join`.`product_id` WHERE (products_projects_join.project_id IS NOT 1)

如何让这个查询在 Rails 3 中工作?

提前非常感谢你。

更新

多亏了@Sebastian帕尔马,视图现在已构建,但查询结果不正确。

@products = Product.joins(:projects => :products).where('products_projects_join.project_id != ?', @project.id).uniq.order(:id, :order => 'id ASC')

这将生成以下查询:

SELECT DISTINCT 'products'.* FROM 'products' INNER JOIN 'projects_products' ON 'projects_products'.'product_id' = 'products'.'id' INNER JOIN 'projects' ON 'projects'.'id' = 'projects_products'.'project_id' INNER JOIN 'projects_products' 'products_projects_join' ON 'products_projects_join'.'project_id' = 'projects'.'id' INNER JOIN 'products' 'products_projects' ON 'products_projects'.'id' = 'products_projects_join'.'product_id' WHERE (products_projects_join.project_id != 2) ORDER BY id, '--- n:order: id ASCn'

我有 14 个产品条目,项目 2 已经连接到其中的 4 个。

projects_products_id / project_id / product_id
3 2 1
4 2 2
5 2 3
6 2 12

我的查询应显示 ID 为的产品:4、5、6、7、8、9、10、11、13、14。

它目前显示产品 3、4、5、6、7、8、9、10、12、13。

产品 14 当前不在连接表上,3 和 12 已连接,11 未连接。

就好像查询结果向左滑动了一个 id 值。

理想情况下,我想从产品表中查找尚未在连接表上连接到给定项目的所有条目。

您已将关联设置为has_and_belongs_to_many但HABTM是无头的,因此实际上也有一个连接模型是没有意义的。

相反,您希望将协会设置为has_many through:,并为表使用常规名称。

首先将联接表重命名为project_products,并确保它具有 id 列。

class Product < ActiveRecord::Base
has_many :project_products 
has_many :projects, through: project_products 
end
class Project < ActiveRecord::Base
has_many :project_products 
has_many :products, through: project_products 
end
class ProjectProduct < ActiveRecord::Base
attr_accessible :project_id, :product_id
belongs_to :project
belongs_to :product
end

这将允许您通过关联project_products加入。

执行此操作的最简单方法是执行子查询。在现代版本的 Rails 中,您可以执行以下操作:

Product.where.not(
id: project.products
)

这将创建以下查询:

SELECT "products".* FROM "products" 
WHERE "products"."id" NOT IN (
SELECT "products"."id" FROM "products" 
INNER JOIN "project_products" ON "products"."id" = "project_products"."product_id" 
WHERE "project_products"."project_id" = $1
) LIMIT $2

然而,where.not是在Rails 4.0中引入的。作为解决方法,您可以在 Rails 3 中使用字符串。

class Product < ApplicationRecord
has_many :project_products
has_many :projects, through: :project_products
def self.not_assigned_to_project(project)
Product.where("
products.id NOT IN (#{project.products.select(:id).to_sql})
")
end
end

相关内容

  • 没有找到相关文章

最新更新