This repository provides a frontend client to consume the Market Access API. It's built with python django.
The Market Access frontend was originally a node project: https://github.com/uktrade/market-access-frontend
It was converted into Django from December 2019 to around February 2020. As a result, most of the css, javascript and html markup were copied directly across.
Market Access Python Frontend uses Docker compose to setup and run all the necessary components.
The docker-compose.yml file provided is meant to be used for running tests and development.
- Install
docker
&docker compose
- https://docs.docker.com/install/ - Add the following to your
hosts
file:# Market Access Frontend Client (Python) 127.0.0.1 market-access.local # Mock SSO 127.0.0.1 mocksso
- Clone the repository:
git clone https://github.com/uktrade/market-access-python-frontend.git cd market-access-python-frontend
- Copy the env file -
cp docker-compose.local-template.env docker-compose.env
Please note that as of Jan 2020 you will need to run the containers from https://github.com/uktrade/market-access-api/ first
as they currently share some dependencies (this only applies for local development).
- Build the images and spin up the containers by running -
docker-compose up --build
- Set up git hooks by running -
make git-hooks
- Enter bash within the django container using
docker-compose exec web bash
then create a superuserpy3 manage.py createsuperuser --email [email protected]
- Whilst still in the container
run npm install
thenexit
the container - To start the dev server run -
make django-run
- The fronted client is now accessible via http://market-access.local:9880
- run
make dev
- this will run the relevant gulp tasks (compile & watch CSS and JS files) and launch BrowserSync
When you visit the site via http://127.0.0.1:9881 BrowserSync will automatically reload the page when you modify scss or js files.
The installation steps above will require 2 terminal windows to be open to run the processes. If desired this can be reduced to 0 via the following commands:
- Start the containers in detached mode -
docker-compose up -d
- Start django in detached mode -
make django-run-detached
- The frontend client is now accessible via http://market-access.local:9880
Now even if you closed your terminal, the server would be still running.
There's a set of make commands that you can utilize straight away.
To list all available commands with help text type make help
in terminal and hit Enter
.
The project is using CSS and fonts from govuk-frontend
npm package.
Resources for GOV.UK Frontend:
- https://frontend.design-system.service.gov.uk/#gov-uk-frontend
- https://github.com/alphagov/govuk-frontend/tree/master/src/govuk
Fonts are copied while css is imported from node modules.
To prepare staticfiles for local run make dev
Staticfiles are compressed offline for most environments, so it makes sense that you could run the same way locally to test things out. To do that, just:
- stop the django development server (if it's running)
- set
DEBUG
toFalse
inconfig/settings/local.py
- run
npm run build
for a one off run (ormake dev
if you want to recompile css and js real time when changes are saved) - run
make django-static
- start the django development server
Note: this is a good way to mimic how files are generated and served in an environment,
but please note, lazy loading of static files is also disabled in offline mode, so your changes to templates, js, scss
might not take effect unless you run step 3 from above and restart your dev server.
To keep watching and recompiling css and js file use make dev
from step 3.
Builds can be initiated from Jenkins or from the command line using cf
CLI tool (using cf push <app_name>
).
To use cf push
you will need to be in the root of the project.
The preferred way to deploy apps remains Jenkins as of now because Jenkins will set environment variables as part of the flow.
Tasks that should be run at app initialisation can be defined in .profile
file.
If you would like to check the output of that you can do so via cf logs <app_name> --recent
, but
please note that these logs get trimmed so it's best to check straight after deployment.
Front end tests are grouped under ./test
directory. When writing tests please use the corresponding app name to keep the same folder structure as the main app so it's easy to tell which test belongs to which app.
The project's testrunner is pytest - https://docs.pytest.org/en/latest/
- You can run all or a subset of tests via
make django-test
, if you pass in a value inpath
then it will run that subset of tests. Example usage.:make django-test
- run all testsmake django-test path=barriers
- run a subset of tests just for the barriers appmake django-test path=assessments/test_assessment_detail.py::EmptyAssessmentDetailTestCase::test_view
- run a specific test case
- To run tests with coverage use
make django-test-coverage
- this will output the report to the console.
Playwright documentation - https://playwright.dev/python/docs/api/class-playwright
The end-to-end frontend tests reside in the test_frontend directory and are designed to operate independently of the rest of the application. This autonomy is facilitated through a local pytest.ini configuration file located within the same directory. The pytest.ini file configures specific parameters and settings essential for the execution of these tests, ensuring they can run in a self-contained environment. For detailed customization options and further information on pytest configuration files, refer to the pytest configuration docs
If you are running the docker build
-
Ensure the API is running & the frondent service is runing and can be access on
http://localhost:{frontend_port}
or http://host.docker.internal:9880 if runing within the docker container -
Ensure the frontend server is up and has reached the point where the Django development server is running.
By default the tests DO NOT RUN in headless mode, to activate headless mode the variable --is-headless will be required.
-
Run the tests:
make test-end-to-end target_url=<target-url>
e.g target_url:http://localhost:9880/
orhttps://market-access.uat.uktrade.digital/
-
To run a specific suite of frontend tests, specify the desired module:
make test-end-to-end target_url=<target-url> target=test_examples.py
To run headless:
make test-end-to-end target_url=http://localhost:9880/ is-headless=true
module structure
frontend_test/
├── .gitignore # Specifies intentionally untracked files to ignore
├── requirements.txt # Project dependencies
├── conftest.py # the pytest config file (the most important file to get things going)
├── README.md # The top-level README for developers using this project
└── pytest.ini # Configuration file for pytest
└── test_file.py # one test file for a specific end to end functionality
...
Inside the conftest.py
file the functions below would be required to be implemented as shown.
# conftest.py
BASE_URL = os.getenv("BASE_FRONTEND_TESTING_URL", "http://testserver/")
HEADLESS = os.getenv("TEST_HEADLESS", "false").lower() == "true"
@pytest.fixture(scope="session")
def session_data():
"""Return a dictionary to store session data."""
return {
"cookies": None,
"barrier_id": None,
}
@pytest.fixture(scope="session")
def playwright_instance():
"""Return a Playwright instance."""
with sync_playwright() as p:
yield p
@pytest.fixture(scope="session")
def browser(playwright_instance):
"""Return a browser instance."""
if HEADLESS:
browser = playwright_instance.chromium.launch(headless=True)
else:
browser = playwright_instance.chromium.launch(slow_mo=100, headless=HEADLESS)
yield browser
browser.close()
@pytest.fixture(scope="session")
def context(browser, session_data):
# Create a new browser context
context = browser.new_context()
context.set_default_timeout(0)
# Initially, session_data["cookies"] will be None.
# Check if "cookies" key exists and has a value; if not, it means it's the first test run.
if session_data.get("cookies") is None:
# Since it's the first run, let the browser context initiate and capture the cookies.
session_data["cookies"] = context.cookies()
else:
# If it's not the first run, load the initially captured cookies into the context.
context.add_cookies(session_data["cookies"])
yield context
context.close()
@pytest.fixture(scope="session")
def page(context):
# Create a new page in the provided context
_page = context.new_page()
_page.goto(BASE_URL, wait_until="domcontentloaded")
# Wait for the page to load
_page.wait_for_timeout(10000)
yield _page
# the rest of the test fixures
# ...
now we can easily use the page fixture in our tests like you would in playwright examples here
# test_file.py
def test_home_page(page):
...
def test_dashboard(page):
...
def test_fill_form(page):
...
Testing code coverage is automatically ran as part of the CircleCI build and sent to codecov.io. You can run the tests locally and generate a coverage report by running:
With docker:
docker compose run --rm web pytest tests --cov-report term
Or for local builds:
poetry run pytest tests --cov-report term