FIX: Create BaseDropper functions in a different schema.

https://meta.discourse.org/t/error-when-restore-db-backup/93145/25?u=tgxworld
This commit is contained in:
Guo Xiang Tan 2018-08-23 12:51:22 +08:00
parent db05ab1868
commit 212ee15804
3 changed files with 24 additions and 25 deletions

View File

@ -65,24 +65,6 @@ module BackupRestore
BackupRestore.move_tables_between_schemas("public", "backup") BackupRestore.move_tables_between_schemas("public", "backup")
# This is a temp fix to allow restores to work again.
# @tgxworld is currently working on a fix that namespaces functions
# created by Discourse so that we can alter the schema of those
# functions before restoring.
%w{
raise_email_logs_reply_key_readonly
raise_email_logs_skipped_reason_readonly
}.each do |function|
begin
DB.exec(<<~SQL)
DROP FUNCTION IF EXISTS backup.#{function};
ALTER FUNCTION public.#{function} SET SCHEMA "backup";
SQL
rescue PG::UndefinedFunction
# the function does not exist, no need to worry about this
end
end
@db_was_changed = true @db_was_changed = true
restore_dump restore_dump
migrate_database migrate_database

View File

@ -1,5 +1,7 @@
module Migration module Migration
class BaseDropper class BaseDropper
FUNCTION_SCHEMA_NAME = "discourse_functions".freeze
def initialize(after_migration, delay, on_drop, after_drop) def initialize(after_migration, delay, on_drop, after_drop)
@after_migration = after_migration @after_migration = after_migration
@on_drop = on_drop @on_drop = on_drop
@ -46,6 +48,10 @@ module Migration
end end
def self.create_readonly_function(table_name, column_name = nil) def self.create_readonly_function(table_name, column_name = nil)
DB.exec <<~SQL
CREATE SCHEMA IF NOT EXISTS #{FUNCTION_SCHEMA_NAME};
SQL
message = column_name ? message = column_name ?
"Discourse: #{column_name} in #{table_name} is readonly" : "Discourse: #{column_name} in #{table_name} is readonly" :
"Discourse: #{table_name} is read only" "Discourse: #{table_name} is read only"
@ -69,7 +75,14 @@ module Migration
end end
def self.readonly_function_name(table_name, column_name = nil) def self.readonly_function_name(table_name, column_name = nil)
["raise", table_name, column_name, "readonly()"].compact.join("_") function_name = [
"raise",
table_name,
column_name,
"readonly()"
].compact.join("_")
"#{FUNCTION_SCHEMA_NAME}.#{function_name}"
end end
def self.readonly_trigger_name(table_name, column_name = nil) def self.readonly_trigger_name(table_name, column_name = nil)

View File

@ -152,19 +152,23 @@ RSpec.describe Migration::ColumnDropper do
) )
expect(dropped_proc_called).to eq(true) expect(dropped_proc_called).to eq(true)
end end
it 'should prevent updates to the readonly column' do it 'should prevent updates to the readonly column' do
expect do begin
DB.exec <<~SQL DB.exec <<~SQL
UPDATE #{table_name} UPDATE #{table_name}
SET email = 'testing@email.com' SET email = 'testing@email.com'
WHERE topic_id = 1; WHERE topic_id = 1;
SQL SQL
end.to raise_error( rescue PG::RaiseException => e
PG::RaiseException, [
/Discourse: email in #{table_name} is readonly/ "Discourse: email in #{table_name} is readonly",
) 'discourse_functions.raise_table_with_readonly_column_email_readonly()'
].each do |message|
expect(e.message).to include(message)
end
end
end end
it 'should allow updates to the other columns' do it 'should allow updates to the other columns' do