mirror of
https://github.com/Garmelon/PFERD.git
synced 2023-12-21 10:23:01 +01:00
Add a keyring authenticator
This commit is contained in:
parent
2fdf24495b
commit
6e9f8fd391
@ -154,6 +154,15 @@ This authenticator prompts the user on the console for a two-factor
|
||||
authentication token. The token is provided as password and it is not cached.
|
||||
This authenticator does not support usernames.
|
||||
|
||||
### The `keyring` authenticator
|
||||
|
||||
This authenticator uses the system keyring to store passwords. It expects a
|
||||
username in the config and will prompt *once* for the password. After that it
|
||||
receives the password from the system keyring.
|
||||
|
||||
- `username`: The username. (Required)
|
||||
- `keyring_name`: The service name PFERD uses for storing credentials. (Optional)
|
||||
|
||||
## Transformation rules
|
||||
|
||||
Transformation rules are rules for renaming and excluding files and directories.
|
||||
|
@ -3,6 +3,7 @@ from typing import Callable, Dict
|
||||
|
||||
from ..config import Config
|
||||
from .authenticator import Authenticator, AuthSection
|
||||
from .keyring_authenticator import KeyringAuthenticator, KeyringAuthSection
|
||||
from .simple import SimpleAuthenticator, SimpleAuthSection
|
||||
from .tfa import TfaAuthenticator
|
||||
|
||||
@ -17,4 +18,6 @@ AUTHENTICATORS: Dict[str, AuthConstructor] = {
|
||||
SimpleAuthenticator(n, SimpleAuthSection(s), c),
|
||||
"tfa": lambda n, s, c:
|
||||
TfaAuthenticator(n, AuthSection(s), c),
|
||||
"keyring": lambda n, s, c:
|
||||
KeyringAuthenticator(n, KeyringAuthSection(s), c)
|
||||
}
|
||||
|
56
PFERD/auth/keyring_authenticator.py
Normal file
56
PFERD/auth/keyring_authenticator.py
Normal file
@ -0,0 +1,56 @@
|
||||
from typing import Optional, Tuple
|
||||
|
||||
import keyring
|
||||
|
||||
from ..config import Config
|
||||
from ..logging import log
|
||||
from ..utils import agetpass
|
||||
from ..version import NAME
|
||||
from .authenticator import Authenticator, AuthException, AuthSection
|
||||
|
||||
|
||||
class KeyringAuthSection(AuthSection):
|
||||
def username(self) -> str:
|
||||
name = self.s.get("username")
|
||||
if name is None:
|
||||
self.missing_value("username")
|
||||
return name
|
||||
|
||||
def keyring_name(self) -> str:
|
||||
return self.s.get("keyring_name", fallback=NAME)
|
||||
|
||||
|
||||
class KeyringAuthenticator(Authenticator):
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
name: str,
|
||||
section: KeyringAuthSection,
|
||||
config: Config,
|
||||
) -> None:
|
||||
super().__init__(name, section, config)
|
||||
|
||||
self._username = section.username()
|
||||
self._password: Optional[str] = None
|
||||
self._keyring_name = section.keyring_name()
|
||||
|
||||
async def credentials(self) -> Tuple[str, str]:
|
||||
if self._password is not None:
|
||||
return self._username, self._password
|
||||
|
||||
password = keyring.get_password(self._keyring_name, self._username)
|
||||
|
||||
if not password:
|
||||
async with log.exclusive_output():
|
||||
password = await agetpass("Password: ")
|
||||
keyring.set_password(self._keyring_name, self._username, password)
|
||||
|
||||
self._password = password
|
||||
|
||||
return self._username, password
|
||||
|
||||
def invalidate_credentials(self) -> None:
|
||||
self.invalidate_password()
|
||||
|
||||
def invalidate_password(self) -> None:
|
||||
raise AuthException("Invalid password")
|
Loading…
Reference in New Issue
Block a user