feat: full project sync - CI fixes, frontend, workspace API, and all changes

This commit is contained in:
Tomas Dvorak
2026-04-27 09:08:07 +02:00
parent a07fca997e
commit 89b9390c14
109 changed files with 21120 additions and 545 deletions
+258
View File
@@ -0,0 +1,258 @@
-- +goose Up
CREATE TABLE IF NOT EXISTS documents (
id TEXT PRIMARY KEY,
data BYTEA NOT NULL
);
CREATE TABLE IF NOT EXISTS canvases (
id TEXT NOT NULL,
user_id TEXT NOT NULL,
name TEXT,
thumbnail TEXT,
data BYTEA,
created_at TIMESTAMPTZ,
updated_at TIMESTAMPTZ,
PRIMARY KEY (user_id, id)
);
CREATE TABLE IF NOT EXISTS workspace_users (
id TEXT PRIMARY KEY,
name TEXT NOT NULL,
username TEXT NOT NULL UNIQUE,
email TEXT NOT NULL UNIQUE,
password_hash TEXT NOT NULL,
avatar_url TEXT,
locale TEXT NOT NULL DEFAULT 'en',
timezone TEXT NOT NULL DEFAULT 'UTC',
created_at TIMESTAMPTZ NOT NULL,
updated_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE IF NOT EXISTS workspace_sessions (
id TEXT PRIMARY KEY,
user_id TEXT NOT NULL REFERENCES workspace_users(id) ON DELETE CASCADE,
token_hash TEXT NOT NULL UNIQUE,
expires_at TIMESTAMPTZ NOT NULL,
created_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE IF NOT EXISTS workspace_auth_identities (
id TEXT PRIMARY KEY,
user_id TEXT NOT NULL REFERENCES workspace_users(id) ON DELETE CASCADE,
provider TEXT NOT NULL,
provider_user_id TEXT NOT NULL,
email_verified_at TIMESTAMPTZ,
created_at TIMESTAMPTZ NOT NULL,
UNIQUE(provider, provider_user_id)
);
CREATE TABLE IF NOT EXISTS workspace_teams (
id TEXT PRIMARY KEY,
name TEXT NOT NULL,
slug TEXT NOT NULL UNIQUE,
owner_user_id TEXT NOT NULL REFERENCES workspace_users(id),
plan_type TEXT NOT NULL DEFAULT 'free',
created_at TIMESTAMPTZ NOT NULL,
updated_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE IF NOT EXISTS workspace_team_memberships (
id TEXT PRIMARY KEY,
team_id TEXT NOT NULL REFERENCES workspace_teams(id) ON DELETE CASCADE,
user_id TEXT NOT NULL REFERENCES workspace_users(id) ON DELETE CASCADE,
role TEXT NOT NULL,
joined_at TIMESTAMPTZ NOT NULL,
UNIQUE(team_id, user_id)
);
CREATE TABLE IF NOT EXISTS workspace_team_invites (
id TEXT PRIMARY KEY,
team_id TEXT NOT NULL REFERENCES workspace_teams(id) ON DELETE CASCADE,
email TEXT NOT NULL,
role TEXT NOT NULL,
token_hash TEXT NOT NULL UNIQUE,
invited_by TEXT NOT NULL REFERENCES workspace_users(id),
expires_at TIMESTAMPTZ NOT NULL,
accepted_at TIMESTAMPTZ,
revoked_at TIMESTAMPTZ,
created_at TIMESTAMPTZ NOT NULL
);
CREATE INDEX IF NOT EXISTS idx_workspace_invites_team ON workspace_team_invites(team_id, created_at);
CREATE TABLE IF NOT EXISTS workspace_projects (
id TEXT PRIMARY KEY,
team_id TEXT NOT NULL REFERENCES workspace_teams(id) ON DELETE CASCADE,
name TEXT NOT NULL,
slug TEXT NOT NULL,
description TEXT,
created_by TEXT NOT NULL REFERENCES workspace_users(id),
created_at TIMESTAMPTZ NOT NULL,
updated_at TIMESTAMPTZ NOT NULL,
UNIQUE(team_id, slug)
);
CREATE TABLE IF NOT EXISTS workspace_folders (
id TEXT PRIMARY KEY,
team_id TEXT NOT NULL REFERENCES workspace_teams(id) ON DELETE CASCADE,
project_id TEXT REFERENCES workspace_projects(id) ON DELETE SET NULL,
parent_folder_id TEXT REFERENCES workspace_folders(id) ON DELETE SET NULL,
name TEXT NOT NULL,
slug TEXT NOT NULL,
path_cache TEXT NOT NULL,
visibility TEXT NOT NULL,
created_by TEXT NOT NULL REFERENCES workspace_users(id),
created_at TIMESTAMPTZ NOT NULL,
updated_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE IF NOT EXISTS workspace_drawings (
id TEXT PRIMARY KEY,
team_id TEXT NOT NULL REFERENCES workspace_teams(id) ON DELETE CASCADE,
folder_id TEXT REFERENCES workspace_folders(id) ON DELETE SET NULL,
project_id TEXT REFERENCES workspace_projects(id) ON DELETE SET NULL,
slug TEXT,
title TEXT NOT NULL,
description TEXT,
owner_user_id TEXT NOT NULL REFERENCES workspace_users(id),
latest_revision_id TEXT,
visibility TEXT NOT NULL,
is_archived BOOLEAN NOT NULL DEFAULT false,
thumbnail_asset_id TEXT,
created_at TIMESTAMPTZ NOT NULL,
updated_at TIMESTAMPTZ NOT NULL,
deleted_at TIMESTAMPTZ
);
CREATE INDEX IF NOT EXISTS idx_workspace_drawings_team ON workspace_drawings(team_id, updated_at);
CREATE TABLE IF NOT EXISTS workspace_drawing_revisions (
id TEXT PRIMARY KEY,
drawing_id TEXT NOT NULL REFERENCES workspace_drawings(id) ON DELETE CASCADE,
revision_number INTEGER NOT NULL,
snapshot_path TEXT NOT NULL,
snapshot_size BIGINT NOT NULL,
content_hash TEXT NOT NULL,
snapshot_json BYTEA NOT NULL,
created_by TEXT NOT NULL REFERENCES workspace_users(id),
created_at TIMESTAMPTZ NOT NULL,
change_summary TEXT,
UNIQUE(drawing_id, revision_number)
);
CREATE TABLE IF NOT EXISTS workspace_drawing_assets (
id TEXT PRIMARY KEY,
drawing_id TEXT NOT NULL REFERENCES workspace_drawings(id) ON DELETE CASCADE,
kind TEXT NOT NULL,
path TEXT NOT NULL,
mime_type TEXT NOT NULL,
size BIGINT NOT NULL,
width INTEGER,
height INTEGER,
uploaded_by TEXT NOT NULL REFERENCES workspace_users(id),
created_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE IF NOT EXISTS workspace_share_links (
id TEXT PRIMARY KEY,
resource_type TEXT NOT NULL,
resource_id TEXT NOT NULL,
token_hash TEXT NOT NULL UNIQUE,
permission TEXT NOT NULL,
expires_at TIMESTAMPTZ,
password_hash TEXT,
created_by TEXT NOT NULL REFERENCES workspace_users(id),
revoked_at TIMESTAMPTZ,
created_at TIMESTAMPTZ NOT NULL
);
CREATE INDEX IF NOT EXISTS idx_workspace_share_links_resource ON workspace_share_links(resource_type, resource_id);
CREATE TABLE IF NOT EXISTS workspace_permission_grants (
id TEXT PRIMARY KEY,
resource_type TEXT NOT NULL,
resource_id TEXT NOT NULL,
subject_type TEXT NOT NULL,
subject_id TEXT NOT NULL,
permission TEXT NOT NULL,
inherited_from TEXT,
created_at TIMESTAMPTZ NOT NULL,
UNIQUE(resource_type, resource_id, subject_type, subject_id, permission)
);
CREATE INDEX IF NOT EXISTS idx_workspace_permission_grants_subject ON workspace_permission_grants(subject_type, subject_id);
CREATE TABLE IF NOT EXISTS workspace_embeds (
id TEXT PRIMARY KEY,
drawing_id TEXT NOT NULL REFERENCES workspace_drawings(id) ON DELETE CASCADE,
source_url TEXT NOT NULL,
canonical_url TEXT NOT NULL,
provider TEXT NOT NULL,
embed_type TEXT NOT NULL,
title TEXT,
preview_asset_id TEXT,
safe_embed_html TEXT,
created_by TEXT NOT NULL REFERENCES workspace_users(id),
created_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE IF NOT EXISTS workspace_link_references (
id TEXT PRIMARY KEY,
source_resource_type TEXT NOT NULL,
source_resource_id TEXT NOT NULL,
target_resource_type TEXT NOT NULL,
target_resource_id TEXT NOT NULL,
label TEXT,
created_by TEXT NOT NULL REFERENCES workspace_users(id),
created_at TIMESTAMPTZ NOT NULL
);
CREATE INDEX IF NOT EXISTS idx_workspace_links_source ON workspace_link_references(source_resource_type, source_resource_id);
CREATE TABLE IF NOT EXISTS workspace_templates (
id TEXT PRIMARY KEY,
team_id TEXT REFERENCES workspace_teams(id) ON DELETE CASCADE,
scope TEXT NOT NULL,
type TEXT NOT NULL,
name TEXT NOT NULL,
description TEXT,
snapshot_path TEXT NOT NULL,
metadata_json TEXT NOT NULL,
created_by TEXT NOT NULL,
created_at TIMESTAMPTZ NOT NULL,
updated_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE IF NOT EXISTS workspace_activity_events (
id TEXT PRIMARY KEY,
actor_user_id TEXT REFERENCES workspace_users(id) ON DELETE SET NULL,
team_id TEXT REFERENCES workspace_teams(id) ON DELETE CASCADE,
resource_type TEXT NOT NULL,
resource_id TEXT NOT NULL,
event_type TEXT NOT NULL,
metadata_json TEXT NOT NULL,
created_at TIMESTAMPTZ NOT NULL
);
CREATE INDEX IF NOT EXISTS idx_workspace_activity_team ON workspace_activity_events(team_id, created_at);
-- +goose Down
DROP TABLE IF EXISTS workspace_activity_events;
DROP TABLE IF EXISTS workspace_templates;
DROP TABLE IF EXISTS workspace_link_references;
DROP TABLE IF EXISTS workspace_embeds;
DROP TABLE IF EXISTS workspace_permission_grants;
DROP TABLE IF EXISTS workspace_share_links;
DROP TABLE IF EXISTS workspace_drawing_assets;
DROP TABLE IF EXISTS workspace_drawing_revisions;
DROP TABLE IF EXISTS workspace_drawings;
DROP TABLE IF EXISTS workspace_folders;
DROP TABLE IF EXISTS workspace_projects;
DROP TABLE IF EXISTS workspace_team_invites;
DROP TABLE IF EXISTS workspace_team_memberships;
DROP TABLE IF EXISTS workspace_teams;
DROP TABLE IF EXISTS workspace_auth_identities;
DROP TABLE IF EXISTS workspace_sessions;
DROP TABLE IF EXISTS workspace_users;
DROP TABLE IF EXISTS canvases;
DROP TABLE IF EXISTS documents;
+116
View File
@@ -0,0 +1,116 @@
package postgres
import (
"context"
"database/sql"
"embed"
"fmt"
"strconv"
"time"
_ "github.com/jackc/pgx/v5/stdlib"
"github.com/pressly/goose/v3"
)
//go:embed migrations/*.sql
var migrationFS embed.FS
type DB struct {
*sql.DB
}
type Tx struct {
*sql.Tx
}
func Open(databaseURL string) (*DB, error) {
if databaseURL == "" {
return nil, fmt.Errorf("DATABASE_URL is required")
}
db, err := sql.Open("pgx", databaseURL)
if err != nil {
return nil, err
}
db.SetMaxOpenConns(25)
db.SetMaxIdleConns(5)
db.SetConnMaxLifetime(30 * time.Minute)
return &DB{DB: db}, nil
}
func Migrate(ctx context.Context, db *sql.DB) error {
goose.SetBaseFS(migrationFS)
if err := goose.SetDialect("postgres"); err != nil {
return err
}
return goose.UpContext(ctx, db, "migrations")
}
func (db *DB) ExecContext(ctx context.Context, query string, args ...any) (sql.Result, error) {
return db.DB.ExecContext(ctx, Rebind(query), args...)
}
func (db *DB) QueryContext(ctx context.Context, query string, args ...any) (*sql.Rows, error) {
return db.DB.QueryContext(ctx, Rebind(query), args...)
}
func (db *DB) QueryRowContext(ctx context.Context, query string, args ...any) *sql.Row {
return db.DB.QueryRowContext(ctx, Rebind(query), args...)
}
func (db *DB) BeginTx(ctx context.Context, opts *sql.TxOptions) (*Tx, error) {
tx, err := db.DB.BeginTx(ctx, opts)
if err != nil {
return nil, err
}
return &Tx{Tx: tx}, nil
}
func (tx *Tx) ExecContext(ctx context.Context, query string, args ...any) (sql.Result, error) {
return tx.Tx.ExecContext(ctx, Rebind(query), args...)
}
func (tx *Tx) QueryContext(ctx context.Context, query string, args ...any) (*sql.Rows, error) {
return tx.Tx.QueryContext(ctx, Rebind(query), args...)
}
func (tx *Tx) QueryRowContext(ctx context.Context, query string, args ...any) *sql.Row {
return tx.Tx.QueryRowContext(ctx, Rebind(query), args...)
}
func Rebind(query string) string {
out := make([]byte, 0, len(query)+8)
arg := 1
inSingle := false
inDouble := false
for i := 0; i < len(query); i++ {
ch := query[i]
switch ch {
case '\'':
out = append(out, ch)
if !inDouble {
if inSingle && i+1 < len(query) && query[i+1] == '\'' {
i++
out = append(out, query[i])
continue
}
inSingle = !inSingle
}
case '"':
out = append(out, ch)
if !inSingle {
inDouble = !inDouble
}
case '?':
if inSingle || inDouble {
out = append(out, ch)
continue
}
out = append(out, '$')
out = strconv.AppendInt(out, int64(arg), 10)
arg++
default:
out = append(out, ch)
}
}
return string(out)
}