2018-05-20 00:10:01 +02:00
|
|
|
"""
|
|
|
|
Flask Documentation: http://flask.pocoo.org/docs/
|
|
|
|
Jinja2 Documentation: http://jinja.pocoo.org/2/documentation/
|
|
|
|
Werkzeug Documentation: http://werkzeug.pocoo.org/documentation/
|
|
|
|
This file creates your application.
|
|
|
|
"""
|
|
|
|
|
|
|
|
from app import app, db
|
|
|
|
from flask import Flask, render_template, request, redirect, url_for, flash, send_from_directory, jsonify, abort
|
2018-05-29 16:18:00 +02:00
|
|
|
import json
|
2018-05-31 18:26:22 +02:00
|
|
|
from sqlalchemy.sql.expression import func, select
|
|
|
|
from app.forms import UploadForm, EditForm, SearchForm
|
2018-06-01 23:27:26 +02:00
|
|
|
from app.models import Book, BookSchema, Author, AuthorSchema, Stack, StackSchema, UserIns
|
2018-05-23 14:12:24 +02:00
|
|
|
from app.cover import get_cover
|
2018-05-20 00:10:01 +02:00
|
|
|
|
|
|
|
import os
|
|
|
|
from werkzeug.utils import secure_filename
|
2018-05-23 14:12:24 +02:00
|
|
|
|
2018-05-20 00:10:01 +02:00
|
|
|
# import sqlite3
|
2018-05-31 20:06:13 +02:00
|
|
|
ALLOWED_EXTENSIONS = set(['txt', 'pdf', 'epub', 'chm', 'mobi'])
|
2018-05-20 00:10:01 +02:00
|
|
|
|
2018-05-27 22:11:35 +02:00
|
|
|
author_schema = AuthorSchema()
|
|
|
|
authors_schema = AuthorSchema(many=True)
|
2018-05-20 00:10:01 +02:00
|
|
|
book_schema = BookSchema()
|
|
|
|
books_schema = BookSchema(many=True)
|
2018-06-01 20:26:37 +02:00
|
|
|
stack_schema = StackSchema()
|
|
|
|
stacks_schema = StackSchema(many=True)
|
2018-05-20 00:10:01 +02:00
|
|
|
|
|
|
|
def allowed_file(filename):
|
|
|
|
return '.' in filename and \
|
|
|
|
filename.rsplit('.', 1)[1].lower() in ALLOWED_EXTENSIONS
|
|
|
|
###
|
|
|
|
# Routing for your application.
|
|
|
|
###
|
|
|
|
|
|
|
|
@app.route('/')
|
|
|
|
def home():
|
|
|
|
"""Render website's home page."""
|
|
|
|
return render_template('home.html')
|
|
|
|
|
2018-05-23 14:12:24 +02:00
|
|
|
@app.route('/hello/<name>')
|
|
|
|
def hello(name):
|
|
|
|
return "Hello " + name
|
2018-05-20 00:10:01 +02:00
|
|
|
|
|
|
|
@app.route('/about/')
|
|
|
|
def about():
|
|
|
|
"""Render the website's about page."""
|
|
|
|
return render_template('about.html', name="Mary Jane")
|
|
|
|
|
|
|
|
@app.route('/uploads/<filename>')
|
|
|
|
def uploaded_file(filename):
|
|
|
|
return send_from_directory(app.config['UPLOAD_FOLDER'],
|
|
|
|
filename)
|
|
|
|
|
2018-05-23 14:12:24 +02:00
|
|
|
@app.route('/uploads/cover/<filename>')
|
|
|
|
def uploaded_file_cover(filename):
|
|
|
|
return send_from_directory(app.config['UPLOAD_FOLDER_COVER'],
|
|
|
|
filename)
|
|
|
|
|
2018-05-31 18:26:22 +02:00
|
|
|
@app.route('/books', methods= ['POST','GET'])
|
2018-05-20 00:10:01 +02:00
|
|
|
def show_books():
|
2018-05-31 18:26:22 +02:00
|
|
|
books = db.session.query(Book).all()
|
|
|
|
search = SearchForm(request.form)
|
|
|
|
if request.method == 'POST':
|
2018-06-01 17:20:12 +02:00
|
|
|
if search.select.data == 'Title':
|
|
|
|
return redirect((url_for('search_results', query=search.search.data)))
|
|
|
|
if search.select.data == 'Category':
|
|
|
|
return redirect((url_for('search_cat', query=search.search.data)))
|
2018-06-01 20:40:45 +02:00
|
|
|
|
2018-05-31 18:26:22 +02:00
|
|
|
return render_template('show_books.html', books=books, form=search)
|
2018-05-20 00:10:01 +02:00
|
|
|
|
2018-06-01 23:27:26 +02:00
|
|
|
@app.route('/updates', methods=['POST', 'GET'])
|
|
|
|
def get_updates():
|
|
|
|
userin = UserIns.query.filter_by(title="lastViewed").first()
|
2018-06-01 23:53:25 +02:00
|
|
|
return "XPPL / / / / / / / Last viewed: " + userin.info + " / / / / / / / "
|
2018-06-01 23:27:26 +02:00
|
|
|
|
2018-05-29 16:18:00 +02:00
|
|
|
@app.route('/scape', methods=['POST', 'GET'])
|
|
|
|
def scape():
|
|
|
|
if request.method == 'POST':
|
|
|
|
data = request.form
|
|
|
|
book = Book.query.get(data['id'])
|
|
|
|
print(book.scapeX)
|
|
|
|
book.scapeX = data['x']
|
|
|
|
book.scapeY = data['y']
|
|
|
|
db.session.commit()
|
|
|
|
books = db.session.query(Book).all() # or you could have used User.query.all()
|
|
|
|
return render_template('scape.html', books=books)
|
|
|
|
|
|
|
|
|
|
|
|
@app.route('/books_grid')
|
|
|
|
def show_books_grid():
|
|
|
|
books = db.session.query(Book).all() # or you could have used User.query.all()
|
|
|
|
return render_template('show_books_grid.html', books=books)
|
|
|
|
|
2018-05-20 00:10:01 +02:00
|
|
|
@app.route('/books/<int:id>')
|
|
|
|
def show_book_by_id(id):
|
|
|
|
book = Book.query.get(id)
|
2018-06-01 23:27:26 +02:00
|
|
|
userin = UserIns.query.filter_by(title="lastViewed").first()
|
|
|
|
if userin != None:
|
|
|
|
userin.info = book.title
|
|
|
|
db.session.commit()
|
|
|
|
else:
|
|
|
|
user_info = UserIns("lastViewed", book.title)
|
|
|
|
db.session.add(user_info)
|
|
|
|
db.session.commit()
|
2018-05-20 00:10:01 +02:00
|
|
|
if not book:
|
2018-05-23 14:12:24 +02:00
|
|
|
return render_template('red_link.html', id=id)
|
2018-05-20 00:10:01 +02:00
|
|
|
else:
|
|
|
|
return render_template('show_book_detail.html', book=book)
|
|
|
|
|
2018-05-23 17:50:01 +02:00
|
|
|
|
2018-05-23 14:12:24 +02:00
|
|
|
@app.route('/books/<int:id>/delete', methods=['POST', 'GET'])
|
|
|
|
def remove_book_by_id(id):
|
|
|
|
book_to_edit = Book.query.filter_by(id=id).first()
|
|
|
|
title = book_to_edit.title
|
|
|
|
Book.query.filter_by(id=id).delete()
|
2018-06-01 20:26:37 +02:00
|
|
|
#author_table = Author.query.filter_by(book_id=book_to_edit.id).delete()
|
2018-05-23 14:12:24 +02:00
|
|
|
db.session.commit()
|
|
|
|
flash("%s deleted from library" % (title))
|
|
|
|
return redirect(url_for('show_books'))
|
|
|
|
|
|
|
|
@app.route('/books/<int:id>/edit', methods=['POST', 'GET'])
|
|
|
|
def edit_book_by_id(id):
|
|
|
|
book_to_edit = Book.query.filter_by(id=id).first()
|
2018-06-01 21:37:45 +02:00
|
|
|
user_form = EditForm(title = book_to_edit.title, author =book_to_edit.authors, category = book_to_edit.category, year_published= book_to_edit.year_published)
|
2018-05-23 14:12:24 +02:00
|
|
|
|
|
|
|
if request.method == 'POST':
|
|
|
|
if user_form.validate_on_submit():
|
|
|
|
# check if the post request has the file part
|
|
|
|
title = user_form.title.data # You could also have used request.form['name']
|
2018-05-31 17:30:16 +02:00
|
|
|
input_authors = user_form.author.data # You could also have used request.form['email']
|
2018-05-27 22:11:35 +02:00
|
|
|
category = user_form.category.data
|
2018-06-01 21:37:45 +02:00
|
|
|
year_published = user_form.year_published.data
|
|
|
|
if year_published=="":
|
|
|
|
year_published = None
|
2018-05-23 14:12:24 +02:00
|
|
|
# save user to database
|
|
|
|
#book = Book(title, author, filename, cover, file_extension)
|
|
|
|
|
2018-05-23 18:31:03 +02:00
|
|
|
book = Book.query.filter_by(id=id).first()
|
|
|
|
book.title = title
|
2018-05-27 22:11:35 +02:00
|
|
|
book.category = category
|
2018-06-01 21:37:45 +02:00
|
|
|
book.year_published = year_published
|
2018-05-31 17:30:16 +02:00
|
|
|
|
2018-06-01 21:37:45 +02:00
|
|
|
#authors update
|
2018-06-01 20:26:37 +02:00
|
|
|
book.authors.clear()
|
|
|
|
for i, author in enumerate(input_authors):
|
|
|
|
author_name = author.get("author_name")
|
|
|
|
if author_name:
|
|
|
|
a = db.session.query(Author).filter_by(author_name=author_name).first()
|
|
|
|
if a == None:
|
|
|
|
a = Author(author_name=author_name)
|
|
|
|
db.session.add(a)
|
|
|
|
book.authors.append(a)
|
2018-05-23 18:31:03 +02:00
|
|
|
db.session.commit()
|
2018-05-23 14:12:24 +02:00
|
|
|
flash("%s updated" % (title))
|
2018-06-01 21:37:45 +02:00
|
|
|
return redirect(url_for('show_book_by_id', id=id))
|
2018-05-23 14:12:24 +02:00
|
|
|
|
|
|
|
return render_template('edit_book_detail.html', book=book_to_edit, form=user_form)
|
|
|
|
|
2018-05-20 00:10:01 +02:00
|
|
|
|
|
|
|
@app.route('/add-book', methods=['POST', 'GET'])
|
|
|
|
def add_book():
|
2018-05-31 18:26:22 +02:00
|
|
|
upload_form = UploadForm()
|
2018-05-31 20:06:13 +02:00
|
|
|
|
2018-05-20 00:10:01 +02:00
|
|
|
if request.method == 'POST':
|
2018-05-31 18:26:22 +02:00
|
|
|
if upload_form.validate_on_submit():
|
2018-06-01 21:37:45 +02:00
|
|
|
#get data from form
|
|
|
|
title = upload_form.title.data
|
|
|
|
authors = upload_form.author.data
|
|
|
|
category = upload_form.category.data
|
|
|
|
year_published = upload_form.year_published.data
|
|
|
|
if year_published=="":
|
|
|
|
year_published = None
|
|
|
|
|
|
|
|
#if upload with file
|
2018-05-31 18:26:22 +02:00
|
|
|
if upload_form.upload.data:
|
|
|
|
# check if the post request has the file part
|
|
|
|
if 'file' not in request.files:
|
|
|
|
flash('No file part')
|
|
|
|
return redirect(request.url)
|
|
|
|
file = request.files['file']
|
|
|
|
# if user does not select file, browser also
|
|
|
|
# submit a empty part without filename
|
|
|
|
if file.filename == '':
|
|
|
|
flash('No selected file')
|
|
|
|
return redirect(request.url)
|
|
|
|
if file and allowed_file(file.filename):
|
|
|
|
filename = secure_filename(file.filename)
|
2018-05-31 20:06:13 +02:00
|
|
|
allbooks = db.session.query(Book).all()
|
|
|
|
id = len(allbooks)+1
|
|
|
|
new_filename = str(id) +"_"+ filename
|
|
|
|
fullpath = os.path.join(app.config['UPLOAD_FOLDER'], new_filename)
|
|
|
|
name, file_extension = os.path.splitext(new_filename)
|
2018-05-31 18:26:22 +02:00
|
|
|
file.save(fullpath)
|
|
|
|
cover = get_cover(fullpath, name)
|
|
|
|
else:
|
|
|
|
flash('allowed file formats: %s' % ALLOWED_EXTENSIONS)
|
2018-06-01 21:37:45 +02:00
|
|
|
#if upload without file -> wishform, with potential PDF
|
2018-05-31 18:26:22 +02:00
|
|
|
if upload_form.wish.data:
|
2018-06-01 21:37:45 +02:00
|
|
|
#TO DO: make pdf generator
|
|
|
|
#file = open('app/uploads/potential.pdf')
|
|
|
|
#filename = 'potential.pdf'
|
|
|
|
#file_extension = '.pdf'
|
|
|
|
filename = ''
|
|
|
|
file_extension = ''
|
2018-06-01 17:20:12 +02:00
|
|
|
cover = ''
|
2018-06-01 21:37:45 +02:00
|
|
|
|
|
|
|
book = Book(title, filename, cover, file_extension, category,year_published)
|
|
|
|
db.session.add(book)
|
|
|
|
for author in authors:
|
|
|
|
author_name = author.get("author_name")
|
|
|
|
if author_name:
|
|
|
|
a = db.session.query(Author).filter_by(author_name=author_name).first()
|
|
|
|
if a == None:
|
|
|
|
a = Author(author_name=author_name)
|
|
|
|
db.session.add(a)
|
|
|
|
book.authors.append(a)
|
|
|
|
db.session.commit()
|
2018-05-23 14:12:24 +02:00
|
|
|
|
2018-05-31 18:26:22 +02:00
|
|
|
flash("%s added to the library" % (title))
|
|
|
|
return redirect(url_for('show_books'))
|
2018-05-23 14:12:24 +02:00
|
|
|
|
2018-05-31 18:26:22 +02:00
|
|
|
flash_errors(upload_form)
|
|
|
|
return render_template('add_book.html', form=upload_form)
|
|
|
|
|
2018-05-20 00:10:01 +02:00
|
|
|
|
|
|
|
# Flash errors from the form if validation fails
|
|
|
|
def flash_errors(form):
|
|
|
|
for field, errors in form.errors.items():
|
|
|
|
for error in errors:
|
|
|
|
flash(u"Error in the %s field - %s" % (
|
|
|
|
getattr(form, field).label.text,
|
|
|
|
error
|
|
|
|
))
|
|
|
|
|
2018-05-23 18:11:17 +02:00
|
|
|
#Authors
|
2018-05-23 17:50:01 +02:00
|
|
|
|
2018-05-23 17:53:30 +02:00
|
|
|
@app.route('/authors/<int:id>')
|
|
|
|
def show_author_by_id(id):
|
|
|
|
author = Author.query.get(id)
|
|
|
|
if not author:
|
2018-05-23 17:50:01 +02:00
|
|
|
abort (404)
|
2018-05-23 17:53:30 +02:00
|
|
|
else:
|
|
|
|
return render_template('show_author_detail.html', author=author)
|
2018-05-20 00:10:01 +02:00
|
|
|
|
2018-05-23 17:50:01 +02:00
|
|
|
|
|
|
|
@app.route('/authors/<int:id>/edit', methods=['POST', 'GET'])
|
|
|
|
def edit_author_by_id(id):
|
2018-05-27 22:11:35 +02:00
|
|
|
return "Ask the programmer."
|
2018-05-23 17:50:01 +02:00
|
|
|
|
2018-05-31 18:04:05 +02:00
|
|
|
##stacks
|
|
|
|
|
|
|
|
@app.route('/stacks')
|
|
|
|
def show_stacks():
|
|
|
|
stacks = db.session.query(Stack).all()
|
|
|
|
return render_template('show_stacks.html', stacks=stacks)
|
|
|
|
|
|
|
|
@app.route('/stacks/<int:id>', methods=['POST', 'GET'])
|
|
|
|
def show_stack_by_id(id):
|
|
|
|
stack = Stack.query.get(id)
|
|
|
|
if not stack:
|
|
|
|
abort (404)
|
|
|
|
else:
|
|
|
|
return render_template('show_stack_detail.html', stack=stack)
|
2018-05-23 17:50:01 +02:00
|
|
|
|
2018-05-31 18:26:22 +02:00
|
|
|
## search
|
|
|
|
|
2018-06-01 17:20:12 +02:00
|
|
|
@app.route('/search/titles/<query>/', methods=['POST', 'GET'])
|
2018-05-31 18:26:22 +02:00
|
|
|
def search_results(query):
|
|
|
|
search = SearchForm(request.form)
|
|
|
|
random_order=Book.query.order_by(func.random()).limit(10)
|
|
|
|
results=Book.query.filter(Book.title.contains(query)).all()
|
|
|
|
|
|
|
|
if not results:
|
|
|
|
upload_form = UploadForm(title= query, author='')
|
|
|
|
return render_template('red_link.html', form=upload_form, title=query)
|
|
|
|
|
|
|
|
if request.method == 'POST':
|
|
|
|
query = search.search.data
|
|
|
|
results = []
|
2018-06-01 17:20:12 +02:00
|
|
|
if search.select.data == 'Title':
|
|
|
|
return redirect((url_for('search_results', query=search.search.data)))
|
|
|
|
if search.select.data == 'Category':
|
|
|
|
return redirect((url_for('search_cat', query=search.search.data)))
|
2018-05-31 18:26:22 +02:00
|
|
|
|
|
|
|
return render_template('results.html', form=search, books=results, books_all=random_order, query=query)
|
|
|
|
|
2018-06-01 17:20:12 +02:00
|
|
|
@app.route('/search/cat/<query>/', methods=['POST', 'GET'])
|
|
|
|
def search_cat(query):
|
|
|
|
search = SearchForm(request.form)
|
|
|
|
random_order=Book.query.order_by(func.random()).limit(10)
|
|
|
|
results=Book.query.filter(Book.category.contains(query)).all()
|
|
|
|
|
|
|
|
if not results:
|
|
|
|
upload_form = UploadForm(category=query)
|
|
|
|
return render_template('red_link.html', form=upload_form, category=query)
|
|
|
|
|
|
|
|
if request.method == 'POST':
|
|
|
|
query = search.search.data
|
|
|
|
results = []
|
|
|
|
if search.select.data == 'Title':
|
|
|
|
return redirect((url_for('search_results', query=search.search.data)))
|
|
|
|
if search.select.data == 'Category':
|
|
|
|
return redirect((url_for('search_cat', query=search.search.data)))
|
|
|
|
|
|
|
|
return render_template('results.html', form=search, books=results, books_all=random_order, query=query)
|
2018-05-20 00:10:01 +02:00
|
|
|
|
|
|
|
###
|
|
|
|
# The API
|
|
|
|
###
|
|
|
|
|
|
|
|
@app.route('/api/books', methods=['GET'])
|
|
|
|
def get_books():
|
|
|
|
books = Book.query.all()
|
2018-06-01 20:26:37 +02:00
|
|
|
data = books_schema.dump(books)
|
|
|
|
#print(errors)
|
2018-05-20 00:10:01 +02:00
|
|
|
return jsonify({'books': data})
|
|
|
|
|
|
|
|
@app.route('/api/books/<int:id>', methods=['GET'])
|
|
|
|
def get_book_by_id(id):
|
|
|
|
book = Book.query.get(id)
|
2018-06-01 20:40:45 +02:00
|
|
|
data = book_schema.dump(book)
|
2018-05-27 22:11:35 +02:00
|
|
|
if not data:
|
2018-05-20 00:10:01 +02:00
|
|
|
return jsonify({"message": "Book could not be found."}), 400
|
|
|
|
else:
|
2018-05-27 22:11:35 +02:00
|
|
|
return jsonify({'book': data })
|
2018-05-20 00:10:01 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
###
|
|
|
|
# The functions below should be applicable to all Flask apps.
|
|
|
|
###
|
|
|
|
|
|
|
|
@app.route('/<file_name>.txt')
|
|
|
|
def send_text_file(file_name):
|
|
|
|
"""Send your static text file."""
|
|
|
|
file_dot_text = file_name + '.txt'
|
|
|
|
return app.send_static_file(file_dot_text)
|
|
|
|
|
|
|
|
|
|
|
|
@app.after_request
|
|
|
|
def add_header(response):
|
|
|
|
"""
|
|
|
|
Add headers to both force latest IE rendering engine or Chrome Frame,
|
|
|
|
and also to cache the rendered page for 10 minutes.
|
|
|
|
"""
|
|
|
|
response.headers['X-UA-Compatible'] = 'IE=Edge,chrome=1'
|
|
|
|
response.headers['Cache-Control'] = 'public, max-age=600'
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
|
@app.errorhandler(404)
|
|
|
|
def page_not_found(error):
|
|
|
|
"""Custom 404 page."""
|
|
|
|
return render_template('404.html'), 404
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
app.run(debug=True,host="0.0.0.0",port="8080")
|