Source code for user.auth.clients.kerberos

from __future__ import annotations

import kerberos  # type:ignore
import os

from attr import attrs, attrib
from contextlib import contextmanager
from webob.exc import HTTPUnauthorized


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Iterator
    from onegov.core.request import CoreRequest
    from webob import Response


@attrs()
[docs] class KerberosClient: """ Kerberos is a computer-network authentication protocol that works on the basis of tickets to allow nodes communicating over a non-secure network to prove their identity to one another in a secure manner. """
[docs] keytab: str = attrib()
[docs] hostname: str = attrib()
[docs] service: str = attrib()
[docs] def try_configuration(self) -> None: """ Tries to use the configuration to get the principal. If this fails with an exception, the client was not configured corectly, so this is a good way to check for that. """ with self.context(): kerberos.getServerPrincipalDetails(self.service, self.hostname)
@contextmanager
[docs] def context(self) -> Iterator[None]: """ Runs the block inside the context manager with the keytab set to the provider's keytab. All functions that interact with kerberos must be run inside this context. For convenience, this context returns the kerberos module when invoked. """ previous = os.environ.pop('KRB5_KTNAME', None) os.environ['KRB5_KTNAME'] = self.keytab yield if previous is not None: os.environ['KRB5_KTNAME'] = previous
[docs] def authenticated_username( self, request: CoreRequest ) -> Response | str | None: """ Authenticates the given request using Kerberos. The kerberos handshake is as follows: 1. An HTTPUnauthorized response (401) is returned, with the WWW-Authenticate header set to "Negotiate" 2. The client sends a request with the Authorization header set to the kerberos ticket. The result is an authenticated username or None. Note that this username is a username separate from our users table (in most cases). The kerberos environment defines this username and it is most likely the Windows login username. """ # extract the token token = request.headers.get('Authorization') token = token and ''.join(token.split()[1:]).strip() def with_header( response: Response, include_token: bool = True ) -> Response: if include_token and token: negotiate = f'Negotiate {token}' else: negotiate = 'Negotiate' response.headers['WWW-Authenticate'] = negotiate return response def negotiate() -> Response: # only mirror the token back, if it is valid, which is never # the case in the negotiate step return with_header(HTTPUnauthorized(), include_token=False) # ask for a token if not token: return negotiate() # verify the token with self.context(): # initialization step result, state = kerberos.authGSSServerInit(self.service) if result != kerberos.AUTH_GSS_COMPLETE: return negotiate() # challenge step result = kerberos.authGSSServerStep(state, token) if result != kerberos.AUTH_GSS_COMPLETE: return negotiate() # extract the final token token = kerberos.authGSSServerResponse(state) # include the token in the response request.after(with_header) # extract the user if possible return kerberos.authGSSServerUserName(state) or None