CREATE TYPE "public"."achievement_status" AS ENUM('draft', 'published');--> statement-breakpoint CREATE TYPE "public"."user_role" AS ENUM('model', 'viewer', 'admin');--> statement-breakpoint CREATE TYPE "public"."recording_status" AS ENUM('draft', 'published', 'archived');--> statement-breakpoint CREATE TABLE "articles" ( "id" text PRIMARY KEY NOT NULL, "slug" text NOT NULL, "title" text NOT NULL, "excerpt" text, "content" text, "image" text, "tags" text[] DEFAULT '{}', "publish_date" timestamp DEFAULT now() NOT NULL, "author" text, "category" text, "featured" boolean DEFAULT false, "date_created" timestamp DEFAULT now() NOT NULL, "date_updated" timestamp ); --> statement-breakpoint CREATE TABLE "comments" ( "id" integer PRIMARY KEY GENERATED ALWAYS AS IDENTITY (sequence name "comments_id_seq" INCREMENT BY 1 MINVALUE 1 MAXVALUE 2147483647 START WITH 1 CACHE 1), "collection" text NOT NULL, "item_id" text NOT NULL, "comment" text NOT NULL, "user_id" text NOT NULL, "date_created" timestamp DEFAULT now() NOT NULL, "date_updated" timestamp ); --> statement-breakpoint CREATE TABLE "files" ( "id" text PRIMARY KEY NOT NULL, "title" text, "description" text, "filename" text NOT NULL, "mime_type" text, "filesize" bigint, "duration" integer, "uploaded_by" text, "date_created" timestamp DEFAULT now() NOT NULL ); --> statement-breakpoint CREATE TABLE "achievements" ( "id" text PRIMARY KEY NOT NULL, "code" text NOT NULL, "name" text NOT NULL, "description" text, "icon" text, "category" text, "required_count" integer DEFAULT 1 NOT NULL, "points_reward" integer DEFAULT 0 NOT NULL, "status" "achievement_status" DEFAULT 'published' NOT NULL, "sort" integer DEFAULT 0 ); --> statement-breakpoint CREATE TABLE "user_achievements" ( "id" integer PRIMARY KEY GENERATED ALWAYS AS IDENTITY (sequence name "user_achievements_id_seq" INCREMENT BY 1 MINVALUE 1 MAXVALUE 2147483647 START WITH 1 CACHE 1), "user_id" text NOT NULL, "achievement_id" text NOT NULL, "progress" integer DEFAULT 0, "date_unlocked" timestamp ); --> statement-breakpoint CREATE TABLE "user_points" ( "id" integer PRIMARY KEY GENERATED ALWAYS AS IDENTITY (sequence name "user_points_id_seq" INCREMENT BY 1 MINVALUE 1 MAXVALUE 2147483647 START WITH 1 CACHE 1), "user_id" text NOT NULL, "action" text NOT NULL, "points" integer NOT NULL, "recording_id" text, "date_created" timestamp DEFAULT now() NOT NULL ); --> statement-breakpoint CREATE TABLE "user_stats" ( "id" integer PRIMARY KEY GENERATED ALWAYS AS IDENTITY (sequence name "user_stats_id_seq" INCREMENT BY 1 MINVALUE 1 MAXVALUE 2147483647 START WITH 1 CACHE 1), "user_id" text NOT NULL, "total_raw_points" integer DEFAULT 0, "total_weighted_points" real DEFAULT 0, "recordings_count" integer DEFAULT 0, "playbacks_count" integer DEFAULT 0, "comments_count" integer DEFAULT 0, "achievements_count" integer DEFAULT 0, "last_updated" timestamp DEFAULT now() ); --> statement-breakpoint CREATE TABLE "user_photos" ( "id" integer PRIMARY KEY GENERATED ALWAYS AS IDENTITY (sequence name "user_photos_id_seq" INCREMENT BY 1 MINVALUE 1 MAXVALUE 2147483647 START WITH 1 CACHE 1), "user_id" text NOT NULL, "file_id" text NOT NULL, "sort" integer DEFAULT 0 ); --> statement-breakpoint CREATE TABLE "users" ( "id" text PRIMARY KEY NOT NULL, "email" text NOT NULL, "password_hash" text NOT NULL, "first_name" text, "last_name" text, "artist_name" text, "slug" text, "description" text, "tags" text[] DEFAULT '{}', "role" "user_role" DEFAULT 'viewer' NOT NULL, "avatar" text, "banner" text, "email_verified" boolean DEFAULT false NOT NULL, "email_verify_token" text, "password_reset_token" text, "password_reset_expiry" timestamp, "date_created" timestamp DEFAULT now() NOT NULL, "date_updated" timestamp ); --> statement-breakpoint CREATE TABLE "video_likes" ( "id" text PRIMARY KEY NOT NULL, "video_id" text NOT NULL, "user_id" text NOT NULL, "date_created" timestamp DEFAULT now() NOT NULL ); --> statement-breakpoint CREATE TABLE "video_models" ( "video_id" text NOT NULL, "user_id" text NOT NULL, CONSTRAINT "video_models_video_id_user_id_pk" PRIMARY KEY("video_id","user_id") ); --> statement-breakpoint CREATE TABLE "video_plays" ( "id" text PRIMARY KEY NOT NULL, "video_id" text NOT NULL, "user_id" text, "session_id" text, "duration_watched" integer, "completed" boolean DEFAULT false, "date_created" timestamp DEFAULT now() NOT NULL, "date_updated" timestamp ); --> statement-breakpoint CREATE TABLE "videos" ( "id" text PRIMARY KEY NOT NULL, "slug" text NOT NULL, "title" text NOT NULL, "description" text, "image" text, "movie" text, "tags" text[] DEFAULT '{}', "upload_date" timestamp DEFAULT now() NOT NULL, "premium" boolean DEFAULT false, "featured" boolean DEFAULT false, "likes_count" integer DEFAULT 0, "plays_count" integer DEFAULT 0 ); --> statement-breakpoint CREATE TABLE "recording_plays" ( "id" text PRIMARY KEY NOT NULL, "recording_id" text NOT NULL, "user_id" text, "duration_played" integer DEFAULT 0, "completed" boolean DEFAULT false, "date_created" timestamp DEFAULT now() NOT NULL, "date_updated" timestamp ); --> statement-breakpoint CREATE TABLE "recordings" ( "id" text PRIMARY KEY NOT NULL, "title" text NOT NULL, "description" text, "slug" text NOT NULL, "duration" integer NOT NULL, "events" jsonb DEFAULT '[]'::jsonb, "device_info" jsonb DEFAULT '[]'::jsonb, "user_id" text NOT NULL, "status" "recording_status" DEFAULT 'draft' NOT NULL, "tags" text[] DEFAULT '{}', "linked_video" text, "featured" boolean DEFAULT false, "public" boolean DEFAULT false, "original_recording_id" text, "date_created" timestamp DEFAULT now() NOT NULL, "date_updated" timestamp ); --> statement-breakpoint ALTER TABLE "articles" ADD CONSTRAINT "articles_image_files_id_fk" FOREIGN KEY ("image") REFERENCES "public"."files"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "articles" ADD CONSTRAINT "articles_author_users_id_fk" FOREIGN KEY ("author") REFERENCES "public"."users"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "comments" ADD CONSTRAINT "comments_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "user_achievements" ADD CONSTRAINT "user_achievements_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "user_achievements" ADD CONSTRAINT "user_achievements_achievement_id_achievements_id_fk" FOREIGN KEY ("achievement_id") REFERENCES "public"."achievements"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "user_points" ADD CONSTRAINT "user_points_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "user_points" ADD CONSTRAINT "user_points_recording_id_recordings_id_fk" FOREIGN KEY ("recording_id") REFERENCES "public"."recordings"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "user_stats" ADD CONSTRAINT "user_stats_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "user_photos" ADD CONSTRAINT "user_photos_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "user_photos" ADD CONSTRAINT "user_photos_file_id_files_id_fk" FOREIGN KEY ("file_id") REFERENCES "public"."files"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "users" ADD CONSTRAINT "users_avatar_files_id_fk" FOREIGN KEY ("avatar") REFERENCES "public"."files"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "users" ADD CONSTRAINT "users_banner_files_id_fk" FOREIGN KEY ("banner") REFERENCES "public"."files"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "video_likes" ADD CONSTRAINT "video_likes_video_id_videos_id_fk" FOREIGN KEY ("video_id") REFERENCES "public"."videos"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "video_likes" ADD CONSTRAINT "video_likes_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "video_models" ADD CONSTRAINT "video_models_video_id_videos_id_fk" FOREIGN KEY ("video_id") REFERENCES "public"."videos"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "video_models" ADD CONSTRAINT "video_models_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "video_plays" ADD CONSTRAINT "video_plays_video_id_videos_id_fk" FOREIGN KEY ("video_id") REFERENCES "public"."videos"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "video_plays" ADD CONSTRAINT "video_plays_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "videos" ADD CONSTRAINT "videos_image_files_id_fk" FOREIGN KEY ("image") REFERENCES "public"."files"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "videos" ADD CONSTRAINT "videos_movie_files_id_fk" FOREIGN KEY ("movie") REFERENCES "public"."files"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "recording_plays" ADD CONSTRAINT "recording_plays_recording_id_recordings_id_fk" FOREIGN KEY ("recording_id") REFERENCES "public"."recordings"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "recording_plays" ADD CONSTRAINT "recording_plays_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint ALTER TABLE "recordings" ADD CONSTRAINT "recordings_user_id_users_id_fk" FOREIGN KEY ("user_id") REFERENCES "public"."users"("id") ON DELETE cascade ON UPDATE no action;--> statement-breakpoint ALTER TABLE "recordings" ADD CONSTRAINT "recordings_linked_video_videos_id_fk" FOREIGN KEY ("linked_video") REFERENCES "public"."videos"("id") ON DELETE set null ON UPDATE no action;--> statement-breakpoint CREATE UNIQUE INDEX "articles_slug_idx" ON "articles" USING btree ("slug");--> statement-breakpoint CREATE INDEX "articles_publish_date_idx" ON "articles" USING btree ("publish_date");--> statement-breakpoint CREATE INDEX "articles_featured_idx" ON "articles" USING btree ("featured");--> statement-breakpoint CREATE INDEX "comments_collection_item_idx" ON "comments" USING btree ("collection","item_id");--> statement-breakpoint CREATE INDEX "comments_user_idx" ON "comments" USING btree ("user_id");--> statement-breakpoint CREATE INDEX "files_uploaded_by_idx" ON "files" USING btree ("uploaded_by");--> statement-breakpoint CREATE UNIQUE INDEX "achievements_code_idx" ON "achievements" USING btree ("code");--> statement-breakpoint CREATE INDEX "user_achievements_user_idx" ON "user_achievements" USING btree ("user_id");--> statement-breakpoint CREATE UNIQUE INDEX "user_achievements_unique_idx" ON "user_achievements" USING btree ("user_id","achievement_id");--> statement-breakpoint CREATE INDEX "user_points_user_idx" ON "user_points" USING btree ("user_id");--> statement-breakpoint CREATE INDEX "user_points_date_idx" ON "user_points" USING btree ("date_created");--> statement-breakpoint CREATE UNIQUE INDEX "user_stats_user_idx" ON "user_stats" USING btree ("user_id");--> statement-breakpoint CREATE INDEX "user_photos_user_idx" ON "user_photos" USING btree ("user_id");--> statement-breakpoint CREATE UNIQUE INDEX "users_email_idx" ON "users" USING btree ("email");--> statement-breakpoint CREATE UNIQUE INDEX "users_slug_idx" ON "users" USING btree ("slug");--> statement-breakpoint CREATE INDEX "users_role_idx" ON "users" USING btree ("role");--> statement-breakpoint CREATE INDEX "video_likes_video_idx" ON "video_likes" USING btree ("video_id");--> statement-breakpoint CREATE INDEX "video_likes_user_idx" ON "video_likes" USING btree ("user_id");--> statement-breakpoint CREATE INDEX "video_plays_video_idx" ON "video_plays" USING btree ("video_id");--> statement-breakpoint CREATE INDEX "video_plays_user_idx" ON "video_plays" USING btree ("user_id");--> statement-breakpoint CREATE INDEX "video_plays_date_idx" ON "video_plays" USING btree ("date_created");--> statement-breakpoint CREATE UNIQUE INDEX "videos_slug_idx" ON "videos" USING btree ("slug");--> statement-breakpoint CREATE INDEX "videos_upload_date_idx" ON "videos" USING btree ("upload_date");--> statement-breakpoint CREATE INDEX "videos_featured_idx" ON "videos" USING btree ("featured");--> statement-breakpoint CREATE INDEX "recording_plays_recording_idx" ON "recording_plays" USING btree ("recording_id");--> statement-breakpoint CREATE INDEX "recording_plays_user_idx" ON "recording_plays" USING btree ("user_id");--> statement-breakpoint CREATE UNIQUE INDEX "recordings_slug_idx" ON "recordings" USING btree ("slug");--> statement-breakpoint CREATE INDEX "recordings_user_idx" ON "recordings" USING btree ("user_id");--> statement-breakpoint CREATE INDEX "recordings_status_idx" ON "recordings" USING btree ("status");--> statement-breakpoint CREATE INDEX "recordings_public_idx" ON "recordings" USING btree ("public");