Phil Dawson pushed to branch phil/tox-vev-environment at BuildStream / buildstream
Commits:
- 
59ce7bfb
by Phil Dawson at 2019-02-20T11:10:31Z
- 
4cec3d4a
by Phil Dawson at 2019-02-20T12:09:20Z
- 
5735b154
by Phil Dawson at 2019-02-20T12:19:56Z
9 changed files:
- CONTRIBUTING.rst
- buildstream/_versions.py
- buildstream/plugin.py
- + tests/plugins/deprecationwarnings/deprecationwarnings.py
- + tests/plugins/deprecationwarnings/project/elements/deprecated.bst
- + tests/plugins/deprecationwarnings/project/plugins/elements/deprecated_plugin.py
- + tests/plugins/deprecationwarnings/project/plugins/elements/deprecated_plugin.yaml
- + tests/plugins/deprecationwarnings/project/project.conf
- tox.ini
Changes:
| ... | ... | @@ -1589,6 +1589,15 @@ can run ``tox`` with ``-r`` or ``--recreate`` option. | 
| 1589 | 1589 |  | 
| 1590 | 1590 |       ./setup.py test --addopts 'tests/frontend/buildtrack.py::test_build_track'
 | 
| 1591 | 1591 |  | 
| 1592 | +.. tip::
 | |
| 1593 | + | |
| 1594 | +   We also have an environment called 'venv' which takes any arguments
 | |
| 1595 | +   you give it and runs them inside the same virtualenv we use for our
 | |
| 1596 | +   tests::
 | |
| 1597 | + | |
| 1598 | +     tox -e venv -- <your command(s) here>
 | |
| 1599 | +     
 | |
| 1600 | +   Any commands after ``--`` will be run a virtualenv managed by tox.
 | |
| 1592 | 1601 |  | 
| 1593 | 1602 |  Observing coverage
 | 
| 1594 | 1603 |  ~~~~~~~~~~~~~~~~~~
 | 
| ... | ... | @@ -23,7 +23,7 @@ | 
| 23 | 23 |  # This version is bumped whenever enhancements are made
 | 
| 24 | 24 |  # to the `project.conf` format or the core element format.
 | 
| 25 | 25 |  #
 | 
| 26 | -BST_FORMAT_VERSION = 21
 | |
| 26 | +BST_FORMAT_VERSION = 22
 | |
| 27 | 27 |  | 
| 28 | 28 |  | 
| 29 | 29 |  # The base BuildStream artifact version
 | 
| ... | ... | @@ -164,6 +164,25 @@ class Plugin(): | 
| 164 | 164 |         core format version :ref:`core format version <project_format_version>`.
 | 
| 165 | 165 |      """
 | 
| 166 | 166 |  | 
| 167 | +    BST_PLUGIN_DEPRECATED = False
 | |
| 168 | +    """True if this element plugin has been deprecated.
 | |
| 169 | + | |
| 170 | +    If this is set to true, BuildStream will emmit a deprecation
 | |
| 171 | +    warning when this plugin is loaded. This deprecation warning may
 | |
| 172 | +    be suppressed on a plugin by plugin basis by setting
 | |
| 173 | +    ``suppress-deprecation-warnings: true`` in the relevent section of
 | |
| 174 | +    the project's :ref:`plugin configuration overrides <project_overrides>`.
 | |
| 175 | + | |
| 176 | +    """
 | |
| 177 | + | |
| 178 | +    BST_PLUGIN_DEPRECATION_MESSAGE = ""
 | |
| 179 | +    """ The message printed when this element shows a deprecation warning.
 | |
| 180 | + | |
| 181 | +    This should be set if BST_PLUGIN_DEPRECATED is True and should direct the user
 | |
| 182 | +    to the deprecated plug-in's replacement.
 | |
| 183 | + | |
| 184 | +    """
 | |
| 185 | + | |
| 167 | 186 |      def __init__(self, name, context, project, provenance, type_tag):
 | 
| 168 | 187 |  | 
| 169 | 188 |          self.name = name
 | 
| ... | ... | @@ -188,6 +207,12 @@ class Plugin(): | 
| 188 | 207 |          self.__kind = modulename.split('.')[-1]
 | 
| 189 | 208 |          self.debug("Created: {}".format(self))
 | 
| 190 | 209 |  | 
| 210 | +        # If this plugin has been deprecated, emit a warning.
 | |
| 211 | +        if self.BST_PLUGIN_DEPRECATED and not self.__deprecation_warning_silenced():
 | |
| 212 | +            detail = "Using deprecated plugin {}: {}".format(self.__kind,
 | |
| 213 | +                                                             self.BST_PLUGIN_DEPRECATION_MESSAGE)
 | |
| 214 | +            self.__message(MessageType.WARN, detail)
 | |
| 215 | + | |
| 191 | 216 |      def __del__(self):
 | 
| 192 | 217 |          # Dont send anything through the Message() pipeline at destruction time,
 | 
| 193 | 218 |          # any subsequent lookup of plugin by unique id would raise KeyError.
 | 
| ... | ... | @@ -767,6 +792,20 @@ class Plugin(): | 
| 767 | 792 |          else:
 | 
| 768 | 793 |              return self.name
 | 
| 769 | 794 |  | 
| 795 | +    def __deprecation_warning_silenced(self):
 | |
| 796 | +        if not self.BST_PLUGIN_DEPRECATED:
 | |
| 797 | +            return False
 | |
| 798 | +        else:
 | |
| 799 | +            silenced_warnings = set()
 | |
| 800 | +            project = self.__project
 | |
| 801 | +            plugin_overrides = {**project.element_overrides, **project.source_overrides}
 | |
| 802 | + | |
| 803 | +            for key, value in self.node_items(plugin_overrides):
 | |
| 804 | +                if value.get('suppress-deprecation-warnings', False):
 | |
| 805 | +                    silenced_warnings.add(key)
 | |
| 806 | + | |
| 807 | +            return self.get_kind() in silenced_warnings
 | |
| 808 | + | |
| 770 | 809 |  | 
| 771 | 810 |  # Hold on to a lookup table by counter of all instantiated plugins.
 | 
| 772 | 811 |  # We use this to send the id back from child processes so we can lookup
 | 
| 1 | +import pytest
 | |
| 2 | +import tempfile
 | |
| 3 | +import os
 | |
| 4 | +from buildstream.plugintestutils import cli
 | |
| 5 | +from buildstream import _yaml
 | |
| 6 | +import buildstream.plugins.elements.manual
 | |
| 7 | + | |
| 8 | + | |
| 9 | +DATA_DIR = os.path.join(
 | |
| 10 | +    os.path.dirname(os.path.realpath(__file__)),
 | |
| 11 | +    "project"
 | |
| 12 | +)
 | |
| 13 | + | |
| 14 | +_DEPRECATION_MESSAGE = "Here is some detail."
 | |
| 15 | +_DEPRECATION_WARNING = "Using deprecated plugin deprecated_plugin: {}".format(_DEPRECATION_MESSAGE)
 | |
| 16 | + | |
| 17 | + | |
| 18 | +@pytest.mark.datafiles(DATA_DIR)
 | |
| 19 | +def test_deprecation_warning_present(cli, datafiles):
 | |
| 20 | +    project = os.path.join(datafiles.dirname, datafiles.basename)
 | |
| 21 | +    result = cli.run(project=project, args=['show', 'deprecated.bst'])
 | |
| 22 | +    result.assert_success()
 | |
| 23 | +    assert _DEPRECATION_WARNING in result.stderr
 | |
| 24 | + | |
| 25 | + | |
| 26 | +@pytest.mark.datafiles(DATA_DIR)
 | |
| 27 | +def test_suppress_deprecation_warning(cli, datafiles):
 | |
| 28 | +    project = os.path.join(datafiles.dirname, datafiles.basename)
 | |
| 29 | +    result = cli.run(project=project, args=['show', 'manual.bst'])
 | |
| 30 | + | |
| 31 | +    element_overrides = "elements:\n" \
 | |
| 32 | +                        "  deprecated_plugin:\n" \
 | |
| 33 | +                        "    suppress-deprecation-warnings : True\n"
 | |
| 34 | + | |
| 35 | +    project_conf = os.path.join(project, 'project.conf')
 | |
| 36 | +    with open(project_conf, 'a') as f:
 | |
| 37 | +        f.write(element_overrides)
 | |
| 38 | + | |
| 39 | +    result = cli.run(project=project, args=['show', 'deprecated.bst'])
 | |
| 40 | +    result.assert_success()
 | |
| 41 | +    assert _DEPRECATION_WARNING not in result.stderr | 
| 1 | +kind: deprecated_plugin | |
| \ No newline at end of file | 
| 1 | +from buildstream import BuildElement, SandboxFlags
 | |
| 2 | + | |
| 3 | + | |
| 4 | +class DeprecatedPlugin(BuildElement):
 | |
| 5 | +    BST_PLUGIN_DEPRECATED = True
 | |
| 6 | +    BST_PLUGIN_DEPRECATION_MESSAGE = "Here is some detail."
 | |
| 7 | + | |
| 8 | + | |
| 9 | +# Plugin entry point
 | |
| 10 | +def setup():
 | |
| 11 | +    return DeprecatedPlugin | 
| 1 | +# Deprecated-plugin build element does not provide any default
 | |
| 2 | +# build commands
 | |
| 3 | +config:
 | |
| 4 | + | |
| 5 | +  # Commands for configuring the software
 | |
| 6 | +  #
 | |
| 7 | +  configure-commands: []
 | |
| 8 | + | |
| 9 | +  # Commands for building the software
 | |
| 10 | +  #
 | |
| 11 | +  build-commands: []
 | |
| 12 | + | |
| 13 | +  # Commands for installing the software into a
 | |
| 14 | +  # destination folder
 | |
| 15 | +  #
 | |
| 16 | +  install-commands: []
 | |
| 17 | + | |
| 18 | +  # Commands for stripping installed binaries
 | |
| 19 | +  #
 | |
| 20 | +  strip-commands:
 | |
| 21 | +  - |
 | |
| 22 | +    %{strip-binaries} | |
| \ No newline at end of file | 
| 1 | +# Unique project name
 | |
| 2 | +name: deprecation-warnings
 | |
| 3 | + | |
| 4 | +# Required BuildStream format version
 | |
| 5 | +format-version: 20
 | |
| 6 | + | |
| 7 | +# Subdirectory where elements are stored
 | |
| 8 | +element-path: elements
 | |
| 9 | + | |
| 10 | +plugins:
 | |
| 11 | + | |
| 12 | +- origin: local
 | |
| 13 | +  path: plugins/elements
 | |
| 14 | +  elements:
 | |
| 15 | +    deprecated_plugin: 0 | 
| ... | ... | @@ -91,3 +91,14 @@ commands = | 
| 91 | 91 |  deps =
 | 
| 92 | 92 |      click-man >= 0.3.0
 | 
| 93 | 93 |      -rrequirements/requirements.txt
 | 
| 94 | + | |
| 95 | +#
 | |
| 96 | +# Usefull for running arbitrary scripts in a BuildStream virtual env
 | |
| 97 | +#
 | |
| 98 | +[testenv:venv]
 | |
| 99 | +commands = {posargs}
 | |
| 100 | +deps =
 | |
| 101 | +    -rrequirements/requirements.txt
 | |
| 102 | +    -rrequirements/dev-requirements.txt
 | |
| 103 | +    -rrequirements/plugin-requirements.txt
 | |
| 104 | +whitelist_externals = * | 
