From 06bc485b98a3ba6c7e27a24a7501a664d92e0a07 Mon Sep 17 00:00:00 2001 From: Evaclaire Wamitu Date: Wed, 31 Jul 2024 12:06:32 +0300 Subject: [PATCH 1/3] Add deployment section --- Streamflix_data_report.docx | Bin 2416684 -> 2417521 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/Streamflix_data_report.docx b/Streamflix_data_report.docx index 03885da7231e8e480426ac45ec4da1da5e7e6c0b..d0308dbd617cebbea8e010272dfc62bcfdee2b91 100644 GIT binary patch delta 16539 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 YQd_qvWWh`(QXzLJNTCWhD87&WUlARZ3IG5A delta 15675 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 From a72da5853b012e0d0b3016d2f62c17da15c24d1b Mon Sep 17 00:00:00 2001 From: Evaclaire Wamitu Date: Wed, 31 Jul 2024 12:07:38 +0300 Subject: [PATCH 2/3] Add about section --- app.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app.py b/app.py index c83d023..f022ad8 100644 --- a/app.py +++ b/app.py @@ -228,9 +228,9 @@ def main(): elif page == 'About': st.header('📚 About Streamflix') st.write( - "Streamflix is a movie recommendation system that combines collaborative filtering and content-based methods " + "Streamflix is a movie recommendation system that utilizes collaborative filtering methods " "to provide personalized movie suggestions. Our system uses your ratings and movie genres to recommend " - "movies you might enjoy. Explore top trending movies, get personalized recommendations, and search for " + "movies you might enjoy. Explore top trending movies, get personalized recommendations and search for " "your favorite films all in one place.") st.subheader('Developers') From c3584f2ed8c9b4da98b057a9aa601da069d3112d Mon Sep 17 00:00:00 2001 From: Evaclaire Wamitu Date: Wed, 31 Jul 2024 12:10:39 +0300 Subject: [PATCH 3/3] Add tracking files --- .DS_Store | Bin 8196 -> 8196 bytes .ipynb_checkpoints/app-checkpoint.py | 270 +++++++++++++++++++-------- 2 files changed, 194 insertions(+), 76 deletions(-) diff --git a/.DS_Store b/.DS_Store index 2b1e2cae7b25153fbe1dfb6959eab4855acd828f..9b7e7807dab88cac5a244b5d53d4f760d569be3e 100644 GIT binary patch delta 111 zcmV-#0FeKLK!iY$PXQ3IP`eKS5R>i@AhRqG3<3;dcXM!JY;R%!006j?KoM~Z7<+p$ zFf=S6G%+xfx)B=zij&R|9|4P#91^>eOA;9jc2jb0Y;SI700001lTH*Mv-A?T0kLva R0<#Ag_5_pl6SlE(Q37_KBclKS delta 50 zcmV-20L}k|K!iY$PXP|GP`eKS4wLQ>AhRqG3<8rF6Q7fb6&;M1& diff --git a/.ipynb_checkpoints/app-checkpoint.py b/.ipynb_checkpoints/app-checkpoint.py index d523071..f022ad8 100644 --- a/.ipynb_checkpoints/app-checkpoint.py +++ b/.ipynb_checkpoints/app-checkpoint.py @@ -1,129 +1,247 @@ +import os import streamlit as st import pandas as pd import numpy as np -from surprise import Dataset, Reader, SVD -import requests import pickle +import requests +from googleapiclient.discovery import build +from surprise import Reader, Dataset, SVD +from surprise.model_selection import train_test_split +from dotenv import load_dotenv + +# Load environment variables from .env file +load_dotenv() # Set page config st.set_page_config(page_title='STREAMFLIX', page_icon="🎬", layout='wide') -# Load your data +# Load data and model @st.cache_data -def load_data(): - df = pd.read_csv('movies_data/movies.csv') - ratings = pd.read_csv('movies_data/ratings.csv') - return df, ratings - -# Train your model -@st.cache_resource -def train_model(ratings): - reader = Reader(rating_scale=(1, 5)) - data = Dataset.load_from_df(ratings[['userId_x', 'movieId', 'rating']], reader) - model = SVD() - model.fit(data.build_full_trainset()) - return model - -# Get recommendations -def get_recommendations(model, df, user_ratings, n=5, genre=None): - new_user_id = df['userId_x'].max() + 1 - movies_to_predict = df[~df['movieId'].isin([x[0] for x in user_ratings])]['movieId'].unique() - - predictions = [] - for movie_id in movies_to_predict: - predicted_rating = model.predict(new_user_id, movie_id).est - predictions.append((movie_id, predicted_rating)) - - recommendations = sorted(predictions, key=lambda x: x[1], reverse=True) - - if genre: - genre_recommendations = [ - (movie_id, rating) for movie_id, rating in recommendations - if genre.lower() in df[df['movieId'] == movie_id]['genres'].iloc[0].lower() - ] - return genre_recommendations[:n] - else: +def load_data_and_model(): + try: + collab_df = pd.read_csv('modelling_data/collab_movies.csv') + content_df = pd.read_csv('modelling_data/content_movies.csv') + merged_df = pd.merge(collab_df, content_df, on='movieId').drop_duplicates(subset=['movieId']) + with open('pickle_files/collaborative_model1.pkl', 'rb') as f: + collab_model = pickle.load(f) + return merged_df, collab_model + except Exception as e: + st.error(f"Error loading data and model: {e}") + return None, None + +merged_df, collab_model = load_data_and_model() + +class CollabBasedModel: + def __init__(self, collab_df, model): + self.df = collab_df + self.model = model + + def get_recommendations(self, user_ratings, n=5): + new_user_id = self.df['user_id'].max() + 1 + # Create a DataFrame for new user ratings + new_ratings_df = pd.DataFrame(user_ratings, columns=['movieId', 'rating']) + new_ratings_df['user_id'] = new_user_id + + # Append new user ratings to the dataset + self.df = pd.concat([self.df, new_ratings_df[['user_id', 'movieId', 'rating']]], ignore_index=True) + + # Train the model with the updated dataset + reader = Reader(rating_scale=(1, 5)) + data = Dataset.load_from_df(self.df[['user_id', 'movieId', 'rating']], reader) + trainset = data.build_full_trainset() + self.model.fit(trainset) + + # Get recommendations + movies_to_predict = self.df[~self.df['movieId'].isin([x[0] for x in user_ratings])]['movieId'].unique() + predictions = [(movie_id, self.model.predict(new_user_id, movie_id).est) for movie_id in movies_to_predict] + recommendations = sorted(predictions, key=lambda x: x[1], reverse=True) return recommendations[:n] # Fetch movie poster @st.cache_data -def fetch_poster(movie_id): - url = f"https://api.themoviedb.org/3/movie/{movie_id}?api_key=your_api_key" - response = requests.get(url) - data = response.json() - return "https://image.tmdb.org/t/p/w500/" + data.get('poster_path', '') +def fetch_poster(movie_title): + try: + tmdb_api_key = os.getenv('TMDB_API_KEY') + url = f"https://api.themoviedb.org/3/search/movie?api_key={tmdb_api_key}&query={movie_title}" + response = requests.get(url) + data = response.json() + if data['results']: + poster_path = data['results'][0].get('poster_path', '') + return "https://image.tmdb.org/t/p/w500/" + poster_path if poster_path else "https://via.placeholder.com/500x750.png?text=No+Poster+Available" + else: + return "https://via.placeholder.com/500x750.png?text=No+Poster+Available" + except Exception as e: + st.warning(f"Error fetching poster for movie {movie_title}: {e}") + return "https://via.placeholder.com/500x750.png?text=No+Poster+Available" + +# Get trailer URL +@st.cache_data +def get_trailer_url(movie_title): + try: + youtube_api_key = os.getenv('YOUTUBE_API_KEY') + youtube = build('youtube', 'v3', developerKey=youtube_api_key) + + # Search for the movie trailer + search_response = youtube.search().list( + q=f"{movie_title} official trailer", + type='video', + part='id,snippet', + maxResults=1 + ).execute() + + # Get the first search result + if search_response['items']: + video_id = search_response['items'][0]['id']['videoId'] + return f"https://www.youtube.com/watch?v={video_id}" + else: + return None + except Exception as e: + st.warning(f"Error fetching trailer for {movie_title}: {e}") + return None # Main app def main(): - st.title("🎬 Streamflix: Hybrid Movie Recommendation System") - - # Load data - df, ratings = load_data() - model = train_model(ratings) + st.title("🎬 Streamflix Movie Recommendation System") # Sidebar st.sidebar.title('Navigation') - page = st.sidebar.radio('Go to', ['Home', 'Get Recommendations', 'Search Movie']) + page = st.sidebar.radio('Go to', ['Home', 'Get Recommendations', 'Search Movies', 'About']) if page == 'Home': st.header('🔥 Top Trending Movies') - top_movies = df.sort_values('popularity', ascending=False).head(10) - - for _, movie in top_movies.iterrows(): + top_movies = merged_df['movieId'].value_counts().head(20).index + for movie_id in top_movies: + movie = merged_df[merged_df['movieId'] == movie_id].iloc[0] col1, col2 = st.columns([1, 3]) with col1: - poster_url = fetch_poster(movie['id']) + poster_url = fetch_poster(movie['title']) st.image(poster_url, width=150) with col2: st.subheader(movie['title']) st.write(f"Genres: {movie['genres']}") - st.write(f"Average Rating: {movie['vote_average']:.1f}/10") - if st.button(f"Rate {movie['title']}", key=f"rate_{movie['id']}"): - rating = st.slider('Your rating', 0.5, 5.0, 3.0, 0.5, key=f"slider_{movie['id']}") + st.write(f"Release Year: {movie['release_year']}") + if st.button(f"Rate {movie['title']}", key=f"rate_{movie_id}"): + rating = st.number_input('Your rating', min_value=0.5, max_value=5.0, value=3.0, step=0.5, key=f"slider_{movie_id}") st.write(f"You rated {movie['title']} {rating} stars!") - st.write(''---'') + st.write('---') elif page == 'Get Recommendations': st.header('🎯 Get Personalized Recommendations') - user_id = st.number_input('Please enter your user ID', min_value=1, step=1) - genres = st.multiselect('Select genres', df['genres'].explode().unique()) - + num_ratings = 6 + num_recommendations = 5 + + # Initialize session state for user ratings and sampled movies + if 'user_ratings' not in st.session_state: + st.session_state.user_ratings = [3.0] * num_ratings # Use float here + + if 'sampled_movies' not in st.session_state: + st.session_state.sampled_movies = merged_df.sample(num_ratings).reset_index(drop=True) + + # Show all six movies for rating + for i in range(num_ratings): + movie = st.session_state.sampled_movies.iloc[i] + st.write(f"\nMovie: {movie['title']} ({movie['release_year']})") + st.write(f"Genre: {movie['genres']}") + st.session_state.user_ratings[i] = st.number_input( + f"Rate {movie['title']}", + min_value=0.5, # Float + max_value=5.0, # Float + value=float(st.session_state.user_ratings[i]), # Convert to float + step=0.5, # Float + key=f"rating_{movie['movieId']}" + ) + if st.button('Get Recommendations'): - recommendations = get_recommendations(user_id, model, df, ratings) - if genres: - recommendations = recommendations[recommendations['genres'].apply(lambda x: any(genre in x for genre in genres))] - + model = CollabBasedModel(merged_df, collab_model) + user_ratings = [(st.session_state.sampled_movies.iloc[i]['movieId'], st.session_state.user_ratings[i]) for i in range(num_ratings)] + recommendations = model.get_recommendations(user_ratings, n=num_recommendations) + st.subheader('Your Recommended Movies:') - for _, movie in recommendations.iterrows(): + for movie_id, score in recommendations: + movie = merged_df[merged_df['movieId'] == movie_id].iloc[0] col1, col2 = st.columns([1, 3]) with col1: - poster_url = fetch_poster(movie['id']) + poster_url = fetch_poster(movie['title']) st.image(poster_url, width=150) with col2: - st.write(f'**{movie['title']}**') - st.write(f'Genres: {movie['genres']}') - st.write(f'Average Rating: {movie['vote_average']:.1f}/10') - if st.button(f'Watch Trailer for {movie['title']}', key=f'trailer_{movie['id']}'): - # You would need to implement a function to fetch and display the trailer - st.video('https://www.youtube.com/watch?v=dQw4w9WgXcQ') # Placeholder + st.write(f"**{movie['title']}**") + st.write(f"Genres: {movie['genres']}") + st.write(f"Predicted Rating: {score:.2f}") + trailer_url = get_trailer_url(movie['title']) + if trailer_url: + st.write(f"[Watch Trailer]({trailer_url})") + else: + st.write("Sorry, couldn't find a trailer for this movie.") st.write('---') elif page == 'Search Movies': st.header('🔍 Search Movies') search_term = st.text_input('Enter a movie title') if search_term: - results = df[df['title'].str.contains(search_term, case=False)] - for _, movie in results.iterrows(): + results = merged_df[merged_df['title'].str.contains(search_term, case=False)] + if results.empty: + st.write("No movies found matching your search term.") + else: + for _, movie in results.iterrows(): + col1, col2 = st.columns([1, 3]) + with col1: + poster_url = fetch_poster(movie['title']) + st.image(poster_url, width=150) + with col2: + st.subheader(movie['title']) + st.write(f"Genres: {movie['genres']}") + st.write(f"Release Year: {movie['release_year']}") + trailer_url = get_trailer_url(movie['title']) + if trailer_url: + st.write(f"[Watch Trailer]({trailer_url})") + else: + st.write("Sorry, couldn't find a trailer for this movie.") + st.write('---') + + st.subheader('Browse by Genre') + genres = merged_df['genres'].str.get_dummies(sep=',').columns.tolist() + genres.insert(0, 'All') # Add 'All' option to the list + selected_genre = st.selectbox('Select a Genre', genres) + + if selected_genre: + st.subheader(f'Top 10 {selected_genre.capitalize()} Movies') + if selected_genre == 'All': + genre_results = merged_df.nlargest(10, 'rating') + else: + genre_results = merged_df[merged_df['genres'].str.contains(selected_genre)].nlargest(10, 'rating') + for _, movie in genre_results.iterrows(): col1, col2 = st.columns([1, 3]) with col1: - poster_url = fetch_poster(movie['id']) + poster_url = fetch_poster(movie['title']) st.image(poster_url, width=150) with col2: st.subheader(movie['title']) - st.write(f'Genres: {movie['genres']}') - st.write(f'Average Rating: {movie['vote_average']:.1f}/10') - st.write(f'Overview: {movie['overview'][:200]}...') + st.write(f"Genres: {movie['genres']}") + st.write(f"Release Year: {movie['release_year']}") + trailer_url = get_trailer_url(movie['title']) + if trailer_url: + st.write(f"[Watch Trailer]({trailer_url})") + else: + st.write("Sorry, couldn't find a trailer for this movie.") st.write('---') + elif page == 'About': + st.header('📚 About Streamflix') + st.write( + "Streamflix is a movie recommendation system that utilizes collaborative filtering methods " + "to provide personalized movie suggestions. Our system uses your ratings and movie genres to recommend " + "movies you might enjoy. Explore top trending movies, get personalized recommendations and search for " + "your favorite films all in one place.") + + st.subheader('Developers') + st.write( + "- **Evaclaire Wamitu**\n" + " - [GitHub](https://github.com/Eva-Claire)\n" + " - Email: [evamunyika@gmail.com](mailto:evamunyika@gmail.com)\n\n" + "- **Simon Makumi**\n" + " - [GitHub](https://github.com/simonMakumi)\n" + " - Email: [simonmakumi5@gmail.com](mailto:simonmakumi5@gmail.com)" + ) + if __name__ == '__main__': main() \ No newline at end of file