From 7cc7dda2b7b50ae6cc7f5037b8a181c599ea9700 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Tue, 22 Jan 2019 19:10:31 +0100 Subject: [PATCH 01/11] Default on Docker image: readthedocs/build:3.0 --- readthedocs/settings/base.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/readthedocs/settings/base.py b/readthedocs/settings/base.py index 35b5480e198..6648cdea978 100644 --- a/readthedocs/settings/base.py +++ b/readthedocs/settings/base.py @@ -274,7 +274,7 @@ def USE_PROMOS(self): # noqa # Docker DOCKER_ENABLE = False DOCKER_DEFAULT_IMAGE = 'readthedocs/build' - DOCKER_DEFAULT_VERSION = '2.0' + DOCKER_DEFAULT_VERSION = '3.0' DOCKER_IMAGE = '{}:{}'.format(DOCKER_DEFAULT_IMAGE, DOCKER_DEFAULT_VERSION) DOCKER_IMAGE_SETTINGS = { 'readthedocs/build:1.0': { From df9ef542b22009fbc68d302e933ac76c4ef2eff5 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Tue, 22 Jan 2019 19:12:10 +0100 Subject: [PATCH 02/11] Include new Docker images in settings --- readthedocs/settings/base.py | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/readthedocs/settings/base.py b/readthedocs/settings/base.py index 6648cdea978..4776294f582 100644 --- a/readthedocs/settings/base.py +++ b/readthedocs/settings/base.py @@ -286,13 +286,17 @@ def USE_PROMOS(self): # noqa 'readthedocs/build:3.0': { 'python': {'supported_versions': [2, 2.7, 3, 3.3, 3.4, 3.5, 3.6]}, }, - 'readthedocs/build:stable': { - 'python': {'supported_versions': [2, 2.7, 3, 3.3, 3.4, 3.5, 3.6]}, - }, - 'readthedocs/build:latest': { - 'python': {'supported_versions': [2, 2.7, 3, 3.3, 3.4, 3.5, 3.6]}, + 'readthedocs/build:4.0': { + 'python': {'supported_versions': [2, 2.7, 3, 3.5, 3.6, 3.7]}, }, } + DOCKER_IMAGE_SETTINGS.update({ + 'readthedocs/build:stable': DOCKER_IMAGE_SETTINGS.get('readthedocs/build:3.0'), + 'readthedocs/build:latest': DOCKER_IMAGE_SETTINGS.get('readthedocs/build:4.0'), + 'readthedocs/build:testing': { # 5.0rc1 + 'python': {'supported_versions': [2, 2.7, 3, 3.6, 3.7]}, + }, + }) # All auth ACCOUNT_ADAPTER = 'readthedocs.core.adapters.AccountAdapter' From 293c4b81cf6828219b2c7d2c362b3ba97bc5defb Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Tue, 22 Jan 2019 19:55:58 +0100 Subject: [PATCH 03/11] Use DOCKER_* settings for config validation Remove all the hardcoded valid options/choices from the Config objects and get them from Django settings instead. --- readthedocs/config/config.py | 49 ++++++++++++++++++++++++------------ readthedocs/settings/base.py | 4 ++- 2 files changed, 36 insertions(+), 17 deletions(-) diff --git a/readthedocs/config/config.py b/readthedocs/config/config.py index 601e29c93f9..9183979c22f 100644 --- a/readthedocs/config/config.py +++ b/readthedocs/config/config.py @@ -146,6 +146,8 @@ class BuildConfigBase: 'mkdocs', 'submodules', ] + valid_build_images = [k.split(':')[1] for k in DOCKER_IMAGE_SETTINGS] + default_build_image = DOCKER_DEFAULT_VERSION version = None def __init__(self, env_config, raw_config, source_file): @@ -246,6 +248,24 @@ def python_full_version(self): ) return ver + def get_valid_python_versions_for_image(self, build_image): + """ + Return all the valid Python versions for a Docker image. + + The Docker image (``build_image``) has to be its complete name, already + validated: ``readthedocs/build:4.0``, not just ``4.0``. + + Returns supported versions for the ``DOCKER_DEFAULT_VERSION`` if not + ``build_image`` found. + """ + + if build_image not in DOCKER_IMAGE_SETTINGS: + build_image = '{}:{}'.format( + DOCKER_DEFAULT_IMAGE, + self.default_build_image, + ) + return DOCKER_IMAGE_SETTINGS[build_image]['python']['supported_versions'] + def as_dict(self): config = {} for name in self.PUBLIC_ATTRIBUTES: @@ -268,18 +288,23 @@ class BuildConfigV1(BuildConfigBase): '"python.extra_requirements" section must be a list.' ) - PYTHON_SUPPORTED_VERSIONS = [2, 2.7, 3, 3.5] - DOCKER_SUPPORTED_VERSIONS = ['1.0', '2.0', 'latest'] - version = '1' def get_valid_python_versions(self): - """Get all valid python versions.""" + """ + Return all valid Python versions. + + .. note:: + + It does not take current build image used into account. + """ try: return self.env_config['python']['supported_versions'] except (KeyError, TypeError): - pass - return self.PYTHON_SUPPORTED_VERSIONS + versions = set() + for name, options in DOCKER_IMAGE_SETTINGS.items(): + versions = versions.union(options['python']['supported_versions']) + return versions def get_valid_formats(self): # noqa """Get all valid documentation formats.""" @@ -339,7 +364,7 @@ def validate_build(self): with self.catch_validation_error('build'): build['image'] = validate_choice( str(_build['image']), - self.DOCKER_SUPPORTED_VERSIONS, + self.valid_build_images, ) if ':' not in build['image']: # Prepend proper image name to user's image name @@ -577,8 +602,6 @@ class BuildConfigV2(BuildConfigBase): version = '2' valid_formats = ['htmlzip', 'pdf', 'epub'] - valid_build_images = ['1.0', '2.0', '3.0', 'stable', 'latest'] - default_build_image = 'latest' valid_install_method = [PIP, SETUPTOOLS] valid_sphinx_builders = { 'html': 'sphinx', @@ -793,13 +816,7 @@ def get_valid_python_versions(self): This should be called after ``validate_build()``. """ build_image = self.build.image - if build_image not in DOCKER_IMAGE_SETTINGS: - build_image = '{}:{}'.format( - DOCKER_DEFAULT_IMAGE, - self.default_build_image, - ) - python = DOCKER_IMAGE_SETTINGS[build_image]['python'] - return python['supported_versions'] + return self.get_valid_python_versions_for_image(build_image) def validate_doc_types(self): """ diff --git a/readthedocs/settings/base.py b/readthedocs/settings/base.py index 4776294f582..a43862ac7c7 100644 --- a/readthedocs/settings/base.py +++ b/readthedocs/settings/base.py @@ -274,7 +274,7 @@ def USE_PROMOS(self): # noqa # Docker DOCKER_ENABLE = False DOCKER_DEFAULT_IMAGE = 'readthedocs/build' - DOCKER_DEFAULT_VERSION = '3.0' + DOCKER_DEFAULT_VERSION = 'stable' DOCKER_IMAGE = '{}:{}'.format(DOCKER_DEFAULT_IMAGE, DOCKER_DEFAULT_VERSION) DOCKER_IMAGE_SETTINGS = { 'readthedocs/build:1.0': { @@ -290,6 +290,8 @@ def USE_PROMOS(self): # noqa 'python': {'supported_versions': [2, 2.7, 3, 3.5, 3.6, 3.7]}, }, } + + # Alias tagged via ``docker tag`` on the build servers DOCKER_IMAGE_SETTINGS.update({ 'readthedocs/build:stable': DOCKER_IMAGE_SETTINGS.get('readthedocs/build:3.0'), 'readthedocs/build:latest': DOCKER_IMAGE_SETTINGS.get('readthedocs/build:4.0'), From 9bb3ced53784a53ef756d23517b8cdb0ef94cc4f Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Wed, 23 Jan 2019 11:38:30 +0100 Subject: [PATCH 04/11] Accept numbered versions for V1 and named ones for V2 V1 reads the versions from DOCKER_IMAGE_SETTINGS and keeps only the numbered ones. V2 only accepts `stable` and `latest` (they are hardcoded because they don't change). --- readthedocs/config/config.py | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/readthedocs/config/config.py b/readthedocs/config/config.py index 9183979c22f..fab23a1538f 100644 --- a/readthedocs/config/config.py +++ b/readthedocs/config/config.py @@ -4,6 +4,7 @@ import copy import os +import re from contextlib import contextmanager from django.conf import settings @@ -146,7 +147,6 @@ class BuildConfigBase: 'mkdocs', 'submodules', ] - valid_build_images = [k.split(':')[1] for k in DOCKER_IMAGE_SETTINGS] default_build_image = DOCKER_DEFAULT_VERSION version = None @@ -288,6 +288,13 @@ class BuildConfigV1(BuildConfigBase): '"python.extra_requirements" section must be a list.' ) + # ConfigV1 accepts only numbered versions + valid_build_images = set() + for k in DOCKER_IMAGE_SETTINGS: + image, version = k.split(':') + if re.fullmatch(r'[\d\.]+', version): + valid_build_images.add(version) + version = '1' def get_valid_python_versions(self): @@ -610,6 +617,9 @@ class BuildConfigV2(BuildConfigBase): } builders_display = dict(DOCUMENTATION_CHOICES) + # Config V2 only accepts named versions + valid_build_images = ['stable', 'latest'] + def validate(self): """ Validates and process ``raw_config`` and ``env_config``. From ed9eee86f4c86f444ff5cf4515ea8ea01d704a04 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Wed, 23 Jan 2019 18:38:54 +0100 Subject: [PATCH 05/11] Allow all numbered versions plus stable and latest as valid images YAML v1 and v2 files allow the same Docker build images now, - 1.0 - 2.0 - 3.0 - 4.0 - stable - latest Any new image added to DOCKER_IMAGE_SETTINGS will be automatically picked as accepted for both YAML files. --- readthedocs/config/config.py | 17 +++++++---------- 1 file changed, 7 insertions(+), 10 deletions(-) diff --git a/readthedocs/config/config.py b/readthedocs/config/config.py index fab23a1538f..a06d62bae87 100644 --- a/readthedocs/config/config.py +++ b/readthedocs/config/config.py @@ -147,6 +147,13 @@ class BuildConfigBase: 'mkdocs', 'submodules', ] + + valid_build_images = {'stable', 'latest'} + for k in DOCKER_IMAGE_SETTINGS: + image, version = k.split(':') + if re.fullmatch(r'^[\d\.]+$', version): + valid_build_images.add(version) + default_build_image = DOCKER_DEFAULT_VERSION version = None @@ -288,13 +295,6 @@ class BuildConfigV1(BuildConfigBase): '"python.extra_requirements" section must be a list.' ) - # ConfigV1 accepts only numbered versions - valid_build_images = set() - for k in DOCKER_IMAGE_SETTINGS: - image, version = k.split(':') - if re.fullmatch(r'[\d\.]+', version): - valid_build_images.add(version) - version = '1' def get_valid_python_versions(self): @@ -617,9 +617,6 @@ class BuildConfigV2(BuildConfigBase): } builders_display = dict(DOCUMENTATION_CHOICES) - # Config V2 only accepts named versions - valid_build_images = ['stable', 'latest'] - def validate(self): """ Validates and process ``raw_config`` and ``env_config``. From 2bf0f88e42f8ad5b26b3c979182921752ea6b614 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Wed, 23 Jan 2019 18:41:34 +0100 Subject: [PATCH 06/11] Do not accept `testing` or `5.0rc1` as valid image --- readthedocs/settings/base.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/readthedocs/settings/base.py b/readthedocs/settings/base.py index a43862ac7c7..3586fbde508 100644 --- a/readthedocs/settings/base.py +++ b/readthedocs/settings/base.py @@ -295,9 +295,6 @@ def USE_PROMOS(self): # noqa DOCKER_IMAGE_SETTINGS.update({ 'readthedocs/build:stable': DOCKER_IMAGE_SETTINGS.get('readthedocs/build:3.0'), 'readthedocs/build:latest': DOCKER_IMAGE_SETTINGS.get('readthedocs/build:4.0'), - 'readthedocs/build:testing': { # 5.0rc1 - 'python': {'supported_versions': [2, 2.7, 3, 3.6, 3.7]}, - }, }) # All auth From ee640023bc9897ebb5c209af42f630700acb6ae0 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Wed, 23 Jan 2019 18:41:58 +0100 Subject: [PATCH 07/11] Default Docker image to `latest` in YAML v1 and v2 By making `latest` as default, and considering that this will be `4.0` which has python `3.7`, the default python selected when using `3` will be `3.7`. --- readthedocs/settings/base.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/readthedocs/settings/base.py b/readthedocs/settings/base.py index 3586fbde508..0134ae93299 100644 --- a/readthedocs/settings/base.py +++ b/readthedocs/settings/base.py @@ -274,7 +274,7 @@ def USE_PROMOS(self): # noqa # Docker DOCKER_ENABLE = False DOCKER_DEFAULT_IMAGE = 'readthedocs/build' - DOCKER_DEFAULT_VERSION = 'stable' + DOCKER_DEFAULT_VERSION = 'latest' DOCKER_IMAGE = '{}:{}'.format(DOCKER_DEFAULT_IMAGE, DOCKER_DEFAULT_VERSION) DOCKER_IMAGE_SETTINGS = { 'readthedocs/build:1.0': { From b829d70ce0b5c87365fcdb7c238725c0f6c27652 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Wed, 23 Jan 2019 18:43:02 +0100 Subject: [PATCH 08/11] Make test pass the decisions made --- readthedocs/config/tests/test_config.py | 12 ++++++------ .../rtd_tests/tests/test_config_integration.py | 4 ++-- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/readthedocs/config/tests/test_config.py b/readthedocs/config/tests/test_config.py index 0d4d0e92d37..b3d93dc83f4 100644 --- a/readthedocs/config/tests/test_config.py +++ b/readthedocs/config/tests/test_config.py @@ -363,8 +363,8 @@ def test_it_validates_wrong_type_right_value(self): ) build.validate() assert build.python.version == 3 - assert build.python_interpreter == 'python3.5' - assert build.python_full_version == 3.5 + assert build.python_interpreter == 'python3.7' + assert build.python_full_version == 3.7 def test_it_validates_env_supported_versions(self): build = get_build_config( @@ -483,7 +483,7 @@ def test_it_fails_if_build_is_invalid_option(self, tmpdir): apply_fs(tmpdir, yaml_config_dir) build = BuildConfigV1( {}, - {'build': {'image': 3.0}}, + {'build': {'image': 3.2}}, source_file=str(tmpdir.join('readthedocs.yml')), ) with raises(InvalidConfig) as excinfo: @@ -513,7 +513,7 @@ def test_it_works_on_python_validation(self, tmpdir): {}, { 'build': {'image': 'latest'}, - 'python': {'version': '3.3'}, + 'python': {'version': '3.6'}, }, source_file=str(tmpdir.join('readthedocs.yml')), ) @@ -538,7 +538,7 @@ def test_default(self, tmpdir): source_file=str(tmpdir.join('readthedocs.yml')), ) build.validate() - assert build.build.image == 'readthedocs/build:2.0' + assert build.build.image == 'readthedocs/build:latest' @pytest.mark.parametrize( 'image', ['latest', 'readthedocs/build:3.0', 'rtd/build:latest'], @@ -712,7 +712,7 @@ def test_as_dict(tmpdir): 'use_system_site_packages': False, }, 'build': { - 'image': 'readthedocs/build:2.0', + 'image': 'readthedocs/build:latest', }, 'conda': None, 'sphinx': { diff --git a/readthedocs/rtd_tests/tests/test_config_integration.py b/readthedocs/rtd_tests/tests/test_config_integration.py index bfae7d8dbe1..748d56906a3 100644 --- a/readthedocs/rtd_tests/tests/test_config_integration.py +++ b/readthedocs/rtd_tests/tests/test_config_integration.py @@ -145,7 +145,7 @@ def test_python_supported_versions_image_latest(self, load_config): config = load_yaml_config(self.version) self.assertEqual( config.get_valid_python_versions(), - [2, 2.7, 3, 3.3, 3.4, 3.5, 3.6], + [2, 2.7, 3, 3.5, 3.6, 3.7], ) @mock.patch('readthedocs.doc_builder.config.load_config') @@ -507,7 +507,7 @@ def test_python_version(self, checkout_path, tmpdir): config = self.get_update_docs_task().config assert config.python.version == 3 - assert config.python_full_version == 3.6 + assert config.python_full_version == 3.7 @patch('readthedocs.doc_builder.environments.BuildEnvironment.run') def test_python_install_requirements(self, run, checkout_path, tmpdir): From 4cd913397a2a640b0b488159252ada43844ace7c Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Wed, 23 Jan 2019 18:50:27 +0100 Subject: [PATCH 09/11] Lint issue fixed --- readthedocs/config/config.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/readthedocs/config/config.py b/readthedocs/config/config.py index a06d62bae87..109410ea5fb 100644 --- a/readthedocs/config/config.py +++ b/readthedocs/config/config.py @@ -309,7 +309,7 @@ def get_valid_python_versions(self): return self.env_config['python']['supported_versions'] except (KeyError, TypeError): versions = set() - for name, options in DOCKER_IMAGE_SETTINGS.items(): + for _, options in DOCKER_IMAGE_SETTINGS.items(): versions = versions.union(options['python']['supported_versions']) return versions From caaff03c98e185baf956f5fa4b2e04984713a6c4 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Wed, 23 Jan 2019 19:04:14 +0100 Subject: [PATCH 10/11] Make valid_build_images a @property --- readthedocs/config/config.py | 22 ++++++++++++++++------ 1 file changed, 16 insertions(+), 6 deletions(-) diff --git a/readthedocs/config/config.py b/readthedocs/config/config.py index 109410ea5fb..7c686614d98 100644 --- a/readthedocs/config/config.py +++ b/readthedocs/config/config.py @@ -148,12 +148,6 @@ class BuildConfigBase: 'submodules', ] - valid_build_images = {'stable', 'latest'} - for k in DOCKER_IMAGE_SETTINGS: - image, version = k.split(':') - if re.fullmatch(r'^[\d\.]+$', version): - valid_build_images.add(version) - default_build_image = DOCKER_DEFAULT_VERSION version = None @@ -255,6 +249,22 @@ def python_full_version(self): ) return ver + @property + def valid_build_images(self): + """ + Return all the valid Docker image choices for ``build.image`` option. + + The user can use any of this values in the YAML file. These values are + the keys of ``DOCKER_IMAGE_SETTINGS`` Django setting (without the + ``readthedocs/build`` part) plus ``stable`` and ``latest``. + """ + images = {'stable', 'latest'} + for k in DOCKER_IMAGE_SETTINGS: + image, version = k.split(':') + if re.fullmatch(r'^[\d\.]+$', version): + images.add(version) + return images + def get_valid_python_versions_for_image(self, build_image): """ Return all the valid Python versions for a Docker image. From a7770c0d48f05814bc4f99967ab12bf5ea29a884 Mon Sep 17 00:00:00 2001 From: Manuel Kaufmann Date: Fri, 25 Jan 2019 17:14:13 +0100 Subject: [PATCH 11/11] Lint --- readthedocs/config/config.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/readthedocs/config/config.py b/readthedocs/config/config.py index 7c686614d98..5db36829734 100644 --- a/readthedocs/config/config.py +++ b/readthedocs/config/config.py @@ -260,7 +260,7 @@ def valid_build_images(self): """ images = {'stable', 'latest'} for k in DOCKER_IMAGE_SETTINGS: - image, version = k.split(':') + _, version = k.split(':') if re.fullmatch(r'^[\d\.]+$', version): images.add(version) return images