mirror of
https://github.com/multica-ai/multica.git
synced 2026-06-17 11:48:42 +02:00
- Add migration 106: CREATE INDEX CONCURRENTLY on member(user_id, workspace_id) - Rewrite ListWorkspaces to drive from member table with explicit fields - Regenerate all sqlc code with v1.31.1 (intentional version upgrade) Co-authored-by: multica-agent <github@multica.ai>
316 lines
9.3 KiB
Go
316 lines
9.3 KiB
Go
// Code generated by sqlc. DO NOT EDIT.
|
|
// versions:
|
|
// sqlc v1.31.1
|
|
// source: user.sql
|
|
|
|
package db
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/jackc/pgx/v5/pgtype"
|
|
)
|
|
|
|
const createUser = `-- name: CreateUser :one
|
|
INSERT INTO "user" (name, email, avatar_url)
|
|
VALUES ($1, $2, $3)
|
|
RETURNING id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone
|
|
`
|
|
|
|
type CreateUserParams struct {
|
|
Name string `json:"name"`
|
|
Email string `json:"email"`
|
|
AvatarUrl pgtype.Text `json:"avatar_url"`
|
|
}
|
|
|
|
func (q *Queries) CreateUser(ctx context.Context, arg CreateUserParams) (User, error) {
|
|
row := q.db.QueryRow(ctx, createUser, arg.Name, arg.Email, arg.AvatarUrl)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|
|
|
|
const getUser = `-- name: GetUser :one
|
|
SELECT id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone FROM "user"
|
|
WHERE id = $1
|
|
`
|
|
|
|
func (q *Queries) GetUser(ctx context.Context, id pgtype.UUID) (User, error) {
|
|
row := q.db.QueryRow(ctx, getUser, id)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|
|
|
|
const getUserByEmail = `-- name: GetUserByEmail :one
|
|
SELECT id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone FROM "user"
|
|
WHERE email = $1
|
|
`
|
|
|
|
func (q *Queries) GetUserByEmail(ctx context.Context, email string) (User, error) {
|
|
row := q.db.QueryRow(ctx, getUserByEmail, email)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|
|
|
|
const joinCloudWaitlist = `-- name: JoinCloudWaitlist :one
|
|
UPDATE "user" SET
|
|
cloud_waitlist_email = $2,
|
|
cloud_waitlist_reason = $3,
|
|
updated_at = now()
|
|
WHERE id = $1
|
|
RETURNING id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone
|
|
`
|
|
|
|
type JoinCloudWaitlistParams struct {
|
|
ID pgtype.UUID `json:"id"`
|
|
CloudWaitlistEmail pgtype.Text `json:"cloud_waitlist_email"`
|
|
CloudWaitlistReason pgtype.Text `json:"cloud_waitlist_reason"`
|
|
}
|
|
|
|
// Records interest in cloud runtimes. Does NOT mark onboarding
|
|
// complete — the user still has to pick a real path (CLI / Skip)
|
|
// in Step 3. Repeating the call overwrites email + reason.
|
|
func (q *Queries) JoinCloudWaitlist(ctx context.Context, arg JoinCloudWaitlistParams) (User, error) {
|
|
row := q.db.QueryRow(ctx, joinCloudWaitlist, arg.ID, arg.CloudWaitlistEmail, arg.CloudWaitlistReason)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|
|
|
|
const markUserOnboarded = `-- name: MarkUserOnboarded :one
|
|
UPDATE "user" SET
|
|
onboarded_at = COALESCE(onboarded_at, now()),
|
|
updated_at = now()
|
|
WHERE id = $1
|
|
RETURNING id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone
|
|
`
|
|
|
|
func (q *Queries) MarkUserOnboarded(ctx context.Context, id pgtype.UUID) (User, error) {
|
|
row := q.db.QueryRow(ctx, markUserOnboarded, id)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|
|
|
|
const patchUserOnboarding = `-- name: PatchUserOnboarding :one
|
|
UPDATE "user" SET
|
|
onboarding_questionnaire = COALESCE($1, onboarding_questionnaire),
|
|
updated_at = now()
|
|
WHERE id = $2
|
|
RETURNING id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone
|
|
`
|
|
|
|
type PatchUserOnboardingParams struct {
|
|
Questionnaire []byte `json:"questionnaire"`
|
|
ID pgtype.UUID `json:"id"`
|
|
}
|
|
|
|
// Partial update of the user's onboarding decision fields. Currently only the
|
|
// questionnaire JSONB is patchable — the v2 attempt at persisting Step 3
|
|
// runtime choice on the user row was reverted; that state now lives in a
|
|
// frontend Zustand transient store.
|
|
func (q *Queries) PatchUserOnboarding(ctx context.Context, arg PatchUserOnboardingParams) (User, error) {
|
|
row := q.db.QueryRow(ctx, patchUserOnboarding, arg.Questionnaire, arg.ID)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|
|
|
|
const setStarterContentState = `-- name: SetStarterContentState :one
|
|
UPDATE "user" SET
|
|
starter_content_state = $2,
|
|
updated_at = now()
|
|
WHERE id = $1
|
|
RETURNING id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone
|
|
`
|
|
|
|
type SetStarterContentStateParams struct {
|
|
ID pgtype.UUID `json:"id"`
|
|
StarterContentState pgtype.Text `json:"starter_content_state"`
|
|
}
|
|
|
|
// Atomically transition starter_content_state. The handler is
|
|
// responsible for checking the current value first (to decide between
|
|
// "transition NULL -> imported and run the seeding" vs "already
|
|
// decided, short-circuit"). Using COALESCE here would swallow the
|
|
// transition, so this is a straight assignment.
|
|
func (q *Queries) SetStarterContentState(ctx context.Context, arg SetStarterContentStateParams) (User, error) {
|
|
row := q.db.QueryRow(ctx, setStarterContentState, arg.ID, arg.StarterContentState)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|
|
|
|
const updateUser = `-- name: UpdateUser :one
|
|
UPDATE "user" SET
|
|
name = COALESCE($2, name),
|
|
avatar_url = COALESCE($3, avatar_url),
|
|
language = COALESCE($4, language),
|
|
profile_description = COALESCE($5, profile_description),
|
|
timezone = CASE
|
|
WHEN $6::text IS NULL THEN timezone
|
|
WHEN $6::text = '' THEN NULL
|
|
ELSE $6::text
|
|
END,
|
|
updated_at = now()
|
|
WHERE id = $1
|
|
RETURNING id, name, email, avatar_url, created_at, updated_at, onboarded_at, onboarding_questionnaire, cloud_waitlist_email, cloud_waitlist_reason, starter_content_state, language, profile_description, timezone
|
|
`
|
|
|
|
type UpdateUserParams struct {
|
|
ID pgtype.UUID `json:"id"`
|
|
Name string `json:"name"`
|
|
AvatarUrl pgtype.Text `json:"avatar_url"`
|
|
Language pgtype.Text `json:"language"`
|
|
ProfileDescription pgtype.Text `json:"profile_description"`
|
|
Timezone pgtype.Text `json:"timezone"`
|
|
}
|
|
|
|
// Patches the user-controlled profile fields. Each parameter follows
|
|
// COALESCE-on-NULL semantics so the handler can omit any field it
|
|
// doesn't intend to write.
|
|
//
|
|
// `timezone` (Viewing-tz preference) participates in
|
|
// the same shape but uses sqlc.narg + a sentinel-string convention:
|
|
// the handler passes the empty string "" to mean "clear back to NULL"
|
|
// (browser-detected fallback), an IANA name like "Asia/Shanghai" to
|
|
// pin a value, and `sqlc.narg('timezone') IS NULL` (no value at all)
|
|
// to leave the existing column untouched. Folding it into UpdateUser
|
|
// rather than carrying a dedicated UpdateUserTimezone keeps the
|
|
// profile-patch shape uniform between Preferences fields.
|
|
func (q *Queries) UpdateUser(ctx context.Context, arg UpdateUserParams) (User, error) {
|
|
row := q.db.QueryRow(ctx, updateUser,
|
|
arg.ID,
|
|
arg.Name,
|
|
arg.AvatarUrl,
|
|
arg.Language,
|
|
arg.ProfileDescription,
|
|
arg.Timezone,
|
|
)
|
|
var i User
|
|
err := row.Scan(
|
|
&i.ID,
|
|
&i.Name,
|
|
&i.Email,
|
|
&i.AvatarUrl,
|
|
&i.CreatedAt,
|
|
&i.UpdatedAt,
|
|
&i.OnboardedAt,
|
|
&i.OnboardingQuestionnaire,
|
|
&i.CloudWaitlistEmail,
|
|
&i.CloudWaitlistReason,
|
|
&i.StarterContentState,
|
|
&i.Language,
|
|
&i.ProfileDescription,
|
|
&i.Timezone,
|
|
)
|
|
return i, err
|
|
}
|