2019-06-14 15:56:02 +00:00
|
|
|
"""
|
|
|
|
Testtools-style matchers useful to the Tahoe-LAFS test suite.
|
|
|
|
"""
|
|
|
|
|
2019-06-14 19:48:28 +00:00
|
|
|
import attr
|
|
|
|
|
2019-06-14 15:56:02 +00:00
|
|
|
from testtools.matchers import (
|
2019-06-14 19:48:28 +00:00
|
|
|
Mismatch,
|
2019-06-14 15:56:02 +00:00
|
|
|
AfterPreprocessing,
|
|
|
|
MatchesStructure,
|
|
|
|
MatchesDict,
|
2019-06-28 15:51:32 +00:00
|
|
|
MatchesListwise,
|
2019-06-14 15:56:02 +00:00
|
|
|
Always,
|
|
|
|
Equals,
|
|
|
|
)
|
|
|
|
|
|
|
|
from foolscap.furl import (
|
|
|
|
decode_furl,
|
|
|
|
)
|
|
|
|
|
|
|
|
from allmydata.util import (
|
|
|
|
base32,
|
|
|
|
)
|
2019-06-14 19:48:28 +00:00
|
|
|
from allmydata.node import (
|
|
|
|
read_config,
|
|
|
|
)
|
|
|
|
from allmydata.crypto import (
|
|
|
|
ed25519,
|
|
|
|
error,
|
|
|
|
)
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class MatchesNodePublicKey(object):
|
|
|
|
"""
|
|
|
|
Match an object representing the node's private key.
|
|
|
|
|
|
|
|
To verify, the private key is loaded from the node's private config
|
|
|
|
directory at the time the match is checked.
|
|
|
|
"""
|
|
|
|
basedir = attr.ib()
|
|
|
|
|
|
|
|
def match(self, other):
|
2019-08-03 10:28:38 +00:00
|
|
|
"""
|
|
|
|
Match a private key which is the same as the private key in the node at
|
|
|
|
``self.basedir``.
|
|
|
|
|
|
|
|
:param other: A signing key (aka "private key") from
|
|
|
|
``allmydata.crypto.ed25519``. This is the key to check against
|
|
|
|
the node's key.
|
|
|
|
|
|
|
|
:return Mismatch: If the keys don't match.
|
|
|
|
"""
|
2019-06-14 19:48:28 +00:00
|
|
|
config = read_config(self.basedir, u"tub.port")
|
|
|
|
privkey_bytes = config.get_private_config("node.privkey")
|
|
|
|
private_key = ed25519.signing_keypair_from_string(privkey_bytes)[0]
|
|
|
|
signature = ed25519.sign_data(private_key, b"")
|
|
|
|
other_public_key = ed25519.verifying_key_from_signing_key(other)
|
|
|
|
try:
|
|
|
|
ed25519.verify_signature(other_public_key, signature, b"")
|
|
|
|
except error.BadSignature:
|
|
|
|
return Mismatch("The signature did not verify.")
|
2019-06-14 15:56:02 +00:00
|
|
|
|
|
|
|
|
2019-07-03 19:11:06 +00:00
|
|
|
def matches_storage_announcement(basedir, anonymous=True, options=None):
|
2019-06-14 15:56:02 +00:00
|
|
|
"""
|
2019-07-03 19:11:06 +00:00
|
|
|
Match a storage announcement.
|
|
|
|
|
|
|
|
:param bytes basedir: The path to the node base directory which is
|
|
|
|
expected to emit the announcement. This is used to determine the key
|
|
|
|
which is meant to sign the announcement.
|
|
|
|
|
|
|
|
:param bool anonymous: If True, matches a storage announcement containing
|
|
|
|
an anonymous access fURL. Otherwise, fails to match such an
|
|
|
|
announcement.
|
|
|
|
|
|
|
|
:param list[matcher]|NoneType options: If a list, matches a storage
|
|
|
|
announcement containing a list of storage plugin options matching the
|
|
|
|
elements of the list. If None, fails to match an announcement with
|
|
|
|
storage plugin options.
|
|
|
|
|
|
|
|
:return: A matcher with the requested behavior.
|
2019-06-14 15:56:02 +00:00
|
|
|
"""
|
2019-06-28 15:51:32 +00:00
|
|
|
announcement = {
|
|
|
|
u"permutation-seed-base32": matches_base32(),
|
|
|
|
}
|
2019-07-03 19:11:06 +00:00
|
|
|
if anonymous:
|
|
|
|
announcement[u"anonymous-storage-FURL"] = matches_furl()
|
2019-06-28 15:51:32 +00:00
|
|
|
if options:
|
|
|
|
announcement[u"storage-options"] = MatchesListwise(options)
|
2019-06-14 15:56:02 +00:00
|
|
|
return MatchesStructure(
|
|
|
|
# Has each of these keys with associated values that match
|
2019-06-28 15:51:32 +00:00
|
|
|
service_name=Equals(u"storage"),
|
|
|
|
ann=MatchesDict(announcement),
|
2019-06-14 19:48:28 +00:00
|
|
|
signing_key=MatchesNodePublicKey(basedir),
|
2019-06-14 15:56:02 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def matches_furl():
|
|
|
|
"""
|
|
|
|
Match any Foolscap fURL byte string.
|
|
|
|
"""
|
|
|
|
return AfterPreprocessing(decode_furl, Always())
|
|
|
|
|
|
|
|
|
|
|
|
def matches_base32():
|
|
|
|
"""
|
|
|
|
Match any base32 encoded byte string.
|
|
|
|
"""
|
|
|
|
return AfterPreprocessing(base32.a2b, Always())
|
2019-06-14 20:34:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class MatchesSameElements(object):
|
|
|
|
"""
|
|
|
|
Match if the two-tuple value given contains two elements that are equal to
|
|
|
|
each other.
|
|
|
|
"""
|
|
|
|
def match(self, value):
|
|
|
|
left, right = value
|
|
|
|
return Equals(left).match(right)
|