Commit fe04d523 authored by Chris Molozian's avatar Chris Molozian
Browse files

Add cascade constraint to wallet ledger.

parent e12ec7b7
Loading
Loading
Loading
Loading
+3 −0
Original line number Diff line number Diff line
@@ -13,6 +13,9 @@ The format is based on [keep a changelog](http://keepachangelog.com) and this pr
- The avatar URL fields in various domain objects now support up to 512 characters for FBIG.
- Runtime modules are now loaded in a consistent order.

### Fixed
- Add ON DELETE CASCADE to foreign key user constraint on wallet ledger.

## [2.0.0] - 2018-05-14

This release brings a large number of changes and new features to the server. It cannot be upgraded from v1.0 - reach out for help to upgrade.
+3 −1
Original line number Diff line number Diff line
// Code generated by github.com/gobuffalo/packr. DO NOT EDIT

package migrate

import "github.com/gobuffalo/packr"
@@ -9,5 +11,5 @@ import "github.com/gobuffalo/packr"
// Go binary. You can use the "packr clean" command to clean up this,
// and any other packr generated files.
func init() {
	packr.PackJSONBytes("./sql", "20180103142001_initial_schema.sql", "\"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\"")
		packr.PackJSONBytes("./sql", "20180103142001_initial_schema.sql", "\"/*
 * Copyright 2018 The Nakama Authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

-- +migrate Up
CREATE TABLE IF NOT EXISTS users (
    PRIMARY KEY (id),

    id            UUID          NOT NULL,
    username      VARCHAR(128)  CONSTRAINT users_username_key UNIQUE NOT NULL,
    display_name  VARCHAR(255),
    avatar_url    VARCHAR(512),
    -- https://tools.ietf.org/html/bcp47
    lang_tag      VARCHAR(18)   DEFAULT 'en',
    location      VARCHAR(255), -- e.g. "San Francisco, CA"
    timezone      VARCHAR(255), -- e.g. "Pacific Time (US & Canada)"
    metadata      JSONB         DEFAULT '{}' NOT NULL,
    wallet        JSONB         DEFAULT '{}' NOT NULL,
    email         VARCHAR(255)  UNIQUE,
    password      BYTEA         CHECK (length(password) < 32000),
    facebook_id   VARCHAR(128)  UNIQUE,
    google_id     VARCHAR(128)  UNIQUE,
    gamecenter_id VARCHAR(128)  UNIQUE,
    steam_id      VARCHAR(128)  UNIQUE,
    custom_id     VARCHAR(128)  UNIQUE,
    edge_count    INT           DEFAULT 0 CHECK (edge_count >= 0) NOT NULL,
    create_time   TIMESTAMPTZ   DEFAULT now() NOT NULL,
    update_time   TIMESTAMPTZ   DEFAULT now() NOT NULL,
    verify_time   TIMESTAMPTZ   DEFAULT CAST(0 AS TIMESTAMPTZ) NOT NULL,
    disable_time  TIMESTAMPTZ   DEFAULT CAST(0 AS TIMESTAMPTZ) NOT NULL
);

-- Setup System user.
INSERT INTO users (id, username)
    VALUES ('00000000-0000-0000-0000-000000000000', '')
    ON CONFLICT(id) DO NOTHING;

CREATE TABLE IF NOT EXISTS user_device (
    PRIMARY KEY (id),
    FOREIGN KEY (user_id) REFERENCES users (id) ON DELETE CASCADE,

    id      VARCHAR(128) NOT NULL,
    user_id UUID         NOT NULL
);

CREATE TABLE IF NOT EXISTS user_edge (
    PRIMARY KEY (source_id, state, position),
    FOREIGN KEY (source_id)      REFERENCES users (id) ON DELETE CASCADE,
    FOREIGN KEY (destination_id) REFERENCES users (id) ON DELETE CASCADE,

    source_id      UUID        NOT NULL,
    position       BIGINT      NOT NULL, -- Used for sort order on rows.
    update_time    TIMESTAMPTZ DEFAULT now() NOT NULL,
    destination_id UUID        NOT NULL,
    state          SMALLINT    DEFAULT 0 NOT NULL, -- friend(0), invite(1), invited(2), blocked(3), deleted(4), archived(5)

    UNIQUE (source_id, destination_id)
);

CREATE TABLE IF NOT EXISTS notification (
    -- FIXME: cockroach's analyser is not clever enough when create_time has DESC mode on the index.
    PRIMARY KEY (user_id, create_time, id),
    FOREIGN KEY (user_id) REFERENCES users (id) ON DELETE CASCADE,

    id          UUID         CONSTRAINT notification_id_key UNIQUE NOT NULL,
    user_id     UUID         NOT NULL,
    subject     VARCHAR(255) NOT NULL,
    content     JSONB        DEFAULT '{}' NOT NULL,
    code        SMALLINT     NOT NULL, -- Negative values are system reserved.
    sender_id   UUID         NOT NULL,
    create_time TIMESTAMPTZ  DEFAULT now() NOT NULL
);

CREATE TABLE IF NOT EXISTS storage (
    PRIMARY KEY (collection, read, key, user_id),
    FOREIGN KEY (user_id) REFERENCES users (id) ON DELETE CASCADE,

    collection  VARCHAR(128) NOT NULL,
    key         VARCHAR(128) NOT NULL,
    user_id     UUID         NOT NULL,
    value       JSONB        DEFAULT '{}' NOT NULL,
    version     VARCHAR(32)  NOT NULL, -- md5 hash of value object.
    read        SMALLINT     DEFAULT 1 CHECK (read >= 0) NOT NULL,
    write       SMALLINT     DEFAULT 1 CHECK (write >= 0) NOT NULL,
    create_time TIMESTAMPTZ  DEFAULT now() NOT NULL,
    update_time TIMESTAMPTZ  DEFAULT now() NOT NULL,

    UNIQUE (collection, key, user_id)
);
CREATE INDEX IF NOT EXISTS collection_read_user_id_key_idx ON storage (collection, read, user_id, key);
CREATE INDEX IF NOT EXISTS value_ginidx ON storage USING GIN (value);

CREATE TABLE IF NOT EXISTS message (
  PRIMARY KEY (stream_mode, stream_subject, stream_descriptor, stream_label, create_time, id),
  FOREIGN KEY (sender_id) REFERENCES users (id) ON DELETE CASCADE,

  id                UUID         UNIQUE NOT NULL,
  -- chat(0), chat_update(1), chat_remove(2), group_join(3), group_add(4), group_leave(5), group_kick(6), group_promoted(7)
  code              SMALLINT     DEFAULT 0 NOT NULL,
  sender_id         UUID         NOT NULL,
  username          VARCHAR(128) NOT NULL,
  stream_mode       SMALLINT     NOT NULL,
  stream_subject    UUID         NOT NULL,
  stream_descriptor UUID         NOT NULL,
  stream_label      VARCHAR(128) NOT NULL,
  content           JSONB        DEFAULT '{}' NOT NULL,
  create_time       TIMESTAMPTZ  DEFAULT now() NOT NULL,
  update_time       TIMESTAMPTZ  DEFAULT now() NOT NULL,

  UNIQUE (sender_id, id)
);

CREATE TABLE IF NOT EXISTS leaderboard (
  PRIMARY KEY (id),

  id             VARCHAR(128) NOT NULL,
  authoritative  BOOLEAN      DEFAULT FALSE,
  sort_order     SMALLINT     DEFAULT 1 NOT NULL, -- asc(0), desc(1)
  operator       SMALLINT     DEFAULT 0 NOT NULL, -- best(0), set(1), increment(2), decrement(3)
  reset_schedule VARCHAR(64), -- e.g. cron format: "* * * * * * *"
  metadata       JSONB        DEFAULT '{}' NOT NULL,
  create_time    TIMESTAMPTZ  DEFAULT now() NOT NULL
);

CREATE TABLE IF NOT EXISTS leaderboard_record (
  PRIMARY KEY (leaderboard_id, expiry_time, score, subscore, owner_id),
  FOREIGN KEY (leaderboard_id) REFERENCES leaderboard (id) ON DELETE CASCADE,

  leaderboard_id VARCHAR(128)  NOT NULL,
  owner_id       UUID          NOT NULL,
  username       VARCHAR(128),
  score          BIGINT        DEFAULT 0 CHECK (score >= 0) NOT NULL,
  subscore       BIGINT        DEFAULT 0 CHECK (subscore >= 0) NOT NULL,
  num_score      INT           DEFAULT 1 CHECK (num_score >= 0) NOT NULL,
  metadata       JSONB         DEFAULT '{}' NOT NULL,
  create_time    TIMESTAMPTZ   DEFAULT now() NOT NULL,
  update_time    TIMESTAMPTZ   DEFAULT now() NOT NULL,
  expiry_time    TIMESTAMPTZ   DEFAULT CAST(0 AS TIMESTAMPTZ) NOT NULL,

  UNIQUE (owner_id, leaderboard_id, expiry_time)
);

CREATE TABLE IF NOT EXISTS wallet_ledger (
  PRIMARY KEY (user_id, create_time, id),
  FOREIGN KEY (user_id) REFERENCES users (id) ON DELETE CASCADE,

  id          UUID        UNIQUE NOT NULL,
  user_id     UUID        NOT NULL,
  changeset   JSONB       NOT NULL,
  metadata    JSONB       NOT NULL,
  create_time TIMESTAMPTZ DEFAULT now() NOT NULL,
  update_time TIMESTAMPTZ DEFAULT now() NOT NULL
);

CREATE TABLE IF NOT EXISTS user_tombstone (
  PRIMARY KEY (create_time, user_id),

  user_id        UUID        UNIQUE NOT NULL,
  create_time    TIMESTAMPTZ DEFAULT now() NOT NULL
);

CREATE TABLE IF NOT EXISTS groups (
  PRIMARY KEY (disable_time, lang_tag, edge_count, id),

  id           UUID          UNIQUE NOT NULL,
  creator_id   UUID          NOT NULL,
  name         VARCHAR(255)  CONSTRAINT groups_name_key UNIQUE NOT NULL,
  description  VARCHAR(255),
  avatar_url   VARCHAR(512),
  -- https://tools.ietf.org/html/bcp47
  lang_tag     VARCHAR(18)   DEFAULT 'en',
  metadata     JSONB         DEFAULT '{}' NOT NULL,
  state        SMALLINT      DEFAULT 0 CHECK (state >= 0) NOT NULL, -- open(0), closed(1)
  edge_count   INT           DEFAULT 0 CHECK (edge_count >= 1 AND edge_count <= max_count) NOT NULL,
  max_count    INT           DEFAULT 100 CHECK (max_count >= 1) NOT NULL,
  create_time  TIMESTAMPTZ   DEFAULT now() NOT NULL,
  update_time  TIMESTAMPTZ   DEFAULT now() NOT NULL,
  disable_time TIMESTAMPTZ   DEFAULT CAST(0 AS TIMESTAMPTZ) NOT NULL
);
CREATE INDEX IF NOT EXISTS edge_count_update_time_id_idx ON groups (disable_time, edge_count, update_time, id);
CREATE INDEX IF NOT EXISTS update_time_edge_count_id_idx ON groups (disable_time, update_time, edge_count, id);

CREATE TABLE IF NOT EXISTS group_edge (
  PRIMARY KEY (source_id, state, position),

  source_id      UUID        NOT NULL,
  position       BIGINT      NOT NULL, -- Used for sort order on rows.
  update_time    TIMESTAMPTZ DEFAULT now() NOT NULL,
  destination_id UUID        NOT NULL,
  state          SMALLINT    DEFAULT 0 NOT NULL, -- superadmin(0), admin(1), member(2), join_request(3), archived(4)

  UNIQUE (source_id, destination_id)
);

-- +migrate Down
DROP TABLE IF EXISTS group_edge;
DROP TABLE IF EXISTS groups;
DROP TABLE IF EXISTS user_tombstone;
DROP TABLE IF EXISTS wallet_ledger;
DROP TABLE IF EXISTS leaderboard_record;
DROP TABLE IF EXISTS leaderboard;
DROP TABLE IF EXISTS message;
DROP TABLE IF EXISTS storage;
DROP TABLE IF EXISTS notification;
DROP TABLE IF EXISTS user_edge;
DROP TABLE IF EXISTS user_device;
DROP TABLE IF EXISTS users;
\"")
}
+45 −45
Original line number Diff line number Diff line
@@ -154,7 +154,7 @@ CREATE TABLE IF NOT EXISTS leaderboard_record (

CREATE TABLE IF NOT EXISTS wallet_ledger (
  PRIMARY KEY (user_id, create_time, id),
  FOREIGN KEY (user_id) REFERENCES users (id),
  FOREIGN KEY (user_id) REFERENCES users (id) ON DELETE CASCADE,

  id          UUID        UNIQUE NOT NULL,
  user_id     UUID        NOT NULL,