First commit
This commit is contained in:
commit
53ccb57208
42
.gitignore
vendored
Normal file
42
.gitignore
vendored
Normal file
@ -0,0 +1,42 @@
|
||||
# Project-specific
|
||||
conf
|
||||
mfapi_tokens.json
|
||||
|
||||
# Byte-compiled / optimized / DLL files
|
||||
__pycache__/
|
||||
*.py[cod]
|
||||
|
||||
# Translations
|
||||
*.mo
|
||||
*.pot
|
||||
|
||||
# pyenv
|
||||
.python-version
|
||||
|
||||
# Environments
|
||||
.env
|
||||
.venv
|
||||
env/
|
||||
venv/
|
||||
ENV/
|
||||
env.bak/
|
||||
venv.bak/
|
||||
|
||||
## Editors:
|
||||
*.bak
|
||||
*.sav
|
||||
*.backup
|
||||
.*.swp
|
||||
*~
|
||||
|
||||
# Secret configuration
|
||||
settings.cfg
|
||||
settings.py
|
||||
*.sqlite
|
||||
|
||||
# Personal conf
|
||||
static/jquery.js
|
||||
static/socket.io.js
|
||||
templates/head-perso.html
|
||||
templates/scripts-perso.html
|
||||
|
54
README.md
Normal file
54
README.md
Normal file
@ -0,0 +1,54 @@
|
||||
# Easy access to Meteo-France API
|
||||
|
||||
It currently covers downloading of:
|
||||
|
||||
- Hourly climatotlogical data for one station
|
||||
- BRA (Bulletin d'Estimation du risque d'avalanche)
|
||||
|
||||
## 0. Get an account on Meteo-France API
|
||||
|
||||
Have a look to https://portail-api.meteofrance.fr/web/fr/faq !
|
||||
|
||||
## 1.Install
|
||||
```bash
|
||||
pip install git+https://git.vln-glr.fr/leo.viallon/mfapi.git
|
||||
```
|
||||
|
||||
## 2.Configuration
|
||||
Create a file containing;
|
||||
|
||||
```ini
|
||||
[DEFAULT]
|
||||
application_id = the-application-id-you-got-on-mf-api-website
|
||||
```
|
||||
|
||||
L'application id se trouve sur le site API de Météo-France en cliquant sur user, Mes API, puis Générer Token. Il vous sera affiché une commande curl de la forme suivante:
|
||||
``curl -k -X POST https://portail-api.meteofrance.fr/token -d "grant_type=client_credentials" -H "Authorization: Basic APPLICATION_ID"``
|
||||
|
||||
## 3.Examples
|
||||
|
||||
### 3.1 The BRA
|
||||
|
||||
```python
|
||||
import mfapi
|
||||
c = mfapi.MFAPIClient(config_file='path_to_your_config_file')
|
||||
c.get_bra(23, format='pdf') # 23 is the massif code of Mercantour (Alps : 1-23, Pyrenees: 64-74, Corsica: 40-41)
|
||||
```
|
||||
The file is stored as `BRA_23.pdf`.
|
||||
|
||||
You can also get XML format (and images needed to render with):
|
||||
|
||||
```python
|
||||
c.get_bra(23, format='xml', images=True)
|
||||
```
|
||||
|
||||
### 3.2 Climatological data
|
||||
|
||||
```python
|
||||
import mfapi
|
||||
import datetime
|
||||
c = mfapi.MFAPIClient(config_file='path_to_your_config_file')
|
||||
c.get_clim_h('65413400', begin=datetime.datetime(2024, 9, 1), end=datetime.datetime(2024, 12, 1)) # You first have to know the Station id
|
||||
```
|
||||
|
||||
The data is stored under csv format.
|
4
mfapi/__init__.py
Normal file
4
mfapi/__init__.py
Normal file
@ -0,0 +1,4 @@
|
||||
#!/usr/bin/env python3
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
from mfapi.client import MFAPIClient
|
230
mfapi/client.py
Normal file
230
mfapi/client.py
Normal file
@ -0,0 +1,230 @@
|
||||
#!/usr/bin/env python3
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
"""
|
||||
This file is largely inspired from the example provided on
|
||||
https://portail-api.meteofrance.fr/web/fr/faq
|
||||
|
||||
It allow to connect to Météo-France API with oauth2.
|
||||
"""
|
||||
|
||||
import json
|
||||
import time
|
||||
import os
|
||||
import logging
|
||||
import configparser
|
||||
import datetime
|
||||
|
||||
import requests
|
||||
|
||||
DEFAULT_EXPIRATION_TIME = 3590 # Default validity of the JWT token
|
||||
|
||||
# url to obtain acces token
|
||||
TOKEN_URL = "https://portail-api.meteofrance.fr/token"
|
||||
URL_BRA = "https://public-api.meteofrance.fr/public/DPBRA/v1/massif/BRA?id-massif={id_massif:d}&format={format}"
|
||||
URL_BRA_IMAGES = {
|
||||
"montagne_risques_{id_massif}.png": "https://public-api.meteofrance.fr/public/DPBRA/v1/massif/image/montagne-risques?id-massif={id_massif:d}",
|
||||
"rose_pentes_{id_massif}.png": "https://public-api.meteofrance.fr/public/DPBRA/v1/massif/image/rose-pentes?id-massif={id_massif:d}",
|
||||
"montagne_enneigement_{id_massif}.png": "https://public-api.meteofrance.fr/public/DPBRA/v1/massif/image/?id-massif={id_massif:d}",
|
||||
"graphe_neige_fraiche_{id_massif}.png": "https://public-api.meteofrance.fr/public/DPBRA/v1/massif/image/graphe-neige-fraiche?id-massif={id_massif:d}",
|
||||
"apercu_meteo_{id_massif}.png": "https://public-api.meteofrance.fr/public/DPBRA/v1/massif/image/apercu-meteo?id-massif={id_massif:d}",
|
||||
"sept_derniers_jours_{id_massif}.png": "https://public-api.meteofrance.fr/public/DPBRA/v1/massif/image/sept-derniers-jours?id-massif={id_massif:d}",
|
||||
}
|
||||
URL_CLIM_H = "https://public-api.meteofrance.fr/public/DPClim/v1/commande-station/horaire"
|
||||
URL_CLIM_H_DATA = 'https://public-api.meteofrance.fr/public/DPClim/v1/commande/fichier'
|
||||
|
||||
|
||||
class MFAPIClient(object):
|
||||
def __init__(self, config_file='~/.mfapi', token_file='./mfapi_tokens.json'):
|
||||
"""
|
||||
Client object to provide transparent authentication.
|
||||
|
||||
The application_id have to be provided in a configuration file
|
||||
(default is ~/.mfapi).
|
||||
The config file is a ini-like format.
|
||||
Keys are:
|
||||
|
||||
- application_id for your personal unique application id that can
|
||||
be found in the curl's command to generate the JWT token
|
||||
- default_expiration_time : the default JWT expiration time
|
||||
|
||||
:param config_file: Path to the configuration file
|
||||
:param token_file: Path to the temporary file to store token.
|
||||
"""
|
||||
self.session = requests.Session()
|
||||
self.token_file = token_file
|
||||
|
||||
if not os.path.isfile(config_file):
|
||||
logging.critical(f'Config file {config_file} not found.')
|
||||
raise ValueError(f'Config file {config_file} not found.')
|
||||
|
||||
config = configparser.ConfigParser()
|
||||
config.read(config_file)
|
||||
self.application_id = config.get('DEFAULT', 'application_id')
|
||||
self.default_expire = config.getint('DEFAULT', 'default_expiration_time',
|
||||
fallback=DEFAULT_EXPIRATION_TIME)
|
||||
|
||||
self.token = None
|
||||
self.token_expire = time.time()
|
||||
|
||||
if os.path.isfile(self.token_file):
|
||||
try:
|
||||
with open(self.token_file, 'r') as ff:
|
||||
token_json = json.load(ff)
|
||||
if 'token' in token_json:
|
||||
self.token = token_json['token']
|
||||
if 'token_expire' in token_json:
|
||||
self.token_expire = int(token_json['token_expire'])
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
def request(self, method, url, **kwargs):
|
||||
# If token is supposed to be expired or does not exist yet
|
||||
if self.token is None or time.time() > self.token_expire:
|
||||
self._obtain_token()
|
||||
|
||||
# Optimistically attempt to dispatch reqest
|
||||
if 'headers' not in kwargs:
|
||||
kwargs['headers'] = {}
|
||||
kwargs['headers']['Authorization'] = 'Bearer %s' % self.token
|
||||
response = self.session.request(method, url, **kwargs)
|
||||
if self._token_has_expired(response):
|
||||
self._obtain_token()
|
||||
response = self.session.request(method, url, **kwargs)
|
||||
|
||||
return response
|
||||
|
||||
def get(self, url, **kwargs):
|
||||
"""
|
||||
The function to interrogate an API endpoint.
|
||||
|
||||
:param url: The URL to interrogate
|
||||
:trturns: raw requests Response from API
|
||||
"""
|
||||
return self.request('GET', url, **kwargs)
|
||||
|
||||
def _token_has_expired(self, response):
|
||||
status = response.status_code
|
||||
content_type = response.headers['Content-Type']
|
||||
if status == 401 and 'application/json' in content_type:
|
||||
repJson = response.json()
|
||||
if 'description' in repJson and 'Invalid JWT token' in repJson['description']:
|
||||
return True
|
||||
return False
|
||||
|
||||
def _obtain_token(self):
|
||||
"""
|
||||
Obtain a new token and store it into self.token_file
|
||||
"""
|
||||
data = {'grant_type': 'client_credentials'}
|
||||
headers = {'Authorization': 'Basic ' + self.application_id}
|
||||
access_token_response = requests.post(TOKEN_URL, data=data,
|
||||
allow_redirects=False,
|
||||
headers=headers)
|
||||
self.token = access_token_response.json()['access_token']
|
||||
self.token_expire = time.time() + self.default_expire
|
||||
with open(self.token_file, 'w') as ff:
|
||||
json.dump({'token': self.token, 'token_expire': self.token_expire},
|
||||
ff)
|
||||
|
||||
def get_bra(self, id_massif: int,
|
||||
format: str = 'xml',
|
||||
dest_filename: str | None = None,
|
||||
images: bool = False) -> str | None:
|
||||
"""
|
||||
Get BRA (Bulletin d'estimation du risque d'avlanche).
|
||||
|
||||
You have to subscribe to this API first.
|
||||
|
||||
:param id_massif: Massif number (int)
|
||||
:param format: xml or pdf
|
||||
:param dest_filename: A custom output filename (default is BRA_{id_massif}).
|
||||
:param images: Download images also (for XML display, only if format=xml)
|
||||
:returns: The created filename or None if not found
|
||||
"""
|
||||
if format not in ['xml', 'pdf']:
|
||||
raise ValueError('Format should be either pdf or xml')
|
||||
if dest_filename is None:
|
||||
dest_filename = f"BRA_{id_massif}.{format}"
|
||||
|
||||
r = self.get(URL_BRA.format(id_massif=id_massif, format=format),
|
||||
headers={'Accept': '*/*'})
|
||||
s = r.status_code
|
||||
|
||||
if s == 404:
|
||||
return
|
||||
|
||||
with open(dest_filename, 'wb') as ff:
|
||||
ff.write(r.content)
|
||||
|
||||
if images and format == 'xml':
|
||||
for filename, url in URL_BRA_IMAGES.items():
|
||||
filename = filename.format(id_massif=id_massif)
|
||||
r = self.get(url.format(id_massif=id_massif),
|
||||
headers={'Accept': '*/*'})
|
||||
if r.status_code == 404:
|
||||
continue
|
||||
else:
|
||||
with open(filename, 'wb') as ff:
|
||||
ff.write(r.content)
|
||||
return dest_filename
|
||||
|
||||
def get_clim_h(self, id_station: str,
|
||||
begin: datetime.datetime,
|
||||
end: datetime.datetime | None = None,
|
||||
filename: str = None) -> str | None:
|
||||
"""
|
||||
Get hourly data from one station
|
||||
|
||||
:param id_station: The station ID (string, 9 numeric chars)
|
||||
:param begin: begin date, supposed to be UTC.
|
||||
:param end: end date (default to now)
|
||||
:param filename: output filename (default is id_station.csv).
|
||||
:returns: Output filename if succeded else None.
|
||||
Return the raw csv if '-' is provided as filename.
|
||||
"""
|
||||
if filename is None:
|
||||
filename = f'{id_station}.csv'
|
||||
begin = begin.isoformat(sep='T', timespec='seconds') + 'Z'
|
||||
if end is None:
|
||||
end = datetime.datetime.utcnow()
|
||||
end = datetime.datetime(end.year, end.month, end.day, end.hour)
|
||||
end = end.isoformat(sep='T', timespec='seconds') + 'Z'
|
||||
|
||||
data = {
|
||||
'id-station': id_station,
|
||||
'date-deb-periode': begin,
|
||||
'date-fin-periode': end,
|
||||
}
|
||||
|
||||
# Send request
|
||||
r = self.get(URL_CLIM_H, params=data)
|
||||
content_type = r.headers['Content-Type']
|
||||
if r.status_code == 202 and 'application/json' in content_type:
|
||||
j = r.json()
|
||||
if 'elaboreProduitAvecDemandeResponse' in j and 'return' in j['elaboreProduitAvecDemandeResponse']:
|
||||
id_cmde = j['elaboreProduitAvecDemandeResponse']['return']
|
||||
else:
|
||||
logging.error(f'Incorrect JSON answer: {j}')
|
||||
return None
|
||||
else:
|
||||
logging.error(f'Error in request. Answer: {r.text}')
|
||||
return None
|
||||
|
||||
# Get data
|
||||
rcode = 204
|
||||
while rcode == 204:
|
||||
r = self.get(URL_CLIM_H_DATA, params={'id-cmde': id_cmde})
|
||||
rcode = r.status_code
|
||||
if rcode == 201:
|
||||
if filename == '-':
|
||||
return r.text
|
||||
else:
|
||||
with open(filename, 'w') as ff:
|
||||
ff.write(r.text)
|
||||
break
|
||||
elif rcode != 204:
|
||||
logging.error('Error while downloading the data: {r.text}')
|
||||
break
|
||||
logging.info('Waiting for the data...')
|
||||
time.sleep(10)
|
15
pyproject.toml
Normal file
15
pyproject.toml
Normal file
@ -0,0 +1,15 @@
|
||||
[build-system]
|
||||
requires = ["setuptools >= 61.0"]
|
||||
build-backend = "setuptools.build_meta"
|
||||
|
||||
[project]
|
||||
name = "mfapi"
|
||||
version = "0.1"
|
||||
requires-python = ">=3.8"
|
||||
dependencies = ['requests']
|
||||
description = "Easy access to Meteo-France API"
|
||||
readme = "README.md"
|
||||
license = "CECILL-C"
|
||||
authors = [
|
||||
{name = "LVG"},
|
||||
]
|
Loading…
x
Reference in New Issue
Block a user