2018-11-29 15:57:57 -05:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-07-22 00:02:18 -04:00
|
|
|
class InitialMigration < ActiveRecord::Migration[5.2]
|
2018-11-29 15:57:57 -05:00
|
|
|
def change
|
2019-07-21 22:56:28 -04:00
|
|
|
reversible do |dir|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
|
|
|
CREATE TYPE sex AS ENUM ('male', 'female');
|
|
|
|
|
|
|
|
CREATE TYPE relationship_status AS ENUM (
|
|
|
|
'supporter',
|
|
|
|
'excluded',
|
|
|
|
'member'
|
|
|
|
);
|
|
|
|
|
|
|
|
CREATE TYPE relationship_role AS ENUM ('manager', 'supervisor');
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
|
|
|
dir.down do
|
|
|
|
execute <<~SQL
|
|
|
|
DROP TYPE sex;
|
|
|
|
DROP TYPE relationship_status;
|
|
|
|
DROP TYPE relationship_role;
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-21 23:52:56 -04:00
|
|
|
create_table :contacts_lists do |t|
|
|
|
|
t.timestamps 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-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-21 23:07:47 -04:00
|
|
|
t.references :federal_subject, null: false, index: { unique: 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
|
|
|
|
t.column :sex, :sex, null: false
|
|
|
|
t.date :date_of_birth, null: false
|
|
|
|
t.string :place_of_birth, null: false
|
2019-07-21 23:52:56 -04:00
|
|
|
|
|
|
|
t.references :contacts_list, null: false, index: { unique: 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
|
|
|
|
|
|
|
|
t.references :person, index: true, foreign_key: true
|
|
|
|
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-07-19 20:40:21 -04:00
|
|
|
t.references :person, index: { unique: true }
|
2019-07-21 23:52:56 -04:00
|
|
|
|
|
|
|
t.references :contacts_list, null: false, index: { unique: 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
|
|
|
|
end
|
|
|
|
|
2018-11-29 15:57:57 -05:00
|
|
|
create_table :users do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
2019-07-19 19:58:29 -04:00
|
|
|
t.references :account, null: false, index: { unique: 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
|
|
|
|
|
|
|
create_table :roles do |t|
|
|
|
|
t.timestamps null: false
|
2019-07-19 20:10:44 -04:00
|
|
|
t.string :name, null: false
|
2019-07-19 19:44:21 -04:00
|
|
|
t.references :resource, polymorphic: true
|
|
|
|
|
|
|
|
t.index %i[name resource_type resource_id], unique: true
|
|
|
|
end
|
|
|
|
|
2019-07-19 19:47:56 -04:00
|
|
|
create_table :account_roles do |t|
|
2019-07-19 19:44:21 -04:00
|
|
|
t.timestamps null: false
|
2019-07-21 23:02:15 -04:00
|
|
|
|
2019-07-21 23:09:23 -04:00
|
|
|
t.references :account, null: false, index: true
|
|
|
|
t.references :role, null: false, index: true
|
2019-07-21 23:02:15 -04:00
|
|
|
|
|
|
|
t.datetime :deleted_at
|
|
|
|
t.datetime :expires_at
|
2019-07-19 19:44:21 -04:00
|
|
|
end
|
|
|
|
|
2019-07-19 20:08:50 -04:00
|
|
|
create_table :user_omniauths do |t|
|
|
|
|
t.timestamps null: false
|
|
|
|
|
|
|
|
t.references :user, foreign_key: true
|
|
|
|
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
|
|
|
|
|
|
|
|
t.references :person,
|
|
|
|
null: false, index: false, foreign_key: true
|
|
|
|
|
|
|
|
t.references :regional_office,
|
|
|
|
null: false, index: true, foreign_key: true
|
|
|
|
|
|
|
|
t.date :from_date, null: false, index: true
|
|
|
|
t.date :until_date, null: true, index: false
|
|
|
|
|
|
|
|
t.column :status, :relationship_status, null: false, index: true
|
|
|
|
t.column :role, :relationship_role, null: true, index: true
|
|
|
|
|
|
|
|
t.index %i[person_id from_date], unique: true
|
|
|
|
end
|
|
|
|
|
2019-07-19 20:45:55 -04:00
|
|
|
add_foreign_key :users, :accounts
|
|
|
|
add_foreign_key :account_roles, :accounts
|
|
|
|
add_foreign_key :account_roles, :roles
|
|
|
|
add_foreign_key :accounts, :people
|
2019-07-21 23:07:47 -04:00
|
|
|
add_foreign_key :regional_offices, :federal_subjects
|
2019-07-21 23:52:56 -04:00
|
|
|
add_foreign_key :accounts, :contacts_lists
|
|
|
|
add_foreign_key :people, :contacts_lists
|
2019-07-21 23:56:55 -04:00
|
|
|
|
|
|
|
reversible do |dir|
|
|
|
|
dir.up do
|
|
|
|
execute <<~SQL
|
|
|
|
ALTER TABLE relationships ADD CONSTRAINT dates CHECK (
|
|
|
|
until_date IS NULL OR from_date < until_date
|
|
|
|
);
|
|
|
|
|
|
|
|
ALTER TABLE relationships ADD CONSTRAINT role CHECK (
|
|
|
|
status = 'member' OR role IS NULL
|
|
|
|
);
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|
2019-07-22 00:01:01 -04:00
|
|
|
|
|
|
|
constraint :accounts, :guest_token, <<~SQL
|
|
|
|
guest_token ~ '^[0-9a-f]{32}$'
|
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :accounts, :nickname, <<~SQL
|
|
|
|
length(nickname) BETWEEN 3 AND 36
|
|
|
|
AND
|
|
|
|
nickname ~ '^[a-z][a-z0-9]*(_[a-z0-9]+)*$'
|
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :accounts, :public_name, <<~SQL
|
|
|
|
public_name IS NULL
|
|
|
|
OR
|
|
|
|
length(public_name) BETWEEN 3 AND 255
|
|
|
|
AND
|
|
|
|
public_name !~ '^[[:space:]]*$'
|
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :accounts, :biography, <<~SQL
|
|
|
|
biography IS NULL
|
|
|
|
OR
|
|
|
|
length(biography) BETWEEN 3 AND 10000
|
|
|
|
AND
|
|
|
|
biography !~ '^[[:space:]]*$'
|
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :federal_subjects, :english_name, <<~SQL
|
|
|
|
length(english_name) BETWEEN 1 AND 255
|
|
|
|
AND
|
|
|
|
english_name !~ '^[[:space:]]{1,}'
|
|
|
|
AND
|
|
|
|
english_name !~ '[[:space:]]{1,}$'
|
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :federal_subjects, :native_name, <<~SQL
|
|
|
|
length(native_name) BETWEEN 1 AND 255
|
|
|
|
AND
|
|
|
|
native_name !~ '^[[:space:]]{1,}'
|
|
|
|
AND
|
|
|
|
native_name !~ '[[:space:]]{1,}$'
|
|
|
|
SQL
|
|
|
|
|
|
|
|
constraint :federal_subjects, :number, 'number > 0'
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
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
|