C-6: ConnectorRuntime enforces capability_mask per operation.
READ-only ACs cannot invoke MUTATE operations (wipe, lock, retire).
C-7: AC validated against database (exists, active, not expired)
before connector invocation.
C-9: Delegated AC capability bounded by delegator's capability.
C-10: Command counter uses atomic SQL increment with limit check.
M-23: expire_stale() uses same atomic SQL pattern.
H-1: Sensitive credential fields hidden from repr/logs via repr=False.
H-2: Stub backend requires ALLOW_STUB_CREDENTIALS=true to activate.
H-3: Kerberos backend raises CredentialResolutionError instead of
returning stub ticket.
H-4: Chronicle INTENT emitted before execution, RESULT after.
H-5: device_id validated as UUID before Graph API URL interpolation.
H-8: ConnectorRuntime enforces governance for all connector invocations.
Signed-off-by: Tyler King <tking@guildhouse.dev>
83 lines
3.3 KiB
Python
83 lines
3.3 KiB
Python
"""
|
|
certifi.py
|
|
~~~~~~~~~~
|
|
|
|
This module returns the installation location of cacert.pem or its contents.
|
|
"""
|
|
import sys
|
|
import atexit
|
|
|
|
def exit_cacert_ctx() -> None:
|
|
_CACERT_CTX.__exit__(None, None, None) # type: ignore[union-attr]
|
|
|
|
|
|
if sys.version_info >= (3, 11):
|
|
|
|
from importlib.resources import as_file, files
|
|
|
|
_CACERT_CTX = None
|
|
_CACERT_PATH = None
|
|
|
|
def where() -> str:
|
|
# This is slightly terrible, but we want to delay extracting the file
|
|
# in cases where we're inside of a zipimport situation until someone
|
|
# actually calls where(), but we don't want to re-extract the file
|
|
# on every call of where(), so we'll do it once then store it in a
|
|
# global variable.
|
|
global _CACERT_CTX
|
|
global _CACERT_PATH
|
|
if _CACERT_PATH is None:
|
|
# This is slightly janky, the importlib.resources API wants you to
|
|
# manage the cleanup of this file, so it doesn't actually return a
|
|
# path, it returns a context manager that will give you the path
|
|
# when you enter it and will do any cleanup when you leave it. In
|
|
# the common case of not needing a temporary file, it will just
|
|
# return the file system location and the __exit__() is a no-op.
|
|
#
|
|
# We also have to hold onto the actual context manager, because
|
|
# it will do the cleanup whenever it gets garbage collected, so
|
|
# we will also store that at the global level as well.
|
|
_CACERT_CTX = as_file(files("certifi").joinpath("cacert.pem"))
|
|
_CACERT_PATH = str(_CACERT_CTX.__enter__())
|
|
atexit.register(exit_cacert_ctx)
|
|
|
|
return _CACERT_PATH
|
|
|
|
def contents() -> str:
|
|
return files("certifi").joinpath("cacert.pem").read_text(encoding="ascii")
|
|
|
|
else:
|
|
|
|
from importlib.resources import path as get_path, read_text
|
|
|
|
_CACERT_CTX = None
|
|
_CACERT_PATH = None
|
|
|
|
def where() -> str:
|
|
# This is slightly terrible, but we want to delay extracting the
|
|
# file in cases where we're inside of a zipimport situation until
|
|
# someone actually calls where(), but we don't want to re-extract
|
|
# the file on every call of where(), so we'll do it once then store
|
|
# it in a global variable.
|
|
global _CACERT_CTX
|
|
global _CACERT_PATH
|
|
if _CACERT_PATH is None:
|
|
# This is slightly janky, the importlib.resources API wants you
|
|
# to manage the cleanup of this file, so it doesn't actually
|
|
# return a path, it returns a context manager that will give
|
|
# you the path when you enter it and will do any cleanup when
|
|
# you leave it. In the common case of not needing a temporary
|
|
# file, it will just return the file system location and the
|
|
# __exit__() is a no-op.
|
|
#
|
|
# We also have to hold onto the actual context manager, because
|
|
# it will do the cleanup whenever it gets garbage collected, so
|
|
# we will also store that at the global level as well.
|
|
_CACERT_CTX = get_path("certifi", "cacert.pem")
|
|
_CACERT_PATH = str(_CACERT_CTX.__enter__())
|
|
atexit.register(exit_cacert_ctx)
|
|
|
|
return _CACERT_PATH
|
|
|
|
def contents() -> str:
|
|
return read_text("certifi", "cacert.pem", encoding="ascii")
|