chorus: on JSON decode error, log resume of response (#43270)

This commit is contained in:
Benjamin Dauvergne 2020-05-25 16:10:40 +02:00
parent 24526c2943
commit 0fdf874e39
3 changed files with 106 additions and 12 deletions

View File

@ -18,6 +18,7 @@ import base64
import logging
from django.conf import settings
from django.utils.encoding import force_text
import requests
@ -94,16 +95,30 @@ def push_to_chorus(pdf_bytes: bytes, pdf_name: str):
'avecSignature': False,
}
response = requests.post(
get_flux_url(),
headers={
'cpro-account': token,
'authorization': 'Bearer ' + get_piste_access_token(),
'content-type': 'application/json;charset=utf-8',
'accept': 'application/json;charset=utf-8',
},
json=payload)
data = {}
logger.debug('invoice %s sent, reveiced %s', pdf_name, response.text)
return response.json()
try:
response = requests.post(
get_flux_url(),
headers={
'cpro-account': token,
'authorization': 'Bearer ' + get_piste_access_token(),
'content-type': 'application/json;charset=utf-8',
'accept': 'application/json;charset=utf-8',
},
json=payload)
except requests.RequestException as e:
logger.warning('invoice %s sent, failure: %s', pdf_name, e)
data['http.requests_error'] = str(e)
else:
data['http.response.status-code'] = response.status_code
try:
data.update(response.json())
logger.debug('invoice %s sent, reveiced %s', pdf_name, response.text)
except ValueError:
logger.warning('response from chorus is not JSON: %r', response.content[:1024])
data['http.response.headers'] = {
force_text(key, errors='replace'): force_text(value, errors='replace') for key, value in response.headers.items()
}
data['http.response.body'] = repr(response.content[:1024])
return data

78
tests/test_chorus.py Normal file
View File

@ -0,0 +1,78 @@
# barbacompta - invoicing for dummies
# Copyright (C) 2010-2020 Entr'ouvert
#
# This program is free software: you can redistribute it and/or modify it
# under the terms of the GNU Affero General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import pytest
import httmock
import requests
from eo_gestion.chorus import chorus
@pytest.fixture(autouse=True)
def chorus_setup(settings, monkeypatch):
settings.CHORUS_PLATFORM = 'qualif'
settings.CHORUS_TECH_USER_LOGIN = 'login'
settings.CHORUS_TECH_USER_PASSWORD = 'password'
monkeypatch.setattr(chorus, 'get_piste_access_token', lambda: 'token')
PDF_BYTES = b'1'
PDF_NAME = 'F202000001.pdf'
@httmock.urlmatch()
def chorus_ok(url, request):
return httmock.response(200, {'ok': 1})
@httmock.urlmatch()
def chorus_connection_error(url, request):
raise requests.ConnectionError('bouh')
@httmock.urlmatch()
def chorus_error_500(url, request):
return httmock.response(500, b'Pas content \xe9', headers={
'Header Pourri': 'Héhé'.encode('latin1'),
'Header-Ok': 'ok',
})
def test_push_to_chorus_ok():
with httmock.HTTMock(chorus_ok):
result = chorus.push_to_chorus(PDF_BYTES, PDF_NAME)
assert result == {'http.response.status-code': 200, 'ok': 1}
def test_push_to_chorus_connection_error():
with httmock.HTTMock(chorus_connection_error):
result = chorus.push_to_chorus(PDF_BYTES, PDF_NAME)
assert result == {'http.requests_error': 'bouh'}
def test_push_to_chorus_error_500():
with httmock.HTTMock(chorus_error_500):
result = chorus.push_to_chorus(PDF_BYTES, PDF_NAME)
assert result == {
'http.response.status-code': 500,
'http.response.body': "b'Pas content \\xe9'",
'http.response.headers': {
'Header Pourri': 'H<EFBFBD>h<EFBFBD>',
'Header-Ok': 'ok',
},
}

View File

@ -13,6 +13,7 @@ deps =
pytest-cov
pytest-django
pytest-freezegun
httmock
pylint
pylint-django
django-webtest