Repository URL to install this package:
|
Version:
4.36.0 ▾
|
import sys
import requests
from base64 import encodebytes
import json
import braintree
from braintree import version
from braintree.environment import Environment
from braintree.util.xml_util import XmlUtil
from braintree.exceptions.authentication_error import AuthenticationError
from braintree.exceptions.authorization_error import AuthorizationError
from braintree.exceptions.gateway_timeout_error import GatewayTimeoutError
from braintree.exceptions.http.connection_error import ConnectionError
from braintree.exceptions.http.invalid_response_error import InvalidResponseError
from braintree.exceptions.http.timeout_error import ConnectTimeoutError
from braintree.exceptions.http.timeout_error import ReadTimeoutError
from braintree.exceptions.http.timeout_error import TimeoutError
from braintree.exceptions.not_found_error import NotFoundError
from braintree.exceptions.request_timeout_error import RequestTimeoutError
from braintree.exceptions.server_error import ServerError
from braintree.exceptions.service_unavailable_error import ServiceUnavailableError
from braintree.exceptions.too_many_requests_error import TooManyRequestsError
from braintree.exceptions.unexpected_error import UnexpectedError
from braintree.exceptions.upgrade_required_error import UpgradeRequiredError
class Http(object):
class ContentType(object):
Xml = "application/xml"
Multipart = "multipart/form-data"
Json = "application/json"
@staticmethod
def is_error_status(status):
return status not in [200, 201, 204, 422]
@staticmethod
def raise_exception_from_status(status, message=None):
if status == 401:
raise AuthenticationError()
elif status == 403:
raise AuthorizationError(message)
elif status == 404:
raise NotFoundError()
elif status == 408:
raise RequestTimeoutError()
elif status == 426:
raise UpgradeRequiredError()
elif status == 429:
raise TooManyRequestsError()
elif status == 500:
raise ServerError()
elif status == 503:
raise ServiceUnavailableError()
elif status == 504:
raise GatewayTimeoutError()
else:
raise UnexpectedError("Unexpected HTTP_RESPONSE " + str(status))
def __init__(self, config, environment=None):
self.config = config
self.environment = environment or self.config.environment
def post(self, path, params=None):
return self._make_request("POST", path, Http.ContentType.Xml, params)
def delete(self, path):
return self._make_request("DELETE", path, Http.ContentType.Xml)
def get(self, path):
return self._make_request("GET", path, Http.ContentType.Xml)
def put(self, path, params=None):
return self._make_request("PUT", path, Http.ContentType.Xml, params)
def post_multipart(self, path, files, params=None):
return self._make_request("POST", path, Http.ContentType.Multipart, params, files)
def _make_request(self, http_verb, path, content_type, params=None, files=None, header_overrides=None):
http_strategy = self.config.http_strategy()
headers = self.__headers(content_type, header_overrides)
request_body = self.__request_body(content_type, params, files)
full_path = self.__full_path(path)
try:
status, response_body = http_strategy.http_do(http_verb, full_path, headers, request_body)
except Exception as e:
if self.config.wrap_http_exceptions:
http_strategy.handle_exception(e)
else:
raise
if Http.is_error_status(status):
Http.raise_exception_from_status(status)
else:
if len(response_body.strip()) == 0:
return {}
else:
if content_type == Http.ContentType.Json:
return json.loads(response_body)
else:
return XmlUtil.dict_from_xml(response_body)
def http_do(self, http_verb, path, headers, request_body):
data = request_body
files = None
full_path = self.__full_path(path)
if type(request_body) is tuple:
data = request_body[0]
files = request_body[1]
if (self.config.environment == Environment.Development):
verify = False
else:
verify = self.environment.ssl_certificate
with requests.Session() as session:
request = requests.Request(
method=http_verb,
url=full_path,
headers=headers,
data=data,
files=files)
prepared_request = request.prepare()
prepared_request.url = full_path
# there's a bug in requests module that requires we manually update proxy settings,
# see https://github.com/psf/requests/issues/5677
session.proxies.update(requests.utils.getproxies())
response = session.send(prepared_request,
verify=verify,
timeout=self.config.timeout)
return [response.status_code, response.text]
def handle_exception(self, exception):
if isinstance(exception, requests.exceptions.ReadTimeout):
raise ReadTimeoutError(exception)
elif isinstance(exception, requests.exceptions.ConnectTimeout):
raise ConnectTimeoutError(exception)
elif isinstance(exception, requests.exceptions.ConnectionError):
raise ConnectionError(exception)
elif isinstance(exception, requests.exceptions.HTTPError):
raise InvalidResponseError(exception)
elif isinstance(exception, requests.exceptions.Timeout):
raise TimeoutError(exception)
else:
raise UnexpectedError(exception)
def __authorization_header(self):
if self.config.has_client_credentials():
return b"Basic " + encodebytes(
self.config.client_id.encode('ascii') +
b":" +
self.config.client_secret.encode('ascii')
).replace(b"\n", b"").strip()
elif self.config.has_access_token():
return b"Bearer " + self.config.access_token.encode('ascii')
else:
return b"Basic " + encodebytes(
self.config.public_key.encode('ascii') +
b":" +
self.config.private_key.encode('ascii')
).replace(b"\n", b"").strip()
def __headers(self, content_type, header_overrides=None):
headers = {
"Accept": "application/xml",
"Authorization": self.__authorization_header(),
"User-Agent": "Braintree Python " + version.Version,
"Accept-Encoding": "gzip",
"X-ApiVersion": braintree.configuration.Configuration.api_version()
}
if content_type == Http.ContentType.Xml:
headers["Content-type"] = Http.ContentType.Xml
headers.update(header_overrides or {})
return headers
def __request_body(self, content_type, params, files):
if content_type == Http.ContentType.Xml:
request_body = XmlUtil.xml_from_dict(params) if params else ''
return request_body
elif files == None:
return params
else:
return (params, files)
def __full_path(self, path):
return path if path.startswith(self.config.base_url()) or path.startswith(self.config.graphql_base_url()) else (self.config.base_url() + path)