forked from SI364-Winter2018/HW3
-
Notifications
You must be signed in to change notification settings - Fork 32
/
Copy pathSI364W18_HW3.py
178 lines (132 loc) · 7.54 KB
/
SI364W18_HW3.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
## SI 364 - Winter 2018
## HW 3
####################
## Import statements
####################
from flask import Flask, render_template, session, redirect, url_for, flash, request
from flask_wtf import FlaskForm
from wtforms import StringField, SubmitField, ValidationError
from wtforms.validators import Required, Length
from flask_sqlalchemy import SQLAlchemy
############################
# Application configurations
############################
app = Flask(__name__)
app.config['SECRET_KEY'] = 'hard to guess string from si364'
## TODO 364: Create a database in postgresql in the code line below, and fill in your app's database URI. It should be of the format: postgresql://localhost/YOUR_DATABASE_NAME
## Your final Postgres database should be your uniqname, plus HW3, e.g. "jczettaHW3" or "maupandeHW3"
app.config["SQLALCHEMY_DATABASE_URI"] = ""
## Provided:
app.config['SQLALCHEMY_COMMIT_ON_TEARDOWN'] = True
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
##################
### App setup ####
##################
db = SQLAlchemy(app) # For database use
#########################
#########################
######### Everything above this line is important/useful setup,
## not problem-solving.##
#########################
#########################
#########################
##### Set up Models #####
#########################
## TODO 364: Set up the following Model classes, as described, with the respective fields (data types).
## The following relationships should exist between them:
# Tweet:User - Many:One
# - Tweet
## -- id (Integer, Primary Key)
## -- text (String, up to 280 chars)
## -- user_id (Integer, ID of user posted -- ForeignKey)
## Should have a __repr__ method that returns strings of a format like:
#### {Tweet text...} (ID: {tweet id})
# - User
## -- id (Integer, Primary Key)
## -- username (String, up to 64 chars, Unique=True)
## -- display_name (String, up to 124 chars)
## ---- Line to indicate relationship between Tweet and User tables (the 1 user: many tweets relationship)
## Should have a __repr__ method that returns strings of a format like:
#### {username} | ID: {id}
########################
##### Set up Forms #####
########################
# TODO 364: Fill in the rest of the below Form class so that someone running this web app will be able to fill in information about tweets they wish existed to save in the database:
## -- text: tweet text (Required, should not be more than 280 characters)
## -- username: the twitter username who should post it (Required, should not be more than 64 characters)
## -- display_name: the display name of the twitter user with that username (Required, + set up custom validation for this -- see below)
# HINT: Check out index.html where the form will be rendered to decide what field names to use in the form class definition
# TODO 364: Set up custom validation for this form such that:
# - the twitter username may NOT start with an "@" symbol (the template will put that in where it should appear)
# - the display name MUST be at least 2 words (this is a useful technique to practice, even though this is not true of everyone's actual full name!)
# TODO 364: Make sure to check out the sample application linked in the readme to check if yours is like it!
###################################
##### Routes & view functions #####
###################################
## Error handling routes - PROVIDED
@app.errorhandler(404)
def page_not_found(e):
return render_template('404.html'), 404
@app.errorhandler(500)
def internal_server_error(e):
return render_template('500.html'), 500
#############
## Main route
#############
## TODO 364: Fill in the index route as described.
# A template index.html has been created and provided to render what this route needs to show -- YOU just need to fill in this view function so it will work.
# Some code already exists at the end of this view function -- but there's a bunch to be filled in.
## HINT: Check out the index.html template to make sure you're sending it the data it needs.
## We have provided comment scaffolding. Translate those comments into code properly and you'll be all set!
# NOTE: The index route should:
# - Show the Tweet form.
# - If you enter a tweet with identical text and username to an existing tweet, it should redirect you to the list of all the tweets and a message that you've already saved a tweet like that.
# - If the Tweet form is entered and validates properly, the data from the form should be saved properly to the database, and the user should see the form again with a message flashed: "Tweet successfully saved!"
# Try it out in the sample app to check against yours!
@app.route('/', methods=['GET', 'POST'])
def index():
# Initialize the form
# Get the number of Tweets
# If the form was posted to this route,
## Get the data from the form
## Find out if there's already a user with the entered username
## If there is, save it in a variable: user
## Or if there is not, then create one and add it to the database
## If there already exists a tweet in the database with this text and this user id (the id of that user variable above...) ## Then flash a message about the tweet already existing
## And redirect to the list of all tweets
## Assuming we got past that redirect,
## Create a new tweet object with the text and user id
## And add it to the database
## Flash a message about a tweet being successfully added
## Redirect to the index page
# PROVIDED: If the form did NOT validate / was not submitted
errors = [v for v in form.errors.values()]
if len(errors) > 0:
flash("!!!! ERRORS IN FORM SUBMISSION - " + str(errors))
return render_template('index.html',) # TODO 364: Add more arguments to the render_template invocation to send data to index.html
@app.route('/all_tweets')
def see_all_tweets():
pass # Replace with code
# TODO 364: Fill in this view function so that it can successfully render the template all_tweets.html, which is provided.
# HINT: Careful about what type the templating in all_tweets.html is expecting! It's a list of... not lists, but...
# HINT #2: You'll have to make a query for the tweet and, based on that, another query for the username that goes with it...
@app.route('/all_users')
def see_all_users():
pass # Replace with code
# TODO 364: Fill in this view function so it can successfully render the template all_users.html, which is provided.
# TODO 364
# Create another route (no scaffolding provided) at /longest_tweet with a view function get_longest_tweet (see details below for what it should do)
# TODO 364
# Create a template to accompany it called longest_tweet.html that extends from base.html.
# NOTE:
# This view function should compute and render a template (as shown in the sample application) that shows the text of the tweet currently saved in the database which has the most NON-WHITESPACE characters in it, and the username AND display name of the user that it belongs to.
# NOTE: This is different (or could be different) from the tweet with the most characters including whitespace!
# Any ties should be broken alphabetically (alphabetically by text of the tweet). HINT: Check out the chapter in the Python reference textbook on stable sorting.
# Check out /longest_tweet in the sample application for an example.
# HINT 2: The chapters in the Python reference textbook on:
## - Dictionary accumulation, the max value pattern
## - Sorting
# may be useful for this problem!
if __name__ == '__main__':
db.create_all() # Will create any defined models when you run the application
app.run(use_reloader=True,debug=True) # The usual