Skip to content

Latest commit

 

History

History
455 lines (310 loc) · 16.7 KB

README.md

File metadata and controls

455 lines (310 loc) · 16.7 KB

Quick Start

Creating a virtual environment to isolate our package dependencies locally

It is suggested to have a dedicated virtual environment for each Django project, and one way to manage a virtual environment is venv, which is included in Python.

  • cd into its_backend directory, run the following command to install Python 3.11

    • pyenv install 3.11.4
    • If you do not have pyenv, follow the guide here to install it on your machine and set up your shell environment

  • Create a virtual environment to isolate our package dependencies locally.

    • ${pyenv root}/versions/3.11.4/bin/python -m venv --upgrade-deps --copies its_env

Install dependencies

  • activate the virtual environment by running

    • source its_env/bin/activate  # On Windows use env\Scripts\activate
  • install dependencies by running

    • pip install -r requirements.txt

Install redis

  • This project uses redis, so make sure you have it installed on your local machine for development

  • For MacOS:

    • brew install redis
  • For Windows

    • You can download a Redis installer for Windows from the Redis website or use the Microsoft archive.

Database Migration

  • Create migration files by running python manage.py makemigrations
  • Run migrations to database with python manage.py migrate
python manage.py makemigrations
python manage.py migrate

Set up third party login

  • Navigate to http://127.0.0.1:8000/admin
  • Modify the entry in Sites table to be
    • Domain name: http://127.0.0.1:8000/
    • Display name: Django Server (or anything else)

Google

  1. Navigate to Google Cloud Developer Console
  2. Go to 'Credentials' -> 'Create Credentials' -> 'OAuth client ID'
  3. Choose 'Web application' as Application Type
  4. Add the following URLS under 'Authorized JavaScript origins'
    • http://127.0.0.1:8000
    • http://localhost:3000
  5. Add the following URLS under 'Authorized redirect URIs'
    • http://127.0.0.1:8000/auth/google/login/callback/
    • http://127.0.0.1:8000/auth/social
  6. Click 'Create' at the bottom
  7. Note down the client ID and client secret values

To see a list of all OAuth Apps created: open the console left side menu and select APIs & services -> Credentials

GitHub

  1. Navigate to https://github.com/settings/applications/new
  2. Application Name: Intelligent Tutoring System
  3. Homepage URL: http://127.0.0.1:8000/
  4. Authorization callback URL: http://127.0.0.1:8000/auth/social/callback
  5. Click 'Register Application'
  6. Note down the client ID and client secret values

To see a list of all OAuth Apps created: settings -> <> Developer settings -> OAuth Apps

Register the Third Party OAuth Providers to Django

  1. Navigate to Django admin site (http://127.0.0.1:8000/admin) and log in as superuser
    • If you have not created a superuser, create one by running python manage.py createsuperuser
  2. Click on + Add button next to Social applications
  3. Choose the 'Provider' (either Google or GitHub)
  4. Enter the corresponding client ID and client secret generated before
  5. Move http://127.0.0.1:8000/ into the Chosen sites
  6. Click Save

Add More Third Party Login Providers

See allauth library's documentation to see all available third party OAuth providers. Follow the guideline and update the SOCIALACCOUNT_PROVIDERS dictionary in the settings.py file and the SocialApp table on Django admin site accordingly.

Start Application

Using Docker

  1. Install Docker into your system
  2. Run docker compose up --build backend to run the Django server
    • The server will be running on http://127.0.0.1:8000
  3. Alternatively, you can run docker compose up --build backend-test to run the Django server with the test entities populated into the database
  4. A superuser account with email=[email protected] and password=CS3213ITS is created for you to access the admin site when you run backend service.
  5. If you wish to start the service from a clean state, you can run the following command to remove any new data entries:
docker-compose down -v [SERVICE_NAMES] # remove containers and any associated volumes

See Dockerfile and Docker compose file

docker compose up --build backend       # start Django server
docker compose up --build backend-test  # start server with test data

Using shell scripts

If you have a terminal that can run bash, you can run the following

./start_project.sh    # run Django server
./run_test_server.sh  # run server with test data

Build Manually

The following steps are basically what happens inside the Docker compose file

  1. Migrate the database
    • Additionally, populate the database with test data
  2. Start the redis server
  3. Start the celery worker
  4. Start the Django server

Together, it looks like this:

python manage.py migrate
python manage.py loaddata test/test_data.json  # optional
redis-server --port 6379 &
celery -A its_backend worker -l info &
python manage.py runserver

Things to Note Before Testing by CS3213 TAs

If you start a fresh Django server (not in test environment) using any of the methods above, there will not be any prepopulated entities, including superusers and managers. To load sample data to test/assess our service, run the following before starting the Django server:

# Create a db.sqlite3 file
python manage.py migrate
# Load sample data for testing/tinkering (also loads manager)
python manage.py loaddata test/test_data.json
# Create superuser to access "/admin" site
python manage.py createsuperuser

# Start server
docker compose up --build backend

See this section to see the entities preloaded by test_data.json.

You can also manually add/delete entities from the database by navigating to the Django admin site (after starting the server). Just create a superuser using the aforementioned command to access the admin site.

View all urls created

run python manage.py show_urls

Swagger Documentation

After running the Django server, navigate to http://127.0.0.1:8000/swagger to see our API documentation.

Test APIs

Our project uses Postman collections and newman to test our Django APIs.

Run tests

Check that you have the newman package installed in your shell environment by running

$ newman --version
6.1.1

If your shell cannot find the newman package, download it using NPM or Homebrew:

npm install [-g] newman  # include -g flag to download globally
brew install newman

Alternatively, you can cd into test folder and install dependencies:

cd test
npm install

After the package is downloaded, run your Postman collection by running either the shell script:

chmod +x ./test/test.sh  # if necessary
./test/test.sh           # or ./test.sh if you cd-ed into test

Or the Python script:

python test/test.py      # or test.py if you cd-ed into test

A sample test Postman collection is already included under the test folder. If you want to test your own collection, add your collection under the test folder and update the POSTMAN_COLLECTION value inside the shell test script or python test script.

Add test cases to Postman Collection

To add test cases to the existing Postman collection, download the json file and import it to your local (or web) Postman application.

You will see that our Postman collection is structured hierarchically, with each collection and subcollection organized according to its respective tests. Here is an overview of how our tests are organized:

[01] Test collection 1
|--- [01] test request 1 - success
|--- [02] test request 2 - fail (reason for failure)
|--- ...
[02] Test collection 2
|--- [01] Test subcollection 1
     |--- [01] Test request 1 - status
     |--- [02] Test request 2 - status
     |--- ...
|--- [02] Test subcollection 2
     |--- ...
...

The [xx] numbering scheme in the names do not really matter as Postman does not inherently sort the collections by name, but it helps give a clear picture on the order of the tests. To add your own test cases, simply add a folder to the end and add test requests under your folder. Try to follow the naming scheme of our test collection.

See Postman's documentation on the basics of writing test scripts in Postman.

To test endpoints that require authorization (student, tutor, or manager), you can use the bearer tokens that are configured in the first test collection ([01] Get access tokens).

{{tutor_bearer}}
{{student_bearer}}

Add one of the above in your request's Authorization tab with type Bearer token.

To test the contents of received response, you can add a Javascript script under Tests tab. For example, the following script tests if the response is of status 200 OK and contains message key with value "Hello World".

// Parse the response body as JSON
var responseBody = pm.response.json();

// Check if the status code is 200 OK
pm.test("Status code is 200 OK", function () {
    pm.response.to.have.status(200);
});

// Check if the response contains the expected message
pm.test("Response message is correct", function () {
    pm.expect(responseBody.message).to.eql("Hello World");
});

You can also use Tests tab to set a new collection variable to be reused inside the Postman collection. For example, the following script stores "Hello World" into the variable message.

pm.collectionVariables.set("message", "Hello World");

Populate DB with test data

To populate the db.sqlite3 file with test data, run:

# Either command works
sqlite3 db.sqlite3 ".read ./test/populate_db.sql"
python manage.py loaddata ./test/test_data.json

Currently, the db.sqlite3 file is pre-populated with the following test data:

  • 10 students
  • 5 tutors
  • 1 manager
  • 2 Teaches relation
  • 6 questions
  • 13 test cases
  • 7 submissions

More teaches relations are added through Postman collection (see [02] Add teaches relations in our collection).

If you need to populate more test data to the DB (e.g., sample questions and submissions), write more INSERT INTO SQL statements inside the script.

If you want to save the current state of the db.sqlite3 file as test data to use in the future, run:

python manage.py dumpdata --exclude socialaccount --output ./test/test_data.json

Add test cases to Django Unit Test

The Postman Collection is unable to test the correctness of utils.py and its_utils.py files within the submission app using Postman Collection tests. These files are involved in an asynchronous process in the API call, rendering them unreachable by the test cases.

To add more testcases about utils.py and its_utils.py files within the submission app, add more functions to the tests_utils.py file under tests folder under the submission app.

Test Code Coverage

Our project's code coverage is tested using the Coverage.py library along with the Postman tests explained above. Similar to testing APIs, testing code coverage also requires your machine to have the newman package installed.

To test and generate reports on the code coverage of the Postman tests, you can run the following:

cd test
chmod +x coverage.sh
./coverage.sh

This will generate 3 artifacts:

Name Description
.coverage File containing the data of the last run coverage result
htmlcov/ Folder containing the coverage report in HTML format

Open index.html under this directory in your browser to see more in-depth code coverage analysis
coverage.json Coverage report in JSON format

These can be cleaned up by running:

chmod +x clean.sh  # ./test/clean.sh if not in test directory
./clean.sh         # ./test/clean.sh if not in test directory

We generate coverage reports for only the following files

  • its_backend/apps/**/models.py
  • its_backend/apps/**/views.py
  • its_backend/apps/submissions/its_utils.py
  • its_backend/apps/submissions/utils.py

because these are the only files related to the actual logic of the backend server.

We check for both statement and branch coverage. To only test for statement coverage, you can remove the --branch flag from the coverage run command inside the coverage script. Coverage.py checks for statement coverage by default, and there is no way to force it to test for only branch coverage as of now.

Ruff Linter

Our project uses ruff to lint our code and maintain a clean codebase.

Run the following to lint your whole code base:

ruff check .

Or the following to lint certain files or directories:

ruff check /path/to/file_or_folder

You can also select certain rules to test for:

ruff check --select F .

If there is no output, it means your codebase passes all lint checks.

To fix fixable lint errors, you can also run:

ruff check --fix .

Or to fix certain set of lint rules, run:

ruff check --select I --fix .  # fixes imports

You can also format files based on format rules defined in ruff.toml:

ruff format .                 # format all files in codebase
ruff format /path/to/file.py  # format a single file

Lint Rules used in ITS Backend

Currently, the lint rules used are as follows:

  • Pyflakes (F)
  • Subset of pycodestyle errors (E)
  • isort (I)
  • pep8-naming (N)
  • flake8-bugbear (B)
  • flake8-builtins (A)
  • Subset of flake8-django (DJ)
  • flake8-return (RET)

You can see the documentation of each here.

Configure ruff Behavior

ruff configuration is laid out inside the ruff.toml file.

To add more lint errors, you can add full rule codes (e.g., F401) or any valid prefix (e.g., F) to the end of the select list.

You can find out more configurations here.

Debug

  • set DEBUG=TRUE in settings.py

  • The typical usage to break into the debugger is to insert

    • import pdb; pdb.set_trace() or
    • breakpoint()

FAQs

  1. I followed all the instructions for third party login setup, but it is still not working. What is wrong?

    The allauth package creates a placeholder OAuth provider (example.com) by default, which may take primary key value of 1 in the Sites table. Try changing the value of the SITE_ID variable in settings.py to 1, 2, 3, and so on until it works. Or you can download a database browser (like DB Browser for SQLite) to see how the table is configured.

  2. Django is not creating migration files for one Django app X. What should I do?

    Sometimes Django might now detect the changes in individual apps, in that case, run:

    python manage.py makemigrations --empty [APP_NAME]
  3. When I try to migrate to database, Django throws the following:

    django.db.migrations.exceptions.InconsistentMigrationHistory: Migration X is applied before its dependency Y

    Refer to this Stack Overflow post for solution. We find that deleting conflicting migration file sometimes works as well.

  4. Can the application run without third-party authentication properly set up?

    Yes, the application will work, but as of now, our website will still have two nonfunctional buttons for Google and GitHub sign in.