Skip to content
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

remove support for Python 2.7 and 3.5 #4229

Merged
merged 50 commits into from
Apr 12, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
50 commits
Select commit Hold shift + click to select a range
68057bc
[EB5] Remove Python 2
branfosj Mar 26, 2023
eab4067
remove py2 deprecation
branfosj Mar 26, 2023
c317efe
flake8 and unit tests
branfosj Mar 26, 2023
f9309b8
remove unecessary restrictions on requirements
branfosj Mar 29, 2023
870cd92
remove use of py2vs3 (pt1)
branfosj Mar 31, 2023
c207173
remove use of py2vs3 (pt2)
branfosj Mar 31, 2023
420a794
remove use of py2vs3 (pt3)
branfosj Mar 31, 2023
7fdea83
remove use of py2vs3 (pt4)
branfosj Mar 31, 2023
452be35
restore import
branfosj Mar 31, 2023
778c21d
need to raise the exception
branfosj Mar 31, 2023
43dd69a
revert change to check if this is the bit causing the test failures
branfosj Mar 31, 2023
59afb81
revert to last working state
branfosj Apr 2, 2023
5d93e85
string_type
branfosj Apr 2, 2023
39b2cd4
revert accidental deletion
branfosj Apr 2, 2023
e2f37a4
reload
branfosj Apr 2, 2023
efa35d5
OrderedDict
branfosj Apr 2, 2023
4c90e9c
Mapping
branfosj Apr 2, 2023
229bc9d
ascii_*
branfosj Apr 2, 2023
0a3090a
StringIO
branfosj Apr 2, 2023
b6e1f3e
do not edit bootstrap script
branfosj Apr 2, 2023
605eb25
urllib
branfosj Apr 2, 2023
8b4dbf9
configparser
branfosj Apr 2, 2023
6510b29
base/meta class
branfosj Apr 2, 2023
f35c2ec
traceback
branfosj Apr 2, 2023
00a84e7
urlopen/html parser
branfosj Apr 2, 2023
01befab
json loads
branfosj Apr 2, 2023
9583500
method name
branfosj Apr 2, 2023
9a6f51e
remove import
branfosj Apr 2, 2023
6294ecb
subprocess terminate and popen
branfosj Apr 2, 2023
0f7fd41
remove sort_looseversions tests
branfosj Apr 2, 2023
29ba208
f8
branfosj Apr 2, 2023
668acef
request
branfosj Apr 2, 2023
f5f58b9
Revert "request"
branfosj Apr 2, 2023
238a160
add deprecation warning
branfosj Apr 2, 2023
ed1e89e
request test
branfosj Apr 2, 2023
85fc1f5
request
branfosj Apr 2, 2023
60266d5
consider python3 before python
branfosj Apr 7, 2023
e9a6955
resolve conflict
branfosj Apr 7, 2023
63f10d0
Merge branch 'py2' of github.com:branfosj/easybuild-framework into py2
branfosj Apr 7, 2023
58d57d1
fix eb python test
branfosj Apr 7, 2023
4179997
Merge branch '5.0.x' into py2
boegel Apr 11, 2023
3aacca4
remove py2 from new apptainer container test
branfosj Apr 11, 2023
3c12303
use log.deprecated in py2vs3 module
boegel Apr 12, 2023
1d61bc0
Merge pull request #7 from boegel/py2
branfosj Apr 12, 2023
eb53b70
fix merge conflict
branfosj Apr 12, 2023
855db89
relocate subprocess_popen_text + subprocess_terminate to easybuild.to…
boegel Apr 12, 2023
9eb6ce2
avoid duplicate copies of create_base_metaclass + mk_wrapper_baseclas…
boegel Apr 12, 2023
117cd51
add #noqa to unused imports in py2vs3/*py
boegel Apr 12, 2023
2ab6d3b
add back python2_is_deprecated to py2vs3/__init__.py, to avoid broken…
boegel Apr 12, 2023
4c1dc23
Merge pull request #8 from boegel/py2
branfosj Apr 12, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/container_tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ jobs:
runs-on: ubuntu-22.04
strategy:
matrix:
python: [2.7, 3.7]
python: [3.7]
fail-fast: false
steps:
- uses: actions/checkout@v3
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/container_tests_apptainer.yml
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ jobs:
runs-on: ubuntu-22.04
strategy:
matrix:
python: [2.7, 3.7]
python: [3.7]
apptainer: [1.0.0, 1.1.7]
fail-fast: false
steps:
Expand Down
4 changes: 2 additions & 2 deletions .github/workflows/eb_command.yml
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ jobs:
runs-on: ubuntu-20.04
strategy:
matrix:
python: [2.7, 3.5, 3.6, 3.7, 3.8, 3.9, '3.10', '3.11']
python: [3.6, 3.7, 3.8, 3.9, '3.10', '3.11']
fail-fast: false
steps:
- uses: actions/checkout@v3
Expand Down Expand Up @@ -91,7 +91,7 @@ jobs:
pymajver=$(python -c 'import sys; print(sys.version_info[0])')
pymajminver=$(python -c 'import sys; print(".".join(str(x) for x in sys.version_info[:2]))')
# check patterns in verbose output
for pattern in "^>> Considering .python.\.\.\." "^>> .python. version: ${pymajminver}\.[0-9]\+, which matches Python ${pymajver} version requirement" "^>> 'python' is able to import 'easybuild.framework', so retaining it" "^>> Selected Python command: python \(.*/bin/python\)" "^This is EasyBuild 4\.[0-9.]\+"; do
for pattern in "^>> Considering .python3.\.\.\." "^>> .python3. version: ${pymajminver}\.[0-9]\+, which matches Python ${pymajver} version requirement" "^>> 'python3' is able to import 'easybuild.framework', so retaining it" "^>> Selected Python command: python3 \(.*/bin/python3\)" "^This is EasyBuild 4\.[0-9.]\+"; do
echo "Looking for pattern \"${pattern}\" in eb_version.out..."
grep "$pattern" eb_version.out
done
Expand Down
10 changes: 2 additions & 8 deletions .github/workflows/linting.yml
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ jobs:
runs-on: ubuntu-20.04
strategy:
matrix:
python-version: [2.7, 3.5, 3.6, 3.7, 3.8, 3.9, '3.10', '3.11']
python-version: [3.6, 3.7, 3.8, 3.9, '3.10', '3.11']

steps:
- uses: actions/checkout@v3
Expand All @@ -30,10 +30,4 @@ jobs:

- name: Run flake8 to verify PEP8-compliance of Python code
run: |
# don't check py2vs3/py3.py when testing with Python 2, and vice versa
if [[ "${{ matrix.python-version }}" =~ "2." ]]; then
py_excl=py3
else
py_excl=py2
fi
flake8 --exclude ./easybuild/tools/py2vs3/${py_excl}.py
flake8
9 changes: 1 addition & 8 deletions .github/workflows/unit_tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ jobs:
runs-on: ubuntu-20.04
strategy:
matrix:
python: [2.7, 3.6]
python: [3.6]
modules_tool:
# use variables defined by 'setup' job above, see also
# https://docs.github.com/en/actions/reference/context-and-expression-syntax-for-github-actions#needs-context
Expand All @@ -46,9 +46,6 @@ jobs:
module_syntax: Lua
include:
# Test different Python 3 versions with Lmod 8.x (with both Lua and Tcl module syntax)
- python: 3.5
modules_tool: ${{needs.setup.outputs.lmod8}}
module_syntax: Lua
- python: 3.7
modules_tool: ${{needs.setup.outputs.lmod8}}
module_syntax: Lua
Expand Down Expand Up @@ -205,10 +202,6 @@ jobs:
# create file owned by root but writable by anyone (used by test_copy_file)
sudo touch /tmp/file_to_overwrite_for_easybuild_test_copy_file.txt
sudo chmod o+w /tmp/file_to_overwrite_for_easybuild_test_copy_file.txt
# silence deprecation warning when using Python 2, since it breaks a bunch of tests
if [ "${{matrix.python}}" == '2.7' ]; then
export TEST_EASYBUILD_SILENCE_DEPRECATION_WARNINGS=python2
fi
# run test suite
python -O -m test.framework.suite 2>&1 | tee test_framework_suite.log
# try and make sure output of running tests is clean (no printed messages/warnings)
Expand Down
11 changes: 5 additions & 6 deletions easybuild/base/fancylogger.py
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,6 @@
import weakref

from easybuild.tools import LooseVersion
from easybuild.tools.py2vs3 import raise_with_traceback, string_type


def _env_to_boolean(varname, default=False):
Expand Down Expand Up @@ -214,11 +213,11 @@ class MissingLevelName(KeyError):

def getLevelInt(level_name):
"""Given a level name, return the int value"""
if not isinstance(level_name, string_type):
if not isinstance(level_name, str):
raise TypeError('Provided name %s is not a string (type %s)' % (level_name, type(level_name)))

level = logging.getLevelName(level_name)
if isinstance(level, string_type):
if isinstance(level, str):
raise MissingLevelName('Unknown loglevel name %s' % level_name)

return level
Expand Down Expand Up @@ -328,7 +327,7 @@ def raiseException(self, message, exception=None, catch=False):
exception = self.RAISE_EXCEPTION_CLASS

self.RAISE_EXCEPTION_LOG_METHOD(fullmessage)
raise_with_traceback(exception, message, tb)
raise exception(message).with_traceback(tb)

# pylint: disable=unused-argument
def deprecated(self, msg, cur_ver, max_ver, depth=2, exception=None, log_callback=None, *args, **kwargs):
Expand Down Expand Up @@ -588,7 +587,7 @@ def logToFile(filename, enable=True, filehandler=None, name=None, max_bytes=MAX_
os.makedirs(directory)
except Exception as ex:
exc, detail, tb = sys.exc_info()
raise_with_traceback(exc, "Cannot create logdirectory %s: %s \n detail: %s" % (directory, ex, detail), tb)
raise exc("Cannot create logdirectory %s: %s \n detail: %s" % (directory, ex, detail)).with_traceback(tb)

return _logToSomething(
logging.handlers.RotatingFileHandler,
Expand Down Expand Up @@ -741,7 +740,7 @@ def setLogLevel(level):
"""
Set a global log level for all FancyLoggers
"""
if isinstance(level, string_type):
if isinstance(level, str):
level = getLevelInt(level)
logger = getLogger(fname=False, clsname=False)
logger.setLevel(level)
Expand Down
2 changes: 1 addition & 1 deletion easybuild/base/frozendict.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,10 +21,10 @@
It can be used as a drop-in replacement for dictionaries where immutability is desired.
"""
import operator
from collections.abc import Mapping
from functools import reduce

from easybuild.base import fancylogger
from easybuild.tools.py2vs3 import Mapping


# minor adjustments:
Expand Down
11 changes: 7 additions & 4 deletions easybuild/base/generaloption.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
* Jens Timmerman (Ghent University)
"""

import configparser
import copy
import difflib
import inspect
Expand All @@ -39,13 +40,15 @@
import re
import sys
import textwrap
from configparser import ConfigParser
from functools import reduce
from io import StringIO
from optparse import Option, OptionGroup, OptionParser, OptionValueError, Values
from optparse import SUPPRESS_HELP as nohelp # supported in optparse of python v2.4

from easybuild.base.fancylogger import getLogger, setroot, setLogLevel, getDetailsLogLevels
from easybuild.base.optcomplete import autocomplete, CompleterOption
from easybuild.tools.py2vs3 import StringIO, configparser, ConfigParser, string_type, subprocess_popen_text
from easybuild.tools.run import subprocess_popen_text
from easybuild.tools.utilities import mk_md_table, mk_rst_table, nub, shell_quote

try:
Expand Down Expand Up @@ -133,7 +136,7 @@ def get_empty_add_flex(allvalues, self=None):
empty = None

if isinstance(allvalues, (list, tuple)):
if isinstance(allvalues[0], string_type):
if isinstance(allvalues[0], str):
empty = ''

if empty is None:
Expand Down Expand Up @@ -464,7 +467,7 @@ def is_value_a_commandline_option(self, opt, value, index=None):
# --longopt=value, so no issues there either.

# following checks assume that value is a string (not a store_or_None)
if not isinstance(value, string_type):
if not isinstance(value, str):
return None

cmdline_index = None
Expand Down Expand Up @@ -1191,7 +1194,7 @@ def add_group_parser(self, opt_dict, description, prefix=None, otherdefaults=Non
# choices
nameds['choices'] = ["%s" % x for x in extra_detail] # force to strings
hlp += ' (choices: %s)' % ', '.join(nameds['choices'])
elif isinstance(extra_detail, string_type) and len(extra_detail) == 1:
elif isinstance(extra_detail, str) and len(extra_detail) == 1:
args.insert(0, "-%s" % extra_detail)
elif isinstance(extra_detail, (dict,)):
# extract any optcomplete completer hints
Expand Down
11 changes: 5 additions & 6 deletions easybuild/base/optcomplete.py
Original file line number Diff line number Diff line change
Expand Up @@ -107,7 +107,6 @@
from optparse import OptionParser, Option
from pprint import pformat

from easybuild.tools.py2vs3 import string_type
from easybuild.tools.utilities import shell_quote

debugfn = None # for debugging only
Expand Down Expand Up @@ -211,7 +210,7 @@ class FileCompleter(Completer):
CALL_ARGS_OPTIONAL = ['prefix']

def __init__(self, endings=None):
if isinstance(endings, string_type):
if isinstance(endings, str):
endings = [endings]
elif endings is None:
endings = []
Expand Down Expand Up @@ -282,11 +281,11 @@ class RegexCompleter(Completer):
def __init__(self, regexlist, always_dirs=True):
self.always_dirs = always_dirs

if isinstance(regexlist, string_type):
if isinstance(regexlist, str):
regexlist = [regexlist]
self.regexlist = []
for regex in regexlist:
if isinstance(regex, string_type):
if isinstance(regex, str):
regex = re.compile(regex)
self.regexlist.append(regex)

Expand Down Expand Up @@ -546,15 +545,15 @@ def autocomplete(parser, arg_completer=None, opt_completer=None, subcmd_complete
# File completion.
if completer and (not prefix or not prefix.startswith('-')):
# Call appropriate completer depending on type.
if isinstance(completer, (string_type, list, tuple)):
if isinstance(completer, (str, list, tuple)):
completer = FileCompleter(completer)
elif not isinstance(completer, (types.FunctionType, types.LambdaType, types.ClassType, types.ObjectType)):
# TODO: what to do here?
pass

completions = completer(**completer_kwargs)

if isinstance(completions, string_type):
if isinstance(completions, str):
# is a bash command, just run it
if SHELL in (BASH,): # TODO: zsh
print(completions)
Expand Down
11 changes: 5 additions & 6 deletions easybuild/base/rest.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,9 +40,10 @@
import copy
import json
from functools import partial
from urllib.parse import urlencode
from urllib.request import HTTPSHandler, Request, build_opener

from easybuild.base import fancylogger
from easybuild.tools.py2vs3 import HTTPSHandler, Request, build_opener, json_loads, string_type, urlencode


class Client(object):
Expand Down Expand Up @@ -180,7 +181,7 @@ def request(self, method, url, body, headers, content_type=None):
else:
body = conn.read()
try:
pybody = json_loads(body)
pybody = json.loads(body)
except ValueError:
pybody = body
fancylogger.getLogger().debug('reponse len: %s ', len(pybody))
Expand All @@ -203,10 +204,8 @@ def get_connection(self, method, url, body, headers):
else:
sep = ''

# value passed to 'data' must be a 'bytes' value (not 'str') in Python 3.x, but a string value in Python 2
# hence, we encode the value obtained (if needed)
# this doesn't affect the value type in Python 2, and makes it a 'bytes' value in Python 3
if isinstance(body, string_type):
# value passed to 'data' must be a 'bytes' value (not 'str') hence, we encode the value obtained (if needed)
if isinstance(body, str):
body = body.encode('utf-8')

request = Request(self.url + sep + url, data=body)
Expand Down
10 changes: 2 additions & 8 deletions easybuild/base/testing.py
Original file line number Diff line number Diff line change
Expand Up @@ -39,16 +39,10 @@
import re
import sys
from contextlib import contextmanager

try:
from cStringIO import StringIO # Python 2
except ImportError:
from io import StringIO # Python 3
from io import StringIO

from unittest import TestCase as OrigTestCase

from easybuild.tools.py2vs3 import string_type


def nicediff(txta, txtb, offset=5):
"""
Expand Down Expand Up @@ -85,7 +79,7 @@ class TestCase(OrigTestCase):
def is_string(self, x):
"""test if the variable x is a string)"""
try:
return isinstance(x, string_type)
return isinstance(x, str)
except NameError:
return isinstance(x, str)

Expand Down
19 changes: 18 additions & 1 deletion easybuild/base/wrapper.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,24 @@
Original code by http://stackoverflow.com/users/416467/kindall from answer 4 of
http://stackoverflow.com/questions/9057669/how-can-i-intercept-calls-to-pythons-magic-methods-in-new-style-classes
"""
from easybuild.tools.py2vs3 import mk_wrapper_baseclass


# based on six's 'with_metaclass' function
# see also https://stackoverflow.com/questions/18513821/python-metaclass-understanding-the-with-metaclass
def create_base_metaclass(base_class_name, metaclass, *bases):
"""Create new class with specified metaclass based on specified base class(es)."""
return metaclass(base_class_name, bases, {})


def mk_wrapper_baseclass(metaclass):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@branfosj If we copy these functions here, we should also import them from easybuild.base.wrapper in py2vs3/py3.py, to avoid having two copies of them?

see branfosj#8


class WrapperBase(object, metaclass=metaclass):
"""
Wrapper class that provides proxy access to an instance of some internal instance.
"""
__wraps__ = None

return WrapperBase


class WrapperMeta(type):
Expand Down
Loading