diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 628c53504..542e3245c 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -46,11 +46,26 @@ jobs: strategy: matrix: variant: [ "integration-dind", "integration-dind-ssl", "integration-dind-ssh" ] + api-version: [1.44, 1.43, 1.42, 1.24] # 1.24 is the oldest supported version + engine-version: [25.0] # latest + include: # ugly repetition because GHA doesn't support lists in these objects so we need a separate entry for each variant + - engine-version: 23.0 # mirantis lts + api-version: 1.42 + variant: "integration-dind" + - engine-version: 23.0 # mirantis lts + api-version: 1.42 + variant: "integration-dind-ssl" + - engine-version: 23.0 # mirantis lts + api-version: 1.42 + variant: "integration-dind-ssh" steps: - uses: actions/checkout@v4 - - name: make ${{ matrix.variant }} + - name: ${{ matrix.variant }} - v${{ matrix.engine-version }}/v${{ matrix.api-version }} run: | docker logout rm -rf ~/.docker make ${{ matrix.variant }} + env: + TEST_ENGINE_VERSION: ${{ matrix.engine-version }} + TEST_API_VERSION: ${{ matrix.api-version }} diff --git a/Makefile b/Makefile index 00ebca05c..25a83205b 100644 --- a/Makefile +++ b/Makefile @@ -1,5 +1,5 @@ -TEST_API_VERSION ?= 1.43 -TEST_ENGINE_VERSION ?= 24.0 +TEST_API_VERSION ?= 1.44 +TEST_ENGINE_VERSION ?= 25.0 ifeq ($(OS),Windows_NT) PLATFORM := Windows diff --git a/tests/integration/api_container_test.py b/tests/integration/api_container_test.py index aa27fbfd7..e62576026 100644 --- a/tests/integration/api_container_test.py +++ b/tests/integration/api_container_test.py @@ -14,7 +14,7 @@ from ..helpers import ctrl_with from ..helpers import requires_api_version, skip_if_desktop from .base import BaseAPIIntegrationTest -from .base import TEST_IMG +from .base import TEST_IMG, TEST_API_VERSION from docker.constants import IS_WINDOWS_PLATFORM from docker.utils.socket import next_frame_header from docker.utils.socket import read_exactly @@ -1285,7 +1285,11 @@ def test_run_container_reading_socket_ws(self): self.client.start(container) data = pty_stdout.recv() - assert data.decode('utf-8') == line + + if TEST_API_VERSION < "1.25": + assert data == line + else: + assert data.decode('utf-8') == line @pytest.mark.timeout(10) def test_attach_no_stream(self): @@ -1294,7 +1298,10 @@ def test_attach_no_stream(self): ) self.tmp_containers.append(container) self.client.start(container) - self.client.wait(container, condition='not-running') + wait_kwargs = {} + if TEST_API_VERSION >= "1.30": + wait_kwargs['condition'] = 'not-running' + self.client.wait(container, **wait_kwargs) output = self.client.attach(container, stream=False, logs=True) assert output == 'hello\n'.encode(encoding='ascii') diff --git a/tests/integration/api_exec_test.py b/tests/integration/api_exec_test.py index 4d7748f5e..f0dfe53eb 100644 --- a/tests/integration/api_exec_test.py +++ b/tests/integration/api_exec_test.py @@ -2,7 +2,7 @@ from ..helpers import ctrl_with from ..helpers import requires_api_version from .base import BaseAPIIntegrationTest -from .base import TEST_IMG +from .base import TEST_IMG, TEST_API_VERSION from docker.utils.proxy import ProxyConfig from docker.utils.socket import next_frame_header from docker.utils.socket import read_exactly @@ -35,17 +35,19 @@ def test_execute_command_with_proxy_env(self): for item in expected: assert item in output - # Overwrite some variables with a custom environment - env = {'https_proxy': 'xxx', 'HTTPS_PROXY': 'XXX'} - - res = self.client.exec_create(container, cmd=cmd, environment=env) - output = self.client.exec_start(res).decode('utf-8').split('\n') - expected = [ - 'ftp_proxy=a', 'https_proxy=xxx', 'http_proxy=c', 'no_proxy=d', - 'FTP_PROXY=a', 'HTTPS_PROXY=XXX', 'HTTP_PROXY=c', 'NO_PROXY=d' - ] - for item in expected: - assert item in output + # Setting environment for exec is not supported in API < 1.25 + if TEST_API_VERSION > "1.24": + # Overwrite some variables with a custom environment + env = {'https_proxy': 'xxx', 'HTTPS_PROXY': 'XXX'} + + res = self.client.exec_create(container, cmd=cmd, environment=env) + output = self.client.exec_start(res).decode('utf-8').split('\n') + expected = [ + 'ftp_proxy=a', 'https_proxy=xxx', 'http_proxy=c', 'no_proxy=d', + 'FTP_PROXY=a', 'HTTPS_PROXY=XXX', 'HTTP_PROXY=c', 'NO_PROXY=d' + ] + for item in expected: + assert item in output def test_execute_command(self): container = self.client.create_container(TEST_IMG, 'cat', diff --git a/tests/integration/api_service_test.py b/tests/integration/api_service_test.py index dec3fa007..cbeeb7f3b 100644 --- a/tests/integration/api_service_test.py +++ b/tests/integration/api_service_test.py @@ -382,6 +382,7 @@ def test_create_service_with_custom_networks(self): {'Target': net1['Id']}, {'Target': net2['Id']} ] + @requires_api_version('1.25') def test_create_service_with_network_attachment_config(self): network = self.client.create_network( 'dockerpytest_1', driver='overlay', ipam={'Driver': 'default'} @@ -1108,6 +1109,7 @@ def test_update_service_with_defaults_update_config(self): assert update_config['Delay'] == uc['Delay'] assert update_config['FailureAction'] == uc['FailureAction'] + @requires_api_version('1.25') def test_update_service_with_defaults_networks(self): net1 = self.client.create_network( 'dockerpytest_1', driver='overlay', ipam={'Driver': 'default'} diff --git a/tests/integration/api_swarm_test.py b/tests/integration/api_swarm_test.py index b4125d24d..69c950567 100644 --- a/tests/integration/api_swarm_test.py +++ b/tests/integration/api_swarm_test.py @@ -27,8 +27,6 @@ def test_init_swarm_simple(self): @requires_api_version('1.24') def test_init_swarm_force_new_cluster(self): - pytest.skip('Test stalls the engine on 1.12.0') - assert self.init_swarm() version_1 = self.client.inspect_swarm()['Version']['Index'] assert self.client.init_swarm(force_new_cluster=True) diff --git a/tests/integration/models_containers_test.py b/tests/integration/models_containers_test.py index 219b9a4cb..cdf380011 100644 --- a/tests/integration/models_containers_test.py +++ b/tests/integration/models_containers_test.py @@ -104,6 +104,7 @@ def test_run_with_network(self): assert 'Networks' in attrs['NetworkSettings'] assert list(attrs['NetworkSettings']['Networks'].keys()) == [net_name] + @requires_api_version('1.32') def test_run_with_networking_config(self): net_name = random_name() client = docker.from_env(version=TEST_API_VERSION) @@ -132,11 +133,12 @@ def test_run_with_networking_config(self): assert 'NetworkSettings' in attrs assert 'Networks' in attrs['NetworkSettings'] assert list(attrs['NetworkSettings']['Networks'].keys()) == [net_name] - assert attrs['NetworkSettings']['Networks'][net_name]['Aliases'] == \ - test_aliases + for alias in test_aliases: + assert alias in attrs['NetworkSettings']['Networks'][net_name]['Aliases'] assert attrs['NetworkSettings']['Networks'][net_name]['DriverOpts'] \ == test_driver_opt + @requires_api_version('1.32') def test_run_with_networking_config_with_undeclared_network(self): net_name = random_name() client = docker.from_env(version=TEST_API_VERSION) @@ -165,6 +167,7 @@ def test_run_with_networking_config_with_undeclared_network(self): ) self.tmp_containers.append(container.id) + @requires_api_version('1.32') def test_run_with_networking_config_only_undeclared_network(self): net_name = random_name() client = docker.from_env(version=TEST_API_VERSION) @@ -190,9 +193,11 @@ def test_run_with_networking_config_only_undeclared_network(self): assert 'NetworkSettings' in attrs assert 'Networks' in attrs['NetworkSettings'] assert list(attrs['NetworkSettings']['Networks'].keys()) == [net_name] - assert attrs['NetworkSettings']['Networks'][net_name]['Aliases'] is None assert (attrs['NetworkSettings']['Networks'][net_name]['DriverOpts'] is None) + # Aliases should include the container's short-id (but it will be removed + # in API v1.45). + assert attrs['NetworkSettings']['Networks'][net_name]['Aliases'] == [attrs["Id"][:12]] def test_run_with_none_driver(self): client = docker.from_env(version=TEST_API_VERSION) diff --git a/tests/integration/models_services_test.py b/tests/integration/models_services_test.py index f1439a418..f017d88e3 100644 --- a/tests/integration/models_services_test.py +++ b/tests/integration/models_services_test.py @@ -23,26 +23,32 @@ def tearDownClass(cls): def test_create(self): client = docker.from_env(version=TEST_API_VERSION) name = helpers.random_name() - service = client.services.create( + create_kwargs = { # create arguments - name=name, - labels={'foo': 'bar'}, + 'name': name, + 'labels': {'foo': 'bar'}, # ContainerSpec arguments - image="alpine", - command="sleep 300", - container_labels={'container': 'label'}, - rollback_config={'order': 'start-first'} - ) + 'image': "alpine", + 'command': "sleep 300", + 'container_labels': {'container': 'label'} + } + if TEST_API_VERSION >= "1.28": + create_kwargs['rollback_config'] = {'order': 'start-first'} + + service = client.services.create(**create_kwargs) + assert service.name == name assert service.attrs['Spec']['Labels']['foo'] == 'bar' container_spec = service.attrs['Spec']['TaskTemplate']['ContainerSpec'] assert "alpine" in container_spec['Image'] assert container_spec['Labels'] == {'container': 'label'} - spec_rollback = service.attrs['Spec'].get('RollbackConfig', None) - assert spec_rollback is not None - assert ('Order' in spec_rollback and - spec_rollback['Order'] == 'start-first') + if TEST_API_VERSION >= "1.28": + spec_rollback = service.attrs['Spec'].get('RollbackConfig', None) + assert spec_rollback is not None + assert ('Order' in spec_rollback and + spec_rollback['Order'] == 'start-first') + @helpers.requires_api_version("1.25") def test_create_with_network(self): client = docker.from_env(version=TEST_API_VERSION) name = helpers.random_name()