generated from ministryofjustice/template-repository
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #4 from ministryofjustice/unit-tests
Unit tests
- Loading branch information
Showing
14 changed files
with
379 additions
and
74 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -9,6 +9,6 @@ env/ | |
*.sha256 | ||
terraform.tfstate | ||
|
||
test_data/**.csv | ||
data_src/**.csv | ||
./idea/* | ||
**/__pycache__/ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,64 +1,23 @@ | ||
# Ministry of Justice Template Repository | ||
# Juniper mist integration script | ||
|
||
[![repo standards badge](https://img.shields.io/endpoint?labelColor=231f20&color=005ea5&style=for-the-badge&label=MoJ%20Compliant&url=https%3A%2F%2Foperations-engineering-reports.cloud-platform.service.justice.gov.uk%2Fapi%2Fv1%2Fcompliant_public_repositories%2Fendpoint%2Ftemplate-repository&logo=)](https://operations-engineering-reports.cloud-platform.service.justice.gov.uk/public-report/template-repository) | ||
This repo has been developed by the DevOps Lan&Wi-Fi to automate site creation on juniper mist. | ||
|
||
This template repository equips you with the default initial files required for a Ministry of Justice GitHub repository. | ||
## Run script | ||
|
||
## Included Files | ||
## Development setup | ||
|
||
The repository comes with the following preset files: | ||
### Prerequisites | ||
|
||
- LICENSE | ||
- .gitignore | ||
- CODEOWNERS | ||
- dependabot.yml | ||
- GitHub Actions example file | ||
- Ministry of Justice Compliance Badge (Public repositories only) | ||
- Docker | ||
- IDE that integrates with docker (We use IntelliJ in this example) | ||
|
||
## Setup Instructions | ||
### Setup | ||
|
||
Once you've created your repository using this template, ensure the following steps: | ||
- Run `make build` | ||
- Integrate built docker container with IDE. [here](https://www.jetbrains.com/help/idea/configuring-remote-python-sdks.html#2546d02c) is the example for intelliJ | ||
- mark src directory & test directory within the IDE. [here](https://www.jetbrains.com/help/idea/content-roots.html) | ||
|
||
### Update README | ||
# Notes | ||
|
||
Edit this README.md file to document your project accurately. Take the time to create a clear, engaging, and informative README.md file. Include information like what your project does, how to install and run it, how to contribute, and any other pertinent details. | ||
|
||
### Update repository description | ||
|
||
After you've created your repository, GitHub provides a brief description field that appears on the top of your repository's main page. This is a summary that gives visitors quick insight into the project. Using this field to provide a succinct overview of your repository is highly recommended. | ||
|
||
This description and your README.md will be one of the first things people see when they visit your repository. It's a good place to make a strong, concise first impression. Remember, this is often visible in search results on GitHub and search engines, so it's also an opportunity to help people discover your project. | ||
|
||
### Grant Team Permissions | ||
|
||
Assign permissions to the appropriate Ministry of Justice teams. Ensure at least one team is granted Admin permissions. Whenever possible, assign permissions to teams rather than individual users. | ||
|
||
### Read about the GitHub repository standards | ||
|
||
Familiarise yourself with the Ministry of Justice GitHub Repository Standards. These standards ensure consistency, maintainability, and best practices across all our repositories. | ||
|
||
You can find the standards [here](https://operations-engineering.service.justice.gov.uk/documentation/services/repository-standards.html). | ||
|
||
Please read and understand these standards thoroughly and enable them when you feel comfortable. | ||
|
||
### Modify the GitHub Standards Badge | ||
|
||
Once you've ensured that all the [GitHub Repository Standards](https://operations-engineering.service.justice.gov.uk/documentation/services/repository-standards.html) have been applied to your repository, it's time to update the Ministry of Justice (MoJ) Compliance Badge located in the README file. | ||
|
||
The badge demonstrates that your repository is compliant with MoJ's standards. Please follow these [instructions](https://operations-engineering.service.justice.gov.uk/documentation/runbooks/services/add-repo-badge.html) to modify the badge URL to reflect the status of your repository correctly. | ||
|
||
**Please note** the badge will not function correctly if your repository is internal or private. In this case, you may remove the badge from your README. | ||
|
||
### Manage Outside Collaborators | ||
|
||
To add an Outside Collaborator to the repository, follow the guidelines detailed [here](https://github.com/ministryofjustice/github-collaborators). | ||
|
||
### Update CODEOWNERS | ||
|
||
(Optional) Modify the CODEOWNERS file to specify the teams or users authorized to approve pull requests. | ||
|
||
### Configure Dependabot | ||
|
||
Adapt the dependabot.yml file to match your project's [dependency manager](https://docs.github.com/en/code-security/dependabot/dependabot-version-updates/configuration-options-for-the-dependabot.yml-file#package-ecosystem) and to enable [automated pull requests for package updates](https://docs.github.com/en/code-security/supply-chain-security). | ||
|
||
If your repository is private with no GitHub Advanced Security license, remove the .github/workflows/dependency-review.yml file. | ||
- To see options run `make help` | ||
- If you update [requirements.txt](src/requirements.txt), you will need to rebuild the docker container. |
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
FROM python:3.11.5 | ||
|
||
# Set the working directory to /app | ||
WORKDIR /app | ||
|
||
# Copy the current directory contents into the container at /app | ||
COPY ../src/requirements.txt /tmp/requirements.txt | ||
|
||
# Copy entrypoint | ||
COPY docker/entrypoint.sh /entrypoint.sh | ||
COPY src/ /app/src | ||
|
||
# Make entrypoint executable | ||
RUN chmod +x /entrypoint.sh | ||
|
||
# Install any needed packages specified in requirements.txt | ||
RUN pip install --no-cache-dir -r ../tmp/requirements.txt | ||
|
||
# Set the src directory | ||
ENV PYTHONPATH=/app/src | ||
|
||
# Run tests when the container launches | ||
ENTRYPOINT ["/entrypoint.sh"] | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
#!/bin/bash | ||
|
||
if [ "$RUN_UNIT_TESTS" = True ] ; then | ||
python -m unittest discover -s /app -t /app | ||
else | ||
python /app/src/main.py | ||
fi |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
#!make | ||
.DEFAULT_GOAL := help | ||
SHELL := '/bin/bash' | ||
|
||
.PHONY: build | ||
build: ## Build the docker container | ||
docker build -t juniper-mist -f docker/Dockerfile . | ||
|
||
.PHONO: create-dir | ||
make create-dir: ## Creates a directory for end user to put CSV file into | ||
mkdir data_src; | ||
echo "Please put csv file into data_src then run 'make-prod'"; | ||
|
||
.PHONY: run-prod | ||
run-prod: ## Run the python script only mounting the host for csv-file. Format: MIST_API_TOKEN=foo ORG_ID=bar make run-prod | ||
docker run -v $(shell pwd)/data_src:/data_src \ | ||
-e MIST_API_TOKEN=$$MIST_API_TOKEN \ | ||
-e ORG_ID=$$ORG_ID \ | ||
juniper-mist | ||
|
||
.PHONY: run-dev | ||
run-dev: ## Run the python script while mounting the host. This enables using the latest local src code without needing to wait for a container build. Format: MIST_API_TOKEN=foo ORG_ID=bar make run-dev | ||
docker run -v $(shell pwd)/src:/app/src \ | ||
-v $(shell pwd)/data_src:/data_src \ | ||
-e MIST_API_TOKEN=$$MIST_API_TOKEN \ | ||
-e ORG_ID=$$ORG_ID \ | ||
juniper-mist | ||
|
||
.PHONY: tests | ||
tests: ## Run unit tests for the python app | ||
docker run -v $(shell pwd)/src:/app/src \ | ||
-v $(shell pwd)/test:/app/test \ | ||
-e RUN_UNIT_TESTS=True juniper-mist | ||
|
||
.PHONY: shell | ||
shell: ## Make interactive docker container | ||
docker run -it --entrypoint /bin/bash \ | ||
-v $(shell pwd)/src:/app/src \ | ||
-v $(shell pwd)/test:/app/test \ | ||
-v $(shell pwd)/data_src:/data_src \ | ||
-e RUN_UNIT_TESTS=True juniper-mist | ||
|
||
help: | ||
@grep -h -E '^[a-zA-Z0-9_-]+:.*?## .*$$' $(MAKEFILE_LIST) | sort | awk 'BEGIN {FS = ":.*?## "}; {printf "\033[36m%-30s\033[0m %s\n", $$1, $$2}' |
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,3 @@ | ||
requests==2.21.0 | ||
requests==2.31.0 | ||
geopy==2.4.1 | ||
timezonefinder==6.2.0 |
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
import unittest | ||
from unittest.mock import patch, MagicMock | ||
from src.geocode import geocode, find_timezone, find_country_code | ||
|
||
|
||
class TestGeocode(unittest.TestCase): | ||
|
||
@patch('src.geocode.Nominatim.geocode') | ||
def test_given_list_of_valid_addresses_when_geocode_called_then_return_relevant_list_of_gps_locations(self, mock_nominatim): | ||
# Define a list of addresses and their expected results | ||
addresses = [ | ||
("40 Mayflower Dr, Plymouth PL2 3DG", (50.3868633, -4.1539256)), | ||
("102 Petty France, London SW1H 9AJ", | ||
(51.499929300000005, -0.13477761285315926)), | ||
("London", (51.4893335, -0.14405508452768728)), | ||
("Met Office, FitzRoy Road, Exeter, Devon, EX1 3PB", | ||
(50.727350349999995, -3.4744726127760086)) | ||
] | ||
# Mock the geocode method to return the corresponding latitude and longitude | ||
for address, (lat, lon) in addresses: | ||
mock_nominatim.return_value.latitude = lat | ||
mock_nominatim.return_value.longitude = lon | ||
|
||
# Call the geocode function for each address | ||
result = geocode(address) | ||
|
||
# Assert that the result matches the expected latitude and longitude | ||
self.assertEqual(result, (lat, lon)) | ||
|
||
@patch('src.geocode.Nominatim.geocode') | ||
def test_geocode_invalid_address(self, mock_geocode): | ||
# Arrange | ||
address = "Invalid Address" | ||
mock_geocode.return_value = None # Simulate geocode returning None | ||
|
||
# Act & Assert | ||
with self.assertRaises(AttributeError) as context: | ||
geocode(address) | ||
|
||
expected_error_message = 'geocode unable to find latitude & longitude for {address}'.format( | ||
address=address) | ||
self.assertEqual(str(context.exception), expected_error_message) | ||
|
||
@patch('src.geocode.TimezoneFinder') | ||
def test_find_timezone_valid_coordinates(self, mock_timezone_finder): | ||
tf_instance = MagicMock() | ||
tf_instance.timezone_at.return_value = 'America/New_York' | ||
mock_timezone_finder.return_value = tf_instance | ||
|
||
gps_coordinates = (40.7128, -74.0060) | ||
result = find_timezone(gps_coordinates) | ||
|
||
self.assertEqual(result, 'America/New_York') | ||
|
||
@patch('src.geocode.TimezoneFinder') | ||
def test_find_timezone_out_of_bounds(self, mock_timezone_finder): | ||
tf_instance = MagicMock() | ||
tf_instance.timezone_at.side_effect = ValueError( | ||
'The coordinates were out of bounds 40.7128:-74.0060') | ||
mock_timezone_finder.return_value = tf_instance | ||
|
||
gps_coordinates = (40.7128, -74.0060) | ||
|
||
with self.assertRaises(ValueError) as context: | ||
find_timezone(gps_coordinates) | ||
|
||
self.assertEqual(str(context.exception), | ||
'The coordinates were out of bounds 40.7128:-74.006') | ||
|
||
@patch('src.geocode.TimezoneFinder') | ||
def test_find_timezone_no_matching_timezone(self, mock_timezone_finder): | ||
tf_instance = MagicMock() | ||
tf_instance.timezone_at.return_value = None | ||
mock_timezone_finder.return_value = tf_instance | ||
|
||
gps_coordinates = (40.7128, -74.0060) | ||
|
||
with self.assertRaises(ValueError) as context: | ||
find_timezone(gps_coordinates) | ||
|
||
self.assertEqual(str(context.exception), | ||
'GPS coordinates did not match a time_zone') | ||
|
||
|
||
class TestFindCountryCode(unittest.TestCase): | ||
@patch('src.geocode.Nominatim.geocode') | ||
def test_find_country_code_valid_coordinates(self, mock_nominatim): | ||
geolocator_instance = MagicMock() | ||
location_instance = MagicMock() | ||
location_instance.raw = {'address': {'country_code': 'us'}} | ||
geolocator_instance.reverse.return_value = location_instance | ||
mock_nominatim.return_value = geolocator_instance | ||
|
||
gps_coordinates = (40.7128, -74.0060) | ||
result = find_country_code(gps_coordinates) | ||
|
||
self.assertEqual(result, 'US') | ||
|
||
@patch('src.geocode.Nominatim.geocode') | ||
def test_find_country_code_invalid_coordinates(self, mock_nominatim): | ||
geolocator_instance = MagicMock() | ||
geolocator_instance.reverse.side_effect = Exception( | ||
'Invalid coordinates') | ||
mock_nominatim.return_value = geolocator_instance | ||
|
||
gps_coordinates = (1000.0, 2000.0) # Invalid coordinates | ||
|
||
with self.assertRaises(Exception) as context: | ||
find_country_code(gps_coordinates) | ||
|
||
self.assertEqual(str(context.exception), | ||
'Must be a coordinate pair or Point') |
Oops, something went wrong.