|
|
|
|
|
# Copyright (C) 2010-2023 RhodeCode GmbH
|
|
|
#
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
# it under the terms of the GNU Affero General Public License, version 3
|
|
|
# (only), as published by the Free Software Foundation.
|
|
|
#
|
|
|
# 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 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/>.
|
|
|
#
|
|
|
# This program is dual-licensed. If you wish to learn more about the
|
|
|
# RhodeCode Enterprise Edition, including its added features, Support services,
|
|
|
# and proprietary license terms, please see https://rhodecode.com/licenses/
|
|
|
|
|
|
import logging
|
|
|
|
|
|
import mock
|
|
|
import msgpack
|
|
|
import pytest
|
|
|
|
|
|
from rhodecode.lib import vcs
|
|
|
from rhodecode.lib.vcs import client_http, exceptions
|
|
|
|
|
|
|
|
|
def is_new_connection(logger, level, message):
|
|
|
return (
|
|
|
logger == 'requests.packages.urllib3.connectionpool' and
|
|
|
message.startswith('Starting new HTTP'))
|
|
|
|
|
|
|
|
|
@pytest.fixture()
|
|
|
def stub_session():
|
|
|
"""
|
|
|
Stub of `requests.Session()`.
|
|
|
"""
|
|
|
session = mock.Mock()
|
|
|
post = session.post()
|
|
|
post.content = msgpack.packb({})
|
|
|
post.status_code = 200
|
|
|
|
|
|
session.reset_mock()
|
|
|
return session
|
|
|
|
|
|
|
|
|
@pytest.fixture()
|
|
|
def stub_fail_session():
|
|
|
"""
|
|
|
Stub of `requests.Session()`.
|
|
|
"""
|
|
|
session = mock.Mock()
|
|
|
post = session.post()
|
|
|
post.content = msgpack.packb({'error': '500'})
|
|
|
post.status_code = 500
|
|
|
|
|
|
session.reset_mock()
|
|
|
return session
|
|
|
|
|
|
|
|
|
@pytest.fixture()
|
|
|
def stub_session_factory(stub_session):
|
|
|
"""
|
|
|
Stub of `rhodecode.lib.vcs.client_http.ThreadlocalSessionFactory`.
|
|
|
"""
|
|
|
session_factory = mock.Mock()
|
|
|
session_factory.return_value = stub_session
|
|
|
return session_factory
|
|
|
|
|
|
|
|
|
@pytest.fixture()
|
|
|
def stub_session_failing_factory(stub_fail_session):
|
|
|
"""
|
|
|
Stub of `rhodecode.lib.vcs.client_http.ThreadlocalSessionFactory`.
|
|
|
"""
|
|
|
session_factory = mock.Mock()
|
|
|
session_factory.return_value = stub_fail_session
|
|
|
return session_factory
|
|
|
|
|
|
|
|
|
def test_uses_persistent_http_connections(caplog, vcsbackend_hg):
|
|
|
repo = vcsbackend_hg.repo
|
|
|
remote_call = repo._remote.branches
|
|
|
|
|
|
with caplog.at_level(logging.INFO):
|
|
|
for x in range(5):
|
|
|
remote_call(normal=True, closed=False)
|
|
|
|
|
|
new_connections = [
|
|
|
r for r in caplog.record_tuples if is_new_connection(*r)]
|
|
|
assert len(new_connections) <= 1
|
|
|
|
|
|
|
|
|
def test_repo_maker_uses_session_for_classmethods(stub_session_factory):
|
|
|
repo_maker = client_http.RemoteVCSMaker(
|
|
|
'server_and_port', 'endpoint', 'test_dummy_scm', stub_session_factory)
|
|
|
repo_maker.example_call()
|
|
|
stub_session_factory().post.assert_called_with(
|
|
|
'http://server_and_port/endpoint', data=mock.ANY,
|
|
|
headers={'X-RC-Method': 'example_call', 'X-RC-Repo-Name': None})
|
|
|
|
|
|
|
|
|
def test_repo_maker_uses_session_for_instance_methods(
|
|
|
stub_session_factory, config):
|
|
|
repo_maker = client_http.RemoteVCSMaker(
|
|
|
'server_and_port', 'endpoint', 'test_dummy_scm', stub_session_factory)
|
|
|
repo = repo_maker('stub_path', 'stub_repo_id', config)
|
|
|
repo.example_call()
|
|
|
stub_session_factory().post.assert_called_with(
|
|
|
'http://server_and_port/endpoint', data=mock.ANY,
|
|
|
headers={'X-RC-Method': 'example_call', 'X-RC-Repo-Name': 'stub_path'})
|
|
|
|
|
|
|
|
|
@mock.patch('rhodecode.lib.vcs.client_http.ThreadlocalSessionFactory')
|
|
|
@mock.patch('rhodecode.lib.vcs.connection')
|
|
|
def test_connect_passes_in_the_same_session(
|
|
|
connection, session_factory_class, stub_session):
|
|
|
session_factory = session_factory_class.return_value
|
|
|
session_factory.return_value = stub_session
|
|
|
|
|
|
vcs.connect_http('server_and_port')
|
|
|
|
|
|
|
|
|
def test_repo_maker_uses_session_that_throws_error(
|
|
|
stub_session_failing_factory, config):
|
|
|
repo_maker = client_http.RemoteVCSMaker(
|
|
|
'server_and_port', 'endpoint', 'test_dummy_scm', stub_session_failing_factory)
|
|
|
repo = repo_maker('stub_path', 'stub_repo_id', config)
|
|
|
|
|
|
with pytest.raises(exceptions.HttpVCSCommunicationError):
|
|
|
repo.example_call()
|
|
|
|