2019-03-21 15:00:57 -04:00
|
|
|
|
|
|
|
from __future__ import (
|
|
|
|
print_function,
|
|
|
|
unicode_literals,
|
|
|
|
absolute_import,
|
|
|
|
division,
|
|
|
|
)
|
|
|
|
|
2019-03-22 13:47:32 -04:00
|
|
|
import attr
|
|
|
|
|
|
|
|
from zope.interface import (
|
|
|
|
implementer,
|
|
|
|
)
|
|
|
|
|
|
|
|
from twisted.python.failure import (
|
|
|
|
Failure,
|
|
|
|
)
|
|
|
|
from twisted.internet.defer import (
|
|
|
|
succeed,
|
|
|
|
fail,
|
|
|
|
)
|
|
|
|
from twisted.cred.credentials import (
|
|
|
|
ICredentials,
|
|
|
|
)
|
|
|
|
from twisted.cred.portal import (
|
|
|
|
IRealm,
|
|
|
|
Portal,
|
|
|
|
)
|
|
|
|
from twisted.cred.checkers import (
|
|
|
|
ANONYMOUS,
|
|
|
|
)
|
|
|
|
from twisted.cred.error import (
|
|
|
|
UnauthorizedLogin,
|
|
|
|
)
|
|
|
|
from twisted.web.iweb import (
|
|
|
|
ICredentialFactory,
|
|
|
|
)
|
2019-03-21 15:00:57 -04:00
|
|
|
from twisted.web.resource import (
|
2019-03-22 13:47:32 -04:00
|
|
|
IResource,
|
2019-03-21 15:00:57 -04:00
|
|
|
Resource,
|
|
|
|
)
|
2019-03-22 13:47:32 -04:00
|
|
|
from twisted.web.guard import (
|
|
|
|
HTTPAuthSessionWrapper,
|
|
|
|
)
|
|
|
|
|
|
|
|
from ..util.hashutil import (
|
|
|
|
timing_safe_compare,
|
|
|
|
)
|
2019-03-22 16:42:50 -04:00
|
|
|
from ..util.assertutil import (
|
|
|
|
precondition,
|
|
|
|
)
|
2019-03-21 15:00:57 -04:00
|
|
|
|
|
|
|
from .logs import (
|
|
|
|
create_log_resources,
|
|
|
|
)
|
|
|
|
|
2019-03-22 13:47:32 -04:00
|
|
|
SCHEME = b"tahoe-lafs"
|
|
|
|
|
|
|
|
class IToken(ICredentials):
|
|
|
|
def check(auth_token):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2020-12-18 10:33:30 -05:00
|
|
|
# Workaround for Shoobx/mypy-zope#26, where without suitable
|
|
|
|
# stubs for twisted classes (ICredentials), IToken does not
|
|
|
|
# appear to be an Interface. The proper fix appears to be to
|
|
|
|
# create stubs for twisted
|
|
|
|
# (https://twistedmatrix.com/trac/ticket/9717). For now,
|
|
|
|
# bypassing the inline decorator syntax works around the issue.
|
2020-11-20 13:47:23 -05:00
|
|
|
_itoken_impl = implementer(IToken)
|
|
|
|
|
|
|
|
|
|
|
|
@_itoken_impl
|
2019-03-22 13:47:32 -04:00
|
|
|
@attr.s
|
|
|
|
class Token(object):
|
|
|
|
proposed_token = attr.ib(type=bytes)
|
|
|
|
|
|
|
|
def equals(self, valid_token):
|
|
|
|
return timing_safe_compare(
|
2019-03-22 15:50:58 -04:00
|
|
|
valid_token,
|
2019-03-22 13:47:32 -04:00
|
|
|
self.proposed_token,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class TokenChecker(object):
|
|
|
|
get_auth_token = attr.ib()
|
|
|
|
|
|
|
|
credentialInterfaces = [IToken]
|
|
|
|
|
|
|
|
def requestAvatarId(self, credentials):
|
2019-03-22 16:42:50 -04:00
|
|
|
required_token = self.get_auth_token()
|
|
|
|
precondition(isinstance(required_token, bytes))
|
|
|
|
if credentials.equals(required_token):
|
2019-03-22 13:47:32 -04:00
|
|
|
return succeed(ANONYMOUS)
|
|
|
|
return fail(Failure(UnauthorizedLogin()))
|
|
|
|
|
|
|
|
|
|
|
|
@implementer(ICredentialFactory)
|
|
|
|
@attr.s
|
|
|
|
class TokenCredentialFactory(object):
|
|
|
|
scheme = SCHEME
|
|
|
|
authentication_realm = b"tahoe-lafs"
|
|
|
|
|
|
|
|
def getChallenge(self, request):
|
|
|
|
return {b"realm": self.authentication_realm}
|
|
|
|
|
|
|
|
def decode(self, response, request):
|
|
|
|
return Token(response)
|
|
|
|
|
|
|
|
|
|
|
|
@implementer(IRealm)
|
|
|
|
@attr.s
|
|
|
|
class PrivateRealm(object):
|
|
|
|
_root = attr.ib()
|
|
|
|
|
|
|
|
def _logout(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def requestAvatar(self, avatarId, mind, *interfaces):
|
|
|
|
if IResource in interfaces:
|
|
|
|
return (IResource, self._root, self._logout)
|
|
|
|
raise NotImplementedError(
|
|
|
|
"PrivateRealm supports IResource not {}".format(interfaces),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def _create_vulnerable_tree():
|
2019-03-21 15:00:57 -04:00
|
|
|
private = Resource()
|
2019-03-22 13:47:32 -04:00
|
|
|
private.putChild(b"logs", create_log_resources())
|
2019-03-21 15:00:57 -04:00
|
|
|
return private
|
2019-03-22 13:47:32 -04:00
|
|
|
|
|
|
|
|
|
|
|
def _create_private_tree(get_auth_token, vulnerable):
|
|
|
|
realm = PrivateRealm(vulnerable)
|
|
|
|
portal = Portal(realm, [TokenChecker(get_auth_token)])
|
|
|
|
return HTTPAuthSessionWrapper(portal, [TokenCredentialFactory()])
|
|
|
|
|
|
|
|
|
|
|
|
def create_private_tree(get_auth_token):
|
|
|
|
"""
|
|
|
|
Create a new resource tree that only allows requests if they include a
|
|
|
|
correct `Authorization: tahoe-lafs <api_auth_token>` header (where
|
|
|
|
`api_auth_token` matches the private configuration value).
|
|
|
|
"""
|
|
|
|
return _create_private_tree(
|
|
|
|
get_auth_token,
|
|
|
|
_create_vulnerable_tree(),
|
|
|
|
)
|