-
Notifications
You must be signed in to change notification settings - Fork 31
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #148 from getAlby/feat/manual-migrations
feat: add manual migrations using gormigration
- Loading branch information
Showing
12 changed files
with
361 additions
and
24 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
package migrations | ||
|
||
import ( | ||
_ "embed" | ||
"log" | ||
|
||
"github.com/go-gormigrate/gormigrate/v2" | ||
"gorm.io/gorm" | ||
) | ||
|
||
//go:embed initial_migration_postgres.sql | ||
var initialMigrationPostgres string | ||
//go:embed initial_migration_sqlite.sql | ||
var initialMigrationSqlite string | ||
|
||
var initialMigrations = map[string]string { | ||
"postgres": initialMigrationPostgres, | ||
"sqlite": initialMigrationSqlite, | ||
} | ||
|
||
// Initial migration | ||
var _202309271616_initial_migration = &gormigrate.Migration { | ||
ID: "202309271616_initial_migration", | ||
Migrate: func(tx *gorm.DB) error { | ||
// only execute migration if apps table doesn't exist | ||
err := tx.Exec("SELECT * FROM apps").Error; | ||
if err != nil { | ||
// find which initial migration should be executed | ||
initialMigration := initialMigrations[tx.Dialector.Name()] | ||
if initialMigration == "" { | ||
log.Fatalf("unsupported database type: %s", tx.Dialector.Name()) | ||
} | ||
|
||
return tx.Exec(initialMigration).Error | ||
} | ||
|
||
return nil | ||
}, | ||
Rollback: func(tx *gorm.DB) error { | ||
return nil; | ||
}, | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
package migrations | ||
|
||
import ( | ||
"github.com/go-gormigrate/gormigrate/v2" | ||
"gorm.io/gorm" | ||
) | ||
|
||
// Update payments with preimage as an empty string to use NULL instead | ||
var _202309271617_fix_preimage_null = &gormigrate.Migration { | ||
ID: "202309271617_fix_preimage_null", | ||
Migrate: func(tx *gorm.DB) error { | ||
return tx.Table("payments").Where("preimage = ?", "").Update("preimage", nil).Error; | ||
}, | ||
Rollback: func(tx *gorm.DB) error { | ||
return nil; | ||
}, | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
package migrations | ||
|
||
import ( | ||
"log" | ||
|
||
"github.com/go-gormigrate/gormigrate/v2" | ||
"gorm.io/gorm" | ||
) | ||
|
||
// Create a composite index to improve performance of summing payments in the current budget period | ||
var _202309271618_add_payment_sum_index = &gormigrate.Migration { | ||
ID: "202309271618_add_payment_sum_index", | ||
Migrate: func(tx *gorm.DB) error { | ||
|
||
var sql string | ||
if tx.Dialector.Name() == "postgres" { | ||
sql = "CREATE INDEX idx_payment_sum ON payments USING btree (app_id, preimage, created_at) INCLUDE(amount)" | ||
} else if tx.Dialector.Name() == "sqlite" { | ||
sql = "CREATE INDEX idx_payment_sum ON payments (app_id, preimage, created_at)" | ||
} else { | ||
log.Fatalf("unsupported database type: %s", tx.Dialector.Name()) | ||
} | ||
|
||
return tx.Exec(sql).Error | ||
}, | ||
Rollback: func(tx *gorm.DB) error { | ||
return nil; | ||
}, | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
# Creating a new migration | ||
|
||
1. Create a new file based on the current date and time (see existing migration format) | ||
2. Copy the following code and update MY_ID_HERE and MY_COMMENT_HERE and DO_SOMETHING_HERE | ||
3. Add the ID to the list of migrations in migrate.go | ||
4. If possible, add a rollback function. | ||
|
||
*For Postgres/Sqlite specific migrations, see the [initial migration](202309271616.go)* | ||
|
||
```go | ||
package migrations | ||
|
||
import ( | ||
"github.com/go-gormigrate/gormigrate/v2" | ||
"gorm.io/gorm" | ||
) | ||
|
||
// MY_COMMENT_HERE | ||
var _MY_ID_HERE = &gormigrate.Migration { | ||
ID: "MY_ID_HERE", | ||
Migrate: func(tx *gorm.DB) error { | ||
return DO_SOMETHING_HERE.Error; | ||
}, | ||
Rollback: func(tx *gorm.DB) error { | ||
return nil; | ||
}, | ||
} | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
CREATE TABLE app_permissions ( | ||
id bigint NOT NULL, | ||
app_id bigint, | ||
request_method text, | ||
max_amount bigint, | ||
budget_renewal text, | ||
expires_at timestamp with time zone, | ||
created_at timestamp with time zone, | ||
updated_at timestamp with time zone | ||
); | ||
|
||
CREATE SEQUENCE app_permissions_id_seq | ||
START WITH 1 | ||
INCREMENT BY 1 | ||
NO MINVALUE | ||
NO MAXVALUE | ||
CACHE 1; | ||
|
||
ALTER SEQUENCE app_permissions_id_seq OWNED BY app_permissions.id; | ||
|
||
CREATE TABLE apps ( | ||
id bigint NOT NULL, | ||
user_id bigint, | ||
name text, | ||
description text, | ||
nostr_pubkey text, | ||
created_at timestamp with time zone, | ||
updated_at timestamp with time zone | ||
); | ||
|
||
CREATE SEQUENCE apps_id_seq | ||
START WITH 1 | ||
INCREMENT BY 1 | ||
NO MINVALUE | ||
NO MAXVALUE | ||
CACHE 1; | ||
|
||
ALTER SEQUENCE apps_id_seq OWNED BY apps.id; | ||
|
||
CREATE TABLE identities ( | ||
id bigint NOT NULL, | ||
created_at timestamp with time zone, | ||
updated_at timestamp with time zone, | ||
deleted_at timestamp with time zone, | ||
privkey text | ||
); | ||
|
||
CREATE SEQUENCE identities_id_seq | ||
START WITH 1 | ||
INCREMENT BY 1 | ||
NO MINVALUE | ||
NO MAXVALUE | ||
CACHE 1; | ||
|
||
ALTER SEQUENCE identities_id_seq OWNED BY identities.id; | ||
|
||
CREATE TABLE nostr_events ( | ||
id bigint NOT NULL, | ||
app_id bigint, | ||
nostr_id text, | ||
reply_id text, | ||
content text, | ||
state text, | ||
replied_at timestamp with time zone, | ||
created_at timestamp with time zone, | ||
updated_at timestamp with time zone | ||
); | ||
|
||
CREATE SEQUENCE nostr_events_id_seq | ||
START WITH 1 | ||
INCREMENT BY 1 | ||
NO MINVALUE | ||
NO MAXVALUE | ||
CACHE 1; | ||
|
||
ALTER SEQUENCE nostr_events_id_seq OWNED BY nostr_events.id; | ||
|
||
CREATE TABLE payments ( | ||
id bigint NOT NULL, | ||
app_id bigint, | ||
nostr_event_id bigint, | ||
amount bigint, | ||
payment_request text, | ||
preimage text, | ||
created_at timestamp with time zone, | ||
updated_at timestamp with time zone | ||
); | ||
|
||
CREATE SEQUENCE payments_id_seq | ||
START WITH 1 | ||
INCREMENT BY 1 | ||
NO MINVALUE | ||
NO MAXVALUE | ||
CACHE 1; | ||
|
||
ALTER SEQUENCE payments_id_seq OWNED BY payments.id; | ||
|
||
CREATE TABLE users ( | ||
id bigint NOT NULL, | ||
alby_identifier text, | ||
access_token text, | ||
refresh_token text, | ||
email text, | ||
expiry timestamp with time zone, | ||
lightning_address text, | ||
created_at timestamp with time zone, | ||
updated_at timestamp with time zone | ||
); | ||
|
||
CREATE SEQUENCE users_id_seq | ||
START WITH 1 | ||
INCREMENT BY 1 | ||
NO MINVALUE | ||
NO MAXVALUE | ||
CACHE 1; | ||
|
||
ALTER SEQUENCE users_id_seq OWNED BY users.id; | ||
|
||
ALTER TABLE ONLY app_permissions ALTER COLUMN id SET DEFAULT nextval('app_permissions_id_seq'::regclass); | ||
|
||
ALTER TABLE ONLY apps ALTER COLUMN id SET DEFAULT nextval('apps_id_seq'::regclass); | ||
|
||
ALTER TABLE ONLY identities ALTER COLUMN id SET DEFAULT nextval('identities_id_seq'::regclass); | ||
|
||
ALTER TABLE ONLY nostr_events ALTER COLUMN id SET DEFAULT nextval('nostr_events_id_seq'::regclass); | ||
|
||
ALTER TABLE ONLY payments ALTER COLUMN id SET DEFAULT nextval('payments_id_seq'::regclass); | ||
|
||
ALTER TABLE ONLY users ALTER COLUMN id SET DEFAULT nextval('users_id_seq'::regclass); | ||
|
||
ALTER TABLE ONLY app_permissions | ||
ADD CONSTRAINT app_permissions_pkey PRIMARY KEY (id); | ||
|
||
ALTER TABLE ONLY apps | ||
ADD CONSTRAINT apps_pkey PRIMARY KEY (id); | ||
|
||
ALTER TABLE ONLY identities | ||
ADD CONSTRAINT identities_pkey PRIMARY KEY (id); | ||
|
||
ALTER TABLE ONLY nostr_events | ||
ADD CONSTRAINT nostr_events_pkey PRIMARY KEY (id); | ||
|
||
ALTER TABLE ONLY payments | ||
ADD CONSTRAINT payments_pkey PRIMARY KEY (id); | ||
|
||
ALTER TABLE ONLY users | ||
ADD CONSTRAINT users_pkey PRIMARY KEY (id); | ||
|
||
CREATE INDEX idx_app_permissions_app_id ON app_permissions USING btree (app_id); | ||
|
||
CREATE INDEX idx_app_permissions_request_method ON app_permissions USING btree (request_method); | ||
|
||
CREATE INDEX idx_apps_nostr_pubkey ON apps USING btree (nostr_pubkey); | ||
|
||
CREATE INDEX idx_apps_user_id ON apps USING btree (user_id); | ||
|
||
CREATE INDEX idx_identities_deleted_at ON identities USING btree (deleted_at); | ||
|
||
CREATE INDEX idx_nostr_events_app_id ON nostr_events USING btree (app_id); | ||
|
||
CREATE UNIQUE INDEX idx_nostr_events_nostr_id ON nostr_events USING btree (nostr_id); | ||
|
||
CREATE INDEX idx_payments_app_id ON payments USING btree (app_id); | ||
|
||
CREATE INDEX idx_payments_nostr_event_id ON payments USING btree (nostr_event_id); | ||
|
||
CREATE UNIQUE INDEX idx_users_alby_identifier ON users USING btree (alby_identifier); | ||
|
||
ALTER TABLE ONLY app_permissions | ||
ADD CONSTRAINT fk_app_permissions_app FOREIGN KEY (app_id) REFERENCES apps(id) ON DELETE CASCADE; | ||
|
||
ALTER TABLE ONLY nostr_events | ||
ADD CONSTRAINT fk_nostr_events_app FOREIGN KEY (app_id) REFERENCES apps(id) ON DELETE CASCADE; | ||
|
||
ALTER TABLE ONLY payments | ||
ADD CONSTRAINT fk_payments_app FOREIGN KEY (app_id) REFERENCES apps(id) ON DELETE CASCADE; | ||
|
||
ALTER TABLE ONLY payments | ||
ADD CONSTRAINT fk_payments_nostr_event FOREIGN KEY (nostr_event_id) REFERENCES nostr_events(id); | ||
|
||
ALTER TABLE ONLY apps | ||
ADD CONSTRAINT fk_users_apps FOREIGN KEY (user_id) REFERENCES users(id); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
CREATE TABLE `apps` (`id` integer,`user_id` integer,`name` text,`description` text,`nostr_pubkey` text,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_users_apps` FOREIGN KEY (`user_id`) REFERENCES `users`(`id`)); | ||
CREATE INDEX `idx_apps_nostr_pubkey` ON `apps`(`nostr_pubkey`); | ||
CREATE INDEX `idx_apps_user_id` ON `apps`(`user_id`); | ||
CREATE TABLE `app_permissions` (`id` integer,`app_id` integer,`request_method` text,`max_amount` integer,`budget_renewal` text,`expires_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_app_permissions_app` FOREIGN KEY (`app_id`) REFERENCES `apps`(`id`) ON DELETE CASCADE); | ||
CREATE INDEX `idx_app_permissions_request_method` ON `app_permissions`(`request_method`); | ||
CREATE INDEX `idx_app_permissions_app_id` ON `app_permissions`(`app_id`); | ||
CREATE TABLE `payments` (`id` integer,`app_id` integer,`nostr_event_id` integer,`amount` integer,`payment_request` text,`preimage` text,`created_at` datetime,`updated_at` datetime, `preimage2` text,PRIMARY KEY (`id`),CONSTRAINT `fk_payments_app` FOREIGN KEY (`app_id`) REFERENCES `apps`(`id`) ON DELETE CASCADE,CONSTRAINT `fk_payments_nostr_event` FOREIGN KEY (`nostr_event_id`) REFERENCES `nostr_events`(`id`)); | ||
CREATE INDEX `idx_payments_nostr_event_id` ON `payments`(`nostr_event_id`); | ||
CREATE INDEX `idx_payments_app_id` ON `payments`(`app_id`); | ||
CREATE TABLE `identities` (`id` integer,`created_at` datetime,`updated_at` datetime,`deleted_at` datetime,`privkey` text,PRIMARY KEY (`id`)); | ||
CREATE INDEX `idx_identities_deleted_at` ON `identities`(`deleted_at`); | ||
CREATE TABLE IF NOT EXISTS "users" (`id` integer,`alby_identifier` text UNIQUE,`access_token` text,`refresh_token` text,`email` text,`expiry` datetime,`lightning_address` text,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`)); | ||
CREATE UNIQUE INDEX `idx_users_alby_identifier` ON `users`(`alby_identifier`); | ||
CREATE TABLE IF NOT EXISTS "nostr_events" (`id` integer,`app_id` integer,`nostr_id` text UNIQUE,`reply_id` text,`content` text,`state` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_nostr_events_app` FOREIGN KEY (`app_id`) REFERENCES `apps`(`id`) ON DELETE CASCADE); | ||
CREATE UNIQUE INDEX `idx_nostr_events_nostr_id` ON `nostr_events`(`nostr_id`); | ||
CREATE INDEX `idx_nostr_events_app_id` ON `nostr_events`(`app_id`); |
Oops, something went wrong.