mirror of
https://github.com/langgenius/dify.git
synced 2026-06-26 23:01:11 +08:00
149 lines
5.8 KiB
Python
149 lines
5.8 KiB
Python
"""Inner API for the agent drive (agent 网盘) control plane.
|
|
|
|
These endpoints are called by the dify-agent server (not the sandbox) with the
|
|
inner API key. The drive ref is the URL segment ``agent-<agent_id>``; the
|
|
path-like file key travels in the query/body, never as a URL path segment (so
|
|
its ``/`` characters do not collide with routing). Drive-owned semantics:
|
|
tenant scoped, no user-level FileAccessScope. Commit still canonicalizes the
|
|
trusted execution-context user through the same EndUser lookup as plugin file
|
|
upload before validating ToolFile ownership.
|
|
"""
|
|
|
|
from flask import request
|
|
from flask_restx import Resource
|
|
from pydantic import BaseModel, ValidationError
|
|
|
|
from controllers.console.wraps import setup_required
|
|
from controllers.inner_api import inner_api_ns
|
|
from controllers.inner_api.plugin.wraps import get_user
|
|
from controllers.inner_api.wraps import plugin_inner_api_only
|
|
from extensions.ext_database import db
|
|
from models.agent import AgentDriveFileKind
|
|
from services.agent.soul_files_service import AgentSoulFilesService
|
|
from services.agent_drive_service import (
|
|
AgentDriveError,
|
|
AgentDriveService,
|
|
DriveCommitItem,
|
|
parse_agent_drive_ref,
|
|
)
|
|
|
|
|
|
class _CommitRequest(BaseModel):
|
|
tenant_id: str
|
|
user_id: str
|
|
items: list[DriveCommitItem]
|
|
|
|
|
|
def _error_response(exc: AgentDriveError) -> tuple[dict[str, str], int]:
|
|
return {"code": exc.code, "message": exc.message}, exc.status_code
|
|
|
|
|
|
def _versioned_manifest(
|
|
*,
|
|
tenant_id: str,
|
|
agent_id: str,
|
|
prefix: str = "",
|
|
include_download_url: bool = False,
|
|
) -> list[dict[str, object]]:
|
|
agent_soul = AgentSoulFilesService.active_agent_soul(session=db.session, tenant_id=tenant_id, agent_id=agent_id)
|
|
normalized_prefix = prefix.strip().lstrip("/")
|
|
items = AgentSoulFilesService.list_manifest_items(
|
|
session=db.session,
|
|
tenant_id=tenant_id,
|
|
agent_id=agent_id,
|
|
prefix=normalized_prefix,
|
|
)
|
|
skill_prefixes = AgentSoulFilesService.allowed_skill_prefixes(agent_soul)
|
|
if normalized_prefix and not any(normalized_prefix.startswith(p) for p in skill_prefixes):
|
|
allowed_keys = AgentSoulFilesService.allowed_drive_keys(agent_soul)
|
|
items = [item for item in items if item.get("key") in allowed_keys]
|
|
if include_download_url:
|
|
for item in items:
|
|
file_kind = item.get("file_kind")
|
|
file_id = item.get("file_id")
|
|
if not file_kind or not file_id:
|
|
continue
|
|
try:
|
|
item["download_url"] = AgentDriveService.resolve_download_url_for_ref(
|
|
tenant_id=tenant_id,
|
|
file_kind=AgentDriveFileKind(str(file_kind)),
|
|
file_id=str(file_id),
|
|
)
|
|
except ValueError:
|
|
item["download_url"] = None
|
|
return items
|
|
|
|
|
|
@inner_api_ns.route("/drive/<string:drive_ref>/manifest")
|
|
class AgentDriveManifestApi(Resource):
|
|
@setup_required
|
|
@plugin_inner_api_only
|
|
@inner_api_ns.doc("agent_drive_manifest")
|
|
@inner_api_ns.doc(description="List an agent drive (optionally with download URLs)")
|
|
def get(self, drive_ref: str):
|
|
try:
|
|
agent_id = parse_agent_drive_ref(drive_ref)
|
|
tenant_id = (request.args.get("tenant_id") or "").strip()
|
|
if not tenant_id:
|
|
raise AgentDriveError("missing_tenant_id", "tenant_id is required", status_code=400)
|
|
include_download_url = (request.args.get("include_download_url") or "").lower() in ("1", "true", "yes")
|
|
items = _versioned_manifest(
|
|
tenant_id=tenant_id,
|
|
agent_id=agent_id,
|
|
prefix=request.args.get("prefix", ""),
|
|
include_download_url=include_download_url,
|
|
)
|
|
except AgentDriveError as exc:
|
|
return _error_response(exc)
|
|
return {"items": items}
|
|
|
|
|
|
@inner_api_ns.route("/drive/<string:drive_ref>/skills")
|
|
class AgentDriveSkillsApi(Resource):
|
|
@setup_required
|
|
@plugin_inner_api_only
|
|
@inner_api_ns.doc("agent_drive_skills")
|
|
@inner_api_ns.doc(description="List the skill catalog of an agent drive")
|
|
def get(self, drive_ref: str):
|
|
try:
|
|
agent_id = parse_agent_drive_ref(drive_ref)
|
|
tenant_id = (request.args.get("tenant_id") or "").strip()
|
|
if not tenant_id:
|
|
raise AgentDriveError("missing_tenant_id", "tenant_id is required", status_code=400)
|
|
items = AgentSoulFilesService.list_skills(session=db.session, tenant_id=tenant_id, agent_id=agent_id)
|
|
except AgentDriveError as exc:
|
|
return _error_response(exc)
|
|
return {"items": items}
|
|
|
|
|
|
@inner_api_ns.route("/drive/<string:drive_ref>/commit")
|
|
class AgentDriveCommitApi(Resource):
|
|
@setup_required
|
|
@plugin_inner_api_only
|
|
@inner_api_ns.doc("agent_drive_commit")
|
|
@inner_api_ns.doc(description="Commit a batch of file refs into an agent drive")
|
|
def post(self, drive_ref: str):
|
|
try:
|
|
agent_id = parse_agent_drive_ref(drive_ref)
|
|
try:
|
|
body = _CommitRequest.model_validate(request.get_json(silent=True) or {})
|
|
except ValidationError as exc:
|
|
raise AgentDriveError("invalid_request", str(exc), status_code=400) from exc
|
|
user = get_user(body.tenant_id, body.user_id)
|
|
items = AgentDriveService().commit(
|
|
tenant_id=body.tenant_id,
|
|
user_id=user.id,
|
|
agent_id=agent_id,
|
|
items=body.items,
|
|
)
|
|
AgentSoulFilesService.sync_drive_commit_to_active_soul(
|
|
tenant_id=body.tenant_id,
|
|
agent_id=agent_id,
|
|
account_id=user.id,
|
|
committed_items=items,
|
|
)
|
|
db.session.commit()
|
|
except AgentDriveError as exc:
|
|
return _error_response(exc)
|
|
return {"items": items}
|