2018-11-29 15:57:57 -05:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-07-22 00:03:06 -04:00
|
|
|
class InitialMigration < ActiveRecord::Migration[6.0]
|
2018-11-29 15:57:57 -05:00
|
|
|
def change
|
2019-07-25 21:58:07 -04:00
|
|
|
change_types
|
|
|
|
change_functions
|
|
|
|
change_tables
|
2019-08-10 20:51:58 -04:00
|
|
|
change_sequences
|
2019-07-25 21:58:07 -04:00
|
|
|
change_constraints
|
2019-07-29 00:41:53 -04:00
|
|
|
change_triggers
|
2019-07-25 21:58:07 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def change_types
|
2019-07-25 21:56:18 -04:00
|
|
|
enum :sex, %i[male female]
|
|
|
|
|
|
|
|
enum :relationship_status, %i[supporter excluded member]
|
|
|
|
|
2019-07-25 22:06:01 -04:00
|
|
|
enum :relationship_role, %i[
|
|
|
|
federal_manager
|
2019-07-25 22:09:52 -04:00
|
|
|
federal_supervisor
|
2019-07-25 22:06:01 -04:00
|
|
|
regional_manager
|
|
|
|
regional_supervisor
|
|
|
|
]
|
2019-07-25 21:56:18 -04:00
|
|
|
|
2019-07-25 23:22:12 -04:00
|
|
|
enum :relationship_federal_secretary_flag, %i[federal_secretary]
|
|
|
|
|
2019-07-25 23:40:06 -04:00
|
|
|
enum :relationship_regional_secretary_flag, %i[regional_secretary]
|
|
|
|
|
2019-07-25 21:56:18 -04:00
|
|
|
enum :person_comment_origin, %i[
|
|
|
|
general_comments
|
|
|
|
first_contact_date
|
|
|
|
latest_contact_date
|
|
|
|
human_readable_id
|
|
|
|
past_experience
|
|
|
|
aid_at_2014_elections
|
|
|
|
aid_at_2015_elections
|
|
|
|
]
|
2019-07-25 21:58:07 -04:00
|
|
|
end
|
2019-07-25 21:56:18 -04:00
|
|
|
|
2019-07-25 21:58:07 -04:00
|
|
|
def change_functions
|
2019-07-22 16:48:36 -04:00
|
|
|
func :is_guest_token, <<~SQL
|
|
|
|
(str text) RETURNS boolean IMMUTABLE LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
|
|
|
RETURN str ~ '^[0-9a-f]{32}$';
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
|
|
|
|
2019-07-22 16:47:01 -04:00
|
|
|
func :is_nickname, <<~SQL
|
|
|
|
(str text) RETURNS boolean IMMUTABLE LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
|
|
|
RETURN length(str) BETWEEN 3 AND 36
|
|
|
|
AND str ~ '^[a-z][a-z0-9]*(_[a-z0-9]+)*$';
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
|
|
|
|
2019-08-17 19:51:03 -04:00
|
|
|
func :is_codename, <<~SQL
|
|
|
|
(str text) RETURNS boolean IMMUTABLE LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
|
|
|
RETURN is_nickname(str);
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
|
|
|
|
2019-07-22 16:35:22 -04:00
|
|
|
func :is_good_text, <<~SQL
|
2019-07-22 16:33:41 -04:00
|
|
|
(str text) RETURNS boolean IMMUTABLE LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
|
|
|
RETURN str ~ '^[^[:space:]]+(.*[^[:space:]])?$';
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
2019-07-22 16:31:19 -04:00
|
|
|
|
2019-07-22 16:42:56 -04:00
|
|
|
func :is_good_limited_text, <<~SQL
|
2019-07-22 16:57:17 -04:00
|
|
|
(str text, max_length integer)
|
2019-07-22 16:42:56 -04:00
|
|
|
RETURNS boolean IMMUTABLE LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
2019-07-22 16:57:17 -04:00
|
|
|
RETURN LENGTH(str) BETWEEN 1 AND max_length AND is_good_text(str);
|
2019-07-22 16:42:56 -04:00
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
|
|
|
|
2019-07-22 17:00:05 -04:00
|
|
|
func :is_good_small_text, <<~SQL
|
|
|
|
(str text) RETURNS boolean IMMUTABLE LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
|
|
|
RETURN is_good_limited_text(str, 255);
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
|
|
|
|
|
|
|
func :is_good_big_text, <<~SQL
|
|
|
|
(str text) RETURNS boolean IMMUTABLE LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
|
|
|
RETURN is_good_limited_text(str, 10000);
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
2019-07-29 00:41:53 -04:00
|
|
|
|
2019-08-11 15:27:06 -04:00
|
|
|
func :ensure_superuser_has_related_user, <<~SQL
|
|
|
|
() RETURNS trigger LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
DECLARE
|
|
|
|
user record;
|
|
|
|
BEGIN
|
|
|
|
IF NOT NEW.superuser THEN
|
|
|
|
RETURN NEW;
|
|
|
|
END IF;
|
|
|
|
|
|
|
|
SELECT * FROM users INTO user WHERE users.account_id = NEW.id;
|
|
|
|
|
|
|
|
IF user IS NULL THEN
|
|
|
|
RAISE EXCEPTION 'does not have related user';
|
|
|
|
END IF;
|
|
|
|
|
|
|
|
RETURN NEW;
|
|
|
|
END;
|
|
|
|
$$
|
|
|
|
SQL
|
|
|
|
|
2019-07-29 01:36:39 -04:00
|
|
|
func :ensure_contact_list_id_remains_unchanged, <<~SQL
|
2019-07-29 00:41:53 -04:00
|
|
|
() RETURNS trigger LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
BEGIN
|
2019-07-29 01:36:39 -04:00
|
|
|
IF NEW.contact_list_id IS DISTINCT FROM OLD.contact_list_id THEN
|
|
|
|
RAISE EXCEPTION 'can not change column "contact_list_id"';
|
2019-07-29 00:41:53 -04:00
|
|
|
END IF;
|
|
|
|
|
|
|
|
RETURN NEW;
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
2019-07-29 01:26:00 -04:00
|
|
|
|
2019-07-29 01:36:39 -04:00
|
|
|
func :ensure_contact_list_id_matches_related_person, <<~SQL
|
2019-07-29 01:26:00 -04:00
|
|
|
() RETURNS trigger LANGUAGE plpgsql AS
|
|
|
|
$$
|
|
|
|
DECLARE
|
|
|
|
person record;
|
|
|
|
BEGIN
|
|
|
|
IF NEW.person_id IS NULL THEN
|
|
|
|
RETURN NEW;
|
|
|
|
END IF;
|
|
|
|
|
|
|
|
SELECT * FROM people INTO person WHERE people.id = NEW.person_id;
|
|
|
|
|
|
|
|
IF person IS NULL THEN
|
|
|
|
RETURN NEW;
|
|
|
|
END IF;
|
|
|
|
|
2019-07-29 01:36:39 -04:00
|
|
|
IF NEW.contact_list_id IS DISTINCT FROM person.contact_list_id THEN
|
2019-07-29 01:26:00 -04:00
|
|
|
RAISE EXCEPTION
|
2019-07-29 01:36:39 -04:00
|
|
|
'column "contact_list_id" does not match related person';
|
2019-07-29 01:26:00 -04:00
|
|
|
END IF;
|
|
|
|
|
|
|
|
RETURN NEW;
|
|
|
|
END;
|
|
|
|
$$;
|
|
|
|
SQL
|
2019-07-25 21:58:07 -04:00
|
|
|
end
|
2019-07-22 17:00:05 -04:00
|
|
|
|
2019-07-25 21:58:07 -04:00
|
|
|
def change_tables
|
2019-08-04 17:05:57 -04:00
|
|
|
create_table :contact_networks do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
2019-08-17 20:00:37 -04:00
|
|
|
t.string :codename, null: false, index: { unique: true }
|
|
|
|
t.string :name, null: false, index: { unique: true }
|
2019-08-04 17:05:57 -04:00
|
|
|
end
|
|
|
|
|
2019-07-29 01:36:39 -04:00
|
|
|
create_table :contact_lists do |t|
|
2019-07-21 23:52:56 -04:00
|
|
|
t.timestamps null: false
|
|
|
|
end
|
|
|
|
|
2019-08-10 20:23:12 -04:00
|
|
|
create_table :contacts do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
|
|
|
t.references :contact_list, null: false, index: true, foreign_key: true
|
|
|
|
t.references :contact_network, null: false, index: true, foreign_key: true
|
|
|
|
|
|
|
|
t.string :value, null: false
|
|
|
|
end
|
|
|
|
|
2019-07-21 23:07:47 -04:00
|
|
|
create_table :federal_subjects do |t|
|
2019-07-19 20:41:59 -04:00
|
|
|
t.timestamps null: false
|
2019-07-19 20:45:55 -04:00
|
|
|
|
2019-07-21 23:04:38 -04:00
|
|
|
t.string :english_name, null: false, index: { unique: true }
|
|
|
|
t.string :native_name, null: false, index: { unique: true }
|
2019-07-22 05:14:14 -04:00
|
|
|
t.string :centre, null: false, index: false
|
2019-07-21 23:59:29 -04:00
|
|
|
|
|
|
|
t.integer :number, null: false, index: { unique: true }
|
|
|
|
t.interval :timezone, null: false, index: false
|
2019-07-19 20:41:59 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
create_table :regional_offices do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
2019-07-22 00:09:41 -04:00
|
|
|
t.references :federal_subject,
|
|
|
|
null: false, index: { unique: true }, foreign_key: true
|
2019-07-19 20:41:59 -04:00
|
|
|
end
|
|
|
|
|
2019-07-19 20:38:41 -04:00
|
|
|
create_table :people do |t|
|
|
|
|
t.timestamps null: false
|
2019-07-21 22:58:07 -04:00
|
|
|
|
2019-07-21 23:06:07 -04:00
|
|
|
t.string :first_name, null: false
|
|
|
|
t.string :middle_name, null: true
|
|
|
|
t.string :last_name, null: false
|
2019-07-25 17:15:40 -04:00
|
|
|
t.column :sex, :sex, null: true
|
|
|
|
t.date :date_of_birth, null: true
|
|
|
|
t.string :place_of_birth, null: true
|
2019-07-21 23:52:56 -04:00
|
|
|
|
2019-07-29 01:36:39 -04:00
|
|
|
t.references :contact_list,
|
2019-07-22 00:09:41 -04:00
|
|
|
null: false, index: { unique: true }, foreign_key: true
|
2019-07-19 20:38:41 -04:00
|
|
|
end
|
|
|
|
|
2019-07-21 23:55:54 -04:00
|
|
|
create_table :passports do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
|
|
|
t.string :last_name, null: false
|
|
|
|
t.string :first_name, null: false
|
|
|
|
t.string :middle_name
|
|
|
|
t.column :sex, :sex, null: false
|
|
|
|
t.date :date_of_birth, null: false
|
|
|
|
t.string :place_of_birth, null: false
|
|
|
|
t.integer :series, null: false
|
|
|
|
t.integer :number, null: false
|
|
|
|
t.text :issued_by, null: false
|
|
|
|
t.string :unit_code, null: false
|
|
|
|
t.date :date_of_issue, null: false
|
|
|
|
|
2019-07-22 09:52:23 -04:00
|
|
|
t.references :person, index: true, foreign_key: true
|
|
|
|
t.references :federal_subject, index: true, foreign_key: true
|
|
|
|
|
|
|
|
t.string :zip_code
|
|
|
|
|
|
|
|
t.string :town_type
|
|
|
|
t.string :town_name
|
|
|
|
t.string :settlement_type
|
|
|
|
t.string :settlement_name
|
|
|
|
t.string :district_type
|
|
|
|
t.string :district_name
|
|
|
|
t.string :street_type
|
|
|
|
t.string :street_name
|
|
|
|
t.string :residence_type
|
|
|
|
t.string :residence_name
|
|
|
|
t.string :building_type
|
|
|
|
t.string :building_name
|
|
|
|
t.string :apartment_type
|
|
|
|
t.string :apartment_name
|
2019-07-21 23:55:54 -04:00
|
|
|
end
|
|
|
|
|
2019-07-19 19:47:56 -04:00
|
|
|
create_table :accounts do |t|
|
|
|
|
t.timestamps null: false
|
2019-07-19 20:37:28 -04:00
|
|
|
|
2019-07-19 20:49:59 -04:00
|
|
|
t.string :guest_token, null: false, index: { unique: true }
|
2019-07-21 22:38:23 -04:00
|
|
|
t.string :nickname, null: false, index: { unique: true }
|
2019-07-19 20:37:28 -04:00
|
|
|
|
2019-07-21 23:00:42 -04:00
|
|
|
t.string :public_name
|
|
|
|
t.text :biography
|
2019-07-21 22:59:35 -04:00
|
|
|
|
2019-08-11 15:27:06 -04:00
|
|
|
t.boolean :superuser, null: false, default: false
|
|
|
|
|
2019-07-22 00:09:41 -04:00
|
|
|
t.references :person, index: { unique: true }, foreign_key: true
|
2019-07-21 23:52:56 -04:00
|
|
|
|
2019-07-29 01:36:39 -04:00
|
|
|
t.references :contact_list,
|
2019-07-22 00:09:41 -04:00
|
|
|
null: false, index: { unique: true }, foreign_key: true
|
2019-07-19 19:47:56 -04:00
|
|
|
end
|
|
|
|
|
2019-07-21 23:54:11 -04:00
|
|
|
create_table :person_comments do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
|
|
|
t.references :person, null: false, index: true, foreign_key: true
|
|
|
|
t.references :account, null: true, index: true, foreign_key: true
|
|
|
|
|
|
|
|
t.text :text, null: false
|
2019-07-22 13:34:33 -04:00
|
|
|
t.column :origin, :person_comment_origin
|
2019-07-21 23:54:11 -04:00
|
|
|
end
|
|
|
|
|
2018-11-29 15:57:57 -05:00
|
|
|
create_table :users do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
2019-07-22 00:09:41 -04:00
|
|
|
t.references :account,
|
|
|
|
null: false, index: { unique: true }, foreign_key: true
|
2019-07-19 19:53:05 -04:00
|
|
|
|
2018-11-29 15:57:57 -05:00
|
|
|
## Database authenticatable
|
|
|
|
t.string :email, null: false, default: ''
|
|
|
|
t.string :encrypted_password, null: false, default: ''
|
|
|
|
|
|
|
|
## Recoverable
|
|
|
|
t.string :reset_password_token
|
|
|
|
t.datetime :reset_password_sent_at
|
|
|
|
|
|
|
|
## Rememberable
|
|
|
|
t.datetime :remember_created_at
|
|
|
|
|
|
|
|
## Trackable
|
|
|
|
t.integer :sign_in_count, default: 0, null: false
|
|
|
|
t.datetime :current_sign_in_at
|
|
|
|
t.datetime :last_sign_in_at
|
|
|
|
t.inet :current_sign_in_ip
|
|
|
|
t.inet :last_sign_in_ip
|
|
|
|
|
|
|
|
## Confirmable
|
|
|
|
t.string :confirmation_token
|
|
|
|
t.datetime :confirmed_at
|
|
|
|
t.datetime :confirmation_sent_at
|
|
|
|
t.string :unconfirmed_email
|
|
|
|
|
|
|
|
## Lockable
|
|
|
|
t.integer :failed_attempts, default: 0, null: false
|
|
|
|
t.string :unlock_token
|
|
|
|
t.datetime :locked_at
|
2019-07-19 19:53:05 -04:00
|
|
|
|
2019-07-19 19:55:57 -04:00
|
|
|
t.index :email, unique: true
|
|
|
|
t.index :reset_password_token, unique: true
|
|
|
|
t.index :confirmation_token, unique: true
|
|
|
|
t.index :unlock_token, unique: true
|
2019-07-19 19:53:55 -04:00
|
|
|
end
|
2019-07-19 19:44:21 -04:00
|
|
|
|
2019-07-19 20:08:50 -04:00
|
|
|
create_table :user_omniauths do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
2019-07-22 00:09:41 -04:00
|
|
|
t.references :user, index: true, foreign_key: true
|
|
|
|
|
2019-07-19 20:08:50 -04:00
|
|
|
t.string :provider, null: false
|
|
|
|
t.string :remote_id, null: false
|
|
|
|
t.string :email, null: false
|
|
|
|
|
|
|
|
t.index %i[remote_id provider], unique: true
|
|
|
|
end
|
|
|
|
|
2019-07-21 23:56:55 -04:00
|
|
|
create_table :relationships do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
2019-08-14 22:59:11 -04:00
|
|
|
t.references :person, null: false,
|
|
|
|
index: false,
|
|
|
|
foreign_key: true
|
2019-07-21 23:56:55 -04:00
|
|
|
|
2019-08-14 22:59:11 -04:00
|
|
|
t.references :regional_office, null: false,
|
|
|
|
index: true,
|
|
|
|
foreign_key: true
|
|
|
|
|
|
|
|
t.references :initiator_account, null: true,
|
|
|
|
index: true,
|
|
|
|
foreign_key: { to_table: :accounts }
|
2019-07-21 23:56:55 -04:00
|
|
|
|
2019-08-14 22:46:13 -04:00
|
|
|
t.date :from_date, null: false, index: true
|
2019-07-21 23:56:55 -04:00
|
|
|
|
|
|
|
t.column :status, :relationship_status, null: false, index: true
|
|
|
|
t.column :role, :relationship_role, null: true, index: true
|
|
|
|
|
2019-07-25 23:22:12 -04:00
|
|
|
t.column :federal_secretary_flag,
|
|
|
|
:relationship_federal_secretary_flag,
|
|
|
|
null: true,
|
|
|
|
index: { unique: true }
|
|
|
|
|
2019-07-25 23:40:06 -04:00
|
|
|
t.column :regional_secretary_flag,
|
|
|
|
:relationship_regional_secretary_flag,
|
|
|
|
null: true,
|
|
|
|
index: true
|
|
|
|
|
2019-07-21 23:56:55 -04:00
|
|
|
t.index %i[person_id from_date], unique: true
|
2019-07-25 23:40:06 -04:00
|
|
|
|
|
|
|
t.index(
|
|
|
|
%i[regional_office_id regional_secretary_flag],
|
|
|
|
name: :index_relationships_on_regional_office_id_and_secretary_flag,
|
|
|
|
unique: true,
|
|
|
|
)
|
2019-07-21 23:56:55 -04:00
|
|
|
end
|
2019-07-25 21:58:07 -04:00
|
|
|
end
|
2019-07-21 23:56:55 -04:00
|
|
|
|
2019-08-10 20:51:58 -04:00
|
|
|
def change_sequences
|
|
|
|
reversible do |dir|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
2019-08-14 09:54:27 -04:00
|
|
|
ALTER SEQUENCE contact_networks_id_seq RESTART WITH 100;
|
|
|
|
ALTER SEQUENCE federal_subjects_id_seq RESTART WITH 100;
|
|
|
|
ALTER SEQUENCE contacts_id_seq RESTART WITH 4000;
|
|
|
|
ALTER SEQUENCE people_id_seq RESTART WITH 3000;
|
2019-08-10 20:51:58 -04:00
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-25 21:58:07 -04:00
|
|
|
def change_constraints
|
2019-08-10 20:23:12 -04:00
|
|
|
constraint :contacts, :value, <<~SQL
|
|
|
|
is_good_small_text(value)
|
|
|
|
SQL
|
|
|
|
|
2019-08-17 19:51:03 -04:00
|
|
|
constraint :contact_networks, :codename, <<~SQL
|
|
|
|
is_codename(codename)
|
2019-08-04 17:05:57 -04:00
|
|
|
SQL
|
|
|
|
|
2019-08-17 20:00:37 -04:00
|
|
|
constraint :contact_networks, :name, <<~SQL
|
|
|
|
is_good_small_text(name)
|
2019-08-04 17:05:57 -04:00
|
|
|
SQL
|
|
|
|
|
2019-07-22 00:04:26 -04:00
|
|
|
constraint :relationships, :role, <<~SQL
|
|
|
|
status = 'member' OR role IS NULL
|
|
|
|
SQL
|
2019-07-22 00:01:01 -04:00
|
|
|
|
2019-07-25 23:22:12 -04:00
|
|
|
constraint :relationships, :federal_secretary_flag, <<~SQL
|
|
|
|
federal_secretary_flag IS NULL OR role = 'federal_manager'
|
|
|
|
SQL
|
|
|
|
|
2019-07-25 23:40:06 -04:00
|
|
|
constraint :relationships, :regional_secretary_flag, <<~SQL
|
|
|
|
regional_secretary_flag IS NULL OR role = 'regional_manager'
|
|
|
|
SQL
|
|
|
|
|
2019-07-22 00:01:01 -04:00
|
|
|
constraint :accounts, :guest_token, <<~SQL
|
2019-07-22 16:48:36 -04:00
|
|
|
is_guest_token(guest_token)
|
2019-07-22 00:01:01 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :accounts, :nickname, <<~SQL
|
2019-07-22 16:47:01 -04:00
|
|
|
is_nickname(nickname)
|
2019-07-22 00:01:01 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :accounts, :public_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
public_name IS NULL OR is_good_small_text(public_name)
|
2019-07-22 00:01:01 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :accounts, :biography, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
biography IS NULL OR is_good_big_text(biography)
|
2019-07-22 00:01:01 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :federal_subjects, :english_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
is_good_small_text(english_name)
|
2019-07-22 00:01:01 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :federal_subjects, :native_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
is_good_small_text(native_name)
|
2019-07-22 00:01:01 -04:00
|
|
|
SQL
|
|
|
|
|
2019-07-22 05:14:14 -04:00
|
|
|
constraint :federal_subjects, :centre, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
is_good_small_text(centre)
|
2019-07-22 05:14:14 -04:00
|
|
|
SQL
|
|
|
|
|
2019-07-22 00:04:26 -04:00
|
|
|
constraint :federal_subjects, :number, <<~SQL
|
|
|
|
number > 0
|
|
|
|
SQL
|
2019-07-22 09:52:23 -04:00
|
|
|
|
|
|
|
constraint :passports, :zip_code, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
zip_code IS NULL OR is_good_small_text(zip_code)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :town_type, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
town_type IS NULL OR is_good_small_text(town_type)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :town_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
town_name IS NULL OR is_good_small_text(town_name)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :settlement_type, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
settlement_type IS NULL OR is_good_small_text(settlement_type)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :settlement_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
settlement_name IS NULL OR is_good_small_text(settlement_name)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :district_type, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
district_type IS NULL OR is_good_small_text(district_type)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :district_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
district_name IS NULL OR is_good_small_text(district_name)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :street_type, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
street_type IS NULL OR is_good_small_text(street_type)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :street_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
street_name IS NULL OR is_good_small_text(street_name)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :residence_type, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
residence_type IS NULL OR is_good_small_text(residence_type)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :residence_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
residence_name IS NULL OR is_good_small_text(residence_name)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :building_type, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
building_type IS NULL OR is_good_small_text(building_type)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :building_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
building_name IS NULL OR is_good_small_text(building_name)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :apartment_type, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
apartment_type IS NULL OR is_good_small_text(apartment_type)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :passports, :apartment_name, <<~SQL
|
2019-07-22 17:00:05 -04:00
|
|
|
apartment_name IS NULL OR is_good_small_text(apartment_name)
|
2019-07-22 09:52:23 -04:00
|
|
|
SQL
|
2019-07-22 00:01:01 -04:00
|
|
|
end
|
|
|
|
|
2019-07-29 00:41:53 -04:00
|
|
|
def change_triggers
|
2019-08-11 15:27:06 -04:00
|
|
|
reversible do |dir|
|
|
|
|
dir.down do
|
|
|
|
execute 'DROP TRIGGER ensure_superuser_has_related_user ON accounts;'
|
|
|
|
end
|
|
|
|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
|
|
|
CREATE TRIGGER ensure_superuser_has_related_user
|
|
|
|
BEFORE INSERT OR UPDATE
|
|
|
|
ON accounts
|
|
|
|
FOR EACH ROW
|
|
|
|
EXECUTE PROCEDURE ensure_superuser_has_related_user();
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-29 00:41:53 -04:00
|
|
|
reversible do |dir|
|
|
|
|
dir.down do
|
|
|
|
execute <<~SQL
|
2019-07-29 01:36:39 -04:00
|
|
|
DROP TRIGGER ensure_contact_list_id_remains_unchanged
|
2019-07-29 00:41:53 -04:00
|
|
|
ON people;
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
2019-07-29 01:36:39 -04:00
|
|
|
CREATE TRIGGER ensure_contact_list_id_remains_unchanged
|
|
|
|
BEFORE UPDATE OF contact_list_id
|
2019-07-29 00:41:53 -04:00
|
|
|
ON people
|
|
|
|
FOR EACH ROW
|
2019-07-29 01:36:39 -04:00
|
|
|
EXECUTE PROCEDURE ensure_contact_list_id_remains_unchanged();
|
2019-07-29 00:41:53 -04:00
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|
2019-07-29 01:26:00 -04:00
|
|
|
|
|
|
|
reversible do |dir|
|
|
|
|
dir.down do
|
|
|
|
execute <<~SQL
|
2019-07-29 01:36:39 -04:00
|
|
|
DROP TRIGGER ensure_contact_list_id_matches_related_person
|
2019-07-29 01:26:00 -04:00
|
|
|
ON accounts;
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
2019-07-29 01:36:39 -04:00
|
|
|
CREATE TRIGGER ensure_contact_list_id_matches_related_person
|
|
|
|
BEFORE INSERT OR UPDATE OF person_id, contact_list_id
|
2019-07-29 01:26:00 -04:00
|
|
|
ON accounts
|
|
|
|
FOR EACH ROW
|
2019-07-29 01:36:39 -04:00
|
|
|
EXECUTE PROCEDURE ensure_contact_list_id_matches_related_person();
|
2019-07-29 01:26:00 -04:00
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|
2019-07-29 00:41:53 -04:00
|
|
|
end
|
|
|
|
|
2019-07-22 16:33:41 -04:00
|
|
|
def func(name, sql)
|
|
|
|
reversible do |dir|
|
|
|
|
dir.up do
|
|
|
|
execute "CREATE FUNCTION #{name} #{sql}"
|
|
|
|
end
|
|
|
|
|
|
|
|
dir.down do
|
|
|
|
execute "DROP FUNCTION #{name}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-22 00:29:48 -04:00
|
|
|
def enum(name, values)
|
|
|
|
reversible do |dir|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
|
|
|
CREATE TYPE #{name}
|
|
|
|
AS ENUM (#{values.map { |s| "'#{s}'" }.join(', ')})
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
|
|
|
dir.down do
|
|
|
|
execute "DROP TYPE #{name}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-22 00:01:01 -04:00
|
|
|
def constraint(table, name, check)
|
|
|
|
reversible do |dir|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
|
|
|
ALTER TABLE #{table} ADD CONSTRAINT #{name} CHECK (#{check})
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
|
|
|
dir.down do
|
|
|
|
execute <<~SQL
|
|
|
|
ALTER TABLE #{table} DROP CONSTRAINT #{name}
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|
2018-11-29 15:57:57 -05:00
|
|
|
end
|
|
|
|
end
|