Fleet Manager for Polish Humanitarian Action.
This project is used by logisticians assigned to particular humanitarian missions associated with PAH, by local drivers willing to support the mission and by passengers helping around the world.
Its purpose is to provide a simple, robust and mobile-first web application that supports submitting drives taken by drivers and cryptographically verifying their authenticity with passengers without the need for an active internet connection. Submitted and verified drives are automatically sent back to database when network connection is reestablished (eg. near WiFi Hot-Spot).
Drives can also be inspected in detail and corrected by administrators in the admin panel. The admin panel also supports a way of exporting records to many popular file formats supported by any popular spreadsheet software.
If you're familiar with GitHub Flow and Django & Vue then you're most likely ready to pick something from the backlog. Use the following filters to find some interesting work:
The project itself is built using modern technologies and common web application patterns. We're trying to avoid any custom logic or homegrown solutions to already solved problems by using well tested and popular libraries/tools.
Application is split between two main directories - backend and frontend. Here's a high-level description of the project's structure:
-
backend
main purpose:
REST API for frontend codelanguage:
Python 3.8framework:
Django
-
frontend
main purpose:
Interactive interface for applicationlanguage:
JavaScriptframework:
React.js
You'll need docker and
docker-compose
installed on your computer in order to set up the stack.
You will also need git
, make
and a decent code editor.
NOTE:
if you're trying to installdocker-compose
withapt
on Linux then you might end up installing obsolete version! Consider usingpip
instead. Try getting one with version>= 1.25.4
.
- Get the source code. If you intend to send a Pull Request then please follow Github Flow.
# get the code:
git clone [email protected]:CodeForPoznan/pah-fm.git
# if you intend to send a patch then clone your fork:
git clone [email protected]:<your github username>/pah-fm.git
# i.e:
git clone [email protected]:arturtamborski/pah-fm.git
- Go to project directory:
cd pah-fm
- Start the project
# This command will download Docker images set up local stack.
make start
- Generate some random data for testing
make manage populate_database
- Open up http://localhost:8080/ and use the app! There's also admin page at http://localhost:8080/admin/.
At this point you'll be able to log in as driver or passenger, submit a drive, optionally verify it, view your previous drives, change language, manage drives from admin panel or export them to common data formats such as CSV.
First, get familiar with project's handy toolset - make
!
We wrapped many common actions in a Makefile
.
You can view them all by running make help
or just make
.
Here's a snippet of the output:
pah-fm$ make
Usage:
make <target>
Targets:
help Display this help
start Start all containers in background
stop Stop all containers
build Build backend & frontend containers
manage Use manage.py, i.e make manage CMD=collectstatic
build-backend Build backend container
build-frontend Build frontend container
...
As you can see, there are quite a lot of useful commands here. I advise you to read and get familiar with at least some of them from the top so you can work on the project more easily.
Here's a handy list with some problems and solutions based on above commands that you might encounter when developing project locally:
Q:
How to start or stop the project?
A:
make start
or make stop
Q:
How to view logs from all containers?
A:
make logs
Q:
My Pull Request build on Travis failed due to lint errors,
how can I fix it?
A:
make lint
and git add . && git commit && git push
Q:
How to generate some random data for testing?
A:
make manage populate_database
Q:
How to debug backend with PDB?
A:
Place import pdb; pdb.set_trace()
in code, save the file,
run make debug-backend
and interact with page to hit your breakpoint.
After debugging detach from shell by pressing CTRL + P
and CTRL + Q
.
Q:
How to debug frontend with Chrome?
A:
Place debugger;
in code, save the file, wait a second or two
until frontend rebuilds itself (you can check it with make logs
),
refresh page in browser, open developer tools (CTRL + F12
) and interact
with the page to hit a breakpoint.
If you have any more questions not described here then please ask us
on Slack (channel name is#pah
).
We have 2 default users who are always present - hello
and ola
,
but you can create a few more default by running make manage populate_database
.
This command will create driver
and passenger
and also few other
random users and basic entities (Cars, Projects).
Every user that's randomly created has the same password -pass123
.
Here's a table with basic users + the default users created when DB is populated:
username | password | Vue app access | Django Admin access |
---|---|---|---|
[email protected] | pass123 | no | yes |
[email protected] | pass123 | no | no |
...after populate_database ... |
... | ... | ... |
[email protected] | pass123 | yes | no |
[email protected] | pass123 | yes | no |
REST API is built with Django Rest Framework. That package has also autogenerated interactive documentation for its endpoints. You can take a look by going to localhost:8080/api/docs/ or dev.pahfm.codeforpoznan.pl/api/docs/.
NOTE:
documentation is available only to logged-in users because of a DjangoRestFramework quirk, so be sure to log in to Admin panel beforehand.
If your editor does not support hooking up to Docker containers for python environments then you'll probably have to create python virtual environment yourself in order to have auto completion features.
NOTE:
The way it's supposed to be configured varies wildly between editors and toolchains so don't rely too much on this guide. Try finding documentation on configuring python environment for your particular editor, or use the links below:
instructions for VS Code.
instructions for PyCharm
instructions for Sublime Text
The general flow for configuring venv looks like so:
cd pah-fm
python3 -m venv ./backend/.venv
source ./backend/.venv/bin/activate
pip3 install -r requirements/dev.txt -r requirements/base.txt
This will create brand new environment safely separated from your
system libraries where all project dependencies are installed
(more precisely here: backend/.venv/lib/python3.X/site-packages
).
You can use them to check what is available in sources,
and add it to your IDE for module resolution / auto completion.
Application is deployed via CI/CD scripts (.travis.yml
and deploy.sh
)
as AWS Lambda Function to
dev.pahfm.codeforpoznan.pl.
It's built and deployed after every successful merge
to master branch. New Django migrations are also applied
automatically after a successful deployment.
You can log in there and test the application but you'll
need to ask for a new account first on
Slack (channel name is#pah
).
Backend code is a classic Django project with one app
called fleet_management
- that's the core of backend code.
Backend exposes two URLs:
/admin/
- Django Admin Panel/api/docs/
- Interactive API docs
and a few RESTful endpoints:
-
/api/api-token-auth
POST
- send credentials, get login token
-
/api/users/me
GET
- fetch info about me
-
/api/passengers
GET
- fetch list of Users who belong to Passenger group
-
/api/cars
GET
- fetch list of cars available to me
-
/api/projects
GET
- fetch list of projects available to me
-
/api/drives
GET
- fetch list of drives submitted by mePOST
- submit a new drive
These endpoints are created and managed with the help of Django Rest Framework and are tested by unit tests.
Each of those endpoints is basically a wrapper around corresponding model defined in models.py.
-
User
This model represents admins, drivers and passengers.
We use groups to differentiate between these user roles. User can belong to Driver group, Passenger group or (in rare cases) both. User model has username and email but they are considered the same for our use.
We also store country, date of last request for user activity tracking and three fields used for drive signing and verification. User with empty country field becomes a global user and is visible for selection in every country. -
Drive
A record representing Driver's completed journey with Passenger from one place to another.
The Drive records are created by filling data inDriveFormView.vue
and then sending it to/api/drives/
. Web form enforces some constraints on the Driver. One needs to select an existing entity from other models (Car
,Passenger
,Project
) and type in some details about the drive itself, such as start & end location and reading from odometer.
Data from this model is also made available for export in admin panel for later inspection or settlements. -
Project
Representation of a Project under which the Drive takes place. We store title, description and country, all of which are required fields. -
Car
Representation of a Driver's Car used during a Drive. We're storing it's car plates, some short description, fuel consumption in liters and country (current location). Only car plates and country are required.
Every User has three fields used for classic RSA message verification - e, d and modulus n. Our message is a string containing serialized comma-separated values from the DriveForm.
Drive signing takes place on frontend and later on is verified again on backend to compare outputs.
Here's a list of steps that happen during a drive verification:
- Drive Form is filled with valid data by the Driver
- Driver submits the form
- DriveVerifyView takes these values and joins them together creating a checksum (message).
- Checksum is displayed on form
- Driver communicates the Checksum to the Passenger
- Passenger inputs the Checksum into his own form on his phone
- Passenger clicks the button to sign the Message with his RSA keys
- Passenger's RSA private key is used for signing the Message
- Signed message is shown back on Passenger's phone
- Passenger communicates back the Signature (signed message) to the Driver
- Driver types in the Signature into his form
- Driver submits the form
- Endpoint receives the data, checks for Signature
- Backend uses selected Passenger's RSA keypair and signs the form again
- Backend compares the local signature with the sent one
- Backend decides whether or not the signature is equal thus valid and sets appropriate flag
- Backend completes saving the new Drive.
Everything before 13th point happens with no use of internet connection. The process can continue from 13th point when connection is established in some undefined time from when the form was submitted.
This process can also be started again after executing 13th point with no affect on any previous drive. This allows the user to be offline virtually for how long he pleases to be, or until he finally decides to synchronize his Drives back with backend.
Again, it's a classic example of Vue's directory layout. We're trying to optimize the app for size due to the business requirements and reality of use cases.
The app can be installed on any mobile phone in the form of a PWA (Progressive Web App) which behaves like a real app but in reality is just a dedicated Chrome window, bit like similarly to electron on desktop.
Not until recently the frontend code became complex enough that we started to write unit tests for some components, though there's still a lot to improve.
Everything that can be seen is presented on several views:
-
HomeView
Home page. Visible to everyone, even if not logged in. -
LoginView
Login form. Visible only to not logged in users. -
SuccessFullLogoutView
Shows simple information upon logout. Visible only to logged in users. -
PassengerSubmitView
Checksum form. Visible only to the Passenger. -
DriveFormView
Shows the main form for submitting Drives.
Visible only to the Driver. -
DrivesView
Shows the list of submitted Drives (including synchronized and not). Visible only to the Driver. -
DriveVerifyView
Form used for next stage of Drive submission, lets us type in the Passenger's checksum.
Visible only to the Driver. -
DriveVerifyView
Form used for next stage of Drive submission, lets us type in the Passenger's checksum.
Visible only to the Driver.
(Please note that this is in progress).
Due to the fact that we don't want to introduce API versioning because we don't have enough resources / manpower to manage it we decided to shift the responsibility of sending right data to frontend by utilising frontend migrations on vuex store.
It's a rather unique solution which might seem overcomplicated
for the scale of our app but the decision was made in order
to decrease costs of future releases and to allow for easier
testing setup (testing store is trivial).
Additionally, due to the core business point that we use very
heavily for decision making - the fact that users are mostly
offline and only come up online once in a while it makes
sense that we provide runnable pieces of code that will
alter the data of old, unsynced drives to possibly newer
format of an endpoint before sending it out tho backend instead
of supporting possibly very outdated versions of API
for very small number of people who couldn't sync yet.
Live instance of this application is running on dedicated server, here, but the staging environment is running on AWS Lambda service here. We're planning to migrate the production environment to lambda through that will require some non-trivial preparation.
Local environment is set up using docker and docker-compose which also relies on DockerHub for pulling the latest images from repository.
If you have any more questions / comments or if some parts of this
applications are still difficult to understand then please ask us
on Slack (channel name is#pah
).