Back End Developer's Guide - WheatonCS/Lexos GitHub Wiki
This guide gives Lexos coders an understanding of some of the Lexos Back-End code.
- Introduction to the Back End
- Switching between Versions of Lexos
- Patching Lexos v.3.1.1
- The Flask Session
- Structure of the Flask App
- White Space and Unicode Issues
Lexos was initially developed in Python 2.7, and, as of version 3.2 has switched to Python 3.x. It uses the Flask web framework. Flask sets up a local server and handles processing tasks, including the rendering of HTML templates and the sending and receiving of http requests to and from the client.
It is possible to run both Lexos 3.1 and Lexos 3.2 on the same machine. How you do it depends on which version of Anaconda you have downloaded. If you have downloaded the Python 2.7 version of Anaconda and wish to run Lexos 3.2, you will need to perform the following steps:
- Switching to the Python 3 environment.
$ conda create -n py36 python=3.6 anaconda
You can use any name you want in place of "py36".
- Activate the environment.
**Linux, OS X:** source activate py36
**Windows:** activate py36
To de-activeate the environment so that you can run Lexos 3.2 or greater, use deactivate py36
or source deactivate py36
.
You can see what environments you have with conda info --envs
. You can see what Python version the active environment is using with python --version
.
You should have access to all default Anaconda packages, but you may have to install individual dependencies for the Python 3 environment.
If you are running Anaconda with Python 3.x, the procedure is the same, except that you will create an environment for Python 2.7. Adjust the commands above accordingly.
Note that Lexos creates a cache
folder in your /tmp/
directory (~\AppData\Local\Temp\USERNAME
in Windows). This may cause errors when you switch between versions of Lexos. Scrubber, for instance, is known to fail. You may need to delete this directory before launching Lexos when switching between versions.
Occasionally it will be necessary to apply patches to the Lexos v3.1.1 release. The procedure is somewhat involved, so the steps are laid out here.
- On the GitHub website, take a note of the release message. Then delete the "Lexos v3.1.1" release.
- Checkout the
3.1_bugfix
branch. Commit changes and push them to GitHub. - On the command line, delete the remote and local tags with
git push --delete origin v3.1.1
git tag --delete v3.1.1
- Create a new tag (replacing the commit number shown below with the latest commit):
git tag -a v3.1.1 c3c946f -m "Message here"
git push --tags origin 3.1_bugfix
- On GitHub, click the "v3.1.1" tag and then click "Add Release Notes". Add the title "Lexos v3.1.1" and restore the old message, adding any new material. The message should specify that this will be the last release for Python 2.7. Click "Publish Release".
You can get the full documentation for Flask here .
When Lexos is first run, Flask establishes a session cookie to handle information about the user's workspace. This cookie holds values that can be accessed from both the front end and the back end by means of the session
variable. Lexos uses the session
cookie for the following purposes:
- To cache user's options and other information.
- To send default information to the front end. Lexos defaults can be found in
Lexos/lexos/helpers/constants.py
.
The session
variable functions like a Python dictionary. Hence session["option"]
will access the value of the specified option in the session.
The session can be renewed by calling session_function.init()
.
When a user enters information through front-end form field, this is sent to the back end through the Flask request
object. Information in the request
object can be accessed in a variety of ways:
-
request.method
: returns methods of the request, typicallyPOST
orGET
; -
request.form
: returns a dict containing the name each form field mapped to its value; -
request.form.getlist
: returns a dict containing the name each form field mapped to multiple values (if there is more than one); -
request.file
: returns a dict containing the id of each file in an HTML file input submission; -
request.json
: returns a json object typically containing the same information asrequest.form
. It is generally sent from an Ajax request.
For a complete set of properties in the Flask request
object, see Accessing Request Data in the Flask documentation.
In Lexos 2, most Lexos tools required the user to submit the form, which sent the form data to the back end and then triggered a page refresh with the back-end response. In Lexos 3, some features have been transfered to Ajax functions, which send data to return a response to the front end without the need for a page refresh.
The main Lexos
folder contains a file (currently called run.py
, but that may change) that runs the app with the command python run.py
. All other files for the app are in the lexos
folder. This contains the file application.py
, which sets up the application environment. It also contain the config.cfg
, which is used to set configuration settings such as LOCAL_MODE
. The lexos
folder contains the following subfolders:
-
/helpers
: Contains Lexos constants and small general functions -
/views
: Contains the main Python code for each of the Lexos tools -
/managers
: Contains Python code for the large data management features -
/processors
: Contains modular logic for each of the Lexos tools -
/resources
: Contains files used to create rule sets for Scrubber -
/frontend
: Contains CSS, Javascript, and fonts -
/templates
: Contains the HTML template files for displaying each of the Lexos tools
Lexos uses Flask Blueprints to modularise its funcitons. Blueprints are implemented by importing the interface in application.py
as follows:
from lexos.interfaces.base_interface import base_view
from lexos.interfaces.bubble_viz_interface import viz_view
etc.
Each interface is then registered as a Blueprint:
app.register_blueprint(base_view)
app.register_blueprint(viz_view)
etc.
The interface then functions like a miniature app. To work, the interface file (e.g. interfaces/base_interface.py
) must have the following elements:
from flask import session, redirect, url_for, render_template, send_file, \
request, Blueprint
from lexos.helpers import constants as constants
from lexos.managers import utility, session_manager as session_manager
base_view = Blueprint('base', __name__)
Note that resources imported from other folders must be prefixed by lexos.
. The Blueprint name defined in the last line is then used in defining Flask routes.
@base_view.route("/", methods=["GET"])
def base():
# Do something...
Lexos employs a flat structure for its blueprints; all of the templates and static files for the interfaces are kept in the templates
and static
folders used by the main app, rather than allocating them to separate folders for each interface.
Lexos handles white space and Unicode pretty well, but, in case we encounter further challenges, the following may be useful.