discourse/lib/db_helper.rb

60 lines
1.8 KiB
Ruby
Raw Normal View History

class DbHelper
REMAP_SQL ||= "
SELECT table_name, column_name
FROM information_schema.columns
WHERE table_schema = 'public'
AND is_updatable = 'YES'
AND (data_type LIKE 'char%' OR data_type LIKE 'text%')
ORDER BY table_name, column_name"
2018-11-08 14:07:54 +08:00
def self.remap(from, to, exclude_tables: [])
results = DB.query(REMAP_SQL).to_a
remappable_columns = {}
results.each do |result|
remappable_columns[result.table_name] ||= []
remappable_columns[result.table_name] << result.column_name
end
exclude_tables = exclude_tables.map(&:to_s)
remappable_columns.each do |table_name, column_names|
next if exclude_tables.include?(table_name)
set_clause = column_names.map do |column_name|
2018-11-08 14:07:54 +08:00
"#{column_name} = REGEXP_REPLACE(#{column_name}, :from, :to)"
end.join(", ")
where_clause = column_names.map do |column_name|
2018-11-08 14:07:54 +08:00
"#{column_name} ~* :from"
end.join(" OR ")
2018-11-08 14:07:54 +08:00
DB.exec(<<~SQL, from: from, to: to, from: from)
UPDATE #{table_name}
SET #{set_clause}
WHERE #{where_clause}
SQL
end
2018-04-23 16:26:33 +08:00
2018-04-23 15:57:13 +08:00
SiteSetting.refresh!
end
def self.find(needle, anchor_left = false, anchor_right = false)
connection = ActiveRecord::Base.connection.raw_connection
text_columns = connection.async_exec(REMAP_SQL).to_a
args = ["#{anchor_left ? '' : "%"}#{needle}#{anchor_right ? '' : "%"}"]
found = {}
text_columns.each do |rc|
table_name = rc["table_name"]
column_name = rc["column_name"]
result = connection.async_exec("SELECT #{column_name} FROM #{table_name} WHERE #{column_name} LIKE $1", args) rescue nil
if result&.ntuples > 0
2018-06-07 23:05:41 +08:00
found["#{table_name}.#{column_name}"] = result.map { |r| r[column_name] }
end
end
found
end
end