config: Add Config.rt_client method.
This commit is contained in:
		
							parent
							
								
									8d3816a8fd
								
							
						
					
					
						commit
						f09f029fc4
					
				
					 4 changed files with 109 additions and 0 deletions
				
			
		| 
						 | 
					@ -15,11 +15,16 @@
 | 
				
			||||||
# along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | 
					# along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
 | 
					import urllib.parse as urlparse
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import requests.auth
 | 
				
			||||||
 | 
					import rt
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from pathlib import Path
 | 
					from pathlib import Path
 | 
				
			||||||
from typing import (
 | 
					from typing import (
 | 
				
			||||||
    NamedTuple,
 | 
					    NamedTuple,
 | 
				
			||||||
    Optional,
 | 
					    Optional,
 | 
				
			||||||
 | 
					    Type,
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class RTCredentials(NamedTuple):
 | 
					class RTCredentials(NamedTuple):
 | 
				
			||||||
| 
						 | 
					@ -73,3 +78,24 @@ class Config:
 | 
				
			||||||
            RTCredentials(auth='rt'),
 | 
					            RTCredentials(auth='rt'),
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        return RTCredentials._make(v0 or v1 or v2 for v0, v1, v2 in all_creds)
 | 
					        return RTCredentials._make(v0 or v1 or v2 for v0, v1, v2 in all_creds)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def rt_client(self,
 | 
				
			||||||
 | 
					                  credentials: RTCredentials=None,
 | 
				
			||||||
 | 
					                  client: Type[rt.Rt]=rt.Rt,
 | 
				
			||||||
 | 
					    ) -> Optional[rt.Rt]:
 | 
				
			||||||
 | 
					        if credentials is None:
 | 
				
			||||||
 | 
					            credentials = self.rt_credentials()
 | 
				
			||||||
 | 
					        if credentials.server is None:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        urlparts = urlparse.urlparse(credentials.server)
 | 
				
			||||||
 | 
					        rest_path = urlparts.path.rstrip('/') + '/REST/1.0/'
 | 
				
			||||||
 | 
					        url = urlparse.urlunparse(urlparts._replace(path=rest_path))
 | 
				
			||||||
 | 
					        if credentials.auth == 'basic':
 | 
				
			||||||
 | 
					            auth = requests.auth.HTTPBasicAuth(credentials.user, credentials.passwd)
 | 
				
			||||||
 | 
					            retval = client(url, http_auth=auth)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            retval = client(url, credentials.user, credentials.passwd)
 | 
				
			||||||
 | 
					        if retval.login():
 | 
				
			||||||
 | 
					            return retval
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										1
									
								
								setup.py
									
										
									
									
									
								
							
							
						
						
									
										1
									
								
								setup.py
									
										
									
									
									
								
							| 
						 | 
					@ -12,6 +12,7 @@ setup(
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    install_requires=[
 | 
					    install_requires=[
 | 
				
			||||||
        'beancount>=2.2',
 | 
					        'beancount>=2.2',
 | 
				
			||||||
 | 
					        'rt>=2.0',
 | 
				
			||||||
    ],
 | 
					    ],
 | 
				
			||||||
    setup_requires=[
 | 
					    setup_requires=[
 | 
				
			||||||
        'pytest-mypy',
 | 
					        'pytest-mypy',
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -16,6 +16,7 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import contextlib
 | 
					import contextlib
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
 | 
					import re
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import pytest
 | 
					import pytest
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -23,6 +24,8 @@ from . import testutil
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from conservancy_beancount import config as config_mod
 | 
					from conservancy_beancount import config as config_mod
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					RT_AUTH_METHODS = frozenset(['basic', 'gssapi', 'rt'])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
RT_ENV_KEYS = (
 | 
					RT_ENV_KEYS = (
 | 
				
			||||||
    'RTSERVER',
 | 
					    'RTSERVER',
 | 
				
			||||||
    'RTUSER',
 | 
					    'RTUSER',
 | 
				
			||||||
| 
						 | 
					@ -44,6 +47,13 @@ RT_FILE_CREDS = (
 | 
				
			||||||
    'basic',
 | 
					    'basic',
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					RT_GENERIC_CREDS = config_mod.RTCredentials(
 | 
				
			||||||
 | 
					    'https://example.org/genericrt',
 | 
				
			||||||
 | 
					    'genericuser',
 | 
				
			||||||
 | 
					    'generic password',
 | 
				
			||||||
 | 
					    None,
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@pytest.fixture
 | 
					@pytest.fixture
 | 
				
			||||||
def rt_environ():
 | 
					def rt_environ():
 | 
				
			||||||
    return dict(zip(RT_ENV_KEYS, RT_ENV_CREDS))
 | 
					    return dict(zip(RT_ENV_KEYS, RT_ENV_CREDS))
 | 
				
			||||||
| 
						 | 
					@ -109,3 +119,42 @@ def test_rt_credentials_from_all_sources_mixed(tmp_path):
 | 
				
			||||||
        config = config_mod.Config()
 | 
					        config = config_mod.Config()
 | 
				
			||||||
        rt_credentials = config.rt_credentials()
 | 
					        rt_credentials = config.rt_credentials()
 | 
				
			||||||
    assert rt_credentials == (server, 'mixedup', 'mixed up', 'rt')
 | 
					    assert rt_credentials == (server, 'mixedup', 'mixed up', 'rt')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def check_rt_client_url(credentials, client):
 | 
				
			||||||
 | 
					    pattern = '^{}/?$'.format(re.escape(credentials[0].rstrip('/') + '/REST/1.0'))
 | 
				
			||||||
 | 
					    assert re.match(pattern, client.url)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@pytest.mark.parametrize('authmethod', RT_AUTH_METHODS)
 | 
				
			||||||
 | 
					def test_rt_client(authmethod):
 | 
				
			||||||
 | 
					    rt_credentials = RT_GENERIC_CREDS._replace(auth=authmethod)
 | 
				
			||||||
 | 
					    config = config_mod.Config()
 | 
				
			||||||
 | 
					    rt_client = config.rt_client(rt_credentials, testutil.RTClient)
 | 
				
			||||||
 | 
					    check_rt_client_url(RT_GENERIC_CREDS, rt_client)
 | 
				
			||||||
 | 
					    assert rt_client.auth_method == ('HTTPBasicAuth' if authmethod == 'basic' else 'login')
 | 
				
			||||||
 | 
					    assert rt_client.last_login == (
 | 
				
			||||||
 | 
					        RT_GENERIC_CREDS.user,
 | 
				
			||||||
 | 
					        RT_GENERIC_CREDS.passwd,
 | 
				
			||||||
 | 
					        True,
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def test_default_rt_client(rt_environ):
 | 
				
			||||||
 | 
					    with update_environ(**rt_environ):
 | 
				
			||||||
 | 
					        config = config_mod.Config()
 | 
				
			||||||
 | 
					        rt_client = config.rt_client(client=testutil.RTClient)
 | 
				
			||||||
 | 
					    check_rt_client_url(RT_ENV_CREDS, rt_client)
 | 
				
			||||||
 | 
					    assert rt_client.last_login[:-1] == RT_ENV_CREDS[1:3]
 | 
				
			||||||
 | 
					    assert rt_client.last_login[-1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@pytest.mark.parametrize('authmethod', RT_AUTH_METHODS)
 | 
				
			||||||
 | 
					def test_rt_client_login_failure(authmethod):
 | 
				
			||||||
 | 
					    rt_credentials = RT_GENERIC_CREDS._replace(
 | 
				
			||||||
 | 
					        auth=authmethod,
 | 
				
			||||||
 | 
					        passwd='bad{}'.format(authmethod),
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    config = config_mod.Config()
 | 
				
			||||||
 | 
					    assert config.rt_client(rt_credentials, testutil.RTClient) is None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def test_no_rt_client_without_server():
 | 
				
			||||||
 | 
					    rt_credentials = RT_GENERIC_CREDS._replace(server=None, auth='rt')
 | 
				
			||||||
 | 
					    config = config_mod.Config()
 | 
				
			||||||
 | 
					    assert config.rt_client(rt_credentials, testutil.RTClient) is None
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -112,3 +112,36 @@ class TestConfig:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def repository_path(self):
 | 
					    def repository_path(self):
 | 
				
			||||||
        return self.repo_path
 | 
					        return self.repo_path
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class RTClient:
 | 
				
			||||||
 | 
					    def __init__(self,
 | 
				
			||||||
 | 
					                 url,
 | 
				
			||||||
 | 
					                 default_login=None,
 | 
				
			||||||
 | 
					                 default_password=None,
 | 
				
			||||||
 | 
					                 proxy=None,
 | 
				
			||||||
 | 
					                 default_queue='General',
 | 
				
			||||||
 | 
					                 skip_login=False,
 | 
				
			||||||
 | 
					                 verify_cert=True,
 | 
				
			||||||
 | 
					                 http_auth=None,
 | 
				
			||||||
 | 
					    ):
 | 
				
			||||||
 | 
					        self.url = url
 | 
				
			||||||
 | 
					        if http_auth is None:
 | 
				
			||||||
 | 
					            self.user = default_login
 | 
				
			||||||
 | 
					            self.password = default_password
 | 
				
			||||||
 | 
					            self.auth_method = 'login'
 | 
				
			||||||
 | 
					            self.login_result = skip_login or None
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            self.user = http_auth.username
 | 
				
			||||||
 | 
					            self.password = http_auth.password
 | 
				
			||||||
 | 
					            self.auth_method = type(http_auth).__name__
 | 
				
			||||||
 | 
					            self.login_result = True
 | 
				
			||||||
 | 
					        self.last_login = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def login(self, login=None, password=None):
 | 
				
			||||||
 | 
					        if login is None and password is None:
 | 
				
			||||||
 | 
					            login = self.user
 | 
				
			||||||
 | 
					            password = self.password
 | 
				
			||||||
 | 
					        self.login_result = bool(login and password and not password.startswith('bad'))
 | 
				
			||||||
 | 
					        self.last_login = (login, password, self.login_result)
 | 
				
			||||||
 | 
					        return self.login_result
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		
		Reference in a new issue