forked from mrlan/EnglishPal
				
			fix bug547
						commit
						6c6734f1db
					
				| 
						 | 
					@ -2,12 +2,20 @@
 | 
				
			||||||
venv/
 | 
					venv/
 | 
				
			||||||
app/__init__.py
 | 
					app/__init__.py
 | 
				
			||||||
app/__pycache__/
 | 
					app/__pycache__/
 | 
				
			||||||
 | 
					.DS_Store
 | 
				
			||||||
 | 
					app/.DS_Store
 | 
				
			||||||
app/sqlite_commands.py
 | 
					app/sqlite_commands.py
 | 
				
			||||||
app/static/usr/*.jpg
 | 
					app/static/usr/*.jpg
 | 
				
			||||||
app/static/img/
 | 
					app/static/img/
 | 
				
			||||||
app/static/frequency/frequency_*.pickle
 | 
					app/static/frequency/frequency_*.pickle
 | 
				
			||||||
app/static/frequency/frequency.p
 | 
					app/static/frequency/frequency.p
 | 
				
			||||||
app/static/wordfreqapp.db
 | 
					app/wordfreqapp.db
 | 
				
			||||||
 | 
					app/db/wordfreqapp.db
 | 
				
			||||||
app/static/donate-the-author.jpg
 | 
					app/static/donate-the-author.jpg
 | 
				
			||||||
app/static/donate-the-author-hidden.jpg
 | 
					app/static/donate-the-author-hidden.jpg
 | 
				
			||||||
app/model/__pycache__/
 | 
					app/model/__pycache__/
 | 
				
			||||||
 | 
					app/test/__pycache__/
 | 
				
			||||||
 | 
					app/test/.pytest_cache/
 | 
				
			||||||
 | 
					app/test/pytest_report.html
 | 
				
			||||||
 | 
					app/test/assets
 | 
				
			||||||
 | 
					app/log.txt
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,4 +1,5 @@
 | 
				
			||||||
FROM tiangolo/uwsgi-nginx-flask:python3.6
 | 
					FROM tiangolo/uwsgi-nginx-flask:python3.8-alpine
 | 
				
			||||||
COPY requirements.txt /app
 | 
					COPY requirements.txt /tmp
 | 
				
			||||||
RUN pip3 install -r requirements.txt -i https://mirrors.aliyun.com/pypi/simple/
 | 
					COPY ./app/ /app/
 | 
				
			||||||
COPY ./app /app
 | 
					RUN pip3 install -U pip -i https://mirrors.aliyun.com/pypi/simple/
 | 
				
			||||||
 | 
					RUN pip3 install -r /tmp/requirements.txt -i https://mirrors.aliyun.com/pypi/simple/
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -10,8 +10,8 @@ pipeline {
 | 
				
			||||||
    stages {
 | 
					    stages {
 | 
				
			||||||
        stage('MakeDatabasefile') {
 | 
					        stage('MakeDatabasefile') {
 | 
				
			||||||
	    steps {
 | 
						    steps {
 | 
				
			||||||
	        sh 'touch ./app/static/wordfreqapp.db && rm -f ./app/static/wordfreqapp.db' 
 | 
						        sh 'touch ./app/wordfreqapp.db && rm -f ./app/wordfreqapp.db' 
 | 
				
			||||||
	        sh 'cat ./app/static/wordfreqapp.sql | sqlite3 ./app/static/wordfreqapp.db'
 | 
						        sh 'cat ./app/static/wordfreqapp.sql | sqlite3 ./app/wordfreqapp.db'
 | 
				
			||||||
	    }
 | 
						    }
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
        stage('BuildIt') {
 | 
					        stage('BuildIt') {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										30
									
								
								README.md
								
								
								
								
							
							
						
						
									
										30
									
								
								README.md
								
								
								
								
							| 
						 | 
					@ -61,15 +61,15 @@ My steps for deploying English on a Ubuntu server.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
All articles are stored in the `article` table in a SQLite file called
 | 
					All articles are stored in the `article` table in a SQLite file called
 | 
				
			||||||
`app/static/wordfreqapp.db`.
 | 
					`app/db/wordfreqapp.db`.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Adding new articles
 | 
					### Adding new articles
 | 
				
			||||||
 | 
					
 | 
				
			||||||
To add articles, open and edit `app/static/wordfreqapp.db` using DB Browser for SQLite (https://sqlitebrowser.org).
 | 
					To add articles, open and edit `app/db/wordfreqapp.db` using DB Browser for SQLite (https://sqlitebrowser.org).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Extending an account's expiry date
 | 
					### Extending an account's expiry date
 | 
				
			||||||
 | 
					
 | 
				
			||||||
By default, an account's expiry is 30 days after first sign-up.  To extend account's expiry date, open and edit `user` table in `app/static/wordfreqapp.db`.  Simply update field `expiry_date`.
 | 
					By default, an account's expiry is 30 days after first sign-up.  To extend account's expiry date, open and edit `user` table in `app/db/wordfreqapp.db`.  Simply update field `expiry_date`.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Exporting the database
 | 
					### Exporting the database
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -95,7 +95,7 @@ sqlite3 wordfreqapp.db`.  Delete wordfreqapp.db first if it exists.
 | 
				
			||||||
### Uploading wordfreqapp.db to the server
 | 
					### Uploading wordfreqapp.db to the server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
`pscp wordfreqapp.db lanhui@118.*.*.118:/home/lanhui/englishpal2/EnglishPal/app/static`
 | 
					`pscp wordfreqapp.db lanhui@118.*.*.118:/home/lanhui/englishpal2/EnglishPal/app/db/`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -129,6 +129,28 @@ We welcome feedback on EnglishPal.  Feedback examples:
 | 
				
			||||||
EnglishPal's bugs and improvement suggestions are recorded in [Bugzilla](http://118.25.96.118/bugzilla/buglist.cgi?bug_status=__all__&list_id=1302&order=Importance&product=EnglishPal&query_format=specific).  Send (lanhui at zjnu.edu.cn) an email message for opening a Bugzilla account or reporting a bug.
 | 
					EnglishPal's bugs and improvement suggestions are recorded in [Bugzilla](http://118.25.96.118/bugzilla/buglist.cgi?bug_status=__all__&list_id=1302&order=Importance&product=EnglishPal&query_format=specific).  Send (lanhui at zjnu.edu.cn) an email message for opening a Bugzilla account or reporting a bug.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## End-to-end testing
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					We use the Selenium test framework to test our app.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					In order to run the test, first we need to download a webdriver executable.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Microsoft Edge's webdriver can be downloaded from [microsoft-edge-tools-webdriver](https://developer.microsoft.com/en-us/microsoft-edge/tools/webdriver/). Make sure the version we download matches the version of the web browser installed on our laptop.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					After extracting the downloaded zip file (e.g., edgedriver_win64.zip), rename msedgedriver.exe to MicrosoftWebDriver.exe.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Add MicrosoftWebDriver.exe's path to system's PATH variable.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Install the following dependencies too:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- pip install -U selenium==3.141.0
 | 
				
			||||||
 | 
					- pip install -U urllib3==1.26.2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Run English Pal first, then run the test using pytest as follows: pytest --html=pytest_report.html test_add_word.py
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The above command will generate a HTML report file pytest_report.html after finishing executing test_add_word.py.  Note: you need to install pytest-html package first: pip install pytest-html.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					You may also want to use [webdriver-manager](https://pypi.org/project/webdriver-manager/) from PyPI, so that you can avoid tediously installing a web driver executable manually.  However, my experience shows that webdriver-manager is too slow.  For example, it took me 16 minutes to run 9 tests, while with the pre-installed web driver executable, it took less than 2 minutes.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## TODO
 | 
					## TODO
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,6 +1,5 @@
 | 
				
			||||||
from WordFreq import WordFreq
 | 
					from WordFreq import WordFreq
 | 
				
			||||||
from wordfreqCMD import youdao_link, sort_in_descending_order
 | 
					from wordfreqCMD import youdao_link, sort_in_descending_order
 | 
				
			||||||
from UseSqlite import InsertQuery, RecordQuery
 | 
					 | 
				
			||||||
import pickle_idea, pickle_idea2
 | 
					import pickle_idea, pickle_idea2
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
import random, glob
 | 
					import random, glob
 | 
				
			||||||
| 
						 | 
					@ -8,18 +7,15 @@ import hashlib
 | 
				
			||||||
from datetime import datetime
 | 
					from datetime import datetime
 | 
				
			||||||
from flask import Flask, request, redirect, render_template, url_for, session, abort, flash, get_flashed_messages
 | 
					from flask import Flask, request, redirect, render_template, url_for, session, abort, flash, get_flashed_messages
 | 
				
			||||||
from difficulty import get_difficulty_level_for_user, text_difficulty_level, user_difficulty_level
 | 
					from difficulty import get_difficulty_level_for_user, text_difficulty_level, user_difficulty_level
 | 
				
			||||||
 | 
					from model.article import get_all_articles, get_article_by_id, get_number_of_articles
 | 
				
			||||||
 | 
					import logging
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					path_prefix = './'
 | 
				
			||||||
path_prefix = '/var/www/wordfreq/wordfreq/'
 | 
					db_path_prefix = './db/'  # comment this line in deployment
 | 
				
			||||||
path_prefix = './'  # comment this line in deployment
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def total_number_of_essays():
 | 
					def total_number_of_essays():
 | 
				
			||||||
    rq = RecordQuery(path_prefix + 'static/wordfreqapp.db')
 | 
					    return get_number_of_articles()
 | 
				
			||||||
    rq.instructions("SELECT * FROM article")
 | 
					 | 
				
			||||||
    rq.do()
 | 
					 | 
				
			||||||
    result = rq.get_results()
 | 
					 | 
				
			||||||
    return len(result)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def get_article_title(s):
 | 
					def get_article_title(s):
 | 
				
			||||||
| 
						 | 
					@ -33,32 +29,36 @@ def get_article_body(s):
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def get_today_article(user_word_list, visited_articles):
 | 
					def get_today_article(user_word_list, visited_articles):
 | 
				
			||||||
    rq = RecordQuery(path_prefix + 'static/wordfreqapp.db')
 | 
					 | 
				
			||||||
    if visited_articles is None:
 | 
					    if visited_articles is None:
 | 
				
			||||||
        visited_articles = {
 | 
					        visited_articles = {
 | 
				
			||||||
            "index" : 0,  # 为 article_ids 的索引
 | 
					            "index" : 0,  # 为 article_ids 的索引
 | 
				
			||||||
            "article_ids": []  # 之前显示文章的id列表,越后越新
 | 
					            "article_ids": []  # 之前显示文章的id列表,越后越新
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    if visited_articles["index"] > len(visited_articles["article_ids"])-1:  # 生成新的文章,因此查找所有的文章
 | 
					    if visited_articles["index"] > len(visited_articles["article_ids"])-1:  # 生成新的文章,因此查找所有的文章
 | 
				
			||||||
        rq.instructions("SELECT * FROM article")
 | 
					        result = get_all_articles()
 | 
				
			||||||
    else:  # 生成阅读过的文章,因此查询指定 article_id 的文章
 | 
					    else:  # 生成阅读过的文章,因此查询指定 article_id 的文章
 | 
				
			||||||
        if visited_articles["article_ids"][visited_articles["index"]] == 'null':  # 可能因为直接刷新页面导致直接去查询了'null',因此当刷新的页面的时候,需要直接进行“上一篇”操作
 | 
					        if visited_articles["article_ids"][visited_articles["index"]] == 'null':  # 可能因为直接刷新页面导致直接去查询了'null',因此当刷新的页面的时候,需要直接进行“上一篇”操作
 | 
				
			||||||
            visited_articles["index"] -= 1
 | 
					            visited_articles["index"] -= 1
 | 
				
			||||||
            visited_articles["article_ids"].pop()
 | 
					            visited_articles["article_ids"].pop()
 | 
				
			||||||
        rq.instructions('SELECT * FROM article WHERE article_id=%d' % (visited_articles["article_ids"][visited_articles["index"]]))
 | 
					        article_id = visited_articles["article_ids"][visited_articles["index"]]
 | 
				
			||||||
    rq.do()
 | 
					        result = get_article_by_id(article_id)
 | 
				
			||||||
    result = rq.get_results()
 | 
					 | 
				
			||||||
    random.shuffle(result)
 | 
					    random.shuffle(result)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # Choose article according to reader's level
 | 
					    # Choose article according to reader's level
 | 
				
			||||||
    d1 = load_freq_history(path_prefix + 'static/frequency/frequency.p')
 | 
					    logging.debug('* get_today_article(): start d1 = ... ')
 | 
				
			||||||
 | 
					    d1 = load_freq_history(user_word_list)
 | 
				
			||||||
    d2 = load_freq_history(path_prefix + 'static/words_and_tests.p')
 | 
					    d2 = load_freq_history(path_prefix + 'static/words_and_tests.p')
 | 
				
			||||||
 | 
					    logging.debug(' ... get_today_article(): get_difficulty_level_for_user() start')
 | 
				
			||||||
    d3 = get_difficulty_level_for_user(d1, d2)
 | 
					    d3 = get_difficulty_level_for_user(d1, d2)
 | 
				
			||||||
 | 
					    logging.debug(' ... get_today_article(): done')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    d = None
 | 
					    d = None
 | 
				
			||||||
    result_of_generate_article = "not found"
 | 
					    result_of_generate_article = "not found"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    d_user = load_freq_history(user_word_list)
 | 
					    d_user = load_freq_history(user_word_list)
 | 
				
			||||||
 | 
					    logging.debug('* get_today_article(): user_difficulty_level() start')
 | 
				
			||||||
    user_level = user_difficulty_level(d_user, d3)  # more consideration as user's behaviour is dynamic. Time factor should be considered.
 | 
					    user_level = user_difficulty_level(d_user, d3)  # more consideration as user's behaviour is dynamic. Time factor should be considered.
 | 
				
			||||||
 | 
					    logging.debug('* get_today_article(): done')
 | 
				
			||||||
    text_level = 0
 | 
					    text_level = 0
 | 
				
			||||||
    if visited_articles["index"] > len(visited_articles["article_ids"])-1:  # 生成新的文章
 | 
					    if visited_articles["index"] > len(visited_articles["article_ids"])-1:  # 生成新的文章
 | 
				
			||||||
        amount_of_visited_articles = len(visited_articles["article_ids"])
 | 
					        amount_of_visited_articles = len(visited_articles["article_ids"])
 | 
				
			||||||
| 
						 | 
					@ -87,8 +87,8 @@ def get_today_article(user_word_list, visited_articles):
 | 
				
			||||||
    today_article = None
 | 
					    today_article = None
 | 
				
			||||||
    if d:
 | 
					    if d:
 | 
				
			||||||
        today_article = {
 | 
					        today_article = {
 | 
				
			||||||
            "user_level": '%4.2f' % user_level,
 | 
					            "user_level": '%4.1f' % user_level,
 | 
				
			||||||
            "text_level": '%4.2f' % text_level,
 | 
					            "text_level": '%4.1f' % text_level,
 | 
				
			||||||
            "date": d['date'],
 | 
					            "date": d['date'],
 | 
				
			||||||
            "article_title": get_article_title(d['text']),
 | 
					            "article_title": get_article_title(d['text']),
 | 
				
			||||||
            "article_body": get_article_body(d['text']),
 | 
					            "article_body": get_article_body(d['text']),
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,7 +1,6 @@
 | 
				
			||||||
import hashlib
 | 
					import hashlib
 | 
				
			||||||
import string
 | 
					import string
 | 
				
			||||||
from datetime import datetime, timedelta
 | 
					from datetime import datetime, timedelta
 | 
				
			||||||
from UseSqlite import InsertQuery, RecordQuery
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
def md5(s):
 | 
					def md5(s):
 | 
				
			||||||
    '''
 | 
					    '''
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,87 +0,0 @@
 | 
				
			||||||
###########################################################################
 | 
					 | 
				
			||||||
# Copyright 2019 (C) Hui Lan <hui.lan@cantab.net>
 | 
					 | 
				
			||||||
# Written permission must be obtained from the author for commercial uses.
 | 
					 | 
				
			||||||
###########################################################################
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Reference: Dusty Phillips.  Python 3 Objected-oriented Programming Second Edition. Pages 326-328.
 | 
					 | 
				
			||||||
# Copyright (C) 2019 Hui Lan
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import sqlite3
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Sqlite3Template:
 | 
					 | 
				
			||||||
    def __init__(self, db_fname):
 | 
					 | 
				
			||||||
        self.db_fname = db_fname
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def connect(self, db_fname):
 | 
					 | 
				
			||||||
        self.conn = sqlite3.connect(self.db_fname)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def instructions(self, query_statement):
 | 
					 | 
				
			||||||
        raise NotImplementedError()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def operate(self):
 | 
					 | 
				
			||||||
        self.conn.row_factory = sqlite3.Row
 | 
					 | 
				
			||||||
        self.results = self.conn.execute(self.query) # self.query is to be given in the child classes
 | 
					 | 
				
			||||||
        self.conn.commit()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def format_results(self):
 | 
					 | 
				
			||||||
        raise NotImplementedError()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def do(self):
 | 
					 | 
				
			||||||
        self.connect(self.db_fname)
 | 
					 | 
				
			||||||
        self.instructions(self.query)
 | 
					 | 
				
			||||||
        self.operate()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def instructions_with_parameters(self, query_statement, parameters):
 | 
					 | 
				
			||||||
        self.query = query_statement
 | 
					 | 
				
			||||||
        self.parameters = parameters
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def do_with_parameters(self):
 | 
					 | 
				
			||||||
        self.connect(self.db_fname)
 | 
					 | 
				
			||||||
        self.instructions_with_parameters(self.query, self.parameters)
 | 
					 | 
				
			||||||
        self.operate_with_parameters()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def operate_with_parameters(self):
 | 
					 | 
				
			||||||
        self.conn.row_factory = sqlite3.Row
 | 
					 | 
				
			||||||
        self.results = self.conn.execute(self.query, self.parameters) # self.query is to be given in the child classes
 | 
					 | 
				
			||||||
        self.conn.commit()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class InsertQuery(Sqlite3Template):
 | 
					 | 
				
			||||||
    def instructions(self, query):
 | 
					 | 
				
			||||||
        self.query = query
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class RecordQuery(Sqlite3Template):
 | 
					 | 
				
			||||||
    def instructions(self, query):
 | 
					 | 
				
			||||||
        self.query = query
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def format_results(self):
 | 
					 | 
				
			||||||
        output = []
 | 
					 | 
				
			||||||
        for row_dict in self.results.fetchall():
 | 
					 | 
				
			||||||
            lst = []
 | 
					 | 
				
			||||||
            for k in dict(row_dict):
 | 
					 | 
				
			||||||
                lst.append( row_dict[k] )
 | 
					 | 
				
			||||||
            output.append(', '.join(lst))
 | 
					 | 
				
			||||||
        return '\n\n'.join(output)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def get_results(self):
 | 
					 | 
				
			||||||
        result = []
 | 
					 | 
				
			||||||
        for row_dict in self.results.fetchall():
 | 
					 | 
				
			||||||
            result.append( dict(row_dict) )
 | 
					 | 
				
			||||||
        return result
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
if __name__ == '__main__':
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    #iq = InsertQuery('RiskDB.db')
 | 
					 | 
				
			||||||
    #iq.instructions("INSERT INTO inspection Values ('FoodSupplies', 'RI2019051301', '2019-05-13', '{}')")
 | 
					 | 
				
			||||||
    #iq.do()
 | 
					 | 
				
			||||||
    #iq.instructions("INSERT INTO inspection Values ('CarSupplies', 'RI2019051302', '2019-05-13', '{[{\"risk_name\":\"elevator\"}]}')")
 | 
					 | 
				
			||||||
    #iq.do()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    rq = RecordQuery('wordfreqapp.db')
 | 
					 | 
				
			||||||
    rq.instructions("SELECT * FROM article WHERE level=3")
 | 
					 | 
				
			||||||
    rq.do()
 | 
					 | 
				
			||||||
    #print(rq.format_results())
 | 
					 | 
				
			||||||
| 
						 | 
					@ -1,4 +1,5 @@
 | 
				
			||||||
from flask import *
 | 
					from flask import *
 | 
				
			||||||
 | 
					from markupsafe import escape
 | 
				
			||||||
from Login import check_username_availability, verify_user, add_user, get_expiry_date, change_password, WarningMessage
 | 
					from Login import check_username_availability, verify_user, add_user, get_expiry_date, change_password, WarningMessage
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,5 +1,6 @@
 | 
				
			||||||
# System Library
 | 
					# System Library
 | 
				
			||||||
from flask import *
 | 
					from flask import *
 | 
				
			||||||
 | 
					from markupsafe import escape
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Personal library
 | 
					# Personal library
 | 
				
			||||||
from Yaml import yml
 | 
					from Yaml import yml
 | 
				
			||||||
| 
						 | 
					@ -37,6 +38,22 @@ def admin():
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@adminService.route("/admin/article", methods=["GET", "POST"])
 | 
					@adminService.route("/admin/article", methods=["GET", "POST"])
 | 
				
			||||||
def article():
 | 
					def article():
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def _make_title_and_content(article_lst):
 | 
				
			||||||
 | 
					        for article in article_lst:
 | 
				
			||||||
 | 
					            text = escape(article.text) # Fix XSS vulnerability, contributed by Xu Xuan
 | 
				
			||||||
 | 
					            article.title = text.split("\n")[0]
 | 
				
			||||||
 | 
					            article.content = '<br/>'.join(text.split("\n")[1:])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def _update_context():
 | 
				
			||||||
 | 
					        article_len = get_number_of_articles()
 | 
				
			||||||
 | 
					        context["article_number"] = article_len
 | 
				
			||||||
 | 
					        context["text_list"] = get_page_articles(_cur_page, _page_size)
 | 
				
			||||||
 | 
					        _articles = get_page_articles(_cur_page, _page_size)
 | 
				
			||||||
 | 
					        _make_title_and_content(_articles)
 | 
				
			||||||
 | 
					        context["text_list"] = _articles
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    global _cur_page, _page_size
 | 
					    global _cur_page, _page_size
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    is_admin = check_is_admin()
 | 
					    is_admin = check_is_admin()
 | 
				
			||||||
| 
						 | 
					@ -44,20 +61,15 @@ def article():
 | 
				
			||||||
        return is_admin
 | 
					        return is_admin
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    _article_number = get_number_of_articles()
 | 
					    _article_number = get_number_of_articles()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        _page_size = min(
 | 
					        _page_size = min(max(1, int(request.args.get("size", 5))), _article_number)  # 最小的size是1
 | 
				
			||||||
            max(1, int(request.args.get("size", 5))), _article_number
 | 
					        _cur_page = min(max(1, int(request.args.get("page", 1))), _article_number // _page_size + (_article_number % _page_size > 0))  # 最小的page是1
 | 
				
			||||||
        )  # 最小的size是1
 | 
					 | 
				
			||||||
        _cur_page = min(
 | 
					 | 
				
			||||||
            max(1, int(request.args.get("page", 1))), _article_number // _page_size + (_article_number % _page_size > 0)
 | 
					 | 
				
			||||||
        )  # 最小的page是1
 | 
					 | 
				
			||||||
    except ValueError:
 | 
					    except ValueError:
 | 
				
			||||||
        return "page parmas must be int!"
 | 
					        return "page parameters must be integer!"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    _articles = get_page_articles(_cur_page, _page_size)
 | 
					    _articles = get_page_articles(_cur_page, _page_size)
 | 
				
			||||||
    for article in _articles:   # 获取每篇文章的title
 | 
					    _make_title_and_content(_articles)
 | 
				
			||||||
        article.title = article.text.split("\n")[0]
 | 
					 | 
				
			||||||
        article.content = '<br/>'.join(article.text.split("\n")[1:])
 | 
					 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
    context = {
 | 
					    context = {
 | 
				
			||||||
        "article_number": _article_number,
 | 
					        "article_number": _article_number,
 | 
				
			||||||
| 
						 | 
					@ -67,23 +79,16 @@ def article():
 | 
				
			||||||
        "username": session.get("username"),
 | 
					        "username": session.get("username"),
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def _update_context():
 | 
					 | 
				
			||||||
        article_len = get_number_of_articles()
 | 
					 | 
				
			||||||
        context["article_number"] = article_len
 | 
					 | 
				
			||||||
        context["text_list"] = get_page_articles(_cur_page, _page_size)
 | 
					 | 
				
			||||||
        _articles = get_page_articles(_cur_page, _page_size)
 | 
					 | 
				
			||||||
        for article in _articles:   # 获取每篇文章的title
 | 
					 | 
				
			||||||
            article.title = article.text.split("\n")[0]
 | 
					 | 
				
			||||||
        context["text_list"] = _articles
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if request.method == "GET":
 | 
					    if request.method == "GET":
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            delete_id = int(request.args.get("delete_id", 0))
 | 
					            delete_id = int(request.args.get("delete_id", 0))
 | 
				
			||||||
        except:
 | 
					        except:
 | 
				
			||||||
            return "Delete article ID must be int!"
 | 
					            return "Delete article ID must be integer!"
 | 
				
			||||||
        if delete_id:  # delete article
 | 
					        if delete_id:  # delete article
 | 
				
			||||||
            delete_article_by_id(delete_id)
 | 
					            delete_article_by_id(delete_id)
 | 
				
			||||||
            _update_context()
 | 
					            _update_context()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    elif request.method == "POST":
 | 
					    elif request.method == "POST":
 | 
				
			||||||
        data = request.form
 | 
					        data = request.form
 | 
				
			||||||
        content = data.get("content", "")
 | 
					        content = data.get("content", "")
 | 
				
			||||||
| 
						 | 
					@ -97,6 +102,7 @@ def article():
 | 
				
			||||||
            _update_context()
 | 
					            _update_context()
 | 
				
			||||||
            title = content.split('\n')[0]
 | 
					            title = content.split('\n')[0]
 | 
				
			||||||
            flash(f'Article added. Title: {title}')
 | 
					            flash(f'Article added. Title: {title}')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return render_template("admin_manage_article.html", **context)
 | 
					    return render_template("admin_manage_article.html", **context)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1 @@
 | 
				
			||||||
 | 
					Put wordfreqapp.db here
 | 
				
			||||||
| 
						 | 
					@ -18,6 +18,7 @@ def load_record(pickle_fname):
 | 
				
			||||||
    return d
 | 
					    return d
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ENGLISH_WORD_DIFFICULTY_DICT = {}
 | 
				
			||||||
def convert_test_type_to_difficulty_level(d):
 | 
					def convert_test_type_to_difficulty_level(d):
 | 
				
			||||||
    """
 | 
					    """
 | 
				
			||||||
    对原本的单词库中的单词进行难度评级
 | 
					    对原本的单词库中的单词进行难度评级
 | 
				
			||||||
| 
						 | 
					@ -39,8 +40,10 @@ def convert_test_type_to_difficulty_level(d):
 | 
				
			||||||
        elif 'BBC' in d[k]:
 | 
					        elif 'BBC' in d[k]:
 | 
				
			||||||
            result[k] = 8
 | 
					            result[k] = 8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return result  # {'apple': 4, ...}
 | 
					    global ENGLISH_WORD_DIFFICULTY_DICT
 | 
				
			||||||
 | 
					    ENGLISH_WORD_DIFFICULTY_DICT = result
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return result  # {'apple': 4, ...}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def get_difficulty_level_for_user(d1, d2):
 | 
					def get_difficulty_level_for_user(d1, d2):
 | 
				
			||||||
    """
 | 
					    """
 | 
				
			||||||
| 
						 | 
					@ -49,7 +52,11 @@ def get_difficulty_level_for_user(d1, d2):
 | 
				
			||||||
    在d2的后面添加单词,没有新建一个新的字典
 | 
					    在d2的后面添加单词,没有新建一个新的字典
 | 
				
			||||||
    """
 | 
					    """
 | 
				
			||||||
    # TODO: convert_test_type_to_difficulty_level() should not be called every time.  Each word's difficulty level should be pre-computed.
 | 
					    # TODO: convert_test_type_to_difficulty_level() should not be called every time.  Each word's difficulty level should be pre-computed.
 | 
				
			||||||
 | 
					    if ENGLISH_WORD_DIFFICULTY_DICT == {}:
 | 
				
			||||||
        d2 = convert_test_type_to_difficulty_level(d2)  # 根据d2的标记评级{'apple': 4, 'abandon': 4, ...}
 | 
					        d2 = convert_test_type_to_difficulty_level(d2)  # 根据d2的标记评级{'apple': 4, 'abandon': 4, ...}
 | 
				
			||||||
 | 
					    else:
 | 
				
			||||||
 | 
					        d2 = ENGLISH_WORD_DIFFICULTY_DICT
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    stemmer = snowballstemmer.stemmer('english')
 | 
					    stemmer = snowballstemmer.stemmer('english')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for k in d1:  # 用户的词
 | 
					    for k in d1:  # 用户的词
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										11
									
								
								app/main.py
								
								
								
								
							
							
						
						
									
										11
									
								
								app/main.py
								
								
								
								
							| 
						 | 
					@ -1,19 +1,19 @@
 | 
				
			||||||
#! /usr/bin/python3
 | 
					 | 
				
			||||||
# -*- coding: utf-8 -*-
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
###########################################################################
 | 
					###########################################################################
 | 
				
			||||||
# Copyright 2019 (C) Hui Lan <hui.lan@cantab.net>
 | 
					# Copyright 2019 (C) Hui Lan <hui.lan@cantab.net>
 | 
				
			||||||
# Written permission must be obtained from the author for commercial uses.
 | 
					# Written permission must be obtained from the author for commercial uses.
 | 
				
			||||||
###########################################################################
 | 
					###########################################################################
 | 
				
			||||||
from flask import escape
 | 
					from flask import abort
 | 
				
			||||||
 | 
					from markupsafe import escape
 | 
				
			||||||
from Login import *
 | 
					from Login import *
 | 
				
			||||||
from Article import *
 | 
					from Article import *
 | 
				
			||||||
import Yaml
 | 
					import Yaml
 | 
				
			||||||
from user_service import userService
 | 
					from user_service import userService
 | 
				
			||||||
from account_service import accountService
 | 
					from account_service import accountService
 | 
				
			||||||
from admin_service import adminService, ADMIN_NAME
 | 
					from admin_service import adminService, ADMIN_NAME
 | 
				
			||||||
 | 
					import os
 | 
				
			||||||
 | 
					
 | 
				
			||||||
app = Flask(__name__)
 | 
					app = Flask(__name__)
 | 
				
			||||||
app.secret_key = 'lunch.time!'
 | 
					app.secret_key = os.urandom(32)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# 将蓝图注册到Lab app
 | 
					# 将蓝图注册到Lab app
 | 
				
			||||||
app.register_blueprint(userService)
 | 
					app.register_blueprint(userService)
 | 
				
			||||||
| 
						 | 
					@ -54,7 +54,6 @@ def appears_in_test(word, d):
 | 
				
			||||||
    else:
 | 
					    else:
 | 
				
			||||||
        return ','.join(d[word])
 | 
					        return ','.join(d[word])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 | 
				
			||||||
@app.route("/mark", methods=['GET', 'POST'])
 | 
					@app.route("/mark", methods=['GET', 'POST'])
 | 
				
			||||||
def mark_word():
 | 
					def mark_word():
 | 
				
			||||||
    '''
 | 
					    '''
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,7 +1,7 @@
 | 
				
			||||||
from pony.orm import *
 | 
					from pony.orm import *
 | 
				
			||||||
 | 
					
 | 
				
			||||||
db = Database()
 | 
					db = Database()
 | 
				
			||||||
db.bind("sqlite", "../static/wordfreqapp.db", create_db=True)  # bind sqlite file
 | 
					db.bind("sqlite", "../db/wordfreqapp.db", create_db=True)  # bind sqlite file
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class User(db.Entity):
 | 
					class User(db.Entity):
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -7,7 +7,7 @@ def add_article(content, source="manual_input", level="5", question="No question
 | 
				
			||||||
        Article(
 | 
					        Article(
 | 
				
			||||||
            text=content,
 | 
					            text=content,
 | 
				
			||||||
            source=source,
 | 
					            source=source,
 | 
				
			||||||
            date=datetime.now().strftime("%-d %b %Y"),  # format style of `5 Oct 2022`
 | 
					            date=datetime.now().strftime("%d %b %Y"),  # format style of `5 Oct 2022`
 | 
				
			||||||
            level=level,
 | 
					            level=level,
 | 
				
			||||||
            question=question,
 | 
					            question=question,
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
| 
						 | 
					@ -32,3 +32,17 @@ def get_page_articles(num, size):
 | 
				
			||||||
            x
 | 
					            x
 | 
				
			||||||
            for x in Article.select().order_by(desc(Article.article_id)).page(num, size)
 | 
					            for x in Article.select().order_by(desc(Article.article_id)).page(num, size)
 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def get_all_articles():
 | 
				
			||||||
 | 
					    articles = []
 | 
				
			||||||
 | 
					    with db_session:
 | 
				
			||||||
 | 
					        for article in Article.select():
 | 
				
			||||||
 | 
					            articles.append(article.to_dict())
 | 
				
			||||||
 | 
					    return articles
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def get_article_by_id(article_id):
 | 
				
			||||||
 | 
					    with db_session:
 | 
				
			||||||
 | 
					        article = Article.get(article_id=article_id)
 | 
				
			||||||
 | 
					    return [article.to_dict()]
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -9,6 +9,7 @@ js:
 | 
				
			||||||
    - ../static/js/jquery.js
 | 
					    - ../static/js/jquery.js
 | 
				
			||||||
    - ../static/js/read.js
 | 
					    - ../static/js/read.js
 | 
				
			||||||
    - ../static/js/word_operation.js
 | 
					    - ../static/js/word_operation.js
 | 
				
			||||||
 | 
					    - ../static/js/checkboxes.js
 | 
				
			||||||
  bottom: # 在页面加载完之后加载
 | 
					  bottom: # 在页面加载完之后加载
 | 
				
			||||||
    - ../static/js/fillword.js
 | 
					    - ../static/js/fillword.js
 | 
				
			||||||
    - ../static/js/highlight.js
 | 
					    - ../static/js/highlight.js
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,5 @@
 | 
				
			||||||
 | 
					function toggleCheckboxSelection(checkStatus) {
 | 
				
			||||||
 | 
					    // used in userpage_post.html
 | 
				
			||||||
 | 
					    const checkBoxes = document.getElementsByName('marked');
 | 
				
			||||||
 | 
					    checkBoxes.forEach((checkbox) => { checkbox.checked = checkStatus;} );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -29,3 +29,8 @@ function onReadClick() {
 | 
				
			||||||
function onChooseClick() {
 | 
					function onChooseClick() {
 | 
				
			||||||
    isChoose = !isChoose;
 | 
					    isChoose = !isChoose;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// 如果网页刷新,停止播放声音
 | 
				
			||||||
 | 
					if (performance.getEntriesByType("navigation")[0].type == "reload") {
 | 
				
			||||||
 | 
					    Reader.stopRead();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -22,62 +22,38 @@ function getWord() {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function highLight() {
 | 
					function highLight() {
 | 
				
			||||||
    if (!isHighlight) return;
 | 
					    if (!isHighlight) return;
 | 
				
			||||||
    let articleContent = document.getElementById("article").innerText; //将原来的.innerText改为.innerHtml,使用innerText会把原文章中所包含的<br>标签去除,导致处理后的文章内容失去了原来的格式
 | 
					    let articleContent = document.getElementById("article").innerHTML; // innerHTML保留HTML标签来保持部分格式,且适配不同的浏览器
 | 
				
			||||||
    let pickedWords = document.getElementById("selected-words");  // words picked to the text area
 | 
					    let pickedWords = document.getElementById("selected-words");  // words picked to the text area
 | 
				
			||||||
    let dictionaryWords = document.getElementById("selected-words2"); // words appearing in the user's new words list
 | 
					    let dictionaryWords = document.getElementById("selected-words2"); // words appearing in the user's new words list
 | 
				
			||||||
    let allWords = "";  //初始化allWords的值,避免进入判断后编译器认为allWords未初始化的问题
 | 
					    let allWords = dictionaryWords === null ? pickedWords.value + " " : pickedWords.value + " " + dictionaryWords.value;
 | 
				
			||||||
    if(dictionaryWords != null){//增加一个判断,检查生词本里面是否为空,如果为空,allWords只添加选中的单词
 | 
					    const list = allWords.split(" "); // 将所有的生词放入一个list中
 | 
				
			||||||
        allWords = pickedWords.value + " " + dictionaryWords.value;
 | 
					    let totalSet = new Set();
 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    else{
 | 
					 | 
				
			||||||
        allWords = pickedWords.value + " ";
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    const list = allWords.split(" ");//将所有的生词放入一个list中,用于后续处理
 | 
					 | 
				
			||||||
    for (let i = 0; i < list.length; ++i) {
 | 
					    for (let i = 0; i < list.length; ++i) {
 | 
				
			||||||
        list[i] = list[i].replace(/(^\s*)|(\s*$)/g, ""); //消除单词两边的空字符
 | 
					        list[i] = list[i].replace(/(^\W*)|(\W*$)/g, ""); // 消除单词两边的非单词字符
 | 
				
			||||||
        list[i] = list[i].replace('|', "");
 | 
					        if (list[i] != "" && !totalSet.has(list[i])) {
 | 
				
			||||||
        list[i] = list[i].replace('?', "");
 | 
					            // 返回所有匹配单词的集合, 正则表达式RegExp()中, "\b"匹配一个单词的边界, g 表示全局匹配, i 表示对大小写不敏感。
 | 
				
			||||||
        if (list[i] !== "" && "<mark>".indexOf(list[i]) === -1 && "</mark>".indexOf(list[i]) === -1) {
 | 
					            let matches = new Set(articleContent.match(new RegExp("\\b" + list[i] + "\\b", "gi")));
 | 
				
			||||||
           //将文章中所有出现该单词word的地方改为:"<mark>" + word + "<mark>"。 正则表达式RegExp()中,"\\b"代表单词边界匹配。
 | 
					            if (matches.has("mark")) {
 | 
				
			||||||
 | 
					                // 优先处理单词为 "mark" 的情况
 | 
				
			||||||
            //修改代码
 | 
					                totalSet = new Set(["mark", ...totalSet]);
 | 
				
			||||||
            let articleContent_fb = articleContent;  //文章副本
 | 
					            }
 | 
				
			||||||
            while(articleContent_fb.toLowerCase().indexOf(list[i].toLowerCase()) !== -1 && list[i]!=""){
 | 
					            totalSet = new Set([...totalSet, ...matches]);
 | 
				
			||||||
                //找到副本中和list[i]匹配的第一个单词(第一种匹配情况),并赋值给list[i]。
 | 
					 | 
				
			||||||
                const index = articleContent_fb.toLowerCase().indexOf(list[i].toLowerCase());
 | 
					 | 
				
			||||||
                list[i] = articleContent_fb.substring(index, index + list[i].length);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
                articleContent_fb = articleContent_fb.substring(index + list[i].length);    // 使用副本中list[i]之后的子串替换掉副本
 | 
					 | 
				
			||||||
                articleContent = articleContent.replace(new RegExp("\\b"+list[i]+"\\b","g"),"<mark>" + list[i] + "</mark>");
 | 
					 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    } 
 | 
					    } 
 | 
				
			||||||
 | 
					    // 删除所有的mark标签,防止标签发生嵌套
 | 
				
			||||||
 | 
					    articleContent = articleContent.replace(/<(mark)[^>]*>/gi, "");
 | 
				
			||||||
 | 
					    articleContent = articleContent.replace(/<(\/mark)[^>]*>/gi, "");
 | 
				
			||||||
 | 
					    // 将文章中所有出现该单词word的地方改为:"<mark>" + word + "<mark>"。
 | 
				
			||||||
 | 
					    for (let word of totalSet) {
 | 
				
			||||||
 | 
					        articleContent = articleContent.replace(new RegExp("\\b" + word + "\\b", "g"), "<mark>" + word + "</mark>");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    document.getElementById("article").innerHTML = articleContent;
 | 
					    document.getElementById("article").innerHTML = articleContent;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function cancelHighlighting() {
 | 
					function cancelHighlighting() {
 | 
				
			||||||
    let articleContent = document.getElementById("article").innerText;//将原来的.innerText改为.innerHtml,原因同上
 | 
					    let articleContent = document.getElementById("article").innerHTML;
 | 
				
			||||||
    let pickedWords = document.getElementById("selected-words");
 | 
					    articleContent = articleContent.replace(/<(mark)[^>]*>/gi, "");
 | 
				
			||||||
    const dictionaryWords = document.getElementById("selected-words2");    
 | 
					    articleContent = articleContent.replace(/<(\/mark)[^>]*>/gi, "");
 | 
				
			||||||
    const list = pickedWords.value.split(" ");    
 | 
					 | 
				
			||||||
    if (pickedWords != null) {
 | 
					 | 
				
			||||||
        for (let i = 0; i < list.length; ++i) {
 | 
					 | 
				
			||||||
            list[i] = list[i].replace(/(^\s*)|(\s*$)/g, "");
 | 
					 | 
				
			||||||
            if (list[i] !== "") { //原来判断的代码中,替换的内容为“list[i]”这个字符串,这明显是错误的,我们需要替换的是list[i]里的内容
 | 
					 | 
				
			||||||
                articleContent = articleContent.replace(new RegExp("<mark>"+list[i]+"</mark>", "g"), list[i]);
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if (dictionaryWords != null) {
 | 
					 | 
				
			||||||
        let list2 = pickedWords.value.split(" ");
 | 
					 | 
				
			||||||
        for (let i = 0; i < list2.length; ++i) {
 | 
					 | 
				
			||||||
            list2 = dictionaryWords.value.split(" ");
 | 
					 | 
				
			||||||
            list2[i] = list2[i].replace(/(^\s*)|(\s*$)/g, "");
 | 
					 | 
				
			||||||
            if (list2[i] !== "") { //原来代码中,替换的内容为“list[i]”这个字符串,这明显是错误的,我们需要替换的是list[i]里的内容
 | 
					 | 
				
			||||||
                articleContent = articleContent.replace(new RegExp("<mark>"+list2[i]+"</mark>", "g"), list2[i]);
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    document.getElementById("article").innerHTML = articleContent;
 | 
					    document.getElementById("article").innerHTML = articleContent;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,13 +3,14 @@ var Reader = (function() {
 | 
				
			||||||
    let current_position = 0;
 | 
					    let current_position = 0;
 | 
				
			||||||
    let original_position = 0;
 | 
					    let original_position = 0;
 | 
				
			||||||
    let to_speak = "";
 | 
					    let to_speak = "";
 | 
				
			||||||
 | 
					    let current_rate = 1; // 添加这一行,设置默认速率为 1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function makeUtterance(str, rate) {
 | 
					    function makeUtterance(str, rate) {
 | 
				
			||||||
        let msg = new SpeechSynthesisUtterance(str);
 | 
					        let msg = new SpeechSynthesisUtterance(str);
 | 
				
			||||||
        msg.rate = rate;
 | 
					        msg.rate = rate;
 | 
				
			||||||
        msg.lang = "en-US";
 | 
					        msg.lang = "en-US";
 | 
				
			||||||
        msg.onboundary = ev => {
 | 
					        msg.onboundary = ev => {
 | 
				
			||||||
            if (ev.name == "word") {
 | 
					            if (ev.name === "word") {
 | 
				
			||||||
                current_position = ev.charIndex;
 | 
					                current_position = ev.charIndex;
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
| 
						 | 
					@ -24,12 +25,24 @@ var Reader = (function() {
 | 
				
			||||||
        reader.speak(msg);
 | 
					        reader.speak(msg);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function updateRate(rate) {
 | 
				
			||||||
 | 
					        // 停止当前的朗读
 | 
				
			||||||
 | 
					        stopRead();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 更新当前速率
 | 
				
			||||||
 | 
					        current_rate = rate;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 重新开始朗读
 | 
				
			||||||
 | 
					        read(to_speak, current_rate);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function stopRead() {
 | 
					    function stopRead() {
 | 
				
			||||||
        reader.cancel();
 | 
					        reader.cancel();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return {
 | 
					    return {
 | 
				
			||||||
        read: read,
 | 
					        read: read,
 | 
				
			||||||
        stopRead: stopRead
 | 
					        stopRead: stopRead,
 | 
				
			||||||
 | 
					        updateRate: updateRate // 添加这一行,将 updateRate 方法暴露出去
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
}) ();
 | 
					}) ();
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -9,7 +9,6 @@ function familiar(theWord) {
 | 
				
			||||||
            let new_freq = freq - 1;
 | 
					            let new_freq = freq - 1;
 | 
				
			||||||
            const allow_move = document.getElementById("move_dynamiclly").checked;
 | 
					            const allow_move = document.getElementById("move_dynamiclly").checked;
 | 
				
			||||||
            if (allow_move) {
 | 
					            if (allow_move) {
 | 
				
			||||||
 | 
					 | 
				
			||||||
                if (new_freq <= 0) {
 | 
					                if (new_freq <= 0) {
 | 
				
			||||||
                    removeWord(theWord);
 | 
					                    removeWord(theWord);
 | 
				
			||||||
                } else {
 | 
					                } else {
 | 
				
			||||||
| 
						 | 
					@ -114,7 +113,7 @@ function removeWord(word) {
 | 
				
			||||||
    // 根据词频信息删除元素
 | 
					    // 根据词频信息删除元素
 | 
				
			||||||
    word = word.replace('&', '&');
 | 
					    word = word.replace('&', '&');
 | 
				
			||||||
    const element_to_remove = document.getElementById(`p_${word}`);
 | 
					    const element_to_remove = document.getElementById(`p_${word}`);
 | 
				
			||||||
    if (element_to_remove != null) {
 | 
					    if (element_to_remove !== null) {
 | 
				
			||||||
        element_to_remove.remove();
 | 
					        element_to_remove.remove();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -129,7 +128,7 @@ function renderWord(word) {
 | 
				
			||||||
    for (const current of container.children) {
 | 
					    for (const current of container.children) {
 | 
				
			||||||
        const cur_word = parseWord(current);
 | 
					        const cur_word = parseWord(current);
 | 
				
			||||||
        // 找到第一个词频比它小的元素,插入到这个元素前面
 | 
					        // 找到第一个词频比它小的元素,插入到这个元素前面
 | 
				
			||||||
        if (compareWord(cur_word, word) == -1) {
 | 
					        if (compareWord(cur_word, word) === -1) {
 | 
				
			||||||
            container.insertBefore(new_element, current);
 | 
					            container.insertBefore(new_element, current);
 | 
				
			||||||
            inserted = true;
 | 
					            inserted = true;
 | 
				
			||||||
            break;
 | 
					            break;
 | 
				
			||||||
| 
						 | 
					@ -165,17 +164,11 @@ function elementFromString(string) {
 | 
				
			||||||
 *  当first大于second时返回1
 | 
					 *  当first大于second时返回1
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
function compareWord(first, second) {
 | 
					function compareWord(first, second) {
 | 
				
			||||||
    if (first.freq < second.freq) {
 | 
					    if (first.freq !== second.freq) {
 | 
				
			||||||
        return -1;
 | 
					        return first.freq < second.freq ? -1 : 1;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (first.freq > second.freq) {
 | 
					    if (first.word !== second.word) {
 | 
				
			||||||
        return 1;
 | 
					        return first.word < second.word ? -1 : 1;
 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if (first.word < second.word) {
 | 
					 | 
				
			||||||
        return -1;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if (first.word > second.word) {
 | 
					 | 
				
			||||||
        return 1;
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    return 0;
 | 
					    return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -1,7 +1,7 @@
 | 
				
			||||||
{% block body %}
 | 
					{% block body %}
 | 
				
			||||||
{% if session['logged_in'] %}
 | 
					{% if session['logged_in'] %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
你已登录 <a href="/{{ session['username'] }}">{{ session['username'] }}</a>。 登出点击<a href="/logout">这里</a>。
 | 
					你已登录 <a href="/{{ session['username'] }}/userpage">{{ session['username'] }}</a>。 登出点击<a href="/logout">这里</a>。
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{% else %}
 | 
					{% else %}
 | 
				
			||||||
<meta charset="utf-8" name="viewport" content="width=device-width, initial-scale=1.0, minimum-scale=0.5, maximum-scale=3.0, user-scalable=yes" />
 | 
					<meta charset="utf-8" name="viewport" content="width=device-width, initial-scale=1.0, minimum-scale=0.5, maximum-scale=3.0, user-scalable=yes" />
 | 
				
			||||||
| 
						 | 
					@ -32,14 +32,13 @@
 | 
				
			||||||
<div class="container">
 | 
					<div class="container">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <section class="signin-heading">
 | 
					  <section class="signin-heading">
 | 
				
			||||||
    <h1>Sign In</h1>
 | 
					    <h1>Sign in</h1>
 | 
				
			||||||
  </section>
 | 
					  </section>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <input type="text" placeholder="用户名" class="username" id="username">
 | 
					  <input type="text" placeholder="用户名" class="username" id="username">
 | 
				
			||||||
  <input type="password" placeholder="密码" class="password"  id="password">
 | 
					  <input type="password" placeholder="密码" class="password"  id="password">
 | 
				
			||||||
  <button type="button" class="btn" onclick="login()">登录</button>
 | 
					  <button type="button" class="btn" onclick="login()">登录</button>
 | 
				
			||||||
  <a class="signup" href="/signup">注册</a>
 | 
					  <a class="signup" href="/signup">注册</a>
 | 
				
			||||||
 | 
					 | 
				
			||||||
</div>
 | 
					</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{% endif %}
 | 
					{% endif %}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -44,6 +44,7 @@
 | 
				
			||||||
                <a href="http://youdao.com/w/eng/{{x[0]}}/#keyfrom=dict2.index">{{x[0]}}</a> {{x[1]}}
 | 
					                <a href="http://youdao.com/w/eng/{{x[0]}}/#keyfrom=dict2.index">{{x[0]}}</a> {{x[1]}}
 | 
				
			||||||
            {% endfor %}
 | 
					            {% endfor %}
 | 
				
			||||||
        {% endif %}
 | 
					        {% endif %}
 | 
				
			||||||
 | 
					        <p class="text-muted">Version: 20230810</p>
 | 
				
			||||||
        <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.0.2/dist/js/bootstrap.bundle.min.js" integrity="sha384-MrcW6ZMFYlzcLA8Nl+NtUVF0sA7MsXsP1UyJoMp4YLEuNSfAP+JcXn/tWtIaxVXM" crossorigin="anonymous"></script>
 | 
					        <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.0.2/dist/js/bootstrap.bundle.min.js" integrity="sha384-MrcW6ZMFYlzcLA8Nl+NtUVF0sA7MsXsP1UyJoMp4YLEuNSfAP+JcXn/tWtIaxVXM" crossorigin="anonymous"></script>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
    {{ yml['footer'] | safe }}
 | 
					    {{ yml['footer'] | safe }}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -53,7 +53,7 @@ You're logged in already! <a href="/logout">Logout</a>.
 | 
				
			||||||
<div class="container">
 | 
					<div class="container">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <section class="signin-heading">
 | 
					  <section class="signin-heading">
 | 
				
			||||||
    <h1>Sign Up</h1>
 | 
					    <h1>Sign up</h1>
 | 
				
			||||||
  </section>
 | 
					  </section>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <p><input type="username" id="username" placeholder="输入用户名" class="username"></p>
 | 
					  <p><input type="username" id="username" placeholder="输入用户名" class="username"></p>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -28,21 +28,11 @@
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      @keyframes shakes {
 | 
					      @keyframes shakes {
 | 
				
			||||||
            10%, 90% {
 | 
					          10%, 90% { transform: translate3d(-1px, 0, 0); }
 | 
				
			||||||
                transform: translate3d(-1px, 0, 0);
 | 
					          20%, 50% { transform: translate3d(+2px, 0, 0); }
 | 
				
			||||||
            }
 | 
					          30%, 70% { transform: translate3d(-4px, 0, 0); }
 | 
				
			||||||
            20%, 50% {
 | 
					          40%, 60% { transform: translate3d(+4px, 0, 0); }
 | 
				
			||||||
                transform: translate3d(+2px, 0, 0);
 | 
					          50% { transform: translate3d(-4px, 0, 0); }
 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
            30%, 70% {
 | 
					 | 
				
			||||||
                transform: translate3d(-4px, 0, 0);
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
            40%, 60% {
 | 
					 | 
				
			||||||
                transform: translate3d(+4px, 0, 0);
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
            50% {
 | 
					 | 
				
			||||||
                transform: translate3d(-4px, 0, 0);
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      .lead{
 | 
					      .lead{
 | 
				
			||||||
| 
						 | 
					@ -82,22 +72,13 @@
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
    {% endfor %}
 | 
					    {% endfor %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <button class="arrow" id="load_next_article" onclick="load_next_article();Reader.stopRead()"
 | 
					        <button class="arrow" id="load_next_article" onclick="load_next_article();Reader.stopRead()" title="下一篇 Next Article">⇨</button>
 | 
				
			||||||
            title="下一篇 Next Article">⇨
 | 
					        <button class="arrow" id="load_pre_article" onclick="load_pre_article();Reader.stopRead()" style="display: none" title="上一篇 Previous Article">⇦</button>
 | 
				
			||||||
    </button>
 | 
					 | 
				
			||||||
    <button class="arrow" id="load_pre_article" onclick="load_pre_article();Reader.stopRead()" style="display: none"
 | 
					 | 
				
			||||||
            title="上一篇 Previous Article">⇦
 | 
					 | 
				
			||||||
    </button>
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <p><b>阅读文章并回答问题</b></p>
 | 
					    <p><b>阅读文章并回答问题</b></p>
 | 
				
			||||||
    <div id="text-content">
 | 
					    <div id="text-content">
 | 
				
			||||||
        <div id="found">
 | 
					        <div id="found">
 | 
				
			||||||
            <div class="alert alert-success" role="alert">According to your word list, your level is <span
 | 
					            <div class="alert alert-success" role="alert">According to your word list, your level is <span class="text-decoration-underline" id="user_level">{{ today_article["user_level"] }}</span>  and we have chosen an article with a difficulty level of <span class="text-decoration-underline" id="text_level">{{ today_article["text_level"] }}</span> for you.</div>
 | 
				
			||||||
                    class="text-decoration-underline" id="user_level">{{ today_article["user_level"] }}</span> and we
 | 
					 | 
				
			||||||
                have chosen an article with a difficulty level of <span class="text-decoration-underline"
 | 
					 | 
				
			||||||
                                                                        id="text_level">{{ today_article["text_level"] }}</span>
 | 
					 | 
				
			||||||
                for you.
 | 
					 | 
				
			||||||
            </div>
 | 
					 | 
				
			||||||
                <p class="text-muted" id="date">Article added on: {{ today_article["date"] }}</p><br/>
 | 
					                <p class="text-muted" id="date">Article added on: {{ today_article["date"] }}</p><br/>
 | 
				
			||||||
            <div class="p-3 mb-2 bg-light text-dark" style="margin: 0 0.5%;"><br/>
 | 
					            <div class="p-3 mb-2 bg-light text-dark" style="margin: 0 0.5%;"><br/>
 | 
				
			||||||
            <p class="display-6" id="article_title">{{ today_article["article_title"] }}</p><br/>
 | 
					            <p class="display-6" id="article_title">{{ today_article["article_title"] }}</p><br/>
 | 
				
			||||||
| 
						 | 
					@ -117,22 +98,20 @@
 | 
				
			||||||
                    }
 | 
					                    }
 | 
				
			||||||
                </script>
 | 
					                </script>
 | 
				
			||||||
                <button onclick="toggle_visibility('answer');">ANSWER</button>
 | 
					                <button onclick="toggle_visibility('answer');">ANSWER</button>
 | 
				
			||||||
                <div id="answer" style="display:none;">{{ today_article['answer'] }}</div>
 | 
					                <div id="answer" style="display:none;">{{ today_article['answer'] }}</div><br/>
 | 
				
			||||||
                <br/>
 | 
					 | 
				
			||||||
            </div>
 | 
					            </div>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
        <div class="alert alert-success" role="alert" id="not_found" style="display:none;">
 | 
					        <div class="alert alert-success" role="alert" id="not_found" style="display:none;">
 | 
				
			||||||
            <p class="text-muted"><span class="badge bg-success">Notes:</span><br>No article is currently available for
 | 
					            <p class="text-muted"><span class="badge bg-success">Notes:</span><br>No article is currently available for you. You can try again a few times or mark new words in the passage to improve your level.</p>
 | 
				
			||||||
                you. You can try again a few times or mark new words in the passage to improve your level.</p>
 | 
					 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
        <div class="alert alert-success" role="alert" id="read_all" style="display:none;">
 | 
					        <div class="alert alert-success" role="alert" id="read_all" style="display:none;">
 | 
				
			||||||
            <p class="text-muted"><span class="badge bg-success">Notes:</span><br>You've read all the articles.</p>
 | 
					            <p class="text-muted"><span class="badge bg-success">Notes:</span><br>You've read all the articles.</p>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <input type="checkbox" id="highlightCheckbox" onclick="toggleHighlighting()"/>生词高亮
 | 
					    <input type="checkbox" onclick="toggleHighlighting()" checked/>生词高亮
 | 
				
			||||||
    <input type="checkbox" id="readCheckbox" onclick="onReadClick()"/>大声朗读
 | 
					    <input type="checkbox" onclick="onReadClick()" checked/>大声朗读
 | 
				
			||||||
    <input type="checkbox" id="chooseCheckbox" onclick="onChooseClick()"/>划词入库
 | 
					    <input type="checkbox" onclick="onChooseClick()" checked/>划词入库
 | 
				
			||||||
    <div class="range">
 | 
					    <div class="range">
 | 
				
			||||||
        <div class="field">
 | 
					        <div class="field">
 | 
				
			||||||
            <div class="sliderValue">
 | 
					            <div class="sliderValue">
 | 
				
			||||||
| 
						 | 
					@ -145,7 +124,7 @@
 | 
				
			||||||
    <form method="post" action="/{{ username }}/userpage">
 | 
					    <form method="post" action="/{{ username }}/userpage">
 | 
				
			||||||
        <textarea name="content" id="selected-words" rows="10" cols="120"></textarea><br/>
 | 
					        <textarea name="content" id="selected-words" rows="10" cols="120"></textarea><br/>
 | 
				
			||||||
        <button class="btn btn-primary btn-lg" type="submit" onclick="Reader.stopRead()">把生词加入我的生词库</button>
 | 
					        <button class="btn btn-primary btn-lg" type="submit" onclick="Reader.stopRead()">把生词加入我的生词库</button>
 | 
				
			||||||
        <button class="btn btn-primary btn-lg" type="reset" onclick="clearSelectedWords()">清除</button>
 | 
					        <button class="btn btn-primary btn-lg" type="reset">清除</button>
 | 
				
			||||||
    </form>
 | 
					    </form>
 | 
				
			||||||
    {% if session.get['thisWord'] %}
 | 
					    {% if session.get['thisWord'] %}
 | 
				
			||||||
        <script type="text/javascript">
 | 
					        <script type="text/javascript">
 | 
				
			||||||
| 
						 | 
					@ -174,8 +153,7 @@
 | 
				
			||||||
                {% if session.get('thisWord') == x[0] and session.get('time') == 1 %}
 | 
					                {% if session.get('thisWord') == x[0] and session.get('time') == 1 %}
 | 
				
			||||||
                {% endif %}
 | 
					                {% endif %}
 | 
				
			||||||
                <p id='p_{{ word }}' class="new-word" >
 | 
					                <p id='p_{{ word }}' class="new-word" >
 | 
				
			||||||
                    <a id="word_{{ word }}" class="btn btn-light"
 | 
					                    <a id="word_{{ word }}"  class="btn btn-light" href='http://youdao.com/w/eng/{{ word }}/#keyfrom=dict2.index'
 | 
				
			||||||
                       href='http://youdao.com/w/eng/{{ word }}/#keyfrom=dict2.index'
 | 
					 | 
				
			||||||
                    role="button">{{ word }}</a>
 | 
					                    role="button">{{ word }}</a>
 | 
				
			||||||
                    ( <a id="freq_{{ word }}" title="{{ word }}">{{ freq }}</a> )
 | 
					                    ( <a id="freq_{{ word }}" title="{{ word }}">{{ freq }}</a> )
 | 
				
			||||||
                    <a class="btn btn-success" onclick="familiar('{{ word }}')" role="button">熟悉</a>
 | 
					                    <a class="btn btn-success" onclick="familiar('{{ word }}')" role="button">熟悉</a>
 | 
				
			||||||
| 
						 | 
					@ -196,56 +174,12 @@
 | 
				
			||||||
{% endif %}
 | 
					{% endif %}
 | 
				
			||||||
<script type="text/javascript">
 | 
					<script type="text/javascript">
 | 
				
			||||||
    window.onload = function () { // 页面加载时执行
 | 
					    window.onload = function () { // 页面加载时执行
 | 
				
			||||||
        const settings = {
 | 
					 | 
				
			||||||
            // initialize settings from localStorage
 | 
					 | 
				
			||||||
            highlightChecked: localStorage.getItem('highlightChecked') !== 'false', // localStorage stores strings, default to true. same below
 | 
					 | 
				
			||||||
            readChecked: localStorage.getItem('readChecked') !== 'false',
 | 
					 | 
				
			||||||
            chooseChecked: localStorage.getItem('chooseChecked') !== 'false',
 | 
					 | 
				
			||||||
            rangeValue: localStorage.getItem('rangeValue') || '1',
 | 
					 | 
				
			||||||
            selectedWords: localStorage.getItem('selectedWords') || ''
 | 
					 | 
				
			||||||
        };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const elements = {
 | 
					 | 
				
			||||||
            highlightCheckbox: document.querySelector('#highlightCheckbox'),
 | 
					 | 
				
			||||||
            readCheckbox: document.querySelector('#readCheckbox'),
 | 
					 | 
				
			||||||
            chooseCheckbox: document.querySelector('#chooseCheckbox'),
 | 
					 | 
				
			||||||
            rangeComponent: document.querySelector('#rangeComponent'),
 | 
					 | 
				
			||||||
            rangeValueDisplay: document.querySelector('#rangeValue'),
 | 
					 | 
				
			||||||
            selectedWordsInput: document.querySelector('#selected-words')
 | 
					 | 
				
			||||||
        };
 | 
					 | 
				
			||||||
        // 应用设置到页面元素
 | 
					 | 
				
			||||||
        elements.highlightCheckbox.checked = settings.highlightChecked;
 | 
					 | 
				
			||||||
        elements.readCheckbox.checked = settings.readChecked;
 | 
					 | 
				
			||||||
        elements.chooseCheckbox.checked = settings.chooseChecked;
 | 
					 | 
				
			||||||
        elements.rangeComponent.value = settings.rangeValue;
 | 
					 | 
				
			||||||
        elements.rangeValueDisplay.textContent = `${settings.rangeValue}x`;
 | 
					 | 
				
			||||||
        elements.selectedWordsInput.value = settings.selectedWords;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        // 刷新页面或进入页面时判断,若不是首篇文章,则上一篇按钮可见
 | 
					        // 刷新页面或进入页面时判断,若不是首篇文章,则上一篇按钮可见
 | 
				
			||||||
        if (sessionStorage.getItem('pre_page_button') !== 'display' && sessionStorage.getItem('pre_page_button')) {
 | 
					        if(sessionStorage.getItem('pre_page_button')!="display" && sessionStorage.getItem('pre_page_button')){
 | 
				
			||||||
            $('#load_pre_article').show();
 | 
					            $('#load_pre_article').show();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					 | 
				
			||||||
        // 事件监听器
 | 
					 | 
				
			||||||
        elements.selectedWordsInput.addEventListener('input', () => {
 | 
					 | 
				
			||||||
            localStorage.setItem('selectedWords', elements.selectedWordsInput.value);
 | 
					 | 
				
			||||||
        });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        elements.rangeComponent.addEventListener('input', () => {
 | 
					 | 
				
			||||||
            const rangeValue = elements.rangeComponent.value;
 | 
					 | 
				
			||||||
            elements.rangeValueDisplay.textContent = `${rangeValue}x`;
 | 
					 | 
				
			||||||
            localStorage.setItem('rangeValue', rangeValue);
 | 
					 | 
				
			||||||
        });
 | 
					 | 
				
			||||||
     };
 | 
					     };
 | 
				
			||||||
 | 
					 | 
				
			||||||
    function clearSelectedWords() {
 | 
					 | 
				
			||||||
        localStorage.removeItem('selectedWords');
 | 
					 | 
				
			||||||
        document.querySelector('#selected-words').value = '';
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    function load_next_article(){
 | 
					    function load_next_article(){
 | 
				
			||||||
        $("#load_next_article").prop("disabled", true)
 | 
					 | 
				
			||||||
        $.ajax({
 | 
					        $.ajax({
 | 
				
			||||||
            url: '/get_next_article/{{username}}',
 | 
					            url: '/get_next_article/{{username}}',
 | 
				
			||||||
            dataType: 'json',
 | 
					            dataType: 'json',
 | 
				
			||||||
| 
						 | 
					@ -255,14 +189,10 @@
 | 
				
			||||||
                    update(data['today_article']);
 | 
					                    update(data['today_article']);
 | 
				
			||||||
                    check_pre(data['visited_articles']);
 | 
					                    check_pre(data['visited_articles']);
 | 
				
			||||||
                    check_next(data['result_of_generate_article']);
 | 
					                    check_next(data['result_of_generate_article']);
 | 
				
			||||||
                    toggleHighlighting();
 | 
					 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
            }, complete: function (xhr, status) {
 | 
					 | 
				
			||||||
                $("#load_next_article").prop("disabled", false)
 | 
					 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					 | 
				
			||||||
    function load_pre_article(){
 | 
					    function load_pre_article(){
 | 
				
			||||||
        $.ajax({
 | 
					        $.ajax({
 | 
				
			||||||
            url: '/get_pre_article/{{username}}',
 | 
					            url: '/get_pre_article/{{username}}',
 | 
				
			||||||
| 
						 | 
					@ -272,12 +202,10 @@
 | 
				
			||||||
                if(data['today_article']){
 | 
					                if(data['today_article']){
 | 
				
			||||||
                    update(data['today_article']);
 | 
					                    update(data['today_article']);
 | 
				
			||||||
                    check_pre(data['visited_articles']);
 | 
					                    check_pre(data['visited_articles']);
 | 
				
			||||||
                    toggleHighlighting();
 | 
					 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					 | 
				
			||||||
    function update(today_article){
 | 
					    function update(today_article){
 | 
				
			||||||
        $('#user_level').html(today_article['user_level']);
 | 
					        $('#user_level').html(today_article['user_level']);
 | 
				
			||||||
        $('#text_level').html(today_article["text_level"]);
 | 
					        $('#text_level').html(today_article["text_level"]);
 | 
				
			||||||
| 
						 | 
					@ -288,15 +216,10 @@
 | 
				
			||||||
        $('#question').html(today_article["question"]);
 | 
					        $('#question').html(today_article["question"]);
 | 
				
			||||||
        $('#answer').html(today_article["answer"]);
 | 
					        $('#answer').html(today_article["answer"]);
 | 
				
			||||||
        document.querySelector('#text_level').classList.add('mark'); // highlight text difficult level for 2 seconds
 | 
					        document.querySelector('#text_level').classList.add('mark'); // highlight text difficult level for 2 seconds
 | 
				
			||||||
        setTimeout(() => {
 | 
					        setTimeout(() => {document.querySelector('#text_level').classList.remove('mark');}, 2000);
 | 
				
			||||||
            document.querySelector('#text_level').classList.remove('mark');
 | 
					 | 
				
			||||||
        }, 2000);
 | 
					 | 
				
			||||||
        document.querySelector('#user_level').classList.add('mark'); // do the same thing for user difficulty level
 | 
					        document.querySelector('#user_level').classList.add('mark'); // do the same thing for user difficulty level
 | 
				
			||||||
        setTimeout(() => {
 | 
					        setTimeout(() => {document.querySelector('#user_level').classList.remove('mark');}, 2000);
 | 
				
			||||||
            document.querySelector('#user_level').classList.remove('mark');
 | 
					 | 
				
			||||||
        }, 2000);
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					 | 
				
			||||||
<!-- 检查是否存在上一篇或下一篇,不存在则对应按钮隐藏-->
 | 
					<!-- 检查是否存在上一篇或下一篇,不存在则对应按钮隐藏-->
 | 
				
			||||||
    function check_pre(visited_articles){
 | 
					    function check_pre(visited_articles){
 | 
				
			||||||
        if((visited_articles=='')||(visited_articles['index']<=0)){
 | 
					        if((visited_articles=='')||(visited_articles['index']<=0)){
 | 
				
			||||||
| 
						 | 
					@ -307,11 +230,9 @@
 | 
				
			||||||
            sessionStorage.setItem('pre_page_button', 'show')
 | 
					            sessionStorage.setItem('pre_page_button', 'show')
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					 | 
				
			||||||
    function check_next(result_of_generate_article){
 | 
					    function check_next(result_of_generate_article){
 | 
				
			||||||
        if(result_of_generate_article == "found"){
 | 
					        if(result_of_generate_article == "found"){
 | 
				
			||||||
            $('#found').show();
 | 
					            $('#found').show();$('#not_found').hide();
 | 
				
			||||||
            $('#not_found').hide();
 | 
					 | 
				
			||||||
            $('#read_all').hide();
 | 
					            $('#read_all').hide();
 | 
				
			||||||
        }else if(result_of_generate_article == "not found"){
 | 
					        }else if(result_of_generate_article == "not found"){
 | 
				
			||||||
            $('#found').hide();
 | 
					            $('#found').hide();
 | 
				
			||||||
| 
						 | 
					@ -323,11 +244,16 @@
 | 
				
			||||||
            $('#read_all').show();
 | 
					            $('#read_all').show();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    document.getElementById('rangeComponent').addEventListener('input', function() {
 | 
				
			||||||
 | 
					    var rate = this.value;
 | 
				
			||||||
 | 
					    Reader.updateRate(rate);
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
</body>
 | 
					</body>
 | 
				
			||||||
<style>
 | 
					<style>
 | 
				
			||||||
    mark {
 | 
					    mark {
 | 
				
			||||||
        color: #{{ yml['highlight']['color'] }};
 | 
					        color: {{ yml['highlight']['color'] }};
 | 
				
			||||||
        background-color: rgba(0, 0, 0, 0);
 | 
					        background-color: rgba(0, 0, 0, 0);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
</style>
 | 
					</style>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -20,9 +20,13 @@
 | 
				
			||||||
	<title>EnglishPal Study Room for {{username}}</title>
 | 
						<title>EnglishPal Study Room for {{username}}</title>
 | 
				
			||||||
    </head>
 | 
					    </head>
 | 
				
			||||||
    <body>
 | 
					    <body>
 | 
				
			||||||
     <p>取消勾选认识的单词</p>
 | 
						<div class="container-fluid">
 | 
				
			||||||
 | 
						    <p class="mt-md-3">
 | 
				
			||||||
 | 
							<input type="button" id="btn-cancel-selection" value="取消勾选" onclick="toggleCheckboxSelection(false)" />
 | 
				
			||||||
 | 
							<input type="button" id="btn-selection" value="全部勾选" onclick="toggleCheckboxSelection(true)" />
 | 
				
			||||||
 | 
						    </p>
 | 
				
			||||||
	    <form method="post" action="/{{username}}/mark">
 | 
						    <form method="post" action="/{{username}}/mark">
 | 
				
			||||||
       <input type="submit" name="add-btn" value="加入我的生词簿"/>
 | 
							<button type="submit" name="add-btn" class="btn btn-outline-primary btn-lg">加入我的生词簿</button>
 | 
				
			||||||
		{% for x in lst %}
 | 
							{% for x in lst %}
 | 
				
			||||||
		{% set word = x[0]%}
 | 
							{% set word = x[0]%}
 | 
				
			||||||
		<p>
 | 
							<p>
 | 
				
			||||||
| 
						 | 
					@ -41,5 +45,6 @@
 | 
				
			||||||
            <script src="{{ js }}" ></script>
 | 
					            <script src="{{ js }}" ></script>
 | 
				
			||||||
            {% endfor %}
 | 
					            {% endfor %}
 | 
				
			||||||
	    {% endif %}
 | 
						    {% endif %}
 | 
				
			||||||
 | 
						</div>
 | 
				
			||||||
    </body>
 | 
					    </body>
 | 
				
			||||||
</html>
 | 
					</html>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,6 +1,9 @@
 | 
				
			||||||
import pytest
 | 
					import pytest
 | 
				
			||||||
 | 
					import sqlite3
 | 
				
			||||||
 | 
					import time
 | 
				
			||||||
from selenium import webdriver
 | 
					from selenium import webdriver
 | 
				
			||||||
from selenium.webdriver.common.desired_capabilities import DesiredCapabilities
 | 
					
 | 
				
			||||||
 | 
					from pathlib import Path
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@pytest.fixture
 | 
					@pytest.fixture
 | 
				
			||||||
def URL():
 | 
					def URL():
 | 
				
			||||||
| 
						 | 
					@ -9,5 +12,24 @@ def URL():
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@pytest.fixture
 | 
					@pytest.fixture
 | 
				
			||||||
def driver():
 | 
					def driver():
 | 
				
			||||||
    my_driver = webdriver.Chrome()
 | 
					    return webdriver.Edge()  # follow the "End-to-end testing" section in README.md to install the web driver executable
 | 
				
			||||||
    return my_driver
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@pytest.fixture
 | 
				
			||||||
 | 
					def restore_sqlite_database():
 | 
				
			||||||
 | 
					    '''
 | 
				
			||||||
 | 
					    Automatically restore SQLite database file app/db/wordfreqapp.db
 | 
				
			||||||
 | 
					    using SQL statements from app/static/wordfreqapp.sql
 | 
				
			||||||
 | 
					    '''
 | 
				
			||||||
 | 
					    con = sqlite3.connect('../db/wordfreqapp.db')
 | 
				
			||||||
 | 
					    with con:
 | 
				
			||||||
 | 
					        con.executescript('DROP TABLE IF EXISTS user;')
 | 
				
			||||||
 | 
					        con.executescript('DROP TABLE IF EXISTS article;')
 | 
				
			||||||
 | 
					        con.executescript(open('../static/wordfreqapp.sql', encoding='utf8').read())
 | 
				
			||||||
 | 
					    con.close()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@pytest.fixture(autouse=True)
 | 
				
			||||||
 | 
					def restart_englishpal(restore_sqlite_database):
 | 
				
			||||||
 | 
					    (Path(__file__).parent / '../main.py').touch()
 | 
				
			||||||
 | 
					    time.sleep(1)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,33 @@
 | 
				
			||||||
 | 
					import uuid
 | 
				
			||||||
 | 
					from selenium.webdriver.support.ui import WebDriverWait
 | 
				
			||||||
 | 
					from selenium.webdriver.support import expected_conditions as EC
 | 
				
			||||||
 | 
					from selenium.common.exceptions import UnexpectedAlertPresentException, NoAlertPresentException
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def signup(URL, driver):
 | 
				
			||||||
 | 
					    username = 'TestUser' + str(uuid.uuid1()).split('-')[0].title()
 | 
				
			||||||
 | 
					    password = '[Abc+123]'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    driver.get(URL)
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					    elem = driver.find_element_by_link_text('注册')
 | 
				
			||||||
 | 
					    elem.click()
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    elem = driver.find_element_by_id('username')
 | 
				
			||||||
 | 
					    elem.send_keys(username)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    elem = driver.find_element_by_id('password')
 | 
				
			||||||
 | 
					    elem.send_keys(password)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    elem = driver.find_element_by_id('password2')
 | 
				
			||||||
 | 
					    elem.send_keys(password)    
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    elem = driver.find_element_by_class_name('btn') # 找到"注册"按钮
 | 
				
			||||||
 | 
					    elem.click()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        WebDriverWait(driver, 1).until(EC.alert_is_present())
 | 
				
			||||||
 | 
					        driver.switch_to.alert.accept()
 | 
				
			||||||
 | 
					    except (UnexpectedAlertPresentException, NoAlertPresentException):
 | 
				
			||||||
 | 
					        pass
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    return username, password
 | 
				
			||||||
| 
						 | 
					@ -1,68 +1,23 @@
 | 
				
			||||||
# -*- coding: utf-8 -*-
 | 
					import time
 | 
				
			||||||
# Run the docker image using the following command:
 | 
					from helper import signup
 | 
				
			||||||
# docker run -d -p 4444:4444 selenium/standalone-chrome
 | 
					 | 
				
			||||||
from selenium import webdriver
 | 
					 | 
				
			||||||
from selenium.webdriver.common.desired_capabilities import DesiredCapabilities
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import random, time
 | 
					 | 
				
			||||||
import string
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
driver = webdriver.Remote('http://localhost:4444/wd/hub', DesiredCapabilities.FIREFOX)
 | 
					 | 
				
			||||||
driver.implicitly_wait(10)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
HOME_PAGE = 'http://121.4.94.30:91/'
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def has_punctuation(s):
 | 
					def test_add_word(URL, driver):
 | 
				
			||||||
    return [c for c in s if c in string.punctuation] != []
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
def test_add_word():
 | 
					 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        driver.get(HOME_PAGE)
 | 
					        username, password = signup(URL, driver) # sign up a new account and automatically log in
 | 
				
			||||||
        assert 'English Pal -' in driver.page_source
 | 
					        time.sleep(1)
 | 
				
			||||||
    
 | 
					 | 
				
			||||||
        # login
 | 
					 | 
				
			||||||
        elem = driver.find_element_by_link_text('登录')
 | 
					 | 
				
			||||||
        elem.click()
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
        uname = 'lanhui'
 | 
					 | 
				
			||||||
        password = 'l0ve1t'
 | 
					 | 
				
			||||||
        elem = driver.find_element_by_name('username')
 | 
					 | 
				
			||||||
        elem.send_keys(uname)
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
        elem = driver.find_element_by_name('password')
 | 
					 | 
				
			||||||
        elem.send_keys(password)
 | 
					 | 
				
			||||||
        
 | 
					 | 
				
			||||||
        elem = driver.find_element_by_xpath('//form[1]/p[3]/input[1]') # 找到登录按钮
 | 
					 | 
				
			||||||
        elem.click()
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
        assert 'EnglishPal Study Room for ' + uname in  driver.title
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
        # get essay content
 | 
					 | 
				
			||||||
        elem = driver.find_element_by_id('text-content')
 | 
					 | 
				
			||||||
        essay_content = elem.text
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # enter the word in the text area
 | 
				
			||||||
        elem = driver.find_element_by_id('selected-words')
 | 
					        elem = driver.find_element_by_id('selected-words')
 | 
				
			||||||
        word = random.choice(essay_content.split())
 | 
					        word = 'devour'
 | 
				
			||||||
        while 'font>' in word or 'br>' in word or 'p>' in word or len(word) < 6 or has_punctuation(word):
 | 
					 | 
				
			||||||
            word = random.choice(essay_content.split())
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        elem.send_keys(word)
 | 
					        elem.send_keys(word)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        elem = driver.find_element_by_xpath('//form[1]//input[1]') # 找到get所有词频按钮
 | 
					        elem = driver.find_element_by_xpath('//form[1]//button[1]') # 找到"把生词加入我的生词库"按钮
 | 
				
			||||||
        elem.click()
 | 
					        elem.click()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        elems = driver.find_elements_by_xpath("//input[@type='checkbox']")
 | 
					        elem = driver.find_element_by_name('add-btn') # 找到"加入我的生词簿"按钮
 | 
				
			||||||
        for elem in elems:
 | 
					 | 
				
			||||||
            if elem.get_attribute('name') == 'marked':
 | 
					 | 
				
			||||||
        elem.click()
 | 
					        elem.click()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        elem = driver.find_element_by_name('add-btn') # 找到加入我的生词簿按钮
 | 
					 | 
				
			||||||
        elem.click()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        driver.refresh()
 | 
					 | 
				
			||||||
        driver.refresh()
 | 
					 | 
				
			||||||
        driver.refresh()        
 | 
					 | 
				
			||||||
        elems = driver.find_elements_by_xpath("//p[@class='new-word']/a")
 | 
					        elems = driver.find_elements_by_xpath("//p[@class='new-word']/a")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        found = 0
 | 
					        found = 0
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,43 @@
 | 
				
			||||||
 | 
					''' Contributed by Lin Junhong et al. 2023-06.'''
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import requests
 | 
				
			||||||
 | 
					import multiprocessing
 | 
				
			||||||
 | 
					import time
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def stress(username):
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        data = {
 | 
				
			||||||
 | 
					            'username': username,
 | 
				
			||||||
 | 
					            'password': '123123'
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        headers = {
 | 
				
			||||||
 | 
					            'User-Agent': 'Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Mobile Safari/537.36 Edg/114.0.1823.51'
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        session = requests.session()
 | 
				
			||||||
 | 
					        response = session.post(url='http://127.0.0.1:5000/signup', data=data, headers=headers)
 | 
				
			||||||
 | 
					        print('Sign up ', response.status_code)
 | 
				
			||||||
 | 
					        time.sleep(0.5)
 | 
				
			||||||
 | 
					        response = session.post(url='http://127.0.0.1:5000/login', data=data, headers=headers)
 | 
				
			||||||
 | 
					        print('Sign in ', response.status_code)
 | 
				
			||||||
 | 
					        time.sleep(0.5)
 | 
				
			||||||
 | 
					        response = session.get(url=f'http://127.0.0.1:5000/{username}/userpage', headers=headers)
 | 
				
			||||||
 | 
					        print('User page', response.status_code)
 | 
				
			||||||
 | 
					        time.sleep(0.5)
 | 
				
			||||||
 | 
					        print(session.cookies)
 | 
				
			||||||
 | 
					        for i in range(5):
 | 
				
			||||||
 | 
					            response = session.get(url=f'http://127.0.0.1:5000/get_next_article/{username}', headers=headers, cookies=session.cookies)
 | 
				
			||||||
 | 
					            time.sleep(0.5)
 | 
				
			||||||
 | 
					            print(f'Next page ({i}) [{username}]')
 | 
				
			||||||
 | 
					            print(response.status_code)
 | 
				
			||||||
 | 
					            print(response.json()['today_article']['article_title'])
 | 
				
			||||||
 | 
					    except Exception as e:
 | 
				
			||||||
 | 
					        print(e)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if __name__ == '__main__':
 | 
				
			||||||
 | 
					    username = 'Learner'
 | 
				
			||||||
 | 
					    pool = multiprocessing.Pool(processes=10)
 | 
				
			||||||
 | 
					    for i in range(10):
 | 
				
			||||||
 | 
					        pool.apply_async(stress, (f'{username}{i}',))
 | 
				
			||||||
 | 
					    pool.close()
 | 
				
			||||||
 | 
					    pool.join()
 | 
				
			||||||
| 
						 | 
					@ -15,6 +15,9 @@ from wordfreqCMD import sort_in_descending_order
 | 
				
			||||||
import pickle_idea
 | 
					import pickle_idea
 | 
				
			||||||
import pickle_idea2
 | 
					import pickle_idea2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import logging
 | 
				
			||||||
 | 
					logging.basicConfig(filename='log.txt', format='%(asctime)s %(message)s', level=logging.DEBUG)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# 初始化蓝图
 | 
					# 初始化蓝图
 | 
				
			||||||
userService = Blueprint("user_bp", __name__)
 | 
					userService = Blueprint("user_bp", __name__)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -32,7 +35,9 @@ def get_next_article(username):
 | 
				
			||||||
        else:  # 当前不为“null”,直接 index+=1
 | 
					        else:  # 当前不为“null”,直接 index+=1
 | 
				
			||||||
            visited_articles["index"] += 1
 | 
					            visited_articles["index"] += 1
 | 
				
			||||||
        session["visited_articles"] = visited_articles
 | 
					        session["visited_articles"] = visited_articles
 | 
				
			||||||
 | 
					        logging.debug('/get_next_article: start calling get_today_arcile()')
 | 
				
			||||||
        visited_articles, today_article, result_of_generate_article = get_today_article(user_freq_record, session.get('visited_articles'))
 | 
					        visited_articles, today_article, result_of_generate_article = get_today_article(user_freq_record, session.get('visited_articles'))
 | 
				
			||||||
 | 
					        logging.debug('/get_next_arcile: done.')
 | 
				
			||||||
        data = {
 | 
					        data = {
 | 
				
			||||||
            'visited_articles': visited_articles,
 | 
					            'visited_articles': visited_articles,
 | 
				
			||||||
            'today_article': today_article,
 | 
					            'today_article': today_article,
 | 
				
			||||||
| 
						 | 
					@ -129,7 +134,7 @@ def userpage(username):
 | 
				
			||||||
    user_freq_record = path_prefix + 'static/frequency/' + 'frequency_%s.pickle' % (username)
 | 
					    user_freq_record = path_prefix + 'static/frequency/' + 'frequency_%s.pickle' % (username)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if request.method == 'POST':  # when we submit a form
 | 
					    if request.method == 'POST':  # when we submit a form
 | 
				
			||||||
        content = escape(request.form['content'])
 | 
					        content = request.form['content']
 | 
				
			||||||
        f = WordFreq(content)
 | 
					        f = WordFreq(content)
 | 
				
			||||||
        lst = f.get_freq()
 | 
					        lst = f.get_freq()
 | 
				
			||||||
        return render_template('userpage_post.html',username=username,lst = lst, yml=Yaml.yml)
 | 
					        return render_template('userpage_post.html',username=username,lst = lst, yml=Yaml.yml)
 | 
				
			||||||
| 
						 | 
					@ -176,7 +181,11 @@ def user_mark_word(username):
 | 
				
			||||||
        for word in request.form.getlist('marked'):
 | 
					        for word in request.form.getlist('marked'):
 | 
				
			||||||
            lst.append((word, [get_time()]))
 | 
					            lst.append((word, [get_time()]))
 | 
				
			||||||
        d = pickle_idea2.merge_frequency(lst, lst_history)
 | 
					        d = pickle_idea2.merge_frequency(lst, lst_history)
 | 
				
			||||||
 | 
					        if len(lst_history) > 999:
 | 
				
			||||||
 | 
					            flash('You have way too many words in your difficult-words book. Delete some first.')
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
            pickle_idea2.save_frequency_to_pickle(d, user_freq_record)
 | 
					            pickle_idea2.save_frequency_to_pickle(d, user_freq_record)
 | 
				
			||||||
 | 
					            flash('Added %s.' % (', '.join(request.form.getlist('marked'))))
 | 
				
			||||||
        return redirect(url_for('user_bp.userpage', username=username))
 | 
					        return redirect(url_for('user_bp.userpage', username=username))
 | 
				
			||||||
    else:
 | 
					    else:
 | 
				
			||||||
        return 'Under construction'
 | 
					        return 'Under construction'
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										7
									
								
								build.sh
								
								
								
								
							
							
						
						
									
										7
									
								
								build.sh
								
								
								
								
							| 
						 | 
					@ -2,10 +2,7 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
DEPLOYMENT_DIR=/home/lanhui/englishpal2/EnglishPal
 | 
					DEPLOYMENT_DIR=/home/lanhui/englishpal2/EnglishPal
 | 
				
			||||||
cd $DEPLOYMENT_DIR
 | 
					cd $DEPLOYMENT_DIR
 | 
				
			||||||
 | 
					pwd
 | 
				
			||||||
# Install dependencies
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
pip3 install -r requirements.txt
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Stop service
 | 
					# Stop service
 | 
				
			||||||
sudo docker stop EnglishPal
 | 
					sudo docker stop EnglishPal
 | 
				
			||||||
| 
						 | 
					@ -15,7 +12,7 @@ sudo docker rm EnglishPal
 | 
				
			||||||
sudo docker build -t englishpal .
 | 
					sudo docker build -t englishpal .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Run the application
 | 
					# Run the application
 | 
				
			||||||
sudo docker run --restart=always -d --name EnglishPal -p 90:80 -v ${DEPLOYMENT_DIR}/app/static/frequency:/app/static/frequency -v ${DEPLOYMENT_DIR}/app/static/:/app/static/ -t englishpal  # for permanently saving data
 | 
					sudo docker run --restart=always -d --name EnglishPal -p 90:80  -v ${DEPLOYMENT_DIR}/app/static/frequency:/app/static/frequency --mount type=volume,src=englishpal-db,target=/app/db -t englishpal # for permanently saving data
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Save space.  Run it after sudo docker run
 | 
					# Save space.  Run it after sudo docker run
 | 
				
			||||||
sudo docker system prune -a -f
 | 
					sudo docker system prune -a -f
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue