summaryrefslogtreecommitdiff
path: root/src/leap/mail/cred.py
blob: 7eab1f0434d322996d265bcb3808bbbf9d515876 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
# -*- coding: utf-8 -*-
# cred.py
# Copyright (C) 2015 LEAP
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
"""
Credentials handling.
"""

from zope.interface import implementer
from twisted.cred.checkers import ICredentialsChecker
from twisted.cred.credentials import IUsernamePassword
from twisted.cred.error import UnauthorizedLogin
from twisted.internet import defer


@implementer(ICredentialsChecker)
class LocalSoledadTokenChecker(object):

    """
    A Credentials Checker for a LocalSoledad store.

    It checks that:

    1) The Local SoledadStorage has been correctly unlocked for the given
       user. This currently means that the right passphrase has been passed
       to the Local SoledadStorage.

    2) The password passed in the credentials matches whatever token has
       been stored in the local encrypted SoledadStorage, associated to the
       Protocol that is requesting the authentication.
    """

    credentialInterfaces = (IUsernamePassword,)
    service = None

    def __init__(self, soledad_sessions):
        """
        :param soledad_sessions: a dict-like object, containing instances
                                 of a Store (soledad instances), indexed by
                                 userid.
        """
        self._soledad_sessions = soledad_sessions

    def requestAvatarId(self, credentials):
        if self.service is None:
            raise NotImplementedError(
                "this checker has not defined its service name")
        username, password = credentials.username, credentials.password
        d = self.checkSoledadToken(username, password, self.service)
        d.addErrback(lambda f: defer.fail(UnauthorizedLogin()))
        return d

    def checkSoledadToken(self, username, password, service):
        soledad = self._soledad_sessions.get(username)
        if not soledad:
            return defer.fail(Exception("No soledad"))

        def match_token(token):
            if token is None:
                raise RuntimeError('no token')
            if token == password:
                return username
            else:
                raise RuntimeError('bad token')

        d = soledad.get_or_create_service_token(service)
        d.addCallback(match_token)
        return d