File, cache, and repository abstractions for AI agents
Project description
kiarina-agi-file
English | 日本語
[!NOTE] What is this? A package for storing AI agent files in a local filesystem and assets in a pluggable asset store, with safe retrieval through access policies and a local cache.
Dependencies
Required Dependencies
| Package | Version | License |
|---|---|---|
| kiarina-agi-base | >=2.4.0 |
MIT |
| kiarina-lib-google | >=2.3.1 |
MIT |
| kiarina-utils-app | >=2.4.0 |
MIT |
| kiarina-utils-common | >=2.3.0 |
MIT |
| kiarina-utils-file | >=2.3.1 |
MIT |
| Pydantic | >=2.11.7 |
MIT |
| pydantic-settings | >=2.10.1 |
MIT |
| pydantic-settings-manager | >=3.2.0 |
MIT |
Optional Dependencies
asset-repository-gcs
Used by the Google Cloud Storage asset repository.
| Package | Version | License |
|---|---|---|
| google-cloud-storage | >=3.4.0 |
Apache-2.0 |
Installation
For local filesystem storage only:
pip install kiarina-agi-file
To also use Google Cloud Storage:
pip install "kiarina-agi-file[asset-repository-gcs]"
Features
- Local file storage Stores files in application data and cache directories and restricts accessible paths with regular expressions.
- Pluggable asset storage Provides an async API independent of the asset store implementation and supports implementations registered by import path.
- Local asset cache Caches retrieved content by URI and fetches it again after its TTL expires.
- Unified file resolution
Detects URIs and local file paths and returns either as a
FileBlob.
Store Local Files
Configure the application name first. The default data and cache paths are generated from the application's user directories and RunContext.agent_id.
from kiarina.agi.base.run_context import RunContext
from kiarina.agi.file.local_repository import create_local_repository
from kiarina.utils.app import configure
configure("example-app", "example-author")
run_context = RunContext()
repository = create_local_repository(run_context)
file_path = repository.generate_data_path("documents/example.txt")
file_blob = await repository.set(file_path, "text/plain", b"hello")
loaded = await repository.get(file_path)
FilePathPolicy.allowed_file_path_patterns uses full regular-expression matches. Its default allows every path, so applications that handle untrusted paths should restrict it explicitly.
Store Assets
The default local preset uses the local filesystem. Stored and retrieved values are FileBlob objects in the local asset cache.
from kiarina.agi.base.run_context import RunContext
from kiarina.agi.file.asset_repository import create_asset_repository
from kiarina.utils.app import configure
configure("example-app", "example-author")
repository = create_asset_repository(RunContext())
uri = repository.generate_data_uri("documents/example.txt")
cached_file = await repository.set(uri, "text/plain", b"hello")
loaded = await repository.get(uri)
get(..., ignore_cache=True) skips reading the cache, retrieves the asset from the backend, and refreshes the cache. delete() removes the asset from both the backend and the cache.
Use an Asset Store Implementation
Select an asset store as a preset or custom implementation. This package includes the local and gcs presets.
For example, to use Google Cloud Storage, select the gcs preset and configure a policy for gs:// URIs. {organization_id}, {user_id}, and {agent_id} are expanded from the RunContext.
export KIARINA_AGI_ASSET_REPOSITORY_DEFAULT=gcs
export KIARINA_AGI_ASSET_REPOSITORY_URI_POLICY='{
"allowed_uri_patterns": ["gs://example-bucket/{agent_id}/.*"],
"data_dir_uri_template": "gs://example-bucket/{agent_id}/data",
"cache_dir_uri_template": "gs://example-bucket/{agent_id}/cache"
}'
Credentials are resolved from kiarina-lib-google settings. To use a specific settings key:
export KIARINA_AGI_ASSET_REPOSITORY_IMPL_GCS_GOOGLE_AUTH_SETTINGS_KEY=service_account
Resolve a URI or File Path
from kiarina.agi.file.file import get_file_blob
file_blob = await get_file_blob(
"gs://example-bucket/agent-1/data/example.txt",
run_context=run_context,
)
URIs use the configured asset repository; other strings use the local repository. The function returns None when the target does not exist.
API Reference
kiarina.agi.file.file
from kiarina.agi.file.file import FilePath, URIOrFilePath, get_file_blob
async def get_file_blob(
uri_or_file_path: URIOrFilePath,
*,
run_context: RunContext,
) -> FileBlob | None: ...
FilePath: TypeAlias = str
URIOrFilePath: TypeAlias = str
kiarina.agi.file.local_repository
from kiarina.agi.file.local_repository import (
FilePathPolicy,
LocalArea,
LocalRepository,
LocalRepositorySettings,
create_local_repository,
resolve_file_path,
settings_manager,
)
def create_local_repository(run_context: RunContext) -> LocalRepository: ...
def resolve_file_path(file_path: str | os.PathLike[str]) -> str: ...
class LocalRepository:
def __init__(
self,
settings: LocalRepositorySettings,
*,
run_context: RunContext,
) -> None: ...
@property
def template_variables(self) -> dict[str, str]: ...
@property
def file_path_policy(self) -> FilePathPolicy: ...
@property
def data_dir(self) -> str: ...
@property
def cache_dir(self) -> str: ...
def generate_data_path(self, relative_path: str | os.PathLike[str]) -> str: ...
def generate_cache_path(self, relative_path: str | os.PathLike[str]) -> str: ...
def generate_time_based_dir_path(
self,
*,
sub_dir_path: str | os.PathLike[str] = "log",
area: LocalArea = "data",
) -> str: ...
def generate_time_based_file_path(
self,
file_name: str,
*,
sub_dir_path: str | os.PathLike[str] = "log",
area: LocalArea = "data",
) -> str: ...
def is_valid_file_path(self, file_path: str | os.PathLike[str]) -> bool: ...
def validate_file_path(self, file_path: str | os.PathLike[str]) -> None: ...
async def exists(self, file_path: str | os.PathLike[str]) -> bool: ...
async def get(self, file_path: str | os.PathLike[str]) -> FileBlob | None: ...
async def set(
self,
file_path: str | os.PathLike[str],
mime_type: str,
raw_data: bytes,
*,
only_not_exists: bool = False,
) -> FileBlob: ...
async def delete(self, file_path: str | os.PathLike[str]) -> None: ...
class FilePathPolicy(BaseModel):
allowed_file_path_patterns: list[str] = [".*"]
data_dir_path_template: str = "{user_data_dir}/agents/{agent_id}"
cache_dir_path_template: str = "{user_cache_dir}/agents/{agent_id}"
class LocalRepositorySettings(BaseSettings):
file_path_policy: FilePathPolicy = FilePathPolicy()
LocalArea = Literal["data", "cache"]
settings_manager: SettingsManager[LocalRepositorySettings]
resolve_file_path() expands environment variables and ~, then returns an absolute path. Operations on a disallowed path raise PermissionError.
kiarina.agi.file.asset_cache
from kiarina.agi.file.asset_cache import (
AssetCache,
AssetCacheSettings,
create_asset_cache,
settings_manager,
)
def create_asset_cache(run_context: RunContext) -> AssetCache: ...
class AssetCache:
def __init__(
self,
settings: AssetCacheSettings,
*,
run_context: RunContext,
) -> None: ...
@property
def local_repository(self) -> LocalRepository: ...
async def get(self, uri: str) -> FileBlob | None: ...
async def set(self, uri: str, mime_type: str, raw_data: bytes) -> FileBlob: ...
async def delete(self, uri: str) -> None: ...
class AssetCacheSettings(BaseSettings):
hash_algorithm: str = "sha256"
cache_ttl: int = 86400
settings_manager: SettingsManager[AssetCacheSettings]
kiarina.agi.file.asset_repository
from kiarina.agi.file.asset_repository import (
AssetArea,
AssetRepository,
AssetRepositoryName,
AssetRepositorySettings,
AssetRepositorySpecifier,
BaseAssetRepository,
CachedFileBlob,
URIPolicy,
asset_repository_registry,
create_asset_repository,
settings_manager,
)
def create_asset_repository(run_context: RunContext) -> AssetRepository: ...
class AssetRepository(Protocol):
uri_policy: URIPolicy
run_context: RunContext
@property
def asset_cache(self) -> AssetCache: ...
def generate_data_uri(self, relative_path: str) -> str: ...
def generate_cache_uri(self, relative_path: str) -> str: ...
def generate_time_based_uri(
self,
file_name: str | None = None,
*,
sub_dir_path: str = "log",
area: AssetArea = "data",
) -> str: ...
def is_valid_uri(self, uri: str) -> bool: ...
def validate_uri(self, uri: str) -> None: ...
async def exists(self, uri: str) -> bool: ...
async def get(
self,
uri: str,
*,
ignore_cache: bool = False,
) -> CachedFileBlob | None: ...
async def set(
self,
uri: str,
mime_type: str,
raw_data: bytes,
*,
only_not_exists: bool = False,
) -> CachedFileBlob: ...
async def delete(self, uri: str) -> None: ...
async def generate_download_url(
self,
uri: str,
*,
expire_seconds: int = 86400,
) -> str: ...
class BaseAssetRepository(AssetRepository):
def __init__(self) -> None: ...
@property
def uri_policy(self) -> URIPolicy: ...
@uri_policy.setter
def uri_policy(self, uri_policy: URIPolicy) -> None: ...
@property
def run_context(self) -> RunContext: ...
@run_context.setter
def run_context(self, run_context: RunContext) -> None: ...
@property
def template_variables(self) -> dict[str, str]: ...
@property
def asset_cache(self) -> AssetCache: ...
@property
def data_uri(self) -> str: ...
@property
def cache_uri(self) -> str: ...
# Implements every AssetRepository method.
class URIPolicy(BaseModel):
allowed_uri_patterns: list[str] = []
data_dir_uri_template: str = "{invalid}"
cache_dir_uri_template: str = "{invalid}"
class AssetRepositorySettings(BaseSettings):
default: AssetRepositorySpecifier = "local"
presets: dict[AssetRepositoryName, ImportPath] = <local and gcs presets>
customs: dict[AssetRepositoryName, ImportPath] = {}
uri_policy: URIPolicy = <local asset policy>
AssetArea = Literal["data", "cache"]
AssetRepositoryName: TypeAlias = str
AssetRepositorySpecifier: TypeAlias = AssetRepositoryName | str
CachedFileBlob: TypeAlias = FileBlob
asset_repository_registry: ComponentRegistry[AssetRepository]
settings_manager: SettingsManager[AssetRepositorySettings]
AssetRepositorySpecifier is a repository name or a string in the "{name}?{config}" form. An absent URI pattern raises ValueError; operations on a disallowed URI raise PermissionError.
kiarina.agi.file.asset_repository_impl.local
from kiarina.agi.file.asset_repository_impl.local import LocalAssetRepository
class LocalAssetRepository(BaseAssetRepository):
@property
def template_variables(self) -> dict[str, str]: ...
@property
def local_repository(self) -> LocalRepository: ...
LocalAssetRepository implements the BaseAssetRepository storage operations on the local filesystem. Its download URL is a file:// URL.
kiarina.agi.file.asset_repository_impl.gcs
Importing this module requires the asset-repository-gcs extra.
from kiarina.agi.file.asset_repository_impl.gcs import (
GCSAssetRepository,
GCSAssetRepositorySettings,
create_gcs_asset_repository,
settings_manager,
)
def create_gcs_asset_repository(**kwargs: Any) -> GCSAssetRepository: ...
class GCSAssetRepository(BaseAssetRepository):
def __init__(self, settings: GCSAssetRepositorySettings) -> None: ...
@property
def client(self) -> google.cloud.storage.Client: ...
class GCSAssetRepositorySettings(BaseSettings):
google_auth_settings_key: str | None = None
settings_manager: SettingsManager[GCSAssetRepositorySettings]
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file kiarina_agi_file-2.5.0.tar.gz.
File metadata
- Download URL: kiarina_agi_file-2.5.0.tar.gz
- Upload date:
- Size: 19.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6f109e063f59137bc882b83e3ba18305d6ba59a5fc0c378b54077e87c8c84401
|
|
| MD5 |
600a9ecd5f377e5b3b253e43a560da07
|
|
| BLAKE2b-256 |
698b88b293bd1da70cb6cc9ff0aed9a4847ca6dffbb42e5787769d65d0328c7e
|
Provenance
The following attestation bundles were made for kiarina_agi_file-2.5.0.tar.gz:
Publisher:
release-pypi.yml on kiarina/kiarina-python
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kiarina_agi_file-2.5.0.tar.gz -
Subject digest:
6f109e063f59137bc882b83e3ba18305d6ba59a5fc0c378b54077e87c8c84401 - Sigstore transparency entry: 2057546681
- Sigstore integration time:
-
Permalink:
kiarina/kiarina-python@d9fc18c7a6cca99acebac9721539daf0fe1d9e23 -
Branch / Tag:
refs/tags/v2.5.0 - Owner: https://github.com/kiarina
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release-pypi.yml@d9fc18c7a6cca99acebac9721539daf0fe1d9e23 -
Trigger Event:
push
-
Statement type:
File details
Details for the file kiarina_agi_file-2.5.0-py3-none-any.whl.
File metadata
- Download URL: kiarina_agi_file-2.5.0-py3-none-any.whl
- Upload date:
- Size: 25.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8d7366d7518d2fa2965c3bbd38a762803b5cd0fbc61113510eced5870db6a590
|
|
| MD5 |
2c65014e62bc46d92fa3e3d2b6d5bc68
|
|
| BLAKE2b-256 |
95fdf064cad738f335831a5d5efc59b87f76f9e78a24ec7c87f31943901fe981
|
Provenance
The following attestation bundles were made for kiarina_agi_file-2.5.0-py3-none-any.whl:
Publisher:
release-pypi.yml on kiarina/kiarina-python
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kiarina_agi_file-2.5.0-py3-none-any.whl -
Subject digest:
8d7366d7518d2fa2965c3bbd38a762803b5cd0fbc61113510eced5870db6a590 - Sigstore transparency entry: 2057547491
- Sigstore integration time:
-
Permalink:
kiarina/kiarina-python@d9fc18c7a6cca99acebac9721539daf0fe1d9e23 -
Branch / Tag:
refs/tags/v2.5.0 - Owner: https://github.com/kiarina
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release-pypi.yml@d9fc18c7a6cca99acebac9721539daf0fe1d9e23 -
Trigger Event:
push
-
Statement type: