Merge pull request #1344 from tahoe-lafs/3072-python-3.12-support

Python 3.12 support, compatability with Eliot 1.15

Fixes ticket:4071
This commit is contained in:
Itamar Turner-Trauring 2023-11-21 08:13:50 -05:00 committed by GitHub
commit e3c8ff0fc6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
24 changed files with 106 additions and 296 deletions

View File

@ -46,8 +46,8 @@ export PIP_FIND_LINKS="file://${WHEELHOUSE_PATH}"
# setuptools 45 requires Python 3.5 or newer. Even though we upgraded pip # setuptools 45 requires Python 3.5 or newer. Even though we upgraded pip
# above, it may still not be able to get us a compatible version unless we # above, it may still not be able to get us a compatible version unless we
# explicitly ask for one. # explicitly ask for one.
"${PIP}" install --upgrade setuptools==44.0.0 wheel "${PIP}" install --upgrade setuptools wheel
# Just about every user of this image wants to use tox from the bootstrap # Just about every user of this image wants to use tox from the bootstrap
# virtualenv so go ahead and install it now. # virtualenv so go ahead and install it now.
"${PIP}" install "tox~=3.0" "${PIP}" install "tox~=4.0"

View File

@ -45,16 +45,17 @@ jobs:
fail-fast: false fail-fast: false
matrix: matrix:
include: include:
# On macOS don't bother with 3.8, just to get faster builds.
- os: macos-12 - os: macos-12
python-version: "3.9" python-version: "3.12"
- os: macos-12
python-version: "3.11"
# We only support PyPy on Linux at the moment. # We only support PyPy on Linux at the moment.
- os: ubuntu-latest - os: ubuntu-latest
python-version: "pypy-3.8" python-version: "pypy-3.8"
- os: ubuntu-latest - os: ubuntu-latest
python-version: "pypy-3.9" python-version: "pypy-3.9"
- os: ubuntu-latest
python-version: "3.12"
- os: windows-latest
python-version: "3.12"
steps: steps:
# See https://github.com/actions/checkout. A fetch-depth of 0 # See https://github.com/actions/checkout. A fetch-depth of 0
@ -72,7 +73,7 @@ jobs:
- name: Install Python packages - name: Install Python packages
run: | run: |
pip install --upgrade "tox<4" tox-gh-actions setuptools pip install --upgrade tox tox-gh-actions setuptools
pip list pip list
- name: Display tool versions - name: Display tool versions
@ -169,7 +170,7 @@ jobs:
- false - false
include: include:
- os: ubuntu-20.04 - os: ubuntu-20.04
python-version: "3.10" python-version: "3.12"
force-foolscap: true force-foolscap: true
steps: steps:
@ -204,7 +205,7 @@ jobs:
- name: Install Python packages - name: Install Python packages
run: | run: |
pip install --upgrade "tox<4" pip install --upgrade tox
pip list pip list
- name: Display tool versions - name: Display tool versions
@ -264,7 +265,7 @@ jobs:
- name: Install Python packages - name: Install Python packages
run: | run: |
pip install --upgrade "tox<4" pip install --upgrade tox
pip list pip list
- name: Display tool versions - name: Display tool versions

View File

@ -30,6 +30,7 @@ from allmydata.util.deferredutil import async_to_deferred
if sys.platform.startswith('win'): if sys.platform.startswith('win'):
pytest.skip('Skipping Tor tests on Windows', allow_module_level=True) pytest.skip('Skipping Tor tests on Windows', allow_module_level=True)
@pytest.mark.skipif(sys.version_info[:2] > (3, 11), reason='Chutney still does not support 3.12')
@pytest_twisted.inlineCallbacks @pytest_twisted.inlineCallbacks
def test_onion_service_storage(reactor, request, temp_dir, flog_gatherer, tor_network, tor_introducer_furl): def test_onion_service_storage(reactor, request, temp_dir, flog_gatherer, tor_network, tor_introducer_furl):
""" """
@ -140,6 +141,7 @@ def _create_anonymous_node(reactor, name, web_port, request, temp_dir, flog_gath
print("okay, launched") print("okay, launched")
return result return result
@pytest.mark.skipif(sys.version_info[:2] > (3, 11), reason='Chutney still does not support 3.12')
@pytest.mark.skipif(sys.platform.startswith('darwin'), reason='This test has issues on macOS') @pytest.mark.skipif(sys.platform.startswith('darwin'), reason='This test has issues on macOS')
@pytest_twisted.inlineCallbacks @pytest_twisted.inlineCallbacks
def test_anonymous_client(reactor, request, temp_dir, flog_gatherer, tor_network, introducer_furl): def test_anonymous_client(reactor, request, temp_dir, flog_gatherer, tor_network, introducer_furl):

View File

@ -0,0 +1 @@
Added support for Python 3.12, and work with Eliot 1.15

3
pyproject.toml Normal file
View File

@ -0,0 +1,3 @@
[build-system]
requires = ["setuptools"]
build-backend = "setuptools.build_meta"

View File

@ -112,7 +112,7 @@ install_requires = [
"magic-wormhole >= 0.10.2", "magic-wormhole >= 0.10.2",
# We want a new enough version to support custom JSON encoders. # We want a new enough version to support custom JSON encoders.
"eliot < 1.15.0", # temporary cap, to be fixed in PR #1344 "eliot >= 1.14.0",
"pyrsistent", "pyrsistent",
@ -157,10 +157,6 @@ install_requires = [
"filelock", "filelock",
] ]
setup_requires = [
'setuptools >= 28.8.0', # for PEP-440 style versions
]
tor_requires = [ tor_requires = [
# 23.5 added support for custom TLS contexts in web_agent(), which is # 23.5 added support for custom TLS contexts in web_agent(), which is
# needed for the HTTP storage client to run over Tor. # needed for the HTTP storage client to run over Tor.
@ -384,8 +380,8 @@ setup(name="tahoe-lafs", # also set in __init__.py
package_dir = {'':'src'}, package_dir = {'':'src'},
packages=find_packages('src') + ['allmydata.test.plugins'], packages=find_packages('src') + ['allmydata.test.plugins'],
classifiers=trove_classifiers, classifiers=trove_classifiers,
# We support Python 3.8 or later, 3.12 is untested for now # We support Python 3.8 or later, 3.13 is untested for now
python_requires=">=3.8, <3.12", python_requires=">=3.8, <3.13",
install_requires=install_requires, install_requires=install_requires,
extras_require={ extras_require={
# Duplicate the Twisted pywin32 dependency here. See # Duplicate the Twisted pywin32 dependency here. See
@ -409,8 +405,8 @@ setup(name="tahoe-lafs", # also set in __init__.py
# selected here are just the current versions at the time. # selected here are just the current versions at the time.
# Bumping them to keep up with future releases is fine as long # Bumping them to keep up with future releases is fine as long
# as those releases are known to actually work. # as those releases are known to actually work.
"pip==22.0.3", "pip==23.3.1",
"wheel==0.37.1", "wheel==0.41.3",
"subunitreporter==23.8.0", "subunitreporter==23.8.0",
"python-subunit==1.4.2", "python-subunit==1.4.2",
"junitxml==0.7", "junitxml==0.7",
@ -446,7 +442,6 @@ setup(name="tahoe-lafs", # also set in __init__.py
"allmydata": ["ported-modules.txt"], "allmydata": ["ported-modules.txt"],
}, },
include_package_data=True, include_package_data=True,
setup_requires=setup_requires,
entry_points={ entry_points={
'console_scripts': [ 'console_scripts': [
'tahoe = allmydata.scripts.runner:run', 'tahoe = allmydata.scripts.runner:run',

View File

@ -3,16 +3,6 @@ Decentralized storage grid.
community web site: U{https://tahoe-lafs.org/} community web site: U{https://tahoe-lafs.org/}
""" """
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
from future.utils import PY2, PY3
if PY2:
# Don't import future str() so we don't break Foolscap serialization on Python 2.
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, max, min # noqa: F401
from past.builtins import unicode as str
__all__ = [ __all__ = [
"__version__", "__version__",
@ -52,12 +42,6 @@ __appname__ = "tahoe-lafs"
# https://tahoe-lafs.org/trac/tahoe-lafs/wiki/Versioning # https://tahoe-lafs.org/trac/tahoe-lafs/wiki/Versioning
__full_version__ = __appname__ + '/' + str(__version__) __full_version__ = __appname__ + '/' + str(__version__)
# Install Python 3 module locations in Python 2:
from future import standard_library
standard_library.install_aliases()
# Monkey-patch 3rd party libraries: # Monkey-patch 3rd party libraries:
from ._monkeypatch import patch from ._monkeypatch import patch
patch() patch()
@ -72,7 +56,6 @@ del patch
# #
# Also note that BytesWarnings only happen if Python is run with -b option, so # Also note that BytesWarnings only happen if Python is run with -b option, so
# in practice this should only affect tests. # in practice this should only affect tests.
if PY3:
import warnings import warnings
# Error on BytesWarnings, to catch things like str(b""), but only for # Error on BytesWarnings, to catch things like str(b""), but only for
# allmydata code. # allmydata code.

View File

@ -12,6 +12,7 @@ from base64 import urlsafe_b64encode
from functools import partial from functools import partial
from configparser import NoSectionError from configparser import NoSectionError
from six import ensure_text
from foolscap.furl import ( from foolscap.furl import (
decode_furl, decode_furl,
) )
@ -989,6 +990,9 @@ class _Client(node.Node, pollmixin.PollMixin):
static_servers = servers_yaml.get("storage", {}) static_servers = servers_yaml.get("storage", {})
log.msg("found %d static servers in private/servers.yaml" % log.msg("found %d static servers in private/servers.yaml" %
len(static_servers)) len(static_servers))
static_servers = {
ensure_text(key): value for (key, value) in static_servers.items()
}
self.storage_broker.set_static_servers(static_servers) self.storage_broker.set_static_servers(static_servers)
except EnvironmentError: except EnvironmentError:
pass pass

View File

@ -78,7 +78,7 @@ _READONLY_PEERS = Field(
def _serialize_existing_shares(existing_shares): def _serialize_existing_shares(existing_shares):
return { return {
server: list(shares) ensure_str(server): list(shares)
for (server, shares) for (server, shares)
in existing_shares.items() in existing_shares.items()
} }
@ -91,7 +91,7 @@ _EXISTING_SHARES = Field(
def _serialize_happiness_mappings(happiness_mappings): def _serialize_happiness_mappings(happiness_mappings):
return { return {
sharenum: base32.b2a(serverid) str(sharenum): ensure_str(base32.b2a(serverid))
for (sharenum, serverid) for (sharenum, serverid)
in happiness_mappings.items() in happiness_mappings.items()
} }
@ -112,7 +112,7 @@ _UPLOAD_TRACKERS = Field(
u"upload_trackers", u"upload_trackers",
lambda trackers: list( lambda trackers: list(
dict( dict(
server=tracker.get_name(), server=ensure_str(tracker.get_name()),
shareids=sorted(tracker.buckets.keys()), shareids=sorted(tracker.buckets.keys()),
) )
for tracker for tracker
@ -123,7 +123,7 @@ _UPLOAD_TRACKERS = Field(
_ALREADY_SERVERIDS = Field( _ALREADY_SERVERIDS = Field(
u"already_serverids", u"already_serverids",
lambda d: d, lambda d: {str(k): v for k, v in d.items()},
u"Some servers which are already holding some shares that we were interested in uploading.", u"Some servers which are already holding some shares that we were interested in uploading.",
) )

View File

@ -32,7 +32,6 @@ Ported to Python 3.
from __future__ import annotations from __future__ import annotations
from six import ensure_text
from typing import Union, Callable, Any, Optional, cast, Dict from typing import Union, Callable, Any, Optional, cast, Dict
from os import urandom from os import urandom
import re import re
@ -273,7 +272,6 @@ class StorageFarmBroker(service.MultiService):
# doesn't really matter but it makes the logging behavior more # doesn't really matter but it makes the logging behavior more
# predictable and easier to test (and at least one test does depend on # predictable and easier to test (and at least one test does depend on
# this sorted order). # this sorted order).
servers = {ensure_text(key): value for (key, value) in servers.items()}
for (server_id, server) in sorted(servers.items()): for (server_id, server) in sorted(servers.items()):
try: try:
storage_server = self._make_storage_server( storage_server = self._make_storage_server(

View File

@ -125,5 +125,5 @@ if sys.platform == "win32":
initialize() initialize()
from eliot import to_file from eliot import to_file
from allmydata.util.eliotutil import eliot_json_encoder from allmydata.util.jsonbytes import AnyBytesJSONEncoder
to_file(open("eliot.log", "wb"), encoder=eliot_json_encoder) to_file(open("eliot.log", "wb"), encoder=AnyBytesJSONEncoder)

View File

@ -1352,6 +1352,26 @@ class _TestCaseMixin(object):
def assertRaises(self, *a, **kw): def assertRaises(self, *a, **kw):
return self._dummyCase.assertRaises(*a, **kw) return self._dummyCase.assertRaises(*a, **kw)
def failUnless(self, *args, **kwargs):
"""Backwards compatibility method."""
self.assertTrue(*args, **kwargs)
def failIf(self, *args, **kwargs):
"""Backwards compatibility method."""
self.assertFalse(*args, **kwargs)
def failIfEqual(self, *args, **kwargs):
"""Backwards compatibility method."""
self.assertNotEqual(*args, **kwargs)
def failUnlessEqual(self, *args, **kwargs):
"""Backwards compatibility method."""
self.assertEqual(*args, **kwargs)
def failUnlessReallyEqual(self, *args, **kwargs):
"""Backwards compatibility method."""
self.assertReallyEqual(*args, **kwargs)
class SyncTestCase(_TestCaseMixin, TestCase): class SyncTestCase(_TestCaseMixin, TestCase):
""" """

View File

@ -29,6 +29,7 @@ from eliot import (
ILogger, ILogger,
) )
from eliot.testing import ( from eliot.testing import (
MemoryLogger,
swap_logger, swap_logger,
check_for_errors, check_for_errors,
) )
@ -37,8 +38,8 @@ from twisted.python.monkey import (
MonkeyPatcher, MonkeyPatcher,
) )
from ..util.eliotutil import ( from ..util.jsonbytes import (
MemoryLogger, AnyBytesJSONEncoder
) )
_NAME = Field.for_types( _NAME = Field.for_types(
@ -146,7 +147,7 @@ def with_logging(
""" """
@wraps(test_method) @wraps(test_method)
def run_with_logging(*args, **kwargs): def run_with_logging(*args, **kwargs):
validating_logger = MemoryLogger() validating_logger = MemoryLogger(encoder=AnyBytesJSONEncoder)
original = swap_logger(None) original = swap_logger(None)
try: try:
swap_logger(_TwoLoggers(original, validating_logger)) swap_logger(_TwoLoggers(original, validating_logger))

View File

@ -850,6 +850,7 @@ class StorageClients(SyncTestCase):
actionType=u"storage-client:broker:set-static-servers", actionType=u"storage-client:broker:set-static-servers",
succeeded=True, succeeded=True,
), ),
encoder_=json.AnyBytesJSONEncoder
) )
def test_static_servers(self, logger): def test_static_servers(self, logger):
""" """
@ -884,6 +885,7 @@ class StorageClients(SyncTestCase):
actionType=u"storage-client:broker:make-storage-server", actionType=u"storage-client:broker:make-storage-server",
succeeded=False, succeeded=False,
), ),
encoder_=json.AnyBytesJSONEncoder
) )
def test_invalid_static_server(self, logger): def test_invalid_static_server(self, logger):
""" """

View File

@ -507,7 +507,7 @@ class TestUtil(unittest.TestCase):
""" """
remove a simple prefix properly remove a simple prefix properly
""" """
self.assertEquals( self.assertEqual(
remove_prefix(b"foobar", b"foo"), remove_prefix(b"foobar", b"foo"),
b"bar" b"bar"
) )
@ -523,7 +523,7 @@ class TestUtil(unittest.TestCase):
""" """
removing a zero-length prefix does nothing removing a zero-length prefix does nothing
""" """
self.assertEquals( self.assertEqual(
remove_prefix(b"foobar", b""), remove_prefix(b"foobar", b""),
b"foobar", b"foobar",
) )
@ -532,7 +532,7 @@ class TestUtil(unittest.TestCase):
""" """
removing a prefix which is the whole string is empty removing a prefix which is the whole string is empty
""" """
self.assertEquals( self.assertEqual(
remove_prefix(b"foobar", b"foobar"), remove_prefix(b"foobar", b"foobar"),
b"", b"",
) )

View File

@ -47,7 +47,6 @@ from eliot import (
Message, Message,
MessageType, MessageType,
fields, fields,
FileDestination,
MemoryLogger, MemoryLogger,
) )
from eliot.twisted import DeferredContext from eliot.twisted import DeferredContext
@ -64,7 +63,6 @@ from twisted.internet.task import deferLater
from twisted.internet import reactor from twisted.internet import reactor
from ..util.eliotutil import ( from ..util.eliotutil import (
eliot_json_encoder,
log_call_deferred, log_call_deferred,
_parse_destination_description, _parse_destination_description,
_EliotLogging, _EliotLogging,
@ -188,8 +186,8 @@ class ParseDestinationDescriptionTests(SyncTestCase):
""" """
reactor = object() reactor = object()
self.assertThat( self.assertThat(
_parse_destination_description("file:-")(reactor), _parse_destination_description("file:-")(reactor).file,
Equals(FileDestination(stdout, encoder=eliot_json_encoder)), Equals(stdout),
) )

View File

@ -28,14 +28,14 @@ INTRODUCERS_CFG_FURLS_COMMENTED="""introducers:
class MultiIntroTests(unittest.TestCase): class MultiIntroTests(unittest.TestCase):
def setUp(self): async def setUp(self):
# setup tahoe.cfg and basedir/private/introducers # setup tahoe.cfg and basedir/private/introducers
# create a custom tahoe.cfg # create a custom tahoe.cfg
self.basedir = os.path.dirname(self.mktemp()) self.basedir = os.path.dirname(self.mktemp())
c = open(os.path.join(self.basedir, "tahoe.cfg"), "w") c = open(os.path.join(self.basedir, "tahoe.cfg"), "w")
config = {'hide-ip':False, 'listen': 'tcp', config = {'hide-ip':False, 'listen': 'tcp',
'port': None, 'location': None, 'hostname': 'example.net'} 'port': None, 'location': None, 'hostname': 'example.net'}
write_node_config(c, config) await write_node_config(c, config)
c.write("[storage]\n") c.write("[storage]\n")
c.write("enabled = false\n") c.write("enabled = false\n")
c.close() c.close()
@ -63,8 +63,7 @@ class MultiIntroTests(unittest.TestCase):
# assertions # assertions
self.failUnlessEqual(ic_count, len(connections["introducers"])) self.failUnlessEqual(ic_count, len(connections["introducers"]))
@defer.inlineCallbacks async def test_read_introducer_furl_from_tahoecfg(self):
def test_read_introducer_furl_from_tahoecfg(self):
""" """
The deprecated [client]introducer.furl item is still read and respected. The deprecated [client]introducer.furl item is still read and respected.
""" """
@ -72,7 +71,7 @@ class MultiIntroTests(unittest.TestCase):
c = open(os.path.join(self.basedir, "tahoe.cfg"), "w") c = open(os.path.join(self.basedir, "tahoe.cfg"), "w")
config = {'hide-ip':False, 'listen': 'tcp', config = {'hide-ip':False, 'listen': 'tcp',
'port': None, 'location': None, 'hostname': 'example.net'} 'port': None, 'location': None, 'hostname': 'example.net'}
write_node_config(c, config) await write_node_config(c, config)
fake_furl = "furl1" fake_furl = "furl1"
c.write("[client]\n") c.write("[client]\n")
c.write("introducer.furl = %s\n" % fake_furl) c.write("introducer.furl = %s\n" % fake_furl)
@ -139,14 +138,14 @@ introducers:
""" """
class NoDefault(unittest.TestCase): class NoDefault(unittest.TestCase):
def setUp(self): async def setUp(self):
# setup tahoe.cfg and basedir/private/introducers # setup tahoe.cfg and basedir/private/introducers
# create a custom tahoe.cfg # create a custom tahoe.cfg
self.basedir = os.path.dirname(self.mktemp()) self.basedir = os.path.dirname(self.mktemp())
c = open(os.path.join(self.basedir, "tahoe.cfg"), "w") c = open(os.path.join(self.basedir, "tahoe.cfg"), "w")
config = {'hide-ip':False, 'listen': 'tcp', config = {'hide-ip':False, 'listen': 'tcp',
'port': None, 'location': None, 'hostname': 'example.net'} 'port': None, 'location': None, 'hostname': 'example.net'}
write_node_config(c, config) await write_node_config(c, config)
c.write("[storage]\n") c.write("[storage]\n")
c.write("enabled = false\n") c.write("enabled = false\n")
c.close() c.close()

View File

@ -26,7 +26,7 @@ from typing import Union, Callable, Tuple, Iterable
from queue import Queue from queue import Queue
from cbor2 import dumps from cbor2 import dumps
from pycddl import ValidationError as CDDLValidationError from pycddl import ValidationError as CDDLValidationError
from hypothesis import assume, given, strategies as st from hypothesis import assume, given, strategies as st, settings as hypothesis_settings
from fixtures import Fixture, TempDir, MonkeyPatch from fixtures import Fixture, TempDir, MonkeyPatch
from treq.testing import StubTreq from treq.testing import StubTreq
from klein import Klein from klein import Klein
@ -442,6 +442,9 @@ class CustomHTTPServerTests(SyncTestCase):
result_of(client.get_version()) result_of(client.get_version())
@given(length=st.integers(min_value=1, max_value=1_000_000)) @given(length=st.integers(min_value=1, max_value=1_000_000))
# On Python 3.12 we're getting weird deadline issues in CI, so disabling
# for now.
@hypothesis_settings(deadline=None)
def test_limited_content_fits(self, length): def test_limited_content_fits(self, length):
""" """
``http_client.limited_content()`` returns the body if it is less than ``http_client.limited_content()`` returns the body if it is less than

View File

@ -23,7 +23,7 @@ def assert_soup_has_favicon(testcase, soup):
``BeautifulSoup`` object ``soup`` contains the tahoe favicon link. ``BeautifulSoup`` object ``soup`` contains the tahoe favicon link.
""" """
links = soup.find_all(u'link', rel=u'shortcut icon') links = soup.find_all(u'link', rel=u'shortcut icon')
testcase.assert_( testcase.assertTrue(
any(t[u'href'] == u'/icon.png' for t in links), soup) any(t[u'href'] == u'/icon.png' for t in links), soup)
@ -92,6 +92,6 @@ def assert_soup_has_text(testcase, soup, text):
``BeautifulSoup`` object ``soup`` contains the passed in ``text`` anywhere ``BeautifulSoup`` object ``soup`` contains the passed in ``text`` anywhere
as a text node. as a text node.
""" """
testcase.assert_( testcase.assertTrue(
soup.find_all(string=re.compile(re.escape(text))), soup.find_all(string=re.compile(re.escape(text))),
soup) soup)

View File

@ -117,7 +117,7 @@ class TestStreamingLogs(AsyncTestCase):
proto.transport.loseConnection() proto.transport.loseConnection()
yield proto.is_closed yield proto.is_closed
self.assertThat(len(messages), Equals(3)) self.assertThat(len(messages), Equals(3), messages)
self.assertThat(messages[0]["action_type"], Equals("test:cli:some-exciting-action")) self.assertThat(messages[0]["action_type"], Equals("test:cli:some-exciting-action"))
self.assertThat(messages[0]["arguments"], self.assertThat(messages[0]["arguments"],
Equals(["hello", "good-\\xff-day", 123, {"a": 35}, [None]])) Equals(["hello", "good-\\xff-day", 123, {"a": 35}, [None]]))

View File

@ -1,195 +0,0 @@
"""
Bring in some Eliot updates from newer versions of Eliot than we can
depend on in Python 2. The implementations are copied from Eliot 1.14 and
only changed enough to add Python 2 compatibility.
Every API in this module (except ``eliot_json_encoder``) should be obsolete as
soon as we depend on Eliot 1.14 or newer.
When that happens:
* replace ``capture_logging``
with ``partial(eliot.testing.capture_logging, encoder_=eliot_json_encoder)``
* replace ``validateLogging``
with ``partial(eliot.testing.validateLogging, encoder_=eliot_json_encoder)``
* replace ``MemoryLogger``
with ``partial(eliot.MemoryLogger, encoder=eliot_json_encoder)``
Ported to Python 3.
"""
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
from future.utils import PY2
if PY2:
from builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
import json as pyjson
from functools import wraps, partial
from eliot import (
MemoryLogger as _MemoryLogger,
)
from eliot.testing import (
check_for_errors,
swap_logger,
)
from .jsonbytes import AnyBytesJSONEncoder
# There are currently a number of log messages that include non-UTF-8 bytes.
# Allow these, at least for now. Later when the whole test suite has been
# converted to our SyncTestCase or AsyncTestCase it will be easier to turn
# this off and then attribute log failures to specific codepaths so they can
# be fixed (and then not regressed later) because those instances will result
# in test failures instead of only garbage being written to the eliot log.
eliot_json_encoder = AnyBytesJSONEncoder
class _CustomEncoderMemoryLogger(_MemoryLogger):
"""
Override message validation from the Eliot-supplied ``MemoryLogger`` to
use our chosen JSON encoder.
This is only necessary on Python 2 where we use an old version of Eliot
that does not parameterize the encoder.
"""
def __init__(self, encoder=eliot_json_encoder):
"""
@param encoder: A JSONEncoder subclass to use when encoding JSON.
"""
self._encoder = encoder
super(_CustomEncoderMemoryLogger, self).__init__()
def _validate_message(self, dictionary, serializer):
"""Validate an individual message.
As a side-effect, the message is replaced with its serialized contents.
@param dictionary: A message C{dict} to be validated. Might be mutated
by the serializer!
@param serializer: C{None} or a serializer.
@raises TypeError: If a field name is not unicode, or the dictionary
fails to serialize to JSON.
@raises eliot.ValidationError: If serializer was given and validation
failed.
"""
if serializer is not None:
serializer.validate(dictionary)
for key in dictionary:
if not isinstance(key, str):
if isinstance(key, bytes):
key.decode("utf-8")
else:
raise TypeError(dictionary, "%r is not unicode" % (key,))
if serializer is not None:
serializer.serialize(dictionary)
try:
pyjson.dumps(dictionary, cls=self._encoder)
except Exception as e:
raise TypeError("Message %s doesn't encode to JSON: %s" % (dictionary, e))
if PY2:
MemoryLogger = partial(_CustomEncoderMemoryLogger, encoder=eliot_json_encoder)
else:
MemoryLogger = partial(_MemoryLogger, encoder=eliot_json_encoder)
def validateLogging(
assertion, *assertionArgs, **assertionKwargs
):
"""
Decorator factory for L{unittest.TestCase} methods to add logging
validation.
1. The decorated test method gets a C{logger} keyword argument, a
L{MemoryLogger}.
2. All messages logged to this logger will be validated at the end of
the test.
3. Any unflushed logged tracebacks will cause the test to fail.
For example:
from unittest import TestCase
from eliot.testing import assertContainsFields, validateLogging
class MyTests(TestCase):
def assertFooLogging(self, logger):
assertContainsFields(self, logger.messages[0], {"key": 123})
@param assertion: A callable that will be called with the
L{unittest.TestCase} instance, the logger and C{assertionArgs} and
C{assertionKwargs} once the actual test has run, allowing for extra
logging-related assertions on the effects of the test. Use L{None} if you
want the cleanup assertions registered but no custom assertions.
@param assertionArgs: Additional positional arguments to pass to
C{assertion}.
@param assertionKwargs: Additional keyword arguments to pass to
C{assertion}.
@param encoder_: C{json.JSONEncoder} subclass to use when validating JSON.
"""
encoder_ = assertionKwargs.pop("encoder_", eliot_json_encoder)
def decorator(function):
@wraps(function)
def wrapper(self, *args, **kwargs):
skipped = False
kwargs["logger"] = logger = MemoryLogger(encoder=encoder_)
self.addCleanup(check_for_errors, logger)
# TestCase runs cleanups in reverse order, and we want this to
# run *before* tracebacks are checked:
if assertion is not None:
self.addCleanup(
lambda: skipped
or assertion(self, logger, *assertionArgs, **assertionKwargs)
)
try:
return function(self, *args, **kwargs)
except self.skipException:
skipped = True
raise
return wrapper
return decorator
# PEP 8 variant:
validate_logging = validateLogging
def capture_logging(
assertion, *assertionArgs, **assertionKwargs
):
"""
Capture and validate all logging that doesn't specify a L{Logger}.
See L{validate_logging} for details on the rest of its behavior.
"""
encoder_ = assertionKwargs.pop("encoder_", eliot_json_encoder)
def decorator(function):
@validate_logging(
assertion, *assertionArgs, encoder_=encoder_, **assertionKwargs
)
@wraps(function)
def wrapper(self, *args, **kwargs):
logger = kwargs["logger"]
previous_logger = swap_logger(logger)
def cleanup():
swap_logger(previous_logger)
self.addCleanup(cleanup)
return function(self, *args, **kwargs)
return wrapper
return decorator

View File

@ -3,17 +3,6 @@ Tools aimed at the interaction between Tahoe-LAFS implementation and Eliot.
Ported to Python 3. Ported to Python 3.
""" """
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
from __future__ import (
unicode_literals,
print_function,
absolute_import,
division,
)
__all__ = [ __all__ = [
"MemoryLogger", "MemoryLogger",
@ -26,11 +15,6 @@ __all__ = [
"capture_logging", "capture_logging",
] ]
from future.utils import PY2
if PY2:
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
from six import ensure_text
from sys import ( from sys import (
stdout, stdout,
) )
@ -42,6 +26,7 @@ from logging import (
) )
from json import loads from json import loads
from six import ensure_text
from zope.interface import ( from zope.interface import (
implementer, implementer,
) )
@ -61,6 +46,11 @@ from eliot import (
write_traceback, write_traceback,
start_action, start_action,
) )
from eliot.testing import (
MemoryLogger,
capture_logging,
)
from eliot._validation import ( from eliot._validation import (
ValidationError, ValidationError,
) )
@ -87,11 +77,8 @@ from twisted.internet.defer import (
) )
from twisted.application.service import Service from twisted.application.service import Service
from ._eliot_updates import ( from .jsonbytes import AnyBytesJSONEncoder
MemoryLogger,
eliot_json_encoder,
capture_logging,
)
def validateInstanceOf(t): def validateInstanceOf(t):
""" """
@ -309,7 +296,7 @@ class _DestinationParser(object):
rotateLength=rotate_length, rotateLength=rotate_length,
maxRotatedFiles=max_rotated_files, maxRotatedFiles=max_rotated_files,
) )
return lambda reactor: FileDestination(get_file(), eliot_json_encoder) return lambda reactor: FileDestination(get_file(), encoder=AnyBytesJSONEncoder)
_parse_destination_description = _DestinationParser().parse _parse_destination_description = _DestinationParser().parse

View File

@ -61,6 +61,9 @@ class UTF8BytesJSONEncoder(json.JSONEncoder):
""" """
A JSON encoder than can also encode UTF-8 encoded strings. A JSON encoder than can also encode UTF-8 encoded strings.
""" """
def default(self, o):
return bytes_to_unicode(False, o)
def encode(self, o, **kwargs): def encode(self, o, **kwargs):
return json.JSONEncoder.encode( return json.JSONEncoder.encode(
self, bytes_to_unicode(False, o), **kwargs) self, bytes_to_unicode(False, o), **kwargs)
@ -77,6 +80,9 @@ class AnyBytesJSONEncoder(json.JSONEncoder):
Bytes are decoded to strings using UTF-8, if that fails to decode then the Bytes are decoded to strings using UTF-8, if that fails to decode then the
bytes are quoted. bytes are quoted.
""" """
def default(self, o):
return bytes_to_unicode(True, o)
def encode(self, o, **kwargs): def encode(self, o, **kwargs):
return json.JSONEncoder.encode( return json.JSONEncoder.encode(
self, bytes_to_unicode(True, o), **kwargs) self, bytes_to_unicode(True, o), **kwargs)

20
tox.ini
View File

@ -11,6 +11,7 @@ python =
3.9: py39-coverage 3.9: py39-coverage
3.10: py310-coverage 3.10: py310-coverage
3.11: py311-coverage 3.11: py311-coverage
3.12: py312-coverage
pypy-3.8: pypy38 pypy-3.8: pypy38
pypy-3.9: pypy39 pypy-3.9: pypy39
@ -18,11 +19,14 @@ python =
twisted = 1 twisted = 1
[tox] [tox]
envlist = typechecks,codechecks,py{38,39,310,311}-{coverage},pypy27,pypy38,pypy39,integration envlist = typechecks,codechecks,py{38,39,310,311,312}-{coverage},pypy27,pypy38,pypy39,integration
minversion = 2.4 minversion = 4
[testenv] [testenv]
passenv = TAHOE_LAFS_* PIP_* SUBUNITREPORTER_* USERPROFILE HOMEDRIVE HOMEPATH # Install code the real way, for maximum realism.
usedevelop = False
passenv = TAHOE_LAFS_*,PIP_*,SUBUNITREPORTER_*,USERPROFILE,HOMEDRIVE,HOMEPATH,COLUMNS
deps = deps =
# We pull in certify *here* to avoid bug #2913. Basically if a # We pull in certify *here* to avoid bug #2913. Basically if a
# `setup_requires=...` causes a package to be installed (with setuptools) # `setup_requires=...` causes a package to be installed (with setuptools)
@ -40,10 +44,6 @@ deps =
# with the above pins. # with the above pins.
certifi certifi
# We add usedevelop=False because testing against a true installation gives
# more useful results.
usedevelop = False
extras = extras =
# Get general testing environment dependencies so we can run the tests # Get general testing environment dependencies so we can run the tests
# how we like. # how we like.
@ -56,6 +56,7 @@ setenv =
# Define TEST_SUITE in the environment as an aid to constructing the # Define TEST_SUITE in the environment as an aid to constructing the
# correct test command below. # correct test command below.
TEST_SUITE = allmydata TEST_SUITE = allmydata
COLUMNS = 80
commands = commands =
# As an aid to debugging, dump all of the Python packages and their # As an aid to debugging, dump all of the Python packages and their
@ -81,6 +82,7 @@ commands =
coverage: coverage xml coverage: coverage xml
[testenv:integration] [testenv:integration]
usedevelop = False
basepython = python3 basepython = python3
platform = mylinux: linux platform = mylinux: linux
mymacos: darwin mymacos: darwin
@ -140,7 +142,7 @@ commands = mypy src
[testenv:draftnews] [testenv:draftnews]
passenv = TAHOE_LAFS_* PIP_* SUBUNITREPORTER_* USERPROFILE HOMEDRIVE HOMEPATH passenv = TAHOE_LAFS_*,PIP_*,SUBUNITREPORTER_*,USERPROFILE,HOMEDRIVE,HOMEPATH,COLUMNS
deps = deps =
# see comment in [testenv] about "certifi" # see comment in [testenv] about "certifi"
certifi certifi
@ -150,7 +152,7 @@ commands =
[testenv:news] [testenv:news]
# On macOS, git invoked from Tox needs $HOME. # On macOS, git invoked from Tox needs $HOME.
passenv = TAHOE_LAFS_* PIP_* SUBUNITREPORTER_* USERPROFILE HOMEDRIVE HOMEPATH HOME passenv = TAHOE_LAFS_*,PIP_*,SUBUNITREPORTER_*,USERPROFILE,HOMEDRIVE,HOMEPATH,COLUMNS
whitelist_externals = whitelist_externals =
git git
deps = deps =