- 
          
- 
                Notifications
    You must be signed in to change notification settings 
- Fork 960
          Fail test_installation on warnings, and remove deprecated license classifier
          #2054
        
          New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
          
     Merged
      
        
      
            EliahKagan
  merged 7 commits into
  gitpython-developers:main
from
EliahKagan:deprecated-classifier
  
      
      
   
  Jun 16, 2025 
      
    
                
     Merged
            
            
  
    Fail test_installation on warnings, and remove deprecated license classifier
  
  #2054
              
                    EliahKagan
  merged 7 commits into
  gitpython-developers:main
from
EliahKagan:deprecated-classifier
  
      
      
   
  Jun 16, 2025 
              
            
      
        
          +38
        
        
          −39
        
        
          
        
      
    
  
Conversation
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
    So that the meaning of the `venv.sources` accesses in `test_installation` is more readily clear.
This creates a function (technically, a callable `partial` object) for `test_installation` to use instead of repeating `subproces.run` keyword arguments all the time. This relates directly to steps in `_set_up_venv`, and it's makes about as much sense to do it there as in `test_installation`, so it is placed (and described) in `_set_up_venv`.
By setting the `PYTHONWARNINGS` environment variable to `error` in each of the subprocess invocations. This is strictly stronger than passing `-Werror` for the `python` commands, because it automatically applies to subprocesses (unless they are created with a sanitized environment or otherwise with one in which `PYTHONWARNINGS` has been customized), and because it works for `pip` automatically. Importantly, this will cause warnings internal to Python subprocesses created by `pip` to be treated as errors. It should thus surface any warnings coming from the `setuptools` backend.
Previously, it attempted to show stderr unless empty, first falling back to stdout unless empty, then falling back to the prewritten summary identifying the specific assertion. This now has the `test_installation` assertions capture stderr as well as stdout, handle standard streams as text rather than binary, and show more information when failing, always distinguishing where the information came from: the summary, then labeled captured stdout (empty or not), then labeled captured stderr (empty or not). That applies to all but the last assertion, which does not try to show information differently when it fails, but is simplified to do the right thing now that `subprocess.run` is using text streams. (This subtly changes its semantics, but overall it should be as effective as before at finding the `sys.path` woe it anticipates.)
GitPython project metadata specify the project's license primarily through the value of `license`, currently passed as an argument to `setuptools.setup`, which holds the string `"BSD-3-Clause"`. This is an SPDX license identifier readily understood by both humans and machines. The PyPI trove classifier "License :: OSI Approved :: BSD License" has also been specified in `setup.py`. However, this is not ideal, because: 1. It does not identify a specific license. There are multiple "BSD" licenses in use, with BSD-2-Clause and BSD-3-Clause both in very wide use. 2. It is no longer recommended to use a trove classifier to indicate a license. The use of license classifiers (even unambiguous ones) has been deprecated. See: - https://packaging.python.org/en/latest/specifications/core-metadata/#classifier-multiple-use - https://peps.python.org/pep-0639/#deprecate-license-classifiers This commit removes the classifier. The license itself is of course unchanged, as is the `license` value of `"BSD-3-Clause"`. (An expected effect of this change is that, starting in the next release of GitPython, PyPI may show "License: BSD-3-Clause" instead of the current text "License: BSD License (BSD-3-Clause)".) This change fixes a warning issued by a subprocess of `pip` when installing the package. The warning, until this change, could be observed by running `pip install . -v` or `pip install -e . -v` and examining the verbose output, or by running `pip install .` or `pip install -e .` with the `PYTHONWARNINGS` environment variable set to `error`: SetuptoolsDeprecationWarning: License classifiers are deprecated. !! ******************************************************************************** Please consider removing the following classifiers in favor of a SPDX license expression: License :: OSI Approved :: BSD License See https://packaging.python.org/en/latest/guides/writing-pyproject-toml/#license for details. ******************************************************************************** !! (In preceding commits, `test_installation` has been augmented to set that environment variable, surfacing the error. This change should allow that test to pass, unless it finds other problems.)
It was originally made explicit like this, but it ended up becoming position in my last few commits. This restores that clearer aspect of how it was written before, while keeping all the other changes.
      
        
              This comment was marked as outdated.
        
        
      
    
  This comment was marked as outdated.
  
    Sign up for free
    to join this conversation on GitHub.
    Already have an account?
    Sign in to comment
  
      
  Add this suggestion to a batch that can be applied as a single commit.
  This suggestion is invalid because no changes were made to the code.
  Suggestions cannot be applied while the pull request is closed.
  Suggestions cannot be applied while viewing a subset of changes.
  Only one suggestion per line can be applied in a batch.
  Add this suggestion to a batch that can be applied as a single commit.
  Applying suggestions on deleted lines is not supported.
  You must change the existing code in this line in order to create a valid suggestion.
  Outdated suggestions cannot be applied.
  This suggestion has been applied or marked resolved.
  Suggestions cannot be applied from pending reviews.
  Suggestions cannot be applied on multi-line comments.
  Suggestions cannot be applied while the pull request is queued to merge.
  Suggestion cannot be applied right now. Please check back later.
  
    
  
    
The license trove classifier was imprecise and deprecated
The main way we indicate the license in the project's metadata is:
GitPython/setup.py
Line 72 in 46c439b
However, we have for some time, until this PR, also had a trove classifier indicating "BSD License":
GitPython/setup.py
Line 98 in 46c439b
This has a couple of problems. One is that there is no specific "BSD License", but instead various BSD licenses, of which two are very popular: BSD-3-Clause, which this project uses, and BSD-2-Clause, which is not what this project uses. (No PyPI-recognized trove classifiers for those specific BSD licenses exist.)
The other problem is that all usage of
License :: ...trove classifiers has been deprecated. When the package is installed, the build backend generates a warning. The warning is typically not shown when usingpip, but it will be shown if-vis passed, and it will be shown and cause an error ifPYTHONWARNINGS=erroris set in the environment. (Runningpython -Werror -m pip …is not sufficient to produce it, because it is not passed down to the subprocess runningsetuptools, which is where the warning occurs.)The
"BSD-3-Clause"value of thelicenseargument we pass tosetuptools.setupis already such an SDPX license expression. So all that is needed to fix both these problems is to remove the deprecated trove classifier. However, there are other related problems, some of which make sense to fix at the same time.The installation test didn't detect warnings, nor report errors clearly
test_installationseeks to detect problems installing GitPython, and with #2053 it is back running and passing on all platforms we test. But it has not treated warnings during installation as errors.Especially considering the small number of dependencies GitPython has (when not installing the
testordocextra), we shouldn't ordinarily have any warnings on any supported platform and Python version, and anytime we do, we probably want to know about it. I think it would not be reasonable to setPYTHONWARNINGS=errorin the top-levelpip install '.[test]'commands in the CI test workflows, but that this is a good thing to do intest_installation.Setting it in
test_installationdoes reveal that an error occurs, but without making other changes to the test, it does not reveal in a readily understandable way what error occurs.test_installationcontains this code (and some other code like it):GitPython/test/test_installation.py
Lines 16 to 25 in 46c439b
The intent of
result.stderr or result.stdout or "Can't install project"is to show what, if anything, was written to stderr; otherwise show what, if anything, was written to stdout; and otherwise show the prewritten message. It's true that the most important information to understand an error is often on stderr.The problem is that it will never show what was written to stderr, because the call to
subprocess.runcaptures only stdout, then attempts to access stderr as though it had been captured too. Here's a simplified demonstration of that problem:Depending on how the test runner (usually
pytest, which is the only runner the whole test suite supports) is run, the stderr message may be captured by the test runner and not shown, captured by the test runner and shown later where it may not be readily identified, or not captured by the test runner and possibly seen at some point either at the intuitive position or interleaved elsewhere.The fixes
This PR:
Refactors
test_installationand its helpers to decrease code duplication and to clarify what the steps that set up the new virtual environment are doing, fixes the test bug where stderr was not captured, interacts with thepythonandpipsubprocesses assuming text-based streams and includes in the assertion message the prewritten summary as well as stdout and stderr (labeling each), and runs the subprocesses withPYTHONWARNINGS=error.This surfaces the deprecation warning and causes it to fail
test_installationwith fully detailed output reporting the problem.Removes the deprecated trove classifier from
setup.py. This is sufficient to make the test pass, as there are no other diagnostic messages of warning or higher level.Outscoped
Improve license metadata by Hugo van Kemenade, though not at all specific to GitPython, covers both the problem with some license classifiers (including "BSD License") being unclear and the deprecation, pointing out that an SPDX string should be used to identify in metadata what license a project has chosen.
However, it recommends that one also specify the paths to one's license files using
license-files. This is already something we were not doing, and removing the deprecated classifier does not worsen that situation. OurLICENSEfile also does not have customized elements besides a listed author, who is also one of the authors listed in metadata about authorship. Most importantly, this is just about metadata: theLICENSEfile is part of the project and listed inMANIFEST.in(and various files in the project also carry notices). Nonetheless, it would be good to do something like this.I don't know if we can straightforwardly achieve the effect of
license-filesinsetup.py. That blog post covers static project definitions inpyproject.toml. In addition, that meaning inpyproject.tomloflicenseandlicense-filesis only recognized in sufficiently new versions of build backends. Usually this isn't a problem, but some of these changes to the way licenses are expected to be indicated are actually fairly recent. GitPython supports some old versions of Python that are older than supported by the oldest version ofsetuptoolsthat recognizespyproject.tomlwith alicensevalue that is an SPDX string (even aside from supportinglicense-files).The packaging guide shows the current status for this (PEP 639). One implication here is that it looks like we will have to choose between:
pyproject.tomlto specifylicenseas an SPDX identifier (and I think maybe not switching topyproject.tomlat all, and maybe not being able to specifylicense-filesor equivalent at all).pyproject.tomlwith an old, less usefullicensevalue, which will generate deprecation warnings, and which will eventually--sometime in 2026, I believe--stop the project from being able to be installed and/or published properly. (This deprecation is a more serious deprecation than the one fixed here with classifiers.)setuptoolsbuild backend to another build backend that continues to support older Python versions and also supports modernlicenseandlicense-files, such asflit-coreor (if we drop 3.7 by then)hatchling.Although the original plan was to stick with
setuptoolsper #1716 (comment), some time has passed since then and I suspect there may end up being other reasons to switch build backends. I think it may be best to decide all this at the same time, and when actually making the project definition declarative. Therefore, I have not attempted to addlicense-filesor equivalent metadata in this PR.