diff --git a/aws/.env b/aws/.env index 233b372..36f836f 100644 --- a/aws/.env +++ b/aws/.env @@ -2,7 +2,7 @@ DB_USER=recipes_user DB_PASSWORD=recipes_password DB_NAME=recipes_db -DB_HOST=my-recipes-rds.chw4omcqsuqv7.eu-central-1.rds.amazonaws.com +DB_HOST=my-recipes-rds.chw4omcguqv7.eu-central-1.rds.amazonaws.com DB_PORT=5432 # Email Configuration @@ -24,11 +24,12 @@ AZURE_CLIENT_SECRET=Zad8Q~qRBxaQq8up0lLXAq4pHzrVM2JFGFJhHaDp AZURE_TENANT_ID=consumers AZURE_REDIRECT_URI=http://localhost:8000/auth/azure/callback -# Cloudflare R2 Backup Configuration -R2_ENDPOINT=https://d4704b8c40b2f95b2c7bf7ee4ecc52f8.r2.cloudflarestorage.com -R2_ACCESS_KEY=1997b1e48a337c0dbe1f7552a08631b5 -R2_SECRET_KEY=369694e39fedfedb254158c147171f5760de84fa2346d5d5d5a961f1f517dbc6 -R2_BUCKET_NAME=recipes-backups +# AWS S3 Backup Configuration +S3_ENDPOINT=https://s3.eu-central-1.amazonaws.com +S3_ACCESS_KEY=1997b1e48a337c0dbe1f7552a08631b5 +S3_SECRET_KEY=369694e39fedfedb254158c147171f5760de84fa2346d5d5d5a961f1f517dbc6 +S3_BUCKET_NAME=recipes-backups +S3_REGION=eu-central-1 # Automatic Backup Schedule # Options: test (every 1 minute), daily, weekly, disabled diff --git a/backend/__pycache__/email_utils.cpython-312.pyc b/backend/__pycache__/email_utils.cpython-312.pyc index 6524825..0f2969f 100644 Binary files a/backend/__pycache__/email_utils.cpython-312.pyc and b/backend/__pycache__/email_utils.cpython-312.pyc differ diff --git a/backend/__pycache__/grocery_db_utils.cpython-312.pyc b/backend/__pycache__/grocery_db_utils.cpython-312.pyc index 6a65603..75e7239 100644 Binary files a/backend/__pycache__/grocery_db_utils.cpython-312.pyc and b/backend/__pycache__/grocery_db_utils.cpython-312.pyc differ diff --git a/backend/__pycache__/main.cpython-312.pyc b/backend/__pycache__/main.cpython-312.pyc index 783ca7e..be98ffa 100644 Binary files a/backend/__pycache__/main.cpython-312.pyc and b/backend/__pycache__/main.cpython-312.pyc differ diff --git a/backend/__pycache__/user_db_utils.cpython-312.pyc b/backend/__pycache__/user_db_utils.cpython-312.pyc index 0a43b6f..012ff9b 100644 Binary files a/backend/__pycache__/user_db_utils.cpython-312.pyc and b/backend/__pycache__/user_db_utils.cpython-312.pyc differ diff --git a/backend/main.py b/backend/main.py index 2a792fb..f520815 100644 --- a/backend/main.py +++ b/backend/main.py @@ -1231,4 +1231,4 @@ def trigger_restore( if __name__ == "__main__": - uvicorn.run("main:app", host="0.0.0.0", port=8000, reload=True) \ No newline at end of file + uvicorn.run("main:app", host="0.0.0.0", port=8001, reload=True) \ No newline at end of file diff --git a/backend/schema.sql b/backend/schema.sql index 5b61e17..a432695 100644 --- a/backend/schema.sql +++ b/backend/schema.sql @@ -8,11 +8,13 @@ CREATE TABLE IF NOT EXISTS users ( 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 ( @@ -20,13 +22,15 @@ CREATE TABLE IF NOT EXISTS recipes ( name TEXT NOT NULL, meal_type TEXT NOT NULL, -- breakfast / lunch / dinner / snack time_minutes INTEGER NOT NULL, - tags TEXT[] NOT NULL DEFAULT '{}', -- {"מהיר", "בריא"} - ingredients TEXT[] NOT NULL DEFAULT '{}', -- {"ביצה", "עגבניה", "מלח"} - steps TEXT[] NOT NULL DEFAULT '{}', -- {"לחתוך", "לבשל", ...} + 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 - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP + 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 @@ -81,6 +85,148 @@ CREATE TABLE IF NOT EXISTS notifications ( 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)