我正在使用pg_search gem将全文搜索添加到我的应用中的表格。
我的问题是在尝试创建触发器以在任何更改后使tsv列保持最新时,这是我的迁移:
class AddStoreItemsIndex < ActiveRecord::Migration[5.0]
def up
add_column :store_items, :tsv, :tsvector
add_index :store_items, :tsv, using: "gin"
execute <<-SQL
CREATE TRIGGER tsvectorupdate BEFORE INSERT OR UPDATE
ON store_items FOR EACH ROW EXECUTE PROCEDURE
tsvector_update_trigger(
tsv, 'pg_catalog.english', title, full_description, recommended_age, related_intelligences
);
SQL
end
...
end
在这种情况下,related_intelligences
列是一个数组,因此在尝试查询时我收到以下错误:
ActiveRecord::StatementInvalid:
PG::DatatypeMismatch: ERROR: column "related_intelligences" is not of a character type
如何在此tsv向量列中包含数组?
答案 0 :(得分:3)
我就这样做了:
class AddStoreItemsIndex < ActiveRecord::Migration[5.0]
def up
add_column :store_items, :tsv, :tsvector
add_index :store_items, :tsv, using: 'gin'
execute <<-SQL
CREATE FUNCTION update_tsv() RETURNS trigger AS $$
BEGIN
new.tsv :=
to_tsvector('pg_catalog.english', coalesce(new.title,'')) ||
to_tsvector('pg_catalog.english', coalesce(new.full_description,'')) ||
to_tsvector('pg_catalog.english', coalesce(new.recommended_age,'')) ||
to_tsvector('pg_catalog.english', coalesce(array_to_string(new.related_intelligences, ' '),''));
return new;
END
$$ LANGUAGE plpgsql;
CREATE TRIGGER tsvectorupdate BEFORE INSERT OR UPDATE
ON store_items FOR EACH ROW EXECUTE PROCEDURE update_tsv();
SQL
end
def down
execute <<-SQL
DROP TRIGGER tsvectorupdate
ON store_items;
DROP FUNCTION update_tsv();
SQL
remove_index :store_items, :tsv
remove_column :store_items, :tsv
end
end
tsvector_update_trigger
不能用于数组列,而是根据postgresql search features page中的代码示例创建了自定义函数。
我使用array_to_string
将数组元素转换为文本。