Vector Vein inspired agent framework with cycle runtime, tools and memory management
Project description
vv-agent
A lightweight agent framework extracted from VectorVein's production runtime. Cycle-based execution with pluggable LLM backends, tool dispatch, memory compression, and distributed scheduling.
Architecture
Agent / RunConfig / ModelSettings
└── Runner
└── AgentRuntime
├── CycleRunner # single LLM turn: context -> completion -> tool calls
├── ToolCallRunner # tool dispatch, directive convergence
├── RuntimeHookManager # before/after hooks
├── MemoryManager # automatic history compression
└── ExecutionBackend # inline, thread, or Celery scheduling
The public SDK entry points are exported from vv_agent: Agent, Runner,
RunConfig, RunHandle, ModelSettings, function_tool, Session,
typed RunEvent objects, ApprovalProvider, ContextProvider,
RunEventStore, and the interactive session API for desktop/runtime
integrations. Extension points that live in package modules include
vv_agent.memory.MemoryProvider and vv_agent.tools.ToolExecutor.
Lower-level runtime implementation details include RuntimeTask, AgentResult,
Message, CycleRecord, and ToolCall.
Task completion is tool-driven: the agent calls task_finish or ask_user to signal terminal states. No implicit "last message = answer" heuristics.
Setup
cp local_settings.example.py local_settings.py
# Fill in your API keys and endpoints in local_settings.py
uv sync --dev
uv run pytest
Quick Start
CLI
uv run vv-agent --prompt "Summarize this framework" --backend moonshot --model kimi-k2.6
# With per-cycle logging
uv run vv-agent --prompt "Summarize this framework" --backend moonshot --model kimi-k2.6 --verbose
CLI flags: --settings-file, --backend, --model, --verbose.
Programmatic SDK
from vv_agent import Agent, RunConfig, Runner, function_tool
@function_tool
def read_order(order_id: str) -> str:
"""Read order information."""
return "order details"
agent = Agent(
name="ops",
instructions="Check facts first, then answer.",
model="kimi-k2.6",
tools=[read_order],
)
result = Runner.run_sync(agent, "Analyze order 123", run_config=RunConfig(
default_backend="moonshot",
))
print(result.status, result.final_output)
Agent.output_type can coerce JSON final output into dict, list,
dataclasses, or Pydantic-style models. Decorated tools may accept a leading
ToolContext parameter; it is passed at invocation time and omitted from the
tool JSON schema.
Streaming And Sessions
RunConfig.workspace controls the workspace for a run. RunConfig.session
accepts MemorySession, SQLiteSession, or RedisSession to persist message
history across runs.
from vv_agent import Agent, MemorySession, RunConfig, Runner
agent = Agent(name="assistant", instructions="Remember context.", model="kimi-k2.6")
session = MemorySession("thread-001")
config = RunConfig(
default_backend="moonshot",
workspace="./workspace/thread-001",
session=session,
)
Runner.run_sync(agent, "Inspect the project", run_config=config)
for event in Runner.stream_sync(agent, "Continue and report progress", run_config=config):
if event.type == "assistant_delta":
print(event.delta, end="")
Use Runner.start() when the host needs a live handle instead of blocking for
the final result. RunHandle.events() yields the same typed RunEvent stream
as Runner.stream_sync(), RunHandle.result() waits for the final
RunResult, RunHandle.cancel() cancels the run, and RunHandle.approve()
resolves pending approval requests. When the handle is attached to an
AgentSession, RunHandle.steer() queues context for the active run and
RunHandle.follow_up() queues the next session turn. Plain one-shot
Runner.start() handles do not own session queues, so those methods require an
interactive session controller.
RunConfig.event_store can persist every typed event. JsonlRunEventStore
stores event dictionaries and replays events by run_id, including child runs
whose parent_run_id points at the requested run. Raw runtime logs remain
available for compatibility through the runtime log callbacks, but RunEvent
is the primary UI and app-state contract.
The lower-level AgentRuntime API remains available for backend integrations
that need direct cycle-loop control.
Install Redis support with uv sync --extra redis or inject a Redis-compatible
client when constructing RedisSession.
App Server
Use the App Server when a desktop app, worker, IDE, or other host process needs
to drive vv-agent through a stable protocol instead of embedding the Python
SDK directly. It runs JSONL over stdio, exposes Thread / Turn / Item lifecycle
events, routes tool approval as server-to-client requests, supports
thread/read and thread/resume replay, and exports JSON Schema files for
client bindings.
uv run vv-agent app-server --listen stdio
uv run vv-agent app-server generate-json-schema --out ./app-server-schema
uv run vv-agent debug app-server send-message "hello"
Product hosts implement AppServerHost to map product profiles, workspace
context, tools, approval UI, memory, and model settings into framework
Agent and RunConfig objects. The App Server remains a runtime boundary; it
does not import product UI, account, billing, browser, or IM modules.
See docs/app-server.md for protocol details and the host migration checklists for v-claw and backend services.
Interactive Sessions
Use Runner for one-shot runs, streamed runs, and conversation history managed
by RunConfig.session. Use InteractiveAgentClient when the host application
needs a stateful, bidirectional runtime session with stable session ids,
runtime listeners, queued steering prompts, follow-up turns, cancellation, and
shared tool state. During a running session, session.active_run_handle exposes
the unified RunHandle control surface for approval, cancellation, steering,
and follow-up.
from pathlib import Path
from vv_agent import (
AgentSessionOptions,
InteractiveAgentClient,
InteractiveAgentDefinition,
)
from vv_agent.runtime.backends import ThreadBackend
client = InteractiveAgentClient(
options=AgentSessionOptions(
settings_file=Path("local_settings.py"),
default_backend="moonshot",
workspace=Path("./workspace/thread-001"),
execution_backend=ThreadBackend(max_workers=4),
)
)
session = client.create_session(
session_id="thread-001",
agent=InteractiveAgentDefinition(
description="Operate in the user's workspace and report progress.",
model="kimi-k2.6",
no_tool_policy="finish",
),
)
unsubscribe = session.subscribe(lambda event, payload: print(event, payload))
try:
run = session.prompt("Inspect the workspace")
print(run.result.status, run.result.final_answer)
finally:
unsubscribe()
Interactive sessions are additive to the normal SDK facade; they do not
reintroduce the old 0.1 AgentSDKClient or AgentSDKOptions names.
Agent As Tool, Handoff, And Policy
Use agent.as_tool() when a child agent should return a result to the parent
agent and let the parent continue. Use handoff() when control should transfer
to the target agent and the target output should finish the run.
from vv_agent import Agent, RunConfig, Runner, ToolPolicy, handoff
from vv_agent.constants import TASK_FINISH_TOOL_NAME
researcher = Agent(name="researcher", instructions="Collect facts.", model="kimi-k2.6")
writer = Agent(
name="writer",
instructions="Write from research.",
model="kimi-k2.6",
tools=[researcher.as_tool(name="research", description="Collect facts.")],
)
triage = Agent(
name="triage",
instructions="Transfer writing tasks.",
model="kimi-k2.6",
handoffs=[handoff(agent=writer, description="Use for writing.")],
)
result = Runner.run_sync(
triage,
"Write a short report.",
run_config=RunConfig(
default_backend="moonshot",
tool_policy=ToolPolicy(allowed_tools=[TASK_FINISH_TOOL_NAME, "transfer_to_writer"]),
),
)
Tools can request approval with @function_tool(needs_approval=True). By
default the run enters WAIT_USER before the tool body is called and emits a
ToolApprovalRequestedEvent. ToolPolicy(approval="never") disables that
approval gate for trusted runs.
Guardrails And Tracing
Input guardrails run before the model provider is called. Output guardrails run after a final output is available. Trace processors receive lightweight run and tool spans.
from vv_agent import Agent, GuardrailResult, RunConfig, Runner, input_guardrail
@input_guardrail
def reject_empty(ctx, input_text: str) -> GuardrailResult:
del ctx
if not input_text.strip():
return GuardrailResult.block("input is required")
return GuardrailResult.allow()
agent = Agent(
name="assistant",
instructions="Answer carefully.",
model="kimi-k2.6",
input_guardrails=[reject_empty],
)
result = Runner.run_sync(
agent,
"Summarize this project.",
run_config=RunConfig(default_backend="moonshot", tracing={"workflow_name": "summary"}),
)
Shell Runtime Configuration (Windows)
bash runtime defaults are a startup/session configuration, not tool-call arguments.
- Run defaults: pass
bash_shell,windows_shell_priority, andbash_envthroughRunConfig.metadata. - Per-agent defaults: put the same keys in
Agent.metadata. - Recommended Windows priority:
["git-bash", "powershell", "cmd"] - On Windows, bash-tool child processes default
PYTHONUTF8=1andPYTHONIOENCODING=utf-8unless already overridden via the parent environment orbash_env. - On Windows, bash-tool child processes are launched with hidden-console flags so GUI hosts can run
bash/powershellcommands without flashing a terminal window. Runner.run_sync(...)andRunner.stream_sync(...)both inherit compiled shell metadata.- The
bashtool schema description includes a runtime shell hint (resolved shell kind + invocation prefix), so the model sees which shell command style is expected before calling the tool. - The runtime shell hint is frozen per task/session-run to keep tool schemas stable across cycles and preserve LLM prompt cache efficiency.
- Runner/CLI-generated runtime tasks attach structured
system_prompt_sectionsmetadata to the system message when prompt sections are available, so Anthropic prompt-cache breakpoints can keep the stable prompt prefix hot while treating current time and session-memory blocks as volatile.
from vv_agent import Agent, RunConfig, Runner
agent = Agent(
name="desktop",
instructions="Desktop helper",
model="kimi-k2.6",
metadata={"bash_env": {"HTTP_PROXY": "http://127.0.0.1:7890"}},
)
result = Runner.run_sync(
agent,
"Check the workspace.",
run_config=RunConfig(
default_backend="moonshot",
metadata={
"windows_shell_priority": ["git-bash", "powershell", "cmd"],
"bash_env": {"PIP_INDEX_URL": "https://pypi.tuna.tsinghua.edu.cn/simple"},
},
),
)
Execution Backends
The cycle loop is delegated to a pluggable ExecutionBackend.
| Backend | Use case |
|---|---|
InlineBackend |
Default. Synchronous, single-process. |
ThreadBackend |
Thread pool. Non-blocking submit() returns a Future. |
CeleryBackend |
Distributed. Each cycle dispatched as an independent Celery task. |
CeleryBackend
Two modes:
- Inline fallback (no
RuntimeRecipe): cycles run in-process, same asInlineBackend. - Distributed (with
RuntimeRecipe): each cycle is a Celery task. Workers rebuild theAgentRuntimefrom the recipe and load state from a sharedStateStore(SQLite or Redis).
from vv_agent.runtime.backends.celery import CeleryBackend, RuntimeRecipe, register_cycle_task
register_cycle_task(celery_app)
recipe = RuntimeRecipe(
settings_file="local_settings.py",
backend="moonshot",
model="kimi-k2.6",
workspace="./workspace",
)
backend = CeleryBackend(celery_app=app, state_store=store, runtime_recipe=recipe)
runtime = AgentRuntime(llm_client=llm, tool_registry=registry, execution_backend=backend)
Install celery extras: uv sync --extra celery.
Cancellation and Streaming
from vv_agent.runtime import CancellationToken, ExecutionContext
# Cancel from another thread
token = CancellationToken()
ctx = ExecutionContext(cancellation_token=token)
result = runtime.run(task, ctx=ctx)
def on_stream_event(event: dict) -> None:
if event.get("event") == "assistant_delta":
print(event.get("content_delta", ""), end="")
# Stream LLM output events, including assistant deltas and tool progress
ctx = ExecutionContext(stream_callback=on_stream_event)
result = runtime.run(task, ctx=ctx)
Runtime Log Payloads
tool_result runtime events carry full tool output in content and any structured tool payload in metadata (no implicit truncation of content).
content_preview and assistant_preview are still emitted for UI convenience.
If you need shorter previews for logs/transport, configure an explicit preview limit:
from vv_agent import RunConfig
config = RunConfig(
log_preview_chars=220, # optional: enable preview truncation explicitly
)
Workspace Backends
Workspace file I/O is delegated to a pluggable WorkspaceBackend protocol. All built-in file tools (read_file, write_file, list_files, etc.) go through this abstraction.
list_files includes built-in safety defaults for large workspaces:
- Returns at most
500paths per call by default (max_resultscan tune this, with hard cap). - Uses
ripgrep(rg) for fast local traversal when available, with automatic fallback to Python walk. workspace_grepalso usesrgfor local workspaces (with Python fallback), defaults to smart-case matching (lowercase patterns are case-insensitive; patterns with uppercase stay case-sensitive), and skips hidden/common dependency roots unless explicitly included.workspace_grepreturns model-facing grep text inToolExecutionResult.content, while structured matches/counts live inToolExecutionResult.metadata.- When listing from workspace root, common dependency/cache roots (for example
node_modules,.venv,.git) are summarized instead of expanded. - You can still inspect those paths explicitly by setting
pathto that directory (or by settinginclude_ignored=true). - Supports
scan_limitto stop early on very large trees; when triggered, response setscount_is_estimate=true.
| Backend | Use case |
|---|---|
LocalWorkspaceBackend |
Default. Reads/writes to a local directory with path-escape protection. |
MemoryWorkspaceBackend |
Pure in-memory dict storage. Great for testing and sandboxed runs. |
S3WorkspaceBackend |
S3-compatible object storage (AWS S3, Aliyun OSS, MinIO, Cloudflare R2). |
from vv_agent.workspace import LocalWorkspaceBackend, MemoryWorkspaceBackend
# Explicit local backend
runtime = AgentRuntime(
llm_client=llm,
tool_registry=registry,
workspace_backend=LocalWorkspaceBackend(Path("./workspace")),
)
# In-memory backend for testing
runtime = AgentRuntime(
llm_client=llm,
tool_registry=registry,
workspace_backend=MemoryWorkspaceBackend(),
)
S3WorkspaceBackend
Install the optional S3 dependency: uv pip install 'vv-agent[s3]'.
from vv_agent.workspace import S3WorkspaceBackend
backend = S3WorkspaceBackend(
bucket="my-bucket",
prefix="agent-workspace",
endpoint_url="https://oss-cn-hangzhou.aliyuncs.com", # or None for AWS
aws_access_key_id="...",
aws_secret_access_key="...",
addressing_style="virtual", # "path" for MinIO
)
Custom Backend
Implement the WorkspaceBackend protocol (8 methods) to plug in any storage:
from vv_agent.workspace import WorkspaceBackend
class MyBackend:
def list_files(self, base: str, glob: str) -> list[str]: ...
def read_text(self, path: str) -> str: ...
def read_bytes(self, path: str) -> bytes: ...
def write_text(self, path: str, content: str, *, append: bool = False) -> int: ...
def file_info(self, path: str) -> FileInfo | None: ...
def exists(self, path: str) -> bool: ...
def is_file(self, path: str) -> bool: ...
def mkdir(self, path: str) -> None: ...
Modules
| Module | Description |
|---|---|
vv_agent.runtime.AgentRuntime |
Top-level state machine (completed / wait_user / max_cycles / failed) |
vv_agent.runtime.CycleRunner |
Single LLM turn and cycle record construction |
vv_agent.runtime.ToolCallRunner |
Tool execution with directive convergence |
vv_agent.runtime.RuntimeHookManager |
Hook dispatch (before/after LLM, tool call, memory compact) |
vv_agent.runtime.StateStore |
Checkpoint persistence protocol (InMemoryStateStore / SqliteStateStore / RedisStateStore) |
vv_agent.memory.MemoryManager |
Context compression when history exceeds threshold |
vv_agent.workspace |
Pluggable file storage: LocalWorkspaceBackend, MemoryWorkspaceBackend, S3WorkspaceBackend |
vv_agent.tools |
Built-in tools plus function_tool, FunctionTool, and structured tool outputs |
vv_agent |
Public SDK: Agent, Runner, RunConfig, ModelSettings, tools, sessions, typed events |
vv_agent.app_server |
JSONL App Server protocol, transport, thread state, replay, approval callbacks, schema export, and host provider boundary |
vv_agent.sdk |
Lower-level runtime compatibility helpers; new user code should not use this as the main entry point |
vv_agent.skills |
Agent Skills support (SKILL.md parsing, validation, unified normalization, prompt rendering with budget management, activate_skill tool) |
vv_agent.llm.VVLlmClient |
Unified LLM interface via vv-llm (endpoint rotation, retry, streaming) |
vv_agent.config |
Model/endpoint/key resolution from local_settings.py |
Runtime Boundary
vv-agent owns the portable agent runtime: prompt assembly, model calls, tool
planning, tool execution, memory compaction, typed events, cancellation,
approval interruption, and replayable run history. Host products own product
UI, user and workspace resolution, product storage, browser or IM integration,
and the product-specific tools exposed to the model.
Host products should implement providers instead of patching vv-agent
internals:
AppServerHostmaps product profiles, workspaces, tools, approval UI, memory, context, and model settings into App ServerAgentandRunConfigobjects when the host uses JSONL process integration.ApprovalProviderdecides whether a tool call needs approval and returns the allow, deny, session-allow, or timeout decision from product UI or rules.ContextProvidercontributes product prompt fragments such as profile, workspace, policy, or feature context before each run is compiled.vv_agent.memory.MemoryProviderconnects product memory stores to memory search/save hooks and compaction lifecycle events.vv_agent.tools.ToolExecutorexposes product tools with schema, approval, timeout, error, and execution behavior.FunctionTooland@function_toolcover normal Python functions; custom executors are routed byToolOrchestrator.RunEventStorepersists typedRunEventhistory so app views can replay completed runs and parent/child run graphs.
This boundary keeps Agent, Runner, RunConfig, RunHandle, and
RunEvent stable while allowing each host to keep its own account model,
workspace model, storage backend, and UI workflow outside the framework.
Memory Compaction
MemoryManager now measures context size in tokens and compacts history when a model-derived auto-compaction threshold is exceeded.
- Task-level knobs:
memory_compact_threshold(default128000, legacy fallback only when token counting is unavailable)memory_threshold_percentage(warning threshold percentage, default90)
- Compile mapping:
AgentCompilerforwards stable agent/run metadata intoRuntimeTask.- Runtime-only compaction knobs remain metadata-backed until promoted into stable public fields.
- Token budget model:
effective_context_window = model_context_window - reserved_output_tokensautocompact_threshold = effective_context_window - autocompact_buffer_tokens- Defaults come from
vv-llmmodel metadata when available, otherwise fall back to200000 / 16000 / 13000
- Effective-length strategy (backend-aligned):
- If previous cycle token usage exists:
effective_length = previous_prompt_tokens + token_count(recent_tool_messages)
- Otherwise fallback to:
vv_llm.chat_clients.utils.get_message_token_counts(...)- If tokenizer resolution fails, use a local CJK-aware estimate
- If previous cycle token usage exists:
- Compaction pipeline:
- Preemptive microcompact: clear old large tool results when usage crosses
microcompact_trigger_ratio - Session Memory extraction: persist key facts before full summarization so they survive later compactions
- Structural cleanup (stale tool calls, orphan tool messages, assistant-no-tool collapse, old tool result artifactization)
- If still over threshold, generate a compressed memory summary that preserves original user messages, file operations, current work state, and resolved errors
- If the provider still returns prompt-too-long, retry with forced compaction once, then progressively stronger emergency tail-dropping
- After full compaction, re-inject relevant workspace files into
<Post-Compaction File Context>under a bounded token budget
- Preemptive microcompact: clear old large tool results when usage crosses
- Session Memory behavior:
- Stored in
workspace/.memory/session/<session-or-task-scope>/session_memory.jsonby default - Scoped to the current session when
metadata.session_idis present; otherwise scoped to the currenttask_id - New sessions/tasks start without inherited Session Memory from previous sessions/tasks
- Injected into the first system message on every cycle as
<Session Memory> - Extraction reuses the configured memory summary backend/model
- Full compaction resets transcript tracking but preserves persisted memory entries
- Sub-tasks disable Session Memory by default to avoid parent/child memory-file contamination
- Stored in
Runtime metadata keys
Pass these via Agent.metadata or RunConfig.metadata; the compiler forwards
them into RuntimeTask.metadata:
memory_keep_recent_messagesmodel_context_windowreserved_output_tokensautocompact_buffer_tokensmicrocompact_trigger_ratiomicrocompact_keep_recent_cyclesmicrocompact_min_result_lengthmicrocompact_compactable_toolsinclude_memory_warningsession_memory_enabled/enable_session_memorysession_memory_min_tokenssession_memory_max_tokenssession_memory_min_text_messagessession_memory_storage_dirtool_result_compact_thresholdtool_result_keep_lasttool_result_excerpt_headtool_result_excerpt_tailtool_calls_keep_lastassistant_no_tool_keep_lasttool_result_artifact_dirsummary_event_limit
Memory summary model selection priority
Priority is strict:
RuntimeTask.metadatamemory_summary_backend/memory_summary_model- aliases:
compress_memory_summary_backend/compress_memory_summary_model - aliases:
memory_compress_backend/memory_compress_model
local_settings.pyconstantsDEFAULT_USER_MEMORY_SUMMARIZE_BACKEND/DEFAULT_USER_MEMORY_SUMMARIZE_MODEL- aliases:
DEFAULT_MEMORY_SUMMARIZE_BACKEND/DEFAULT_MEMORY_SUMMARIZE_MODEL - aliases:
VV_AGENT_MEMORY_SUMMARY_BACKEND/VV_AGENT_MEMORY_SUMMARY_MODEL
- Fallback
- runtime
default_backend+ current taskmodel
- runtime
Built-in Tools
list_files, file_info, read_file, write_file, file_str_replace, workspace_grep, compress_memory, todo_write, task_finish, ask_user, bash, read_image, create_sub_task, sub_task_status.
Custom tools can be registered via ToolRegistry.register().
The bash tool supports two background paths:
- Explicit background: pass
run_in_background=true, receive asession_idimmediately, then poll withcheck_background_command. - Timeout handoff: if a foreground command reaches
timeout, it is moved into a background session instead of failing immediately. The tool returns asession_id, and the session emits terminal background-command events when that process completes, fails, or times out.
Sub-agents
Use Agent.as_tool() when the parent agent should call a child agent and then
continue. Use handoff() when the child agent should take over and finish the
run. The lower-level create_sub_task tools remain available for runtime
compatibility, but they are no longer the primary public SDK contract.
Each delegated sub-task now runs in a real AgentSession (session id defaults to the sub-task id). Tool payloads include session_id, and runtime events include stable identifiers (task_id / session_id) so host apps can subscribe, persist, and stream sub-task progress independently, including sub_agent_assistant_delta and sub_agent_tool_call_progress events.
Batch mode in create_sub_task dispatches valid sub-task items through the runtime execution backend's parallel_map, so synchronous batches run concurrently when the backend supports parallel execution.
Use sub_task_status to query legacy runtime sub-task states, inspect
lightweight progress snapshots (detail_level=snapshot), or send follow-up
messages to running/completed sub-tasks.
Before a completed sub-task is resumed, the runtime now sanitizes the saved session transcript: empty assistant turns, thinking-only turns, orphaned tool results, and unresolved tail tool calls are removed so the next follow-up prompt resumes from a coherent history.
Sub-task runtime metadata now includes task_id, session_id, and browser_scope_key for each sub-agent run, so session-scoped tools (for example, browser controllers) stay isolated across parallel sub-tasks.
Host apps can interrupt a currently running sub-agent by calling vv_agent.runtime.engine.steer_sub_agent_session(session_id=..., prompt=...).
When a sub-agent uses a different model from the parent, the runtime needs settings_file and default_backend to resolve the LLM client.
Examples
The examples/ directory now contains public SDK cookbook scripts plus a small
set of lower-level runtime integration examples. See
examples/README.md for the full list.
uv run python examples/01_quick_start.py
uv run python examples/24_workspace_backends.py
Testing
uv run pytest # unit tests (no network)
uv run ruff check . # lint
uv run ty check # type check
V_AGENT_RUN_LIVE_TESTS=1 uv run pytest -m live # integration tests (needs real LLM)
Environment variables for live tests:
| Variable | Default | Description |
|---|---|---|
V_AGENT_LOCAL_SETTINGS |
local_settings.py |
Settings file path |
V_AGENT_LIVE_BACKEND |
moonshot |
LLM backend |
V_AGENT_LIVE_MODEL |
kimi-k2.6 |
Model name |
V_AGENT_ENABLE_BASE64_KEY_DECODE |
- | Set 1 to enable base64 API key decoding |
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 vv_agent-0.4.0.tar.gz.
File metadata
- Download URL: vv_agent-0.4.0.tar.gz
- Upload date:
- Size: 174.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
535f306241bd7856c2cbf290d500442fab454c274ce6053142fbc21b439cb071
|
|
| MD5 |
d8d08027490c240d6dc820b6fc737335
|
|
| BLAKE2b-256 |
4d793ae3b5ba572790b87eb83d5083927bc8192b076eb590f2662a030cf1b067
|
Provenance
The following attestation bundles were made for vv_agent-0.4.0.tar.gz:
Publisher:
release.yml on AndersonBY/vv-agent
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
vv_agent-0.4.0.tar.gz -
Subject digest:
535f306241bd7856c2cbf290d500442fab454c274ce6053142fbc21b439cb071 - Sigstore transparency entry: 1704562078
- Sigstore integration time:
-
Permalink:
AndersonBY/vv-agent@37d18f6fd940fb1185a656833b959aac96ab4180 -
Branch / Tag:
refs/tags/v0.4.0 - Owner: https://github.com/AndersonBY
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@37d18f6fd940fb1185a656833b959aac96ab4180 -
Trigger Event:
push
-
Statement type:
File details
Details for the file vv_agent-0.4.0-py3-none-any.whl.
File metadata
- Download URL: vv_agent-0.4.0-py3-none-any.whl
- Upload date:
- Size: 229.3 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 |
1c0e3e2b96530fcf7fa88da7507676b767025f80ef734ea967649b05fe2f4352
|
|
| MD5 |
80072c1a5a421f962db0f5426befc061
|
|
| BLAKE2b-256 |
ef76604faf86f05f0a2c2f7f4119faf5180e32b9e5f085fa953157512ddee466
|
Provenance
The following attestation bundles were made for vv_agent-0.4.0-py3-none-any.whl:
Publisher:
release.yml on AndersonBY/vv-agent
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
vv_agent-0.4.0-py3-none-any.whl -
Subject digest:
1c0e3e2b96530fcf7fa88da7507676b767025f80ef734ea967649b05fe2f4352 - Sigstore transparency entry: 1704562101
- Sigstore integration time:
-
Permalink:
AndersonBY/vv-agent@37d18f6fd940fb1185a656833b959aac96ab4180 -
Branch / Tag:
refs/tags/v0.4.0 - Owner: https://github.com/AndersonBY
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@37d18f6fd940fb1185a656833b959aac96ab4180 -
Trigger Event:
push
-
Statement type: