Skip to content

Commit c1425b0

Browse files
author
Kevin Le
committed
advisory endpoint
0 parents  commit c1425b0

File tree

7 files changed

+263
-0
lines changed

7 files changed

+263
-0
lines changed

.gitignore

+134
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,134 @@
1+
api.log
2+
3+
# Byte-compiled / optimized / DLL files
4+
__pycache__/
5+
*.py[cod]
6+
*$py.class
7+
8+
# C extensions
9+
*.so
10+
11+
# Distribution / packaging
12+
.Python
13+
build/
14+
develop-eggs/
15+
dist/
16+
downloads/
17+
eggs/
18+
.eggs/
19+
lib/
20+
lib64/
21+
parts/
22+
sdist/
23+
var/
24+
wheels/
25+
pip-wheel-metadata/
26+
share/python-wheels/
27+
*.egg-info/
28+
.installed.cfg
29+
*.egg
30+
MANIFEST
31+
32+
# PyInstaller
33+
# Usually these files are written by a python script from a template
34+
# before PyInstaller builds the exe, so as to inject date/other infos into it.
35+
*.manifest
36+
*.spec
37+
38+
# Installer logs
39+
pip-log.txt
40+
pip-delete-this-directory.txt
41+
42+
# Unit test / coverage reports
43+
htmlcov/
44+
.tox/
45+
.nox/
46+
.coverage
47+
.coverage.*
48+
.cache
49+
nosetests.xml
50+
coverage.xml
51+
*.cover
52+
*.py,cover
53+
.hypothesis/
54+
.pytest_cache/
55+
56+
# Translations
57+
*.mo
58+
*.pot
59+
60+
# Django stuff:
61+
*.log
62+
local_settings.py
63+
db.sqlite3
64+
db.sqlite3-journal
65+
66+
# Flask stuff:
67+
instance/
68+
.webassets-cache
69+
70+
# Scrapy stuff:
71+
.scrapy
72+
73+
# Sphinx documentation
74+
docs/_build/
75+
76+
# PyBuilder
77+
target/
78+
79+
# Jupyter Notebook
80+
.ipynb_checkpoints
81+
82+
# IPython
83+
profile_default/
84+
ipython_config.py
85+
86+
# pyenv
87+
.python-version
88+
89+
# pipenv
90+
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
91+
# However, in case of collaboration, if having platform-specific dependencies or dependencies
92+
# having no cross-platform support, pipenv may install dependencies that don't work, or not
93+
# install all needed dependencies.
94+
#Pipfile.lock
95+
96+
# PEP 582; used by e.g. github.com/David-OConnor/pyflow
97+
__pypackages__/
98+
99+
# Celery stuff
100+
celerybeat-schedule
101+
celerybeat.pid
102+
103+
# SageMath parsed files
104+
*.sage.py
105+
106+
# Environments
107+
.env
108+
.venv
109+
env/
110+
venv/
111+
ENV/
112+
env.bak/
113+
venv.bak/
114+
115+
# Spyder project settings
116+
.spyderproject
117+
.spyproject
118+
119+
# Rope project settings
120+
.ropeproject
121+
122+
# mkdocs documentation
123+
/site
124+
125+
# mypy
126+
.mypy_cache/
127+
.dmypy.json
128+
dmypy.json
129+
130+
# Pyre type checker
131+
.pyre/
132+
133+
.aws-sam/
134+
.idea/

README.md

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
# Smartraveller API
2+
3+
A simple API for travel advisories published by the Australian Government on [Smartraveller](https://www.smartraveller.gov.au/).
4+
5+
**This is not an official API. Visit the Smartraveller website for latest information**
6+
7+
## Locally running
8+
```
9+
pip install -r requirements.txt
10+
python wsgi.py
11+
```
12+
13+
## Deployment
14+
15+
Available at https://smartraveller.api.kevle.xyz/
16+
17+
OR
18+
19+
[![Deploy to Vercel](https://camo.githubusercontent.com/f209ca5cc3af7dd930b6bfc55b3d7b6a5fde1aff/68747470733a2f2f76657263656c2e636f6d2f627574746f6e)](https://vercel.com/import/project?template=https://github.com/kevinle-1/smartraveller-api)
20+
21+
## Endpoint(s)
22+
23+
### GET /advisory
24+
25+
Fetches an advice summary for a country.
26+
27+
Responses may be cached for up to 1 hour.
28+
29+
#### Query Parameters
30+
31+
- country (required)
32+
- Note: Fuzzy matches for country - E.g. "Spain" or "ES" is accepted. However it is recommend to use Alpha 3 country codes.
33+
34+
#### Example Response
35+
36+
```
37+
{
38+
"advisory": "Exercise normal safety precautions",
39+
"level": 1,
40+
"country": "Spain",
41+
"alpha_2": "ES",
42+
"official_name": "Kingdom of Spain"
43+
}
44+
```
45+
46+
## Copyright
47+
48+
Smartraveller material accessed is provided under the latest [Creative Commons Attribution licence](creativecommons.org/licenses/by/4.0/).

api/api.py

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
import json
2+
import pycountry
3+
from flask import Flask, request, Response
4+
from api.smartraveller.advisory import get_overall_advisory
5+
import logging
6+
7+
app = Flask(__name__, static_url_path='')
8+
@app.route('/')
9+
def index():
10+
return '<h1>Smartraveller API</h1> \
11+
<p>Smartraveller material accessed is provided under the latest Creative Commons Attribution licence.<p/> \
12+
<a href="https://www.smartraveller.gov.au/">Smartraveller</a>'
13+
14+
@app.route('/advisory')
15+
def advisory():
16+
country_query = request.args.get('country')
17+
if country_query:
18+
try:
19+
country = pycountry.countries.lookup(country_query)
20+
except Exception as e:
21+
logging.error(e)
22+
country = None
23+
24+
if country is not None:
25+
advisory = get_overall_advisory()
26+
27+
country_query = country.name.lower().replace(" ", "-")
28+
logging.debug(f"Querying for country: {country_query}")
29+
30+
advisory["country"] = country.name
31+
advisory["alpha_2"] = country.alpha_2
32+
advisory["official_name"] = country.official_name
33+
34+
response = Response(json.dumps(advisory))
35+
response.headers['Cache-Control'] = 's-maxage=3600' # Vercel cache for 1 hour
36+
response.headers['Content-Type'] = 'application/json'
37+
38+
return response
39+
else:
40+
return "Invalid country", 404
41+
else:
42+
return "Missing country query parameter", 400

api/smartraveller/advisory.py

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
import requests
2+
import time
3+
from bs4 import BeautifulSoup
4+
5+
advisory_level ={
6+
"Exercise normal safety precautions": 1,
7+
"Exercise a high degree of caution": 2,
8+
"Reconsider your need to travel": 3,
9+
"Do not travel": 4
10+
}
11+
12+
def get_overall_advisory() -> dict:
13+
country = "austria"
14+
15+
html = requests.get(f'https://www.smartraveller.gov.au/destinations/{country}').content
16+
site = BeautifulSoup(html, 'html.parser')
17+
18+
advisory_block = site.findAll('div', { 'class': 'views-field views-field-field-overall-advice-level'})
19+
advisory = advisory_block[0].find('strong').getText()
20+
21+
return {
22+
"advisory": advisory,
23+
"level": advisory_level[advisory]
24+
}
25+
26+
if __name__ == "__main__":
27+
print(get_overall_advisory())

requirements.txt

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
requests==2.28.2
2+
Flask==2.2.3
3+
beautifulsoup4==4.11.2

vercel.json

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
{
2+
"rewrites": [
3+
{ "source": "/(.*)", "destination": "/api/api" }
4+
]
5+
}

wsgi.py

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
from api.api import app
2+
3+
if __name__ == '__main__':
4+
app.run(host='0.0.0.0')

0 commit comments

Comments
 (0)