Created
September 21, 2022 12:11
-
-
Save machacekondra/be1007b3a99e41a23d997d43e09b3ab3 to your computer and use it in GitHub Desktop.
SAPauth plugin for requests
This file contains hidden or 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
| class SAPAuth(AuthBase): | |
| """ | |
| SAPAuth is class which implements the SAP launchpad authentication. | |
| """ | |
| sso_url = 'https://accounts.sap.com/saml2/idp/sso' | |
| def get_next_url(self, text): | |
| return ET.fromstring(text, parser=etree.HTMLParser()).find(".//body//form[@method='post']").attrib['action'] | |
| def get_all_inputs(self, text): | |
| ret = {} | |
| for i in ET.fromstring(text, parser=etree.HTMLParser()).findall(".//body//form[@method='post']//input"): | |
| if i.attrib.get('name') and i.attrib.get('value'): | |
| ret[i.attrib['name']] = i.attrib['value'] | |
| return ret | |
| def __init__(self, username=None, password=None): | |
| self._username = username | |
| self._password = password | |
| def _next_step(self, response, history, next_url=None, headers=None, **kwargs): | |
| if next_url is None: | |
| next_url = self.get_next_url(response.text) | |
| post_data = self.get_all_inputs(response.text) | |
| for k, v in kwargs.items(): | |
| post_data[k] = v | |
| cookies = dict() | |
| for r in history: | |
| cookies.update(dict(r.cookies.items())) | |
| next_response = requests.post( | |
| next_url, | |
| data=post_data, | |
| cookies=cookies, | |
| headers=headers, | |
| ) | |
| history.append(next_response) | |
| return next_response | |
| def handle_response(self, response, **kwargs): | |
| history = [response] | |
| response = self._next_step(response, history) | |
| response = self._next_step(response, history, j_username=self._username) | |
| # We need to pass the next_url explicitly, because the response only contains relative URL for some reason: | |
| response = self._next_step(response, history, next_url=self.sso_url, j_password=self._password) | |
| response = self._next_step(response, history) | |
| return self._next_step(response, history, headers=self._headers) | |
| def __call__(self, request): | |
| request.register_hook('response', self.handle_response) | |
| self._headers = request.headers | |
| return request |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment