摘要

记载了「通往全栈之路上」的一则edx慕课作业:用Flask框架写一个图书查询的web应用。
【honor code警告】如果你刚巧也注册了这门课,千万不要抄。

成品效果视频 @ 优酷:

缘起

出于不可自拔的技能焦虑,跑到edx上撸起了课——其实论技术课程,Udacity比edx和Coursera要好,但edx也不赖(就是国内访问越来越困难,经常加载不出来)——撸到一门偏前端的纯码农培训课。说出来吓死人,哈佛**【继续教育学院】**(对,就是范玮琪读的那个哈佛)开的用Python和Javascript撸网络编程。它的主要卖点是教小白怎么用Flask框架搭网站,就是号称一个 .py + 一个 .html就能欢快地跑出 hello world 来的小快灵建站利器。

想想看,上一个号称小快灵的神器还是PHP呢。那都是二十年前的事儿了。

Flask最好的好处是可以多快好省地做网站,迅速实现一个原型或弄出一个0.1功能版本。若将来再学一点小程序啥的,起码搞起数据科学工程产品来,能派上一丢丢的用场——也算走向「全网没人肯要的中老年」全栈工程师的第一步罢。

事实很打脸。这个作业只是整个课程的五大作业里的一个,我拿出所有业余时间埋头苦干,做了足足两个礼拜。以这个效率去搬砖,你猜老板会用什么武功揍我?

作业要求

要求原文在此:做个图书+书评查询网络应用。大致要求:

  1. 首先,自己想辙把压缩包里的book.csv 5000条图书信息导进数据库里。

    1. heroku上注册创建数据库实例,订阅一个乞丐版就行;
    2. 数据库用PostgreSQL
  2. 然后,该应用要有以下功能

    1. 能注册
    2. 能登录
    3. 能注销
    4. 能根据ISBN、书名、作者查询书籍
    5. 能点进具体一本书里
      1. 除了固有信息,还要利用Goodreads的API获取平均评分
      2. 能看到其他用户发的书评和评级
    6. 能发书评和评级,但一个用户只能发一次
    7. 能暴露API给人家用,返回一个JSON串

准备

数据库

heroku已经被salesforce.com买了,以傻瓜式建站和贵著称。我也跑上去建了一个实例,但是要翻墙。反正就是个作业,哪家SQL不是SQL?所以我就本地弄了个sqlite数据库db.db

建三张表,mbrbookreviewreview表里mbr_idbook_id分别外键关联到mbrbook

为啥不用'user'?

因为postgresql不同意我用这个表名,所以在sqlite里也这么干。

图 | 数据库设计

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
CREATE TABLE IF NOT EXISTS book (
    id INTEGER PRIMARY KEY,
    isbn TEXT NOT NULL,
    title TEXT NOT NULL,
    author TEXT NOT NULL,
    year INTEGER NOT NULL
);
CREATE TABLE IF NOT EXISTS mbr (
    id INTEGER PRIMARY KEY,
    username TEXT NOT NULL,
    pwd TEXT NOT NULL
);
CREATE TABLE IF NOT EXISTS review (
    id INTEGER PRIMARY KEY,
    rev_at TEXT NOT NULL DEFAULT (strftime('%Y-%m-%d %H:%M:%S', 'now')),
    mbr_id INTEGER NOT NULL,
    book_id INTEGER NOT NULL,
    rating INTEGER,
    review TEXT
);

备注

以上是sqlite的建表DDL,如果用PostgreSQL,语句略有不同。

再把books.csv里的数据导进去。

我是坚定的pandas粉,所以直接把csv读进pandas再一口气灌进sqlite里。面对postgresql我也这么干。传统的方法是用csv包一行一行扫描,再写入数据库。我是向量运算的刀山火海里捶打出来的,轻易才不用循环。

import_local.py部分代码如下。如果用heroku上的PostgreSQL,则用import.py

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
# import_local.py
import sqlite3
import pandas as pd
conn = sqlite3.connect('db.db')
cur = conn.cursor()
if cur.execute('select count(*) from book;').fetchone()[0] == 0:
    df = pd.read_csv(r'books.csv')
    df.to_sql(name='book', con=conn, if_exists='append', index=False)
cur.close()
conn.close()

其它

当然,python那边得把python装好。我就直接用anaconda了。此外,需要把FlaskFlask-Sessionpsycopg2-binarySQKAlchemy几个包都装上,conda/pip爱谁谁。

Goodreads是个书评网站(也被墙了-_-||)。需要自己上去注册账号,申请API开发密钥。

开工

项目结构

源代码托管于Github

戳这里看源码

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
project1
|-- application.py
|--+ static
|  |--+ css
|  |  |-- style.css
|  |  `-- star-rating.min.css
|  `--+ js
|     |-- main.js
|     `-- star-rating.min.js
|--+ templates
|  |-- base.html
|  |-- book.html
|  |-- index.html
|  |-- login.html
|  `-- register.html
`-- db.db

这个应用比较简单,所以结构很扁平。

  • application.py是后端,所有后台功能代码都写在上面。
  • static文件夹放静态文件,css和js这种(叫assets也行)。
    • js: 放了自定义的main.js,和用于打分评级的star-rating的js
    • css: 放了自定义的style.css,和star-rating的css
  • templates文件夹放各类html模板,这些模板都是用html+jinja2语法写的宏。包括基础模板base.html和几个衍生的功能性模板

基础模板

base.html是框架模板。简单写一下。

  • 样式主要靠Bootstrap
  • body部分放了几个通用块(block):head, flash, disp, control, misc。用jinja2结构{% block xxx %}{% endblock %}来占位。
    • 块里面基本都没有进一步定义。只是给导航条加了点功能,如果当前线程有用户登着,就显示个注销按钮,否则就没有。
    • flash块比较特别,定义了一个比较通用的flash渲染宏,到时候只需要在后台.py里套用flash函数就能实现告警框。
    • 后续写其他模板时,引用(extend) base.html就行了。
 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
<!-- templates/base.html -->
<!DOCTYPE html>
<html lang='en'>
    <head>
        <meta CharacterSet='UTF-8'>
        <meta http-equiv="X-UA-Compatible" content="IE=edge">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <title>{% block title %}{% endblock %}</title>
        <link rel="stylesheet" href="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/css/bootstrap.min.css">
        <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@3.3.7/dist/css/bootstrap-theme.min.css">
        <link rel="stylesheet" href="https://cdn.staticfile.org/font-awesome/4.7.0/css/font-awesome.css">
        <link rel="stylesheet" href="{{ url_for('static', filename='css/style.css') }}">
        <link rel="stylesheet" href="{{ url_for('static', filename='css/star-rating.min.css') }}">
        <script src="https://code.jquery.com/jquery-3.4.1.min.js"></script>
        <script src="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/js/bootstrap.min.js"></script>
        <script src="{{ url_for('static', filename='js/main.js') }}"></script>
        <script src="{{ url_for('static', filename='js/star-rating.min.js') }}"></script>
    </head>
    <body>
        <!-- head -->
        <main>
            <nav id="navbar" class="navbar navbar-default" role="navigation">
                <div class="container-fluid">
                    <div class="navbar-header">
                        <a class="navbar-brand mb-0" href="#">Book Review</a>
                    </div>
                    {% if act_user is not none %}
                    <form class="navbar-form navbar-right" 
                     action="{{ url_for('sign_off') }}" method="get">
                        <span>Welcome, {{ act_user['username'] }}.&nbsp;&nbsp;</span>
                        <button id="logout" class="btn btn-default btn-sm">
                          Log out
                        </button>
                    </form>
                    {% endif %}
                </div>
            </nav>
            <!-- flash -->
            {% with messages = get_flashed_messages(with_categories=true) %}
                {% if messages %}
                    {% for category, message in messages %}
                    <div class="alert alert-{{ category }} alert-dismissable">
                        <button type="button" class="close" data-dismiss="alert">
                          &times;
                        </button>
                        {{ message }}
                    </div>
                    {% endfor %}
                {% endif %}
            {% endwith %}
            <!-- body -->
            <div class="container" id="div_ctrl">
                {% block control %}
                {% endblock %}            
            </div>
            <div class="container" id="div_disp">
                {% block disp %}
                {% endblock %}            
            </div>
            <div class="container" id="div_misc">
                {% block misc %}
                {% endblock %}            
            </div>
        </main>
        <!-- bottom -->
        <nav class="navbar navbar-fixed-bottom" id="nav_footer" role="navigation">
            <div class="container" id="elem_foot">
                <footer class="navbar" id="footer">
                    <hr>
                    {% block botm_cont %}
                    {% endblock %}
                    <p>&copy; 2019 madlogos</p>
                </footer>
            </div>
        </nav>
    </body>
</html>

对应地,在application.py里定义一些基本代码。

  • “db.db"被export到环境变量DATABASE_URL,为了避免thread不一致问题,写成"sqlite:////absolute/path/to/db?check_same_thread=false”。
  • application.py被export到环境变量FLASK_APP,方便后面直接命令行flask run启动应用。
 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
# -*- coding: UTF-8 -*-
# application.py
import os
import requests
from flask import Flask, flash, jsonify, render_template, request, \
    redirect, session, url_for
from jinja2 import Markup
from flask_session import Session
from sqlalchemy import create_engine
from sqlalchemy.orm import scoped_session, sessionmaker

app = Flask(__name__, static_folder='static')


# Check for environment variable
if not os.getenv("DATABASE_URL"):
    raise RuntimeError("DATABASE_URL is not set")

# Configure session to use filesystem
app.config["SESSION_PERMANENT"] = False
app.config["SESSION_TYPE"] = "filesystem"
Session(app)

# Set up database
engine = create_engine(os.getenv("DATABASE_URL"))
db = scoped_session(sessionmaker(bind=engine))
sess = db()


@app.teardown_request
def remove_session(ex=None):
    db.remove()

登录

访问首页,先跳转到登录界面。

图 | 登录界面

登录页login.html很简单,首先继承base.html的元素,然后在control块里放一个form-signin控件。套了一些bootstrap的元素。action绑定sign_in,也就是signin()函数。

对前端技术不熟,丑得很。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
<!-- templates/login.html -->
{% extends "base.html" %}

{% block title %}
Sign In
{% endblock %}

{% block control %}
<form class="form-signin" action="{{ url_for('sign_in') }}" method="post">
    <h2 class="form-signin-heading">Please Sign In</h2>
    <label for="username" class="sr-only">Enter Your Username</label>
    <input type="text" name="username" class="form-control" placeholder="Username">
    <label for="password" class="sr-only">Enter Your Password</label>
    <input type="password" name="password" class="form-control" placeholder="Password">
    <label for="signIn" class="sr-only">Click</label>
    <button id="signIn" class="btn btn-lg btn-primary btn-block" >Sign In</button>        
</form>
<form class="form-signin" action="{{ url_for('sign_up') }}" method="get">
    <button id="signUp" class="btn btn-lg btn-default btn-block">Sign up now!</button>
</form>
{% endblock %}

application.py里,后台部分写两个路由函数。

主路由下,如果当前session没有用户登录,就转跳去登录页/login,否则直接进书籍列表页/index。

1
2
3
4
5
6
7
8
9
# application.py
@app.route("/", methods=['GET'])
def home():
    """Home page
    """
    if session.get('act_user') is None:
        return redirect(url_for("sign_in"))
    else:
        return render_template("index.html", act_user=session.get('act_user'))

如果进登录页,那么’GET’方法下跟主路由差不多逻辑,‘POST’方法下(点按钮触发POST),就要校验用户名密码了。成功就进书籍列表,假如不对,就flash一个错误来。利用application.py里定义的sign_in()函数和模板form中的url_for()函数,就把后端功能绑定到前端了。

 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
# application.py
@app.route("/login", methods=['GET', 'POST'])
def sign_in():
    """Sign in
    """
    if request.method == 'GET':
        if session.get('act_user') is None:
            return render_template(
                "login.html", act_user=session.get("act_user"))
        else:
            return render_template(
                "index.html", act_user=session.get("act_user"))
    elif request.method == 'POST':
        username = request.form.get('username')
        pwd = request.form.get('password')
        mbrinfo = sess.execute(
            """SELECT id, username FROM mbr WHERE username = :username
            AND pwd = :pwd;""",
            {"username": username, "pwd": pwd}).fetchone()
        if mbrinfo is None:
            flash(Markup(
                """<i class='fa fa-2x fa-exclamation-circle'></i>
                User not exist or wrong password."""), 
                'danger')
        else:
            session['act_user'] = {'id': mbrinfo[0], 'username': mbrinfo[1]}
        return home()

Flask是用SQLAlchemy的。SQLAlchemy是很高效的ORM工具,坊间一直认为它好过Django自带的ORM。不过这门课要求不用ORM,直接硬写SQL。当然,不是直接字符串拼接那么土,还是套了一个模板,变量装在字典里映射过去。这样能避免SQL注入攻击,算基本操作了。

注销

有登陆就有注销。反正base.html里注销按钮已经绑定了logout路由,所以只要定义logout路由的后台绑定函数就行了。登出后,清空session['act_use']对象,回到登录页。

1
2
3
4
5
6
7
8
# application.py
@app.route('/logout', methods=['GET'])
def sign_off():
    session.pop('act_user', None)
    flash(Markup(
        """<i class='fa fa-2x fa-check-square-o'></i>
        You have logged out."""), 'success')
    return home()

注册

图 | 注册页

注册页register.html和登录页差不多。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
<!-- templates/register.html -->
{% extends "base.html" %}

{% block title %}
Sign Up
{% endblock %}

{% block control %}
<form class="form-signin" action="{{ url_for('sign_up') }}" method="post">
    <h2 class="form-signin-heading">Register An Account</h2>
    <label for="username" class="sr-only">Enter Your Username</label>
    <input type="text" name="username" class="form-control" placeholder="Username" required>
    <label for="password" class="sr-only">Enter Your Password</label>
    <input type="password" name="password" class="form-control" placeholder="Password" required>
    <label for="repassword" class="sr-only">Confirm Your Password</label>
    <input type="password" name="repassword" class="form-control" placeholder="Re-input Password" required>
    <button id="signUp" class="btn btn-lg btn-primary btn-block">Sign Up</button>
</form>
<form class="form-signin" action="{{ url_for('sign_in') }}" method="get">
    <button id="signIn" class="btn btn-lg btn-default btn-block">Sign In Now</button>
</form>
{% endblock %}

后端分别对’GET’和’POST’两种方法做了定义。GET的话,渲染注册页模板而已。POST的话,就比较pwd和repwd的值是否相同,然而提交执行INSERT操作。继续转眺回登录页。

考究点的话当然还要有反机器人的措施。我是那种考究的人嘛?作业又没这要求,就不贴金了。

 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
# application.py
@app.route("/signup", methods=['GET', 'POST'])
def sign_up():
    """Sign up
    """
    if request.method == 'GET':
        return render_template("register.html", act_user=None)
    elif request.method == 'POST':
        username = request.form.get('username')
        pwd = request.form.get('password')
        repwd = request.form.get('repassword')
        mbrinfo = sess.execute(
            """SELECT id, username FROM mbr WHERE username = :username;""", 
            {"username": username}).fetchone()
        if mbrinfo is not None:
            flash(Markup(
                """<i class='fa fa-2x fa-check-square-o'></i>
                The username has been registered. Please change one."""), 
                'warning')
            return redirect(url_for('sign_up'))
        else:
            if pwd == repwd:
                sess.execute(
                    'INSERT INTO mbr (username, pwd) VALUES (:username, :pwd);',
                    {'username': username, 'pwd': pwd})
                sess.commit()
                flash(Markup(
                    """<i class='fa fa-2x fa-check-square-o'></i>
                    You have successfully created a new account. Now sign in."""), 
                    'success')
                return redirect(url_for("sign_in"))
            else:
                flash(Markup(
                    """<i class='fa fa-2x fa-exclamation-circle'></i>
                    You did not input the same password."""), 'danger')
                return redirect(url_for('sign_up'))

检索书籍

图 | 初始载入空列表

登录进去后,进入真正的index.html页。通过三个文本框联合查询。在block disp部分,写一个jinja2宏循环,把books这个对象逐个解析出来填进表格里。如果什么条件都不给,那就会一口气查出5000条来。

此处应有分页。我用了一个插件Flask-Paginate来实现。

安装Flask-Paginate

Flask虽然好上手,但什么功能都要自己写,比较上头。好在还是有好心人做了不少插件。比如这款小巧的Flask_Paginate

1
pip install flask-paginate

模板

前端模板主要就是遍历books,把列表对象逐个填进表格td里。

 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
<!-- templates/books.html -->
{% extends "base.html" %}

{% block title %}
Books
{% endblock %}

{% block control %}
<form class="form-inline" action="{{ url_for('index') }}" method="post">
    <h2 class="form-group-heading">Search Books</h2>
    <div class="form-group">
        <label for="isbn" class="sr-only">ISBN</label>
        <input type="text" name="isbn" class="form-control" placeholder="ISBN">
        <label for="title" class="sr-only">Book title</label>
        <input type="text" name="title" class="form-control" placeholder="Book title">
        <label for="author" class="sr-only">Book author</label>
        <input type="text" name="author" class="form-control" placeholder="Book author">
        <label for="search" class="sr-only">Search</label>
        <button id="search" class="btn btn-lg btn-primary">Search</button>
    </div>       
</form>
{% endblock %}

{% block disp %}

{% if books|length > 0 %}
{{ pagination.info }}
{{ pagination.links }}
<div class="table-responsive">
    <table id="tbl_books" class="table table-striped table-hover" cellspacing="0" width="80%">
        <caption>Books at a glance</caption>
        <thead>
            <tr>
                <th class="th-sm">ID</th>
                <th class="th-sm">ISBN</th>
                <th class="th-sm">Title</th>
                <th class="th-sm">Author</th>
                <th class="th-sm">Year</th>
            </tr>
        </thead>
        <tbody>
            {% for book in books %}
            <tr>
                <td><a href="{{ url_for('review', book_id=book[0]) }}">
                    {{ book[0] }}</a></td>
                <td>{{ book[1] }}</td>
                <td>{{ book[2] }}</td>
                <td>{{ book[3] }}</td>
                <td>{{ book[4] }}</td>
            </tr>
            {% endfor %}
        </tbody>
    </table>
</div>
{{ pagination.links }}
{% else %}
<p>No books found.</p>
{% endif %}

{% endblock %}

图 | 所有书籍分页显示

后端

后端则要做很多事情,要读取控件的值,然后执行查询,把books返回到网页。‘GET’方法返回所有记录,而’POST’方法就是页面查询,要筛出符合条件的记录并返回。最后都是将结果返回前端渲染。

我不太会写查询条件的复合拼接,用了列表解析式(list comprehension)。这是python里我最喜欢的语法。

举个栗子

假如isbn=‘123’,title=‘war’,author=‘rider’,那么["%s LIKE '%%%s%%'" % (x, y) for x, y in (('isbn', isbn), ('title', title), ('author', author)) if y is not None and y != '']的结果就是这么一个列表:

1
["isbn LIKE '%123%'", "title LIKE '%war%'", "author LIKE '%rider%'"]

一下就把三个查询条件都生成好了。Very pythonic.

定义了一个subset_rec()函数,对列表进行切片,用来对查询结果分页。最后几行是人工生成分页对象pagination。最后把pagination和分页筛出的记录返回给前端。随着用户点击翻页,pagination和page_books都会跟着更新。

 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
# application.py
def subset_rec(rec, offset=0, per_page=20):
    return rec[offset: offset + per_page]
    
@app.route('/index', methods=['GET', 'POST'])
def index():
    if request.method == 'GET':
        books = sess.execute(
            'SELECT id, isbn, title, author, year FROM book;').fetchall()
    elif request.method == 'POST':
        # run the query
        isbn = request.form.get('isbn')
        title = request.form.get('title')
        author = request.form.get('author')
        qry = ["%s LIKE '%%%s%%'" % (x, y) for x, y in
            (('isbn', isbn), ('title', title), ('author', author))
            if y is not None and y != '']
        if len(qry) == 0:
            books = sess.execute(
                'SELECT id, isbn, title, author, year FROM book;').fetchall()
        else:
            books = sess.execute(
                'SELECT id, isbn, title, author, year FROM book WHERE '
                + ' AND '.join(qry) + ';').fetchall()

    page, _, offset = get_page_args(page_parameter='page',
        per_page_parameter='per_page')
    page_books = subset_rec(books, offset=offset, per_page=20)
    pagination = Pagination(page=page, total=len(books), bs_version=3, per_page=20)

    return render_template(
        'index.html', act_user=session.get('act_user'), books=page_books,
        pagination=pagination)

书名搜索"china",返回所有标题内含"china"的书。

图 | 搜索标题含china的书籍

书籍明细

图 | 书籍明细

生成的书籍列表,可以点id访问明细。这里包含三部分:

  1. books.csv自带的信息 (book对象)
  2. 用ISBN到Goodreads上查询到的信息 (gr_data对象)
  3. 用户发布的评级、评论 (review对象)

由于作业要求一个用户只能对一本书作评价,所以还有一个用来判断当前用户对此书评论数量的my_review对象。在前端模板里写一个条件,一旦my_review > 0,就禁用提交按钮。

book.html模板:

 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
<!-- templates/book.html -->
{% extends "base.html" %}

{% block title %}
{{ book[1] }}
{% endblock %}


{% block disp %}
<table id="book_info" class="table table-responsive" width="80%">
    <caption><h4>Info of book <strong>{{ book[1] }}</strong></h4></caption>
    <tr>
        <td class="info" style="width:20%">Title</td>
        <td style="width:30%">{{ book[1] }}</td>   
        <td class="info" style="width:20%">Author</td>
        <td style="width:30%">{{ book[2] }}</td>
    </tr>
    <tr>
        <td class="info" style="width:20%">ISBN</td>
        <td style="width:30%">{{ book[3] }}</td>
        <td class="info" style="width:20%">Year</td>
        <td style="width:30%">{{ book[4] }}</td>
    </tr>
    <tr>
        <td class="info" style="width:20%">Number of ratings</td>
        <td style="width:30%">{{ gr_data['work_ratings_count'] }}</td>
        <td class="info" style="width:20%">Average ratings</td>
        <td style="width:30%">{{ gr_data['average_rating'] }}</td>
    </tr>
</table>
<hr>
<table id="review" class="table table-striped table-hover table-responsive" 
 cellspacing="0" width="80%">
 <caption>Reviews of the book <strong>{{ book[1] }}</strong></caption>
<thead>
    <tr>
        <th class="th-sm">ID</th>
        <th class="th-sm">Time</th>
        <th class="th-sm">Reviewer</th>
        <th class="th-sm">Rating</th>
        <th class="th-sm">Review</th>
    </tr>
</thead>
<tbody>
    {% for review in reviews %}
    <tr class='d-flex'>
        <td style="width:10%">{{ review[0] }}</td>
        <td style="width:20%">{{ review[1] }}</td>
        <td style="width:10%">{{ review[2] }}</td>
        <td style="width:10%">{{ review[3] }}</td>
        <td style="width:50%">{{ review[4] }}</td>
    </tr>
    {% endfor %}
</tbody>
</table>
{% endblock %}

{% block misc %}
<form class="form-group" action="{{ url_for('review', book_id=book[0]) }}" method="post">
    <h5>Submit your review comments.</h5>
    <label for="comment" class="sr-only">Input your review comments.</label>
    <textarea name="comment" class="form-control" rows="4" 
        placeholder="Input your review comments for {{ book[1] }}. You can at most submit one comment for a book">
    </textarea>
    <label for="submit" class="sr-only">Submit</label>
    
    <label for="rating">Rating</label>
    <input id="rating" name="rating" class="rating"  min="0" max="5" step="1" 
        data-size="sm" value='0'>
    
    <button id="submit" class="btn btn-lg btn-primary" {% if my_reviews > 0 %} 
     disabled {% endif %}>
      Submit
    </button>
</form>
{% endblock %}

可以看到,reviews是一个嵌套列表,每个review列表都是长度为5的列表。这个review的结构定义在后端代码里(review_qry)。

后端做了很多工作。首先,定义一个get_gr()函数,用来读取Goodreads API。我用了Lantern,所以调用了Lantern的SOCKS代理翻墙访问。它能返回json串或者None。HTTP_PROXY和HTTPS_PROXY都要事先注册进PATH。key和secret原则上不能写进源代码里,我这里偷懒了。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
# application.py
def get_gr(isbn, api='review_counts', success_code=200):
    """Goodreads API data
    return json or None
    """
    urls = {'review_counts': 'https://www.goodreads.com/book/review_counts.json'}
    url = urls[api]
    res = requests.get(
            url, params={'key': 'yFLoMH1lgWCNlYDs1kWA', 'isbns': isbn},
            timeout=10)
    try:
        res = requests.get(
            url, params={'key': 'yFLoMH1lgWCNlYDs1kWA', 'isbns': isbn},
            timeout=10)
    except requests.exceptions.ProxyError:
        res = requests.get(
            url, params={'key': 'yFLoMH1lgWCNlYDs1kWA', 'isbns': isbn},
            proxies={'http': os.getenv('http_proxy'),
                     'https': os.getenv('https_proxy')},
            timeout=10)
    if res.status_code == success_code:
        return res.json()
    else:
        return None

然后,把book/int:book_id路由的处理都写进review()函数里。

  1. 路由要写成"/book/<int:book_id>“这种动态形式,前端收到的book_id对象可以自动匹配过来。
  2. 分别拿到book,gr_data和review三处数据,丢到index()函数处理。
 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
# application.py
@app.route('/book/<int:book_id>', methods=['GET', 'POST'])
def review(book_id):
    """Book detail
    """
    book = sess.execute(
        """SELECT id, title, author, isbn, year FROM book WHERE
        id = :book_id""", {"book_id": book_id}).fetchone()
    gr_data = get_gr(isbn=book[3])
    if gr_data is not None:
        gr_data = gr_data['books'][0]
    else:
        gr_data = {'work_ratings_count': '', 'average_rating': ''}
    my_reviews = sess.execute(
        """SELECT count(*) FROM review WHERE mbr_id = :mbr_id
        and book_id = :book_id;""",
        {'mbr_id': session.get('act_user')['id'], 
         'book_id': book_id}).fetchone()[0]
    if engine.name == 'sqlite':
        reviews_qry = """SELECT review.id, 
            datetime(review.rev_at, 'localtime'),
            mbr.username, review.rating, review.review FROM
            review LEFT JOIN mbr ON review.mbr_id = mbr.id
            WHERE review.book_id = :book_id"""
    elif engine.name == 'postgresql':
        reviews_qry = """SELECT review.id, 
            review.rev_at at time zone 'utc' at time zone 'cst',
            mbr.username, review.rating, review.review FROM
            review LEFT JOIN mbr ON review.mbr_id = mbr.id
            WHERE review.book_id = :book_id"""
    reviews = sess.execute(
        reviews_qry, {"book_id": book_id}).fetchall()
    # methods
    if request.method == 'GET':
        return render_template(
            'book.html', act_user=session.get('act_user'), book=book
            , reviews=reviews, gr_data=gr_data, my_reviews=my_reviews)
    elif request.method == 'POST':
        comment = request.form.get('comment')
        rating = request.form.get('rating')
        if (comment is not None and comment != '') or \
            (rating is not None and rating != ''):
            sess.execute(
                """INSERT INTO review (mbr_id, book_id, rating, review)
                VALUES (:mbr_id, :book_id, :rating, :review);""",
                {'mbr_id': session.get('act_user')['id'],
                 'book_id': book_id, 'rating': rating, 'review': comment})
            sess.commit()
            flash(Markup(
                """<i class='fa fa-2x fa-check-square-o'></i>
                You have successfully submitted the comment."""), 'success')
        else:
            flash(Markup(
                """<i class='fa fa-2x fa-warning'></i>
                You cannot submit empty rating and comments."""), 'warning')
        return redirect(url_for("review", book_id=book_id))

评分评级

如果是提交评分评级,就调用前述review()函数的’POST’方法。这里用到了一个Flask评星插件bootstrap-star-rating。我把它的css和js都下载到了static。

这个插件本质上通过交互,获得一个评分数值。提交表单时,这个数值rating也被传到后端。

图 | 给图书打分

图 | 打完分后Submit按钮失效

API

自己定义一个API方法。当然,验证key和secret这种专业操作我就不弄了。

在这里,我用request.args.get()方法取id,调用起来就变成<url_head>/api/book?id=xx的形式,而不再是默认的<url_head>/api/book/xx

 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
# application.py
@app.route('/api/book', methods=['GET'])
def api():
    if session.get('act_user') is None:
        return render_template(
            "error.html", act_user=None, err="404 Error",
            err_info="You have not logged in!")
    book_isbn = request.args.get('isbn')
    book = sess.execute(
        """SELECT id, title, author, isbn, year FROM book WHERE
        isbn = :book_isbn""", {"book_isbn": book_isbn}).fetchone()
    if book is None or len(book) == 0:
        return render_template(
            "error.html", act_user=session.get('act_user'),
            err="404 Error", err_info="Book with ISBN = %s not found!" % book_isbn)
    else:
        gr_data = get_gr(isbn=book_isbn)
        if gr_data is not None:
            gr_data = gr_data['books'][0]
        else:
            gr_data = {'work_ratings_count': '', 'average_rating': ''}
        rslt = {'result': True, 'book': 
            {'title': book[1], 'author': book[2], 'year': book[4]
            , 'isbn': book[3], 'review_count': gr_data['work_ratings_count']
            , 'average_score': gr_data['average_rating']}}
        return jsonify(rslt)

图 | API请求成功

图 | API请求失败

响应式布局

参考bootstrap案例写了几个可有可无的@media选择器,概念上有响应式布局的意思了。效果如下。

 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
/* static/css/style.css */
.form-signin .form-control {
    margin: 10px auto 10px auto;
}

.form-signin .btn {
    margin: 10px auto 10px auto;
}

body {
    min-height: 100%;
    margin: 0;
    padding: 0;
    position: relative;
}

main {
    padding-bottom: 100px;
}

#nav_footer {
    position: absolute;
    bottom: 0;
    height: 100px;
}


@media (min-width: 1200px){
    .form-signin {
        width: 20%;
        margin: 10px 40% 10px 40%;
        padding: 10px;
    }
}

@media (min-width: 992px){
    .form-signin {
        width: 30%;
        margin: 10px 35% 10px 35%;
        padding: 5px;
    }
}

@media (min-width: 768px){
    .form-signin {
        margin: 10px auto 10px auto;
        padding: 5px;
    }
}

其他

flash自动消失和star-rating需要专门适配一些javascript,在main.js里。主要是一些jQuery。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
/* static/js/main.js */
$(document).ready(function () {
    /* alert-dismissable dismiss automatically in 4s */
    window.setTimeout(function() {
        $(".alert-dismissable").fadeTo(1000, 0).slideUp(1000, function(){
            $(this).remove(); 
        });
    }, 4000);  
});

$(document).ready(function(){
    $("#rating").rating();           
});
 

大致就是这样。

[完]


扫码关注

扫码关注我的公众号