-- Create users table CREATE TABLE IF NOT EXISTS users ( id SERIAL PRIMARY KEY, username TEXT UNIQUE NOT NULL, email TEXT UNIQUE NOT NULL, password_hash TEXT NOT NULL, first_name TEXT, last_name TEXT, display_name TEXT UNIQUE NOT NULL, is_admin BOOLEAN DEFAULT FALSE, auth_provider VARCHAR(50) DEFAULT 'local' NOT NULL, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ); CREATE INDEX IF NOT EXISTS idx_users_username ON users (username); CREATE INDEX IF NOT EXISTS idx_users_email ON users (email); CREATE INDEX IF NOT EXISTS idx_users_auth_provider ON users (auth_provider); -- Create recipes table CREATE TABLE IF NOT EXISTS recipes ( id SERIAL PRIMARY KEY, name TEXT NOT NULL, meal_type TEXT NOT NULL, -- breakfast / lunch / dinner / snack time_minutes INTEGER NOT NULL, tags JSONB NOT NULL DEFAULT '[]', -- ["מהיר", "בריא"] ingredients JSONB NOT NULL DEFAULT '[]', -- ["ביצה", "עגבניה", "מלח"] steps JSONB NOT NULL DEFAULT '[]', -- ["לחתוך", "לבשל", ...] image TEXT, -- Base64-encoded image or image URL made_by TEXT, -- Person who created this recipe version user_id INTEGER REFERENCES users(id) ON DELETE SET NULL, -- Recipe owner visibility VARCHAR(20) DEFAULT 'public', -- public / private / friends / groups created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, CONSTRAINT recipes_visibility_check CHECK (visibility IN ('public', 'private', 'friends', 'groups')) ); -- Optional: index for filters CREATE INDEX IF NOT EXISTS idx_recipes_meal_type ON recipes (meal_type); CREATE INDEX IF NOT EXISTS idx_recipes_time_minutes ON recipes (time_minutes); CREATE INDEX IF NOT EXISTS idx_recipes_made_by ON recipes (made_by); CREATE INDEX IF NOT EXISTS idx_recipes_user_id ON recipes (user_id); -- Create grocery lists table CREATE TABLE IF NOT EXISTS grocery_lists ( id SERIAL PRIMARY KEY, name TEXT NOT NULL, items TEXT[] NOT NULL DEFAULT '{}', -- Array of grocery items owner_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, is_pinned BOOLEAN DEFAULT FALSE, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ); -- Create grocery list shares table CREATE TABLE IF NOT EXISTS grocery_list_shares ( id SERIAL PRIMARY KEY, list_id INTEGER NOT NULL REFERENCES grocery_lists(id) ON DELETE CASCADE, shared_with_user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, can_edit BOOLEAN DEFAULT FALSE, shared_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, UNIQUE(list_id, shared_with_user_id) ); CREATE INDEX IF NOT EXISTS idx_grocery_lists_owner_id ON grocery_lists (owner_id); CREATE INDEX IF NOT EXISTS idx_grocery_list_shares_list_id ON grocery_list_shares (list_id); CREATE INDEX IF NOT EXISTS idx_grocery_list_shares_user_id ON grocery_list_shares (shared_with_user_id); -- Create notifications table CREATE TABLE IF NOT EXISTS notifications ( id SERIAL PRIMARY KEY, user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, type TEXT NOT NULL, -- 'grocery_share', etc. message TEXT NOT NULL, related_id INTEGER, -- Related entity ID (e.g., list_id) is_read BOOLEAN DEFAULT FALSE, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ); CREATE INDEX IF NOT EXISTS idx_notifications_user_id ON notifications (user_id); CREATE INDEX IF NOT EXISTS idx_notifications_is_read ON notifications (is_read); -- Create friend requests table CREATE TABLE IF NOT EXISTS friend_requests ( id SERIAL PRIMARY KEY, sender_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, receiver_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, status VARCHAR(20) DEFAULT 'pending', created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, CONSTRAINT friend_requests_check CHECK (sender_id <> receiver_id), CONSTRAINT friend_requests_status_check CHECK (status IN ('pending', 'accepted', 'rejected')), UNIQUE(sender_id, receiver_id) ); CREATE INDEX IF NOT EXISTS idx_friend_requests_sender_id ON friend_requests (sender_id); CREATE INDEX IF NOT EXISTS idx_friend_requests_receiver_id ON friend_requests (receiver_id); CREATE INDEX IF NOT EXISTS idx_friend_requests_status ON friend_requests (status); -- Create friendships table CREATE TABLE IF NOT EXISTS friendships ( id SERIAL PRIMARY KEY, user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, friend_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, CONSTRAINT friendships_check CHECK (user_id <> friend_id), UNIQUE(user_id, friend_id) ); CREATE INDEX IF NOT EXISTS idx_friendships_user_id ON friendships (user_id); CREATE INDEX IF NOT EXISTS idx_friendships_friend_id ON friendships (friend_id); -- Create groups table CREATE TABLE IF NOT EXISTS groups ( id SERIAL PRIMARY KEY, name VARCHAR(100) NOT NULL, description TEXT, created_by INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, is_private BOOLEAN DEFAULT FALSE, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ); CREATE INDEX IF NOT EXISTS idx_groups_created_by ON groups (created_by); -- Create group members table CREATE TABLE IF NOT EXISTS group_members ( id SERIAL PRIMARY KEY, group_id INTEGER NOT NULL REFERENCES groups(id) ON DELETE CASCADE, user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, role VARCHAR(20) DEFAULT 'member', joined_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, CONSTRAINT group_members_role_check CHECK (role IN ('admin', 'moderator', 'member')), UNIQUE(group_id, user_id) ); CREATE INDEX IF NOT EXISTS idx_group_members_group_id ON group_members (group_id); CREATE INDEX IF NOT EXISTS idx_group_members_user_id ON group_members (user_id); -- Create conversations table CREATE TABLE IF NOT EXISTS conversations ( id SERIAL PRIMARY KEY, name VARCHAR(100), is_group BOOLEAN DEFAULT FALSE, created_by INTEGER REFERENCES users(id) ON DELETE SET NULL, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ); CREATE INDEX IF NOT EXISTS idx_conversations_created_by ON conversations (created_by); -- Create conversation members table CREATE TABLE IF NOT EXISTS conversation_members ( id SERIAL PRIMARY KEY, conversation_id INTEGER NOT NULL REFERENCES conversations(id) ON DELETE CASCADE, user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, joined_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, last_read_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, UNIQUE(conversation_id, user_id) ); CREATE INDEX IF NOT EXISTS idx_conversation_members_conversation_id ON conversation_members (conversation_id); CREATE INDEX IF NOT EXISTS idx_conversation_members_user_id ON conversation_members (user_id); -- Create messages table CREATE TABLE IF NOT EXISTS messages ( id SERIAL PRIMARY KEY, conversation_id INTEGER NOT NULL REFERENCES conversations(id) ON DELETE CASCADE, sender_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, content TEXT NOT NULL, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, edited_at TIMESTAMP, is_deleted BOOLEAN DEFAULT FALSE ); CREATE INDEX IF NOT EXISTS idx_messages_conversation_id ON messages (conversation_id); CREATE INDEX IF NOT EXISTS idx_messages_sender_id ON messages (sender_id); CREATE INDEX IF NOT EXISTS idx_messages_created_at ON messages (created_at); -- Create recipe shares table CREATE TABLE IF NOT EXISTS recipe_shares ( id SERIAL PRIMARY KEY, recipe_id INTEGER NOT NULL REFERENCES recipes(id) ON DELETE CASCADE, group_id INTEGER NOT NULL REFERENCES groups(id) ON DELETE CASCADE, shared_by INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, shared_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, UNIQUE(recipe_id, group_id) ); CREATE INDEX IF NOT EXISTS idx_recipe_shares_recipe_id ON recipe_shares (recipe_id); CREATE INDEX IF NOT EXISTS idx_recipe_shares_group_id ON recipe_shares (group_id); CREATE INDEX IF NOT EXISTS idx_recipe_shares_shared_by ON recipe_shares (shared_by); -- Create recipe ratings table CREATE TABLE IF NOT EXISTS recipe_ratings ( id SERIAL PRIMARY KEY, recipe_id INTEGER NOT NULL REFERENCES recipes(id) ON DELETE CASCADE, user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, rating INTEGER NOT NULL, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, CONSTRAINT recipe_ratings_rating_check CHECK (rating >= 1 AND rating <= 5), UNIQUE(recipe_id, user_id) ); CREATE INDEX IF NOT EXISTS idx_recipe_ratings_recipe_id ON recipe_ratings (recipe_id); CREATE INDEX IF NOT EXISTS idx_recipe_ratings_user_id ON recipe_ratings (user_id); -- Create recipe comments table CREATE TABLE IF NOT EXISTS recipe_comments ( id SERIAL PRIMARY KEY, recipe_id INTEGER NOT NULL REFERENCES recipes(id) ON DELETE CASCADE, user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE, content TEXT NOT NULL, parent_comment_id INTEGER REFERENCES recipe_comments(id) ON DELETE CASCADE, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, is_deleted BOOLEAN DEFAULT FALSE ); CREATE INDEX IF NOT EXISTS idx_recipe_comments_recipe_id ON recipe_comments (recipe_id); CREATE INDEX IF NOT EXISTS idx_recipe_comments_user_id ON recipe_comments (user_id); CREATE INDEX IF NOT EXISTS idx_recipe_comments_parent_comment_id ON recipe_comments (parent_comment_id); -- Create default admin user (password: admin123) -- Password hash generated with bcrypt for 'admin123' INSERT INTO users (username, email, password_hash, first_name, last_name, display_name, is_admin) VALUES ('admin', 'admin@myrecipes.local', '$2b$12$LQv3c1yqBWVHxkd0LHAkCOYz6TtxMQJqhN8/LewY5lE7UGf3rCvHC', 'Admin', 'User', 'מנהל', TRUE) ON CONFLICT (username) DO NOTHING;