Skip to content

Commit

Permalink
Merge pull request #13 from adipai/overhaul_coding_standards
Browse files Browse the repository at this point in the history
Codebase decoupling and pep8 standard conformation
  • Loading branch information
adipai authored Oct 13, 2023
2 parents e8d5e3b + 93569e5 commit 31c3869
Show file tree
Hide file tree
Showing 6 changed files with 208 additions and 104 deletions.
13 changes: 9 additions & 4 deletions Code/recommenderapp/app.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,16 @@
"""
Module for routing all calls from the frontend
"""

import json
import sys
from search import Search
from flask import Flask, jsonify, render_template, request
from flask_cors import CORS
from utils import send_email_to_user, beautify_feedback_data

sys.path.append("../../")
from Code.prediction_scripts.item_based import recommend_for_new_user
from search import Search

app = Flask(__name__)
app.secret_key = "secret key"
Expand Down Expand Up @@ -45,8 +50,8 @@ def search():
Handles movie search requests.
"""
term = request.form["q"]
search = Search()
filtered_dict = search.resultsTop10(term)
finder = Search()
filtered_dict = finder.results_top_ten(term)
resp = jsonify(filtered_dict)
resp.status_code = 200
return resp
Expand All @@ -58,7 +63,7 @@ def feedback():
Handles user feedback submission and mails the results.
"""
data = json.loads(request.data)
user_email = "ananyamantravadi@gmail.com"
user_email = "adipai16@gmail.com"
send_email_to_user(user_email, beautify_feedback_data(data))
return data

Expand Down
31 changes: 31 additions & 0 deletions Code/recommenderapp/constants.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
"""
This module contains all the constants
"""

EMAIL_HTML_CONTENT = """
<html>
<head></head>
<body>
<h1 style="color: #333333;">Movie Recommendations from PopcornPicks</h1>
<p style="color: #555555;">Dear Movie Enthusiast,</p>
<p style="color: #555555;">We hope you're having a fantastic day!</p>
<div style="padding: 10px; border: 1px solid #cccccc; border-radius: 5px; background-color: #f9f9f9;">
<h2>Your Movie Recommendations:</h2>
<h3>Movies Liked:</h3>
<ul style="color: #555555;">
{}
</ul>
<h3>Movies Disliked:</h3>
<ul style="color: #555555;">
{}
</ul>
<h3>Movies Yet to Watch:</h3>
<ul style="color: #555555;">
{}
</ul>
</div>
<p style="color: #555555;">Enjoy your movie time with PopcornPicks!</p>
<p style="color: #555555;">Best regards,<br>PopcornPicks Team 🍿</p>
</body>
</html>
"""
53 changes: 35 additions & 18 deletions Code/recommenderapp/search.py
Original file line number Diff line number Diff line change
@@ -1,23 +1,31 @@
import pandas as pd
"""
Search feature for the web application
"""

# from app import app
from flask import jsonify, request, render_template
import sys
import os
import pandas as pd
#from flask import jsonify, request, render_template


app_dir = os.path.dirname(os.path.abspath(__file__))
code_dir = os.path.dirname(app_dir)
project_dir = os.path.dirname(code_dir)


class Search:
"""
Search feature for landing page
"""

df = pd.read_csv(project_dir + "/data/movies.csv")

def __init__(self):
pass

def startsWith(self, word):
def starts_with(self, word):
"""
Function to check movie prefix
"""
n = len(word)
res = []
word = word.lower()
Expand All @@ -27,28 +35,37 @@ def startsWith(self, word):
res.append(x)
return res

def anywhere(self, word, visitedWords):
def anywhere(self, word, visited_words):
"""
Function to check visited words
"""
res = []
word = word.lower()
for x in self.df["title"]:
if x not in visitedWords:
if x not in visited_words:
curr = x.lower()
if word in curr:
res.append(x)
return res

def results(self, word):
startsWith = self.startsWith(word)
visitedWords = set()
for x in startsWith:
visitedWords.add(x)
anywhere = self.anywhere(word, visitedWords)
startsWith.extend(anywhere)
return startsWith

def resultsTop10(self, word):
"""
Function to serve the result render
"""
starts_with = self.starts_with(word)
visited_words = set()
for x in starts_with:
visited_words.add(x)
anywhere = self.anywhere(word, visited_words)
starts_with.extend(anywhere)
return starts_with

def results_top_ten(self, word):
"""
Function to get top 10 results
"""
return self.results(word)[:10]


if __name__ == "__main__":
app.run()
#if __name__ == "__main__":
# app.run()
55 changes: 17 additions & 38 deletions Code/recommenderapp/utils.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,12 @@
""" Module contains utility functions used for various purposes in the backend """

import logging
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
import logging

import constants as c


def beautify_feedback_data(data):
"""
Expand All @@ -22,12 +27,11 @@ def beautify_feedback_data(data):
dislike.append(movie)

# Create a category-dictionary of liked, disliked and yet to watch movies
categorized_data_dict = {"Liked":like, "Disliked":dislike, "Yet to Watch":yet_to_watch}
categorized_data_dict = {"Liked": like,
"Disliked": dislike, "Yet to Watch": yet_to_watch}

return categorized_data_dict



def send_email_to_user(recipient_email, categorized_data):
"""
Utility function to send movie recommendations to user over email
Expand All @@ -36,11 +40,11 @@ def send_email_to_user(recipient_email, categorized_data):
# Email configuration
smtp_server = 'smtp.gmail.com'
# Port for TLS
smtp_port = 587
smtp_port = 587
sender_email = 'popcornpicks504@gmail.com'

# Use an app password since 2-factor authentication is enabled
sender_password = 'uxnd shis sazo mstj'
sender_password = 'uxnd shis sazo mstj'
subject = 'Your movie recommendation from PopcornPicks'

# Create the email message
Expand All @@ -50,52 +54,27 @@ def send_email_to_user(recipient_email, categorized_data):
message['Subject'] = subject

# Create the email message with HTML content
html_content = """
<html>
<head></head>
<body>
<h1 style="color: #333333;">Movie Recommendations from PopcornPicks</h1>
<p style="color: #555555;">Dear Movie Enthusiast,</p>
<p style="color: #555555;">We hope you're having a fantastic day!</p>
<div style="padding: 10px; border: 1px solid #cccccc; border-radius: 5px; background-color: #f9f9f9;">
<h2>Your Movie Recommendations:</h2>
<h3>Movies Liked:</h3>
<ul style="color: #555555;">
{}
</ul>
<h3>Movies Disliked:</h3>
<ul style="color: #555555;">
{}
</ul>
<h3>Movies Yet to Watch:</h3>
<ul style="color: #555555;">
{}
</ul>
</div>
<p style="color: #555555;">Enjoy your movie time with PopcornPicks!</p>
<p style="color: #555555;">Best regards,<br>PopcornPicks Team 🍿</p>
</body>
</html>
""".format('\n'.join(f'<li>{movie}</li>' for movie in categorized_data['Liked']),
'\n'.join(f'<li>{movie}</li>' for movie in categorized_data['Disliked']),
'\n'.join(f'<li>{movie}</li>' for movie in categorized_data['Yet to Watch']))
html_content = c.EMAIL_HTML_CONTENT.format('\n'.join(f'<li>{movie}</li>' for movie in categorized_data['Liked']),
'\n'.join(
f'<li>{movie}</li>' for movie in categorized_data['Disliked']),
'\n'.join(f'<li>{movie}</li>' for movie in categorized_data['Yet to Watch']))

# Attach the HTML email body
message.attach(MIMEText(html_content, 'html'))

# Connect to the SMTP server
try:
server = smtplib.SMTP(smtp_server, smtp_port)
# Start TLS encryption
server.starttls()
server.starttls()
server.login(sender_email, sender_password)

# Send the email
server.sendmail(sender_email, recipient_email, message.as_string())
logging.info("Email sent successfully!")

except Exception as e:
logging.warning(f'Email could not be sent. Error: {str(e)}')
logging.warning("Email could not be sent. Error: %s", str(e))

finally:
server.quit()
45 changes: 32 additions & 13 deletions test/test_search.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
"""
Test suit for search feature
"""

import unittest
import warnings
import sys
Expand All @@ -8,12 +12,18 @@

warnings.filterwarnings("ignore")


class Tests(unittest.TestCase):
def testSearchToy(self):
"""
Test cases for search feature
"""

def test_search_toy(self):
"""
Test case 1
"""
search_word = "toy"
search = Search()
filtered_dict = search.resultsTop10(search_word)
finder = Search()
filtered_dict = finder.results_top_ten(search_word)
expected_resp = [
"Toy Story (1995)",
"Toys (1992)",
Expand All @@ -26,10 +36,13 @@ def testSearchToy(self):
]
self.assertTrue(filtered_dict == expected_resp)

def testSearchLove(self):
def test_search_love(self):
"""
Test case 2
"""
search_word = "love"
search = Search()
filtered_dict = search.resultsTop10(search_word)
finder = Search()
filtered_dict = finder.results_top_ten(search_word)
expected_resp = [
"Love & Human Remains (1993)",
"Love Affair (1994)",
Expand All @@ -44,17 +57,23 @@ def testSearchLove(self):
]
self.assertTrue(filtered_dict == expected_resp)

def testSearchGibberish(self):
def test_search_gibberish(self):
"""
Test case 3
"""
search_word = "gibberish"
search = Search()
filtered_dict = search.resultsTop10(search_word)
finder = Search()
filtered_dict = finder.results_top_ten(search_word)
expected_resp = []
self.assertTrue(filtered_dict == expected_resp)

def testSearch1995(self):
def test_search_1995(self):
"""
Test case 4
"""
search_word = "1995"
search = Search()
filtered_dict = search.resultsTop10(search_word)
finder = Search()
filtered_dict = finder.results_top_ten(search_word)
expected_resp = [
"Toy Story (1995)",
"Jumanji (1995)",
Expand Down
Loading

0 comments on commit 31c3869

Please sign in to comment.