diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json index 8246544061..ddec42e0ee 100644 --- a/.devcontainer/devcontainer.json +++ b/.devcontainer/devcontainer.json @@ -11,7 +11,7 @@ "nodeGypDependencies": true, "version": "lts" }, - "ghcr.io/devcontainers-contrib/features/npm-package:1": { + "ghcr.io/devcontainers-extra/features/npm-package:1": { "package": "typescript", "version": "latest" }, diff --git a/.devcontainer/post_create_command.sh b/.devcontainer/post_create_command.sh index 2e787ab855..a26fd076ed 100755 --- a/.devcontainer/post_create_command.sh +++ b/.devcontainer/post_create_command.sh @@ -6,11 +6,10 @@ cd web && pnpm install pipx install uv echo "alias start-api=\"cd $WORKSPACE_ROOT/api && uv run python -m flask run --host 0.0.0.0 --port=5001 --debug\"" >> ~/.bashrc -echo "alias start-worker=\"cd $WORKSPACE_ROOT/api && uv run python -m celery -A app.celery worker -P gevent -c 1 --loglevel INFO -Q dataset,generation,mail,ops_trace,app_deletion,plugin,workflow_storage\"" >> ~/.bashrc +echo "alias start-worker=\"cd $WORKSPACE_ROOT/api && uv run python -m celery -A app.celery worker -P threads -c 1 --loglevel INFO -Q dataset,priority_dataset,priority_pipeline,pipeline,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,workflow,schedule_poller,schedule_executor,triggered_workflow_dispatcher,trigger_refresh_executor\"" >> ~/.bashrc echo "alias start-web=\"cd $WORKSPACE_ROOT/web && pnpm dev\"" >> ~/.bashrc echo "alias start-web-prod=\"cd $WORKSPACE_ROOT/web && pnpm build && pnpm start\"" >> ~/.bashrc echo "alias start-containers=\"cd $WORKSPACE_ROOT/docker && docker-compose -f docker-compose.middleware.yaml -p dify --env-file middleware.env up -d\"" >> ~/.bashrc echo "alias stop-containers=\"cd $WORKSPACE_ROOT/docker && docker-compose -f docker-compose.middleware.yaml -p dify --env-file middleware.env down\"" >> ~/.bashrc source /home/vscode/.bashrc - diff --git a/.github/workflows/autofix.yml b/.github/workflows/autofix.yml index 0cae2ef552..2ce8a09a7d 100644 --- a/.github/workflows/autofix.yml +++ b/.github/workflows/autofix.yml @@ -2,6 +2,8 @@ name: autofix.ci on: pull_request: branches: ["main"] + push: + branches: ["main"] permissions: contents: read diff --git a/.github/workflows/style.yml b/.github/workflows/style.yml index 06584c1b78..e652657705 100644 --- a/.github/workflows/style.yml +++ b/.github/workflows/style.yml @@ -103,6 +103,11 @@ jobs: run: | pnpm run lint + - name: Web type check + if: steps.changed-files.outputs.any_changed == 'true' + working-directory: ./web + run: pnpm run type-check + docker-compose-template: name: Docker Compose Template runs-on: ubuntu-latest diff --git a/.gitignore b/.gitignore index 22a2c42566..c6067e96cd 100644 --- a/.gitignore +++ b/.gitignore @@ -6,6 +6,9 @@ __pycache__/ # C extensions *.so +# *db files +*.db + # Distribution / packaging .Python build/ @@ -97,6 +100,7 @@ __pypackages__/ # Celery stuff celerybeat-schedule +celerybeat-schedule.db celerybeat.pid # SageMath parsed files @@ -234,4 +238,7 @@ scripts/stress-test/reports/ # mcp .playwright-mcp/ -.serena/ \ No newline at end of file +.serena/ + +# settings +*.local.json diff --git a/.vscode/launch.json.template b/.vscode/launch.json.template index f5a7f0893b..bd5a787d4c 100644 --- a/.vscode/launch.json.template +++ b/.vscode/launch.json.template @@ -8,8 +8,7 @@ "module": "flask", "env": { "FLASK_APP": "app.py", - "FLASK_ENV": "development", - "GEVENT_SUPPORT": "True" + "FLASK_ENV": "development" }, "args": [ "run", @@ -28,9 +27,7 @@ "type": "debugpy", "request": "launch", "module": "celery", - "env": { - "GEVENT_SUPPORT": "True" - }, + "env": {}, "args": [ "-A", "app.celery", @@ -40,7 +37,7 @@ "-c", "1", "-Q", - "dataset,generation,mail,ops_trace", + "dataset,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,priority_pipeline,pipeline", "--loglevel", "INFO" ], diff --git a/README.md b/README.md index 7c194e065a..e5cc05fbc0 100644 --- a/README.md +++ b/README.md @@ -63,7 +63,7 @@ Dify is an open-source platform for developing LLM applications. Its intuitive i > - CPU >= 2 Core > - RAM >= 4 GiB -
+
The easiest way to start the Dify server is through [Docker Compose](docker/docker-compose.yaml). Before running Dify with the following commands, make sure that [Docker](https://docs.docker.com/get-docker/) and [Docker Compose](https://docs.docker.com/compose/install/) are installed on your machine: @@ -109,15 +109,15 @@ All of Dify's offerings come with corresponding APIs, so you could effortlessly ## Using Dify -- **Cloud
** +- **Cloud
** We host a [Dify Cloud](https://dify.ai) service for anyone to try with zero setup. It provides all the capabilities of the self-deployed version, and includes 200 free GPT-4 calls in the sandbox plan. -- **Self-hosting Dify Community Edition
** +- **Self-hosting Dify Community Edition
** Quickly get Dify running in your environment with this [starter guide](#quick-start). Use our [documentation](https://docs.dify.ai) for further references and more in-depth instructions. -- **Dify for enterprise / organizations
** - We provide additional enterprise-centric features. [Log your questions for us through this chatbot](https://udify.app/chat/22L1zSxg6yW1cWQg) or [send us an email](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) to discuss enterprise needs.
+- **Dify for enterprise / organizations
** + We provide additional enterprise-centric features. [Send us an email](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) to discuss your enterprise needs.
> For startups and small businesses using AWS, check out [Dify Premium on AWS Marketplace](https://aws.amazon.com/marketplace/pp/prodview-t22mebxzwjhu6) and deploy it to your own AWS VPC with one click. It's an affordable AMI offering with the option to create apps with custom logo and branding. diff --git a/api/.env.example b/api/.env.example index 4df6adf348..b1ac15d25b 100644 --- a/api/.env.example +++ b/api/.env.example @@ -27,6 +27,9 @@ FILES_URL=http://localhost:5001 # Example: INTERNAL_FILES_URL=http://api:5001 INTERNAL_FILES_URL=http://127.0.0.1:5001 +# TRIGGER URL +TRIGGER_URL=http://localhost:5001 + # The time in seconds after the signature is rejected FILES_ACCESS_TIMEOUT=300 @@ -156,6 +159,9 @@ SUPABASE_URL=your-server-url # CORS configuration WEB_API_CORS_ALLOW_ORIGINS=http://localhost:3000,* CONSOLE_CORS_ALLOW_ORIGINS=http://localhost:3000,* +# Set COOKIE_DOMAIN when the console frontend and API are on different subdomains. +# Provide the registrable domain (e.g. example.com); leading dots are optional. +COOKIE_DOMAIN= # Vector database configuration # Supported values are `weaviate`, `qdrant`, `milvus`, `myscale`, `relyt`, `pgvector`, `pgvecto-rs`, `chroma`, `opensearch`, `oracle`, `tencent`, `elasticsearch`, `elasticsearch-ja`, `analyticdb`, `couchbase`, `vikingdb`, `oceanbase`, `opengauss`, `tablestore`,`vastbase`,`tidb`,`tidb_on_qdrant`,`baidu`,`lindorm`,`huawei_cloud`,`upstash`, `matrixone`. @@ -368,6 +374,12 @@ UPLOAD_IMAGE_FILE_SIZE_LIMIT=10 UPLOAD_VIDEO_FILE_SIZE_LIMIT=100 UPLOAD_AUDIO_FILE_SIZE_LIMIT=50 +# Comma-separated list of file extensions blocked from upload for security reasons. +# Extensions should be lowercase without dots (e.g., exe,bat,sh,dll). +# Empty by default to allow all file types. +# Recommended: exe,bat,cmd,com,scr,vbs,ps1,msi,dll +UPLOAD_FILE_EXTENSION_BLACKLIST= + # Model configuration MULTIMODAL_SEND_FORMAT=base64 PROMPT_GENERATION_MAX_TOKENS=512 @@ -457,6 +469,9 @@ HTTP_REQUEST_NODE_MAX_BINARY_SIZE=10485760 HTTP_REQUEST_NODE_MAX_TEXT_SIZE=1048576 HTTP_REQUEST_NODE_SSL_VERIFY=True +# Webhook request configuration +WEBHOOK_REQUEST_BODY_MAX_SIZE=10485760 + # Respect X-* headers to redirect clients RESPECT_XFORWARD_HEADERS_ENABLED=false @@ -512,7 +527,7 @@ API_WORKFLOW_NODE_EXECUTION_REPOSITORY=repositories.sqlalchemy_api_workflow_node API_WORKFLOW_RUN_REPOSITORY=repositories.sqlalchemy_api_workflow_run_repository.DifyAPISQLAlchemyWorkflowRunRepository # Workflow log cleanup configuration # Enable automatic cleanup of workflow run logs to manage database size -WORKFLOW_LOG_CLEANUP_ENABLED=true +WORKFLOW_LOG_CLEANUP_ENABLED=false # Number of days to retain workflow run logs (default: 30 days) WORKFLOW_LOG_RETENTION_DAYS=30 # Batch size for workflow log cleanup operations (default: 100) @@ -534,6 +549,12 @@ ENABLE_CLEAN_MESSAGES=false ENABLE_MAIL_CLEAN_DOCUMENT_NOTIFY_TASK=false ENABLE_DATASETS_QUEUE_MONITOR=false ENABLE_CHECK_UPGRADABLE_PLUGIN_TASK=true +ENABLE_WORKFLOW_SCHEDULE_POLLER_TASK=true +# Interval time in minutes for polling scheduled workflows(default: 1 min) +WORKFLOW_SCHEDULE_POLLER_INTERVAL=1 +WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE=100 +# Maximum number of scheduled workflows to dispatch per tick (0 for unlimited) +WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK=0 # Position configuration POSITION_TOOL_PINS= @@ -605,3 +626,9 @@ SWAGGER_UI_PATH=/swagger-ui.html # Whether to encrypt dataset IDs when exporting DSL files (default: true) # Set to false to export dataset IDs as plain text for easier cross-environment import DSL_EXPORT_ENCRYPT_DATASET_ID=true + +# Tenant isolated task queue configuration +TENANT_ISOLATED_TASK_CONCURRENCY=1 + +# Maximum number of segments for dataset segments API (0 for unlimited) +DATASET_MAX_SEGMENTS_PER_REQUEST=0 diff --git a/api/.vscode/launch.json.example b/api/.vscode/launch.json.example index b9e32e2511..092c66e798 100644 --- a/api/.vscode/launch.json.example +++ b/api/.vscode/launch.json.example @@ -54,7 +54,7 @@ "--loglevel", "DEBUG", "-Q", - "dataset,generation,mail,ops_trace,app_deletion" + "dataset,priority_pipeline,pipeline,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,workflow,schedule_poller,schedule_executor,triggered_workflow_dispatcher,trigger_refresh_executor" ] } ] diff --git a/api/AGENTS.md b/api/AGENTS.md new file mode 100644 index 0000000000..17398ec4b8 --- /dev/null +++ b/api/AGENTS.md @@ -0,0 +1,62 @@ +# Agent Skill Index + +Start with the section that best matches your need. Each entry lists the problems it solves plus key files/concepts so you know what to expect before opening it. + +______________________________________________________________________ + +## Platform Foundations + +- **[Infrastructure Overview](agent_skills/infra.md)**\ + When to read this: + + - You need to understand where a feature belongs in the architecture. + - You’re wiring storage, Redis, vector stores, or OTEL. + - You’re about to add CLI commands or async jobs.\ + What it covers: configuration stack (`configs/app_config.py`, remote settings), storage entry points (`extensions/ext_storage.py`, `core/file/file_manager.py`), Redis conventions (`extensions/ext_redis.py`), plugin runtime topology, vector-store factory (`core/rag/datasource/vdb/*`), observability hooks, SSRF proxy usage, and core CLI commands. + +- **[Coding Style](agent_skills/coding_style.md)**\ + When to read this: + + - You’re writing or reviewing backend code and need the authoritative checklist. + - You’re unsure about Pydantic validators, SQLAlchemy session usage, or logging patterns. + - You want the exact lint/type/test commands used in PRs.\ + Includes: Ruff & BasedPyright commands, no-annotation policy, session examples (`with Session(db.engine, ...)`), `@field_validator` usage, logging expectations, and the rule set for file size, helpers, and package management. + +______________________________________________________________________ + +## Plugin & Extension Development + +- **[Plugin Systems](agent_skills/plugin.md)**\ + When to read this: + + - You’re building or debugging a marketplace plugin. + - You need to know how manifests, providers, daemons, and migrations fit together.\ + What it covers: plugin manifests (`core/plugin/entities/plugin.py`), installation/upgrade flows (`services/plugin/plugin_service.py`, CLI commands), runtime adapters (`core/plugin/impl/*` for tool/model/datasource/trigger/endpoint/agent), daemon coordination (`core/plugin/entities/plugin_daemon.py`), and how provider registries surface capabilities to the rest of the platform. + +- **[Plugin OAuth](agent_skills/plugin_oauth.md)**\ + When to read this: + + - You must integrate OAuth for a plugin or datasource. + - You’re handling credential encryption or refresh flows.\ + Topics: credential storage, encryption helpers (`core/helper/provider_encryption.py`), OAuth client bootstrap (`services/plugin/oauth_service.py`, `services/plugin/plugin_parameter_service.py`), and how console/API layers expose the flows. + +______________________________________________________________________ + +## Workflow Entry & Execution + +- **[Trigger Concepts](agent_skills/trigger.md)**\ + When to read this: + - You’re debugging why a workflow didn’t start. + - You’re adding a new trigger type or hook. + - You need to trace async execution, draft debugging, or webhook/schedule pipelines.\ + Details: Start-node taxonomy, webhook & schedule internals (`core/workflow/nodes/trigger_*`, `services/trigger/*`), async orchestration (`services/async_workflow_service.py`, Celery queues), debug event bus, and storage/logging interactions. + +______________________________________________________________________ + +## Additional Notes for Agents + +- All skill docs assume you follow the coding style guide—run Ruff/BasedPyright/tests listed there before submitting changes. +- When you cannot find an answer in these briefs, search the codebase using the paths referenced (e.g., `core/plugin/impl/tool.py`, `services/dataset_service.py`). +- If you run into cross-cutting concerns (tenancy, configuration, storage), check the infrastructure guide first; it links to most supporting modules. +- Keep multi-tenancy and configuration central: everything flows through `configs.dify_config` and `tenant_id`. +- When touching plugins or triggers, consult both the system overview and the specialised doc to ensure you adjust lifecycle, storage, and observability consistently. diff --git a/api/Dockerfile b/api/Dockerfile index 79a4892768..ed61923a40 100644 --- a/api/Dockerfile +++ b/api/Dockerfile @@ -15,7 +15,11 @@ FROM base AS packages # RUN sed -i 's@deb.debian.org@mirrors.aliyun.com@g' /etc/apt/sources.list.d/debian.sources RUN apt-get update \ - && apt-get install -y --no-install-recommends gcc g++ libc-dev libffi-dev libgmp-dev libmpfr-dev libmpc-dev + && apt-get install -y --no-install-recommends \ + # basic environment + g++ \ + # for building gmpy2 + libmpfr-dev libmpc-dev # Install Python dependencies COPY pyproject.toml uv.lock ./ @@ -49,7 +53,9 @@ RUN \ # Install dependencies && apt-get install -y --no-install-recommends \ # basic environment - curl nodejs libgmp-dev libmpfr-dev libmpc-dev \ + curl nodejs \ + # for gmpy2 \ + libgmp-dev libmpfr-dev libmpc-dev \ # For Security expat libldap-2.5-0 perl libsqlite3-0 zlib1g \ # install fonts to support the use of tools like pypdfium2 diff --git a/api/README.md b/api/README.md index e75ea3d354..45dad07af0 100644 --- a/api/README.md +++ b/api/README.md @@ -80,7 +80,7 @@ 1. If you need to handle and debug the async tasks (e.g. dataset importing and documents indexing), please start the worker service. ```bash -uv run celery -A app.celery worker -P gevent -c 2 --loglevel INFO -Q dataset,generation,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation +uv run celery -A app.celery worker -P threads -c 2 --loglevel INFO -Q dataset,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,priority_pipeline,pipeline ``` Additionally, if you want to debug the celery scheduled tasks, you can run the following command in another terminal to start the beat service: diff --git a/api/agent_skills/coding_style.md b/api/agent_skills/coding_style.md new file mode 100644 index 0000000000..a2b66f0bd5 --- /dev/null +++ b/api/agent_skills/coding_style.md @@ -0,0 +1,115 @@ +## Linter + +- Always follow `.ruff.toml`. +- Run `uv run ruff check --fix --unsafe-fixes`. +- Keep each line under 100 characters (including spaces). + +## Code Style + +- `snake_case` for variables and functions. +- `PascalCase` for classes. +- `UPPER_CASE` for constants. + +## Rules + +- Use Pydantic v2 standard. +- Use `uv` for package management. +- Do not override dunder methods like `__init__`, `__iadd__`, etc. +- Never launch services (`uv run app.py`, `flask run`, etc.); running tests under `tests/` is allowed. +- Prefer simple functions over classes for lightweight helpers. +- Keep files below 800 lines; split when necessary. +- Keep code readable—no clever hacks. +- Never use `print`; log with `logger = logging.getLogger(__name__)`. + +## Guiding Principles + +- Mirror the project’s layered architecture: controller → service → core/domain. +- Reuse existing helpers in `core/`, `services/`, and `libs/` before creating new abstractions. +- Optimise for observability: deterministic control flow, clear logging, actionable errors. + +## SQLAlchemy Patterns + +- Models inherit from `models.base.Base`; never create ad-hoc metadata or engines. + +- Open sessions with context managers: + + ```python + from sqlalchemy.orm import Session + + with Session(db.engine, expire_on_commit=False) as session: + stmt = select(Workflow).where( + Workflow.id == workflow_id, + Workflow.tenant_id == tenant_id, + ) + workflow = session.execute(stmt).scalar_one_or_none() + ``` + +- Use SQLAlchemy expressions; avoid raw SQL unless necessary. + +- Introduce repository abstractions only for very large tables (e.g., workflow executions) to support alternative storage strategies. + +- Always scope queries by `tenant_id` and protect write paths with safeguards (`FOR UPDATE`, row counts, etc.). + +## Storage & External IO + +- Access storage via `extensions.ext_storage.storage`. +- Use `core.helper.ssrf_proxy` for outbound HTTP fetches. +- Background tasks that touch storage must be idempotent and log the relevant object identifiers. + +## Pydantic Usage + +- Define DTOs with Pydantic v2 models and forbid extras by default. + +- Use `@field_validator` / `@model_validator` for domain rules. + +- Example: + + ```python + from pydantic import BaseModel, ConfigDict, HttpUrl, field_validator + + class TriggerConfig(BaseModel): + endpoint: HttpUrl + secret: str + + model_config = ConfigDict(extra="forbid") + + @field_validator("secret") + def ensure_secret_prefix(cls, value: str) -> str: + if not value.startswith("dify_"): + raise ValueError("secret must start with dify_") + return value + ``` + +## Generics & Protocols + +- Use `typing.Protocol` to define behavioural contracts (e.g., cache interfaces). +- Apply generics (`TypeVar`, `Generic`) for reusable utilities like caches or providers. +- Validate dynamic inputs at runtime when generics cannot enforce safety alone. + +## Error Handling & Logging + +- Raise domain-specific exceptions (`services/errors`, `core/errors`) and translate to HTTP responses in controllers. +- Declare `logger = logging.getLogger(__name__)` at module top. +- Include tenant/app/workflow identifiers in log context. +- Log retryable events at `warning`, terminal failures at `error`. + +## Tooling & Checks + +- Format/lint: `uv run --project api --dev ruff format ./api` and `uv run --project api --dev ruff check --fix --unsafe-fixes ./api`. +- Type checks: `uv run --directory api --dev basedpyright`. +- Tests: `uv run --project api --dev dev/pytest/pytest_unit_tests.sh`. +- Run all of the above before submitting your work. + +## Controllers & Services + +- Controllers: parse input via Pydantic, invoke services, return serialised responses; no business logic. +- Services: coordinate repositories, providers, background tasks; keep side effects explicit. +- Avoid repositories unless necessary; direct SQLAlchemy usage is preferred for typical tables. +- Document non-obvious behaviour with concise comments. + +## Miscellaneous + +- Use `configs.dify_config` for configuration—never read environment variables directly. +- Maintain tenant awareness end-to-end; `tenant_id` must flow through every layer touching shared resources. +- Queue async work through `services/async_workflow_service`; implement tasks under `tasks/` with explicit queue selection. +- Keep experimental scripts under `dev/`; do not ship them in production builds. diff --git a/api/agent_skills/infra.md b/api/agent_skills/infra.md new file mode 100644 index 0000000000..bc36c7bf64 --- /dev/null +++ b/api/agent_skills/infra.md @@ -0,0 +1,96 @@ +## Configuration + +- Import `configs.dify_config` for every runtime toggle. Do not read environment variables directly. +- Add new settings to the proper mixin inside `configs/` (deployment, feature, middleware, etc.) so they load through `DifyConfig`. +- Remote overrides come from the optional providers in `configs/remote_settings_sources`; keep defaults in code safe when the value is missing. +- Example: logging pulls targets from `extensions/ext_logging.py`, and model provider URLs are assembled in `services/entities/model_provider_entities.py`. + +## Dependencies + +- Runtime dependencies live in `[project].dependencies` inside `pyproject.toml`. Optional clients go into the `storage`, `tools`, or `vdb` groups under `[dependency-groups]`. +- Always pin versions and keep the list alphabetised. Shared tooling (lint, typing, pytest) belongs in the `dev` group. +- When code needs a new package, explain why in the PR and run `uv lock` so the lockfile stays current. + +## Storage & Files + +- Use `extensions.ext_storage.storage` for all blob IO; it already respects the configured backend. +- Convert files for workflows with helpers in `core/file/file_manager.py`; they handle signed URLs and multimodal payloads. +- When writing controller logic, delegate upload quotas and metadata to `services/file_service.py` instead of touching storage directly. +- All outbound HTTP fetches (webhooks, remote files) must go through the SSRF-safe client in `core/helper/ssrf_proxy.py`; it wraps `httpx` with the allow/deny rules configured for the platform. + +## Redis & Shared State + +- Access Redis through `extensions.ext_redis.redis_client`. For locking, reuse `redis_client.lock`. +- Prefer higher-level helpers when available: rate limits use `libs.helper.RateLimiter`, provider metadata uses caches in `core/helper/provider_cache.py`. + +## Models + +- SQLAlchemy models sit in `models/` and inherit from the shared declarative `Base` defined in `models/base.py` (metadata configured via `models/engine.py`). +- `models/__init__.py` exposes grouped aggregates: account/tenant models, app and conversation tables, datasets, providers, workflow runs, triggers, etc. Import from there to avoid deep path churn. +- Follow the DDD boundary: persistence objects live in `models/`, repositories under `repositories/` translate them into domain entities, and services consume those repositories. +- When adding a table, create the model class, register it in `models/__init__.py`, wire a repository if needed, and generate an Alembic migration as described below. + +## Vector Stores + +- Vector client implementations live in `core/rag/datasource/vdb/`, with a common factory in `core/rag/datasource/vdb/vector_factory.py` and enums in `core/rag/datasource/vdb/vector_type.py`. +- Retrieval pipelines call these providers through `core/rag/datasource/retrieval_service.py` and dataset ingestion flows in `services/dataset_service.py`. +- The CLI helper `flask vdb-migrate` orchestrates bulk migrations using routines in `commands.py`; reuse that pattern when adding new backend transitions. +- To add another store, mirror the provider layout, register it with the factory, and include any schema changes in Alembic migrations. + +## Observability & OTEL + +- OpenTelemetry settings live under the observability mixin in `configs/observability`. Toggle exporters and sampling via `dify_config`, not ad-hoc env reads. +- HTTP, Celery, Redis, SQLAlchemy, and httpx instrumentation is initialised in `extensions/ext_app_metrics.py` and `extensions/ext_request_logging.py`; reuse these hooks when adding new workers or entrypoints. +- When creating background tasks or external calls, propagate tracing context with helpers in the existing instrumented clients (e.g. use the shared `httpx` session from `core/helper/http_client_pooling.py`). +- If you add a new external integration, ensure spans and metrics are emitted by wiring the appropriate OTEL instrumentation package in `pyproject.toml` and configuring it in `extensions/`. + +## Ops Integrations + +- Langfuse support and other tracing bridges live under `core/ops/opik_trace`. Config toggles sit in `configs/observability`, while exporters are initialised in the OTEL extensions mentioned above. +- External monitoring services should follow this pattern: keep client code in `core/ops`, expose switches via `dify_config`, and hook initialisation in `extensions/ext_app_metrics.py` or sibling modules. +- Before instrumenting new code paths, check whether existing context helpers (e.g. `extensions/ext_request_logging.py`) already capture the necessary metadata. + +## Controllers, Services, Core + +- Controllers only parse HTTP input and call a service method. Keep business rules in `services/`. +- Services enforce tenant rules, quotas, and orchestration, then call into `core/` engines (workflow execution, tools, LLMs). +- When adding a new endpoint, search for an existing service to extend before introducing a new layer. Example: workflow APIs pipe through `services/workflow_service.py` into `core/workflow`. + +## Plugins, Tools, Providers + +- In Dify a plugin is a tenant-installable bundle that declares one or more providers (tool, model, datasource, trigger, endpoint, agent strategy) plus its resource needs and version metadata. The manifest (`core/plugin/entities/plugin.py`) mirrors what you see in the marketplace documentation. +- Installation, upgrades, and migrations are orchestrated by `services/plugin/plugin_service.py` together with helpers such as `services/plugin/plugin_migration.py`. +- Runtime loading happens through the implementations under `core/plugin/impl/*` (tool/model/datasource/trigger/endpoint/agent). These modules normalise plugin providers so that downstream systems (`core/tools/tool_manager.py`, `services/model_provider_service.py`, `services/trigger/*`) can treat builtin and plugin capabilities the same way. +- For remote execution, plugin daemons (`core/plugin/entities/plugin_daemon.py`, `core/plugin/impl/plugin.py`) manage lifecycle hooks, credential forwarding, and background workers that keep plugin processes in sync with the main application. +- Acquire tool implementations through `core/tools/tool_manager.py`; it resolves builtin, plugin, and workflow-as-tool providers uniformly, injecting the right context (tenant, credentials, runtime config). +- To add a new plugin capability, extend the relevant `core/plugin/entities` schema and register the implementation in the matching `core/plugin/impl` module rather than importing the provider directly. + +## Async Workloads + +see `agent_skills/trigger.md` for more detailed documentation. + +- Enqueue background work through `services/async_workflow_service.py`. It routes jobs to the tiered Celery queues defined in `tasks/`. +- Workers boot from `celery_entrypoint.py` and execute functions in `tasks/workflow_execution_tasks.py`, `tasks/trigger_processing_tasks.py`, etc. +- Scheduled workflows poll from `schedule/workflow_schedule_tasks.py`. Follow the same pattern if you need new periodic jobs. + +## Database & Migrations + +- SQLAlchemy models live under `models/` and map directly to migration files in `migrations/versions`. +- Generate migrations with `uv run --project api flask db revision --autogenerate -m ""`, then review the diff; never hand-edit the database outside Alembic. +- Apply migrations locally using `uv run --project api flask db upgrade`; production deploys expect the same history. +- If you add tenant-scoped data, confirm the upgrade includes tenant filters or defaults consistent with the service logic touching those tables. + +## CLI Commands + +- Maintenance commands from `commands.py` are registered on the Flask CLI. Run them via `uv run --project api flask `. +- Use the built-in `db` commands from Flask-Migrate for schema operations (`flask db upgrade`, `flask db stamp`, etc.). Only fall back to custom helpers if you need their extra behaviour. +- Custom entries such as `flask reset-password`, `flask reset-email`, and `flask vdb-migrate` handle self-hosted account recovery and vector database migrations. +- Before adding a new command, check whether an existing service can be reused and ensure the command guards edition-specific behaviour (many enforce `SELF_HOSTED`). Document any additions in the PR. +- Ruff helpers are run directly with `uv`: `uv run --project api --dev ruff format ./api` for formatting and `uv run --project api --dev ruff check ./api` (add `--fix` if you want automatic fixes). + +## When You Add Features + +- Check for an existing helper or service before writing a new util. +- Uphold tenancy: every service method should receive the tenant ID from controller wrappers such as `controllers/console/wraps.py`. +- Update or create tests alongside behaviour changes (`tests/unit_tests` for fast coverage, `tests/integration_tests` when touching orchestrations). +- Run `uv run --project api --dev ruff check ./api`, `uv run --directory api --dev basedpyright`, and `uv run --project api --dev dev/pytest/pytest_unit_tests.sh` before submitting changes. diff --git a/api/agent_skills/plugin.md b/api/agent_skills/plugin.md new file mode 100644 index 0000000000..954ddd236b --- /dev/null +++ b/api/agent_skills/plugin.md @@ -0,0 +1 @@ +// TBD diff --git a/api/agent_skills/plugin_oauth.md b/api/agent_skills/plugin_oauth.md new file mode 100644 index 0000000000..954ddd236b --- /dev/null +++ b/api/agent_skills/plugin_oauth.md @@ -0,0 +1 @@ +// TBD diff --git a/api/agent_skills/trigger.md b/api/agent_skills/trigger.md new file mode 100644 index 0000000000..f4b076332c --- /dev/null +++ b/api/agent_skills/trigger.md @@ -0,0 +1,53 @@ +## Overview + +Trigger is a collection of nodes that we called `Start` nodes, also, the concept of `Start` is the same as `RootNode` in the workflow engine `core/workflow/graph_engine`, On the other hand, `Start` node is the entry point of workflows, every workflow run always starts from a `Start` node. + +## Trigger nodes + +- `UserInput` +- `Trigger Webhook` +- `Trigger Schedule` +- `Trigger Plugin` + +### UserInput + +Before `Trigger` concept is introduced, it's what we called `Start` node, but now, to avoid confusion, it was renamed to `UserInput` node, has a strong relation with `ServiceAPI` in `controllers/service_api/app` + +1. `UserInput` node introduces a list of arguments that need to be provided by the user, finally it will be converted into variables in the workflow variable pool. +1. `ServiceAPI` accept those arguments, and pass through them into `UserInput` node. +1. For its detailed implementation, please refer to `core/workflow/nodes/start` + +### Trigger Webhook + +Inside Webhook Node, Dify provided a UI panel that allows user define a HTTP manifest `core/workflow/nodes/trigger_webhook/entities.py`.`WebhookData`, also, Dify generates a random webhook id for each `Trigger Webhook` node, the implementation was implemented in `core/trigger/utils/endpoint.py`, as you can see, `webhook-debug` is a debug mode for webhook, you may find it in `controllers/trigger/webhook.py`. + +Finally, requests to `webhook` endpoint will be converted into variables in workflow variable pool during workflow execution. + +### Trigger Schedule + +`Trigger Schedule` node is a node that allows user define a schedule to trigger the workflow, detailed manifest is here `core/workflow/nodes/trigger_schedule/entities.py`, we have a poller and executor to handle millions of schedules, see `docker/entrypoint.sh` / `schedule/workflow_schedule_task.py` for help. + +To Achieve this, a `WorkflowSchedulePlan` model was introduced in `models/trigger.py`, and a `events/event_handlers/sync_workflow_schedule_when_app_published.py` was used to sync workflow schedule plans when app is published. + +### Trigger Plugin + +`Trigger Plugin` node allows user define there own distributed trigger plugin, whenever a request was received, Dify forwards it to the plugin and wait for parsed variables from it. + +1. Requests were saved in storage by `services/trigger/trigger_request_service.py`, referenced by `services/trigger/trigger_service.py`.`TriggerService`.`process_endpoint` +1. Plugins accept those requests and parse variables from it, see `core/plugin/impl/trigger.py` for details. + +A `subscription` concept was out here by Dify, it means an endpoint address from Dify was bound to thirdparty webhook service like `Github` `Slack` `Linear` `GoogleDrive` `Gmail` etc. Once a subscription was created, Dify continually receives requests from the platforms and handle them one by one. + +## Worker Pool / Async Task + +All the events that triggered a new workflow run is always in async mode, a unified entrypoint can be found here `services/async_workflow_service.py`.`AsyncWorkflowService`.`trigger_workflow_async`. + +The infrastructure we used is `celery`, we've already configured it in `docker/entrypoint.sh`, and the consumers are in `tasks/async_workflow_tasks.py`, 3 queues were used to handle different tiers of users, `PROFESSIONAL_QUEUE` `TEAM_QUEUE` `SANDBOX_QUEUE`. + +## Debug Strategy + +Dify divided users into 2 groups: builders / end users. + +Builders are the users who create workflows, in this stage, debugging a workflow becomes a critical part of the workflow development process, as the start node in workflows, trigger nodes can `listen` to the events from `WebhookDebug` `Schedule` `Plugin`, debugging process was created in `controllers/console/app/workflow.py`.`DraftWorkflowTriggerNodeApi`. + +A polling process can be considered as combine of few single `poll` operations, each `poll` operation fetches events cached in `Redis`, returns `None` if no event was found, more detailed implemented: `core/trigger/debug/event_bus.py` was used to handle the polling process, and `core/trigger/debug/event_selectors.py` was used to select the event poller based on the trigger type. diff --git a/api/app.py b/api/app.py index e0a903b10d..99f70f32d5 100644 --- a/api/app.py +++ b/api/app.py @@ -1,7 +1,7 @@ import sys -def is_db_command(): +def is_db_command() -> bool: if len(sys.argv) > 1 and sys.argv[0].endswith("flask") and sys.argv[1] == "db": return True return False @@ -13,23 +13,12 @@ if is_db_command(): app = create_migrations_app() else: - # It seems that JetBrains Python debugger does not work well with gevent, - # so we need to disable gevent in debug mode. - # If you are using debugpy and set GEVENT_SUPPORT=True, you can debug with gevent. - # if (flask_debug := os.environ.get("FLASK_DEBUG", "0")) and flask_debug.lower() in {"false", "0", "no"}: - # from gevent import monkey + # Gunicorn and Celery handle monkey patching automatically in production by + # specifying the `gevent` worker class. Manual monkey patching is not required here. # - # # gevent - # monkey.patch_all() + # See `api/docker/entrypoint.sh` (lines 33 and 47) for details. # - # from grpc.experimental import gevent as grpc_gevent # type: ignore - # - # # grpc gevent - # grpc_gevent.init_gevent() - - # import psycogreen.gevent # type: ignore - # - # psycogreen.gevent.patch_psycopg() + # For third-party library patching, refer to `gunicorn.conf.py` and `celery_entrypoint.py`. from app_factory import create_app diff --git a/api/commands.py b/api/commands.py index 8ca19e1dac..e15c996a34 100644 --- a/api/commands.py +++ b/api/commands.py @@ -15,12 +15,12 @@ from sqlalchemy.orm import sessionmaker from configs import dify_config from constants.languages import languages from core.helper import encrypter +from core.plugin.entities.plugin_daemon import CredentialType from core.plugin.impl.plugin import PluginInstaller from core.rag.datasource.vdb.vector_factory import Vector from core.rag.datasource.vdb.vector_type import VectorType from core.rag.index_processor.constant.built_in_field import BuiltInField from core.rag.models.document import Document -from core.tools.entities.tool_entities import CredentialType from core.tools.utils.system_oauth_encryption import encrypt_system_oauth_params from events.app_event import app_was_created from extensions.ext_database import db @@ -321,6 +321,8 @@ def migrate_knowledge_vector_database(): ) datasets = db.paginate(select=stmt, page=page, per_page=50, max_per_page=50, error_out=False) + if not datasets.items: + break except SQLAlchemyError: raise @@ -1227,6 +1229,55 @@ def setup_system_tool_oauth_client(provider, client_params): click.echo(click.style(f"OAuth client params setup successfully. id: {oauth_client.id}", fg="green")) +@click.command("setup-system-trigger-oauth-client", help="Setup system trigger oauth client.") +@click.option("--provider", prompt=True, help="Provider name") +@click.option("--client-params", prompt=True, help="Client Params") +def setup_system_trigger_oauth_client(provider, client_params): + """ + Setup system trigger oauth client + """ + from models.provider_ids import TriggerProviderID + from models.trigger import TriggerOAuthSystemClient + + provider_id = TriggerProviderID(provider) + provider_name = provider_id.provider_name + plugin_id = provider_id.plugin_id + + try: + # json validate + click.echo(click.style(f"Validating client params: {client_params}", fg="yellow")) + client_params_dict = TypeAdapter(dict[str, Any]).validate_json(client_params) + click.echo(click.style("Client params validated successfully.", fg="green")) + + click.echo(click.style(f"Encrypting client params: {client_params}", fg="yellow")) + click.echo(click.style(f"Using SECRET_KEY: `{dify_config.SECRET_KEY}`", fg="yellow")) + oauth_client_params = encrypt_system_oauth_params(client_params_dict) + click.echo(click.style("Client params encrypted successfully.", fg="green")) + except Exception as e: + click.echo(click.style(f"Error parsing client params: {str(e)}", fg="red")) + return + + deleted_count = ( + db.session.query(TriggerOAuthSystemClient) + .filter_by( + provider=provider_name, + plugin_id=plugin_id, + ) + .delete() + ) + if deleted_count > 0: + click.echo(click.style(f"Deleted {deleted_count} existing oauth client params.", fg="yellow")) + + oauth_client = TriggerOAuthSystemClient( + provider=provider_name, + plugin_id=plugin_id, + encrypted_oauth_params=oauth_client_params, + ) + db.session.add(oauth_client) + db.session.commit() + click.echo(click.style(f"OAuth client params setup successfully. id: {oauth_client.id}", fg="green")) + + def _find_orphaned_draft_variables(batch_size: int = 1000) -> list[str]: """ Find draft variables that reference non-existent apps. @@ -1420,7 +1471,10 @@ def setup_datasource_oauth_client(provider, client_params): @click.command("transform-datasource-credentials", help="Transform datasource credentials.") -def transform_datasource_credentials(): +@click.option( + "--environment", prompt=True, help="the environment to transform datasource credentials", default="online" +) +def transform_datasource_credentials(environment: str): """ Transform datasource credentials """ @@ -1431,9 +1485,14 @@ def transform_datasource_credentials(): notion_plugin_id = "langgenius/notion_datasource" firecrawl_plugin_id = "langgenius/firecrawl_datasource" jina_plugin_id = "langgenius/jina_datasource" - notion_plugin_unique_identifier = plugin_migration._fetch_plugin_unique_identifier(notion_plugin_id) # pyright: ignore[reportPrivateUsage] - firecrawl_plugin_unique_identifier = plugin_migration._fetch_plugin_unique_identifier(firecrawl_plugin_id) # pyright: ignore[reportPrivateUsage] - jina_plugin_unique_identifier = plugin_migration._fetch_plugin_unique_identifier(jina_plugin_id) # pyright: ignore[reportPrivateUsage] + if environment == "online": + notion_plugin_unique_identifier = plugin_migration._fetch_plugin_unique_identifier(notion_plugin_id) # pyright: ignore[reportPrivateUsage] + firecrawl_plugin_unique_identifier = plugin_migration._fetch_plugin_unique_identifier(firecrawl_plugin_id) # pyright: ignore[reportPrivateUsage] + jina_plugin_unique_identifier = plugin_migration._fetch_plugin_unique_identifier(jina_plugin_id) # pyright: ignore[reportPrivateUsage] + else: + notion_plugin_unique_identifier = None + firecrawl_plugin_unique_identifier = None + jina_plugin_unique_identifier = None oauth_credential_type = CredentialType.OAUTH2 api_key_credential_type = CredentialType.API_KEY @@ -1599,7 +1658,7 @@ def transform_datasource_credentials(): "integration_secret": api_key, } datasource_provider = DatasourceProvider( - provider="jina", + provider="jinareader", tenant_id=tenant_id, plugin_id=jina_plugin_id, auth_type=api_key_credential_type.value, diff --git a/api/configs/feature/__init__.py b/api/configs/feature/__init__.py index a02f8a4d49..ff1f983f94 100644 --- a/api/configs/feature/__init__.py +++ b/api/configs/feature/__init__.py @@ -174,6 +174,33 @@ class CodeExecutionSandboxConfig(BaseSettings): ) +class TriggerConfig(BaseSettings): + """ + Configuration for trigger + """ + + WEBHOOK_REQUEST_BODY_MAX_SIZE: PositiveInt = Field( + description="Maximum allowed size for webhook request bodies in bytes", + default=10485760, + ) + + +class AsyncWorkflowConfig(BaseSettings): + """ + Configuration for async workflow + """ + + ASYNC_WORKFLOW_SCHEDULER_GRANULARITY: int = Field( + description="Granularity for async workflow scheduler, " + "sometime, few users could block the queue due to some time-consuming tasks, " + "to avoid this, workflow can be suspended if needed, to achieve" + "this, a time-based checker is required, every granularity seconds, " + "the checker will check the workflow queue and suspend the workflow", + default=120, + ge=1, + ) + + class PluginConfig(BaseSettings): """ Plugin configs @@ -263,6 +290,8 @@ class EndpointConfig(BaseSettings): description="Template url for endpoint plugin", default="http://localhost:5002/e/{hook_id}" ) + TRIGGER_URL: str = Field(description="Template url for triggers", default="http://localhost:5001") + class FileAccessConfig(BaseSettings): """ @@ -331,12 +360,42 @@ class FileUploadConfig(BaseSettings): default=10, ) + inner_UPLOAD_FILE_EXTENSION_BLACKLIST: str = Field( + description=( + "Comma-separated list of file extensions that are blocked from upload. " + "Extensions should be lowercase without dots (e.g., 'exe,bat,sh,dll'). " + "Empty by default to allow all file types." + ), + validation_alias=AliasChoices("UPLOAD_FILE_EXTENSION_BLACKLIST"), + default="", + ) + + @computed_field # type: ignore[misc] + @property + def UPLOAD_FILE_EXTENSION_BLACKLIST(self) -> set[str]: + """ + Parse and return the blacklist as a set of lowercase extensions. + Returns an empty set if no blacklist is configured. + """ + if not self.inner_UPLOAD_FILE_EXTENSION_BLACKLIST: + return set() + return { + ext.strip().lower().strip(".") + for ext in self.inner_UPLOAD_FILE_EXTENSION_BLACKLIST.split(",") + if ext.strip() + } + class HttpConfig(BaseSettings): """ HTTP-related configurations for the application """ + COOKIE_DOMAIN: str = Field( + description="Explicit cookie domain for console/service cookies when sharing across subdomains", + default="", + ) + API_COMPRESSION_ENABLED: bool = Field( description="Enable or disable gzip compression for HTTP responses", default=False, @@ -915,6 +974,11 @@ class DataSetConfig(BaseSettings): default=True, ) + DATASET_MAX_SEGMENTS_PER_REQUEST: NonNegativeInt = Field( + description="Maximum number of segments for dataset segments API (0 for unlimited)", + default=0, + ) + class WorkspaceConfig(BaseSettings): """ @@ -990,6 +1054,44 @@ class CeleryScheduleTasksConfig(BaseSettings): description="Enable check upgradable plugin task", default=True, ) + ENABLE_WORKFLOW_SCHEDULE_POLLER_TASK: bool = Field( + description="Enable workflow schedule poller task", + default=True, + ) + WORKFLOW_SCHEDULE_POLLER_INTERVAL: int = Field( + description="Workflow schedule poller interval in minutes", + default=1, + ) + WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE: int = Field( + description="Maximum number of schedules to process in each poll batch", + default=100, + ) + WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK: int = Field( + description="Maximum schedules to dispatch per tick (0=unlimited, circuit breaker)", + default=0, + ) + + # Trigger provider refresh (simple version) + ENABLE_TRIGGER_PROVIDER_REFRESH_TASK: bool = Field( + description="Enable trigger provider refresh poller", + default=True, + ) + TRIGGER_PROVIDER_REFRESH_INTERVAL: int = Field( + description="Trigger provider refresh poller interval in minutes", + default=1, + ) + TRIGGER_PROVIDER_REFRESH_BATCH_SIZE: int = Field( + description="Max trigger subscriptions to process per tick", + default=200, + ) + TRIGGER_PROVIDER_CREDENTIAL_THRESHOLD_SECONDS: int = Field( + description="Proactive credential refresh threshold in seconds", + default=180, + ) + TRIGGER_PROVIDER_SUBSCRIPTION_THRESHOLD_SECONDS: int = Field( + description="Proactive subscription refresh threshold in seconds", + default=60 * 60, + ) class PositionConfig(BaseSettings): @@ -1088,7 +1190,7 @@ class AccountConfig(BaseSettings): class WorkflowLogConfig(BaseSettings): - WORKFLOW_LOG_CLEANUP_ENABLED: bool = Field(default=True, description="Enable workflow run log cleanup") + WORKFLOW_LOG_CLEANUP_ENABLED: bool = Field(default=False, description="Enable workflow run log cleanup") WORKFLOW_LOG_RETENTION_DAYS: int = Field(default=30, description="Retention days for workflow run logs") WORKFLOW_LOG_CLEANUP_BATCH_SIZE: int = Field( default=100, description="Batch size for workflow run log cleanup operations" @@ -1107,12 +1209,21 @@ class SwaggerUIConfig(BaseSettings): ) +class TenantIsolatedTaskQueueConfig(BaseSettings): + TENANT_ISOLATED_TASK_CONCURRENCY: int = Field( + description="Number of tasks allowed to be delivered concurrently from isolated queue per tenant", + default=1, + ) + + class FeatureConfig( # place the configs in alphabet order AppExecutionConfig, AuthConfig, # Changed from OAuthConfig to AuthConfig BillingConfig, CodeExecutionSandboxConfig, + TriggerConfig, + AsyncWorkflowConfig, PluginConfig, MarketplaceConfig, DataSetConfig, @@ -1131,6 +1242,7 @@ class FeatureConfig( RagEtlConfig, RepositoryConfig, SecurityConfig, + TenantIsolatedTaskQueueConfig, ToolConfig, UpdateConfig, WorkflowConfig, diff --git a/api/configs/middleware/vdb/weaviate_config.py b/api/configs/middleware/vdb/weaviate_config.py index 6a79412ab8..aa81c870f6 100644 --- a/api/configs/middleware/vdb/weaviate_config.py +++ b/api/configs/middleware/vdb/weaviate_config.py @@ -22,6 +22,11 @@ class WeaviateConfig(BaseSettings): default=True, ) + WEAVIATE_GRPC_ENDPOINT: str | None = Field( + description="URL of the Weaviate gRPC server (e.g., 'grpc://localhost:50051' or 'grpcs://weaviate.example.com:443')", + default=None, + ) + WEAVIATE_BATCH_SIZE: PositiveInt = Field( description="Number of objects to be processed in a single batch operation (default is 100)", default=100, diff --git a/api/constants/__init__.py b/api/constants/__init__.py index 248cdfc09f..e441395afc 100644 --- a/api/constants/__init__.py +++ b/api/constants/__init__.py @@ -56,11 +56,15 @@ else: } DOCUMENT_EXTENSIONS: set[str] = convert_to_lower_and_upper_set(_doc_extensions) +# console COOKIE_NAME_ACCESS_TOKEN = "access_token" COOKIE_NAME_REFRESH_TOKEN = "refresh_token" -COOKIE_NAME_PASSPORT = "passport" COOKIE_NAME_CSRF_TOKEN = "csrf_token" +# webapp +COOKIE_NAME_WEBAPP_ACCESS_TOKEN = "webapp_access_token" +COOKIE_NAME_PASSPORT = "passport" + HEADER_NAME_CSRF_TOKEN = "X-CSRF-Token" HEADER_NAME_APP_CODE = "X-App-Code" HEADER_NAME_PASSPORT = "X-App-Passport" diff --git a/api/constants/pipeline_templates.json b/api/constants/pipeline_templates.json new file mode 100644 index 0000000000..32b42769e3 --- /dev/null +++ b/api/constants/pipeline_templates.json @@ -0,0 +1,7343 @@ +{ + "pipeline_templates": { + "en-US": { + "pipeline_templates": [ + { + "id": "9f5ea5a7-7796-49f3-9e9a-ae2d8e84cfa3", + "name": "General Mode-ECO", + "description": "In this template, the document content is divided into smaller paragraphs, known as general chunks, which are directly used for matching user queries and retrieval in Economical indexing mode.", + "icon": { + "icon_type": "image", + "icon": "52064ff0-26b6-47d0-902f-e331f94d959b", + "icon_background": null, + "icon_url": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAAT1klEQVR4Ae1dzXPcRBbvlsZ2xo6dcbwXinyMC+IDW5WY08IJh2NyIFRxJLvhHyDxaWv3kuS0e4v5CwjLHqmCHMgxMbVbBZxIOEAVCWXnq7hsMiaJPf4aad9Pmh5rNBqPPmdamtdVdkutVuv1r396\/fX0RgpNwspvterurqjatqiatlWxhKgYUhyHeLaQFYrwh5OqE3v+SSkqtrruSS\/yoRRijbBa89bRSZN7aVLYq7hu2eKBgfzSWLXpeqkkVmdfmXau4fogA8nc37CyUqs0TLEghfUOEatKhJoXspNU\/ZVqOJ8mbXGHCLlq2\/ZdKY07ZkMsz85Ot5E6a2T6QsB7j2oL9Aa+QxVdoArhryMYhiEMUnmmaQpJKg1\/SEMgcJxzHJumm4ZjFVR+dT4MMWEp8OcNOLdI3algWQ3KQ52GbTl5LcuNGw2L8lEfExBASiHt5YZhfDZ3ZPpOQJZUkzIjIDSdZVgXbCnfI4kXlNQgS6lkOkQD2UZGRlqEU3k47g8CjUZDgIy7uzsUN8TOzm7bg4kcq0Tpq68f+8P1tgspnqROQId4JXGRXrlLalwG0o2NjRLZRh3y4ZyDngiAhNvbWw4ZlZYEEUlLXH\/t6PTVtKVOlQn3H\/7vnLSNazSuqELQkZGSOHCg7MRpC87lZY\/A1tZ2i4x4GoiYtkZMhYCk9aoN0\/6UZFyAoEw8oFCcAK24vr7uHTd+ZY7IxTRm0okJuPKodtGy7SvobtG1lstl0npjxUGfa9JCABqxXq8rItJs2VpMOj6MTUBnrGeKyzQXuwQJR0dHxMTERGu22pKaDwqFAMaFICHIiEDtv3Ti2Mxi3ErGIiC6XMuwv6Sx3jxrvbjQ5\/u+zc0th4hY+sHSjTEq34\/TJUcmYJN8tzHRwDrd1NRka70u35Cy9FERgDZ8\/vyF0yUTkVaNEXk6KgkjEdBLPqzhTU4eZPJFbbWC5QcJX7x46awjxiFhaAL6yQfNx+t5BWNTzOqgG4YmxGJ2VBKGIiCTL2bLDNFtcUnYubEaAFpzwlFFt8uaLwAgTnJ6Q3ADHKEluaq1bX9JiqvSC5qeBPz1YQ07G\/OYcGDMx91uL0iH9zq4oeYF4MyuaV3uhca+XTBtrV0QwvgUBR86NMUTjl5o8nUHAUxMfv\/9uWOBQ13z4onjM0vdoOlKQGfcZ9o\/YIdjfHycdze6IcjpgQhgnXBjYwPX1mjb7s1uyzNdu2Da270G8sGKhbfWAjHmxH0QAGewO0ah0thx7AQCcwcS0O16xTmM+7C3y4ERiIOAZ2t24f7D2rmgMgIJSCZVzuAR5FNWyUE3cxojsB8CmDsoBUbfp1wLmhV3EPDXR7XLapsN3S8HRiAJAuiKYZ5Hw7nqrmE5hive8joISJ9QXUAGqE8OjEAaCMAoGYE04kW\/FmwjIMZ+0H5gLP44MAJpIODhU4W04AVvmW0EVGO\/0VE2KPWCxMfJEfBoQXyk1gotAq48rs3z2K+FCx+kjAC0ICYlFBbwma4qvkVA+jzvAhK561XQcJw2Aq1JrWUtqLJbBJSGfAeJ3P0qaDhOGwF8lotAmtDhGo4dAmJmQiZd80hgDQgUOGSBABwSqG5YzYYdAjbMxgIeyOTLAnYuUyEA8oGECPAPhNghoG1LR\/sZhnsRFzgwAlkgAHtBJ9juONAhIDHzFBLhp4UDI5AlAoqAjmc0elCTgKKKhwZ5nkI6B0YgLQSUkqPe2FF6zS7YnYAodqb1MC6HEfAj0JyEILmKfyWajVTJixxbvQCNnISNDUvcvl0X9+7tiKfPGuLp04Yj+fi4IY68WhKnTo2KkyfHxMyMfmN6EBAWVrCahldciVVpadu3MQOenJzMSRMMp5gg2uefvxC\/3HPdYvRC4a23DoizZya0IyLM9fEJJ\/mOPF2SdqOCoaBHNfaqV9+v443\/\/vtN8csvO+Lxk93WG3\/kSEnMHDbpjR8TADvrMEg5bt3eEDdvbpCZe7Bn06C6f\/fdprh7d8sh4bvvjgdlGUgalmKcb4jtRlX++uDpJWLitbGxMTLB0kdIhQwA\/PzfL3oCj+4Gb3tWRBykHF\/fXBdff72uIIkVA5uzZ\/UwscO3IvhmBB8sleCNHlvE8M+sW\/jii5cCb36YgO7pX58\/d7Rj2kAPUg7UP4h8cydonEdjvVOesd7jx7viEf3dvPmScGjXlCBxuSyFDprQ09tWSrBUBfU8iWHaO\/M8ACws+bzC4L563RIffJDOeHaQcuClQrfrDePjUpwhbfbu6c7eCkMS\/L1Nw5FbNEm5SVpzg7BQAXXBcGXQkxP1mYchjePOMgwE1ImAGLsEvfUKyF4xwEeXmTQMWg4QxjvmA\/kuXZwOJJ+\/ru+eLotLlypivNxqYnoxbZrEPPdnHeg59bzyOCTQaRsOwCcN6I69b3+c8gYpB7QfXgBvgOaDhgsbkPeMb9z3Cy3dJMUl7PO75VPKjjzrTu+9Ht1y9zkdoAP8pAFv+3fftjdglDIHLcfdH9s1+MyMEUrz+esITTh3on2L9fatuj9bX8\/xuy8ItCR4SDsC3kmh61Rohl0vU\/m98aDl+PFu+1rfmTMHveJFOj5J4z5vuBdyHdF7T1bH1AO7v8Gmyyy4Riv7aYUnT+KXNWg5MKP1BuxwxA2YKXvD02d7ExNver+OPTYHVYN+xYkWovWZhGAZIa2QpCsftBz+cdrRo\/EJ6J\/1JsElrbZR5WjXBSvBOB4OBLQjoP9tTdIMRyPMGP3PGbQc\/ucn0Vp+bY4FaV2CdgR8NcFYxw\/q9OH41Ru0HDM+2ZOsaz7xDWuOHmmfFftx6+d5axKi1mb6+fCgZ83NpQfOqVPxDRQGLceJuXa\/PD\/6lmWCsOuW5l\/PPHmyvexu92WV7uFaxaCtOK0mIW+\/VW5bvY8LAtbNsCUVNwxaDv9WGxaQb91q35YLUzdsZ\/q7b2zHDTK0EXCQggQ9G+OT839Ovo+bZN0Mcg1aDjzfv4AMTeYfzwVhqNKwlOPfS4a1kH98qfIPIo4\/SMpQWqxbJbHagOlREu2nqjZoOc6fn2rrDbC7s7RUC6UJofmWPlnr2EsGNjoF8+PFv16BQMqRoC7CvfEGjVNosgaz8yjhNFmJnDsXf9fA\/6xBygET+9KIFD\/9tLcrskvLpD\/9vC2+IwNdZWgwNeXqEXS1MNy9cWNd\/Oe\/dfrRaRpgecJ77x0Uf3xjsN2vEqded7dJ5f2HzxwpDx+eVte0ir+lveEg+za\/kLAU+fDDKTGf0fhmkHKg601iHQSsdDJIhTzPntUQCe0J6EhJ\/0CAH2mf+Blt1alxEMYy2KI6QTPnt\/50QEBjZB0GJUeQfV+Yuu5nPxjm\/qzy5I6AWQGRp3LRxUIb+s20utUBVtPnz09qNelQsjIBFRI5jEFEmGvBYubxE7Lv23DHeugR8JEWeoTTC7Sc1YceIS58TMC4yPF9qSCgCJj9oCkVcbmQoiLABCxqy+akXkzAnDRUUcVkAha1ZXNSLyZgThqqqGIyAYvasjmpFxMwJw1VVDGZgEVt2ZzUiwmYk4Yqqpjxv\/UrKiL71At+WnTwTKqLHPtAFfpSbqxhQtcog4zYe9XBM6kucqQBsdqKywUB8cYHeUhV5lhZekiFZXFUz6RoIJjUwwYviWW3t6F1kcMrU5Lj3BCQPZMKxwSrqAapWo8B2TOpcJx0BpEvzx5SvZpT2y44iRk6XJIl8ZCKsdY\/\/lnr+KCnm2dSL6BBlsvojv\/+t8ORDUN1kcNbv7SOVRes5TIMLH6D3vqwlU\/qIRXk18EzqS5yhMU9Tj4tCQjgk4a4HlKhdfwm74PwTKqLHEnbodf92hGQPZO6TVZkD6leUmpHQPZM6jbP0HhI9bJRh2P2TOq2QpE9pHp5pp0GVN\/8eoWMe4xxVNSgi2dSXeSIil\/U\/NoRMGoFOH++EdCOgGl6borjIdX\/\/DhaVFHCr82xHhg26CJHWHnj5tOOgOyZ1G3KofGQGpe5Wd3HnkldZIvsIdXLHe00IHsmdZunyB5StSYgxkmD9JCK5+vgmVQXObxkyeJYOw2ISrJnUrep2UNqFpQPWSZ7JhWOdyv2kBqSMFllY8+kxTZI1dYe0E\/oYfdMGmRn6Mco6Jw9pAahkrM0LEbDRMxvptWtGll5JtVFjm71jpKuDFJzowGjVC6rvCCADp5JdZEjCc5MwCTo8b2JEVAE1HIZJnHtuIDcIMAEzE1TFVNQJmAx2zU3tWIC5qapiikoE7CY7ZqbWjEBc9NUxRSUCVjMds1NrZiAuWmqYgrKBCxmu+amVlp7x1Io6uIRlOVQLZJerPVeMPY82TPpXmPrgseeRPGP1FactgTUxSMoyxGfZPvdqQhofrz41yvIWC6X98vf12swfbpxY13s7Li\/gxvl4bu7Qvz087Zzy9zcaJRbO\/KyHB2QpJZQr286ZWk3BoTGCfIN2G+PoCxHalzbtyCtumCMcdgz6V576YLHnkTpHakuWKtlGHR57Jl0r5F1wWNPovSPtCEg3na\/yfsweybVBY\/0KddeokHuctaQZNvRB\/ztRSU708UjKMuRrB3D3O3h2ppBvNOCgLp4BGU5wlAoWZ42AiYrKr27dfEIynKk16ZhStJmDKiLR1CWIwxt0sujDQHTqxKXlCcEtCGgLh5BWY7s6WtZ7oRX0vzDEFKs4pGNhpX9k\/d5gi4eQVmOfRoppUtqEmJLEFCToItHUJajv4QAAbVYhtHFIyjL0WcCWrb9Ox5p24PtgnXxCMpyZE9Ay3J\/v0UKuapNF4xq6+IRlOXIloTeTTfYA85LKRdKJVOMjIxk++QepY+PG0IHj6AsR4+GSnh5Z2dH7JLhJk1GbshfHzy9ZEt5bWxsTExMjCcsOp3bYQUSZBMYpfSzZybE2bMTUW7pyMtydECSSsLGxobY3NwCARdLDWk7azE0Ckyl8DQKAXnKZUPc\/JrMs+rRxqZpegRlOdJozc4yLMttUymNVXnvUW1B2vZt0zTFoUNTnbkHmAKTJGghv5lWN5GK7plUFzy64R82\/cWLF\/S5BXXBUp6WKyu1asO0VwzDEJXKobBl9DUfgGfPpHuQ64LHnkTRjtbWfhfQguaInHV+Pe\/+w2dO\/zs9XRE0IYlWGudmBCIioMzxXz92WLrLMLa7Hae2SCKWx9kZgdAI7O421wBtcQc3uQSU7gmmxxwYgSwRUIvQNA15gOc0NaDtnCh2ZikAlz3cCGD9zw22VwPay0hU7HQz8H9GIH0EGo1mFyyNPQKaDXMZj4IG5HFg+qBziXsIYPkFwWyIZcROFzw7Ow2LmGWQj7thwMIhCwQU+cgQ9U6Tc80xID2NyPcNHrq97fpVyUIALnO4Edje3nIAsIXLNZy4kxDnyFhGxAQEChyyQEBpQMsyrqvyWwQ8cXR6mRKdblhlVJk4ZgSSIrC1teXsftA2x+rc7LQzAUGZLQLihPaEbyDe3Kwj4sAIpIaA6lltIa96C20joEGqkRi6Bg3IWtALEx8nQUDxCdrv9WPT171ltREQMxMy0f8EGVgLemHi4yQIrK+vO7cTtz7zl0OkbA9kHVOxDPsH+mSuOj5eFgcOHGjPwGeMQAQEMPZbX9+gr3\/F6mvHDs\/6b23TgLgILUh2Wos4hhtVXpgGEhziIIBvzZUrXv\/YT5XXQUBcoH76K4qcGfHLl676VDdwzAiERQDDuKb181f+sZ8qI5CAuGg25EekNmlCskPjQdehtLqJY0agFwL45mNraxtd7xoZnjo9atA9XQlIXfEq2UxfxU1Qo4N23REkPKfpiYDb9bpLedT1Ls6+QlzqEroSEPlfOz69RIPATzAOhB0\/k7ALipzcQgAcAVecuQNxp1vXq24gDbl\/aM6Kb9OseB4fLk1NTbLZ\/v6QDe1VkO75cyiqBm1qiDuvHT\/8Zi8w9tWAuBmzYsOS71OBqygYD+CZcS9Yh+96G\/loycUYle+HQaGnBlSF4Os5Wh+EJqyyJlSocAwEOsg3Ik\/vN+7zohaagLjJT8KDBw8K0+ypRL3P4+OCIYAx38uXL91uF5ovAvkARSQC4gYvCfEt8eTkJJMQwAxhUBMOrPURkSKTD5BFJiBuapLwS0xM8B1xuXyAt+wAzBAFrPPV63Wn+8WEA2O+sN2uF6ZYBFQF3H\/wdImmxBdxPjY2SiQsszZU4BQ0xngPxgXb281PeGmpxbSMK5isxqlyIgLigfcf1i5IYV8j1woVdMnQhvC0xaF4CLRpPdrhIOuWqyeOzywlqWliAuLh6JIbprhG86FzOAcRJyYmyN+gdr8GC\/E4REQA9nzY1\/XYiC7T9tpHcbpc\/6NTIaAq1NGGtn0ZSzVIAwFHR0dZIyqAchb7iUdkWcXWWtNYJZXapEpAJdG9B0+v0O8\/\/EURERrRJeMYa0UFkoYxxnf4LHdnZ9sxJMA5ApHEMVQuWcZS3LFet+pmQkD1ML9GVOkgIxazS6USddeITXWJ4z4hAHLhD9ZO2OHCX4BjgmVpyxuGJa6nTTxVzUwJqB6y8rg2T2tGNFmR72DpRqV7Y2hJLGpjWQfHiNUfSKqCe71dbJVP5RmGWBHIX1eszSHgVw+UBsM6ncqvSNa00\/PfjvNlyvsNNcJy80vJoDyppbW3ZGrFdi+IJiwVmrAsEEBYQzxFa0jVbqTsXgpfSQUBuOWDZzSbnFNJYxnuMrLSdN3k7TsBuwmy8lutSo6TqkTICkhpCatCv6Z9HPlp4FulyAm4jiUfdY6YlGVHmvd6EY+p4daoB13rqFvzp9cofY2Wx5zr9NNsDwxhrDXop7EIq1Ua+aymMYPteHaMhP8DKleEJHlBQFwAAAAASUVORK5CYII=" + }, + "copyright": "Copyright 2023 Dify", + "privacy_policy": "https:\/\/dify.ai\n", + "position": 1, + "chunk_structure": "text_model", + "language": "en-US" + }, + { + "id": "9553b1e0-0c26-445b-9e18-063ad7eca0b4", + "name": "Parent-child-HQ", + "description": "This template uses an advanced chunking strategy that organizes document text into a hierarchical structure of larger \"parent\" chunks and smaller \"child\" chunks to balance retrieval precision and contextual richness.", + "icon": { + "icon_type": "image", + "icon": "ab8da246-37ba-4bbb-9b24-e7bda0778005", + "icon_background": null, + "icon_url": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAAYkklEQVR4Ae2dz28cx5XHq2f4m5JIyo4R2+t46B+H1S5gGUiwa1\/EAFmvkUtsIHGOq6y9Z1vJHyDpD0iknG2vneMmBmxfFo5twPTFzmIDRAYS7cFKSMU\/FCS2RVKiSIpk975PNWtYU9M9nB\/dM8PueoLY3TXVVV2vv\/N+1auaQA0JLV27XpNHqe3K\/yAIZ1WkZitK3c\/jhUEwG8g150I1\/df+E8hn+5\/bnxT3PFArMuaVhgFyTfkeBSpa5jRU6irlUVhZrsafL8\/fPac\/4\/NBUtDvzpeWrs\/ujquFqgpPhZWgJsA6Kc9Q6\/dz+P6EA5G6FFXUsoqij6Kocqm6pRbn5+fqAO4Hj\/oCQJFuCzKYU5GKOPK\/iSqViqoEgaqOVFUgR\/5TBgVy5Bqq7pXpi70\/pr5dVvTzKBJuyn+buA6tsnB3V+oIzqJQ1w1DOYaR2pUj54kkoBTJuahGKr+Yv2vuUmKdDAtzAyCSLpwMTwdR8D153gXzzIBlpFrVQKvKcXR0tA44U8cf+8OBXQEoYNzZ3la7O7tqe2fH7XhZoHr+obvvfNX9IKvrzAEI8NSEej4KoheMXQboxsfH1OjYmAafkWZZDcK3kx0HAOHtrS21vb1jS8ll0Umvit14Prue4pYyBeCVz794qhJULkjTNZofHRlRE1OT+si1p8PFga2t2zEY9yVj5hIxEwDiwYpF8oqwdwEWe+DBheIQUnH95npdIkaBeqMSBWey8KR7BuDVv1x\/Xkzdc6hbVOvk5KSamBgvDvf9SOocQCJubGzEQJRwThiFZ3q1D7sGoLb1JtVZ8bxe4AnHxkbV9PR03VutP7U\/KRQH8J4BIWCExNa\/+ODX7zjT7SC7AqBWuVH0ugQ3T3qp1y3rD\/d9m5tbGog6FEToJgie7kYldwzAPXvvPWFfjTjdsWNH6\/G6w81S\/\/SdcgBpuLZ2w9iGeMrf7hSEHQHQBh8xvKNHj3jwdfrWClYfEN64cVMRUxTqGIRtA9AFH5LPx\/MKhqYuh4MaRhJ2A8K2AOjB1+WbKdFt3YIwnmw9gFHS+OtSpYba9ZLvAGaV9GO0IdgAI2AFzOhIyQH8OBCAS3+5fkGJt4vDgc3n1e4BHC3xx2Cj7hcIZiQX4OxB7Gipgq9c++K05Ki8QsMzM8e8w3EQN\/3nmgM4JqurazoDRyThmQfvueNiGmtSAajtviD6HTMcU1NTfnYjjYO+PJEDxAlv3boluXRqRTKiHk0Lz6Sr4CC6APjIYvFTa4k89oUtOABmmB0DQ3t5Aom1EwGI6hXP+insPuZ2PXkOdMMBa2p24crn159KaiMRgGL3aeMR8Jms5KSbfZnnQCsO4DsYAVYRjZrkFTcBUGw\/wFcDeKhfT54DvXAAVUx6nlAtnAh14ordXhMARV+fpsL0kWm7nj\/3HOiaAyQlQyIRn3elYAMAsf2kXg3E7qGW+zx5DvTEgTqexCEJx8PTdmMNADS239i4Tyi1meTPe+eAJQVZpFanOgCXPr1+Ukq97VdnjT\/JkgNIQZwSoQXxMxZM23UAhpVYNI6OaoPRfO6PngOZccA4tbLUc8E0WgegJBOeotCrX8Maf8yaAyzLhQzWONcA1J6JTB5T4J0PuOApDw6wIUFdDbN+XEgDcHd8d4ELDz644CkvDgA+QKhpSi1w1ACUD7T0q8i+LJ48B\/LkAHv\/QOFubAdqAMraukcoHB2RyWNPngM5cmAvYRU7sEY32uUV51hfVKsxHvnA0z4H1rYj9dZnW+ry6q7683qoLq\/sqFUpo9zQfVMV9XfTVfWPs1V1YmZEPXbXqKLMUyMH2IxKU6C00ItjLnsOiEFn4y3lvAJcL368qT7827b+fxAXPrkVKv5T39A\/CBife2jSg9EwRI57TgglNf4EewuOlkg+mJ2doazUZID30scbDRKuV6Y8UxtXPz4x5aWiMHJlZVWvJRY1PI8ErMHcpI0fKS8T\/fTyhsoaeIZ\/v1zeUvwHhD85Ue4cS1sKVnajXR2PCSpiCZaUUJ1PvLuifnb5VqrUe\/xro+o\/Hp5Q\/\/n4UYU0S6L7pqoaXNRNI\/r45\/++rtV1Wp2il4\/secKyPWZtpFoJZAmd6GJRwWUkpNLZj9YTgXdsNNCge+7hScU59FMBEPe49OQ9Y+rcyem6itX24F+3E9vWgH9nRV381hH1r3Jf2chIQFkrMjsiWwbPwlr2Zy4bAaafidp1CbChJgGeIUDz7Ac31B\/EA3bpJ6JWf5ygVl+6spkIbO7H1vx3aa+MKtkAUGIxsyMCuxoMqRdyUQJKAx9qFlAYiQcrfv35bXX20nqT2kTlPvfweANQW9WnTTt0Q11UMlQmu9As85D0v\/vrqS9lAiCASpJ85x+ZagJTGlAB368WjtVVrkaR\/Dmo\/q8\/EzCLyrcJEBIzTLMt7bpFOxfXI7ifQVXMHF3RRuiMB1X6wv\/ebChFMr126lgD+Kh39qNkFY2954Kv3frPiYR9+zuzDRKWhwGUtFEGMsJOFq3P1SVgGQbOGH+wuNqkBl87NaMIGhsCCNRLAkSSvddp\/WNjstOEo45Rzc9+sKbBaZ6jqMe6wytsKBUAUY8uqFC7Nvio85LMgLi2Gir35cePSN1GlmVVH7D9YWVXmwZJDk1RwViREEycl1VwLxjguXYfNpft6Rr7LQl8qNwk8NFmr\/VtcL2oZ2CKrYqtSY+aJOrHADR62WZGkc6Nt2nGhETD24UAZ6sQC3ab7RVnWR+v+78krmhAzPGlj5kx2Q8BmWcu4rEU0WcA4waPecF4nnyGvdcqvueCL8v65x6ZlhBM\/EUwACuDFDRjbTRoTGnBjh\/KjIRNSD\/Ub1b2W6\/2IRKWZymjFCyFBHz5SuNsxzO1sXqIxbx0A1ATYrHtPaSkCcnkVd\/uj2f5wErrMs9WxGNsAzIXLP+KSIDn9+Jd2kTWSxJlEWIxKp2jS520T17h2nYotmfxZETd3xD\/o8L+bTCqqNkwrvp1QcE1KpRwjGv4M2OSFA\/Mu755xrdk1qSIVAegYK\/wNuDl1ebkAfulAiZ3VoPPTUjGrst53vXt\/lgCUHQqPABd9Wu\/UFRiUoiFQDSJqS7lXf8xySO0U\/pZf1J0KjwAP11PliKd2GOAoB\/1fyCeOcmqhlj8VHQqPABdZwAVmueUWi\/tux42K++KToUHoPsCh8nec+1JO+DNc7uAdMdShOvSAdBeq4t0HNQUXJo9WQRQdTKGwgMQqWJLEhNbyyrLGSnWSVb0QfU7eXlFqFt4ALp5d6syK\/fix8mJpq5KNC94UCEZW1qbZynasfAAZIrrk1v7Ad0zkg1thzrMC3VXtVGOik4LyeRdn\/7vk60+ik6FB+B9041TWUng60eIxZ1lAdxJsyw24OxEWbu8SOeFB+CJmXQpgspNCsm0sg\/zrO8Ci02Oik6FH+GT946rM79tXIXGSx02ey8JaOywVXQqPADxgt0pLnYjYFcCO+426JAMz2Iv18R29U5IQb5+j39tpMHxwA50wZdmj\/XLPrSn4GD7cw9NFIT7rYdReAmoX6ZsscFefyYeyJFr1mMMQ1Y0ywWQwDaVQf0y3lIAEGkXg20\/w4VFSp\/qMMt+mQFA3iEWu32A5y6YYrlAGdRvaQDIQFl+6UrBtJSrTkImvapowOdKP7Naz3whinxsDJIVeKRGCqYNEa+431nRfCHc1XoAuizSj3dRChVsQIdkeevz7aYlmIMIybALwjlnkyKew5W+5tmLeiyNBDQv8GXZ4dT2gClflcU\/a7f3nQBUolkFZ+4zR+w3N6Wr0\/p44d9\/f9U0qY88E+2WjUolAXm5qLfzshj8zG\/3d8jCK37i3VXFIvEn7x1LnSLr1d6jf9SuK\/kop98yqV7GDAV\/uvaVTrs9fnwuLinJXwDo2l8MHUlkwjWGFajGpCm4TkI4tGk2QTftukdMhLJsVPnVV\/HSg9JJQF46KjNtuWYS+FyVSxudpGgh9fB23bZpxybqHOQs2fWLcF46AAK+tFkP94UCBpJNbeL+drKoARvAS\/vZBwM06tjARD2Tw1iW3VJLpYLTwEeQ+q3PtkUyJq+gA4DMJzOllzRrAZgADD\/PgIPBUtCktC8DZOZ5cYaw+WKHZM18VD9e+OaRQoPQqOBDA0CkBL\/X9uEXOzqM8omsmTWSAwCQ98eLfezOUW3QU2YTdfE8CX\/YZDsWqMC0bTvse7o9N1LPDTQDatspMu3bIOx1\/KbNYTkeGgAitV6WReL2HnrtMBGJxIs2nuX3319rkkrU4SXbRH8AMclBset1cm6AZ\/\/eiHt\/GggZww0JE\/U6fre\/QV8PPQD5xh\/kNbbDRHY+oC0XUEjLt7+T\/tt4ABFH5WX5rY\/fd7lAHJX8mKjtVsCzx5AGQrtOp+eMH8962DY5GmoAptlqnTI\/rT7gY1d8V02n1TdgZJ8ZVPgnstsCZYZoB8eBdjEFyMImEbbd9k07HPMAIVrgVwszdW1g9zeocwPAofOCecHsFm+\/YMMko8pwCPhtXqNekXDscEoq\/UHORBzTa54NMX0kHennPlHXSu17xPe+9mW9Kv3\/3\/eO1697OQHEjJM2Xep2\/OYLjeND+8NEQ+WEGEa54AM0F741rT3RdpiHFGHz8CSvFskHgHslG4C09dn37+i1Sf2lSwoRZTX+YZKERgIOzVww3\/gk5hMieftfZjoCDc4F93CvSyzLZHH6sFE\/xm++4MM0\/qEBIA6HK\/kIkTA\/240txT3xBuCNu83TR56hlm6BXdbxDwUAAYWbHIr0yiI1iTCGKwlZbO6CvVvgZHFfmcc\/FAAk7mYTNo8brLU\/7\/Q8jgc2rg8mtjgsVObxDxyA2D5ujA7J143aTQMUbeHE2BQHdgdvC5Z9\/AMHoLsRN9IPJyJrwvO1Qc2Ld\/vOus922nOfoWzjHzgAP\/yi8Udknry39xBJ2ot3bUHmlQdNZR\/\/wAHo7oPMrgV5kRv\/cxMT8uq3VbtlH\/\/AAejuBJ\/njlDMntjElNqgqezjHzgAscVsynPS3Ezdmf7cvk15P4\/uM5Rt\/AMHYD9ftu9r+DgwcADaninsyTNA3CxtGpNWB\/F6yj7+gQPwG84Opmk\/LJMFONzfBB6GLXDLPv6BA\/CEkx704d\/yC42QrmVTng6P3U+r87KPf+AAfOzOxvw0fi08L3KDvqwfaZdQ379c3tRrN554d6XpNsrMWmNX1TdVtgoOy\/itR870dOAAdDOHeXmtVpR1O3qm+1z7sp2gN\/ewVPKf5Dfc2OqXdpLih5TxGSD8+ze\/0ke3v6RnH\/bxJz1zlmUDByBG+A+dqbesc\/YAtTvhz3Rfq5AH97A\/DDuXumt323kBgJF72Xa3Vf7dsI6\/nTFmUWfgAGQQz8refTYhObLM2UvKtWuVbUP\/T7yz0pQiZj9ju+ekfj3xzmqT9LXvH7bx28+W93mjAZZ3byntEyBmnhZJY4gXh4Tqda+UeP+WRruSvtygtOk3jzUpAJps77Q1GcM0fsOHfh2HZk0IKi+WFI3TY90uK6Q9JJ+b6Eq2Cen6bvwNhhugcLSJe7JYkwLQ0lanDcP47THnfW7WhAwNABlwDABWxDWCkBeHymw3TQsnBjsyCUhJGw3RdwyAlaZ7kJb0nQRY7ksj2sPutKU6dRlL\/AVotn4GOf60ceRVPpQAZLCxCrzRBEI+4+Wxjx4ZM2b5IuW8OALYH0gMMW0zIKRYrAIbExK4H8LhcKWlvW1HXKvzv4DQtWeR6uxRmESDGn\/Ss+RZNrQAZNBpkqBbhgC+NMln+nN\/pwPJx6KmLIgwjisJf\/PduVQ7tN\/jz2KMnbZhANisBzptKYf6Rk0Bgl6JNlB5tJlGbogGwLbyktPaSSunLdq0qdWalH6P336ufp8PlQ2YNHikAQAhrtYumdga4Y1WwKM9bDUCxzbZu1LZ5b2cu9uw8Yz\/893ZlrFI+st7\/L2MqZd7jQQcegCaQQIUptJIYb8ssw5\/FpuPMoiX+Q1JNj0xW5Xt2UY62pfFzF6YfpBUvxFg5EEA3Twz7V\/45rQ4Vu1J+bzGn8c422nTAHAo4oDtPDAgwwtu1xNup03q9HtNhu2QsCblmVp7T5rX+NvrPb9a6YZRfn0OVctlX5Mx6JdRUYHSqR1R2JgaP+gH61f\/ZV+T0S8+2\/1E0R7WBHsVFe0BUE7KSLZNxvhbJSj0yh\/XIXL77rX9w3J\/HYCCvdKr4MPy0or6nKUHIMa9TYQ98iJX4rl959XvMLdbegCWfU3GoMFZegCWfU3GIAAY2k6IKKBlHmI3zE\/1DGKQ7fZZ9jUZ7fIpy3reCbG4WfY1GRYrBnJakfBfqeOAOALDuCZlIGgYQKeVIIj0LydHUTlVMDwv85qMAWBOhbtxwnGgguXSOyG8AALEbuoXa1LsedtuX1Sna1K67ecw3Wd8EJ65IvMfy5yEJXVCGDuUlLNHGthByyrju5v\/EvMjy5rfK7Ep61xDu+3Dcm60bajCq5XK3lxw3TU+LKPI+DmxBeOs6cbEUbOsspN8RHL\/kpZ1Aj76KHsA2vaCgyvXvjhdUZVXxsfH1PR0NinoGWOjr82VZU1GX5nqdHbzxk11e3tbBZXg6WDp2vWFSEXvVatVNTNzzKlazssyrMkY5Ju9sXZDbe\/sSCJW8G2ckGUepi4WuSg5lWlNxiBetTXpsaxn4v907SudizU3O4tYHMQzDW2fRV2TMUiGm3T8B+4+HhgALskD1WZnZ1Sl4iMzSS8HrzaPNSlJfRW5bEdigGura0r076UHvn78Ub0mROIylwSKtW0xDMfHs\/+RmCIwFM81jzUpReBNJ2MwQWgVqqvctyfuIn0BOj15DuTJgR1xPqAoiC5x1AAUL3iRi3DHAxA+eMqPA7t7GBNTbx+A1a3qIl0iAcu6OCk\/lvuWbQ4QftF0Sy1y1BJwfn5uRbyRRUIxO6GXgppB\/k\/mHKiDTxwQMEcHdZc3VNH7FNy+3biTPGWePAey4MDtzXh7FdGyGmu0WQegTMctUnB7ywMQPnjKngNGAlZGKq+a1usAnL97btGoYVPRVPJHz4FeObC1tWUyrpbn75rTDght1gGoOwiiNzlu3mpMIdKf+T+eAz1wwGhWmf89bzfTCMANEY2SnoUE9FLQZpM\/74UDFp6WRdO+arfVAEA8E\/GEf04FLwVtNvnzXjiwfnNd3y7x5l+47YjZ10hLS9dno4nod1Jam5qaVBMT7e1f19iKv\/IciDmA7be+fouLZUk+mHf50iAB+VDHBKPgDOcbG5s+MA0jPHXFAdKuwBDk2n6mwSYA8sH8PXNviGjUgemb67H4NDf4o+dAuxzAjGOtURSoN1zbz7SRCMD4w+BH2iGRDJnNzf1fMDI3+qPnQCsObErQeYtJDfYA3NOoSfVTASiIXQ7C2GVGjFpZrEnt+DLPgToHYtUbh\/ICAR9Yqn\/onKQCkHqiii\/iFTNHTB6\/B6HDPX\/ZxAEwAlbADNhJU73mxiYv2HxgjtorHo\/eE1F6koVLx44e9Wn7hjn+2MABQLeGoCKvVJKcH7jn+KMNFRIuWkpA6muvOAieltNlGl67Iegu6X7SCfzzRXscaACfYCWIMXMgfw6UgKYFWb5ZY\/mmXNe8JDRc8Uc40AQ+WW7Zyu6zudY2ALnJBeGRo0dU1S9isvlZunNsPhaaa7WL5OsAfDCrIwBygw1CVtAdPXbUgxDGlJCMw7G3r1DH4INlHQOQmzQIo+h1ufuk6Ho1OTnhp+xgTImION\/GxoZWvzgc2Hztql2bTV0B0DTwx8+\/vCgdP8\/1+NiYmpC5Y6+SDXeKecTeI7mAvV0guf55ZatyzqTYdzrqngBIZyINT8sSuwvyLZhFJSMN\/driTl\/D4ajfIPVkhkOiIecfvOeOi708fc8ApHNUsqjjC\/JteIprgDh9ZFqNjhya30LksT2lcIB8PuZ1rRzRRXE2ftSNynW7yASAplEtDVV0Vq5rlAHAMdn2zUtEuHH4KAF4y3pqTZJVshpNpgA0D\/XHa1+ek2\/Iv8l1jTIkogbjxLiXijBkSAn7jrXh25JEsCWL07jWhLrF1tusXOzW1ksbci4ANJ25EtGUA8bqSFWNyLEi03sj8t9TfzkAuPjPfkDE8NixQG9MYEAXP86iOJlvqg31atbAM6PNFYCmk6W\/Xj8Z7oSnRSqeUhK6MeX2ESmJB01Yp1KNj5zH1\/sA1ddSbpOpZ5cV\/dwAyB2nSRiJyMPbA5POydsD3I4AjfIWe4IvCjTfZ5mu2HiLbvtZXze+yaxbT2iP5AY1rhbCIDwpvxHxiPw6BA5MIigTbvdF2XJA5mzVpTCMrup14VtqMS9Jl\/bYfQdg2oNoTxqbUcI5sli0FkbhrGRK3B\/XD2rmPvnyyi6a8t8mrikvE4ldJmNecYcsL3RZl+nPI\/25\/ALM1UpQWdmV+qJL+JzVaXE9XXlwf\/4f1AC7LPmFaqYAAAAASUVORK5CYII=" + }, + "copyright": "Copyright 2023 Dify", + "privacy_policy": "https:\/\/dify.ai\n", + "position": 2, + "chunk_structure": "hierarchical_model", + "language": "en-US" + }, + { + "id": "9ef3e66a-11c7-4227-897c-3b0f9a42da1a", + "name": "Simple Q&A", + "description": "This template generates structured Q&A pairs by extracting selected columns from a table. These pairs are indexed by questions, enabling efficient retrieval of relevant answers based on query similarity.", + "icon": { + "icon_type": "image", + "icon": "ae0993dc-ff90-48ac-9e35-c31ebae5124b", + "icon_background": null, + "icon_url": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAAUPklEQVR4Ae1dW4wcxRWt6pl92rseQ7xgYocdIALFeRglkSBEYkkkwF\/YEoT8RDiKwkd+wEryG+P8JpHNTySEQuwkHzEgYX6C4AM2UghISYTzMMrDySzYeION4\/Wu7X3NdOWe6qnempru3Znpefbca427uroe3afP3lv3Vk2NFF0ihdnZSZEVkyUpJqWSOSFUzlPezbg9X6qcFILySOi6Plb8R+WVCq5X5Kf4RMo5wog+liiB8zCPcJzBVV\/67xFwc0r6MxlF9YpiJr99u76G650Ueq\/tlcKlQq5UGprKKO9eXxDZpNgtVBSp2ntffdrbSSXEDBH5z0qqk5nM8nR+az4kcDswaQsBCxdmp4Tw7lVC0VHgUyWe5wmP2JjJZoSkIz7Ig0g64hySKefpk\/J\/prydl\/a0UoQmfWzBuW\/l+aUSlSF6KV+X9X06+kqU6Ih0jJwkpKeF8o7lJyZOxpRpWnbLCAhN5xdH9lMHD9HdTpk7BlmymYwmWoaOAwMDIeFMGT62B4ESERRkLK6uilKxJFaLxcqOpZjxfXXotontRysvNO+s6QQE8URx9AklxZP0Z5fDrYJ0Q0ODYmBwUJPPaLPmPQa31CwEQMKV5WWxulpc05JERBpPHs1vu+FQs\/ox7TSVgKc\/PLfXy3iHzZhuIJsVw6MjAkeW3kNgeXklIKPRjC3QiE0hYOHS7KQqyp8TxFOAmYkHFNIj0IpXr1wNNSINK094WXUgvzW5J52YgO9dPP9ESamnYG5hWkdGRsTw8FB60OcnCRGARlxcXDREnCOH50DS8WHDBAzGeiMH6a\/hSdzh4OCA2LRpU+ithnfNiVQhAO8ZJAQZIUp4R27dNnGg0YdsiIBlk\/sSdbqbtV6j0Pd2vaWlZU3EcijopMyqfY2Y5LoJqMlXkm\/A0UCcbnx8LIzX9TakfPf1IgBtOD+\/EJhkeMoZdV+9JKyLgDb5EMMbG9vM5Kv3raWsPEi4sHBFIKZI06R1k7BmArrkg+bjeF7K2NTg48AMQxM2QsKaCMjka\/DN9FG1RkkYTLZuABTF+F7CmA9mlzXfBmD16WVYQ3ADHAFXwBkdKdkAjw0JWLjw38PUxm44HBjzsdndANE+vgxuWH7Bbr+46eBGcKxrgk+fn91PK1R+joa3bBlnh2MjNPm6RgCOyeXL83oFjiqJA7feeOOROGhiCRiM+7x3MMMxOjrKsxtxCHJ+JAKIE167dg3X5ihGeGdceCbeBBexqEDlsIqFp9YiMebMdRAAZzA7RpIrrxOILB1JQJheWu64F+M+zO2yMAKNIGBNzU6d\/ujc3qg2IgnoeVIPHkE+syo5qjLnMQLrIQDfwSgwWu9+OMorriJg4eKHB800G8wvCyOQBAGYYr0elEIz\/sqwXrhit1dFQAoo7keBTZs32eU4zQg0jAAWJUOkJ59wtWAFATH2g\/YDY3kVc8N4c0UHAYtP+ntC9uUKApqx3+AQLyi1QeJ0cgRCLRh8SS1sMCRg4fxZ\/f1cOB089gvx4USTEIAWLM+iTQVf0w0aDgnoe95+ZA0M8BeIAmj4\/2YjYBQbTZRMmbZDAkqVuReZbH4NNHxsNgL4Wi6EnBHNNaQ1AQuXLuVoCcNuZLDzARRYWoEANiQIzTC+P06iCVgqrUzhhMkHFFhahQDIBxJqKY1O4agJKJWvtZ9H+7KwMAKtRAB7\/0B8vzSFY3kMKD+Hk4GsnjxGkoURaAkCesEqtSwp3owOAg0o5CSlaTVrmY84YWEEWoAANqPSkvG00iszLnBADDtb0C83yQhoBMpOiF62jwxP70yKBAWgWRiBViMAAhqugXsetsVFp1EbP7b6Zrj9\/kQg1ILEPa8kPR2PoeBgf6LBT912BLJlTxj7gXsZpSZxB9gGl4URaAcCRgNiM3qPdg0OItJkm1kYgXYgYAhInkjOM\/GYtcx23AL30c8IGCfEk97Nod1lAvYzJTr37PS9c3kzuvfMHF3n7oV77hMEjLJTpdLWUAP2ybPzY3YBAqHD63lbmIBd8EL6+RaySujfZdO\/UtQNQHzipz\/qhttI7T28\/53vd\/zZwkkPxAFpWUIQiOYwTMdfTD\/eAJvgfnzrXfTMTMAuehn9eCtMwH586130zJ7QPw5Nc8H0j4URaAcCJg5Iu3DkSAOWnRBeDdMO7LkPQiAkIO0dyCaYKdFRBJiAHYWfO2cCMgc6igATsKPwc+dMQOZARxFgAnYUfu6cCcgc6CgCTMCOws+dMwGZAx1FgAnYUfi5cyYgc6CjCDABOwo\/d84EZA50FIGu3xK\/G77D0NE3lLDzbv+ODWvAhC+YqydDgAmYDD+unRABJmBCALl6MgSYgMnw49oJEWACJgSQqydDgAmYDD+unRABJmBCALl6MgSYgMnw49oJEWACJgSQqydDgAmYDD+unRABJmBCALl6MgS6fi64kcd769z74t2PLoiz85fF\/Mqy2DE2LsaHhsVdN+0Uuz420UiTus788rJ4tfBPcXZhPmzjro\/vFHff9InwPEkC9+3Krusn9L27+Wk5Tw0BQY6f\/eWP9PmTQDpOdoxtEQe++CXxyB2fjisSmY92D\/\/hzeq2\/yCI4FvE8Ye+LnaOj0fWrSUT5Hv0xPGqorjXA1+8pyo\/LRmpMMGnPjov9jx\/jAjy+2qCOG\/q7MJl8d3XX6GX\/WtxZn5NkznFKk5BvEO\/ez22bbT56Mu1t1fRePnkxb+fisoWrxVOR+anJbPnCQjy6ZdPJKhH3jp3pibSwNyC2LaMDw2JnWTWbQEJv\/f6b+ysutKvFv4VWR7P99YHZyKvpSGzp00wyPH4KyeqNBNMIkzsp2i8B7JAXvz738Tb9CLPWEQ1pDm+9+ux7xLaz5Zvffbz2oRjTKk1H5lN0yZIPb+8VPeY7dX\/nK56BrvPt8k8301jzTRKT2tAkMO8fPNyQJDff+NxTZIH8reRgwAnYaf4yVf2iON7HxUP5D9piuojSIOxY5zAkTECMh\/88ldCgoHoT9IYzRbbQbHz10u\/+I+\/VVx2HSWMP9MqPUtAvOgXSKvZAvKBIHECwjy7Z2+VJxyMHZfiqoX544PDYdokovLMtVqOgWddaX4Pfvm+UHOjDZRJqxnuWQK6phHkgsdYi\/zgnkqSBiSIHuzD1BqByXUdlx+++bq5rL1hmP16xB374TnuorAOtLctr8WMEe0yvZjuWQJicG4Lxkg2WexrbhplYZZteZtMcZQgzmeLcTSggbUnbY0p6w3toF2MTW0xxHv49s\/Y2eIFMtMYX6ZNepKA0FjvOgR8uM643v23OGPBGE\/zkds\/TR7vlvC9Y8z47VdeEg8+f1QgbQQB41o1sKkDEtttIN+QOPiDChwo5OOZT1FwPW3SkwQ8dfHDqvew6\/ptVXnrZezYvEYqlIN5jRI4Hj8mB8aWVyk2B0IYgTaFg1OvvPXB+xVVYH5tEw7y2\/LcX+OdJbtcL6V7koBRANdqfk3dXduqCXvG8nhNORyhjVzv2VyH04MwTr39o36c+TVt3+967KSl02aGU0NA89JaccQsiOssoB9ox\/snK015rf2vZ35NG1FmGNo3TdK3BLy8vFL1HreUg9bmAszsnuPH9PyyybOPuP44jQdtrQRTji+Dm48bKjL1XUK75teUc82wqzVNuV499iQBbafAAB9nPs1192gHmM0114weohDLqYuV3jYWBtj94\/qh371hmqgKjJuZmLBAOfHcnyuDy9B2CKq7H3tMiKpwWmzCu+322nlPTsVFBX\/fJSLsHK90LNZ7Ge86jow7+4DpMVd7YawHh+ORO3aRF3wsdEQQItlBK2FATiwDs8UlNa7Bm3VncNCX25\/djp1Gf9\/67BfsrJ5N96QGhFapiuNFhFG+S4sD7vnlM\/oDU2oHkd3VJ66mcafHEB4xfcJcYvmVLZhNwZSeq9mivPPn1pn6s9uMS79GfxxpkZ4kIMB3A8TQCjbBUAYa6TItSD1D8TaYSozXINA0rgZy44iumXOvQ2NiftkWmGK73QduuS3SO8aiiCSSJjPckyYYLw8myF58ahwCxOOM2YOmevbBfXrZFeqAhFgL6BIA5Yx2Q7ko0WNGZ\/YEWhHerDstaOpHechYeGqTFGWf3bNPe9SmXtQRwW879ohnT8NC1Z7VgDDDWHxgCwiGVcW2JsTg3n5RUdovagbDNckwra5WRN+oGxUjxJSamdWw79E1\/dCk9qod\/CFEfVxv2P0jsvvopXTPEhAgg1iu8wAS3vOrZ\/Q8LTQTPiBOnDcKEkcRxQ0Co90Hn\/8FeaHva00EbYQ0NKobUsG9naXV1lGEdYnzMDk0tYh7PzDDaVgh07Mm2Lw0LK\/SWs+ZStMvyJqrNeXtIzRX3PItaM7AzK9Nf5kFqHYbcWkQFmPCn3x1bZwIz9o1v1FmOqpNE5S2zXAaFqr2tAbEi8L47ZWvPRapxaJepJ0XFQu0r2NdXj3hDmhTO0YIx8geH742U7nuD9q7ntCRa4bTsFC15wkIwsC8wiPFSmiY0zhzi3x7vBZoqbX1fDb5TBokRNuuqTfX0SbGbIgRBvPCcILWVrEgPINxJzSXG+er1fyavlwzrIcBCT1q03anjvI\/F\/6r0Pl1123t1D1U9OvuadzoHtEgF14QtNwOClBDU5ovEmEdH0y0kVo1HcZ0py4G3zdG3U9tIw22OfjOsWmr247NwrPZz\/W\/\/13STfb8GDAOGKzP0+KETpCHsAe+xmnGY9BSWIUcp+WChqBph4NwTUSbpgwf60MgtQRcDwaYyDfJXLN0HoFUjAE7DyPfQaMIMAEbRY7rNQUBJmBTYORGGkWACdgoclyvKQgwAZsCIzfSKAJMwEaR43pNQYAJ2BQYuZFGESACyjlUVr6eEGm0Ha7HCNSMgFIh1+Y8IVVAQBFm1twQF2QEGkEgJKAUc10\/E+LOZTbywFynexHgMWD3vpu+uDMmYF+85u59SCZg976b1N6Zb5wQJeeyUokZcj8mS74vPK\/zfGx0\/V9q31YKHyx0QoQiL5iFEeggAp4vBMcBO\/gC+r1rTyqld2ZUiqjIwgi0AQG\/VNK9SCln2AS3AXDuohIB44Mg11NSzCDhkxPCwgi0AwFjbX3lv0d+bzAXHLrG7bgD7qOvEVjzguWcVyrPBQtbL\/Y1PPzwrUbALwXW1sMYMENxQHRYLAYDw1Z3zu0zAqEGVD7FAYsBAcNMxocRaDECmPTQQtzz8tu3z+AETgivCdSw8H8tRsA4vOBeEIYpe8KK1wS2GHpuvliOAdJC6JNAQxOQ\/A99srq6yggxAi1FwAShhV96Dx2VNaCvT9bY2dJ74Mb7GIFisaifXnm2BhSZaeT67AlrcPi\/1iFQKnPMk96aCc5kBqfRJTQgOyJAgqVVCKyWNaDIXJtGH9oE57dunZNCTCMUU\/Q5Htgq8Pu93ZB85IDkt+bnQgIiQUGY3+K4slL9G2rIZ2EEkiKwshT8xK1SJc01tBc4IUFiGhkrET\/ih3wWRiApAkYDeiJ71LQVEjC\/bfu0McOmoCnER0YgKQLLtF2yDkDT1G9+YkI7IGgzJGC5g5dxXLq2WD7lAyPQHASMZZVCHbJbrCRgdugotuqABmQtaMPE6SQIhHzS2m87cWxNKggIb1gJ\/2lcZi24BhKnkiFw9cpV3QBFWY65LdGwr1IKly7l1OryO0KKydHRETE8PFxZgM8YgToQwNjv6tVrtPuVmLll4sa8W7VCA+Kijgl68gDSi4tLHJgGECwNIYBlV+AQxB37mQarCIgL+Y\/dcIJUow5MX7kaqE9TgY+MQK0IYBinl\/kJcSI\/UTn2M21EElBfLKpvaoeEVsgsLQUsNpX4yAhshMASBZ2X9aQGfe+jqLRFjaoTS0AsFpSidAiVoEbDVaxRrXAeI2AhEJjeIJQnlX\/ALHq2ioTJWAKiRH7bTUeU9J\/GHPHC\/AKTMISNE3EIgHzgCjhDX798Os70mvo01FtftFdcXHmD3JjdmUxGjI+NCeltWG39RvlqKhEA6eahqLDqmRY5k9d750YPuq4GRGXtFRf9fXCj0fD8ArGb95PeCNe+u+6Qb0aW1L5aQKhZlRVmZydVRr6B+CBrwlqg7Z8yEeS7b71xn41MzQREJZeEm8c2i0wX7CloPxCn24sAxnxXFq4YswvNVzP5cKd1ERAVbBJiQ8ux8TEmIYDpQzEOh1nlUi\/5AFndBEQlTcKs9xIcE9piS4yMDPOUHYDpI0Gcb3FxUXu7cDgw5qvV7NowNURA08C\/Pzp3RCrvCZwPDQ6KYZo7ZpNs0EnnEeM9LC5YKX+FF6EW7+ryU\/l8sMS+3qdOREB0Vjg\/u19J7zBpwxxMMrThEP0iOUv6EKjQerRsjyJ9h27dduORJE+amIDoHCZZZOVh2ux8L85BxE2bN4mBbNf\/Dg5ul2UDBLCeD\/O61hrRaUlTtY2YXLerphDQNKq1oZAHEapBHgg4ODTIGtEA1GPHKuJRLFhPrd1w04lmPUpTCWhu6t8XZp+SSj5miAiNqMk4PMRa0YDUhUeM7\/Dd8FVaRLBMX07DeSAwtzTWu7J0pNGxXtzjtoSApjNXI5p8kDGTzYgsHT2a3svSh6W9CIBc+GA\/IMxwYccCvTFBSDp9P9NEkJfFlcWjzSaeedqWEtB0Ujh\/frcvivulzNyL0I3Jt4\/QkvCgEdbxMsER6eB8jaD6nPJtMeXsvLSnDYHc50RsDqLoaDSYXpNXJhw2IkW+jt25lYPzaaLmb2mOdhrflIwu0rzcyjfZvHZjWyoUCjkxNjpFG1Tv9oT3OVLyk3GkjG2ELzQHAdqWj4ZKJ31Vos3CaX+ghWvTrdJ0cTfcdgLG3UjgSRMZpZejP9FJ+vvNecq7WZeXatLUU0LmhFQ5c66PivKofEVe6k9oc3mzv7f1rPjpteCUrqvgR4h8SbvRU9gE+4HrLZlpZ9JmeLBWtw0n\/w+IOsoy1qfzJgAAAABJRU5ErkJggg==" + }, + "copyright": "Copyright 2023 Dify", + "privacy_policy": "https:\/\/dify.ai\n", + "position": 3, + "chunk_structure": "qa_model", + "language": "en-US" + }, + { + "id": "982d1788-837a-40c8-b7de-d37b09a9b2bc", + "name": "Convert to Markdown", + "description": "This template is designed for converting native Office files such as DOCX, XLSX, and PPTX into Markdown to facilitate better information processing. PDF files are not recommended.", + "icon": { + "icon_type": "image", + "icon": "9d658c3a-b22f-487d-8223-db51e9012505", + "icon_background": null, + "icon_url": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAAQfElEQVR4Ae2dT4wbVx3H35vxrjd\/dmMnIZA0UrxtilQuTYUEB5CySD2CSJE4Vl0uHIpQk1sFh7YHqt7aCsGBS7fqEQlSwRGpi8QFJMRyQoKEdaR2U9qkdva\/vfYMv+8b\/7zjsZ2xPTP22PN70u6bP2\/en+\/7+Pf+zMwbrVLiNu9XSpSVUpP+tOsUlKsKtH\/l4Z6rXNrW2uyrc6cthAs6hMVfllyVCou\/Y+eq6sM9x3+sfO6Uxvl7Squqq6yyTT7tl5cvFss4MWmXG3cGNjcrhWZerWjlXFdKlyj9a\/RXcogyOCMX\/nsbBJ93vOWZMPLPKFCg\/\/g7dqRZl070y2Wn6VfteHKqu1tfUGC1QTqX6aJ\/utrasGtqfXm5CEDH5o5zl2CSZN1WKPrrBNMKlR\/bXc6yLKUtrXK2rTSJhj8c+3zboeN0riXkVwrdvxkO3xXpDB\/AD5N\/nFxM7P\/vEbUhLec0m+r8okXhHBPWcRwCkCBskk\/bPZ2B0l23ctb7yxeKGz3DxHgwMQBh6Zy8s0oofd8PHWCxc7YBzSbY5ubm2sD1KtdnBKDfXViy\/LuyHVBgGL2aBChgPGocqQZtN44agdhU2XWcN65ePr8WPBHXfuwAAjy1oF6hX9pNyqRpIgBdPj+v5ufmDXxszQYpxDCCDhLfrIeJqhcgrNVr6oh8n5UsW1qvUb\/xjbj1ixXAO1sPblDD+TZlsoSM5uZy6uTCCeNjfxQXVdBR0pzma+LUq1arGxh9ljF2ixgLgBjBUv\/jPW5q4wCPIYhTUI5zlv0k9AKAu3t7fot4myzirThG0pE7VJufVtDc\/gPwoWk9efKkWlpcjGT1ZhmQaSwbDEqhcEadOnXKDAypDDdQ53c+frAatTwjA4i+3uZW5W3Hcd+hTBTm5+dMJhcW8lHzJNenVAH045eWFk1\/HnVOsxPv3d16iC7XyG6kJhhNLoH3e5pDugard+LECZUUeEk0KSOrNQUXjkuvw8OaOjg48KaCaOrGsvQLozTJQ1tAA5\/rfgT4ME935sxSYvBNQX1nNoswOKh7MAAWqEn+CGwMK8hQALbho1Eu5vBgjk0Ghk1Vws+EAqh7MAAWyOFu1tAQDgygwDcTzMReiKgQDgRgL\/iGmUyOvdQSYaoUAAujWsKBADQDDl+zK\/Clqv5TkZkuCGmQau6KheQuFEBMtaCTCVO7uHi6\/VBASLxyOoMKAEIwYsYFGJjkndfCZHgsgHfuP1il5yhuMt0m4rAY5XymFeA+oddK6ps0T4hnAvq6vgCi36ddc1\/XzPMJfH01lBMBBcAK5oY9p18DS4Eg7d2+ANKQGjPcBcx+JzXJ3M6FbMycAmAGd8fIFfCcQL8C9gQQTS9dcKOT5H5RyHFRoLcCuHeMphjPCdzZqtzoFaongNT0ms4jzKg0vb1kk2ODKAD4uCkmDN\/uNSruAvDu\/QrgKwE8NL\/iRIEoCqApxtM05ErOvNM1IOkCkO4uryL0aTKf4kSBOBTAQ8nGaf1K0Ap2ANjq+5VAbIvaONKXODKugI8n856QX44OALnvl5+XZ\/r8Isl2dAXYCuIlNX9sbQA3P65coxPS9\/OrI9uxKQAryCNimhdc4YjbANKboqs4OOd1GPm8+KJAbArwoJbetlvhSNsAKktfx0Fpflka8eNWAK\/lwpElNKyZbfzDyMTJuxVsnz1bhJcaF3zEPDUZm5KMpOlFfqzcUK0+Mo\/xWzVdxDIgxgI2880V6Ckj3ymhakqziT4gVsWAw\/pA8A2A2tUYgKic5Z3EtjhRIAkFsPaPca1+oNcH1PpZHMzROi3iRIEkFWi9P4KOYAnp8FJTZse2PR5xIi0uTX2YtGgyzfnAYlRw1Bobo8fEmSa4Tec0l1DynmoF0A9suRJ8ix8WlKdeWrKIl6gCAJBZA3sWrQhXQopWCpvfRJWQyCemgN8KWtptFpATWu1oYhmShLOlQI6nYprNEi2Kq0sovqW5O4g9caJAcgqwBaQlmQu0gHBrFVNCUZwoMA4FGECwZ7na6wO2D44jB5JGphXgQYilrCvtdlcAzDQTEys8AaivIHVbbsNNrBKyljAbu6Zyi20LmDURpLyTU4AHvDTsOCMATq4eJGVSAGNfMw+IrxSJEwXGoQDf9HDxCggl6AEoE9Hj0F7SCCggTXBAENkdrwIC4Hj1ltQCCuQ+33EVlo+pWw49pRA4G8Nu1Of5vvpqNYZcZDeKf79lelgjC5DEOzn4Bt32jvcRShp6uNIHHLl65MJRFOB5QLqW7gXLIGQUDeWaCAoEAYwQlVwqCkRTIIcvasOdjelD0En0GaIVUa6OU4GofXrOS67hcZfAsIOTEF8UCFdAAAzXSEIkqIAAmKC4EnW4AgJguEYSIkEFBMAExZWowxUQAMM1khAJKiAAJiiuRB2ugAAYrpGESFABATBBcSXqcAUEwHCNJESCCgiACYorUYcrIACGayQhElRAAExQXIk6XAEBMFwjCZGgAgJgguJK1OEK8BrR4SGnNETwnYhXf7uvfvf3+kilWf12Xv3su\/wpei+KqO+sBPMXNb6RCjbBizJnAd\/64Un1zMXhP0fxzCW7C74J1tvMJJ05AFFzH\/z4tLo8xLI4CPvrF+X7yUlQn0kAl05oA+HSQvhyJIAPwD4xBLBJVNSsxplJAFGZAApghblfkeUT+MJUGv18ZgGEZOjXoU\/Yz\/38eydMmH7n5Xh0BTIH4F\/\/Sx+m8LkffH1e\/fT5Bd8RbxPHXvpW55fj\/7XV7AonB6IpkDkAf\/LBnvq44i0LwdIFYcN0SxBKXPMyXSsuXgUyB+D2gate\/M1uF4Robr\/5ZM40ucG5PsCHaz4JgBtvVWQztswBiGoGSLCE24e0RKLPYcARnG5BGIQV+HxCxbiZSQChH\/pzb\/7hoENKTM8ER7wII32\/Dpli3cksgFARt+R++afDvoLi3Ki37fyRYqCDv1Hd81+bi3T9qOmO47qZvxccJiIgg+ULjnjX\/lJ7LJxh8fJ5gOef6hkW6KjXcz7S6mfaAnKl\/IKaWf\/0zN9oqubNP3Y2zxx2GD8ID0AcxhL2uh4DpVlys1WaCDWDUe44HFvDMEsYhI\/z9g0C0P9j4ePT6osFTLDmABke\/wq6MEvYDz50Fx7XZw2mMw37YgETriW2dGz5OLngPh\/PEnwos1hArvkE\/cdZwmCyvcCcRcvH5RYLyEok7PezhGHJRnmCOyzuNJwXCzjGWuhnCftlYdbhQ7kFwH61n9DxQSHMAnwCYEKQhUUbBmFW4BMAw0hJ8Hw\/CLMEnwCYIGCDRB2EMGvwQaOZHwXH\/Z5t3PEBQnb+bT426\/7MAzgNFZhF8LheZBTMSog\/EQUEwInILomyAgIgKyH+RBQQACciuyTKCgiArIT4E1FAAJyI7JIoKyAAshLiT0QBAXAiskuirIAAyEqIPxEFBMCJyC6JsgICICsh\/kQUEAAnIrskygoIgKyE+BNRQACciOySKCuQe7DjLdbYyHUu2sgBxBcF\/Ap8th0PJ9UWd2IB\/erK9tgVAIBVpOq6nYs1jj0nkmBmFPCxVrVcpQXAzFR9OgrqB1Df3fpik7JVKhTOKMuSFjkdVTTbuXAcR1Wrj1DIshA323Wd+tIJgKmvotnOoAA42\/WbytK5TnvAi0GIKiOXTjOe+Z1UllgylSoFeBBCn4qsigVMVdVkLzMWKESxHZkHzF7tp6DE1AS7ZjzsutIEp6A+MpGFpuN99FG7WqZhMlHjKSukv7G1tNsahNDkoDhRYBwKcGvrKOeepXTrXvDx0HgceZA0MqwAj4LBnuVq17sXrNpzMxmWRoo+DgWardbWVVaZBiF2GYk2GvI18HGIL2kcP3llwwLSAoFliNI2i6KQKJCwAr6bHmVr+WKxjPTwhILMBSasvERvFABrcGCP74SUzRH\/+NgckH+iQLwKNI+7ehuImZfoxU7p6OhI5fP5eFOMGFtc7yBEzMbUXn5hiW1MOorAk9Bk6+4hR17uHNfs+OhMR24lFzOnQKPRMGXSyjUW0ADoWu46jjZat0hMCPknCiSgQKPpzba42joG0K7Z60gLFlAGIgmoLlG2FWgceRbQrql1HDR9wOXlYvXO1hfrNBez4hCE1hx3DdvXpWYjbX2a1AjTykia+8wMH2V1A8why+0eKs0D\/hkH6vXjD6dgX5woEJcCh\/WaiYqeiDasYacNIL0St44DNQEQMohLQAG2gPa8tcbRtwF8+mJxne4Gr+OOCAfkQOKLAlEVqNVq5mYHxVNevlA0AxDE2QYQOzQ0\/hD+\/uEBPHGiQGwKcMvqOvoNf6QdAFo1YxqrsIBiBf0yyXYUBXw8la9eLq754+oAECMTmoZ5FwHECvplku0oCuzu7XmXu+77wXg6AMTJXN16h7wyqD08PAyGl31RYCgF\/H2\/p54493rw4i4AYQVpwaJbCHhwcCgT00HFZH9gBfDYFRiCC\/b9OJIuAHHi6qXibR4R7+22zCdfIb4oMKAC6Ma1Hr26Hez7cRQ9AcRJW+sfkVfFEzLSFLNc4g+qwOFhTdVr5qZG1dJei9rr2r4Aeg+qekNm0xTL0h299JNjPRTwml5vKo+a3lv80HOPoJ3zgMEAT10qvkO3Td7F5PT2zo6sHxMUSPa7FAB8YAXMgJ1+TS9f2NcCcgD7yHpd081jtOU7u7syKGFhxO9SANAZRvDIvas2rl4+d7MrUOBAKIAYFWutX6Dryk16lmtnmywhJSROFPArYFpJYgOMkCtblmHGH6TndiiAuMq8PKL1d2hTIOwpY7YPdsFHrDyu3+dXayAAcUFPCGVg4tcyk9umz+e3fEPAB8EGBhCBgxDKwASqZNfxgKPd7A4JH5QbCkBcwBDywOTR9rbME0KYjDnM86HuzUQzDThorm\/gZtcv1dAA4mJA+OSls8\/xFM3+\/oHCDWf8IsTNtgI80t3f329PtVj10eCDUiMByBJjmO227phg1htNMm4+i5tNBWD18H2Po\/oRClh1lHsLDPD7HaOUOhKASPDqxeIamd\/n6HHW2zDHe3v7JpPyPOEo1ZHOa1CXMC5s9aj7tY46f\/rSOTw5FclRXPG5O\/crq9p1X6MYS4g1R2\/X5efnI622EHzLS96Kg7L9XZx6ATw8UOAzJmU8KYWHVfrnYLgzsQLISf\/nk4ev0y\/kJdov4Rg+AQYYF+bzxsexQV2cgg6a5jSHi6IX+nd4N7x+VKeuVN308VpamAeV8axolOa2l66JAMgJBS0iHweMOdtWuVxO2Zat7JzNp7r8KIJ2RZaBA4PqBdjwh6edMI2CFQsAH46xIzjoRTX9oVVTa3GD50uDN5PzNz+rXGvWnVW6PXOdinetV0qwkpZNKwZrTVB6PrYf7NA6mgQpuy+fsZXGxyV8DuHwlyXHAAXL\/GnFW3kA6zAjzJdocSL0zTk8FiLFtpk+CV5M+4CuiXfE6TVdvCnZI0ish8Zea5ublUIzr1a061wjap6lDJT6QYmS8hfdudTnFyOPmziqmfSH1KtMImzQdNo9AIflMpKydP3EHjuA\/TKyeb9Sot9uiVbtLwBKepanQGGvPNwzTUKJrzt\/2irQEZzzO+wHj\/nPz+J2lQqFvw73cNcp4wAZOXqIRFXPnTJVfI+ajapL+6RdmRZeKWMuF+Em7f4PpXL0Ed9VCt8AAAAASUVORK5CYII=" + }, + "copyright": "Copyright 2023 Dify", + "privacy_policy": "https:\/\/dify.ai\n", + "position": 5, + "chunk_structure": "hierarchical_model", + "language": "en-US" + }, + { + "id": "98374ab6-9dcd-434d-983e-268bec156b43", + "name": "LLM Generated Q&A", + "description": "This template is designed to use LLM to extract key information from the input document and generate Q&A pairs indexed by questions, enabling efficient retrieval of relevant answers based on query similarity.", + "icon": { + "icon_type": "image", + "icon": "e4ea16ed-9690-4de9-ab80-5b622ecbcc04", + "icon_background": null, + "icon_url": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAAQjUlEQVR4Ae1dTYwcxRWuqpnd2R\/veqzgxXaw2YEgRSDBEkJEwsFLDkE5xRwicogUR0g55GJWKGfjXBPJyyU3hLkFKRLmkohD4uVgHIVEOCggRTGZNTbesDbysj\/end3prryveqq3Z6bnv3t2tvu91Uz9dHVV99ffvqpX9bpGigGR4tLStMiKaUeKaallXgidV1o9iMtzpc5LISiPhI6bsOqLymvtHa\/KT3BCyhXCiD4B0QJpP49wXMRRV7rXCbgVLd3FjKbzymKxcPSoOYbjeyn0XPsrxbvFvOPkZjNanXQFkU2KGaHDSNXf60ppa1e1EItE5H9qqa9mMqWFwqGCT+B+YNIXAhZvL80KoU5qoSkU+NSJUkooYmMmmxGSQnyQB5EUIg3JVPJMovJlywfzkh7XmtCkT1CQdgN5ruNQGaKXdk1Z16XQ1cKhEPEGcpWQXhBavVmYmrraoExk2bEREJrOLY+epgZ+RFc7a68YZMlmMoZoGQqHhoZ8wtkyHPYHAYcICjKWd3aEU3bETrlc3bAUi66rz31j6uiF6gPRpSInIIgnymNntBQv079dHpcK0uVyw2JoeNiQz2qz6G6Da4oKAZBwu1QSOzvlXS1JRKTx5IXC4fvPRdWOrSdSAl774tYplVHn7ZhuKJsVI2OjAiHL\/kOgVNr2yGg1YwwaMRICFu8uTeuyfIMgngXMTDygkByBVtxY3\/A1Ig0rL6qsnisc6t2S7pmA179cPuNo\/Sq6W3Sto6OjYmQklxz0+U58BKARNzc3LRFXyOCZ63V82DUBvbHe6Fn6b3gZVzg8PCTGx8d9a9W\/ao4kCgFYzyAhyAjRQs0\/fHhqrtub7IqAlS73bWp0hrVet9Dv7\/O2tkqGiJWpoKsyq1\/opkvumICGfI68BEMD83STkxP+fN3+hpSvvlMEoA1XV9e8LhmWckY\/1ykJOyJgkHyYw5uYOMDk6\/SpJaw8SLi2ti4wp0jLpB2TsG0C1pIPmo\/n8xLGpi5vB90wNGE3JGyLgEy+Lp9Mik7rloTeYmsLoGiO722M+dDtsuZrAVZKD6M3BDfAEXAFnDEzJS3waEnA4u3\/nac6ZmBwYMzH3W4LRFN8GNwI2AUzbnn8bCs4mnbB15aXTpOHyhuo+ODBSTY4WqHJxw0CMEy++mrVeOBoR8w9fOTIfCNoGhLQG\/epD7HCMTY2xqsbjRDk\/FAEME947949HFuhOcInG03PNO6Cy3Aq0Hl4sfDSWijGnNkEAXAGq2Mk+YqfQGjpUAKi6yV3x1MY92Ftl4UR6AaBwNLs7LU7t06F1RFKQKWkGTyCfNYrOexkzmMEmiEA28EqMPJ3Px9mFdcRsPjlF2ftMhu6XxZGoBcE0BUbf1CamnG3R4zjSrC+OgLShOJpFBg\/MB4sx3FGoGsE4JQMkUqeqdWCVQTE2A\/aD4xlL+au8eYTaxAI8Mm8JxQ8XEVAO\/YbzrFDaRAkjveOgK8FvZfU\/Ap9AhaXb5r3c2F08NjPx4cjESEALVhZRZv1XtP1KvYJ6Cp1GllDQ\/wCkQcNf0eNgFVstFAya+v2CSh15iQyufu10HAYNQJ4LRdCxojhGuKGgMW7d\/PkwjCDDDY+gAJLHAhgQwK\/G8b74ySGgI6zPYsEkw8osMSFAMgHEhpxxmYRGgJK7Rrtp2hfFhZGIE4EsPcPxHWdWYSVMaB8AomhrFk8RpSFEYgFAeOwSjVLmm9GA54GFHKa4uTNWuEjEiyMQAwIYDMqIxlllF6FcZ4BYtkZQ7tcJSNgEKgYIcZtHxnK7EyKCE1AszACcSMAAlqugXsK2+Ki0bCNH+O+GK4\/nQj4WpC4pxypzHwMTQ6mEw2+674jkK1YwtgPXGW0nsYVYBtcFkagHwhYDYjN6BXtGuzNSFPfzMII9AMBS0CyRPLKzsfsZvbjEriNNCNgjRAl1YN+v8sETDMl9u7e6b1z+SCaV3aNbu+uhVtOCQJW2WnHOeRrwJTcO9\/mACDgG7xKHWQCDsADSfMlKC3wu2zUBbMVnGYe9PXe\/UUPzAOSW4I3Ec0E7OtD4MY8BFL7AsiJ3\/0m0Rz47Je\/2hf3x2PAffGYknuRTMDkPtt9cWdKmB+HprVg+mNhBPqBgJ0HpF048qQBK0YIe8P0A3tugxDwCUh7B3IXzJTYUwSYgHsKPzfOBGQO7CkCTMA9hZ8bZwIyB\/YUASbgnsLPjTMBmQN7isDArgUnfa12T5\/6ADXOGnCAHkYaL4UJmManPkD3zAQcoIeRxksZ2DFg7cPYL\/5ttdfdbjqtY17WgO0yhMvFggATMBZYudJ2EWACtosUl4sFASZgLLBype0iwARsFykuFwsC+8YKjuXuG1R65dZn4sWLb1UdfevUT8R3jx2vyuNE7wiwBgzBcHVruy735upXdXmc0TsCTMAQDFe3t0JyOSsOBJiAIajeXKvXdmF5IadyVocIMAFDAPvkzu263Jtrq3V5nNE7AkzAEAxvhGjAK5\/fCCnJWb0iwASsQRCa7pM7yzW5QqALvsGGSB0uvWYwAWsQvPL5ZzU5u8k\/\/PtfuwmORYIAE7AGxvkP3q\/J2U2+\/tE\/xGqJLeRdRHqPMQEDGJ7\/4LIIG\/\/ZIqulkjjfhKC2HIftI8AErGAF8rVDLmhBlGWJBoHUL8V5Wu2yALHaFRAV5809\/T0xmRtp9zQuF4JAagkIAr3+0d8N8RDvVEDYd4vXDAmfOXZCHJ+c7LQKLk8IJJ6AcCyw67iYYsHnr2Tp3ohgYhlTM6\/85U+GSI99bUo8QCR89D4KJyaNZpzM5ciB4QQTrQkCiSdgrVdLEyx6OvTxl8sCH2jFoCT9XZbgvXYTZyOkG9T4nMgQYAJGBiVX1A0CTMBuUONzIkMg8WNAeDLDysUKBowGeLog\/DhkvbcXVI+T4fHM108YA+SBiYOmqgcmvbCXepN+buIJ2MiNHiSEhwuW3pqtfjQjAKzclx7\/Nn2+xfOBzYBqcizxBGx079BSP\/7mQfF84REzF9jp6sZLjz8V60R0Wqzn1BLQEhNaDCsakHZJOPf0s\/45th4Ou0OAjZAKbiAhutNWYjVfq3J8vD0EmIABnLy13VwgpzqKbttqy+ojnOoWASZgADnPqHgqkFMdfekJNjaqEek9xQSswbBZN\/yD6UdqSnOyVwSYgDUIQguGebY8Rk4Gx3lerwat3pNMwBAMnwnZggOeLizRI8AEDMHUrmQEDz1K7lYs0SPABAzBNIyAYXkhp3JWhwgwAUMAmxyud7PH2JAlegSYgCGYTo4M1+Xyux91kESSkfqluDAU4UaflrXYsPvvZx5rwH6izW3VIbBvNGC3v6PRjSbr9Y25OpQ5oyEC+4aADe8g4gPv\/vc\/4teXL3XtIxjx5SS+OiZg5RHj9c35v70vrtzibdj6yfrUExDvCb\/y5z8y8frJukBbA0vAbsZuuK92x4p2nNdsPxg4nrK7fYAtMUQHloAx3Kup0hLP22otfEsOvEfy2+\/\/kJ0P4noIgXpTRcBWBgaI9\/J3nuXfAwkQJO5oKgjYysDAOu\/ZZ58Tzz\/E\/n5xE662fiKgXBFC57WrhVSy9vi+T7948fcNDQzPA5pfq+z3Q9Za2yZXskLqFaFFXtOXpL+kSaNpFTYw9u5J+wSUggiYMmEDY7AeeGoIyAbGYBHPXk3iCcgGhn3UgxkmloBsYAwm4XBVrjVCtFzJSi0WySaZdlxXKJUM7yw2MAaXfLgy3wgROnlGyOWf\/oJXMAabf1VXp1whaB6QWEnzgEkQfnd3fz1FJbU2P46rNVGRhRHoAwKu45hWpJSLyRj09QE0biI6BKwNghqVlmIREZeMEBZGoB8I2N7W1e51snuxFhwwjftxBdxGqhHYtYLlinKwFgwJ6sVUw8M3HzcCruP1tgpjwAzNA6LBctkbGMbdONfPCPgaULsrSpQ9AvqZjA8jEDMCWPQwQtxThaNHF5GAEZKUuUBzc\/w1sAhYgxfc86ZhKpYwfAJZGIE4EShX5gDJEfoq2jEEJPvDJHZ2duJsm+tmBISdhKbIdcBR0YCuSeyyk5FiBOJBoFwum4q1CmpAkVlArsuWsAGHv+JDwKlwTEm12wVnMsMLaBIakA0RIMESFwI7FQ0oMvcW0IbpgguHDq3Q60gLmIopuzwfGBf4aa\/XJx8ZIIVDhRWfgIjQJMx7CLe3txGwMAKRI7C95e1EobVjuIYGPCPEiywgY7vEBAQOLNEjYDWgEtkLtnafgIXDRxdsN2wL2kIcMgK9IlCiHw03E9C09FuYmjIGCOr0CVhp4B2EW\/c2K0kOGIFoELA9qxT6XLDGagJmcxewVQc0IGvBIEwc7wUBn09G+x0lju1KFQFhDWvhvobDrAV3QeJYbwhsrG+YCmiW5c3ammjYVy3Fu3fzeqf0IW0TMz02NipGRup\/tKX6DE4xAo0RwNhvY+Me+ZuKxYemjhRqS1ZpQBw0c4JKziG+ubnFE9MAgqUrBOB2BQ5Basd+tsI6AuJA4b77L5JqNBPT6xue+rQncMgItIsAhnHGzU+Ii4Wp6rGfrSOUgOZgWf\/cGCTkIbO15bHYnsQhI9AKgS2adC6ZRQ1676OsTY8adk5DAsJZUArnHE6CGvW9WMNq4TxGIICA1\/V6U3lSu3PW6TlQxI82JCBKFA4fm9fSfQ1rxGura0xCHzaONEIA5ANXwBl6\/fK1Rl2vPZ+Ges3FWMXl7UtkxsxkMhkxOTGRyK18m6PAR9tBAKRbhaKC1zM5OZPV+2Sr85pqQJxsrOKy+wLMaFS8ukbsTsg+Mq3A4ePtI1BDvkXp6BfaObulBrSVFJeWpnVGXsL8IGtCiwqHQCCEfM81G\/cFUWubgDiploQHJg6ITEL2FAyCwvH2EcCYb31t3Xa70Hxtkw+tdERAnBAkITa0nJicYBICmBSKNTisl0un5ANkHRMQJxkSZtXbMExoiy0xOjrCS3YAJkWCeb7NzU3T\/cLgwJiv3W43CFNXBLQVfHrn1rzU6gzSueFhMUJrx9wlW3SSGWK8B+eC7corvJhqURulVwsFz8W+07vuiYBorLi8dFpLdZ60YR5dMrRhLpfr9Dq4\/D5AoErrkdsezfSde\/jwkfleLr1nAqJxdMkiK8\/TvgqnkAYRxw+Mi6FsYjfhx22mRuDPh3XdgI\/ogqSl2m663FrQIiGgrdRoQyHPYqoGeSDgcG6YNaIFaJ+FdcSjuWCztHb\/sYtR3UqkBLQX9entpVellj+zRIRGNGQcybFWtCANYIjxHd4N3yEnghK9nIa0J+huaay3vjXf7Viv0e3GQkDbWK1GtPkgYyabEVkKFS3vZenD0l8EQC58sB8QVriwY4HZmMAnnbmeBSLIO2J980LUxLN3GysBbSPF5eUZV5RPS5k5iakbmx8MoSVhQWNaR2W8EHEvvUtQk6b8oNhywbykxy2Bau8Tc3MQTaHVYMYnr0I4bESKfDN3V3uyl14gar5Ha7QLeFMyvEh0udVPMrp6G9ZULBbzYmJsljaonlFCPUFKfroRKRtWwgeiQYC25aOh0lVXO7RZOO0PtHZvIS5N1+iC+07ARhfiWdJERqny9C86Tf+\/eaXVg6a81NP2PC1kXkidt2kTasqj8lV5iU\/Q5vJ2f+\/AveKn17wkHdfejxC5knajp2kT7AdutmSmnUmjsGADzXYd\/T+j7cbUE7Qx3wAAAABJRU5ErkJggg==" + }, + "copyright": "Copyright 2023 Dify", + "privacy_policy": "https:\/\/dify.ai\n", + "position": 6, + "chunk_structure": "qa_model", + "language": "en-US" + } + ] + }, + "9f5ea5a7-7796-49f3-9e9a-ae2d8e84cfa3": { + "chunk_structure": "text_model", + "description": "In this template, the document content is divided into smaller paragraphs, known as general chunks, which are directly used for matching user queries and retrieval in Economical indexing mode.", + "export_data": "dependencies:\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/dify_extractor:0.0.5@ba7e2fd9165eda73bfcc68e31a108855197e88706e5556c058e0777ab08409b3\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/general_chunker:0.0.7@a685cc66820d0471545499d2ff5c87ed7e51525470155dbc2f82e1114cd2a9d6\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/notion_datasource:0.1.12@2855c4a7cffd3311118ebe70f095e546f99935e47f12c841123146f728534f55\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/jina_datasource:0.0.5@75942f5bbde870ad28e0345ff5ebf54ebd3aec63f0e66344ef76b88cf06b85c3\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/google_drive:0.1.6@4bc0cf8f8979ebd7321b91506b4bc8f090b05b769b5d214f2da4ce4c04ce30bd\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/firecrawl_datasource:0.2.4@37b490ebc52ac30d1c6cbfa538edcddddcfed7d5f5de58982edbd4e2094eb6e2\nkind: rag_pipeline\nrag_pipeline:\n description: ''\n icon: d86a91f4-9a03-4680-a040-e5210e5595e6\n icon_background: '#FFEAD5'\n icon_type: image\n icon_url: \n name: General Mode-ECO\nversion: 0.1.0\nworkflow:\n conversation_variables: []\n environment_variables: []\n features: {}\n graph:\n edges:\n - data:\n isInLoop: false\n sourceType: tool\n targetType: knowledge-index\n id: 1751337124089-source-1750836372241-target\n selected: false\n source: '1751337124089'\n sourceHandle: source\n target: '1750836372241'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: variable-aggregator\n targetType: tool\n id: 1753346901505-source-1751337124089-target\n selected: false\n source: '1753346901505'\n sourceHandle: source\n target: '1751337124089'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: variable-aggregator\n id: 1750836391776-source-1753346901505-target\n selected: false\n source: '1750836391776'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: document-extractor\n targetType: variable-aggregator\n id: 1753349228522-source-1753346901505-target\n selected: false\n source: '1753349228522'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1754023419266-source-1753346901505-target\n selected: false\n source: '1754023419266'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756442998557-source-1756442986174-target\n selected: false\n source: '1756442998557'\n sourceHandle: source\n target: '1756442986174'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: variable-aggregator\n targetType: if-else\n id: 1756442986174-source-1756443014860-target\n selected: false\n source: '1756442986174'\n sourceHandle: source\n target: '1756443014860'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1750836380067-source-1756442986174-target\n selected: false\n source: '1750836380067'\n sourceHandle: source\n target: '1756442986174'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: if-else\n targetType: tool\n id: 1756443014860-true-1750836391776-target\n selected: false\n source: '1756443014860'\n sourceHandle: 'true'\n target: '1750836391776'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: if-else\n targetType: document-extractor\n id: 1756443014860-false-1753349228522-target\n selected: false\n source: '1756443014860'\n sourceHandle: 'false'\n target: '1753349228522'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756896212061-source-1753346901505-target\n source: '1756896212061'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756907397615-source-1753346901505-target\n source: '1756907397615'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n nodes:\n - data:\n chunk_structure: text_model\n index_chunk_variable_selector:\n - '1751337124089'\n - result\n indexing_technique: economy\n keyword_number: 10\n retrieval_model:\n score_threshold: 0.5\n score_threshold_enabled: false\n search_method: keyword_search\n top_k: 3\n selected: false\n title: Knowledge Base\n type: knowledge-index\n height: 114\n id: '1750836372241'\n position:\n x: 479.7628208876065\n y: 326\n positionAbsolute:\n x: 479.7628208876065\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: File\n datasource_name: upload-file\n datasource_parameters: {}\n fileExtensions:\n - txt\n - markdown\n - mdx\n - pdf\n - html\n - xlsx\n - xls\n - vtt\n - properties\n - doc\n - docx\n - csv\n - eml\n - msg\n - pptx\n - xml\n - epub\n - ppt\n - md\n plugin_id: langgenius/file\n provider_name: file\n provider_type: local_file\n selected: false\n title: File\n type: datasource\n height: 52\n id: '1750836380067'\n position:\n x: -1371.6520723158733\n y: 224.87938381325645\n positionAbsolute:\n x: -1371.6520723158733\n y: 224.87938381325645\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n documents:\n description: the documents extracted from the file\n items:\n type: object\n type: array\n images:\n description: The images extracted from the file\n items:\n type: object\n type: array\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n ja_JP: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n pt_BR: o arquivo a ser analisado (suporta pdf, ppt, pptx, doc, docx, png,\n jpg, jpeg)\n zh_Hans: 用于解析的文件(支持 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)\n label:\n en_US: file\n ja_JP: file\n pt_BR: file\n zh_Hans: file\n llm_description: the file to be parsed (support pdf, ppt, pptx, doc, docx,\n png, jpg, jpeg)\n max: null\n min: null\n name: file\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: file\n params:\n file: ''\n provider_id: langgenius/dify_extractor/dify_extractor\n provider_name: langgenius/dify_extractor/dify_extractor\n provider_type: builtin\n selected: false\n title: Dify Extractor\n tool_configurations: {}\n tool_description: Dify Extractor\n tool_label: Dify Extractor\n tool_name: dify_extractor\n tool_node_version: '2'\n tool_parameters:\n file:\n type: variable\n value:\n - '1756442986174'\n - output\n type: tool\n height: 52\n id: '1750836391776'\n position:\n x: -417.5334221022782\n y: 268.1692071834485\n positionAbsolute:\n x: -417.5334221022782\n y: 268.1692071834485\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 252\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge\n Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n starts with Data Source as the starting node and ends with the knowledge\n base node. The general steps are: import documents from the data source\n → use extractor to extract document content → split and clean content into\n structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n user input variables required by the Knowledge Pipeline node must be predefined\n and managed via the Input Field section located in the top-right corner\n of the orchestration canvas. It determines what input fields the end users\n will see and need to fill in when importing files to the knowledge base\n through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique\n Inputs: Input fields defined here are only available to the selected data\n source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global\n Inputs: These input fields are shared across all subsequent nodes after\n the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For\n more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 1124\n height: 252\n id: '1751252161631'\n position:\n x: -1371.6520723158733\n y: -123.758428116601\n positionAbsolute:\n x: -1371.6520723158733\n y: -123.758428116601\n selected: true\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 1124\n - data:\n author: TenTen\n desc: ''\n height: 388\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently\n we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data\n Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\":\n File Upload, Online Drive, Online Doc, and Web Crawler. Different types\n of Data Sources have different input and output types. The output of File\n Upload and Online Drive are files, while the output of Online Doc and WebCrawler\n are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n Knowledge Pipeline can have multiple data sources. Each data source can\n be selected more than once with different settings. Each added data source\n is a tab on the add file interface. However, each time the user can only\n select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 285\n height: 388\n id: '1751252440357'\n position:\n x: -1723.9942193415582\n y: 224.87938381325645\n positionAbsolute:\n x: -1723.9942193415582\n y: 224.87938381325645\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 285\n - data:\n author: TenTen\n desc: ''\n height: 430\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n document extractor in Retrieval-Augmented Generation (RAG) is a tool or\n component that automatically identifies, extracts, and structures text and\n data from various types of documents—such as PDFs, images, scanned files,\n handwritten notes, and more—into a format that can be effectively used by\n language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Dify\n Extractor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is\n a built-in document parser developed by Dify. It supports a wide range of\n common file formats and offers specialized handling for certain formats,\n such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\".docx\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\n In addition to text extraction, it can extract images embedded within documents,\n store them, and return their accessible URLs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 430\n id: '1751253091602'\n position:\n x: -417.5334221022782\n y: 532.832924599999\n positionAbsolute:\n x: -417.5334221022782\n y: 532.832924599999\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 265\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"General\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" divides\n content into chunks and retrieves the most relevant ones based on the user’s\n query for LLM processing. You can customize chunking rules—such as delimiter,\n maximum length, and overlap—to fit different document formats or scenarios.\n Preprocessing options are also available to clean up the text by removing\n excess spaces, URLs, and emails.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 265\n id: '1751253953926'\n position:\n x: 184.46657789772178\n y: 407.42301051148354\n positionAbsolute:\n x: 184.46657789772178\n y: 407.42301051148354\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 344\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n each with different retrieval strategies. High-Quality mode uses embeddings\n for vectorization and supports vector, full-text, and hybrid retrieval,\n offering more accurate results but higher resource usage. Economical mode\n uses keyword-based inverted indexing with no token consumption but lower\n accuracy; upgrading to High-Quality is possible, but downgrading requires\n creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 344\n id: '1751254117904'\n position:\n x: 479.7628208876065\n y: 472.46585541244207\n positionAbsolute:\n x: 479.7628208876065\n y: 472.46585541244207\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n result:\n description: The result of the general chunk tool.\n properties:\n general_chunks:\n items:\n description: The chunk of the text.\n type: string\n type: array\n type: object\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The text you want to chunk.\n ja_JP: The text you want to chunk.\n pt_BR: The text you want to chunk.\n zh_Hans: 你想要分块的文本。\n label:\n en_US: Input Content\n ja_JP: Input Content\n pt_BR: Input Content\n zh_Hans: 输入变量\n llm_description: The text you want to chunk.\n max: null\n min: null\n name: input_variable\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The delimiter of the chunks.\n ja_JP: The delimiter of the chunks.\n pt_BR: The delimiter of the chunks.\n zh_Hans: 块的分隔符。\n label:\n en_US: Delimiter\n ja_JP: Delimiter\n pt_BR: Delimiter\n zh_Hans: 分隔符\n llm_description: The delimiter of the chunks, the format of the delimiter\n must be a string.\n max: null\n min: null\n name: delimiter\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The maximum chunk length.\n ja_JP: The maximum chunk length.\n pt_BR: The maximum chunk length.\n zh_Hans: 最大块的长度。\n label:\n en_US: Maximum Chunk Length\n ja_JP: Maximum Chunk Length\n pt_BR: Maximum Chunk Length\n zh_Hans: 最大块的长度\n llm_description: The maximum chunk length, the format of the chunk size\n must be an integer.\n max: null\n min: null\n name: max_chunk_length\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The chunk overlap length.\n ja_JP: The chunk overlap length.\n pt_BR: The chunk overlap length.\n zh_Hans: 块的重叠长度。\n label:\n en_US: Chunk Overlap Length\n ja_JP: Chunk Overlap Length\n pt_BR: Chunk Overlap Length\n zh_Hans: 块的重叠长度\n llm_description: The chunk overlap length, the format of the chunk overlap\n length must be an integer.\n max: null\n min: null\n name: chunk_overlap_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: Replace consecutive spaces, newlines and tabs\n ja_JP: Replace consecutive spaces, newlines and tabs\n pt_BR: Replace consecutive spaces, newlines and tabs\n zh_Hans: 替换连续的空格、换行符和制表符\n label:\n en_US: Replace consecutive spaces, newlines and tabs\n ja_JP: Replace consecutive spaces, newlines and tabs\n pt_BR: Replace consecutive spaces, newlines and tabs\n zh_Hans: 替换连续的空格、换行符和制表符\n llm_description: Replace consecutive spaces, newlines and tabs, the format\n of the replace must be a boolean.\n max: null\n min: null\n name: replace_consecutive_spaces_newlines_tabs\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: Delete all URLs and email addresses\n ja_JP: Delete all URLs and email addresses\n pt_BR: Delete all URLs and email addresses\n zh_Hans: 删除所有URL和电子邮件地址\n label:\n en_US: Delete all URLs and email addresses\n ja_JP: Delete all URLs and email addresses\n pt_BR: Delete all URLs and email addresses\n zh_Hans: 删除所有URL和电子邮件地址\n llm_description: Delete all URLs and email addresses, the format of the\n delete must be a boolean.\n max: null\n min: null\n name: delete_all_urls_and_email_addresses\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n params:\n chunk_overlap_length: ''\n delete_all_urls_and_email_addresses: ''\n delimiter: ''\n input_variable: ''\n max_chunk_length: ''\n replace_consecutive_spaces_newlines_tabs: ''\n provider_id: langgenius/general_chunker/general_chunker\n provider_name: langgenius/general_chunker/general_chunker\n provider_type: builtin\n selected: false\n title: General Chunker\n tool_configurations: {}\n tool_description: A tool for general text chunking mode, the chunks retrieved\n and recalled are the same.\n tool_label: General Chunker\n tool_name: general_chunker\n tool_node_version: '2'\n tool_parameters:\n chunk_overlap_length:\n type: variable\n value:\n - rag\n - shared\n - Chunk_Overlap_Length\n delete_all_urls_and_email_addresses:\n type: variable\n value:\n - rag\n - shared\n - clean_2\n delimiter:\n type: mixed\n value: '{{#rag.shared.Dilmiter#}}'\n input_variable:\n type: mixed\n value: '{{#1753346901505.output#}}'\n max_chunk_length:\n type: variable\n value:\n - rag\n - shared\n - Maximum_Chunk_Length\n replace_consecutive_spaces_newlines_tabs:\n type: variable\n value:\n - rag\n - shared\n - clean_1\n type: tool\n height: 52\n id: '1751337124089'\n position:\n x: 184.46657789772178\n y: 326\n positionAbsolute:\n x: 184.46657789772178\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n output_type: string\n selected: false\n title: Variable Aggregator\n type: variable-aggregator\n variables:\n - - '1750836391776'\n - text\n - - '1753349228522'\n - text\n - - '1754023419266'\n - content\n - - '1756896212061'\n - content\n height: 187\n id: '1753346901505'\n position:\n x: -117.24452412456148\n y: 326\n positionAbsolute:\n x: -117.24452412456148\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_array_file: false\n selected: false\n title: Doc Extractor\n type: document-extractor\n variable_selector:\n - '1756442986174'\n - output\n height: 92\n id: '1753349228522'\n position:\n x: -417.5334221022782\n y: 417.25474169825833\n positionAbsolute:\n x: -417.5334221022782\n y: 417.25474169825833\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Notion\n datasource_name: notion_datasource\n datasource_parameters: {}\n plugin_id: langgenius/notion_datasource\n provider_name: notion_datasource\n provider_type: online_document\n selected: false\n title: Notion\n type: datasource\n height: 52\n id: '1754023419266'\n position:\n x: -1369.6904698303242\n y: 440.01452302398053\n positionAbsolute:\n x: -1369.6904698303242\n y: 440.01452302398053\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n output_type: file\n selected: false\n title: Variable Aggregator\n type: variable-aggregator\n variables:\n - - '1750836380067'\n - file\n - - '1756442998557'\n - file\n height: 135\n id: '1756442986174'\n position:\n x: -1067.06980963949\n y: 236.10252072775984\n positionAbsolute:\n x: -1067.06980963949\n y: 236.10252072775984\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Google Drive\n datasource_name: google_drive\n datasource_parameters: {}\n plugin_id: langgenius/google_drive\n provider_name: google_drive\n provider_type: online_drive\n selected: false\n title: Google Drive\n type: datasource\n height: 52\n id: '1756442998557'\n position:\n x: -1371.6520723158733\n y: 326\n positionAbsolute:\n x: -1371.6520723158733\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n cases:\n - case_id: 'true'\n conditions:\n - comparison_operator: is\n id: 1581dd11-7898-41f4-962f-937283ba7e01\n value: .xlsx\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 92abb46d-d7e4-46e7-a5e1-8a29bb45d528\n value: .xls\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 1dde5ae7-754d-4e83-96b2-fe1f02995d8b\n value: .md\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 7e1a80e5-c32a-46a4-8f92-8912c64972aa\n value: .markdown\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 53abfe95-c7d0-4f63-ad37-17d425d25106\n value: .mdx\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 436877b8-8c0a-4cc6-9565-92754db08571\n value: .html\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 5e3e375e-750b-4204-8ac3-9a1174a5ab7c\n value: .htm\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 1a84a784-a797-4f96-98a0-33a9b48ceb2b\n value: .docx\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 62d11445-876a-493f-85d3-8fc020146bdd\n value: .csv\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 02c4bce8-7668-4ccd-b750-4281f314b231\n value: .txt\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n id: 'true'\n logical_operator: or\n selected: false\n title: IF/ELSE\n type: if-else\n height: 358\n id: '1756443014860'\n position:\n x: -733.5977815139424\n y: 236.10252072775984\n positionAbsolute:\n x: -733.5977815139424\n y: 236.10252072775984\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Jina Reader\n datasource_name: jina_reader\n datasource_parameters:\n crawl_sub_pages:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jina_subpages\n limit:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jina_limit\n url:\n type: mixed\n value: '{{#rag.1756896212061.jina_url#}}'\n use_sitemap:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jian_sitemap\n plugin_id: langgenius/jina_datasource\n provider_name: jinareader\n provider_type: website_crawl\n selected: false\n title: Jina Reader\n type: datasource\n height: 52\n id: '1756896212061'\n position:\n x: -1371.6520723158733\n y: 538.9988445953813\n positionAbsolute:\n x: -1371.6520723158733\n y: 538.9988445953813\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Firecrawl\n datasource_name: crawl\n datasource_parameters:\n crawl_subpages:\n type: variable\n value:\n - rag\n - '1756907397615'\n - firecrawl_subpages\n exclude_paths:\n type: mixed\n value: '{{#rag.1756907397615.exclude_paths#}}'\n include_paths:\n type: mixed\n value: '{{#rag.1756907397615.include_paths#}}'\n limit:\n type: variable\n value:\n - rag\n - '1756907397615'\n - max_pages\n max_depth:\n type: variable\n value:\n - rag\n - '1756907397615'\n - max_depth\n only_main_content:\n type: variable\n value:\n - rag\n - '1756907397615'\n - main_content\n url:\n type: mixed\n value: '{{#rag.1756907397615.firecrawl_url1#}}'\n plugin_id: langgenius/firecrawl_datasource\n provider_name: firecrawl\n provider_type: website_crawl\n selected: false\n title: Firecrawl\n type: datasource\n height: 52\n id: '1756907397615'\n position:\n x: -1371.6520723158733\n y: 644.3296146102903\n positionAbsolute:\n x: -1371.6520723158733\n y: 644.3296146102903\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n viewport:\n x: 1463.3408543698197\n y: 224.29398382646679\n zoom: 0.6387381963193622\n rag_pipeline_variables:\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: jina_reader_url\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: pages\n variable: jina_reader_imit\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: true\n label: Crawl sub-pages\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: checkbox\n unit: null\n variable: Crawl_sub_pages_2\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: true\n label: Use sitemap\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: Use_sitemap\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: jina_url\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: pages\n variable: jina_limit\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: true\n label: Use sitemap\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: Follow the sitemap to crawl the site. If not, Jina Reader will crawl\n iteratively based on page relevance, yielding fewer but higher-quality pages.\n type: checkbox\n unit: null\n variable: jian_sitemap\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: true\n label: Crawl subpages\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: jina_subpages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: firecrawl_url1\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: true\n label: firecrawl_subpages\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: firecrawl_subpages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: Exclude paths\n max_length: 256\n options: []\n placeholder: blog/*,/about/*\n required: false\n tooltips: null\n type: text-input\n unit: null\n variable: exclude_paths\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: include_paths\n max_length: 256\n options: []\n placeholder: articles/*\n required: false\n tooltips: null\n type: text-input\n unit: null\n variable: include_paths\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: 0\n label: Max depth\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: Maximum depth to crawl relative to the entered URL. Depth 0 just scrapes\n the page of the entered url, depth 1 scrapes the url and everything after enteredURL\n + one /, and so on.\n type: number\n unit: null\n variable: max_depth\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: null\n variable: max_pages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: true\n label: Extract only main content (no headers, navs, footers, etc.)\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: main_content\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\\n\n label: Dilmiter\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: A delimiter is the character used to separate text. \\n\\n and \\n are\n commonly used delimiters for separating paragraphs and lines. Combined with\n commas (\\n\\n,\\n), paragraphs will be segmented by lines when exceeding the maximum\n chunk length. You can also use special delimiters defined by yourself (e.g.\n ***).\n type: text-input\n unit: null\n variable: Dilmiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 1024\n label: Maximum Chunk Length\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: tokens\n variable: Maximum_Chunk_Length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 128\n label: Chunk Overlap Length\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: Setting the chunk overlap can maintain the semantic relevance between\n them, enhancing the retrieve effect. It is recommended to set 10%-25% of the\n maximum chunk size.\n type: number\n unit: tokens\n variable: Chunk_Overlap_Length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: true\n label: Replace consecutive spaces, newlines and tabs.\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: clean_1\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: null\n label: Delete all URLs and email addresses.\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: clean_2\n", + "graph": { + "edges": [ + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "knowledge-index" + }, + "id": "1751337124089-source-1750836372241-target", + "selected": false, + "source": "1751337124089", + "sourceHandle": "source", + "target": "1750836372241", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "variable-aggregator", + "targetType": "tool" + }, + "id": "1753346901505-source-1751337124089-target", + "selected": false, + "source": "1753346901505", + "sourceHandle": "source", + "target": "1751337124089", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "variable-aggregator" + }, + "id": "1750836391776-source-1753346901505-target", + "selected": false, + "source": "1750836391776", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "document-extractor", + "targetType": "variable-aggregator" + }, + "id": "1753349228522-source-1753346901505-target", + "selected": false, + "source": "1753349228522", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1754023419266-source-1753346901505-target", + "selected": false, + "source": "1754023419266", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756442998557-source-1756442986174-target", + "selected": false, + "source": "1756442998557", + "sourceHandle": "source", + "target": "1756442986174", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "variable-aggregator", + "targetType": "if-else" + }, + "id": "1756442986174-source-1756443014860-target", + "selected": false, + "source": "1756442986174", + "sourceHandle": "source", + "target": "1756443014860", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1750836380067-source-1756442986174-target", + "selected": false, + "source": "1750836380067", + "sourceHandle": "source", + "target": "1756442986174", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "if-else", + "targetType": "tool" + }, + "id": "1756443014860-true-1750836391776-target", + "selected": false, + "source": "1756443014860", + "sourceHandle": "true", + "target": "1750836391776", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "if-else", + "targetType": "document-extractor" + }, + "id": "1756443014860-false-1753349228522-target", + "selected": false, + "source": "1756443014860", + "sourceHandle": "false", + "target": "1753349228522", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756896212061-source-1753346901505-target", + "source": "1756896212061", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756907397615-source-1753346901505-target", + "source": "1756907397615", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + } + ], + "nodes": [ + { + "data": { + "chunk_structure": "text_model", + "index_chunk_variable_selector": [ + "1751337124089", + "result" + ], + "indexing_technique": "economy", + "keyword_number": 10, + "retrieval_model": { + "score_threshold": 0.5, + "score_threshold_enabled": false, + "search_method": "keyword_search", + "top_k": 3 + }, + "selected": false, + "title": "Knowledge Base", + "type": "knowledge-index" + }, + "height": 114, + "id": "1750836372241", + "position": { + "x": 479.7628208876065, + "y": 326 + }, + "positionAbsolute": { + "x": 479.7628208876065, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "File", + "datasource_name": "upload-file", + "datasource_parameters": {}, + "fileExtensions": [ + "txt", + "markdown", + "mdx", + "pdf", + "html", + "xlsx", + "xls", + "vtt", + "properties", + "doc", + "docx", + "csv", + "eml", + "msg", + "pptx", + "xml", + "epub", + "ppt", + "md" + ], + "plugin_id": "langgenius/file", + "provider_name": "file", + "provider_type": "local_file", + "selected": false, + "title": "File", + "type": "datasource" + }, + "height": 52, + "id": "1750836380067", + "position": { + "x": -1371.6520723158733, + "y": 224.87938381325645 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 224.87938381325645 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "documents": { + "description": "the documents extracted from the file", + "items": { + "type": "object" + }, + "type": "array" + }, + "images": { + "description": "The images extracted from the file", + "items": { + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "ja_JP": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "pt_BR": "o arquivo a ser analisado (suporta pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "zh_Hans": "用于解析的文件(支持 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)" + }, + "label": { + "en_US": "file", + "ja_JP": "file", + "pt_BR": "file", + "zh_Hans": "file" + }, + "llm_description": "the file to be parsed (support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "max": null, + "min": null, + "name": "file", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "file" + } + ], + "params": { + "file": "" + }, + "provider_id": "langgenius/dify_extractor/dify_extractor", + "provider_name": "langgenius/dify_extractor/dify_extractor", + "provider_type": "builtin", + "selected": false, + "title": "Dify Extractor", + "tool_configurations": {}, + "tool_description": "Dify Extractor", + "tool_label": "Dify Extractor", + "tool_name": "dify_extractor", + "tool_node_version": "2", + "tool_parameters": { + "file": { + "type": "variable", + "value": [ + "1756442986174", + "output" + ] + } + }, + "type": "tool" + }, + "height": 52, + "id": "1750836391776", + "position": { + "x": -417.5334221022782, + "y": 268.1692071834485 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 268.1692071834485 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 252, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" starts with Data Source as the starting node and ends with the knowledge base node. The general steps are: import documents from the data source → use extractor to extract document content → split and clean content into structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The user input variables required by the Knowledge Pipeline node must be predefined and managed via the Input Field section located in the top-right corner of the orchestration canvas. It determines what input fields the end users will see and need to fill in when importing files to the knowledge base through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique Inputs: Input fields defined here are only available to the selected data source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global Inputs: These input fields are shared across all subsequent nodes after the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 1124 + }, + "height": 252, + "id": "1751252161631", + "position": { + "x": -1371.6520723158733, + "y": -123.758428116601 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": -123.758428116601 + }, + "selected": true, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 1124 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 388, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\": File Upload, Online Drive, Online Doc, and Web Crawler. Different types of Data Sources have different input and output types. The output of File Upload and Online Drive are files, while the output of Online Doc and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A Knowledge Pipeline can have multiple data sources. Each data source can be selected more than once with different settings. Each added data source is a tab on the add file interface. However, each time the user can only select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 285 + }, + "height": 388, + "id": "1751252440357", + "position": { + "x": -1723.9942193415582, + "y": 224.87938381325645 + }, + "positionAbsolute": { + "x": -1723.9942193415582, + "y": 224.87938381325645 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 285 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 430, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A document extractor in Retrieval-Augmented Generation (RAG) is a tool or component that automatically identifies, extracts, and structures text and data from various types of documents—such as PDFs, images, scanned files, handwritten notes, and more—into a format that can be effectively used by language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Dify Extractor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is a built-in document parser developed by Dify. It supports a wide range of common file formats and offers specialized handling for certain formats, such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\".docx\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\". In addition to text extraction, it can extract images embedded within documents, store them, and return their accessible URLs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 430, + "id": "1751253091602", + "position": { + "x": -417.5334221022782, + "y": 532.832924599999 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 532.832924599999 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 265, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"General Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" divides content into chunks and retrieves the most relevant ones based on the user’s query for LLM processing. You can customize chunking rules—such as delimiter, maximum length, and overlap—to fit different document formats or scenarios. Preprocessing options are also available to clean up the text by removing excess spaces, URLs, and emails.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 265, + "id": "1751253953926", + "position": { + "x": 184.46657789772178, + "y": 407.42301051148354 + }, + "positionAbsolute": { + "x": 184.46657789772178, + "y": 407.42301051148354 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 344, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", each with different retrieval strategies. High-Quality mode uses embeddings for vectorization and supports vector, full-text, and hybrid retrieval, offering more accurate results but higher resource usage. Economical mode uses keyword-based inverted indexing with no token consumption but lower accuracy; upgrading to High-Quality is possible, but downgrading requires creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 344, + "id": "1751254117904", + "position": { + "x": 479.7628208876065, + "y": 472.46585541244207 + }, + "positionAbsolute": { + "x": 479.7628208876065, + "y": 472.46585541244207 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "result": { + "description": "The result of the general chunk tool.", + "properties": { + "general_chunks": { + "items": { + "description": "The chunk of the text.", + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The text you want to chunk.", + "ja_JP": "The text you want to chunk.", + "pt_BR": "The text you want to chunk.", + "zh_Hans": "你想要分块的文本。" + }, + "label": { + "en_US": "Input Content", + "ja_JP": "Input Content", + "pt_BR": "Input Content", + "zh_Hans": "输入变量" + }, + "llm_description": "The text you want to chunk.", + "max": null, + "min": null, + "name": "input_variable", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The delimiter of the chunks.", + "ja_JP": "The delimiter of the chunks.", + "pt_BR": "The delimiter of the chunks.", + "zh_Hans": "块的分隔符。" + }, + "label": { + "en_US": "Delimiter", + "ja_JP": "Delimiter", + "pt_BR": "Delimiter", + "zh_Hans": "分隔符" + }, + "llm_description": "The delimiter of the chunks, the format of the delimiter must be a string.", + "max": null, + "min": null, + "name": "delimiter", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The maximum chunk length.", + "ja_JP": "The maximum chunk length.", + "pt_BR": "The maximum chunk length.", + "zh_Hans": "最大块的长度。" + }, + "label": { + "en_US": "Maximum Chunk Length", + "ja_JP": "Maximum Chunk Length", + "pt_BR": "Maximum Chunk Length", + "zh_Hans": "最大块的长度" + }, + "llm_description": "The maximum chunk length, the format of the chunk size must be an integer.", + "max": null, + "min": null, + "name": "max_chunk_length", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The chunk overlap length.", + "ja_JP": "The chunk overlap length.", + "pt_BR": "The chunk overlap length.", + "zh_Hans": "块的重叠长度。" + }, + "label": { + "en_US": "Chunk Overlap Length", + "ja_JP": "Chunk Overlap Length", + "pt_BR": "Chunk Overlap Length", + "zh_Hans": "块的重叠长度" + }, + "llm_description": "The chunk overlap length, the format of the chunk overlap length must be an integer.", + "max": null, + "min": null, + "name": "chunk_overlap_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "Replace consecutive spaces, newlines and tabs", + "ja_JP": "Replace consecutive spaces, newlines and tabs", + "pt_BR": "Replace consecutive spaces, newlines and tabs", + "zh_Hans": "替换连续的空格、换行符和制表符" + }, + "label": { + "en_US": "Replace consecutive spaces, newlines and tabs", + "ja_JP": "Replace consecutive spaces, newlines and tabs", + "pt_BR": "Replace consecutive spaces, newlines and tabs", + "zh_Hans": "替换连续的空格、换行符和制表符" + }, + "llm_description": "Replace consecutive spaces, newlines and tabs, the format of the replace must be a boolean.", + "max": null, + "min": null, + "name": "replace_consecutive_spaces_newlines_tabs", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "Delete all URLs and email addresses", + "ja_JP": "Delete all URLs and email addresses", + "pt_BR": "Delete all URLs and email addresses", + "zh_Hans": "删除所有URL和电子邮件地址" + }, + "label": { + "en_US": "Delete all URLs and email addresses", + "ja_JP": "Delete all URLs and email addresses", + "pt_BR": "Delete all URLs and email addresses", + "zh_Hans": "删除所有URL和电子邮件地址" + }, + "llm_description": "Delete all URLs and email addresses, the format of the delete must be a boolean.", + "max": null, + "min": null, + "name": "delete_all_urls_and_email_addresses", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + } + ], + "params": { + "chunk_overlap_length": "", + "delete_all_urls_and_email_addresses": "", + "delimiter": "", + "input_variable": "", + "max_chunk_length": "", + "replace_consecutive_spaces_newlines_tabs": "" + }, + "provider_id": "langgenius/general_chunker/general_chunker", + "provider_name": "langgenius/general_chunker/general_chunker", + "provider_type": "builtin", + "selected": false, + "title": "General Chunker", + "tool_configurations": {}, + "tool_description": "A tool for general text chunking mode, the chunks retrieved and recalled are the same.", + "tool_label": "General Chunker", + "tool_name": "general_chunker", + "tool_node_version": "2", + "tool_parameters": { + "chunk_overlap_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Chunk_Overlap_Length" + ] + }, + "delete_all_urls_and_email_addresses": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_2" + ] + }, + "delimiter": { + "type": "mixed", + "value": "{{#rag.shared.Dilmiter#}}" + }, + "input_variable": { + "type": "mixed", + "value": "{{#1753346901505.output#}}" + }, + "max_chunk_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Maximum_Chunk_Length" + ] + }, + "replace_consecutive_spaces_newlines_tabs": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_1" + ] + } + }, + "type": "tool" + }, + "height": 52, + "id": "1751337124089", + "position": { + "x": 184.46657789772178, + "y": 326 + }, + "positionAbsolute": { + "x": 184.46657789772178, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "output_type": "string", + "selected": false, + "title": "Variable Aggregator", + "type": "variable-aggregator", + "variables": [ + [ + "1750836391776", + "text" + ], + [ + "1753349228522", + "text" + ], + [ + "1754023419266", + "content" + ], + [ + "1756896212061", + "content" + ] + ] + }, + "height": 187, + "id": "1753346901505", + "position": { + "x": -117.24452412456148, + "y": 326 + }, + "positionAbsolute": { + "x": -117.24452412456148, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_array_file": false, + "selected": false, + "title": "Doc Extractor", + "type": "document-extractor", + "variable_selector": [ + "1756442986174", + "output" + ] + }, + "height": 92, + "id": "1753349228522", + "position": { + "x": -417.5334221022782, + "y": 417.25474169825833 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 417.25474169825833 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Notion", + "datasource_name": "notion_datasource", + "datasource_parameters": {}, + "plugin_id": "langgenius/notion_datasource", + "provider_name": "notion_datasource", + "provider_type": "online_document", + "selected": false, + "title": "Notion", + "type": "datasource" + }, + "height": 52, + "id": "1754023419266", + "position": { + "x": -1369.6904698303242, + "y": 440.01452302398053 + }, + "positionAbsolute": { + "x": -1369.6904698303242, + "y": 440.01452302398053 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "output_type": "file", + "selected": false, + "title": "Variable Aggregator", + "type": "variable-aggregator", + "variables": [ + [ + "1750836380067", + "file" + ], + [ + "1756442998557", + "file" + ] + ] + }, + "height": 135, + "id": "1756442986174", + "position": { + "x": -1067.06980963949, + "y": 236.10252072775984 + }, + "positionAbsolute": { + "x": -1067.06980963949, + "y": 236.10252072775984 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Google Drive", + "datasource_name": "google_drive", + "datasource_parameters": {}, + "plugin_id": "langgenius/google_drive", + "provider_name": "google_drive", + "provider_type": "online_drive", + "selected": false, + "title": "Google Drive", + "type": "datasource" + }, + "height": 52, + "id": "1756442998557", + "position": { + "x": -1371.6520723158733, + "y": 326 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "cases": [ + { + "case_id": "true", + "conditions": [ + { + "comparison_operator": "is", + "id": "1581dd11-7898-41f4-962f-937283ba7e01", + "value": ".xlsx", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "92abb46d-d7e4-46e7-a5e1-8a29bb45d528", + "value": ".xls", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "1dde5ae7-754d-4e83-96b2-fe1f02995d8b", + "value": ".md", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "7e1a80e5-c32a-46a4-8f92-8912c64972aa", + "value": ".markdown", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "53abfe95-c7d0-4f63-ad37-17d425d25106", + "value": ".mdx", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "436877b8-8c0a-4cc6-9565-92754db08571", + "value": ".html", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "5e3e375e-750b-4204-8ac3-9a1174a5ab7c", + "value": ".htm", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "1a84a784-a797-4f96-98a0-33a9b48ceb2b", + "value": ".docx", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "62d11445-876a-493f-85d3-8fc020146bdd", + "value": ".csv", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "02c4bce8-7668-4ccd-b750-4281f314b231", + "value": ".txt", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + } + ], + "id": "true", + "logical_operator": "or" + } + ], + "selected": false, + "title": "IF/ELSE", + "type": "if-else" + }, + "height": 358, + "id": "1756443014860", + "position": { + "x": -733.5977815139424, + "y": 236.10252072775984 + }, + "positionAbsolute": { + "x": -733.5977815139424, + "y": 236.10252072775984 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Jina Reader", + "datasource_name": "jina_reader", + "datasource_parameters": { + "crawl_sub_pages": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jina_subpages" + ] + }, + "limit": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jina_limit" + ] + }, + "url": { + "type": "mixed", + "value": "{{#rag.1756896212061.jina_url#}}" + }, + "use_sitemap": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jian_sitemap" + ] + } + }, + "plugin_id": "langgenius/jina_datasource", + "provider_name": "jinareader", + "provider_type": "website_crawl", + "selected": false, + "title": "Jina Reader", + "type": "datasource" + }, + "height": 52, + "id": "1756896212061", + "position": { + "x": -1371.6520723158733, + "y": 538.9988445953813 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 538.9988445953813 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Firecrawl", + "datasource_name": "crawl", + "datasource_parameters": { + "crawl_subpages": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "firecrawl_subpages" + ] + }, + "exclude_paths": { + "type": "mixed", + "value": "{{#rag.1756907397615.exclude_paths#}}" + }, + "include_paths": { + "type": "mixed", + "value": "{{#rag.1756907397615.include_paths#}}" + }, + "limit": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "max_pages" + ] + }, + "max_depth": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "max_depth" + ] + }, + "only_main_content": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "main_content" + ] + }, + "url": { + "type": "mixed", + "value": "{{#rag.1756907397615.firecrawl_url1#}}" + } + }, + "plugin_id": "langgenius/firecrawl_datasource", + "provider_name": "firecrawl", + "provider_type": "website_crawl", + "selected": false, + "title": "Firecrawl", + "type": "datasource" + }, + "height": 52, + "id": "1756907397615", + "position": { + "x": -1371.6520723158733, + "y": 644.3296146102903 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 644.3296146102903 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + } + ], + "viewport": { + "x": 1463.3408543698197, + "y": 224.29398382646679, + "zoom": 0.6387381963193622 + } + }, + "icon_info": { + "icon": "52064ff0-26b6-47d0-902f-e331f94d959b", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "id": "9f5ea5a7-7796-49f3-9e9a-ae2d8e84cfa3", + "name": "General Mode-ECO", + "icon": { + "icon": "52064ff0-26b6-47d0-902f-e331f94d959b", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "language": "zh-Hans", + "position": 1 + }, + "9553b1e0-0c26-445b-9e18-063ad7eca0b4": { + "chunk_structure": "hierarchical_model", + "description": "This template uses an advanced chunking strategy that organizes document text into a hierarchical structure of larger \"parent\" chunks and smaller \"child\" chunks to balance retrieval precision and contextual richness.", + "export_data": "dependencies:\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/notion_datasource:0.1.12@2855c4a7cffd3311118ebe70f095e546f99935e47f12c841123146f728534f55\n version: null\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/dify_extractor:0.0.5@ba7e2fd9165eda73bfcc68e31a108855197e88706e5556c058e0777ab08409b3\n version: null\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/jina:0.0.8@d3a6766fbb80890d73fea7ea04803f3e1702c6e6bd621aafb492b86222a193dd\n version: null\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/parentchild_chunker:0.0.7@ee9c253e7942436b4de0318200af97d98d094262f3c1a56edbe29dcb01fbc158\n version: null\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/google_drive:0.1.6@4bc0cf8f8979ebd7321b91506b4bc8f090b05b769b5d214f2da4ce4c04ce30bd\n version: null\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/jina_datasource:0.0.5@75942f5bbde870ad28e0345ff5ebf54ebd3aec63f0e66344ef76b88cf06b85c3\n version: null\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/firecrawl_datasource:0.2.4@37b490ebc52ac30d1c6cbfa538edcddddcfed7d5f5de58982edbd4e2094eb6e2\n version: null\nkind: rag_pipeline\nrag_pipeline:\n description: ''\n icon: 6509176c-def5-421c-b966-5122ad6bf658\n icon_background: '#FFEAD5'\n icon_type: image\n icon_url: \n name: Parent-child-HQ\nversion: 0.1.0\nworkflow:\n conversation_variables: []\n environment_variables: []\n features: {}\n graph:\n edges:\n - data:\n isInLoop: false\n sourceType: tool\n targetType: variable-aggregator\n id: 1750836391776-source-1753346901505-target\n selected: false\n source: '1750836391776'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: document-extractor\n targetType: variable-aggregator\n id: 1753349228522-source-1753346901505-target\n selected: false\n source: '1753349228522'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1754023419266-source-1753346901505-target\n selected: false\n source: '1754023419266'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756442998557-source-1756442986174-target\n selected: false\n source: '1756442998557'\n sourceHandle: source\n target: '1756442986174'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: variable-aggregator\n targetType: if-else\n id: 1756442986174-source-1756443014860-target\n selected: false\n source: '1756442986174'\n sourceHandle: source\n target: '1756443014860'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1750836380067-source-1756442986174-target\n selected: false\n source: '1750836380067'\n sourceHandle: source\n target: '1756442986174'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: if-else\n targetType: tool\n id: 1756443014860-true-1750836391776-target\n selected: false\n source: '1756443014860'\n sourceHandle: 'true'\n target: '1750836391776'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: if-else\n targetType: document-extractor\n id: 1756443014860-false-1753349228522-target\n selected: false\n source: '1756443014860'\n sourceHandle: 'false'\n target: '1753349228522'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756896212061-source-1753346901505-target\n source: '1756896212061'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756907397615-source-1753346901505-target\n source: '1756907397615'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: variable-aggregator\n targetType: tool\n id: 1753346901505-source-1756972161593-target\n source: '1753346901505'\n sourceHandle: source\n target: '1756972161593'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: knowledge-index\n id: 1756972161593-source-1750836372241-target\n source: '1756972161593'\n sourceHandle: source\n target: '1750836372241'\n targetHandle: target\n type: custom\n zIndex: 0\n nodes:\n - data:\n chunk_structure: hierarchical_model\n embedding_model: jina-embeddings-v2-base-en\n embedding_model_provider: langgenius/jina/jina\n index_chunk_variable_selector:\n - '1756972161593'\n - result\n indexing_technique: high_quality\n keyword_number: 10\n retrieval_model:\n reranking_enable: true\n reranking_mode: reranking_model\n reranking_model:\n reranking_model_name: jina-reranker-v1-base-en\n reranking_provider_name: langgenius/jina/jina\n score_threshold: 0\n score_threshold_enabled: false\n search_method: hybrid_search\n top_k: 3\n weights: null\n selected: false\n title: Knowledge Base\n type: knowledge-index\n height: 114\n id: '1750836372241'\n position:\n x: 479.7628208876065\n y: 326\n positionAbsolute:\n x: 479.7628208876065\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: File\n datasource_name: upload-file\n datasource_parameters: {}\n fileExtensions:\n - txt\n - markdown\n - mdx\n - pdf\n - html\n - xlsx\n - xls\n - vtt\n - properties\n - doc\n - docx\n - csv\n - eml\n - msg\n - pptx\n - xml\n - epub\n - ppt\n - md\n plugin_id: langgenius/file\n provider_name: file\n provider_type: local_file\n selected: false\n title: File\n type: datasource\n height: 52\n id: '1750836380067'\n position:\n x: -1371.6520723158733\n y: 224.87938381325645\n positionAbsolute:\n x: -1371.6520723158733\n y: 224.87938381325645\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n documents:\n description: the documents extracted from the file\n items:\n type: object\n type: array\n images:\n description: The images extracted from the file\n items:\n type: object\n type: array\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n ja_JP: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n pt_BR: o arquivo a ser analisado (suporta pdf, ppt, pptx, doc, docx, png,\n jpg, jpeg)\n zh_Hans: 用于解析的文件(支持 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)\n label:\n en_US: file\n ja_JP: file\n pt_BR: file\n zh_Hans: file\n llm_description: the file to be parsed (support pdf, ppt, pptx, doc, docx,\n png, jpg, jpeg)\n max: null\n min: null\n name: file\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: file\n params:\n file: ''\n provider_id: langgenius/dify_extractor/dify_extractor\n provider_name: langgenius/dify_extractor/dify_extractor\n provider_type: builtin\n selected: false\n title: Dify Extractor\n tool_configurations: {}\n tool_description: Dify Extractor\n tool_label: Dify Extractor\n tool_name: dify_extractor\n tool_node_version: '2'\n tool_parameters:\n file:\n type: variable\n value:\n - '1756442986174'\n - output\n type: tool\n height: 52\n id: '1750836391776'\n position:\n x: -417.5334221022782\n y: 268.1692071834485\n positionAbsolute:\n x: -417.5334221022782\n y: 268.1692071834485\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 252\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge\n Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n starts with Data Source as the starting node and ends with the knowledge\n base node. The general steps are: import documents from the data source\n → use extractor to extract document content → split and clean content into\n structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n user input variables required by the Knowledge Pipeline node must be predefined\n and managed via the Input Field section located in the top-right corner\n of the orchestration canvas. It determines what input fields the end users\n will see and need to fill in when importing files to the knowledge base\n through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique\n Inputs: Input fields defined here are only available to the selected data\n source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global\n Inputs: These input fields are shared across all subsequent nodes after\n the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For\n more information, see https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 1124\n height: 252\n id: '1751252161631'\n position:\n x: -1371.6520723158733\n y: -123.758428116601\n positionAbsolute:\n x: -1371.6520723158733\n y: -123.758428116601\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 1124\n - data:\n author: TenTen\n desc: ''\n height: 388\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently\n we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data\n Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\":\n File Upload, Online Drive, Online Doc, and Web Crawler. Different types\n of Data Sources have different input and output types. The output of File\n Upload and Online Drive are files, while the output of Online Doc and WebCrawler\n are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n Knowledge Pipeline can have multiple data sources. Each data source can\n be selected more than once with different settings. Each added data source\n is a tab on the add file interface. However, each time the user can only\n select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 285\n height: 388\n id: '1751252440357'\n position:\n x: -1723.9942193415582\n y: 224.87938381325645\n positionAbsolute:\n x: -1723.9942193415582\n y: 224.87938381325645\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 285\n - data:\n author: TenTen\n desc: ''\n height: 430\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n document extractor in Retrieval-Augmented Generation (RAG) is a tool or\n component that automatically identifies, extracts, and structures text and\n data from various types of documents—such as PDFs, images, scanned files,\n handwritten notes, and more—into a format that can be effectively used by\n language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Dify\n Extractor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is\n a built-in document parser developed by Dify. It supports a wide range of\n common file formats and offers specialized handling for certain formats,\n such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\".docx\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\n In addition to text extraction, it can extract images embedded within documents,\n store them, and return their accessible URLs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 430\n id: '1751253091602'\n position:\n x: -417.5334221022782\n y: 547.4103414077279\n positionAbsolute:\n x: -417.5334221022782\n y: 547.4103414077279\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 638\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n addresses the dilemma of context and precision by leveraging a two-tier\n hierarchical approach that effectively balances the trade-off between accurate\n matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here\n is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Query Matching with Child Chunks: Small, focused pieces of information,\n often as concise as a single sentence within a paragraph, are used to match\n the user''s query. These child chunks enable precise and relevant initial\n retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Contextual Enrichment with Parent Chunks: Larger, encompassing sections—such\n as a paragraph, a section, or even an entire document—that include the matched\n child chunks are then retrieved. These parent chunks provide comprehensive\n context for the Language Model (LLM). length, and overlap—to fit different\n document formats or scenarios. Preprocessing options are also available\n to clean up the text by removing excess spaces, URLs, and emails.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 638\n id: '1751253953926'\n position:\n x: 184.46657789772178\n y: 407.42301051148354\n positionAbsolute:\n x: 184.46657789772178\n y: 407.42301051148354\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 410\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n each with different retrieval strategies. High-Quality mode uses embeddings\n for vectorization and supports vector, full-text, and hybrid retrieval,\n offering more accurate results but higher resource usage. Economical mode\n uses keyword-based inverted indexing with no token consumption but lower\n accuracy; upgrading to High-Quality is possible, but downgrading requires\n creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"*\n Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only\n support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing\n method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 410\n id: '1751254117904'\n position:\n x: 479.7628208876065\n y: 472.46585541244207\n positionAbsolute:\n x: 479.7628208876065\n y: 472.46585541244207\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n output_type: string\n selected: false\n title: Variable Aggregator\n type: variable-aggregator\n variables:\n - - '1750836391776'\n - text\n - - '1753349228522'\n - text\n - - '1754023419266'\n - content\n - - '1756896212061'\n - content\n - - '1756907397615'\n - content\n height: 213\n id: '1753346901505'\n position:\n x: -117.24452412456148\n y: 326\n positionAbsolute:\n x: -117.24452412456148\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_array_file: false\n selected: false\n title: Doc Extractor\n type: document-extractor\n variable_selector:\n - '1756442986174'\n - output\n height: 92\n id: '1753349228522'\n position:\n x: -417.5334221022782\n y: 417.25474169825833\n positionAbsolute:\n x: -417.5334221022782\n y: 417.25474169825833\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Notion\n datasource_name: notion_datasource\n datasource_parameters: {}\n plugin_id: langgenius/notion_datasource\n provider_name: notion_datasource\n provider_type: online_document\n selected: false\n title: Notion\n type: datasource\n height: 52\n id: '1754023419266'\n position:\n x: -1369.6904698303242\n y: 440.01452302398053\n positionAbsolute:\n x: -1369.6904698303242\n y: 440.01452302398053\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n output_type: file\n selected: false\n title: Variable Aggregator\n type: variable-aggregator\n variables:\n - - '1750836380067'\n - file\n - - '1756442998557'\n - file\n height: 135\n id: '1756442986174'\n position:\n x: -1054.415447856335\n y: 236.10252072775984\n positionAbsolute:\n x: -1054.415447856335\n y: 236.10252072775984\n selected: true\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Google Drive\n datasource_name: google_drive\n datasource_parameters: {}\n plugin_id: langgenius/google_drive\n provider_name: google_drive\n provider_type: online_drive\n selected: false\n title: Google Drive\n type: datasource\n height: 52\n id: '1756442998557'\n position:\n x: -1371.6520723158733\n y: 326\n positionAbsolute:\n x: -1371.6520723158733\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n cases:\n - case_id: 'true'\n conditions:\n - comparison_operator: is\n id: 1581dd11-7898-41f4-962f-937283ba7e01\n value: .xlsx\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 92abb46d-d7e4-46e7-a5e1-8a29bb45d528\n value: .xls\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 1dde5ae7-754d-4e83-96b2-fe1f02995d8b\n value: .md\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 7e1a80e5-c32a-46a4-8f92-8912c64972aa\n value: .markdown\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 53abfe95-c7d0-4f63-ad37-17d425d25106\n value: .mdx\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 436877b8-8c0a-4cc6-9565-92754db08571\n value: .html\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 5e3e375e-750b-4204-8ac3-9a1174a5ab7c\n value: .htm\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 1a84a784-a797-4f96-98a0-33a9b48ceb2b\n value: .docx\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 62d11445-876a-493f-85d3-8fc020146bdd\n value: .csv\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 02c4bce8-7668-4ccd-b750-4281f314b231\n value: .txt\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n id: 'true'\n logical_operator: or\n selected: false\n title: IF/ELSE\n type: if-else\n height: 358\n id: '1756443014860'\n position:\n x: -733.5977815139424\n y: 236.10252072775984\n positionAbsolute:\n x: -733.5977815139424\n y: 236.10252072775984\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Jina Reader\n datasource_name: jina_reader\n datasource_parameters:\n crawl_sub_pages:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jina_subpages\n limit:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jina_limit\n url:\n type: mixed\n value: '{{#rag.1756896212061.jina_url#}}'\n use_sitemap:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jian_sitemap\n plugin_id: langgenius/jina_datasource\n provider_name: jinareader\n provider_type: website_crawl\n selected: false\n title: Jina Reader\n type: datasource\n height: 52\n id: '1756896212061'\n position:\n x: -1371.6520723158733\n y: 538.9988445953813\n positionAbsolute:\n x: -1371.6520723158733\n y: 538.9988445953813\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Firecrawl\n datasource_name: crawl\n datasource_parameters:\n crawl_subpages:\n type: variable\n value:\n - rag\n - '1756907397615'\n - firecrawl_subpages\n exclude_paths:\n type: mixed\n value: '{{#rag.1756907397615.exclude_paths#}}'\n include_paths:\n type: mixed\n value: '{{#rag.1756907397615.include_paths#}}'\n limit:\n type: variable\n value:\n - rag\n - '1756907397615'\n - max_pages\n max_depth:\n type: variable\n value:\n - rag\n - '1756907397615'\n - max_depth\n only_main_content:\n type: variable\n value:\n - rag\n - '1756907397615'\n - main_content\n url:\n type: mixed\n value: '{{#rag.1756907397615.firecrawl_url1#}}'\n plugin_id: langgenius/firecrawl_datasource\n provider_name: firecrawl\n provider_type: website_crawl\n selected: false\n title: Firecrawl\n type: datasource\n height: 52\n id: '1756907397615'\n position:\n x: -1371.6520723158733\n y: 644.3296146102903\n positionAbsolute:\n x: -1371.6520723158733\n y: 644.3296146102903\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The text you want to chunk.\n ja_JP: The text you want to chunk.\n pt_BR: Conteúdo de Entrada\n zh_Hans: 输入文本\n label:\n en_US: Input Content\n ja_JP: Input Content\n pt_BR: Conteúdo de Entrada\n zh_Hans: 输入文本\n llm_description: The text you want to chunk.\n max: null\n min: null\n name: input_text\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: paragraph\n form: llm\n human_description:\n en_US: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n ja_JP: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n pt_BR: Dividir texto em parágrafos com base no separador e no comprimento\n máximo do bloco, usando o texto dividido como bloco pai ou documento\n completo como bloco pai e diretamente recuperá-lo.\n zh_Hans: 根据分隔符和最大块长度将文本拆分为段落,使用拆分文本作为检索的父块或整个文档用作父块并直接检索。\n label:\n en_US: Parent Mode\n ja_JP: Parent Mode\n pt_BR: Modo Pai\n zh_Hans: 父块模式\n llm_description: Split text into paragraphs based on separator and maximum\n chunk length, using split text as parent block or entire document as parent\n block and directly retrieve.\n max: null\n min: null\n name: parent_mode\n options:\n - icon: ''\n label:\n en_US: paragraph\n ja_JP: paragraph\n pt_BR: paragraph\n zh_Hans: paragraph\n value: paragraph\n - icon: ''\n label:\n en_US: full_doc\n ja_JP: full_doc\n pt_BR: full_doc\n zh_Hans: full_doc\n value: full_doc\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: '\n\n\n '\n form: llm\n human_description:\n en_US: Separator used for chunking\n ja_JP: Separator used for chunking\n pt_BR: Separador usado para divisão\n zh_Hans: 用于分块的分隔符\n label:\n en_US: Parent Delimiter\n ja_JP: Parent Delimiter\n pt_BR: Separador de Pai\n zh_Hans: 父块分隔符\n llm_description: The separator used to split chunks\n max: null\n min: null\n name: separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 1024\n form: llm\n human_description:\n en_US: Maximum length for chunking\n ja_JP: Maximum length for chunking\n pt_BR: Comprimento máximo para divisão\n zh_Hans: 用于分块的最大长度\n label:\n en_US: Maximum Parent Chunk Length\n ja_JP: Maximum Parent Chunk Length\n pt_BR: Comprimento Máximo do Bloco Pai\n zh_Hans: 最大父块长度\n llm_description: Maximum length allowed per chunk\n max: null\n min: null\n name: max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: '. '\n form: llm\n human_description:\n en_US: Separator used for subchunking\n ja_JP: Separator used for subchunking\n pt_BR: Separador usado para subdivisão\n zh_Hans: 用于子分块的分隔符\n label:\n en_US: Child Delimiter\n ja_JP: Child Delimiter\n pt_BR: Separador de Subdivisão\n zh_Hans: 子分块分隔符\n llm_description: The separator used to split subchunks\n max: null\n min: null\n name: subchunk_separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 512\n form: llm\n human_description:\n en_US: Maximum length for subchunking\n ja_JP: Maximum length for subchunking\n pt_BR: Comprimento máximo para subdivisão\n zh_Hans: 用于子分块的最大长度\n label:\n en_US: Maximum Child Chunk Length\n ja_JP: Maximum Child Chunk Length\n pt_BR: Comprimento Máximo de Subdivisão\n zh_Hans: 子分块最大长度\n llm_description: Maximum length allowed per subchunk\n max: null\n min: null\n name: subchunk_max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove consecutive spaces, newlines and tabs\n ja_JP: Whether to remove consecutive spaces, newlines and tabs\n pt_BR: Se deve remover espaços extras no texto\n zh_Hans: 是否移除文本中的连续空格、换行符和制表符\n label:\n en_US: Replace consecutive spaces, newlines and tabs\n ja_JP: Replace consecutive spaces, newlines and tabs\n pt_BR: Substituir espaços consecutivos, novas linhas e guias\n zh_Hans: 替换连续空格、换行符和制表符\n llm_description: Whether to remove consecutive spaces, newlines and tabs\n max: null\n min: null\n name: remove_extra_spaces\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove URLs and emails in the text\n ja_JP: Whether to remove URLs and emails in the text\n pt_BR: Se deve remover URLs e e-mails no texto\n zh_Hans: 是否移除文本中的URL和电子邮件地址\n label:\n en_US: Delete all URLs and email addresses\n ja_JP: Delete all URLs and email addresses\n pt_BR: Remover todas as URLs e e-mails\n zh_Hans: 删除所有URL和电子邮件地址\n llm_description: Whether to remove URLs and emails in the text\n max: null\n min: null\n name: remove_urls_emails\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n params:\n input_text: ''\n max_length: ''\n parent_mode: ''\n remove_extra_spaces: ''\n remove_urls_emails: ''\n separator: ''\n subchunk_max_length: ''\n subchunk_separator: ''\n provider_id: langgenius/parentchild_chunker/parentchild_chunker\n provider_name: langgenius/parentchild_chunker/parentchild_chunker\n provider_type: builtin\n selected: false\n title: Parent-child Chunker\n tool_configurations: {}\n tool_description: Process documents into parent-child chunk structures\n tool_label: Parent-child Chunker\n tool_name: parentchild_chunker\n tool_node_version: '2'\n tool_parameters:\n input_text:\n type: mixed\n value: '{{#1753346901505.output#}}'\n max_length:\n type: variable\n value:\n - rag\n - shared\n - parent_length\n parent_mode:\n type: variable\n value:\n - rag\n - shared\n - parent_mode\n remove_extra_spaces:\n type: variable\n value:\n - rag\n - shared\n - clean_1\n remove_urls_emails:\n type: variable\n value:\n - rag\n - shared\n - clean_2\n separator:\n type: mixed\n value: '{{#rag.shared.parent_dilmiter#}}'\n subchunk_max_length:\n type: variable\n value:\n - rag\n - shared\n - child_length\n subchunk_separator:\n type: mixed\n value: '{{#rag.shared.child_delimiter#}}'\n type: tool\n height: 52\n id: '1756972161593'\n position:\n x: 184.46657789772178\n y: 326\n positionAbsolute:\n x: 184.46657789772178\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n viewport:\n x: 947.2141381290828\n y: 179.30600859363653\n zoom: 0.47414481289660987\n rag_pipeline_variables:\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: jina_reader_url\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: pages\n variable: jina_reader_imit\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: true\n label: Crawl sub-pages\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: checkbox\n unit: null\n variable: Crawl_sub_pages_2\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: true\n label: Use sitemap\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: Use_sitemap\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: jina_url\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: pages\n variable: jina_limit\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: true\n label: Use sitemap\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: Follow the sitemap to crawl the site. If not, Jina Reader will crawl\n iteratively based on page relevance, yielding fewer but higher-quality pages.\n type: checkbox\n unit: null\n variable: jian_sitemap\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: true\n label: Crawl subpages\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: jina_subpages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: firecrawl_url1\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: true\n label: firecrawl_subpages\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: firecrawl_subpages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: Exclude paths\n max_length: 256\n options: []\n placeholder: blog/*,/about/*\n required: false\n tooltips: null\n type: text-input\n unit: null\n variable: exclude_paths\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: include_paths\n max_length: 256\n options: []\n placeholder: articles/*\n required: false\n tooltips: null\n type: text-input\n unit: null\n variable: include_paths\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: 0\n label: Max depth\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: Maximum depth to crawl relative to the entered URL. Depth 0 just scrapes\n the page of the entered url, depth 1 scrapes the url and everything after enteredURL\n + one /, and so on.\n type: number\n unit: null\n variable: max_depth\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: null\n variable: max_pages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: true\n label: Extract only main content (no headers, navs, footers, etc.)\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: main_content\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: paragraph\n label: Parent Mode\n max_length: 48\n options:\n - paragraph\n - full_doc\n placeholder: null\n required: true\n tooltips: 'Parent Mode provides two options: paragraph mode splits text into paragraphs\n as parent chunks for retrieval, while full_doc mode uses the entire document\n as a single parent chunk (text beyond 10,000 tokens will be truncated).'\n type: select\n unit: null\n variable: parent_mode\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\\n\n label: Parent Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: A delimiter is the character used to separate text. \\n\\n is recommended\n for splitting the original document into large parent chunks. You can also use\n special delimiters defined by yourself.\n type: text-input\n unit: null\n variable: parent_dilmiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 1024\n label: Maximum Parent Length\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: number\n unit: tokens\n variable: parent_length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\n label: Child Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: A delimiter is the character used to separate text. \\n is recommended\n for splitting parent chunks into small child chunks. You can also use special\n delimiters defined by yourself.\n type: text-input\n unit: null\n variable: child_delimiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 256\n label: Maximum Child Length\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: tokens\n variable: child_length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: true\n label: Replace consecutive spaces, newlines and tabs.\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: clean_1\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: null\n label: Delete all URLs and email addresses.\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: clean_2\n", + "graph": { + "edges": [ + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "variable-aggregator" + }, + "id": "1750836391776-source-1753346901505-target", + "selected": false, + "source": "1750836391776", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "document-extractor", + "targetType": "variable-aggregator" + }, + "id": "1753349228522-source-1753346901505-target", + "selected": false, + "source": "1753349228522", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1754023419266-source-1753346901505-target", + "selected": false, + "source": "1754023419266", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756442998557-source-1756442986174-target", + "selected": false, + "source": "1756442998557", + "sourceHandle": "source", + "target": "1756442986174", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "variable-aggregator", + "targetType": "if-else" + }, + "id": "1756442986174-source-1756443014860-target", + "selected": false, + "source": "1756442986174", + "sourceHandle": "source", + "target": "1756443014860", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1750836380067-source-1756442986174-target", + "selected": false, + "source": "1750836380067", + "sourceHandle": "source", + "target": "1756442986174", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "if-else", + "targetType": "tool" + }, + "id": "1756443014860-true-1750836391776-target", + "selected": false, + "source": "1756443014860", + "sourceHandle": "true", + "target": "1750836391776", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "if-else", + "targetType": "document-extractor" + }, + "id": "1756443014860-false-1753349228522-target", + "selected": false, + "source": "1756443014860", + "sourceHandle": "false", + "target": "1753349228522", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756896212061-source-1753346901505-target", + "source": "1756896212061", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756907397615-source-1753346901505-target", + "source": "1756907397615", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "variable-aggregator", + "targetType": "tool" + }, + "id": "1753346901505-source-1756972161593-target", + "source": "1753346901505", + "sourceHandle": "source", + "target": "1756972161593", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "knowledge-index" + }, + "id": "1756972161593-source-1750836372241-target", + "source": "1756972161593", + "sourceHandle": "source", + "target": "1750836372241", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + } + ], + "nodes": [ + { + "data": { + "chunk_structure": "hierarchical_model", + "embedding_model": "jina-embeddings-v2-base-en", + "embedding_model_provider": "langgenius/jina/jina", + "index_chunk_variable_selector": [ + "1756972161593", + "result" + ], + "indexing_technique": "high_quality", + "keyword_number": 10, + "retrieval_model": { + "reranking_enable": true, + "reranking_mode": "reranking_model", + "reranking_model": { + "reranking_model_name": "jina-reranker-v1-base-en", + "reranking_provider_name": "langgenius/jina/jina" + }, + "score_threshold": 0, + "score_threshold_enabled": false, + "search_method": "hybrid_search", + "top_k": 3, + "weights": null + }, + "selected": false, + "title": "Knowledge Base", + "type": "knowledge-index" + }, + "height": 114, + "id": "1750836372241", + "position": { + "x": 479.7628208876065, + "y": 326 + }, + "positionAbsolute": { + "x": 479.7628208876065, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "File", + "datasource_name": "upload-file", + "datasource_parameters": {}, + "fileExtensions": [ + "txt", + "markdown", + "mdx", + "pdf", + "html", + "xlsx", + "xls", + "vtt", + "properties", + "doc", + "docx", + "csv", + "eml", + "msg", + "pptx", + "xml", + "epub", + "ppt", + "md" + ], + "plugin_id": "langgenius/file", + "provider_name": "file", + "provider_type": "local_file", + "selected": false, + "title": "File", + "type": "datasource" + }, + "height": 52, + "id": "1750836380067", + "position": { + "x": -1371.6520723158733, + "y": 224.87938381325645 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 224.87938381325645 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "documents": { + "description": "the documents extracted from the file", + "items": { + "type": "object" + }, + "type": "array" + }, + "images": { + "description": "The images extracted from the file", + "items": { + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "ja_JP": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "pt_BR": "o arquivo a ser analisado (suporta pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "zh_Hans": "用于解析的文件(支持 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)" + }, + "label": { + "en_US": "file", + "ja_JP": "file", + "pt_BR": "file", + "zh_Hans": "file" + }, + "llm_description": "the file to be parsed (support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "max": null, + "min": null, + "name": "file", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "file" + } + ], + "params": { + "file": "" + }, + "provider_id": "langgenius/dify_extractor/dify_extractor", + "provider_name": "langgenius/dify_extractor/dify_extractor", + "provider_type": "builtin", + "selected": false, + "title": "Dify Extractor", + "tool_configurations": {}, + "tool_description": "Dify Extractor", + "tool_label": "Dify Extractor", + "tool_name": "dify_extractor", + "tool_node_version": "2", + "tool_parameters": { + "file": { + "type": "variable", + "value": [ + "1756442986174", + "output" + ] + } + }, + "type": "tool" + }, + "height": 52, + "id": "1750836391776", + "position": { + "x": -417.5334221022782, + "y": 268.1692071834485 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 268.1692071834485 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 252, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" starts with Data Source as the starting node and ends with the knowledge base node. The general steps are: import documents from the data source → use extractor to extract document content → split and clean content into structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The user input variables required by the Knowledge Pipeline node must be predefined and managed via the Input Field section located in the top-right corner of the orchestration canvas. It determines what input fields the end users will see and need to fill in when importing files to the knowledge base through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique Inputs: Input fields defined here are only available to the selected data source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global Inputs: These input fields are shared across all subsequent nodes after the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For more information, see https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 1124 + }, + "height": 252, + "id": "1751252161631", + "position": { + "x": -1371.6520723158733, + "y": -123.758428116601 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": -123.758428116601 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 1124 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 388, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\": File Upload, Online Drive, Online Doc, and Web Crawler. Different types of Data Sources have different input and output types. The output of File Upload and Online Drive are files, while the output of Online Doc and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A Knowledge Pipeline can have multiple data sources. Each data source can be selected more than once with different settings. Each added data source is a tab on the add file interface. However, each time the user can only select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 285 + }, + "height": 388, + "id": "1751252440357", + "position": { + "x": -1723.9942193415582, + "y": 224.87938381325645 + }, + "positionAbsolute": { + "x": -1723.9942193415582, + "y": 224.87938381325645 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 285 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 430, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A document extractor in Retrieval-Augmented Generation (RAG) is a tool or component that automatically identifies, extracts, and structures text and data from various types of documents—such as PDFs, images, scanned files, handwritten notes, and more—into a format that can be effectively used by language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Dify Extractor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is a built-in document parser developed by Dify. It supports a wide range of common file formats and offers specialized handling for certain formats, such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\".docx\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\". In addition to text extraction, it can extract images embedded within documents, store them, and return their accessible URLs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 430, + "id": "1751253091602", + "position": { + "x": -417.5334221022782, + "y": 547.4103414077279 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 547.4103414077279 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 638, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" addresses the dilemma of context and precision by leveraging a two-tier hierarchical approach that effectively balances the trade-off between accurate matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Query Matching with Child Chunks: Small, focused pieces of information, often as concise as a single sentence within a paragraph, are used to match the user's query. These child chunks enable precise and relevant initial retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Contextual Enrichment with Parent Chunks: Larger, encompassing sections—such as a paragraph, a section, or even an entire document—that include the matched child chunks are then retrieved. These parent chunks provide comprehensive context for the Language Model (LLM). length, and overlap—to fit different document formats or scenarios. Preprocessing options are also available to clean up the text by removing excess spaces, URLs, and emails.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 638, + "id": "1751253953926", + "position": { + "x": 184.46657789772178, + "y": 407.42301051148354 + }, + "positionAbsolute": { + "x": 184.46657789772178, + "y": 407.42301051148354 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 410, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", each with different retrieval strategies. High-Quality mode uses embeddings for vectorization and supports vector, full-text, and hybrid retrieval, offering more accurate results but higher resource usage. Economical mode uses keyword-based inverted indexing with no token consumption but lower accuracy; upgrading to High-Quality is possible, but downgrading requires creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"* Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 410, + "id": "1751254117904", + "position": { + "x": 479.7628208876065, + "y": 472.46585541244207 + }, + "positionAbsolute": { + "x": 479.7628208876065, + "y": 472.46585541244207 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "output_type": "string", + "selected": false, + "title": "Variable Aggregator", + "type": "variable-aggregator", + "variables": [ + [ + "1750836391776", + "text" + ], + [ + "1753349228522", + "text" + ], + [ + "1754023419266", + "content" + ], + [ + "1756896212061", + "content" + ], + [ + "1756907397615", + "content" + ] + ] + }, + "height": 213, + "id": "1753346901505", + "position": { + "x": -117.24452412456148, + "y": 326 + }, + "positionAbsolute": { + "x": -117.24452412456148, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_array_file": false, + "selected": false, + "title": "Doc Extractor", + "type": "document-extractor", + "variable_selector": [ + "1756442986174", + "output" + ] + }, + "height": 92, + "id": "1753349228522", + "position": { + "x": -417.5334221022782, + "y": 417.25474169825833 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 417.25474169825833 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Notion", + "datasource_name": "notion_datasource", + "datasource_parameters": {}, + "plugin_id": "langgenius/notion_datasource", + "provider_name": "notion_datasource", + "provider_type": "online_document", + "selected": false, + "title": "Notion", + "type": "datasource" + }, + "height": 52, + "id": "1754023419266", + "position": { + "x": -1369.6904698303242, + "y": 440.01452302398053 + }, + "positionAbsolute": { + "x": -1369.6904698303242, + "y": 440.01452302398053 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "output_type": "file", + "selected": false, + "title": "Variable Aggregator", + "type": "variable-aggregator", + "variables": [ + [ + "1750836380067", + "file" + ], + [ + "1756442998557", + "file" + ] + ] + }, + "height": 135, + "id": "1756442986174", + "position": { + "x": -1054.415447856335, + "y": 236.10252072775984 + }, + "positionAbsolute": { + "x": -1054.415447856335, + "y": 236.10252072775984 + }, + "selected": true, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Google Drive", + "datasource_name": "google_drive", + "datasource_parameters": {}, + "plugin_id": "langgenius/google_drive", + "provider_name": "google_drive", + "provider_type": "online_drive", + "selected": false, + "title": "Google Drive", + "type": "datasource" + }, + "height": 52, + "id": "1756442998557", + "position": { + "x": -1371.6520723158733, + "y": 326 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "cases": [ + { + "case_id": "true", + "conditions": [ + { + "comparison_operator": "is", + "id": "1581dd11-7898-41f4-962f-937283ba7e01", + "value": ".xlsx", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "92abb46d-d7e4-46e7-a5e1-8a29bb45d528", + "value": ".xls", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "1dde5ae7-754d-4e83-96b2-fe1f02995d8b", + "value": ".md", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "7e1a80e5-c32a-46a4-8f92-8912c64972aa", + "value": ".markdown", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "53abfe95-c7d0-4f63-ad37-17d425d25106", + "value": ".mdx", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "436877b8-8c0a-4cc6-9565-92754db08571", + "value": ".html", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "5e3e375e-750b-4204-8ac3-9a1174a5ab7c", + "value": ".htm", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "1a84a784-a797-4f96-98a0-33a9b48ceb2b", + "value": ".docx", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "62d11445-876a-493f-85d3-8fc020146bdd", + "value": ".csv", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "02c4bce8-7668-4ccd-b750-4281f314b231", + "value": ".txt", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + } + ], + "id": "true", + "logical_operator": "or" + } + ], + "selected": false, + "title": "IF/ELSE", + "type": "if-else" + }, + "height": 358, + "id": "1756443014860", + "position": { + "x": -733.5977815139424, + "y": 236.10252072775984 + }, + "positionAbsolute": { + "x": -733.5977815139424, + "y": 236.10252072775984 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Jina Reader", + "datasource_name": "jina_reader", + "datasource_parameters": { + "crawl_sub_pages": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jina_subpages" + ] + }, + "limit": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jina_limit" + ] + }, + "url": { + "type": "mixed", + "value": "{{#rag.1756896212061.jina_url#}}" + }, + "use_sitemap": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jian_sitemap" + ] + } + }, + "plugin_id": "langgenius/jina_datasource", + "provider_name": "jinareader", + "provider_type": "website_crawl", + "selected": false, + "title": "Jina Reader", + "type": "datasource" + }, + "height": 52, + "id": "1756896212061", + "position": { + "x": -1371.6520723158733, + "y": 538.9988445953813 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 538.9988445953813 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Firecrawl", + "datasource_name": "crawl", + "datasource_parameters": { + "crawl_subpages": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "firecrawl_subpages" + ] + }, + "exclude_paths": { + "type": "mixed", + "value": "{{#rag.1756907397615.exclude_paths#}}" + }, + "include_paths": { + "type": "mixed", + "value": "{{#rag.1756907397615.include_paths#}}" + }, + "limit": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "max_pages" + ] + }, + "max_depth": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "max_depth" + ] + }, + "only_main_content": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "main_content" + ] + }, + "url": { + "type": "mixed", + "value": "{{#rag.1756907397615.firecrawl_url1#}}" + } + }, + "plugin_id": "langgenius/firecrawl_datasource", + "provider_name": "firecrawl", + "provider_type": "website_crawl", + "selected": false, + "title": "Firecrawl", + "type": "datasource" + }, + "height": 52, + "id": "1756907397615", + "position": { + "x": -1371.6520723158733, + "y": 644.3296146102903 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 644.3296146102903 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The text you want to chunk.", + "ja_JP": "The text you want to chunk.", + "pt_BR": "Conteúdo de Entrada", + "zh_Hans": "输入文本" + }, + "label": { + "en_US": "Input Content", + "ja_JP": "Input Content", + "pt_BR": "Conteúdo de Entrada", + "zh_Hans": "输入文本" + }, + "llm_description": "The text you want to chunk.", + "max": null, + "min": null, + "name": "input_text", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": "paragraph", + "form": "llm", + "human_description": { + "en_US": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "ja_JP": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "pt_BR": "Dividir texto em parágrafos com base no separador e no comprimento máximo do bloco, usando o texto dividido como bloco pai ou documento completo como bloco pai e diretamente recuperá-lo.", + "zh_Hans": "根据分隔符和最大块长度将文本拆分为段落,使用拆分文本作为检索的父块或整个文档用作父块并直接检索。" + }, + "label": { + "en_US": "Parent Mode", + "ja_JP": "Parent Mode", + "pt_BR": "Modo Pai", + "zh_Hans": "父块模式" + }, + "llm_description": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "max": null, + "min": null, + "name": "parent_mode", + "options": [ + { + "icon": "", + "label": { + "en_US": "paragraph", + "ja_JP": "paragraph", + "pt_BR": "paragraph", + "zh_Hans": "paragraph" + }, + "value": "paragraph" + }, + { + "icon": "", + "label": { + "en_US": "full_doc", + "ja_JP": "full_doc", + "pt_BR": "full_doc", + "zh_Hans": "full_doc" + }, + "value": "full_doc" + } + ], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": "\n\n", + "form": "llm", + "human_description": { + "en_US": "Separator used for chunking", + "ja_JP": "Separator used for chunking", + "pt_BR": "Separador usado para divisão", + "zh_Hans": "用于分块的分隔符" + }, + "label": { + "en_US": "Parent Delimiter", + "ja_JP": "Parent Delimiter", + "pt_BR": "Separador de Pai", + "zh_Hans": "父块分隔符" + }, + "llm_description": "The separator used to split chunks", + "max": null, + "min": null, + "name": "separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 1024, + "form": "llm", + "human_description": { + "en_US": "Maximum length for chunking", + "ja_JP": "Maximum length for chunking", + "pt_BR": "Comprimento máximo para divisão", + "zh_Hans": "用于分块的最大长度" + }, + "label": { + "en_US": "Maximum Parent Chunk Length", + "ja_JP": "Maximum Parent Chunk Length", + "pt_BR": "Comprimento Máximo do Bloco Pai", + "zh_Hans": "最大父块长度" + }, + "llm_description": "Maximum length allowed per chunk", + "max": null, + "min": null, + "name": "max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": ". ", + "form": "llm", + "human_description": { + "en_US": "Separator used for subchunking", + "ja_JP": "Separator used for subchunking", + "pt_BR": "Separador usado para subdivisão", + "zh_Hans": "用于子分块的分隔符" + }, + "label": { + "en_US": "Child Delimiter", + "ja_JP": "Child Delimiter", + "pt_BR": "Separador de Subdivisão", + "zh_Hans": "子分块分隔符" + }, + "llm_description": "The separator used to split subchunks", + "max": null, + "min": null, + "name": "subchunk_separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 512, + "form": "llm", + "human_description": { + "en_US": "Maximum length for subchunking", + "ja_JP": "Maximum length for subchunking", + "pt_BR": "Comprimento máximo para subdivisão", + "zh_Hans": "用于子分块的最大长度" + }, + "label": { + "en_US": "Maximum Child Chunk Length", + "ja_JP": "Maximum Child Chunk Length", + "pt_BR": "Comprimento Máximo de Subdivisão", + "zh_Hans": "子分块最大长度" + }, + "llm_description": "Maximum length allowed per subchunk", + "max": null, + "min": null, + "name": "subchunk_max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove consecutive spaces, newlines and tabs", + "ja_JP": "Whether to remove consecutive spaces, newlines and tabs", + "pt_BR": "Se deve remover espaços extras no texto", + "zh_Hans": "是否移除文本中的连续空格、换行符和制表符" + }, + "label": { + "en_US": "Replace consecutive spaces, newlines and tabs", + "ja_JP": "Replace consecutive spaces, newlines and tabs", + "pt_BR": "Substituir espaços consecutivos, novas linhas e guias", + "zh_Hans": "替换连续空格、换行符和制表符" + }, + "llm_description": "Whether to remove consecutive spaces, newlines and tabs", + "max": null, + "min": null, + "name": "remove_extra_spaces", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove URLs and emails in the text", + "ja_JP": "Whether to remove URLs and emails in the text", + "pt_BR": "Se deve remover URLs e e-mails no texto", + "zh_Hans": "是否移除文本中的URL和电子邮件地址" + }, + "label": { + "en_US": "Delete all URLs and email addresses", + "ja_JP": "Delete all URLs and email addresses", + "pt_BR": "Remover todas as URLs e e-mails", + "zh_Hans": "删除所有URL和电子邮件地址" + }, + "llm_description": "Whether to remove URLs and emails in the text", + "max": null, + "min": null, + "name": "remove_urls_emails", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + } + ], + "params": { + "input_text": "", + "max_length": "", + "parent_mode": "", + "remove_extra_spaces": "", + "remove_urls_emails": "", + "separator": "", + "subchunk_max_length": "", + "subchunk_separator": "" + }, + "provider_id": "langgenius/parentchild_chunker/parentchild_chunker", + "provider_name": "langgenius/parentchild_chunker/parentchild_chunker", + "provider_type": "builtin", + "selected": false, + "title": "Parent-child Chunker", + "tool_configurations": {}, + "tool_description": "Process documents into parent-child chunk structures", + "tool_label": "Parent-child Chunker", + "tool_name": "parentchild_chunker", + "tool_node_version": "2", + "tool_parameters": { + "input_text": { + "type": "mixed", + "value": "{{#1753346901505.output#}}" + }, + "max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "parent_length" + ] + }, + "parent_mode": { + "type": "variable", + "value": [ + "rag", + "shared", + "parent_mode" + ] + }, + "remove_extra_spaces": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_1" + ] + }, + "remove_urls_emails": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_2" + ] + }, + "separator": { + "type": "mixed", + "value": "{{#rag.shared.parent_dilmiter#}}" + }, + "subchunk_max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "child_length" + ] + }, + "subchunk_separator": { + "type": "mixed", + "value": "{{#rag.shared.child_delimiter#}}" + } + }, + "type": "tool" + }, + "height": 52, + "id": "1756972161593", + "position": { + "x": 184.46657789772178, + "y": 326 + }, + "positionAbsolute": { + "x": 184.46657789772178, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + } + ], + "viewport": { + "x": 947.2141381290828, + "y": 179.30600859363653, + "zoom": 0.47414481289660987 + } + }, + "icon_info": { + "icon": "ab8da246-37ba-4bbb-9b24-e7bda0778005", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "id": "9553b1e0-0c26-445b-9e18-063ad7eca0b4", + "name": "Parent-child-HQ", + "icon": { + "icon": "ab8da246-37ba-4bbb-9b24-e7bda0778005", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "language": "zh-Hans", + "position": 2 + }, + "9ef3e66a-11c7-4227-897c-3b0f9a42da1a": { + "chunk_structure": "qa_model", + "description": "This template generates structured Q&A pairs by extracting selected columns from a table. These pairs are indexed by questions, enabling efficient retrieval of relevant answers based on query similarity.", + "export_data": "dependencies:\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/qa_chunk:0.0.8@1fed9644646bdd48792cdf5a1d559a3df336bd3a8edb0807227499fb56dce3af\n version: null\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/jina:0.0.8@d3a6766fbb80890d73fea7ea04803f3e1702c6e6bd621aafb492b86222a193dd\n version: null\nkind: rag_pipeline\nrag_pipeline:\n description: ''\n icon: 769900fc-8a31-4584-94f6-f227357c00c8\n icon_background: null\n icon_type: image\n icon_url: \n name: Simple Q&A\nversion: 0.1.0\nworkflow:\n conversation_variables: []\n environment_variables: []\n features: {}\n graph:\n edges:\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: datasource\n targetType: tool\n id: 1750836380067-source-1753253430271-target\n source: '1750836380067'\n sourceHandle: source\n target: '1753253430271'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: knowledge-index\n id: 1753253430271-source-1750836372241-target\n source: '1753253430271'\n sourceHandle: source\n target: '1750836372241'\n targetHandle: target\n type: custom\n zIndex: 0\n nodes:\n - data:\n chunk_structure: qa_model\n embedding_model: jina-embeddings-v2-base-en\n embedding_model_provider: langgenius/jina/jina\n index_chunk_variable_selector:\n - '1753253430271'\n - result\n indexing_technique: high_quality\n keyword_number: 10\n retrieval_model:\n reranking_enable: false\n reranking_mode: reranking_model\n reranking_model:\n reranking_model_name: null\n reranking_provider_name: null\n score_threshold: 0\n score_threshold_enabled: false\n search_method: semantic_search\n top_k: 3\n weights: null\n selected: true\n title: Knowledge Base\n type: knowledge-index\n height: 114\n id: '1750836372241'\n position:\n x: 160\n y: 326\n positionAbsolute:\n x: 160\n y: 326\n selected: true\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: File\n datasource_name: upload-file\n datasource_parameters: {}\n fileExtensions:\n - csv\n plugin_id: langgenius/file\n provider_name: file\n provider_type: local_file\n selected: false\n title: File\n type: datasource\n height: 52\n id: '1750836380067'\n position:\n x: -714.4192784522008\n y: 326\n positionAbsolute:\n x: -714.4192784522008\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 249\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge\n Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n starts with Data Source as the starting node and ends with the knowledge\n base node. The general steps are: import documents from the data source\n → use extractor to extract document content → split and clean content into\n structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n user input variables required by the Knowledge Pipeline node must be predefined\n and managed via the Input Field section located in the top-right corner\n of the orchestration canvas. It determines what input fields the end users\n will see and need to fill in when importing files to the knowledge base\n through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique\n Inputs: Input fields defined here are only available to the selected data\n source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global\n Inputs: These input fields are shared across all subsequent nodes after\n the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For\n more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 1115\n height: 249\n id: '1751252161631'\n position:\n x: -714.4192784522008\n y: -19.94142868660783\n positionAbsolute:\n x: -714.4192784522008\n y: -19.94142868660783\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 1115\n - data:\n author: TenTen\n desc: ''\n height: 281\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently\n we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data\n Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\":\n File Upload, Online Drive, Online Doc, and Web Crawler. Different types\n of Data Sources have different input and output types. The output of File\n Upload and Online Drive are files, while the output of Online Doc and WebCrawler\n are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n Knowledge Pipeline can have multiple data sources. Each data source can\n be selected more than once with different settings. Each added data source\n is a tab on the add file interface. However, each time the user can only\n select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 415\n height: 281\n id: '1751252440357'\n position:\n x: -1206.996048993409\n y: 311.5998178583933\n positionAbsolute:\n x: -1206.996048993409\n y: 311.5998178583933\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 415\n - data:\n author: TenTen\n desc: ''\n height: 403\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n each with different retrieval strategies. High-Quality mode uses embeddings\n for vectorization and supports vector, full-text, and hybrid retrieval,\n offering more accurate results but higher resource usage. Economical mode\n uses keyword-based inverted indexing with no token consumption but lower\n accuracy; upgrading to High-Quality is possible, but downgrading requires\n creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"*\n Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only\n support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing\n method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 403\n id: '1751254117904'\n position:\n x: 160\n y: 471.1516409864865\n positionAbsolute:\n x: 160\n y: 471.1516409864865\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 341\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Processor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" extracts\n specified columns from tables to generate structured Q&A pairs. Users can\n independently designate which columns to use for questions and which for\n answers.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"These\n pairs are indexed by the question field, so user queries are matched directly\n against the questions to retrieve the corresponding answers. This \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q-to-Q\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" matching\n strategy improves clarity and precision, especially in scenarios involving\n high-frequency or highly similar user questions.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 341\n id: '1751356019653'\n position:\n x: -282.74494795239\n y: 411.6979750489463\n positionAbsolute:\n x: -282.74494795239\n y: 411.6979750489463\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n result:\n description: The result of the general chunk tool.\n properties:\n qa_chunks:\n items:\n description: The QA chunk.\n properties:\n answer:\n description: The answer of the QA chunk.\n type: string\n question:\n description: The question of the QA chunk.\n type: string\n type: object\n type: array\n type: object\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The file you want to extract QA from.\n ja_JP: The file you want to extract QA from.\n pt_BR: The file you want to extract QA from.\n zh_Hans: 你想要提取 QA 的文件。\n label:\n en_US: Input File\n ja_JP: Input File\n pt_BR: Input File\n zh_Hans: 输入文件\n llm_description: The file you want to extract QA from.\n max: null\n min: null\n name: input_file\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: file\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Column number for question.\n ja_JP: Column number for question.\n pt_BR: Column number for question.\n zh_Hans: 问题所在的列。\n label:\n en_US: Column number for question\n ja_JP: Column number for question\n pt_BR: Column number for question\n zh_Hans: 问题所在的列\n llm_description: The column number for question, the format of the column\n number must be an integer.\n max: null\n min: null\n name: question_column\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: 1\n form: llm\n human_description:\n en_US: Column number for answer.\n ja_JP: Column number for answer.\n pt_BR: Column number for answer.\n zh_Hans: 答案所在的列。\n label:\n en_US: Column number for answer\n ja_JP: Column number for answer\n pt_BR: Column number for answer\n zh_Hans: 答案所在的列\n llm_description: The column number for answer, the format of the column\n number must be an integer.\n max: null\n min: null\n name: answer_column\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: number\n params:\n answer_column: ''\n input_file: ''\n question_column: ''\n provider_id: langgenius/qa_chunk/qa_chunk\n provider_name: langgenius/qa_chunk/qa_chunk\n provider_type: builtin\n selected: false\n title: Q&A PROCESSOR\n tool_configurations: {}\n tool_description: A tool for QA chunking mode.\n tool_label: QA Chunk\n tool_name: qa_chunk\n tool_node_version: '2'\n tool_parameters:\n answer_column:\n type: variable\n value:\n - rag\n - shared\n - Column_Number_for_Answers\n input_file:\n type: variable\n value:\n - '1750836380067'\n - file\n question_column:\n type: variable\n value:\n - rag\n - shared\n - Column_Number_for_Questions\n type: tool\n height: 52\n id: '1753253430271'\n position:\n x: -282.74494795239\n y: 326\n positionAbsolute:\n x: -282.74494795239\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 173\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Simple\n Q&A Template\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" requires\n a pre-prepared table of question-answer pairs. As a result, it only supports\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"File\n Upload\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" data\n source, accepting \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\"csv\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" file\n formats.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 173\n id: '1753411065636'\n position:\n x: -714.4192784522008\n y: 411.6979750489463\n positionAbsolute:\n x: -714.4192784522008\n y: 411.6979750489463\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n viewport:\n x: 698.8920691163195\n y: 311.46417000656925\n zoom: 0.41853867943092266\n rag_pipeline_variables:\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 1\n label: Column Number for Questions\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: Specify a column in the table as Questions. The number of first column is\n 0.\n type: number\n unit: ''\n variable: Column_Number_for_Questions\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 2\n label: Column Number for Answers\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: Specify a column in the table as Answers. The number of first column is\n 0.\n type: number\n unit: null\n variable: Column_Number_for_Answers\n", + "graph": { + "edges": [ + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "datasource", + "targetType": "tool" + }, + "id": "1750836380067-source-1753253430271-target", + "source": "1750836380067", + "sourceHandle": "source", + "target": "1753253430271", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "knowledge-index" + }, + "id": "1753253430271-source-1750836372241-target", + "source": "1753253430271", + "sourceHandle": "source", + "target": "1750836372241", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + } + ], + "nodes": [ + { + "data": { + "chunk_structure": "qa_model", + "embedding_model": "jina-embeddings-v2-base-en", + "embedding_model_provider": "langgenius/jina/jina", + "index_chunk_variable_selector": [ + "1753253430271", + "result" + ], + "indexing_technique": "high_quality", + "keyword_number": 10, + "retrieval_model": { + "reranking_enable": false, + "reranking_mode": "reranking_model", + "reranking_model": { + "reranking_model_name": null, + "reranking_provider_name": null + }, + "score_threshold": 0, + "score_threshold_enabled": false, + "search_method": "semantic_search", + "top_k": 3, + "weights": null + }, + "selected": true, + "title": "Knowledge Base", + "type": "knowledge-index" + }, + "height": 114, + "id": "1750836372241", + "position": { + "x": 160, + "y": 326 + }, + "positionAbsolute": { + "x": 160, + "y": 326 + }, + "selected": true, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "File", + "datasource_name": "upload-file", + "datasource_parameters": {}, + "fileExtensions": [ + "csv" + ], + "plugin_id": "langgenius/file", + "provider_name": "file", + "provider_type": "local_file", + "selected": false, + "title": "File", + "type": "datasource" + }, + "height": 52, + "id": "1750836380067", + "position": { + "x": -714.4192784522008, + "y": 326 + }, + "positionAbsolute": { + "x": -714.4192784522008, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 249, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" starts with Data Source as the starting node and ends with the knowledge base node. The general steps are: import documents from the data source → use extractor to extract document content → split and clean content into structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The user input variables required by the Knowledge Pipeline node must be predefined and managed via the Input Field section located in the top-right corner of the orchestration canvas. It determines what input fields the end users will see and need to fill in when importing files to the knowledge base through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique Inputs: Input fields defined here are only available to the selected data source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global Inputs: These input fields are shared across all subsequent nodes after the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 1115 + }, + "height": 249, + "id": "1751252161631", + "position": { + "x": -714.4192784522008, + "y": -19.94142868660783 + }, + "positionAbsolute": { + "x": -714.4192784522008, + "y": -19.94142868660783 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 1115 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 281, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\": File Upload, Online Drive, Online Doc, and Web Crawler. Different types of Data Sources have different input and output types. The output of File Upload and Online Drive are files, while the output of Online Doc and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A Knowledge Pipeline can have multiple data sources. Each data source can be selected more than once with different settings. Each added data source is a tab on the add file interface. However, each time the user can only select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 415 + }, + "height": 281, + "id": "1751252440357", + "position": { + "x": -1206.996048993409, + "y": 311.5998178583933 + }, + "positionAbsolute": { + "x": -1206.996048993409, + "y": 311.5998178583933 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 415 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 403, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", each with different retrieval strategies. High-Quality mode uses embeddings for vectorization and supports vector, full-text, and hybrid retrieval, offering more accurate results but higher resource usage. Economical mode uses keyword-based inverted indexing with no token consumption but lower accuracy; upgrading to High-Quality is possible, but downgrading requires creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"* Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 403, + "id": "1751254117904", + "position": { + "x": 160, + "y": 471.1516409864865 + }, + "positionAbsolute": { + "x": 160, + "y": 471.1516409864865 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 341, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Processor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" extracts specified columns from tables to generate structured Q&A pairs. Users can independently designate which columns to use for questions and which for answers.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"These pairs are indexed by the question field, so user queries are matched directly against the questions to retrieve the corresponding answers. This \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q-to-Q\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" matching strategy improves clarity and precision, especially in scenarios involving high-frequency or highly similar user questions.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 341, + "id": "1751356019653", + "position": { + "x": -282.74494795239, + "y": 411.6979750489463 + }, + "positionAbsolute": { + "x": -282.74494795239, + "y": 411.6979750489463 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "result": { + "description": "The result of the general chunk tool.", + "properties": { + "qa_chunks": { + "items": { + "description": "The QA chunk.", + "properties": { + "answer": { + "description": "The answer of the QA chunk.", + "type": "string" + }, + "question": { + "description": "The question of the QA chunk.", + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The file you want to extract QA from.", + "ja_JP": "The file you want to extract QA from.", + "pt_BR": "The file you want to extract QA from.", + "zh_Hans": "你想要提取 QA 的文件。" + }, + "label": { + "en_US": "Input File", + "ja_JP": "Input File", + "pt_BR": "Input File", + "zh_Hans": "输入文件" + }, + "llm_description": "The file you want to extract QA from.", + "max": null, + "min": null, + "name": "input_file", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "file" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Column number for question.", + "ja_JP": "Column number for question.", + "pt_BR": "Column number for question.", + "zh_Hans": "问题所在的列。" + }, + "label": { + "en_US": "Column number for question", + "ja_JP": "Column number for question", + "pt_BR": "Column number for question", + "zh_Hans": "问题所在的列" + }, + "llm_description": "The column number for question, the format of the column number must be an integer.", + "max": null, + "min": null, + "name": "question_column", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": 1, + "form": "llm", + "human_description": { + "en_US": "Column number for answer.", + "ja_JP": "Column number for answer.", + "pt_BR": "Column number for answer.", + "zh_Hans": "答案所在的列。" + }, + "label": { + "en_US": "Column number for answer", + "ja_JP": "Column number for answer", + "pt_BR": "Column number for answer", + "zh_Hans": "答案所在的列" + }, + "llm_description": "The column number for answer, the format of the column number must be an integer.", + "max": null, + "min": null, + "name": "answer_column", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "number" + } + ], + "params": { + "answer_column": "", + "input_file": "", + "question_column": "" + }, + "provider_id": "langgenius/qa_chunk/qa_chunk", + "provider_name": "langgenius/qa_chunk/qa_chunk", + "provider_type": "builtin", + "selected": false, + "title": "Q&A PROCESSOR", + "tool_configurations": {}, + "tool_description": "A tool for QA chunking mode.", + "tool_label": "QA Chunk", + "tool_name": "qa_chunk", + "tool_node_version": "2", + "tool_parameters": { + "answer_column": { + "type": "variable", + "value": [ + "rag", + "shared", + "Column_Number_for_Answers" + ] + }, + "input_file": { + "type": "variable", + "value": [ + "1750836380067", + "file" + ] + }, + "question_column": { + "type": "variable", + "value": [ + "rag", + "shared", + "Column_Number_for_Questions" + ] + } + }, + "type": "tool" + }, + "height": 52, + "id": "1753253430271", + "position": { + "x": -282.74494795239, + "y": 326 + }, + "positionAbsolute": { + "x": -282.74494795239, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 173, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Simple Q&A Template\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" requires a pre-prepared table of question-answer pairs. As a result, it only supports \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"File Upload\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" data source, accepting \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\"csv\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" file formats.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 173, + "id": "1753411065636", + "position": { + "x": -714.4192784522008, + "y": 411.6979750489463 + }, + "positionAbsolute": { + "x": -714.4192784522008, + "y": 411.6979750489463 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + } + ], + "viewport": { + "x": 698.8920691163195, + "y": 311.46417000656925, + "zoom": 0.41853867943092266 + } + }, + "icon_info": { + "icon": "ae0993dc-ff90-48ac-9e35-c31ebae5124b", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "id": "9ef3e66a-11c7-4227-897c-3b0f9a42da1a", + "name": "Simple Q&A", + "icon": { + "icon": "ae0993dc-ff90-48ac-9e35-c31ebae5124b", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "language": "zh-Hans", + "position": 3 + }, + "982d1788-837a-40c8-b7de-d37b09a9b2bc": { + "chunk_structure": "hierarchical_model", + "description": "This template is designed for converting native Office files such as DOCX, XLSX, and PPTX into Markdown to facilitate better information processing. PDF files are not recommended.", + "export_data": "dependencies:\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/jina:0.0.8@d3a6766fbb80890d73fea7ea04803f3e1702c6e6bd621aafb492b86222a193dd\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/parentchild_chunker:0.0.7@ee9c253e7942436b4de0318200af97d98d094262f3c1a56edbe29dcb01fbc158\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: yevanchen/markitdown:0.0.4@776b3e2e930e2ffd28a75bb20fecbe7a020849cf754f86e604acacf1258877f6\nkind: rag_pipeline\nrag_pipeline:\n description: ''\n icon: 9d658c3a-b22f-487d-8223-db51e9012505\n icon_background: null\n icon_type: image\n icon_url: \n name: Convert to Markdown\nversion: 0.1.0\nworkflow:\n conversation_variables: []\n environment_variables: []\n features: {}\n graph:\n edges:\n - data:\n isInLoop: false\n sourceType: tool\n targetType: knowledge-index\n id: 1751336942081-source-1750400198569-target\n selected: false\n source: '1751336942081'\n sourceHandle: source\n target: '1750400198569'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: tool\n id: 1750400203722-source-1751359716720-target\n selected: false\n source: '1750400203722'\n sourceHandle: source\n target: '1751359716720'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: tool\n id: 1751359716720-source-1751336942081-target\n source: '1751359716720'\n sourceHandle: source\n target: '1751336942081'\n targetHandle: target\n type: custom\n zIndex: 0\n nodes:\n - data:\n chunk_structure: hierarchical_model\n embedding_model: jina-embeddings-v2-base-en\n embedding_model_provider: langgenius/jina/jina\n index_chunk_variable_selector:\n - '1751336942081'\n - result\n indexing_technique: high_quality\n keyword_number: 10\n retrieval_model:\n hybridSearchMode: weighted_score\n score_threshold: 0.5\n score_threshold_enabled: false\n search_method: hybrid_search\n top_k: 3\n vector_setting:\n embedding_model_name: jina-embeddings-v2-base-en\n embedding_provider_name: langgenius/jina/jina\n selected: true\n title: Knowledge Base\n type: knowledge-index\n height: 114\n id: '1750400198569'\n position:\n x: 357.7591396590142\n y: 282\n positionAbsolute:\n x: 357.7591396590142\n y: 282\n selected: true\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: File\n datasource_name: upload-file\n datasource_parameters: {}\n fileExtensions:\n - html\n - xlsx\n - xls\n - doc\n - docx\n - csv\n - pptx\n - xml\n - ppt\n - txt\n plugin_id: langgenius/file\n provider_name: file\n provider_type: local_file\n selected: false\n title: File\n type: datasource\n height: 52\n id: '1750400203722'\n position:\n x: -580.684520226929\n y: 282\n positionAbsolute:\n x: -580.684520226929\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 316\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently\n we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data\n Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\":\n File Upload, Online Drive, Online Doc, and Web Crawler. Different types\n of Data Sources have different input and output types. The output of File\n Upload and Online Drive are files, while the output of Online Doc and WebCrawler\n are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n Knowledge Pipeline can have multiple data sources. Each data source can\n be selected more than once with different settings. Each added data source\n is a tab on the add file interface. However, each time the user can only\n select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 374\n height: 316\n id: '1751264451381'\n position:\n x: -1034.2054006208518\n y: 282\n positionAbsolute:\n x: -1034.2054006208518\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 374\n - data:\n author: TenTen\n desc: ''\n height: 260\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge\n Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n starts with Data Source as the starting node and ends with the knowledge\n base node. The general steps are: import documents from the data source\n → use extractor to extract document content → split and clean content into\n structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n user input variables required by the Knowledge Pipeline node must be predefined\n and managed via the Input Field section located in the top-right corner\n of the orchestration canvas. It determines what input fields the end users\n will see and need to fill in when importing files to the knowledge base\n through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique\n Inputs: Input fields defined here are only available to the selected data\n source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global\n Inputs: These input fields are shared across all subsequent nodes after\n the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For\n more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 1182\n height: 260\n id: '1751266376760'\n position:\n x: -580.684520226929\n y: -21.891401375096322\n positionAbsolute:\n x: -580.684520226929\n y: -21.891401375096322\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 1182\n - data:\n author: TenTen\n desc: ''\n height: 417\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n document extractor in Retrieval-Augmented Generation (RAG) is a tool or\n component that automatically identifies, extracts, and structures text and\n data from various types of documents—such as PDFs, images, scanned files,\n handwritten notes, and more—into a format that can be effectively used by\n language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Markitdown\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n serves as an excellent alternative to traditional document extraction nodes,\n offering robust file conversion capabilities within the Dify ecosystem.\n It leverages MarkItDown''s plugin-based architecture to provide seamless\n conversion of multiple file formats to Markdown.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 241\n height: 417\n id: '1751266402561'\n position:\n x: -266.96080929383595\n y: 372.64040589639495\n positionAbsolute:\n x: -266.96080929383595\n y: 372.64040589639495\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 241\n - data:\n author: TenTen\n desc: ''\n height: 554\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n addresses the dilemma of context and precision by leveraging a two-tier\n hierarchical approach that effectively balances the trade-off between accurate\n matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here\n is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Query Matching with Child Chunks: Small, focused pieces of information,\n often as concise as a single sentence within a paragraph, are used to match\n the user''s query. These child chunks enable precise and relevant initial\n retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Contextual Enrichment with Parent Chunks: Larger, encompassing sections—such\n as a paragraph, a section, or even an entire document—that include the matched\n child chunks are then retrieved. These parent chunks provide comprehensive\n context for the Language Model (LLM).\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 554\n id: '1751266447821'\n position:\n x: 37.74090119950054\n y: 372.64040589639495\n positionAbsolute:\n x: 37.74090119950054\n y: 372.64040589639495\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 411\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n each with different retrieval strategies. High-Quality mode uses embeddings\n for vectorization and supports vector, full-text, and hybrid retrieval,\n offering more accurate results but higher resource usage. Economical mode\n uses keyword-based inverted indexing with no token consumption but lower\n accuracy; upgrading to High-Quality is possible, but downgrading requires\n creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"*\n Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only\n support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing\n method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 411\n id: '1751266580099'\n position:\n x: 357.7591396590142\n y: 434.3959856026883\n positionAbsolute:\n x: 357.7591396590142\n y: 434.3959856026883\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n result:\n description: Parent child chunks result\n items:\n type: object\n type: array\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: ''\n ja_JP: ''\n pt_BR: ''\n zh_Hans: ''\n label:\n en_US: Input Content\n ja_JP: Input Content\n pt_BR: Conteúdo de Entrada\n zh_Hans: 输入文本\n llm_description: The text you want to chunk.\n max: null\n min: null\n name: input_text\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: paragraph\n form: llm\n human_description:\n en_US: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n ja_JP: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n pt_BR: Dividir texto em parágrafos com base no separador e no comprimento\n máximo do bloco, usando o texto dividido como bloco pai ou documento\n completo como bloco pai e diretamente recuperá-lo.\n zh_Hans: 根据分隔符和最大块长度将文本拆分为段落,使用拆分文本作为检索的父块或整个文档用作父块并直接检索。\n label:\n en_US: Parent Mode\n ja_JP: Parent Mode\n pt_BR: Modo Pai\n zh_Hans: 父块模式\n llm_description: Split text into paragraphs based on separator and maximum\n chunk length, using split text as parent block or entire document as parent\n block and directly retrieve.\n max: null\n min: null\n name: parent_mode\n options:\n - label:\n en_US: Paragraph\n ja_JP: Paragraph\n pt_BR: Parágrafo\n zh_Hans: 段落\n value: paragraph\n - label:\n en_US: Full Document\n ja_JP: Full Document\n pt_BR: Documento Completo\n zh_Hans: 全文\n value: full_doc\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: '\n\n\n '\n form: llm\n human_description:\n en_US: Separator used for chunking\n ja_JP: Separator used for chunking\n pt_BR: Separador usado para divisão\n zh_Hans: 用于分块的分隔符\n label:\n en_US: Parent Delimiter\n ja_JP: Parent Delimiter\n pt_BR: Separador de Pai\n zh_Hans: 父块分隔符\n llm_description: The separator used to split chunks\n max: null\n min: null\n name: separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 1024\n form: llm\n human_description:\n en_US: Maximum length for chunking\n ja_JP: Maximum length for chunking\n pt_BR: Comprimento máximo para divisão\n zh_Hans: 用于分块的最大长度\n label:\n en_US: Maximum Parent Chunk Length\n ja_JP: Maximum Parent Chunk Length\n pt_BR: Comprimento Máximo do Bloco Pai\n zh_Hans: 最大父块长度\n llm_description: Maximum length allowed per chunk\n max: null\n min: null\n name: max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: '. '\n form: llm\n human_description:\n en_US: Separator used for subchunking\n ja_JP: Separator used for subchunking\n pt_BR: Separador usado para subdivisão\n zh_Hans: 用于子分块的分隔符\n label:\n en_US: Child Delimiter\n ja_JP: Child Delimiter\n pt_BR: Separador de Subdivisão\n zh_Hans: 子分块分隔符\n llm_description: The separator used to split subchunks\n max: null\n min: null\n name: subchunk_separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 512\n form: llm\n human_description:\n en_US: Maximum length for subchunking\n ja_JP: Maximum length for subchunking\n pt_BR: Comprimento máximo para subdivisão\n zh_Hans: 用于子分块的最大长度\n label:\n en_US: Maximum Child Chunk Length\n ja_JP: Maximum Child Chunk Length\n pt_BR: Comprimento Máximo de Subdivisão\n zh_Hans: 子分块最大长度\n llm_description: Maximum length allowed per subchunk\n max: null\n min: null\n name: subchunk_max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove consecutive spaces, newlines and tabs\n ja_JP: Whether to remove consecutive spaces, newlines and tabs\n pt_BR: Se deve remover espaços extras no texto\n zh_Hans: 是否移除文本中的连续空格、换行符和制表符\n label:\n en_US: Replace consecutive spaces, newlines and tabs\n ja_JP: Replace consecutive spaces, newlines and tabs\n pt_BR: Substituir espaços consecutivos, novas linhas e guias\n zh_Hans: 替换连续空格、换行符和制表符\n llm_description: Whether to remove consecutive spaces, newlines and tabs\n max: null\n min: null\n name: remove_extra_spaces\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove URLs and emails in the text\n ja_JP: Whether to remove URLs and emails in the text\n pt_BR: Se deve remover URLs e e-mails no texto\n zh_Hans: 是否移除文本中的URL和电子邮件地址\n label:\n en_US: Delete all URLs and email addresses\n ja_JP: Delete all URLs and email addresses\n pt_BR: Remover todas as URLs e e-mails\n zh_Hans: 删除所有URL和电子邮件地址\n llm_description: Whether to remove URLs and emails in the text\n max: null\n min: null\n name: remove_urls_emails\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n params:\n input_text: ''\n max_length: ''\n parent_mode: ''\n remove_extra_spaces: ''\n remove_urls_emails: ''\n separator: ''\n subchunk_max_length: ''\n subchunk_separator: ''\n provider_id: langgenius/parentchild_chunker/parentchild_chunker\n provider_name: langgenius/parentchild_chunker/parentchild_chunker\n provider_type: builtin\n selected: false\n title: Parent-child Chunker\n tool_configurations: {}\n tool_description: Process documents into parent-child chunk structures\n tool_label: Parent-child Chunker\n tool_name: parentchild_chunker\n tool_node_version: '2'\n tool_parameters:\n input_text:\n type: mixed\n value: '{{#1751359716720.text#}}'\n max_length:\n type: variable\n value:\n - rag\n - shared\n - Maximum_Parent_Length\n parent_mode:\n type: variable\n value:\n - rag\n - shared\n - Parent_Mode\n separator:\n type: mixed\n value: '{{#rag.shared.Parent_Delimiter#}}'\n subchunk_max_length:\n type: variable\n value:\n - rag\n - shared\n - Maximum_Child_Length\n subchunk_separator:\n type: mixed\n value: '{{#rag.shared.Child_Delimiter#}}'\n type: tool\n height: 52\n id: '1751336942081'\n position:\n x: 37.74090119950054\n y: 282\n positionAbsolute:\n x: 37.74090119950054\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n output_schema: null\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: Upload files for processing\n ja_JP: Upload files for processing\n pt_BR: Carregar arquivos para processamento\n zh_Hans: 上传文件进行处理\n label:\n en_US: Files\n ja_JP: Files\n pt_BR: Arquivos\n zh_Hans: 文件\n llm_description: ''\n max: null\n min: null\n name: files\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: files\n params:\n files: ''\n provider_id: yevanchen/markitdown/markitdown\n provider_name: yevanchen/markitdown/markitdown\n provider_type: builtin\n selected: false\n title: markitdown\n tool_configurations: {}\n tool_description: Python tool for converting files and office documents to\n Markdown.\n tool_label: markitdown\n tool_name: markitdown\n tool_node_version: '2'\n tool_parameters:\n files:\n type: variable\n value:\n - '1750400203722'\n - file\n type: tool\n height: 52\n id: '1751359716720'\n position:\n x: -266.96080929383595\n y: 282\n positionAbsolute:\n x: -266.96080929383595\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 301\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"MarkItDown\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is\n recommended for converting and handling a wide range of file formats, particularly\n for transforming content into Markdown. It works especially well for converting\n native Office files—such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"DOCX\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"XLSX\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"PPTX\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"—into\n Markdown to facilitate better information processing. However, as some users\n have noted its suboptimal performance in extracting content from PDF files,\n using it for PDFs is not recommended.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 301\n id: '1753425718313'\n position:\n x: -580.684520226929\n y: 372.64040589639495\n positionAbsolute:\n x: -580.684520226929\n y: 372.64040589639495\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n viewport:\n x: 747.6785299994758\n y: 94.6209873206409\n zoom: 0.8152773235379324\n rag_pipeline_variables:\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: paragraph\n label: Parent Mode\n max_length: 48\n options:\n - paragraph\n - full_doc\n placeholder: null\n required: true\n tooltips: 'Parent Mode provides two options: paragraph mode splits text into paragraphs\n as parent chunks for retrieval, while full_doc mode uses the entire document\n as a single parent chunk (text beyond 10,000 tokens will be truncated).'\n type: select\n unit: null\n variable: Parent_Mode\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\\n\n label: Parent Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: A delimiter is the character used to separate text. \\n\\n is recommended\n for splitting the original document into large parent chunks. You can also use\n special delimiters defined by yourself.\n type: text-input\n unit: null\n variable: Parent_Delimiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 1024\n label: Maximum Parent Length\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: number\n unit: tokens\n variable: Maximum_Parent_Length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\n label: Child Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: A delimiter is the character used to separate text. \\n is recommended\n for splitting parent chunks into small child chunks. You can also use special\n delimiters defined by yourself.\n type: text-input\n unit: null\n variable: Child_Delimiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 256\n label: Maximum Child Length\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: tokens\n variable: Maximum_Child_Length\n", + "graph": { + "edges": [ + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "knowledge-index" + }, + "id": "1751336942081-source-1750400198569-target", + "selected": false, + "source": "1751336942081", + "sourceHandle": "source", + "target": "1750400198569", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "tool" + }, + "id": "1750400203722-source-1751359716720-target", + "selected": false, + "source": "1750400203722", + "sourceHandle": "source", + "target": "1751359716720", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "tool" + }, + "id": "1751359716720-source-1751336942081-target", + "source": "1751359716720", + "sourceHandle": "source", + "target": "1751336942081", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + } + ], + "nodes": [ + { + "data": { + "chunk_structure": "hierarchical_model", + "embedding_model": "jina-embeddings-v2-base-en", + "embedding_model_provider": "langgenius/jina/jina", + "index_chunk_variable_selector": [ + "1751336942081", + "result" + ], + "indexing_technique": "high_quality", + "keyword_number": 10, + "retrieval_model": { + "hybridSearchMode": "weighted_score", + "score_threshold": 0.5, + "score_threshold_enabled": false, + "search_method": "hybrid_search", + "top_k": 3, + "vector_setting": { + "embedding_model_name": "jina-embeddings-v2-base-en", + "embedding_provider_name": "langgenius/jina/jina" + } + }, + "selected": true, + "title": "Knowledge Base", + "type": "knowledge-index" + }, + "height": 114, + "id": "1750400198569", + "position": { + "x": 357.7591396590142, + "y": 282 + }, + "positionAbsolute": { + "x": 357.7591396590142, + "y": 282 + }, + "selected": true, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "File", + "datasource_name": "upload-file", + "datasource_parameters": {}, + "fileExtensions": [ + "html", + "xlsx", + "xls", + "doc", + "docx", + "csv", + "pptx", + "xml", + "ppt", + "txt" + ], + "plugin_id": "langgenius/file", + "provider_name": "file", + "provider_type": "local_file", + "selected": false, + "title": "File", + "type": "datasource" + }, + "height": 52, + "id": "1750400203722", + "position": { + "x": -580.684520226929, + "y": 282 + }, + "positionAbsolute": { + "x": -580.684520226929, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 316, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\": File Upload, Online Drive, Online Doc, and Web Crawler. Different types of Data Sources have different input and output types. The output of File Upload and Online Drive are files, while the output of Online Doc and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A Knowledge Pipeline can have multiple data sources. Each data source can be selected more than once with different settings. Each added data source is a tab on the add file interface. However, each time the user can only select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 374 + }, + "height": 316, + "id": "1751264451381", + "position": { + "x": -1034.2054006208518, + "y": 282 + }, + "positionAbsolute": { + "x": -1034.2054006208518, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 374 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 260, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" starts with Data Source as the starting node and ends with the knowledge base node. The general steps are: import documents from the data source → use extractor to extract document content → split and clean content into structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The user input variables required by the Knowledge Pipeline node must be predefined and managed via the Input Field section located in the top-right corner of the orchestration canvas. It determines what input fields the end users will see and need to fill in when importing files to the knowledge base through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique Inputs: Input fields defined here are only available to the selected data source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global Inputs: These input fields are shared across all subsequent nodes after the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 1182 + }, + "height": 260, + "id": "1751266376760", + "position": { + "x": -580.684520226929, + "y": -21.891401375096322 + }, + "positionAbsolute": { + "x": -580.684520226929, + "y": -21.891401375096322 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 1182 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 417, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A document extractor in Retrieval-Augmented Generation (RAG) is a tool or component that automatically identifies, extracts, and structures text and data from various types of documents—such as PDFs, images, scanned files, handwritten notes, and more—into a format that can be effectively used by language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Markitdown\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" serves as an excellent alternative to traditional document extraction nodes, offering robust file conversion capabilities within the Dify ecosystem. It leverages MarkItDown's plugin-based architecture to provide seamless conversion of multiple file formats to Markdown.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 241 + }, + "height": 417, + "id": "1751266402561", + "position": { + "x": -266.96080929383595, + "y": 372.64040589639495 + }, + "positionAbsolute": { + "x": -266.96080929383595, + "y": 372.64040589639495 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 241 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 554, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" addresses the dilemma of context and precision by leveraging a two-tier hierarchical approach that effectively balances the trade-off between accurate matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Query Matching with Child Chunks: Small, focused pieces of information, often as concise as a single sentence within a paragraph, are used to match the user's query. These child chunks enable precise and relevant initial retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Contextual Enrichment with Parent Chunks: Larger, encompassing sections—such as a paragraph, a section, or even an entire document—that include the matched child chunks are then retrieved. These parent chunks provide comprehensive context for the Language Model (LLM).\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 554, + "id": "1751266447821", + "position": { + "x": 37.74090119950054, + "y": 372.64040589639495 + }, + "positionAbsolute": { + "x": 37.74090119950054, + "y": 372.64040589639495 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 411, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", each with different retrieval strategies. High-Quality mode uses embeddings for vectorization and supports vector, full-text, and hybrid retrieval, offering more accurate results but higher resource usage. Economical mode uses keyword-based inverted indexing with no token consumption but lower accuracy; upgrading to High-Quality is possible, but downgrading requires creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"* Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 411, + "id": "1751266580099", + "position": { + "x": 357.7591396590142, + "y": 434.3959856026883 + }, + "positionAbsolute": { + "x": 357.7591396590142, + "y": 434.3959856026883 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "result": { + "description": "Parent child chunks result", + "items": { + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "", + "ja_JP": "", + "pt_BR": "", + "zh_Hans": "" + }, + "label": { + "en_US": "Input Content", + "ja_JP": "Input Content", + "pt_BR": "Conteúdo de Entrada", + "zh_Hans": "输入文本" + }, + "llm_description": "The text you want to chunk.", + "max": null, + "min": null, + "name": "input_text", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": "paragraph", + "form": "llm", + "human_description": { + "en_US": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "ja_JP": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "pt_BR": "Dividir texto em parágrafos com base no separador e no comprimento máximo do bloco, usando o texto dividido como bloco pai ou documento completo como bloco pai e diretamente recuperá-lo.", + "zh_Hans": "根据分隔符和最大块长度将文本拆分为段落,使用拆分文本作为检索的父块或整个文档用作父块并直接检索。" + }, + "label": { + "en_US": "Parent Mode", + "ja_JP": "Parent Mode", + "pt_BR": "Modo Pai", + "zh_Hans": "父块模式" + }, + "llm_description": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "max": null, + "min": null, + "name": "parent_mode", + "options": [ + { + "label": { + "en_US": "Paragraph", + "ja_JP": "Paragraph", + "pt_BR": "Parágrafo", + "zh_Hans": "段落" + }, + "value": "paragraph" + }, + { + "label": { + "en_US": "Full Document", + "ja_JP": "Full Document", + "pt_BR": "Documento Completo", + "zh_Hans": "全文" + }, + "value": "full_doc" + } + ], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": "\n\n", + "form": "llm", + "human_description": { + "en_US": "Separator used for chunking", + "ja_JP": "Separator used for chunking", + "pt_BR": "Separador usado para divisão", + "zh_Hans": "用于分块的分隔符" + }, + "label": { + "en_US": "Parent Delimiter", + "ja_JP": "Parent Delimiter", + "pt_BR": "Separador de Pai", + "zh_Hans": "父块分隔符" + }, + "llm_description": "The separator used to split chunks", + "max": null, + "min": null, + "name": "separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 1024, + "form": "llm", + "human_description": { + "en_US": "Maximum length for chunking", + "ja_JP": "Maximum length for chunking", + "pt_BR": "Comprimento máximo para divisão", + "zh_Hans": "用于分块的最大长度" + }, + "label": { + "en_US": "Maximum Parent Chunk Length", + "ja_JP": "Maximum Parent Chunk Length", + "pt_BR": "Comprimento Máximo do Bloco Pai", + "zh_Hans": "最大父块长度" + }, + "llm_description": "Maximum length allowed per chunk", + "max": null, + "min": null, + "name": "max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": ". ", + "form": "llm", + "human_description": { + "en_US": "Separator used for subchunking", + "ja_JP": "Separator used for subchunking", + "pt_BR": "Separador usado para subdivisão", + "zh_Hans": "用于子分块的分隔符" + }, + "label": { + "en_US": "Child Delimiter", + "ja_JP": "Child Delimiter", + "pt_BR": "Separador de Subdivisão", + "zh_Hans": "子分块分隔符" + }, + "llm_description": "The separator used to split subchunks", + "max": null, + "min": null, + "name": "subchunk_separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 512, + "form": "llm", + "human_description": { + "en_US": "Maximum length for subchunking", + "ja_JP": "Maximum length for subchunking", + "pt_BR": "Comprimento máximo para subdivisão", + "zh_Hans": "用于子分块的最大长度" + }, + "label": { + "en_US": "Maximum Child Chunk Length", + "ja_JP": "Maximum Child Chunk Length", + "pt_BR": "Comprimento Máximo de Subdivisão", + "zh_Hans": "子分块最大长度" + }, + "llm_description": "Maximum length allowed per subchunk", + "max": null, + "min": null, + "name": "subchunk_max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove consecutive spaces, newlines and tabs", + "ja_JP": "Whether to remove consecutive spaces, newlines and tabs", + "pt_BR": "Se deve remover espaços extras no texto", + "zh_Hans": "是否移除文本中的连续空格、换行符和制表符" + }, + "label": { + "en_US": "Replace consecutive spaces, newlines and tabs", + "ja_JP": "Replace consecutive spaces, newlines and tabs", + "pt_BR": "Substituir espaços consecutivos, novas linhas e guias", + "zh_Hans": "替换连续空格、换行符和制表符" + }, + "llm_description": "Whether to remove consecutive spaces, newlines and tabs", + "max": null, + "min": null, + "name": "remove_extra_spaces", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove URLs and emails in the text", + "ja_JP": "Whether to remove URLs and emails in the text", + "pt_BR": "Se deve remover URLs e e-mails no texto", + "zh_Hans": "是否移除文本中的URL和电子邮件地址" + }, + "label": { + "en_US": "Delete all URLs and email addresses", + "ja_JP": "Delete all URLs and email addresses", + "pt_BR": "Remover todas as URLs e e-mails", + "zh_Hans": "删除所有URL和电子邮件地址" + }, + "llm_description": "Whether to remove URLs and emails in the text", + "max": null, + "min": null, + "name": "remove_urls_emails", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + } + ], + "params": { + "input_text": "", + "max_length": "", + "parent_mode": "", + "remove_extra_spaces": "", + "remove_urls_emails": "", + "separator": "", + "subchunk_max_length": "", + "subchunk_separator": "" + }, + "provider_id": "langgenius/parentchild_chunker/parentchild_chunker", + "provider_name": "langgenius/parentchild_chunker/parentchild_chunker", + "provider_type": "builtin", + "selected": false, + "title": "Parent-child Chunker", + "tool_configurations": {}, + "tool_description": "Process documents into parent-child chunk structures", + "tool_label": "Parent-child Chunker", + "tool_name": "parentchild_chunker", + "tool_node_version": "2", + "tool_parameters": { + "input_text": { + "type": "mixed", + "value": "{{#1751359716720.text#}}" + }, + "max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Maximum_Parent_Length" + ] + }, + "parent_mode": { + "type": "variable", + "value": [ + "rag", + "shared", + "Parent_Mode" + ] + }, + "separator": { + "type": "mixed", + "value": "{{#rag.shared.Parent_Delimiter#}}" + }, + "subchunk_max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Maximum_Child_Length" + ] + }, + "subchunk_separator": { + "type": "mixed", + "value": "{{#rag.shared.Child_Delimiter#}}" + } + }, + "type": "tool" + }, + "height": 52, + "id": "1751336942081", + "position": { + "x": 37.74090119950054, + "y": 282 + }, + "positionAbsolute": { + "x": 37.74090119950054, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": null, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "Upload files for processing", + "ja_JP": "Upload files for processing", + "pt_BR": "Carregar arquivos para processamento", + "zh_Hans": "上传文件进行处理" + }, + "label": { + "en_US": "Files", + "ja_JP": "Files", + "pt_BR": "Arquivos", + "zh_Hans": "文件" + }, + "llm_description": "", + "max": null, + "min": null, + "name": "files", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "files" + } + ], + "params": { + "files": "" + }, + "provider_id": "yevanchen/markitdown/markitdown", + "provider_name": "yevanchen/markitdown/markitdown", + "provider_type": "builtin", + "selected": false, + "title": "markitdown", + "tool_configurations": {}, + "tool_description": "Python tool for converting files and office documents to Markdown.", + "tool_label": "markitdown", + "tool_name": "markitdown", + "tool_node_version": "2", + "tool_parameters": { + "files": { + "type": "variable", + "value": [ + "1750400203722", + "file" + ] + } + }, + "type": "tool" + }, + "height": 52, + "id": "1751359716720", + "position": { + "x": -266.96080929383595, + "y": 282 + }, + "positionAbsolute": { + "x": -266.96080929383595, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 301, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"MarkItDown\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is recommended for converting and handling a wide range of file formats, particularly for transforming content into Markdown. It works especially well for converting native Office files—such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"DOCX\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"XLSX\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"PPTX\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"—into Markdown to facilitate better information processing. However, as some users have noted its suboptimal performance in extracting content from PDF files, using it for PDFs is not recommended.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 301, + "id": "1753425718313", + "position": { + "x": -580.684520226929, + "y": 372.64040589639495 + }, + "positionAbsolute": { + "x": -580.684520226929, + "y": 372.64040589639495 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + } + ], + "viewport": { + "x": 747.6785299994758, + "y": 94.6209873206409, + "zoom": 0.8152773235379324 + } + }, + "icon_info": { + "icon": "9d658c3a-b22f-487d-8223-db51e9012505", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "id": "982d1788-837a-40c8-b7de-d37b09a9b2bc", + "name": "Convert to Markdown", + "icon": { + "icon": "9d658c3a-b22f-487d-8223-db51e9012505", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "language": "zh-Hans", + "position": 4 + }, + "98374ab6-9dcd-434d-983e-268bec156b43": { + "chunk_structure": "qa_model", + "description": "This template is designed to use LLM to extract key information from the input document and generate Q&A pairs indexed by questions, enabling efficient retrieval of relevant answers based on query similarity.", + "export_data": "dependencies:\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/dify_extractor:0.0.5@ba7e2fd9165eda73bfcc68e31a108855197e88706e5556c058e0777ab08409b3\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/notion_datasource:0.1.12@2855c4a7cffd3311118ebe70f095e546f99935e47f12c841123146f728534f55\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/jina_datasource:0.0.5@75942f5bbde870ad28e0345ff5ebf54ebd3aec63f0e66344ef76b88cf06b85c3\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/google_drive:0.1.6@4bc0cf8f8979ebd7321b91506b4bc8f090b05b769b5d214f2da4ce4c04ce30bd\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/jina:0.0.8@d3a6766fbb80890d73fea7ea04803f3e1702c6e6bd621aafb492b86222a193dd\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/qa_chunk:0.0.8@1fed9644646bdd48792cdf5a1d559a3df336bd3a8edb0807227499fb56dce3af\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: bowenliang123/md_exporter:2.0.0@13e1aca1995328e41c080ff9f7f6d898df60ff74a3f4d98d6de4b18ab5b92c2e\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/firecrawl_datasource:0.2.4@37b490ebc52ac30d1c6cbfa538edcddddcfed7d5f5de58982edbd4e2094eb6e2\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius/anthropic:0.2.0@a776815b091c81662b2b54295ef4b8a54b5533c2ec1c66c7c8f2feea724f3248\nkind: rag_pipeline\nrag_pipeline:\n description: ''\n icon: 2b887f89-b6c9-4288-be43-635fee45216b\n icon_background: '#FFEAD5'\n icon_type: image\n icon_url: \n name: LLM Generated Q&A\nversion: 0.1.0\nworkflow:\n conversation_variables: []\n environment_variables: []\n features: {}\n graph:\n edges:\n - data:\n isInLoop: false\n sourceType: tool\n targetType: variable-aggregator\n id: 1750836391776-source-1753346901505-target\n selected: false\n source: '1750836391776'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: document-extractor\n targetType: variable-aggregator\n id: 1753349228522-source-1753346901505-target\n selected: false\n source: '1753349228522'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1754023419266-source-1753346901505-target\n selected: false\n source: '1754023419266'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756442998557-source-1756442986174-target\n selected: false\n source: '1756442998557'\n sourceHandle: source\n target: '1756442986174'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: variable-aggregator\n targetType: if-else\n id: 1756442986174-source-1756443014860-target\n selected: false\n source: '1756442986174'\n sourceHandle: source\n target: '1756443014860'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1750836380067-source-1756442986174-target\n selected: false\n source: '1750836380067'\n sourceHandle: source\n target: '1756442986174'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: if-else\n targetType: tool\n id: 1756443014860-true-1750836391776-target\n selected: false\n source: '1756443014860'\n sourceHandle: 'true'\n target: '1750836391776'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: if-else\n targetType: document-extractor\n id: 1756443014860-false-1753349228522-target\n selected: false\n source: '1756443014860'\n sourceHandle: 'false'\n target: '1753349228522'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756896212061-source-1753346901505-target\n source: '1756896212061'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: variable-aggregator\n id: 1756907397615-source-1753346901505-target\n source: '1756907397615'\n sourceHandle: source\n target: '1753346901505'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: variable-aggregator\n targetType: llm\n id: 1753346901505-source-1756912504019-target\n source: '1753346901505'\n sourceHandle: source\n target: '1756912504019'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: llm\n targetType: tool\n id: 1756912504019-source-1756912537172-target\n source: '1756912504019'\n sourceHandle: source\n target: '1756912537172'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: tool\n id: 1756912537172-source-1756912274158-target\n source: '1756912537172'\n sourceHandle: source\n target: '1756912274158'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: knowledge-index\n id: 1756912274158-source-1750836372241-target\n source: '1756912274158'\n sourceHandle: source\n target: '1750836372241'\n targetHandle: target\n type: custom\n zIndex: 0\n nodes:\n - data:\n chunk_structure: qa_model\n embedding_model: jina-embeddings-v2-base-en\n embedding_model_provider: langgenius/jina/jina\n index_chunk_variable_selector:\n - '1756912274158'\n - result\n indexing_technique: high_quality\n keyword_number: 10\n retrieval_model:\n hybridSearchMode: weighted_score\n reranking_enable: false\n score_threshold: 0.5\n score_threshold_enabled: false\n search_method: semantic_search\n top_k: 3\n vector_setting:\n embedding_model_name: jina-embeddings-v2-base-en\n embedding_provider_name: langgenius/jina/jina\n selected: false\n title: Knowledge Base\n type: knowledge-index\n height: 114\n id: '1750836372241'\n position:\n x: 1150.8369138826617\n y: 326\n positionAbsolute:\n x: 1150.8369138826617\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: File\n datasource_name: upload-file\n datasource_parameters: {}\n fileExtensions:\n - txt\n - markdown\n - mdx\n - pdf\n - html\n - xlsx\n - xls\n - vtt\n - properties\n - doc\n - docx\n - csv\n - eml\n - msg\n - pptx\n - xml\n - epub\n - ppt\n - md\n plugin_id: langgenius/file\n provider_name: file\n provider_type: local_file\n selected: false\n title: File\n type: datasource\n height: 52\n id: '1750836380067'\n position:\n x: -1371.6520723158733\n y: 224.87938381325645\n positionAbsolute:\n x: -1371.6520723158733\n y: 224.87938381325645\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n documents:\n description: the documents extracted from the file\n items:\n type: object\n type: array\n images:\n description: The images extracted from the file\n items:\n type: object\n type: array\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n ja_JP: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n pt_BR: o arquivo a ser analisado (suporta pdf, ppt, pptx, doc, docx, png,\n jpg, jpeg)\n zh_Hans: 用于解析的文件(支持 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)\n label:\n en_US: file\n ja_JP: file\n pt_BR: file\n zh_Hans: file\n llm_description: the file to be parsed (support pdf, ppt, pptx, doc, docx,\n png, jpg, jpeg)\n max: null\n min: null\n name: file\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: file\n params:\n file: ''\n provider_id: langgenius/dify_extractor/dify_extractor\n provider_name: langgenius/dify_extractor/dify_extractor\n provider_type: builtin\n selected: false\n title: Dify Extractor\n tool_configurations: {}\n tool_description: Dify Extractor\n tool_label: Dify Extractor\n tool_name: dify_extractor\n tool_node_version: '2'\n tool_parameters:\n file:\n type: variable\n value:\n - '1756442986174'\n - output\n type: tool\n height: 52\n id: '1750836391776'\n position:\n x: -417.5334221022782\n y: 268.1692071834485\n positionAbsolute:\n x: -417.5334221022782\n y: 268.1692071834485\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 252\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge\n Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n starts with Data Source as the starting node and ends with the knowledge\n base node. The general steps are: import documents from the data source\n → use extractor to extract document content → split and clean content into\n structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n user input variables required by the Knowledge Pipeline node must be predefined\n and managed via the Input Field section located in the top-right corner\n of the orchestration canvas. It determines what input fields the end users\n will see and need to fill in when importing files to the knowledge base\n through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique\n Inputs: Input fields defined here are only available to the selected data\n source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global\n Inputs: These input fields are shared across all subsequent nodes after\n the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For\n more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 1124\n height: 252\n id: '1751252161631'\n position:\n x: -1371.6520723158733\n y: -123.758428116601\n positionAbsolute:\n x: -1371.6520723158733\n y: -123.758428116601\n selected: true\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 1124\n - data:\n author: TenTen\n desc: ''\n height: 388\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently\n we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data\n Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\":\n File Upload, Online Drive, Online Doc, and Web Crawler. Different types\n of Data Sources have different input and output types. The output of File\n Upload and Online Drive are files, while the output of Online Doc and WebCrawler\n are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n Knowledge Pipeline can have multiple data sources. Each data source can\n be selected more than once with different settings. Each added data source\n is a tab on the add file interface. However, each time the user can only\n select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 285\n height: 388\n id: '1751252440357'\n position:\n x: -1723.9942193415582\n y: 224.87938381325645\n positionAbsolute:\n x: -1723.9942193415582\n y: 224.87938381325645\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 285\n - data:\n author: TenTen\n desc: ''\n height: 430\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n document extractor in Retrieval-Augmented Generation (RAG) is a tool or\n component that automatically identifies, extracts, and structures text and\n data from various types of documents—such as PDFs, images, scanned files,\n handwritten notes, and more—into a format that can be effectively used by\n language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Dify\n Extractor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is\n a built-in document parser developed by Dify. It supports a wide range of\n common file formats and offers specialized handling for certain formats,\n such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\".docx\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\n In addition to text extraction, it can extract images embedded within documents,\n store them, and return their accessible URLs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 430\n id: '1751253091602'\n position:\n x: -417.5334221022782\n y: 546.5283142529594\n positionAbsolute:\n x: -417.5334221022782\n y: 546.5283142529594\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 336\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Processor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" extracts\n specified columns from tables to generate structured Q&A pairs. Users can\n independently designate which columns to use for questions and which for\n answers.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"These\n pairs are indexed by the question field, so user queries are matched directly\n against the questions to retrieve the corresponding answers. This \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q-to-Q\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" matching\n strategy improves clarity and precision, especially in scenarios involving\n high-frequency or highly similar user questions.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 336\n id: '1751253953926'\n position:\n x: 794.2003154321724\n y: 417.25474169825833\n positionAbsolute:\n x: 794.2003154321724\n y: 417.25474169825833\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 410\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n each with different retrieval strategies. High-Quality mode uses embeddings\n for vectorization and supports vector, full-text, and hybrid retrieval,\n offering more accurate results but higher resource usage. Economical mode\n uses keyword-based inverted indexing with no token consumption but lower\n accuracy; upgrading to High-Quality is possible, but downgrading requires\n creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"*\n Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only\n support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing\n method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 410\n id: '1751254117904'\n position:\n x: 1150.8369138826617\n y: 475.88970282568215\n positionAbsolute:\n x: 1150.8369138826617\n y: 475.88970282568215\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n output_type: string\n selected: false\n title: Variable Aggregator\n type: variable-aggregator\n variables:\n - - '1750836391776'\n - text\n - - '1753349228522'\n - text\n - - '1754023419266'\n - content\n - - '1756896212061'\n - content\n height: 187\n id: '1753346901505'\n position:\n x: -117.24452412456148\n y: 326\n positionAbsolute:\n x: -117.24452412456148\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_array_file: false\n selected: false\n title: Doc Extractor\n type: document-extractor\n variable_selector:\n - '1756442986174'\n - output\n height: 92\n id: '1753349228522'\n position:\n x: -417.5334221022782\n y: 417.25474169825833\n positionAbsolute:\n x: -417.5334221022782\n y: 417.25474169825833\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Notion\n datasource_name: notion_datasource\n datasource_parameters: {}\n plugin_id: langgenius/notion_datasource\n provider_name: notion_datasource\n provider_type: online_document\n selected: false\n title: Notion\n type: datasource\n height: 52\n id: '1754023419266'\n position:\n x: -1369.6904698303242\n y: 440.01452302398053\n positionAbsolute:\n x: -1369.6904698303242\n y: 440.01452302398053\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n output_type: file\n selected: false\n title: Variable Aggregator\n type: variable-aggregator\n variables:\n - - '1750836380067'\n - file\n - - '1756442998557'\n - file\n height: 135\n id: '1756442986174'\n position:\n x: -1067.06980963949\n y: 236.10252072775984\n positionAbsolute:\n x: -1067.06980963949\n y: 236.10252072775984\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Google Drive\n datasource_name: google_drive\n datasource_parameters: {}\n plugin_id: langgenius/google_drive\n provider_name: google_drive\n provider_type: online_drive\n selected: false\n title: Google Drive\n type: datasource\n height: 52\n id: '1756442998557'\n position:\n x: -1371.6520723158733\n y: 326\n positionAbsolute:\n x: -1371.6520723158733\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n cases:\n - case_id: 'true'\n conditions:\n - comparison_operator: is\n id: 1581dd11-7898-41f4-962f-937283ba7e01\n value: .xlsx\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 92abb46d-d7e4-46e7-a5e1-8a29bb45d528\n value: .xls\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 1dde5ae7-754d-4e83-96b2-fe1f02995d8b\n value: .md\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 7e1a80e5-c32a-46a4-8f92-8912c64972aa\n value: .markdown\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 53abfe95-c7d0-4f63-ad37-17d425d25106\n value: .mdx\n varType: string\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 436877b8-8c0a-4cc6-9565-92754db08571\n value: .html\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 5e3e375e-750b-4204-8ac3-9a1174a5ab7c\n value: .htm\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 1a84a784-a797-4f96-98a0-33a9b48ceb2b\n value: .docx\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 62d11445-876a-493f-85d3-8fc020146bdd\n value: .csv\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n - comparison_operator: is\n id: 02c4bce8-7668-4ccd-b750-4281f314b231\n value: .txt\n varType: file\n variable_selector:\n - '1756442986174'\n - output\n - extension\n id: 'true'\n logical_operator: or\n selected: false\n title: IF/ELSE\n type: if-else\n height: 358\n id: '1756443014860'\n position:\n x: -733.5977815139424\n y: 236.10252072775984\n positionAbsolute:\n x: -733.5977815139424\n y: 236.10252072775984\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Jina Reader\n datasource_name: jina_reader\n datasource_parameters:\n crawl_sub_pages:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jina_subpages\n limit:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jina_limit\n url:\n type: mixed\n value: '{{#rag.1756896212061.jina_url#}}'\n use_sitemap:\n type: variable\n value:\n - rag\n - '1756896212061'\n - jian_sitemap\n plugin_id: langgenius/jina_datasource\n provider_name: jinareader\n provider_type: website_crawl\n selected: false\n title: Jina Reader\n type: datasource\n height: 52\n id: '1756896212061'\n position:\n x: -1371.6520723158733\n y: 538.9988445953813\n positionAbsolute:\n x: -1371.6520723158733\n y: 538.9988445953813\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: Firecrawl\n datasource_name: crawl\n datasource_parameters:\n crawl_subpages:\n type: variable\n value:\n - rag\n - '1756907397615'\n - firecrawl_subpages\n exclude_paths:\n type: mixed\n value: '{{#rag.1756907397615.exclude_paths#}}'\n include_paths:\n type: mixed\n value: '{{#rag.1756907397615.include_paths#}}'\n limit:\n type: variable\n value:\n - rag\n - '1756907397615'\n - max_pages\n max_depth:\n type: variable\n value:\n - rag\n - '1756907397615'\n - max_depth\n only_main_content:\n type: variable\n value:\n - rag\n - '1756907397615'\n - main_content\n url:\n type: mixed\n value: '{{#rag.1756907397615.firecrawl_url1#}}'\n plugin_id: langgenius/firecrawl_datasource\n provider_name: firecrawl\n provider_type: website_crawl\n selected: false\n title: Firecrawl\n type: datasource\n height: 52\n id: '1756907397615'\n position:\n x: -1371.6520723158733\n y: 644.3296146102903\n positionAbsolute:\n x: -1371.6520723158733\n y: 644.3296146102903\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The file you want to extract QA from.\n ja_JP: The file you want to extract QA from.\n pt_BR: The file you want to extract QA from.\n zh_Hans: 你想要提取 QA 的文件。\n label:\n en_US: Input File\n ja_JP: Input File\n pt_BR: Input File\n zh_Hans: 输入文件\n llm_description: The file you want to extract QA from.\n max: null\n min: null\n name: input_file\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: file\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Column number for question.\n ja_JP: Column number for question.\n pt_BR: Column number for question.\n zh_Hans: 问题所在的列。\n label:\n en_US: Column number for question\n ja_JP: Column number for question\n pt_BR: Column number for question\n zh_Hans: 问题所在的列\n llm_description: The column number for question, the format of the column\n number must be an integer.\n max: null\n min: null\n name: question_column\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: 1\n form: llm\n human_description:\n en_US: Column number for answer.\n ja_JP: Column number for answer.\n pt_BR: Column number for answer.\n zh_Hans: 答案所在的列。\n label:\n en_US: Column number for answer\n ja_JP: Column number for answer\n pt_BR: Column number for answer\n zh_Hans: 答案所在的列\n llm_description: The column number for answer, the format of the column\n number must be an integer.\n max: null\n min: null\n name: answer_column\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: number\n params:\n answer_column: ''\n input_file: ''\n question_column: ''\n provider_id: langgenius/qa_chunk/qa_chunk\n provider_name: langgenius/qa_chunk/qa_chunk\n provider_type: builtin\n selected: false\n title: Q&A Processor\n tool_configurations: {}\n tool_description: A tool for QA chunking mode.\n tool_label: QA Chunk\n tool_name: qa_chunk\n tool_node_version: '2'\n tool_parameters:\n answer_column:\n type: constant\n value: 2\n input_file:\n type: variable\n value:\n - '1756912537172'\n - files\n question_column:\n type: constant\n value: 1\n type: tool\n height: 52\n id: '1756912274158'\n position:\n x: 794.2003154321724\n y: 326\n positionAbsolute:\n x: 794.2003154321724\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n context:\n enabled: false\n variable_selector: []\n model:\n completion_params:\n temperature: 0.7\n mode: chat\n name: claude-3-5-sonnet-20240620\n provider: langgenius/anthropic/anthropic\n prompt_template:\n - id: 7f8105aa-a37d-4f5a-b581-babeeb31e833\n role: system\n text: '\n\n Generate a list of Q&A pairs based on {{#1753346901505.output#}}. Present\n the output as a Markdown table, where the first column is serial number,\n the second column is Question, and the third column is Question. Ensure\n that the table format can be easily converted into a CSV file.\n\n Example Output Format:\n\n | Index | Question | Answer |\n\n |-------|-----------|--------|\n\n | 1 | What is the main purpose of the document? | The document explains\n the company''s new product launch strategy. ![image](https://cloud.dify.ai/files/xxxxxxx)\n |\n\n | 2 || When will the product be launched? | The product will be launched\n in Q3 of this year. |\n\n\n Instructions:\n\n Read and understand the input text.\n\n Extract key information and generate meaningful questions and answers.\n\n Preserve any ![image] URLs from the input text in the answers.\n\n Keep questions concise and specific.\n\n Ensure answers are accurate, self-contained, and clear.\n\n Output only the Markdown table without any extra explanation.'\n selected: false\n title: LLM\n type: llm\n vision:\n enabled: false\n height: 88\n id: '1756912504019'\n position:\n x: 184.46657789772178\n y: 326\n positionAbsolute:\n x: 184.46657789772178\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: Markdown text\n ja_JP: Markdown text\n pt_BR: Markdown text\n zh_Hans: Markdown格式文本,必须为Markdown表格格式\n label:\n en_US: Markdown text\n ja_JP: Markdown text\n pt_BR: Markdown text\n zh_Hans: Markdown格式文本\n llm_description: ''\n max: null\n min: null\n name: md_text\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: Filename of the output file\n ja_JP: Filename of the output file\n pt_BR: Filename of the output file\n zh_Hans: 输出文件名\n label:\n en_US: Filename of the output file\n ja_JP: Filename of the output file\n pt_BR: Filename of the output file\n zh_Hans: 输出文件名\n llm_description: ''\n max: null\n min: null\n name: output_filename\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n params:\n md_text: ''\n output_filename: ''\n provider_id: bowenliang123/md_exporter/md_exporter\n provider_name: bowenliang123/md_exporter/md_exporter\n provider_type: builtin\n selected: false\n title: Markdown to CSV file\n tool_configurations: {}\n tool_description: Generate CSV file from Markdown text\n tool_label: Markdown to CSV file\n tool_name: md_to_csv\n tool_node_version: '2'\n tool_parameters:\n md_text:\n type: mixed\n value: '{{#1756912504019.text#}}'\n output_filename:\n type: mixed\n value: LLM Generated Q&A\n type: tool\n height: 52\n id: '1756912537172'\n position:\n x: 484.75465419110174\n y: 326\n positionAbsolute:\n x: 484.75465419110174\n y: 326\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 174\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n LLM-generated Q&A pairs are designed to extract key information from the\n input text and present it in a structured, easy-to-use format. Each pair\n consists of a concise question that captures an important point or detail,\n and a clear, self-contained answer that provides the relevant information\n without requiring additional context. The output is formatted as a Markdown\n table with three columns—Index, Question, and Answer—so that it can be easily\n converted into a CSV file for further processing. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 528\n height: 174\n id: '1756912556940'\n position:\n x: 184.46657789772178\n y: 462.64405262857747\n positionAbsolute:\n x: 184.46657789772178\n y: 462.64405262857747\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 528\n viewport:\n x: 1149.1394490177502\n y: 317.2338302699771\n zoom: 0.4911032886685182\n rag_pipeline_variables:\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: jina_reader_url\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: pages\n variable: jina_reader_imit\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: true\n label: Crawl sub-pages\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: checkbox\n unit: null\n variable: Crawl_sub_pages_2\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1753688365254'\n default_value: true\n label: Use sitemap\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: Use_sitemap\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: jina_url\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: pages\n variable: jina_limit\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: true\n label: Use sitemap\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: Follow the sitemap to crawl the site. If not, Jina Reader will crawl\n iteratively based on page relevance, yielding fewer but higher-quality pages.\n type: checkbox\n unit: null\n variable: jian_sitemap\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756896212061'\n default_value: true\n label: Crawl subpages\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: jina_subpages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: URL\n max_length: 256\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: text-input\n unit: null\n variable: firecrawl_url1\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: true\n label: firecrawl_subpages\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: firecrawl_subpages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: Exclude paths\n max_length: 256\n options: []\n placeholder: blog/*,/about/*\n required: false\n tooltips: null\n type: text-input\n unit: null\n variable: exclude_paths\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: include_paths\n max_length: 256\n options: []\n placeholder: articles/*\n required: false\n tooltips: null\n type: text-input\n unit: null\n variable: include_paths\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: 0\n label: Max depth\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: Maximum depth to crawl relative to the entered URL. Depth 0 just scrapes\n the page of the entered url, depth 1 scrapes the url and everything after enteredURL\n + one /, and so on.\n type: number\n unit: null\n variable: max_depth\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: 10\n label: Limit\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: null\n variable: max_pages\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: true\n label: Extract only main content (no headers, navs, footers, etc.)\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: main_content\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: '1756907397615'\n default_value: null\n label: depthtest\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: null\n variable: depthtest\n", + "graph": { + "edges": [ + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "variable-aggregator" + }, + "id": "1750836391776-source-1753346901505-target", + "selected": false, + "source": "1750836391776", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "document-extractor", + "targetType": "variable-aggregator" + }, + "id": "1753349228522-source-1753346901505-target", + "selected": false, + "source": "1753349228522", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1754023419266-source-1753346901505-target", + "selected": false, + "source": "1754023419266", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756442998557-source-1756442986174-target", + "selected": false, + "source": "1756442998557", + "sourceHandle": "source", + "target": "1756442986174", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "variable-aggregator", + "targetType": "if-else" + }, + "id": "1756442986174-source-1756443014860-target", + "selected": false, + "source": "1756442986174", + "sourceHandle": "source", + "target": "1756443014860", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1750836380067-source-1756442986174-target", + "selected": false, + "source": "1750836380067", + "sourceHandle": "source", + "target": "1756442986174", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "if-else", + "targetType": "tool" + }, + "id": "1756443014860-true-1750836391776-target", + "selected": false, + "source": "1756443014860", + "sourceHandle": "true", + "target": "1750836391776", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "if-else", + "targetType": "document-extractor" + }, + "id": "1756443014860-false-1753349228522-target", + "selected": false, + "source": "1756443014860", + "sourceHandle": "false", + "target": "1753349228522", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756896212061-source-1753346901505-target", + "source": "1756896212061", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "variable-aggregator" + }, + "id": "1756907397615-source-1753346901505-target", + "source": "1756907397615", + "sourceHandle": "source", + "target": "1753346901505", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "variable-aggregator", + "targetType": "llm" + }, + "id": "1753346901505-source-1756912504019-target", + "source": "1753346901505", + "sourceHandle": "source", + "target": "1756912504019", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "llm", + "targetType": "tool" + }, + "id": "1756912504019-source-1756912537172-target", + "source": "1756912504019", + "sourceHandle": "source", + "target": "1756912537172", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "tool" + }, + "id": "1756912537172-source-1756912274158-target", + "source": "1756912537172", + "sourceHandle": "source", + "target": "1756912274158", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "knowledge-index" + }, + "id": "1756912274158-source-1750836372241-target", + "source": "1756912274158", + "sourceHandle": "source", + "target": "1750836372241", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + } + ], + "nodes": [ + { + "data": { + "chunk_structure": "qa_model", + "embedding_model": "jina-embeddings-v2-base-en", + "embedding_model_provider": "langgenius/jina/jina", + "index_chunk_variable_selector": [ + "1756912274158", + "result" + ], + "indexing_technique": "high_quality", + "keyword_number": 10, + "retrieval_model": { + "hybridSearchMode": "weighted_score", + "reranking_enable": false, + "score_threshold": 0.5, + "score_threshold_enabled": false, + "search_method": "semantic_search", + "top_k": 3, + "vector_setting": { + "embedding_model_name": "jina-embeddings-v2-base-en", + "embedding_provider_name": "langgenius/jina/jina" + } + }, + "selected": false, + "title": "Knowledge Base", + "type": "knowledge-index" + }, + "height": 114, + "id": "1750836372241", + "position": { + "x": 1150.8369138826617, + "y": 326 + }, + "positionAbsolute": { + "x": 1150.8369138826617, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "File", + "datasource_name": "upload-file", + "datasource_parameters": {}, + "fileExtensions": [ + "txt", + "markdown", + "mdx", + "pdf", + "html", + "xlsx", + "xls", + "vtt", + "properties", + "doc", + "docx", + "csv", + "eml", + "msg", + "pptx", + "xml", + "epub", + "ppt", + "md" + ], + "plugin_id": "langgenius/file", + "provider_name": "file", + "provider_type": "local_file", + "selected": false, + "title": "File", + "type": "datasource" + }, + "height": 52, + "id": "1750836380067", + "position": { + "x": -1371.6520723158733, + "y": 224.87938381325645 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 224.87938381325645 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "documents": { + "description": "the documents extracted from the file", + "items": { + "type": "object" + }, + "type": "array" + }, + "images": { + "description": "The images extracted from the file", + "items": { + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "ja_JP": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "pt_BR": "o arquivo a ser analisado (suporta pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "zh_Hans": "用于解析的文件(支持 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)" + }, + "label": { + "en_US": "file", + "ja_JP": "file", + "pt_BR": "file", + "zh_Hans": "file" + }, + "llm_description": "the file to be parsed (support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "max": null, + "min": null, + "name": "file", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "file" + } + ], + "params": { + "file": "" + }, + "provider_id": "langgenius/dify_extractor/dify_extractor", + "provider_name": "langgenius/dify_extractor/dify_extractor", + "provider_type": "builtin", + "selected": false, + "title": "Dify Extractor", + "tool_configurations": {}, + "tool_description": "Dify Extractor", + "tool_label": "Dify Extractor", + "tool_name": "dify_extractor", + "tool_node_version": "2", + "tool_parameters": { + "file": { + "type": "variable", + "value": [ + "1756442986174", + "output" + ] + } + }, + "type": "tool" + }, + "height": 52, + "id": "1750836391776", + "position": { + "x": -417.5334221022782, + "y": 268.1692071834485 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 268.1692071834485 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 252, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" starts with Data Source as the starting node and ends with the knowledge base node. The general steps are: import documents from the data source → use extractor to extract document content → split and clean content into structured chunks → store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The user input variables required by the Knowledge Pipeline node must be predefined and managed via the Input Field section located in the top-right corner of the orchestration canvas. It determines what input fields the end users will see and need to fill in when importing files to the knowledge base through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique Inputs: Input fields defined here are only available to the selected data source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global Inputs: These input fields are shared across all subsequent nodes after the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https://docs.dify.ai/en/guides/knowledge-base/knowledge-pipeline/knowledge-pipeline-orchestration\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 1124 + }, + "height": 252, + "id": "1751252161631", + "position": { + "x": -1371.6520723158733, + "y": -123.758428116601 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": -123.758428116601 + }, + "selected": true, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 1124 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 388, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\": File Upload, Online Drive, Online Doc, and Web Crawler. Different types of Data Sources have different input and output types. The output of File Upload and Online Drive are files, while the output of Online Doc and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A Knowledge Pipeline can have multiple data sources. Each data source can be selected more than once with different settings. Each added data source is a tab on the add file interface. However, each time the user can only select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 285 + }, + "height": 388, + "id": "1751252440357", + "position": { + "x": -1723.9942193415582, + "y": 224.87938381325645 + }, + "positionAbsolute": { + "x": -1723.9942193415582, + "y": 224.87938381325645 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 285 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 430, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A document extractor in Retrieval-Augmented Generation (RAG) is a tool or component that automatically identifies, extracts, and structures text and data from various types of documents—such as PDFs, images, scanned files, handwritten notes, and more—into a format that can be effectively used by language models within RAG Pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Dify Extractor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is a built-in document parser developed by Dify. It supports a wide range of common file formats and offers specialized handling for certain formats, such as \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":16,\"mode\":\"normal\",\"style\":\"\",\"text\":\".docx\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\". In addition to text extraction, it can extract images embedded within documents, store them, and return their accessible URLs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 430, + "id": "1751253091602", + "position": { + "x": -417.5334221022782, + "y": 546.5283142529594 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 546.5283142529594 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 336, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Processor\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" extracts specified columns from tables to generate structured Q&A pairs. Users can independently designate which columns to use for questions and which for answers.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"These pairs are indexed by the question field, so user queries are matched directly against the questions to retrieve the corresponding answers. This \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q-to-Q\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" matching strategy improves clarity and precision, especially in scenarios involving high-frequency or highly similar user questions.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 336, + "id": "1751253953926", + "position": { + "x": 794.2003154321724, + "y": 417.25474169825833 + }, + "positionAbsolute": { + "x": 794.2003154321724, + "y": 417.25474169825833 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 410, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The knowledge base provides two indexing methods: \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", each with different retrieval strategies. High-Quality mode uses embeddings for vectorization and supports vector, full-text, and hybrid retrieval, offering more accurate results but higher resource usage. Economical mode uses keyword-based inverted indexing with no token consumption but lower accuracy; upgrading to High-Quality is possible, but downgrading requires creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"* Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" and \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" only support the \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" indexing method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 410, + "id": "1751254117904", + "position": { + "x": 1150.8369138826617, + "y": 475.88970282568215 + }, + "positionAbsolute": { + "x": 1150.8369138826617, + "y": 475.88970282568215 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "output_type": "string", + "selected": false, + "title": "Variable Aggregator", + "type": "variable-aggregator", + "variables": [ + [ + "1750836391776", + "text" + ], + [ + "1753349228522", + "text" + ], + [ + "1754023419266", + "content" + ], + [ + "1756896212061", + "content" + ] + ] + }, + "height": 187, + "id": "1753346901505", + "position": { + "x": -117.24452412456148, + "y": 326 + }, + "positionAbsolute": { + "x": -117.24452412456148, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_array_file": false, + "selected": false, + "title": "Doc Extractor", + "type": "document-extractor", + "variable_selector": [ + "1756442986174", + "output" + ] + }, + "height": 92, + "id": "1753349228522", + "position": { + "x": -417.5334221022782, + "y": 417.25474169825833 + }, + "positionAbsolute": { + "x": -417.5334221022782, + "y": 417.25474169825833 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Notion", + "datasource_name": "notion_datasource", + "datasource_parameters": {}, + "plugin_id": "langgenius/notion_datasource", + "provider_name": "notion_datasource", + "provider_type": "online_document", + "selected": false, + "title": "Notion", + "type": "datasource" + }, + "height": 52, + "id": "1754023419266", + "position": { + "x": -1369.6904698303242, + "y": 440.01452302398053 + }, + "positionAbsolute": { + "x": -1369.6904698303242, + "y": 440.01452302398053 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "output_type": "file", + "selected": false, + "title": "Variable Aggregator", + "type": "variable-aggregator", + "variables": [ + [ + "1750836380067", + "file" + ], + [ + "1756442998557", + "file" + ] + ] + }, + "height": 135, + "id": "1756442986174", + "position": { + "x": -1067.06980963949, + "y": 236.10252072775984 + }, + "positionAbsolute": { + "x": -1067.06980963949, + "y": 236.10252072775984 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Google Drive", + "datasource_name": "google_drive", + "datasource_parameters": {}, + "plugin_id": "langgenius/google_drive", + "provider_name": "google_drive", + "provider_type": "online_drive", + "selected": false, + "title": "Google Drive", + "type": "datasource" + }, + "height": 52, + "id": "1756442998557", + "position": { + "x": -1371.6520723158733, + "y": 326 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "cases": [ + { + "case_id": "true", + "conditions": [ + { + "comparison_operator": "is", + "id": "1581dd11-7898-41f4-962f-937283ba7e01", + "value": ".xlsx", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "92abb46d-d7e4-46e7-a5e1-8a29bb45d528", + "value": ".xls", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "1dde5ae7-754d-4e83-96b2-fe1f02995d8b", + "value": ".md", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "7e1a80e5-c32a-46a4-8f92-8912c64972aa", + "value": ".markdown", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "53abfe95-c7d0-4f63-ad37-17d425d25106", + "value": ".mdx", + "varType": "string", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "436877b8-8c0a-4cc6-9565-92754db08571", + "value": ".html", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "5e3e375e-750b-4204-8ac3-9a1174a5ab7c", + "value": ".htm", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "1a84a784-a797-4f96-98a0-33a9b48ceb2b", + "value": ".docx", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "62d11445-876a-493f-85d3-8fc020146bdd", + "value": ".csv", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + }, + { + "comparison_operator": "is", + "id": "02c4bce8-7668-4ccd-b750-4281f314b231", + "value": ".txt", + "varType": "file", + "variable_selector": [ + "1756442986174", + "output", + "extension" + ] + } + ], + "id": "true", + "logical_operator": "or" + } + ], + "selected": false, + "title": "IF/ELSE", + "type": "if-else" + }, + "height": 358, + "id": "1756443014860", + "position": { + "x": -733.5977815139424, + "y": 236.10252072775984 + }, + "positionAbsolute": { + "x": -733.5977815139424, + "y": 236.10252072775984 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Jina Reader", + "datasource_name": "jina_reader", + "datasource_parameters": { + "crawl_sub_pages": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jina_subpages" + ] + }, + "limit": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jina_limit" + ] + }, + "url": { + "type": "mixed", + "value": "{{#rag.1756896212061.jina_url#}}" + }, + "use_sitemap": { + "type": "variable", + "value": [ + "rag", + "1756896212061", + "jian_sitemap" + ] + } + }, + "plugin_id": "langgenius/jina_datasource", + "provider_name": "jinareader", + "provider_type": "website_crawl", + "selected": false, + "title": "Jina Reader", + "type": "datasource" + }, + "height": 52, + "id": "1756896212061", + "position": { + "x": -1371.6520723158733, + "y": 538.9988445953813 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 538.9988445953813 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "Firecrawl", + "datasource_name": "crawl", + "datasource_parameters": { + "crawl_subpages": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "firecrawl_subpages" + ] + }, + "exclude_paths": { + "type": "mixed", + "value": "{{#rag.1756907397615.exclude_paths#}}" + }, + "include_paths": { + "type": "mixed", + "value": "{{#rag.1756907397615.include_paths#}}" + }, + "limit": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "max_pages" + ] + }, + "max_depth": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "max_depth" + ] + }, + "only_main_content": { + "type": "variable", + "value": [ + "rag", + "1756907397615", + "main_content" + ] + }, + "url": { + "type": "mixed", + "value": "{{#rag.1756907397615.firecrawl_url1#}}" + } + }, + "plugin_id": "langgenius/firecrawl_datasource", + "provider_name": "firecrawl", + "provider_type": "website_crawl", + "selected": false, + "title": "Firecrawl", + "type": "datasource" + }, + "height": 52, + "id": "1756907397615", + "position": { + "x": -1371.6520723158733, + "y": 644.3296146102903 + }, + "positionAbsolute": { + "x": -1371.6520723158733, + "y": 644.3296146102903 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The file you want to extract QA from.", + "ja_JP": "The file you want to extract QA from.", + "pt_BR": "The file you want to extract QA from.", + "zh_Hans": "你想要提取 QA 的文件。" + }, + "label": { + "en_US": "Input File", + "ja_JP": "Input File", + "pt_BR": "Input File", + "zh_Hans": "输入文件" + }, + "llm_description": "The file you want to extract QA from.", + "max": null, + "min": null, + "name": "input_file", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "file" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Column number for question.", + "ja_JP": "Column number for question.", + "pt_BR": "Column number for question.", + "zh_Hans": "问题所在的列。" + }, + "label": { + "en_US": "Column number for question", + "ja_JP": "Column number for question", + "pt_BR": "Column number for question", + "zh_Hans": "问题所在的列" + }, + "llm_description": "The column number for question, the format of the column number must be an integer.", + "max": null, + "min": null, + "name": "question_column", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": 1, + "form": "llm", + "human_description": { + "en_US": "Column number for answer.", + "ja_JP": "Column number for answer.", + "pt_BR": "Column number for answer.", + "zh_Hans": "答案所在的列。" + }, + "label": { + "en_US": "Column number for answer", + "ja_JP": "Column number for answer", + "pt_BR": "Column number for answer", + "zh_Hans": "答案所在的列" + }, + "llm_description": "The column number for answer, the format of the column number must be an integer.", + "max": null, + "min": null, + "name": "answer_column", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "number" + } + ], + "params": { + "answer_column": "", + "input_file": "", + "question_column": "" + }, + "provider_id": "langgenius/qa_chunk/qa_chunk", + "provider_name": "langgenius/qa_chunk/qa_chunk", + "provider_type": "builtin", + "selected": false, + "title": "Q&A Processor", + "tool_configurations": {}, + "tool_description": "A tool for QA chunking mode.", + "tool_label": "QA Chunk", + "tool_name": "qa_chunk", + "tool_node_version": "2", + "tool_parameters": { + "answer_column": { + "type": "constant", + "value": 2 + }, + "input_file": { + "type": "variable", + "value": [ + "1756912537172", + "files" + ] + }, + "question_column": { + "type": "constant", + "value": 1 + } + }, + "type": "tool" + }, + "height": 52, + "id": "1756912274158", + "position": { + "x": 794.2003154321724, + "y": 326 + }, + "positionAbsolute": { + "x": 794.2003154321724, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "context": { + "enabled": false, + "variable_selector": [] + }, + "model": { + "completion_params": { + "temperature": 0.7 + }, + "mode": "chat", + "name": "claude-3-5-sonnet-20240620", + "provider": "langgenius/anthropic/anthropic" + }, + "prompt_template": [ + { + "id": "7f8105aa-a37d-4f5a-b581-babeeb31e833", + "role": "system", + "text": "\nGenerate a list of Q&A pairs based on {{#1753346901505.output#}}. Present the output as a Markdown table, where the first column is serial number, the second column is Question, and the third column is Question. Ensure that the table format can be easily converted into a CSV file.\nExample Output Format:\n| Index | Question | Answer |\n|-------|-----------|--------|\n| 1 | What is the main purpose of the document? | The document explains the company's new product launch strategy. ![image](https://cloud.dify.ai/files/xxxxxxx) |\n| 2 || When will the product be launched? | The product will be launched in Q3 of this year. |\n\nInstructions:\nRead and understand the input text.\nExtract key information and generate meaningful questions and answers.\nPreserve any ![image] URLs from the input text in the answers.\nKeep questions concise and specific.\nEnsure answers are accurate, self-contained, and clear.\nOutput only the Markdown table without any extra explanation." + } + ], + "selected": false, + "title": "LLM", + "type": "llm", + "vision": { + "enabled": false + } + }, + "height": 88, + "id": "1756912504019", + "position": { + "x": 184.46657789772178, + "y": 326 + }, + "positionAbsolute": { + "x": 184.46657789772178, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "Markdown text", + "ja_JP": "Markdown text", + "pt_BR": "Markdown text", + "zh_Hans": "Markdown格式文本,必须为Markdown表格格式" + }, + "label": { + "en_US": "Markdown text", + "ja_JP": "Markdown text", + "pt_BR": "Markdown text", + "zh_Hans": "Markdown格式文本" + }, + "llm_description": "", + "max": null, + "min": null, + "name": "md_text", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "Filename of the output file", + "ja_JP": "Filename of the output file", + "pt_BR": "Filename of the output file", + "zh_Hans": "输出文件名" + }, + "label": { + "en_US": "Filename of the output file", + "ja_JP": "Filename of the output file", + "pt_BR": "Filename of the output file", + "zh_Hans": "输出文件名" + }, + "llm_description": "", + "max": null, + "min": null, + "name": "output_filename", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + } + ], + "params": { + "md_text": "", + "output_filename": "" + }, + "provider_id": "bowenliang123/md_exporter/md_exporter", + "provider_name": "bowenliang123/md_exporter/md_exporter", + "provider_type": "builtin", + "selected": false, + "title": "Markdown to CSV file", + "tool_configurations": {}, + "tool_description": "Generate CSV file from Markdown text", + "tool_label": "Markdown to CSV file", + "tool_name": "md_to_csv", + "tool_node_version": "2", + "tool_parameters": { + "md_text": { + "type": "mixed", + "value": "{{#1756912504019.text#}}" + }, + "output_filename": { + "type": "mixed", + "value": "LLM Generated Q&A" + } + }, + "type": "tool" + }, + "height": 52, + "id": "1756912537172", + "position": { + "x": 484.75465419110174, + "y": 326 + }, + "positionAbsolute": { + "x": 484.75465419110174, + "y": 326 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 174, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The LLM-generated Q&A pairs are designed to extract key information from the input text and present it in a structured, easy-to-use format. Each pair consists of a concise question that captures an important point or detail, and a clear, self-contained answer that provides the relevant information without requiring additional context. The output is formatted as a Markdown table with three columns—Index, Question, and Answer—so that it can be easily converted into a CSV file for further processing. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 528 + }, + "height": 174, + "id": "1756912556940", + "position": { + "x": 184.46657789772178, + "y": 462.64405262857747 + }, + "positionAbsolute": { + "x": 184.46657789772178, + "y": 462.64405262857747 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 528 + } + ], + "viewport": { + "x": 1149.1394490177502, + "y": 317.2338302699771, + "zoom": 0.4911032886685182 + } + }, + "icon_info": { + "icon": "e4ea16ed-9690-4de9-ab80-5b622ecbcc04", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "id": "98374ab6-9dcd-434d-983e-268bec156b43", + "name": "LLM Generated Q&A", + "icon": { + "icon": "e4ea16ed-9690-4de9-ab80-5b622ecbcc04", + "icon_background": null, + "icon_type": "image", + "icon_url": "" + }, + "language": "zh-Hans", + "position": 5 + }, + { + "chunk_structure": "hierarchical_model", + "description": "This knowledge pipeline uses LLMs to extract content from images and tables in documents and automatically generate descriptive annotations for contextual enrichment.", + "export_data": "dependencies:\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius\/jina:0.0.8@d3a6766fbb80890d73fea7ea04803f3e1702c6e6bd621aafb492b86222a193dd\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius\/parentchild_chunker:0.0.7@ee9c253e7942436b4de0318200af97d98d094262f3c1a56edbe29dcb01fbc158\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius\/mineru:0.5.0@ca04f2dceb4107e3adf24839756954b7c5bcb7045d035dbab5821595541c093d\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius\/anthropic:0.2.0@a776815b091c81662b2b54295ef4b8a54b5533c2ec1c66c7c8f2feea724f3248\nkind: rag_pipeline\nrag_pipeline:\n description: ''\n icon: e642577f-da15-4c03-81b9-c9dec9189a3c\n icon_background: null\n icon_type: image\n icon_url: data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAAP9UlEQVR4Ae2dTXPbxhnHdwFRr5ZN2b1kJraouk57i\/IJrJx6jDPT9Fpnkrvj3DOOv0DsXDvJxLk2nUnSW09hPkGc6aWdOBEtpZNLE9Gy3iiSQJ\/\/gg8DQnyFFiAAPjtDLbAA9uWPn5595VKrjLjtn\/YqrZaq+L6quL5X9pQqO1qtI3u+0mXy8MFJxfihP1qrss\/XQ+FFPtRK1UmreriMJkz\/GqaVX8N1z1dPHdyvnZpP1+fmVG3jhTVzDden6SjP6brt7b1y21VbWnk3CawKAbWp9Fmo0s3VbKamffWYgKz5vv+t1s5jt62qGxtrPVAnrUwqAH63u7dF\/4E3qaBbVCB8zjjHcZRDJs91XaXJpOGDMDgSx5zj2HWDMByz4\/v5fBZ80lLhE3Y498jcsfO8Nt1DlYbvmXs9L\/DbbY\/uozqmjwOUSvvVtuN8+tKLa4\/73GI1KDEAYek8x7vta\/0a5XiLcw1Y5uZcAxpgK5VKXeD4HvHTUaDdbivA2Go1yW+rZrPVkzDBUSOk7\/\/u2m8e9VyweGIdQAPenLpD\/3LvcLsM0C0szBNs8wY+nIvLpgKA8PS0YWBkKwkQyUo8un517b7tXFsl4cnO\/25p33lA7YoKMloqzanFxSXj2864xJe8Ao3GaRdGpAYQbVtEKwCS1au0Xf8TyuMWMirgQYXiOFjFw8PDcLvxC7ek79roSZ8bwO3dvTue77+P6hZV69LSElm9heKoLyXpKgCLeHx8zCBSb9m7e972YWwATVvPVfeoL\/YOcjg\/X1IrKyvd3mo313JQKAXQLgSEgBGO3v\/DG9eu3I1byFgAosr1HP9zauttitWLK32+nzs5aRgQMfSDoRtnXr8ep0qeGMAOfF+ho4FxuosXV7vjdfmWVHI\/qQKwhvv7z02VTCDVnJJ+dVIIJwIwDB\/G8FZXLwh8k761gt0PCJ8\/PzDjiHEgHBvAKHywfDKeVzCaYhYH1TAsIQazJ4VwLAAFvphvZoYeiwvh2YnVPqJ1OhwVVLti+foIJEGmNgQbYISG5Creqf85Ga7yKGlGAvj9zh5mNjbR4UCbT6rdUZLO7nWwwf0CMNNyvXuj1BhaBdPU2m2lnE8Q8aVLF6XDMUpNuW4UQMfk2bN9swKHqua7N9avPBwkzUAATbvP9b\/BDMfy8rLMbgxSUML7KoBxwqOjI1yr07TdK4OGZwZWwTS3+wDwYRWLTK311VgChygAZjA7Rq7cbpp1An3v7gtgUPWqW2j3YW5XnCgQR4HQ1OzWk529W\/3i6AsgLakyjUfAx6uS+z0sYaLAMAXQd2ADRt9PedCvV3wGwO939+7xNBuqX3GiwHkUQFWM5XnUnKu0HM8sXAnHdwZA+grVbdwA8ylOFLChABYlw5FFvBO1gj0Aou0H6wdi8REnCthQIMRTmazg7XCcPQBy229+XhaUhkWS4\/MrELKC+JJa13UB3P5xb1Pafl1d5MCyArCC6JSQ28LXdDn6LoD09bzbCJSql6UR37YC3U6t521x3F0AtaNvIlCqX5ZGfNsK4Gu5cGQJDWs4NgCiZ0JLujYRIBYQKohLQgFsSMDVMPeGDYBtt72FBAW+JGSXOFkBwAcI4bA\/EHwDoO9rY\/0cJ7iIC+JEgSQUwHpB4\/ygHWgAJDJfRiD2aREnCiSpAANodkajhDoAqgoS7bfzFMLFiQK2FGAjR7WxMXqdKjjogDCdthKTeESBqAKdTgiCK\/jjUG8kOOjsxYdAcaJAUgoAQF5hhV1xndacVL9JiS3x9leArSC2ZHa03y7jNg7s\/4iEigL2FOChGGIPAOoKosY2uOJEgTQUYGNHw39lB7vRI1HszyxOFEhDAQaQ0io7fqc3EgpMIw+SxgwrwJ0QRzvr3XpXAJxhIqZYdKp59TrSl2m4Kb6FGUuajR3trLvWtYAzpoEUd4oKcIeXhgQvCYBTfBGStFJzm\/\/EWkDqiiw1qR6W1TC7r11JlIurX\/6caPy5iJx+uUkd7SOrFYfgM8MwNBKYi7xLJoulgFTBxXqfuSuNAJi7V1asDM99+8fLpvYtly91VykUq4jDSzPtNpntNme0PLbjH67meFexf2C9Hmx8QMOAwVQcj82MF4XcJQrEVyDEmpmKk9Uw8bWUJ2Mo0ANgjOflEVHAmgLSCbEmpUQURwEBMI5q8ow1BQRAa1JKRHEUyAWAPx7Rj+I1afpGXOEUyAWAn+2cqI9\/aBROfCkQLT\/Iugiwfp\/tNtRH3x+LFcz6y4qRv8wDCOu3a6pgX6xgjBec9UcyDSBbPxZRrCArURw\/0wCy9WO595tiBVmLoviZBTBq\/VhwsYKsRDH8zAIYtX4st1hBVqIYfiYBHGT9WHKxgqxE\/v1MAjjI+rHcYgVZifz7mfo5pACsE\/XRDycjlYUVhPvT1QV1dTmT\/0cjyyA30LfisiBCFzwz2Ezf0BvD4ZkP\/n2k\/kbjhH++tiggjqFZFm+ZKoBxwIuKiPaigBhVJT\/n+snOL8bkXL68llqubYA3KLMvUnU8iUVM+zsU0fQGlaPw4Yd1U8RULWCS4PELE4vISuTDT7X1DgCxC8OlUvLJ\/pqWfOE+yyimagFRPb77h2VTRaLz8PfdU1po0Laqz8WSVm\/9dlG9fX1J4VhcthVIFUCWIgkQ8wqe7e\/tRtuYtuPnd3he\/5dfglpwKgBy5m2AmFfwWINZ96cKIIsfBfFjGohGG26YE\/CGqZOfa5kAkOViENFy++A\/wUwHX4v6b1Eb793fL0WD5TxnCiTfHY0hCOAa1oF4cdlVb9AUnLj8K3AuAD\/baSh8bDvA9zb1ZAe5N67J\/O8gbfIWHrsKBnjvfnPQLS+gsOlgBbEoIdoWFOtnU+XpxxXLAkbhA4i2LeEgKyjWb\/rQ2MzBxABG4ePMJAFhtC0o1o\/VLo4\/EYCD4GM5bEMYtYJi\/Vjp4vhjAzgKPpbENoRsBcX6scLF8sfqhIwLH0sDCOFsdEzYCvq0lausfGaFi+OPBHBS+FgamxDCCj4bMTPC6YqfLwWGAhgXPpbAFoSwgviIK54CA9uA54WPpbLdJuR4xS+GAn0BtAUfSyQQshLiRxU4A6Bt+DhBgZCVED+sQA+AScHHCQqErIT4rEAXwKTh4wQFQlZCfChgesH\/+G9DvfdDenswA0I4G+OEJiL5k1sFHAPfvw5TL4BYwtQlz2SCzntTgI+VEAhZidn1u23AaUkgEE5L+WykO3UAIYNAmA0YppGLTAAoEE7j1WcjzcwAKBBmA4i0c5EpAAXCtF\/\/9NPLHIAC4fShSDMHmQRQIEwTgemmlVkABcLpgpFW6pkGUCBMC4PppZN5AAXC6cGRRsq5AFAgTAOF6aSRGwAFwukAknSquQJQIEwah\/Tjzx2AAmH6kCSZYi4BFAiTRCLduHMLoECYLihJpUYA6uAna+j3O\/LoZClX\/t4afium4+oEoJ9rAFEQgZDfZz78MIB65a9PtinbFbV0USkn1zWyFfWT\/l2N6O94WMl03iLx6QtwR\/vIdU2Iy9vLK1h+BcCCvdC8FUcAzNsbK0J+u50QXcfvBX9FZdpaXV1VpdLQ3dqKUHQpQwYUaDZb6vnz58hJVSxgBl7ILGcBAJphmFDXeJb1kLKnrIDj+f4zpOmjayxOFEhBAc8LfiNaKy3DMCnoLUlEFOj2QSjcoZ2Xa7jueWIBoYO45BXg2tbzvaeY+zBtQM\/rzs8lnwNJYaYVCPU36k5bd+aClQA401SkWHiubbV2ao7Wbg1pt1pBwzDFfEhSM6oAW0Bfq7oz1wragBw4o5pIsVNUoN0O+htzc7QYYWNjrYa0YRYFwhTfwgwnxVXwxgtrnWEYX6zgDPOQatG5qad99RgJB1NxOjhpNpupZkYSmz0FeBCaKuGnKH0AoO+bE6Zz9mSREqelQKvV6iTlhy2gX0Uo09m5QzxRwLoC7XZnGk47vwLott0qUoIFlI6Idc0lwpACWIoF57ZVFb6pgqknjNmQKuCTahiyiEtCAYYPHZAOc502IKVG8H2NRE9PT5NIW+IUBYithlHBVwFrOAk6IebIqcITAKGCuCQUYAvoec4jjr8L4I2ra1UKNNUw38g3iS8KnFeBRqNhJjuw+uqljTXTAUGcXQBxon3\/S\/gnJ8fwxIkC1hTgmtVX+n440h4AHTKNRGgdFlCsYFgmOT6PAswTrN\/vrq09CsfVAyB6JrRE\/0PcIFYwLJMcn0eBw8Pg11iJrU+j8RCUvW57e6\/sOf43tFSmsry8pBYXF3tvkDNRYAIF0PY7PDxSsH7Xr13eiD7aYwFxEVbQ1\/oujo+PT2RgGkKIi6UAll2BIbho248jPAMgLlA9\/QV5pkd8cJD+j1lz5sTPtwJoxnWWXn0RbftxyfoCiItuW79JZpM6JE1qDwYU80PiiwKjFDg5aahG4xRVb90tBTVqv2cGAkhVcU35QZcZZpRXsfaLRMJEgbACQdUbDOVR1XsXC0\/D18PHAwHETdfX1x5SI\/BDzBFjLw+BMCydHPdTAIyAFbOohdgZVPXys2Qhh7tOr\/gr6hVvuq6rLl5cVVqPfGx4pHK1kAoAuv19GKo2TWqox9fXL78yqqBDLSAeRq\/Y8fTrFGENESMBQ\/eomOX6TCnQAx8NuTjz+vVxBBjblJElrND4ICxhRSzhONLOzj1n4CvpV4e1+8LKjA0gHopCeOHCBeW6I41oOD05LpgCaPMdHBwE1S4s3wTwQYqJAMQDYQgd2tgDG1sKhFBm9hx3ODDWRyBNDB8UmxhAPNSB8HN0TNAhWVpalCk7CDNDDuN8x8fHpj+ADgfafONWu2GZYgHIETx5+vND6hLfwfnCwjxBuCTWkMUpqI\/2HhYXnJ52vsJLQy2u57yPzmqcIp8LQCT4ZGfvtlb+A9raqIwqGdZwYWEhTl7kmYwr0GP1aIaDVrfcv7F+5eF5sn1uAJE4quS2qx7QlPMtnAPElZUV2fQcYhTAYT0f5nVDa0SrNL32ZpwqNyqHFQA5UmMNff8ehmoQhl335+fnxSKyQDnzo+ARLDVMrXUWq1gpjVUAOUffPf35fUfpvzCIsIgBjAtiFVmkDPpo3+Fruc3mqVlIgHM4gsQsVJ7znIdx23qDipsIgJxY1CJyOGDEYPYc7c\/lOPBdviR+SgoALnyw2gkzXPj02Zigqn39peOpR7bB42ImCiAnsv3j3iaNGVFnRd\/E0A2Hh31YSYwnYlgHx\/D5A0jZBdd7s8338T2z4DNA0bJibA4O+zCzBeOt93DOkPEWadHn6bxK931NL6Ha+aZkn1vsBfW+SXvxDoyJOixl6rBskUAYQ3yZxpAqg6AcGIlcsKMAtuXDzmjYnEo7VWyXkZSlG5Th1AEclJHtn\/YqtHFShYAsA0pPeWXawn8d91PDt0KecbiOIR8+h0\/G8kxY+HoRj+nF1cmg1c+UTQd7PVJ4nYbHzHXaf\/6po5x6m7bEJa1q2JnURg\/2TNoxAv4PoGedQHqhulIAAAAASUVORK5CYII=\n name: Contextual Enrichment Using LLM\nversion: 0.1.0\nworkflow:\n conversation_variables: []\n environment_variables: []\n features: {}\n graph:\n edges:\n - data:\n isInLoop: false\n sourceType: tool\n targetType: knowledge-index\n id: 1751336942081-source-1750400198569-target\n selected: false\n source: '1751336942081'\n sourceHandle: source\n target: '1750400198569'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: llm\n targetType: tool\n id: 1758002850987-source-1751336942081-target\n source: '1758002850987'\n sourceHandle: source\n target: '1751336942081'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInIteration: false\n isInLoop: false\n sourceType: datasource\n targetType: tool\n id: 1756915693835-source-1758027159239-target\n source: '1756915693835'\n sourceHandle: source\n target: '1758027159239'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: llm\n id: 1758027159239-source-1758002850987-target\n source: '1758027159239'\n sourceHandle: source\n target: '1758002850987'\n targetHandle: target\n type: custom\n zIndex: 0\n nodes:\n - data:\n chunk_structure: hierarchical_model\n embedding_model: jina-embeddings-v2-base-en\n embedding_model_provider: langgenius\/jina\/jina\n index_chunk_variable_selector:\n - '1751336942081'\n - result\n indexing_technique: high_quality\n keyword_number: 10\n retrieval_model:\n reranking_enable: true\n reranking_mode: reranking_model\n reranking_model:\n reranking_model_name: jina-reranker-v1-base-en\n reranking_provider_name: langgenius\/jina\/jina\n score_threshold: 0\n score_threshold_enabled: false\n search_method: hybrid_search\n top_k: 3\n weights: null\n selected: false\n title: Knowledge Base\n type: knowledge-index\n height: 114\n id: '1750400198569'\n position:\n x: 474.7618603027596\n y: 282\n positionAbsolute:\n x: 474.7618603027596\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 458\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently\n we support 5 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data\n Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\":\n File Upload, Text Input, Online Drive, Online Doc, and Web Crawler. Different\n types of Data Sources have different input and output types. The output\n of File Upload and Online Drive are files, while the output of Online Doc\n and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n Knowledge Pipeline can have multiple data sources. Each data source can\n be selected more than once with different settings. Each added data source\n is a tab on the add file interface. However, each time the user can only\n select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 458\n id: '1751264451381'\n position:\n x: -893.2836123260277\n y: 378.2537898330178\n positionAbsolute:\n x: -893.2836123260277\n y: 378.2537898330178\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 260\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge\n Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n starts with Data Source as the starting node and ends with the knowledge\n base node. The general steps are: import documents from the data source\n \u2192 use extractor to extract document content \u2192 split and clean content into\n structured chunks \u2192 store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n user input variables required by the Knowledge Pipeline node must be predefined\n and managed via the Input Field section located in the top-right corner\n of the orchestration canvas. It determines what input fields the end users\n will see and need to fill in when importing files to the knowledge base\n through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique\n Inputs: Input fields defined here are only available to the selected data\n source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global\n Inputs: These input fields are shared across all subsequent nodes after\n the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For\n more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 1182\n height: 260\n id: '1751266376760'\n position:\n x: -704.0614991386192\n y: -73.30453110517956\n positionAbsolute:\n x: -704.0614991386192\n y: -73.30453110517956\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 1182\n - data:\n author: TenTen\n desc: ''\n height: 304\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"MinerU\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n is an advanced open-source document extractor designed specifically to convert\n complex, unstructured documents\u2014such as PDFs, Word files, and PPTs\u2014into\n high-quality, machine-readable formats like Markdown and JSON. MinerU addresses\n challenges in document parsing such as layout detection, formula recognition,\n and multi-language support, which are critical for generating high-quality\n training corpora for LLMs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 304\n id: '1751266402561'\n position:\n x: -555.2228329530462\n y: 592.0458661166498\n positionAbsolute:\n x: -555.2228329530462\n y: 592.0458661166498\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 554\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n addresses the dilemma of context and precision by leveraging a two-tier\n hierarchical approach that effectively balances the trade-off between accurate\n matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here\n is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Query Matching with Child Chunks: Small, focused pieces of information,\n often as concise as a single sentence within a paragraph, are used to match\n the user''s query. These child chunks enable precise and relevant initial\n retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Contextual Enrichment with Parent Chunks: Larger, encompassing sections\u2014such\n as a paragraph, a section, or even an entire document\u2014that include the matched\n child chunks are then retrieved. These parent chunks provide comprehensive\n context for the Language Model (LLM).\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 554\n id: '1751266447821'\n position:\n x: 153.2996965006646\n y: 378.2537898330178\n positionAbsolute:\n x: 153.2996965006646\n y: 378.2537898330178\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 411\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n knowledge base provides two indexing methods:\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n each with different retrieval strategies. High-Quality mode uses embeddings\n for vectorization and supports vector, full-text, and hybrid retrieval,\n offering more accurate results but higher resource usage. Economical mode\n uses keyword-based inverted indexing with no token consumption but lower\n accuracy; upgrading to High-Quality is possible, but downgrading requires\n creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"*\n Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0only\n support the\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0indexing\n method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 411\n id: '1751266580099'\n position:\n x: 482.3389174180554\n y: 437.9839361130071\n positionAbsolute:\n x: 482.3389174180554\n y: 437.9839361130071\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n result:\n description: Parent child chunks result\n items:\n type: object\n type: array\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: ''\n ja_JP: ''\n pt_BR: ''\n zh_Hans: ''\n label:\n en_US: Input Content\n ja_JP: Input Content\n pt_BR: Conte\u00fado de Entrada\n zh_Hans: \u8f93\u5165\u6587\u672c\n llm_description: The text you want to chunk.\n max: null\n min: null\n name: input_text\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: paragraph\n form: llm\n human_description:\n en_US: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n ja_JP: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n pt_BR: Dividir texto em par\u00e1grafos com base no separador e no comprimento\n m\u00e1ximo do bloco, usando o texto dividido como bloco pai ou documento\n completo como bloco pai e diretamente recuper\u00e1-lo.\n zh_Hans: \u6839\u636e\u5206\u9694\u7b26\u548c\u6700\u5927\u5757\u957f\u5ea6\u5c06\u6587\u672c\u62c6\u5206\u4e3a\u6bb5\u843d\uff0c\u4f7f\u7528\u62c6\u5206\u6587\u672c\u4f5c\u4e3a\u68c0\u7d22\u7684\u7236\u5757\u6216\u6574\u4e2a\u6587\u6863\u7528\u4f5c\u7236\u5757\u5e76\u76f4\u63a5\u68c0\u7d22\u3002\n label:\n en_US: Parent Mode\n ja_JP: Parent Mode\n pt_BR: Modo Pai\n zh_Hans: \u7236\u5757\u6a21\u5f0f\n llm_description: Split text into paragraphs based on separator and maximum\n chunk length, using split text as parent block or entire document as parent\n block and directly retrieve.\n max: null\n min: null\n name: parent_mode\n options:\n - label:\n en_US: Paragraph\n ja_JP: Paragraph\n pt_BR: Par\u00e1grafo\n zh_Hans: \u6bb5\u843d\n value: paragraph\n - label:\n en_US: Full Document\n ja_JP: Full Document\n pt_BR: Documento Completo\n zh_Hans: \u5168\u6587\n value: full_doc\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: '\n\n\n '\n form: llm\n human_description:\n en_US: Separator used for chunking\n ja_JP: Separator used for chunking\n pt_BR: Separador usado para divis\u00e3o\n zh_Hans: \u7528\u4e8e\u5206\u5757\u7684\u5206\u9694\u7b26\n label:\n en_US: Parent Delimiter\n ja_JP: Parent Delimiter\n pt_BR: Separador de Pai\n zh_Hans: \u7236\u5757\u5206\u9694\u7b26\n llm_description: The separator used to split chunks\n max: null\n min: null\n name: separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 1024\n form: llm\n human_description:\n en_US: Maximum length for chunking\n ja_JP: Maximum length for chunking\n pt_BR: Comprimento m\u00e1ximo para divis\u00e3o\n zh_Hans: \u7528\u4e8e\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6\n label:\n en_US: Maximum Parent Chunk Length\n ja_JP: Maximum Parent Chunk Length\n pt_BR: Comprimento M\u00e1ximo do Bloco Pai\n zh_Hans: \u6700\u5927\u7236\u5757\u957f\u5ea6\n llm_description: Maximum length allowed per chunk\n max: null\n min: null\n name: max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: '. '\n form: llm\n human_description:\n en_US: Separator used for subchunking\n ja_JP: Separator used for subchunking\n pt_BR: Separador usado para subdivis\u00e3o\n zh_Hans: \u7528\u4e8e\u5b50\u5206\u5757\u7684\u5206\u9694\u7b26\n label:\n en_US: Child Delimiter\n ja_JP: Child Delimiter\n pt_BR: Separador de Subdivis\u00e3o\n zh_Hans: \u5b50\u5206\u5757\u5206\u9694\u7b26\n llm_description: The separator used to split subchunks\n max: null\n min: null\n name: subchunk_separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 512\n form: llm\n human_description:\n en_US: Maximum length for subchunking\n ja_JP: Maximum length for subchunking\n pt_BR: Comprimento m\u00e1ximo para subdivis\u00e3o\n zh_Hans: \u7528\u4e8e\u5b50\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6\n label:\n en_US: Maximum Child Chunk Length\n ja_JP: Maximum Child Chunk Length\n pt_BR: Comprimento M\u00e1ximo de Subdivis\u00e3o\n zh_Hans: \u5b50\u5206\u5757\u6700\u5927\u957f\u5ea6\n llm_description: Maximum length allowed per subchunk\n max: null\n min: null\n name: subchunk_max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove consecutive spaces, newlines and tabs\n ja_JP: Whether to remove consecutive spaces, newlines and tabs\n pt_BR: Se deve remover espa\u00e7os extras no texto\n zh_Hans: \u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26\n label:\n en_US: Replace consecutive spaces, newlines and tabs\n ja_JP: Replace consecutive spaces, newlines and tabs\n pt_BR: Substituir espa\u00e7os consecutivos, novas linhas e guias\n zh_Hans: \u66ff\u6362\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26\n llm_description: Whether to remove consecutive spaces, newlines and tabs\n max: null\n min: null\n name: remove_extra_spaces\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove URLs and emails in the text\n ja_JP: Whether to remove URLs and emails in the text\n pt_BR: Se deve remover URLs e e-mails no texto\n zh_Hans: \u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740\n label:\n en_US: Delete all URLs and email addresses\n ja_JP: Delete all URLs and email addresses\n pt_BR: Remover todas as URLs e e-mails\n zh_Hans: \u5220\u9664\u6240\u6709URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740\n llm_description: Whether to remove URLs and emails in the text\n max: null\n min: null\n name: remove_urls_emails\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n params:\n input_text: ''\n max_length: ''\n parent_mode: ''\n remove_extra_spaces: ''\n remove_urls_emails: ''\n separator: ''\n subchunk_max_length: ''\n subchunk_separator: ''\n provider_id: langgenius\/parentchild_chunker\/parentchild_chunker\n provider_name: langgenius\/parentchild_chunker\/parentchild_chunker\n provider_type: builtin\n selected: false\n title: Parent-child Chunker\n tool_configurations: {}\n tool_description: Process documents into parent-child chunk structures\n tool_label: Parent-child Chunker\n tool_name: parentchild_chunker\n tool_node_version: '2'\n tool_parameters:\n input_text:\n type: mixed\n value: '{{#1758002850987.text#}}'\n max_length:\n type: variable\n value:\n - rag\n - shared\n - Maximum_Parent_Length\n parent_mode:\n type: variable\n value:\n - rag\n - shared\n - Parent_Mode\n remove_extra_spaces:\n type: variable\n value:\n - rag\n - shared\n - clean_1\n remove_urls_emails:\n type: variable\n value:\n - rag\n - shared\n - clean_2\n separator:\n type: mixed\n value: '{{#rag.shared.Parent_Delimiter#}}'\n subchunk_max_length:\n type: variable\n value:\n - rag\n - shared\n - Maximum_Child_Length\n subchunk_separator:\n type: mixed\n value: '{{#rag.shared.Child_Delimiter#}}'\n type: tool\n height: 52\n id: '1751336942081'\n position:\n x: 144.55897745117755\n y: 282\n positionAbsolute:\n x: 144.55897745117755\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 446\n selected: true\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"In\n this step, the LLM is responsible for enriching and reorganizing content,\n along with images and tables. The goal is to maintain the integrity of image\n URLs and tables while providing contextual descriptions and summaries to\n enhance understanding. The content should be structured into well-organized\n paragraphs, using double newlines to separate them. The LLM should enrich\n the document by adding relevant descriptions for images and extracting key\n insights from tables, ensuring the content remains easy to retrieve within\n a Retrieval-Augmented Generation (RAG) system. The final output should preserve\n the original structure, making it more accessible for knowledge retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 446\n id: '1753967810859'\n position:\n x: -176.67459682201036\n y: 405.2790698865377\n positionAbsolute:\n x: -176.67459682201036\n y: 405.2790698865377\n selected: true\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n datasource_configurations: {}\n datasource_label: File\n datasource_name: upload-file\n datasource_parameters: {}\n fileExtensions:\n - pdf\n - doc\n - docx\n - pptx\n - ppt\n - jpg\n - png\n - jpeg\n plugin_id: langgenius\/file\n provider_name: file\n provider_type: local_file\n selected: false\n title: File\n type: datasource\n height: 52\n id: '1756915693835'\n position:\n x: -893.2836123260277\n y: 282\n positionAbsolute:\n x: -893.2836123260277\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n context:\n enabled: false\n variable_selector: []\n model:\n completion_params:\n temperature: 0.7\n mode: chat\n name: claude-3-5-sonnet-20240620\n provider: langgenius\/anthropic\/anthropic\n prompt_template:\n - id: beb97761-d30d-4549-9b67-de1b8292e43d\n role: system\n text: \"You are an AI document assistant. \\nYour tasks are:\\nEnrich the content\\\n \\ contextually:\\nAdd meaningful descriptions for each image.\\nSummarize\\\n \\ key information from each table.\\nOutput the enriched content\u00a0with clear\\\n \\ annotations showing the\u00a0corresponding image and table positions, so\\\n \\ the text can later be aligned back into the original document. Preserve\\\n \\ any ![image] URLs from the input text.\\nYou will receive two inputs:\\n\\\n The file and text\u00a0(may contain images url and tables).\\nThe final output\\\n \\ should be a\u00a0single, enriched version of the original document with ![image]\\\n \\ url preserved.\\nGenerate output directly without saying words like:\\\n \\ Here's the enriched version of the original text with the image description\\\n \\ inserted.\"\n - id: f92ef0cd-03a7-48a7-80e8-bcdc965fb399\n role: user\n text: The file is {{#1756915693835.file#}} and the text are\u00a0{{#1758027159239.text#}}.\n selected: false\n title: LLM\n type: llm\n vision:\n configs:\n detail: high\n variable_selector:\n - '1756915693835'\n - file\n enabled: true\n height: 88\n id: '1758002850987'\n position:\n x: -176.67459682201036\n y: 282\n positionAbsolute:\n x: -176.67459682201036\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: The file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n ja_JP: \u89e3\u6790\u3059\u308b\u30d5\u30a1\u30a4\u30eb(pdf\u3001ppt\u3001pptx\u3001doc\u3001docx\u3001png\u3001jpg\u3001jpeg\u3092\u30b5\u30dd\u30fc\u30c8)\n pt_BR: The file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n zh_Hans: \u7528\u4e8e\u89e3\u6790\u7684\u6587\u4ef6(\u652f\u6301 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)\n label:\n en_US: file\n ja_JP: file\n pt_BR: file\n zh_Hans: file\n llm_description: The file to be parsed (support pdf, ppt, pptx, doc, docx,\n png, jpg, jpeg)\n max: null\n min: null\n name: file\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: file\n - auto_generate: null\n default: auto\n form: form\n human_description:\n en_US: (For local deployment v1 and v2) Parsing method, can be auto, ocr,\n or txt. Default is auto. If results are not satisfactory, try ocr\n ja_JP: \uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v1\u3068v2\u7528\uff09\u89e3\u6790\u65b9\u6cd5\u306f\u3001auto\u3001ocr\u3001\u307e\u305f\u306ftxt\u306e\u3044\u305a\u308c\u304b\u3067\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fauto\u3067\u3059\u3002\u7d50\u679c\u304c\u6e80\u8db3\u3067\u304d\u306a\u3044\u5834\u5408\u306f\u3001ocr\u3092\u8a66\u3057\u3066\u304f\u3060\u3055\u3044\n pt_BR: (For local deployment v1 and v2) Parsing method, can be auto, ocr,\n or txt. Default is auto. If results are not satisfactory, try ocr\n zh_Hans: \uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72v1\u548cv2\u7248\u672c\uff09\u89e3\u6790\u65b9\u6cd5\uff0c\u53ef\u4ee5\u662fauto, ocr, \u6216 txt\u3002\u9ed8\u8ba4\u662fauto\u3002\u5982\u679c\u7ed3\u679c\u4e0d\u7406\u60f3\uff0c\u8bf7\u5c1d\u8bd5ocr\n label:\n en_US: parse method\n ja_JP: \u89e3\u6790\u65b9\u6cd5\n pt_BR: parse method\n zh_Hans: \u89e3\u6790\u65b9\u6cd5\n llm_description: (For local deployment v1 and v2) Parsing method, can be\n auto, ocr, or txt. Default is auto. If results are not satisfactory, try\n ocr\n max: null\n min: null\n name: parse_method\n options:\n - icon: ''\n label:\n en_US: auto\n ja_JP: auto\n pt_BR: auto\n zh_Hans: auto\n value: auto\n - icon: ''\n label:\n en_US: ocr\n ja_JP: ocr\n pt_BR: ocr\n zh_Hans: ocr\n value: ocr\n - icon: ''\n label:\n en_US: txt\n ja_JP: txt\n pt_BR: txt\n zh_Hans: txt\n value: txt\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: 1\n form: form\n human_description:\n en_US: (For official API and local deployment v2) Whether to enable formula\n recognition\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\u3068\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b\n pt_BR: (For official API and local deployment v2) Whether to enable formula\n recognition\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\u548c\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u662f\u5426\u5f00\u542f\u516c\u5f0f\u8bc6\u522b\n label:\n en_US: Enable formula recognition\n ja_JP: \u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\n pt_BR: Enable formula recognition\n zh_Hans: \u5f00\u542f\u516c\u5f0f\u8bc6\u522b\n llm_description: (For official API and local deployment v2) Whether to enable\n formula recognition\n max: null\n min: null\n name: enable_formula\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: 1\n form: form\n human_description:\n en_US: (For official API and local deployment v2) Whether to enable table\n recognition\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\u3068\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b\n pt_BR: (For official API and local deployment v2) Whether to enable table\n recognition\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\u548c\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u662f\u5426\u5f00\u542f\u8868\u683c\u8bc6\u522b\n label:\n en_US: Enable table recognition\n ja_JP: \u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\n pt_BR: Enable table recognition\n zh_Hans: \u5f00\u542f\u8868\u683c\u8bc6\u522b\n llm_description: (For official API and local deployment v2) Whether to enable\n table recognition\n max: null\n min: null\n name: enable_table\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: auto\n form: form\n human_description:\n en_US: '(For official API and local deployment v2) Specify document language,\n default ch, can be set to auto(local deployment need to specify the\n language, default ch), other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5'\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\u3068\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u6307\u5b9a\u3057\u307e\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fch\u3067\u3001auto\u306b\u8a2d\u5b9a\u3067\u304d\u307e\u3059\u3002auto\u306e\u5834\u5408\uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8\u3067\u306f\u8a00\u8a9e\u3092\u6307\u5b9a\u3059\u308b\u5fc5\u8981\u304c\u3042\u308a\u307e\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fch\u3067\u3059\uff09\u3001\u30e2\u30c7\u30eb\u306f\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u81ea\u52d5\u7684\u306b\u8b58\u5225\u3057\u307e\u3059\u3002\u4ed6\u306e\u30aa\u30d7\u30b7\u30e7\u30f3\u5024\u30ea\u30b9\u30c8\u306b\u3064\u3044\u3066\u306f\u3001\u6b21\u3092\u53c2\u7167\u3057\u3066\u304f\u3060\u3055\u3044\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5\n pt_BR: '(For official API and local deployment v2) Specify document language,\n default ch, can be set to auto(local deployment need to specify the\n language, default ch), other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5'\n zh_Hans: \uff08\u4ec5\u9650\u5b98\u65b9api\u548c\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u6307\u5b9a\u6587\u6863\u8bed\u8a00\uff0c\u9ed8\u8ba4 ch\uff0c\u53ef\u4ee5\u8bbe\u7f6e\u4e3aauto\uff0c\u5f53\u4e3aauto\u65f6\u6a21\u578b\u4f1a\u81ea\u52a8\u8bc6\u522b\u6587\u6863\u8bed\u8a00\uff08\u672c\u5730\u90e8\u7f72\u9700\u8981\u6307\u5b9a\u660e\u786e\u7684\u8bed\u8a00\uff0c\u9ed8\u8ba4ch\uff09\uff0c\u5176\u4ed6\u53ef\u9009\u503c\u5217\u8868\u8be6\u89c1\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5\n label:\n en_US: Document language\n ja_JP: \u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\n pt_BR: Document language\n zh_Hans: \u6587\u6863\u8bed\u8a00\n llm_description: '(For official API and local deployment v2) Specify document\n language, default ch, can be set to auto(local deployment need to specify\n the language, default ch), other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5'\n max: null\n min: null\n name: language\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 0\n form: form\n human_description:\n en_US: (For official API) Whether to enable OCR recognition\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b\n pt_BR: (For official API) Whether to enable OCR recognition\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542fOCR\u8bc6\u522b\n label:\n en_US: Enable OCR recognition\n ja_JP: OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\n pt_BR: Enable OCR recognition\n zh_Hans: \u5f00\u542fOCR\u8bc6\u522b\n llm_description: (For official API) Whether to enable OCR recognition\n max: null\n min: null\n name: enable_ocr\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: '[]'\n form: form\n human_description:\n en_US: '(For official API) Example: [\"docx\",\"html\"], markdown, json are\n the default export formats, no need to set, this parameter only supports\n one or more of docx, html, latex'\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09\u4f8b\uff1a[\"docx\",\"html\"]\u3001markdown\u3001json\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f\u3067\u3042\u308a\u3001\u8a2d\u5b9a\u3059\u308b\u5fc5\u8981\u306f\u3042\u308a\u307e\u305b\u3093\u3002\u3053\u306e\u30d1\u30e9\u30e1\u30fc\u30bf\u306f\u3001docx\u3001html\u3001latex\u306e3\u3064\u306e\u5f62\u5f0f\u306e\u3044\u305a\u308c\u304b\u307e\u305f\u306f\u8907\u6570\u306e\u307f\u3092\u30b5\u30dd\u30fc\u30c8\u3057\u307e\u3059\n pt_BR: '(For official API) Example: [\"docx\",\"html\"], markdown, json are\n the default export formats, no need to set, this parameter only supports\n one or more of docx, html, latex'\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u793a\u4f8b\uff1a[\"docx\",\"html\"],markdown\u3001json\u4e3a\u9ed8\u8ba4\u5bfc\u51fa\u683c\u5f0f\uff0c\u65e0\u987b\u8bbe\u7f6e\uff0c\u8be5\u53c2\u6570\u4ec5\u652f\u6301docx\u3001html\u3001latex\u4e09\u79cd\u683c\u5f0f\u4e2d\u7684\u4e00\u4e2a\u6216\u591a\u4e2a\n label:\n en_US: Extra export formats\n ja_JP: \u8ffd\u52a0\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f\n pt_BR: Extra export formats\n zh_Hans: \u989d\u5916\u5bfc\u51fa\u683c\u5f0f\n llm_description: '(For official API) Example: [\"docx\",\"html\"], markdown,\n json are the default export formats, no need to set, this parameter only\n supports one or more of docx, html, latex'\n max: null\n min: null\n name: extra_formats\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: pipeline\n form: form\n human_description:\n en_US: '(For local deployment v2) Example: pipeline, vlm-transformers,\n vlm-sglang-engine, vlm-sglang-client, default is pipeline'\n ja_JP: \uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u4f8b\uff1apipeline\u3001vlm-transformers\u3001vlm-sglang-engine\u3001vlm-sglang-client\u3001\u30c7\u30d5\u30a9\u30eb\u30c8\u306fpipeline\n pt_BR: '(For local deployment v2) Example: pipeline, vlm-transformers,\n vlm-sglang-engine, vlm-sglang-client, default is pipeline'\n zh_Hans: \uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u793a\u4f8b\uff1apipeline\u3001vlm-transformers\u3001vlm-sglang-engine\u3001vlm-sglang-client\uff0c\u9ed8\u8ba4\u503c\u4e3apipeline\n label:\n en_US: Backend type\n ja_JP: \u30d0\u30c3\u30af\u30a8\u30f3\u30c9\u30bf\u30a4\u30d7\n pt_BR: Backend type\n zh_Hans: \u89e3\u6790\u540e\u7aef\n llm_description: '(For local deployment v2) Example: pipeline, vlm-transformers,\n vlm-sglang-engine, vlm-sglang-client, default is pipeline'\n max: null\n min: null\n name: backend\n options:\n - icon: ''\n label:\n en_US: pipeline\n ja_JP: pipeline\n pt_BR: pipeline\n zh_Hans: pipeline\n value: pipeline\n - icon: ''\n label:\n en_US: vlm-transformers\n ja_JP: vlm-transformers\n pt_BR: vlm-transformers\n zh_Hans: vlm-transformers\n value: vlm-transformers\n - icon: ''\n label:\n en_US: vlm-sglang-engine\n ja_JP: vlm-sglang-engine\n pt_BR: vlm-sglang-engine\n zh_Hans: vlm-sglang-engine\n value: vlm-sglang-engine\n - icon: ''\n label:\n en_US: vlm-sglang-client\n ja_JP: vlm-sglang-client\n pt_BR: vlm-sglang-client\n zh_Hans: vlm-sglang-client\n value: vlm-sglang-client\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: ''\n form: form\n human_description:\n en_US: '(For local deployment v2 when backend is vlm-sglang-client) Example:\n http:\/\/127.0.0.1:8000, default is empty'\n ja_JP: \uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528 \u89e3\u6790\u5f8c\u7aef\u304cvlm-sglang-client\u306e\u5834\u5408\uff09\u4f8b\uff1ahttp:\/\/127.0.0.1:8000\u3001\u30c7\u30d5\u30a9\u30eb\u30c8\u306f\u7a7a\n pt_BR: '(For local deployment v2 when backend is vlm-sglang-client) Example:\n http:\/\/127.0.0.1:8000, default is empty'\n zh_Hans: \uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72v2\u7248\u672c \u89e3\u6790\u540e\u7aef\u4e3avlm-sglang-client\u65f6\uff09\u793a\u4f8b\uff1ahttp:\/\/127.0.0.1:8000\uff0c\u9ed8\u8ba4\u503c\u4e3a\u7a7a\n label:\n en_US: sglang-server url\n ja_JP: sglang-server\u30a2\u30c9\u30ec\u30b9\n pt_BR: sglang-server url\n zh_Hans: sglang-server\u5730\u5740\n llm_description: '(For local deployment v2 when backend is vlm-sglang-client)\n Example: http:\/\/127.0.0.1:8000, default is empty'\n max: null\n min: null\n name: sglang_server_url\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n params:\n backend: ''\n enable_formula: ''\n enable_ocr: ''\n enable_table: ''\n extra_formats: ''\n file: ''\n language: ''\n parse_method: ''\n sglang_server_url: ''\n provider_id: langgenius\/mineru\/mineru\n provider_name: langgenius\/mineru\/mineru\n provider_type: builtin\n selected: false\n title: Parse File\n tool_configurations:\n backend:\n type: constant\n value: pipeline\n enable_formula:\n type: constant\n value: 1\n enable_ocr:\n type: constant\n value: true\n enable_table:\n type: constant\n value: 1\n extra_formats:\n type: mixed\n value: '[]'\n language:\n type: mixed\n value: auto\n parse_method:\n type: constant\n value: auto\n sglang_server_url:\n type: mixed\n value: ''\n tool_description: a tool for parsing text, tables, and images, supporting\n multiple formats such as pdf, pptx, docx, etc. supporting multiple languages\n such as English, Chinese, etc.\n tool_label: Parse File\n tool_name: parse-file\n tool_node_version: '2'\n tool_parameters:\n file:\n type: variable\n value:\n - '1756915693835'\n - file\n type: tool\n height: 270\n id: '1758027159239'\n position:\n x: -544.9739996945534\n y: 282\n positionAbsolute:\n x: -544.9739996945534\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n viewport:\n x: 679.9701291615181\n y: -191.49392257836791\n zoom: 0.8239704766223018\n rag_pipeline_variables:\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: paragraph\n label: Parent Mode\n max_length: 48\n options:\n - paragraph\n - full_doc\n placeholder: null\n required: true\n tooltips: 'Parent Mode provides two options: paragraph mode splits text into paragraphs\n as parent chunks for retrieval, while full_doc mode uses the entire document\n as a single parent chunk (text beyond 10,000 tokens will be truncated).'\n type: select\n unit: null\n variable: Parent_Mode\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\\n\n label: Parent Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: A delimiter is the character used to separate text. \\n\\n is recommended\n for splitting the original document into large parent chunks. You can also use\n special delimiters defined by yourself.\n type: text-input\n unit: null\n variable: Parent_Delimiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 1024\n label: Maximum Parent Length\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: number\n unit: tokens\n variable: Maximum_Parent_Length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\n label: Child Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: A delimiter is the character used to separate text. \\n is recommended\n for splitting parent chunks into small child chunks. You can also use special\n delimiters defined by yourself.\n type: text-input\n unit: null\n variable: Child_Delimiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 256\n label: Maximum Child Length\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: ''\n type: number\n unit: tokens\n variable: Maximum_Child_Length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: true\n label: Replace consecutive spaces, newlines and tabs.\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: clean_1\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: null\n label: Delete all URLs and email addresses.\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: ''\n type: checkbox\n unit: null\n variable: clean_2\n", + "graph": { + "edges": [ + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "knowledge-index" + }, + "id": "1751336942081-source-1750400198569-target", + "selected": false, + "source": "1751336942081", + "sourceHandle": "source", + "target": "1750400198569", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "llm", + "targetType": "tool" + }, + "id": "1758002850987-source-1751336942081-target", + "source": "1758002850987", + "sourceHandle": "source", + "target": "1751336942081", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInIteration": false, + "isInLoop": false, + "sourceType": "datasource", + "targetType": "tool" + }, + "id": "1756915693835-source-1758027159239-target", + "source": "1756915693835", + "sourceHandle": "source", + "target": "1758027159239", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "llm" + }, + "id": "1758027159239-source-1758002850987-target", + "source": "1758027159239", + "sourceHandle": "source", + "target": "1758002850987", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + } + ], + "nodes": [ + { + "data": { + "chunk_structure": "hierarchical_model", + "embedding_model": "jina-embeddings-v2-base-en", + "embedding_model_provider": "langgenius\/jina\/jina", + "index_chunk_variable_selector": [ + "1751336942081", + "result" + ], + "indexing_technique": "high_quality", + "keyword_number": 10, + "retrieval_model": { + "reranking_enable": true, + "reranking_mode": "reranking_model", + "reranking_model": { + "reranking_model_name": "jina-reranker-v1-base-en", + "reranking_provider_name": "langgenius\/jina\/jina" + }, + "score_threshold": 0, + "score_threshold_enabled": false, + "search_method": "hybrid_search", + "top_k": 3, + "weights": null + }, + "selected": false, + "title": "Knowledge Base", + "type": "knowledge-index" + }, + "height": 114, + "id": "1750400198569", + "position": { + "x": 474.7618603027596, + "y": 282 + }, + "positionAbsolute": { + "x": 474.7618603027596, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 458, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently we support 5 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\": File Upload, Text Input, Online Drive, Online Doc, and Web Crawler. Different types of Data Sources have different input and output types. The output of File Upload and Online Drive are files, while the output of Online Doc and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A Knowledge Pipeline can have multiple data sources. Each data source can be selected more than once with different settings. Each added data source is a tab on the add file interface. However, each time the user can only select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 458, + "id": "1751264451381", + "position": { + "x": -893.2836123260277, + "y": 378.2537898330178 + }, + "positionAbsolute": { + "x": -893.2836123260277, + "y": 378.2537898330178 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 260, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" starts with Data Source as the starting node and ends with the knowledge base node. The general steps are: import documents from the data source \u2192 use extractor to extract document content \u2192 split and clean content into structured chunks \u2192 store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The user input variables required by the Knowledge Pipeline node must be predefined and managed via the Input Field section located in the top-right corner of the orchestration canvas. It determines what input fields the end users will see and need to fill in when importing files to the knowledge base through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique Inputs: Input fields defined here are only available to the selected data source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global Inputs: These input fields are shared across all subsequent nodes after the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\"},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\".\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 1182 + }, + "height": 260, + "id": "1751266376760", + "position": { + "x": -704.0614991386192, + "y": -73.30453110517956 + }, + "positionAbsolute": { + "x": -704.0614991386192, + "y": -73.30453110517956 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 1182 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 304, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"MinerU\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is an advanced open-source document extractor designed specifically to convert complex, unstructured documents\u2014such as PDFs, Word files, and PPTs\u2014into high-quality, machine-readable formats like Markdown and JSON. MinerU addresses challenges in document parsing such as layout detection, formula recognition, and multi-language support, which are critical for generating high-quality training corpora for LLMs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 304, + "id": "1751266402561", + "position": { + "x": -555.2228329530462, + "y": 592.0458661166498 + }, + "positionAbsolute": { + "x": -555.2228329530462, + "y": 592.0458661166498 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 554, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" addresses the dilemma of context and precision by leveraging a two-tier hierarchical approach that effectively balances the trade-off between accurate matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Query Matching with Child Chunks: Small, focused pieces of information, often as concise as a single sentence within a paragraph, are used to match the user's query. These child chunks enable precise and relevant initial retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Contextual Enrichment with Parent Chunks: Larger, encompassing sections\u2014such as a paragraph, a section, or even an entire document\u2014that include the matched child chunks are then retrieved. These parent chunks provide comprehensive context for the Language Model (LLM).\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 554, + "id": "1751266447821", + "position": { + "x": 153.2996965006646, + "y": 378.2537898330178 + }, + "positionAbsolute": { + "x": 153.2996965006646, + "y": 378.2537898330178 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 411, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The knowledge base provides two indexing methods:\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", each with different retrieval strategies. High-Quality mode uses embeddings for vectorization and supports vector, full-text, and hybrid retrieval, offering more accurate results but higher resource usage. Economical mode uses keyword-based inverted indexing with no token consumption but lower accuracy; upgrading to High-Quality is possible, but downgrading requires creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"* Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0only support the\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0indexing method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 411, + "id": "1751266580099", + "position": { + "x": 482.3389174180554, + "y": 437.9839361130071 + }, + "positionAbsolute": { + "x": 482.3389174180554, + "y": 437.9839361130071 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "result": { + "description": "Parent child chunks result", + "items": { + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "", + "ja_JP": "", + "pt_BR": "", + "zh_Hans": "" + }, + "label": { + "en_US": "Input Content", + "ja_JP": "Input Content", + "pt_BR": "Conte\u00fado de Entrada", + "zh_Hans": "\u8f93\u5165\u6587\u672c" + }, + "llm_description": "The text you want to chunk.", + "max": null, + "min": null, + "name": "input_text", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": "paragraph", + "form": "llm", + "human_description": { + "en_US": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "ja_JP": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "pt_BR": "Dividir texto em par\u00e1grafos com base no separador e no comprimento m\u00e1ximo do bloco, usando o texto dividido como bloco pai ou documento completo como bloco pai e diretamente recuper\u00e1-lo.", + "zh_Hans": "\u6839\u636e\u5206\u9694\u7b26\u548c\u6700\u5927\u5757\u957f\u5ea6\u5c06\u6587\u672c\u62c6\u5206\u4e3a\u6bb5\u843d\uff0c\u4f7f\u7528\u62c6\u5206\u6587\u672c\u4f5c\u4e3a\u68c0\u7d22\u7684\u7236\u5757\u6216\u6574\u4e2a\u6587\u6863\u7528\u4f5c\u7236\u5757\u5e76\u76f4\u63a5\u68c0\u7d22\u3002" + }, + "label": { + "en_US": "Parent Mode", + "ja_JP": "Parent Mode", + "pt_BR": "Modo Pai", + "zh_Hans": "\u7236\u5757\u6a21\u5f0f" + }, + "llm_description": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "max": null, + "min": null, + "name": "parent_mode", + "options": [ + { + "label": { + "en_US": "Paragraph", + "ja_JP": "Paragraph", + "pt_BR": "Par\u00e1grafo", + "zh_Hans": "\u6bb5\u843d" + }, + "value": "paragraph" + }, + { + "label": { + "en_US": "Full Document", + "ja_JP": "Full Document", + "pt_BR": "Documento Completo", + "zh_Hans": "\u5168\u6587" + }, + "value": "full_doc" + } + ], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": "\n\n", + "form": "llm", + "human_description": { + "en_US": "Separator used for chunking", + "ja_JP": "Separator used for chunking", + "pt_BR": "Separador usado para divis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5206\u5757\u7684\u5206\u9694\u7b26" + }, + "label": { + "en_US": "Parent Delimiter", + "ja_JP": "Parent Delimiter", + "pt_BR": "Separador de Pai", + "zh_Hans": "\u7236\u5757\u5206\u9694\u7b26" + }, + "llm_description": "The separator used to split chunks", + "max": null, + "min": null, + "name": "separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 1024, + "form": "llm", + "human_description": { + "en_US": "Maximum length for chunking", + "ja_JP": "Maximum length for chunking", + "pt_BR": "Comprimento m\u00e1ximo para divis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6" + }, + "label": { + "en_US": "Maximum Parent Chunk Length", + "ja_JP": "Maximum Parent Chunk Length", + "pt_BR": "Comprimento M\u00e1ximo do Bloco Pai", + "zh_Hans": "\u6700\u5927\u7236\u5757\u957f\u5ea6" + }, + "llm_description": "Maximum length allowed per chunk", + "max": null, + "min": null, + "name": "max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": ". ", + "form": "llm", + "human_description": { + "en_US": "Separator used for subchunking", + "ja_JP": "Separator used for subchunking", + "pt_BR": "Separador usado para subdivis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5b50\u5206\u5757\u7684\u5206\u9694\u7b26" + }, + "label": { + "en_US": "Child Delimiter", + "ja_JP": "Child Delimiter", + "pt_BR": "Separador de Subdivis\u00e3o", + "zh_Hans": "\u5b50\u5206\u5757\u5206\u9694\u7b26" + }, + "llm_description": "The separator used to split subchunks", + "max": null, + "min": null, + "name": "subchunk_separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 512, + "form": "llm", + "human_description": { + "en_US": "Maximum length for subchunking", + "ja_JP": "Maximum length for subchunking", + "pt_BR": "Comprimento m\u00e1ximo para subdivis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5b50\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6" + }, + "label": { + "en_US": "Maximum Child Chunk Length", + "ja_JP": "Maximum Child Chunk Length", + "pt_BR": "Comprimento M\u00e1ximo de Subdivis\u00e3o", + "zh_Hans": "\u5b50\u5206\u5757\u6700\u5927\u957f\u5ea6" + }, + "llm_description": "Maximum length allowed per subchunk", + "max": null, + "min": null, + "name": "subchunk_max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove consecutive spaces, newlines and tabs", + "ja_JP": "Whether to remove consecutive spaces, newlines and tabs", + "pt_BR": "Se deve remover espa\u00e7os extras no texto", + "zh_Hans": "\u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26" + }, + "label": { + "en_US": "Replace consecutive spaces, newlines and tabs", + "ja_JP": "Replace consecutive spaces, newlines and tabs", + "pt_BR": "Substituir espa\u00e7os consecutivos, novas linhas e guias", + "zh_Hans": "\u66ff\u6362\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26" + }, + "llm_description": "Whether to remove consecutive spaces, newlines and tabs", + "max": null, + "min": null, + "name": "remove_extra_spaces", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove URLs and emails in the text", + "ja_JP": "Whether to remove URLs and emails in the text", + "pt_BR": "Se deve remover URLs e e-mails no texto", + "zh_Hans": "\u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740" + }, + "label": { + "en_US": "Delete all URLs and email addresses", + "ja_JP": "Delete all URLs and email addresses", + "pt_BR": "Remover todas as URLs e e-mails", + "zh_Hans": "\u5220\u9664\u6240\u6709URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740" + }, + "llm_description": "Whether to remove URLs and emails in the text", + "max": null, + "min": null, + "name": "remove_urls_emails", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + } + ], + "params": { + "input_text": "", + "max_length": "", + "parent_mode": "", + "remove_extra_spaces": "", + "remove_urls_emails": "", + "separator": "", + "subchunk_max_length": "", + "subchunk_separator": "" + }, + "provider_id": "langgenius\/parentchild_chunker\/parentchild_chunker", + "provider_name": "langgenius\/parentchild_chunker\/parentchild_chunker", + "provider_type": "builtin", + "selected": false, + "title": "Parent-child Chunker", + "tool_configurations": {}, + "tool_description": "Process documents into parent-child chunk structures", + "tool_label": "Parent-child Chunker", + "tool_name": "parentchild_chunker", + "tool_node_version": "2", + "tool_parameters": { + "input_text": { + "type": "mixed", + "value": "{{#1758002850987.text#}}" + }, + "max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Maximum_Parent_Length" + ] + }, + "parent_mode": { + "type": "variable", + "value": [ + "rag", + "shared", + "Parent_Mode" + ] + }, + "remove_extra_spaces": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_1" + ] + }, + "remove_urls_emails": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_2" + ] + }, + "separator": { + "type": "mixed", + "value": "{{#rag.shared.Parent_Delimiter#}}" + }, + "subchunk_max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Maximum_Child_Length" + ] + }, + "subchunk_separator": { + "type": "mixed", + "value": "{{#rag.shared.Child_Delimiter#}}" + } + }, + "type": "tool" + }, + "height": 52, + "id": "1751336942081", + "position": { + "x": 144.55897745117755, + "y": 282 + }, + "positionAbsolute": { + "x": 144.55897745117755, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 446, + "selected": true, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"In this step, the LLM is responsible for enriching and reorganizing content, along with images and tables. The goal is to maintain the integrity of image URLs and tables while providing contextual descriptions and summaries to enhance understanding. The content should be structured into well-organized paragraphs, using double newlines to separate them. The LLM should enrich the document by adding relevant descriptions for images and extracting key insights from tables, ensuring the content remains easy to retrieve within a Retrieval-Augmented Generation (RAG) system. The final output should preserve the original structure, making it more accessible for knowledge retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 446, + "id": "1753967810859", + "position": { + "x": -176.67459682201036, + "y": 405.2790698865377 + }, + "positionAbsolute": { + "x": -176.67459682201036, + "y": 405.2790698865377 + }, + "selected": true, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "File", + "datasource_name": "upload-file", + "datasource_parameters": {}, + "fileExtensions": [ + "pdf", + "doc", + "docx", + "pptx", + "ppt", + "jpg", + "png", + "jpeg" + ], + "plugin_id": "langgenius\/file", + "provider_name": "file", + "provider_type": "local_file", + "selected": false, + "title": "File", + "type": "datasource" + }, + "height": 52, + "id": "1756915693835", + "position": { + "x": -893.2836123260277, + "y": 282 + }, + "positionAbsolute": { + "x": -893.2836123260277, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "context": { + "enabled": false, + "variable_selector": [] + }, + "model": { + "completion_params": { + "temperature": 0.7 + }, + "mode": "chat", + "name": "claude-3-5-sonnet-20240620", + "provider": "langgenius\/anthropic\/anthropic" + }, + "prompt_template": [ + { + "id": "beb97761-d30d-4549-9b67-de1b8292e43d", + "role": "system", + "text": "You are an AI document assistant. \nYour tasks are:\nEnrich the content contextually:\nAdd meaningful descriptions for each image.\nSummarize key information from each table.\nOutput the enriched content\u00a0with clear annotations showing the\u00a0corresponding image and table positions, so the text can later be aligned back into the original document. Preserve any ![image] URLs from the input text.\nYou will receive two inputs:\nThe file and text\u00a0(may contain images url and tables).\nThe final output should be a\u00a0single, enriched version of the original document with ![image] url preserved.\nGenerate output directly without saying words like: Here's the enriched version of the original text with the image description inserted." + }, + { + "id": "f92ef0cd-03a7-48a7-80e8-bcdc965fb399", + "role": "user", + "text": "The file is {{#1756915693835.file#}} and the text are\u00a0{{#1758027159239.text#}}." + } + ], + "selected": false, + "title": "LLM", + "type": "llm", + "vision": { + "configs": { + "detail": "high", + "variable_selector": [ + "1756915693835", + "file" + ] + }, + "enabled": true + } + }, + "height": 88, + "id": "1758002850987", + "position": { + "x": -176.67459682201036, + "y": 282 + }, + "positionAbsolute": { + "x": -176.67459682201036, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "The file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "ja_JP": "\u89e3\u6790\u3059\u308b\u30d5\u30a1\u30a4\u30eb(pdf\u3001ppt\u3001pptx\u3001doc\u3001docx\u3001png\u3001jpg\u3001jpeg\u3092\u30b5\u30dd\u30fc\u30c8)", + "pt_BR": "The file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "zh_Hans": "\u7528\u4e8e\u89e3\u6790\u7684\u6587\u4ef6(\u652f\u6301 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)" + }, + "label": { + "en_US": "file", + "ja_JP": "file", + "pt_BR": "file", + "zh_Hans": "file" + }, + "llm_description": "The file to be parsed (support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "max": null, + "min": null, + "name": "file", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "file" + }, + { + "auto_generate": null, + "default": "auto", + "form": "form", + "human_description": { + "en_US": "(For local deployment v1 and v2) Parsing method, can be auto, ocr, or txt. Default is auto. If results are not satisfactory, try ocr", + "ja_JP": "\uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v1\u3068v2\u7528\uff09\u89e3\u6790\u65b9\u6cd5\u306f\u3001auto\u3001ocr\u3001\u307e\u305f\u306ftxt\u306e\u3044\u305a\u308c\u304b\u3067\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fauto\u3067\u3059\u3002\u7d50\u679c\u304c\u6e80\u8db3\u3067\u304d\u306a\u3044\u5834\u5408\u306f\u3001ocr\u3092\u8a66\u3057\u3066\u304f\u3060\u3055\u3044", + "pt_BR": "(For local deployment v1 and v2) Parsing method, can be auto, ocr, or txt. Default is auto. If results are not satisfactory, try ocr", + "zh_Hans": "\uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72v1\u548cv2\u7248\u672c\uff09\u89e3\u6790\u65b9\u6cd5\uff0c\u53ef\u4ee5\u662fauto, ocr, \u6216 txt\u3002\u9ed8\u8ba4\u662fauto\u3002\u5982\u679c\u7ed3\u679c\u4e0d\u7406\u60f3\uff0c\u8bf7\u5c1d\u8bd5ocr" + }, + "label": { + "en_US": "parse method", + "ja_JP": "\u89e3\u6790\u65b9\u6cd5", + "pt_BR": "parse method", + "zh_Hans": "\u89e3\u6790\u65b9\u6cd5" + }, + "llm_description": "(For local deployment v1 and v2) Parsing method, can be auto, ocr, or txt. Default is auto. If results are not satisfactory, try ocr", + "max": null, + "min": null, + "name": "parse_method", + "options": [ + { + "icon": "", + "label": { + "en_US": "auto", + "ja_JP": "auto", + "pt_BR": "auto", + "zh_Hans": "auto" + }, + "value": "auto" + }, + { + "icon": "", + "label": { + "en_US": "ocr", + "ja_JP": "ocr", + "pt_BR": "ocr", + "zh_Hans": "ocr" + }, + "value": "ocr" + }, + { + "icon": "", + "label": { + "en_US": "txt", + "ja_JP": "txt", + "pt_BR": "txt", + "zh_Hans": "txt" + }, + "value": "txt" + } + ], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": 1, + "form": "form", + "human_description": { + "en_US": "(For official API and local deployment v2) Whether to enable formula recognition", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\u3068\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b", + "pt_BR": "(For official API and local deployment v2) Whether to enable formula recognition", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\u548c\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u662f\u5426\u5f00\u542f\u516c\u5f0f\u8bc6\u522b" + }, + "label": { + "en_US": "Enable formula recognition", + "ja_JP": "\u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b", + "pt_BR": "Enable formula recognition", + "zh_Hans": "\u5f00\u542f\u516c\u5f0f\u8bc6\u522b" + }, + "llm_description": "(For official API and local deployment v2) Whether to enable formula recognition", + "max": null, + "min": null, + "name": "enable_formula", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": 1, + "form": "form", + "human_description": { + "en_US": "(For official API and local deployment v2) Whether to enable table recognition", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\u3068\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b", + "pt_BR": "(For official API and local deployment v2) Whether to enable table recognition", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\u548c\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u662f\u5426\u5f00\u542f\u8868\u683c\u8bc6\u522b" + }, + "label": { + "en_US": "Enable table recognition", + "ja_JP": "\u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b", + "pt_BR": "Enable table recognition", + "zh_Hans": "\u5f00\u542f\u8868\u683c\u8bc6\u522b" + }, + "llm_description": "(For official API and local deployment v2) Whether to enable table recognition", + "max": null, + "min": null, + "name": "enable_table", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": "auto", + "form": "form", + "human_description": { + "en_US": "(For official API and local deployment v2) Specify document language, default ch, can be set to auto(local deployment need to specify the language, default ch), other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\u3068\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u6307\u5b9a\u3057\u307e\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fch\u3067\u3001auto\u306b\u8a2d\u5b9a\u3067\u304d\u307e\u3059\u3002auto\u306e\u5834\u5408\uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8\u3067\u306f\u8a00\u8a9e\u3092\u6307\u5b9a\u3059\u308b\u5fc5\u8981\u304c\u3042\u308a\u307e\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fch\u3067\u3059\uff09\u3001\u30e2\u30c7\u30eb\u306f\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u81ea\u52d5\u7684\u306b\u8b58\u5225\u3057\u307e\u3059\u3002\u4ed6\u306e\u30aa\u30d7\u30b7\u30e7\u30f3\u5024\u30ea\u30b9\u30c8\u306b\u3064\u3044\u3066\u306f\u3001\u6b21\u3092\u53c2\u7167\u3057\u3066\u304f\u3060\u3055\u3044\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5", + "pt_BR": "(For official API and local deployment v2) Specify document language, default ch, can be set to auto(local deployment need to specify the language, default ch), other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5", + "zh_Hans": "\uff08\u4ec5\u9650\u5b98\u65b9api\u548c\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u6307\u5b9a\u6587\u6863\u8bed\u8a00\uff0c\u9ed8\u8ba4 ch\uff0c\u53ef\u4ee5\u8bbe\u7f6e\u4e3aauto\uff0c\u5f53\u4e3aauto\u65f6\u6a21\u578b\u4f1a\u81ea\u52a8\u8bc6\u522b\u6587\u6863\u8bed\u8a00\uff08\u672c\u5730\u90e8\u7f72\u9700\u8981\u6307\u5b9a\u660e\u786e\u7684\u8bed\u8a00\uff0c\u9ed8\u8ba4ch\uff09\uff0c\u5176\u4ed6\u53ef\u9009\u503c\u5217\u8868\u8be6\u89c1\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5" + }, + "label": { + "en_US": "Document language", + "ja_JP": "\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e", + "pt_BR": "Document language", + "zh_Hans": "\u6587\u6863\u8bed\u8a00" + }, + "llm_description": "(For official API and local deployment v2) Specify document language, default ch, can be set to auto(local deployment need to specify the language, default ch), other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/version3.x\/pipeline_usage\/OCR.html#5", + "max": null, + "min": null, + "name": "language", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 0, + "form": "form", + "human_description": { + "en_US": "(For official API) Whether to enable OCR recognition", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b", + "pt_BR": "(For official API) Whether to enable OCR recognition", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542fOCR\u8bc6\u522b" + }, + "label": { + "en_US": "Enable OCR recognition", + "ja_JP": "OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b", + "pt_BR": "Enable OCR recognition", + "zh_Hans": "\u5f00\u542fOCR\u8bc6\u522b" + }, + "llm_description": "(For official API) Whether to enable OCR recognition", + "max": null, + "min": null, + "name": "enable_ocr", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": "[]", + "form": "form", + "human_description": { + "en_US": "(For official API) Example: [\"docx\",\"html\"], markdown, json are the default export formats, no need to set, this parameter only supports one or more of docx, html, latex", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09\u4f8b\uff1a[\"docx\",\"html\"]\u3001markdown\u3001json\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f\u3067\u3042\u308a\u3001\u8a2d\u5b9a\u3059\u308b\u5fc5\u8981\u306f\u3042\u308a\u307e\u305b\u3093\u3002\u3053\u306e\u30d1\u30e9\u30e1\u30fc\u30bf\u306f\u3001docx\u3001html\u3001latex\u306e3\u3064\u306e\u5f62\u5f0f\u306e\u3044\u305a\u308c\u304b\u307e\u305f\u306f\u8907\u6570\u306e\u307f\u3092\u30b5\u30dd\u30fc\u30c8\u3057\u307e\u3059", + "pt_BR": "(For official API) Example: [\"docx\",\"html\"], markdown, json are the default export formats, no need to set, this parameter only supports one or more of docx, html, latex", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u793a\u4f8b\uff1a[\"docx\",\"html\"],markdown\u3001json\u4e3a\u9ed8\u8ba4\u5bfc\u51fa\u683c\u5f0f\uff0c\u65e0\u987b\u8bbe\u7f6e\uff0c\u8be5\u53c2\u6570\u4ec5\u652f\u6301docx\u3001html\u3001latex\u4e09\u79cd\u683c\u5f0f\u4e2d\u7684\u4e00\u4e2a\u6216\u591a\u4e2a" + }, + "label": { + "en_US": "Extra export formats", + "ja_JP": "\u8ffd\u52a0\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f", + "pt_BR": "Extra export formats", + "zh_Hans": "\u989d\u5916\u5bfc\u51fa\u683c\u5f0f" + }, + "llm_description": "(For official API) Example: [\"docx\",\"html\"], markdown, json are the default export formats, no need to set, this parameter only supports one or more of docx, html, latex", + "max": null, + "min": null, + "name": "extra_formats", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": "pipeline", + "form": "form", + "human_description": { + "en_US": "(For local deployment v2) Example: pipeline, vlm-transformers, vlm-sglang-engine, vlm-sglang-client, default is pipeline", + "ja_JP": "\uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528\uff09\u4f8b\uff1apipeline\u3001vlm-transformers\u3001vlm-sglang-engine\u3001vlm-sglang-client\u3001\u30c7\u30d5\u30a9\u30eb\u30c8\u306fpipeline", + "pt_BR": "(For local deployment v2) Example: pipeline, vlm-transformers, vlm-sglang-engine, vlm-sglang-client, default is pipeline", + "zh_Hans": "\uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72v2\u7248\u672c\uff09\u793a\u4f8b\uff1apipeline\u3001vlm-transformers\u3001vlm-sglang-engine\u3001vlm-sglang-client\uff0c\u9ed8\u8ba4\u503c\u4e3apipeline" + }, + "label": { + "en_US": "Backend type", + "ja_JP": "\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\u30bf\u30a4\u30d7", + "pt_BR": "Backend type", + "zh_Hans": "\u89e3\u6790\u540e\u7aef" + }, + "llm_description": "(For local deployment v2) Example: pipeline, vlm-transformers, vlm-sglang-engine, vlm-sglang-client, default is pipeline", + "max": null, + "min": null, + "name": "backend", + "options": [ + { + "icon": "", + "label": { + "en_US": "pipeline", + "ja_JP": "pipeline", + "pt_BR": "pipeline", + "zh_Hans": "pipeline" + }, + "value": "pipeline" + }, + { + "icon": "", + "label": { + "en_US": "vlm-transformers", + "ja_JP": "vlm-transformers", + "pt_BR": "vlm-transformers", + "zh_Hans": "vlm-transformers" + }, + "value": "vlm-transformers" + }, + { + "icon": "", + "label": { + "en_US": "vlm-sglang-engine", + "ja_JP": "vlm-sglang-engine", + "pt_BR": "vlm-sglang-engine", + "zh_Hans": "vlm-sglang-engine" + }, + "value": "vlm-sglang-engine" + }, + { + "icon": "", + "label": { + "en_US": "vlm-sglang-client", + "ja_JP": "vlm-sglang-client", + "pt_BR": "vlm-sglang-client", + "zh_Hans": "vlm-sglang-client" + }, + "value": "vlm-sglang-client" + } + ], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": "", + "form": "form", + "human_description": { + "en_US": "(For local deployment v2 when backend is vlm-sglang-client) Example: http:\/\/127.0.0.1:8000, default is empty", + "ja_JP": "\uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8v2\u7528 \u89e3\u6790\u5f8c\u7aef\u304cvlm-sglang-client\u306e\u5834\u5408\uff09\u4f8b\uff1ahttp:\/\/127.0.0.1:8000\u3001\u30c7\u30d5\u30a9\u30eb\u30c8\u306f\u7a7a", + "pt_BR": "(For local deployment v2 when backend is vlm-sglang-client) Example: http:\/\/127.0.0.1:8000, default is empty", + "zh_Hans": "\uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72v2\u7248\u672c \u89e3\u6790\u540e\u7aef\u4e3avlm-sglang-client\u65f6\uff09\u793a\u4f8b\uff1ahttp:\/\/127.0.0.1:8000\uff0c\u9ed8\u8ba4\u503c\u4e3a\u7a7a" + }, + "label": { + "en_US": "sglang-server url", + "ja_JP": "sglang-server\u30a2\u30c9\u30ec\u30b9", + "pt_BR": "sglang-server url", + "zh_Hans": "sglang-server\u5730\u5740" + }, + "llm_description": "(For local deployment v2 when backend is vlm-sglang-client) Example: http:\/\/127.0.0.1:8000, default is empty", + "max": null, + "min": null, + "name": "sglang_server_url", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + } + ], + "params": { + "backend": "", + "enable_formula": "", + "enable_ocr": "", + "enable_table": "", + "extra_formats": "", + "file": "", + "language": "", + "parse_method": "", + "sglang_server_url": "" + }, + "provider_id": "langgenius\/mineru\/mineru", + "provider_name": "langgenius\/mineru\/mineru", + "provider_type": "builtin", + "selected": false, + "title": "Parse File", + "tool_configurations": { + "backend": { + "type": "constant", + "value": "pipeline" + }, + "enable_formula": { + "type": "constant", + "value": 1 + }, + "enable_ocr": { + "type": "constant", + "value": true + }, + "enable_table": { + "type": "constant", + "value": 1 + }, + "extra_formats": { + "type": "mixed", + "value": "[]" + }, + "language": { + "type": "mixed", + "value": "auto" + }, + "parse_method": { + "type": "constant", + "value": "auto" + }, + "sglang_server_url": { + "type": "mixed", + "value": "" + } + }, + "tool_description": "a tool for parsing text, tables, and images, supporting multiple formats such as pdf, pptx, docx, etc. supporting multiple languages such as English, Chinese, etc.", + "tool_label": "Parse File", + "tool_name": "parse-file", + "tool_node_version": "2", + "tool_parameters": { + "file": { + "type": "variable", + "value": [ + "1756915693835", + "file" + ] + } + }, + "type": "tool" + }, + "height": 270, + "id": "1758027159239", + "position": { + "x": -544.9739996945534, + "y": 282 + }, + "positionAbsolute": { + "x": -544.9739996945534, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + } + ], + "viewport": { + "x": 679.9701291615181, + "y": -191.49392257836791, + "zoom": 0.8239704766223018 + } + }, + "icon_info": { + "icon": "e642577f-da15-4c03-81b9-c9dec9189a3c", + "icon_background": null, + "icon_type": "image", + "icon_url": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAAP9UlEQVR4Ae2dTXPbxhnHdwFRr5ZN2b1kJraouk57i\/IJrJx6jDPT9Fpnkrvj3DOOv0DsXDvJxLk2nUnSW09hPkGc6aWdOBEtpZNLE9Gy3iiSQJ\/\/gg8DQnyFFiAAPjtDLbAA9uWPn5595VKrjLjtn\/YqrZaq+L6quL5X9pQqO1qtI3u+0mXy8MFJxfihP1qrss\/XQ+FFPtRK1UmreriMJkz\/GqaVX8N1z1dPHdyvnZpP1+fmVG3jhTVzDden6SjP6brt7b1y21VbWnk3CawKAbWp9Fmo0s3VbKamffWYgKz5vv+t1s5jt62qGxtrPVAnrUwqAH63u7dF\/4E3qaBbVCB8zjjHcZRDJs91XaXJpOGDMDgSx5zj2HWDMByz4\/v5fBZ80lLhE3Y498jcsfO8Nt1DlYbvmXs9L\/DbbY\/uozqmjwOUSvvVtuN8+tKLa4\/73GI1KDEAYek8x7vta\/0a5XiLcw1Y5uZcAxpgK5VKXeD4HvHTUaDdbivA2Go1yW+rZrPVkzDBUSOk7\/\/u2m8e9VyweGIdQAPenLpD\/3LvcLsM0C0szBNs8wY+nIvLpgKA8PS0YWBkKwkQyUo8un517b7tXFsl4cnO\/25p33lA7YoKMloqzanFxSXj2864xJe8Ao3GaRdGpAYQbVtEKwCS1au0Xf8TyuMWMirgQYXiOFjFw8PDcLvxC7ek79roSZ8bwO3dvTue77+P6hZV69LSElm9heKoLyXpKgCLeHx8zCBSb9m7e972YWwATVvPVfeoL\/YOcjg\/X1IrKyvd3mo313JQKAXQLgSEgBGO3v\/DG9eu3I1byFgAosr1HP9zauttitWLK32+nzs5aRgQMfSDoRtnXr8ep0qeGMAOfF+ho4FxuosXV7vjdfmWVHI\/qQKwhvv7z02VTCDVnJJ+dVIIJwIwDB\/G8FZXLwh8k761gt0PCJ8\/PzDjiHEgHBvAKHywfDKeVzCaYhYH1TAsIQazJ4VwLAAFvphvZoYeiwvh2YnVPqJ1OhwVVLti+foIJEGmNgQbYISG5Creqf85Ga7yKGlGAvj9zh5mNjbR4UCbT6rdUZLO7nWwwf0CMNNyvXuj1BhaBdPU2m2lnE8Q8aVLF6XDMUpNuW4UQMfk2bN9swKHqua7N9avPBwkzUAATbvP9b\/BDMfy8rLMbgxSUML7KoBxwqOjI1yr07TdK4OGZwZWwTS3+wDwYRWLTK311VgChygAZjA7Rq7cbpp1An3v7gtgUPWqW2j3YW5XnCgQR4HQ1OzWk529W\/3i6AsgLakyjUfAx6uS+z0sYaLAMAXQd2ADRt9PedCvV3wGwO939+7xNBuqX3GiwHkUQFWM5XnUnKu0HM8sXAnHdwZA+grVbdwA8ylOFLChABYlw5FFvBO1gj0Aou0H6wdi8REnCthQIMRTmazg7XCcPQBy229+XhaUhkWS4\/MrELKC+JJa13UB3P5xb1Pafl1d5MCyArCC6JSQ28LXdDn6LoD09bzbCJSql6UR37YC3U6t521x3F0AtaNvIlCqX5ZGfNsK4Gu5cGQJDWs4NgCiZ0JLujYRIBYQKohLQgFsSMDVMPeGDYBtt72FBAW+JGSXOFkBwAcI4bA\/EHwDoO9rY\/0cJ7iIC+JEgSQUwHpB4\/ygHWgAJDJfRiD2aREnCiSpAANodkajhDoAqgoS7bfzFMLFiQK2FGAjR7WxMXqdKjjogDCdthKTeESBqAKdTgiCK\/jjUG8kOOjsxYdAcaJAUgoAQF5hhV1xndacVL9JiS3x9leArSC2ZHa03y7jNg7s\/4iEigL2FOChGGIPAOoKosY2uOJEgTQUYGNHw39lB7vRI1HszyxOFEhDAQaQ0io7fqc3EgpMIw+SxgwrwJ0QRzvr3XpXAJxhIqZYdKp59TrSl2m4Kb6FGUuajR3trLvWtYAzpoEUd4oKcIeXhgQvCYBTfBGStFJzm\/\/EWkDqiiw1qR6W1TC7r11JlIurX\/6caPy5iJx+uUkd7SOrFYfgM8MwNBKYi7xLJoulgFTBxXqfuSuNAJi7V1asDM99+8fLpvYtly91VykUq4jDSzPtNpntNme0PLbjH67meFexf2C9Hmx8QMOAwVQcj82MF4XcJQrEVyDEmpmKk9Uw8bWUJ2Mo0ANgjOflEVHAmgLSCbEmpUQURwEBMI5q8ow1BQRAa1JKRHEUyAWAPx7Rj+I1afpGXOEUyAWAn+2cqI9\/aBROfCkQLT\/Iugiwfp\/tNtRH3x+LFcz6y4qRv8wDCOu3a6pgX6xgjBec9UcyDSBbPxZRrCArURw\/0wCy9WO595tiBVmLoviZBTBq\/VhwsYKsRDH8zAIYtX4st1hBVqIYfiYBHGT9WHKxgqxE\/v1MAjjI+rHcYgVZifz7mfo5pACsE\/XRDycjlYUVhPvT1QV1dTmT\/0cjyyA30LfisiBCFzwz2Ezf0BvD4ZkP\/n2k\/kbjhH++tiggjqFZFm+ZKoBxwIuKiPaigBhVJT\/n+snOL8bkXL68llqubYA3KLMvUnU8iUVM+zsU0fQGlaPw4Yd1U8RULWCS4PELE4vISuTDT7X1DgCxC8OlUvLJ\/pqWfOE+yyimagFRPb77h2VTRaLz8PfdU1po0Laqz8WSVm\/9dlG9fX1J4VhcthVIFUCWIgkQ8wqe7e\/tRtuYtuPnd3he\/5dfglpwKgBy5m2AmFfwWINZ96cKIIsfBfFjGohGG26YE\/CGqZOfa5kAkOViENFy++A\/wUwHX4v6b1Eb793fL0WD5TxnCiTfHY0hCOAa1oF4cdlVb9AUnLj8K3AuAD\/baSh8bDvA9zb1ZAe5N67J\/O8gbfIWHrsKBnjvfnPQLS+gsOlgBbEoIdoWFOtnU+XpxxXLAkbhA4i2LeEgKyjWb\/rQ2MzBxABG4ePMJAFhtC0o1o\/VLo4\/EYCD4GM5bEMYtYJi\/Vjp4vhjAzgKPpbENoRsBcX6scLF8sfqhIwLH0sDCOFsdEzYCvq0lausfGaFi+OPBHBS+FgamxDCCj4bMTPC6YqfLwWGAhgXPpbAFoSwgviIK54CA9uA54WPpbLdJuR4xS+GAn0BtAUfSyQQshLiRxU4A6Bt+DhBgZCVED+sQA+AScHHCQqErIT4rEAXwKTh4wQFQlZCfChgesH\/+G9DvfdDenswA0I4G+OEJiL5k1sFHAPfvw5TL4BYwtQlz2SCzntTgI+VEAhZidn1u23AaUkgEE5L+WykO3UAIYNAmA0YppGLTAAoEE7j1WcjzcwAKBBmA4i0c5EpAAXCtF\/\/9NPLHIAC4fShSDMHmQRQIEwTgemmlVkABcLpgpFW6pkGUCBMC4PppZN5AAXC6cGRRsq5AFAgTAOF6aSRGwAFwukAknSquQJQIEwah\/Tjzx2AAmH6kCSZYi4BFAiTRCLduHMLoECYLihJpUYA6uAna+j3O\/LoZClX\/t4afium4+oEoJ9rAFEQgZDfZz78MIB65a9PtinbFbV0USkn1zWyFfWT\/l2N6O94WMl03iLx6QtwR\/vIdU2Iy9vLK1h+BcCCvdC8FUcAzNsbK0J+u50QXcfvBX9FZdpaXV1VpdLQ3dqKUHQpQwYUaDZb6vnz58hJVSxgBl7ILGcBAJphmFDXeJb1kLKnrIDj+f4zpOmjayxOFEhBAc8LfiNaKy3DMCnoLUlEFOj2QSjcoZ2Xa7jueWIBoYO45BXg2tbzvaeY+zBtQM\/rzs8lnwNJYaYVCPU36k5bd+aClQA401SkWHiubbV2ao7Wbg1pt1pBwzDFfEhSM6oAW0Bfq7oz1wragBw4o5pIsVNUoN0O+htzc7QYYWNjrYa0YRYFwhTfwgwnxVXwxgtrnWEYX6zgDPOQatG5qad99RgJB1NxOjhpNpupZkYSmz0FeBCaKuGnKH0AoO+bE6Zz9mSREqelQKvV6iTlhy2gX0Uo09m5QzxRwLoC7XZnGk47vwLott0qUoIFlI6Idc0lwpACWIoF57ZVFb6pgqknjNmQKuCTahiyiEtCAYYPHZAOc502IKVG8H2NRE9PT5NIW+IUBYithlHBVwFrOAk6IebIqcITAKGCuCQUYAvoec4jjr8L4I2ra1UKNNUw38g3iS8KnFeBRqNhJjuw+uqljTXTAUGcXQBxon3\/S\/gnJ8fwxIkC1hTgmtVX+n440h4AHTKNRGgdFlCsYFgmOT6PAswTrN\/vrq09CsfVAyB6JrRE\/0PcIFYwLJMcn0eBw8Pg11iJrU+j8RCUvW57e6\/sOf43tFSmsry8pBYXF3tvkDNRYAIF0PY7PDxSsH7Xr13eiD7aYwFxEVbQ1\/oujo+PT2RgGkKIi6UAll2BIbho248jPAMgLlA9\/QV5pkd8cJD+j1lz5sTPtwJoxnWWXn0RbftxyfoCiItuW79JZpM6JE1qDwYU80PiiwKjFDg5aahG4xRVb90tBTVqv2cGAkhVcU35QZcZZpRXsfaLRMJEgbACQdUbDOVR1XsXC0\/D18PHAwHETdfX1x5SI\/BDzBFjLw+BMCydHPdTAIyAFbOohdgZVPXys2Qhh7tOr\/gr6hVvuq6rLl5cVVqPfGx4pHK1kAoAuv19GKo2TWqox9fXL78yqqBDLSAeRq\/Y8fTrFGENESMBQ\/eomOX6TCnQAx8NuTjz+vVxBBjblJElrND4ICxhRSzhONLOzj1n4CvpV4e1+8LKjA0gHopCeOHCBeW6I41oOD05LpgCaPMdHBwE1S4s3wTwQYqJAMQDYQgd2tgDG1sKhFBm9hx3ODDWRyBNDB8UmxhAPNSB8HN0TNAhWVpalCk7CDNDDuN8x8fHpj+ADgfafONWu2GZYgHIETx5+vND6hLfwfnCwjxBuCTWkMUpqI\/2HhYXnJ52vsJLQy2u57yPzmqcIp8LQCT4ZGfvtlb+A9raqIwqGdZwYWEhTl7kmYwr0GP1aIaDVrfcv7F+5eF5sn1uAJE4quS2qx7QlPMtnAPElZUV2fQcYhTAYT0f5nVDa0SrNL32ZpwqNyqHFQA5UmMNff8ehmoQhl335+fnxSKyQDnzo+ARLDVMrXUWq1gpjVUAOUffPf35fUfpvzCIsIgBjAtiFVmkDPpo3+Fruc3mqVlIgHM4gsQsVJ7znIdx23qDipsIgJxY1CJyOGDEYPYc7c\/lOPBdviR+SgoALnyw2gkzXPj02Zigqn39peOpR7bB42ImCiAnsv3j3iaNGVFnRd\/E0A2Hh31YSYwnYlgHx\/D5A0jZBdd7s8338T2z4DNA0bJibA4O+zCzBeOt93DOkPEWadHn6bxK931NL6Ha+aZkn1vsBfW+SXvxDoyJOixl6rBskUAYQ3yZxpAqg6AcGIlcsKMAtuXDzmjYnEo7VWyXkZSlG5Th1AEclJHtn\/YqtHFShYAsA0pPeWXawn8d91PDt0KecbiOIR8+h0\/G8kxY+HoRj+nF1cmg1c+UTQd7PVJ4nYbHzHXaf\/6po5x6m7bEJa1q2JnURg\/2TNoxAv4PoGedQHqhulIAAAAASUVORK5CYII=" + }, + "id": "103825d3-7018-43ae-bcf0-f3c001f3eb69", + "name": "Contextual Enrichment Using LLM" +}, +{ + "chunk_structure": "hierarchical_model", + "description": "This Knowledge Pipeline extracts images and tables from complex PDF documents for downstream processing.", + "export_data": "dependencies:\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius\/jina:0.0.8@d3a6766fbb80890d73fea7ea04803f3e1702c6e6bd621aafb492b86222a193dd\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius\/parentchild_chunker:0.0.7@ee9c253e7942436b4de0318200af97d98d094262f3c1a56edbe29dcb01fbc158\n- current_identifier: null\n type: marketplace\n value:\n marketplace_plugin_unique_identifier: langgenius\/mineru:0.5.0@ca04f2dceb4107e3adf24839756954b7c5bcb7045d035dbab5821595541c093d\nkind: rag_pipeline\nrag_pipeline:\n description: ''\n icon: 87426868-91d6-4774-a535-5fd4595a77b3\n icon_background: null\n icon_type: image\n icon_url: data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAARwElEQVR4Ae1dvXPcxhVfLMAP0RR1pL7MGVu8G7sXXdszotNYne1x6kgpktZSiiRNIrtMilgqnNZSb4\/lzm4i5i8w1TvDE+UZyZIlnihKOvIAbN5v7\/aIw93xPvBBHPDezBHYBbC7+O2Pb9++\/YAlMiIPHjwoO65btpQqK6VKVKySsqwV9fQpSliy6IcTubhYxrFTrJJqXe+Mz2+I8KgJoeh3IIRBTW1vt+MoXLWWlgRheo\/uqlmWVSVMa67jVJeXl6sHTx7dGb1HurK9uVnybHtNKXFBWAKEW1XCKvcrhb+tCdi+LBeX2ud80o3AaHipDUGkFErdJXJu2J63vliptAncnXr8MakQ8PH9+2tU9Av0omtCCZx3iZSSsLCE49j6iHPE+U+fCEnnCEOmTp\/uehbXzPWuizmNoFaC4CQdFxCE3V9\/bcd4vk8txpLwW\/f6FPZ9RT8c\/fZ9nSdESmGtK1veOvPGG3SerCRGQGg6V8rLxIwPg6QDUWzb1kTDcXrKaROu16v6T550RMuTJzvCHOhEYBS8PM8TIGmj4QrX9ejndiRG5Kj6lvj8zLlzNzsuxBiInYCaeI7zqeWrK8YuA+lmZqbF9PSUcIh0o2irUQCNEZeJTSoqXg0i4d7evial0ZIgopLWzdNvvvl53MDESsBfNrc+sqX6wth0juOIublZMUXHcSUqoOPmO6nPxYkXiFinn9GMIGLcGjEWApLWK7u2\/ZVpauMgniFAnICaNPN8TAIvaMXd3ZcHdqMlbjve1NXFSvSetIxaGU\/u3\/\/Uk\/aPIB+a1rm5Y+LEwnwkrRe1TPx8vAigBVssLYj51+Z0x5Dq+iNXNn58tLV1OWpOYxMQtt7jra0vqFd1HbYe7DsU8tjsTNQy8fMZRQB2PJQLjiQlS4mvwIEoxR2rCdZNrpTfUnd9FVrv2LHZxIiXRJMSBbCsP5sWXvX6nnj1qq5dPOQQ33D86Y\/HaZJH1oAgnyflHZAPfrrSieOJkS\/rlV3k8s1SS3eC6h4cABc82bizvfmgPComIxHQkA+9XPjwoI6bBRg1W74\/Dwig7sEBuNbIDCPFNDoJhyYgky8PlIn\/HUDChQgkHIqAvcg3ijM5\/tfmFLOEALgwLgmHIiANqX0bbHaZfFmq\/myUJUxCV+5\/S4qrNKh0AwnY7GY3OxwLx18baRhtUOZ8PV8IgITHiSOmY0KDE9cGveGhBHy0SY5GJa4gYe5wDIKSrwMB0zHBDCZw5+G9e1cOQ6YvAWH3kX2pnYzw8zVZfVhSfI0RaCIAroAzEJp6cu0w90xfApL6pEkFogSvN49uNIHlv8MjAD8hRsdISq7d+Krfkz0J2Gp6PwKT51pM7pcAxzMC\/RDQY8fNpnjtV5op1eu+ngSUUmnjEeTjprcXbBw3DALoO5imWJA516tX3EVAmt1yDS4XEK816DxMXnwPI9ATATTFmJ5H5lx5X8quDkkXAZXvX0ZK8\/NzPRPkSEZgVAQwKRlCq34+DWvBDgLC9oP2w\/yvKLOYdW78hxFoIQAuQQuSNNcJBZDpIKCx\/bjpDSDEp7EgYLQgjWR8GEywTcBHmz\/r9bls+wXh4fO4EIAWbDmn1x5v3l8z6bYJKKV3GZFTtEyShRFIAoHp5kxq4Ut\/zaTfJqAS8gIiufk10PAxbgRajmloQs01pK+n5KNn4kp7GxEnlwZOYMBtqUl4inlqGeckoywt5MfODbXajp7G7\/jeIrYB0RoQe7UAb+755oR1GX0NOKYlzZ6GGM5pAhIzVxFp074sLIxAkghg7x8I7VezhmPTBrSs8wiwBgQKLEkigLVEEIyM4Njs8iqLAtQNsdt9ElzLhGTJhskEIBNeCGxG9YLegaZpaaXXYlyzCcbqJhZGIEkEYAdCjAaUD2jiKSJ41gtQYEkaAd0RoYkuEOyKK2mMroyA3YrEOQsjkCQCRgs6dbcsaYtc7fizZFM1Jpkxp80IAAHTE7ZsVZbkgikjkptgoMCSBgJGAxL3SmiMmxqwZRymUQDOo9gIGAKCe9L0RgKRxUaH3z5xBExrS5xbaTv+9FSZxLPmDBiBTgSId9YKorLohO4sKofygoBRdp5Si20NmJeX4\/fIPgLG40JEPMEEzH595bqEtF7Ool4wLUWa0F7wr+\/\/JlMVdOrOfzrKY8p3\/C9\/FjMXL3ZcK2rADHrQHtPkiBa+dsOYdrmooCT93s\/\/8U+x9\/33SWczcelzE5xilYGEjY2NFHPMflZMwJTraOdvfxfuTz+lnGt2s3O8bb0URPheA+NxsZeU5\/N1Qqp2d8Wzq38SJ774l3DefrvzYgZDSazJ0V\/r3Hmu3xZTEHgoLuWKNyT0Hj5MOedsZBfo8OqhOCbgEdQLSLhDmrCIJOwg4BFgz1m2EAD5ikpCQwIHX9SGyJjWAydhM5jC5vFoSLhANqH9+uuZf8W4bHppNZd\/xN\/ryDyE2SugIWERm2MmYEb4aEgI27BIwgTMUG2DhDXqmBSJhEzADBEQRfHISV0kEjIBM0ZAQ0KMmBRBmIAZrWWMGWPsOO\/CBMxwDWP2TN5JyATMMAFRNJBw98t\/Z7yU4xePCTg+dqk9Wf\/6a\/Hy1q3U8kszIyZgmmhHyOvlzVu5JCETMAIp0n40jyRkAqbNooj55Y2ETMCIhDiKx0HCV19\/cxRZx54nEzB2SNNJ8MWXX+ZikRMTMB2+JJJLHnyE\/FmkRKhxkGh4nfDBFT4DAqwBmQdHigAT8Ejh58yZgMyBI0WAbcCY4Td7wcScbN\/kJt3GZA3Yt2r5QhoIMAHTQJnz6IsAE7AvNHwhDQSYgGmgzHn0RYAJ2BcavpAGAkzANFDmPPoiwATsCw1fSAOBifcDTrofLI1KznIerAGzXDsFKBsTsACVnOVXZAJmuXYKUDYmYAEqOcuvyATMcu0UoGxMwAJUcpZfkQmY5dopQNkmzg846nw7m77Fge9xzH7wgZhaPT+wSodN35qf1+kibef8eTHz3rsD0+51w7D59Xq2V9yk+UUnjoC9QD8sDhs+4odNfqZWV8U8fTQwjs3AsYsptlDTn96ivVt2iZDT770n5i79Lpb0D3unPF0rVBMMstT+8MdEPpUFQoLkSD8vi8bTIHqhCAhAQRR8KiupHemRPhaN53lLtTiJOfFN8CCbp7FxV9RJM+398EMbN5Bkl3YfxffaBkm\/9P2Hv2gSI2337t0uQmNLNeSD7wSPIv3yGyWNSbp34gk4CGx0PPCD3RfcY8\/Yb7ALxxH5+lmBn+nY7H3\/g04\/qFnRJDtvvSWO\/faTcbIoxDOFaYLnLl\/SnZBgrYI0ccnMxQ9Er68doTnmz7P2R7kwBAQE6KEGpUFNZ5wCLdubhPndYjcqfoUiYPj7vMHmMiqQ5nmQEK6eoKC5hz3I0o1AoQgI53EaArsybFvWY2zu03iHtPIoFAHRIw5KWCMGr0U9n363c2QEznCWbgQKRcB6wBUDKOTZs92IxBRjescmubjtTZPupB9z74YxFQQXDNwiQZm9eDEYjPU8PNznD2kDjjo2POl+w1wTEIa\/+9P\/tH9Oj9kGKAaCTI85gSCQTN\/TsL3JnZDeUE08AUfVGIAB5IC7hOXoESiUDQi4QT4MwYWbyLirIqzxwhox7vwmNb2J14CjAB\/ndKxB+aLpD8qwhJ90my74zsOc556Akmy9GXKJYK5euGc6DEDj3hMefkuyxz1uGbPw3MQTMKsao\/5N54dkZugfgKUbgcLZgN0QxB+DSQ7hYT5niOUA8Zck+yk6\/vZTXUpfedkv7QSUEMQLTvtCkWdoPcqwNmDWX9F\/8iSWIvq1Zzod1oCxwNlMBOTb6THbGlPBWHoj4FhC1JQQJaWUsCwKsYyFwCuy+fARwbD7Ze7Spdxov7GA6fEQuNaSmkOnNQowAQ0kQx4xJb9BEwwwHR\/T8sPEQzJoeln7dQPaQUB7cVGQ7hOytCCk5BY5DNc4Iy2GfMf\/+pdwchMXlidPxl9m3xfSniLWCTHxbpj40YmWIkY80OzyOpDhcGQCDofTwLtAvGOffKKJx8NuA+Fq38AEbEMx2glIBtfKFG3LgVEW5+239DjzaKkU826\/1QlRQtWsx1tbd8gIXFtYmBdTDvOxmJRI960brit2dmiNjCXWudeRLvacWwgBEBBuGKH8tm8mdAsHGYHkEJDkk9FjIgHfTHK5ccqMACHgeb7GgdwwVW6CmRLpI3AwEiIkWIgSeOQcZGEE0kCg3QtW6t6BDRhgZRqF4DyKi0DA3KtJy7eanRAmYHEZkfKb+8YGtKyqVI5VRf6uy\/MBU66HwmbXboI9qyZd160CiYBaLCww\/OLpIOC3+hvurFOVy5VKFdkikn2B6VRA0XMxBFxeXm66YSyhqgCFxuaKjg2\/f8IIuJ4x9dQGstKDv8qyaAM7UW40XDEzM51wEUZLPq41CKPlmp+7E5nPFwEe0wEhp989JKMd0Rb5YxA4YCdCLIxA\/AhgIgKEiKc1YHMkxLLWEelxTxgwsCSIgPG20PqjAwLanreOPKEBuSOSIPqcNLn7mhrQcE7bgIuVSo3mBa6TK2bN9T0xJbM7LzBrNk3WOJVlm9k0v9Td3QDngF2zCcaZUv\/FYX+\/gQMLIxA7Anv1fZ0m+Vo01xA4IKAv1xGxt9e8CecsjECcCLQ1oO\/fNOm2CXi68uY6pkhjRKR9o7mLj4xARASg2PRgB82+OlOp6A4IkmwTUKev1Hc4vnpZ10H+wwjEhUDdtKyW+DyYZgcBnaZqrEEDshYMwsTnURAAl9D7JduveubcuZvBtDoI2OyZqBu4gbVgECY+j4LA7u5L\/Ti5+G6F0+kgIC6SFrxOY8JVsLZe3wvfz2FGYCQEgrbf2crKZ+GHuwgILSh96ypufPmqzo7pMGIcHhoBLPMAh7SEbD+TSBcBceFU5dxt0yPefdFUn+YBPjICwyIAM05PvbLE7bDtZ9LoSUBcpGG539Ohtt9ocFNs0OLj0AjAfNvb1z7lmutN6Ra118N9CagnqvpKd5mhRnnVXC\/4OK4XAsGmV1ni6nJludrrPsT1JSAunq6sXKfJqjfgnMZeHkxCoMJyGALgCLgCzlCv90a\/ptekcSgBcZPt+59h8Bht+fPnL7hTYpDjYxcCIB040hzxUBtnKitXum4KRQwkIHrFru9\/DNeMR9O1nj0ndvM+MiEYOQjyPUMriSl95HD2\/OmPh0FlIAGRCOxBUq3vMwmHgbR493STb+r9w+y+IEJDERAP9CIh24RBKIt5Dg50ar7hyQfEhiYgbg6TkDsmQKW4YjocB83uaOQDciMREA8YEpqOybNnz9lPCGAKJvDzoe5Nh8PzRycfIBuZgHgIJDy9svKOcdG8ePlKYMCZm2Sgk28xPV3UOc7hanlB\/YNhbb4wOmMR0CRyamXlivKFHjGB1xtNMs+oNujk7witt13bERgdI6kJX12Fq6XSWt8xzhtHIiAyPFM5d5MWMr1DY8e3oY4xdoxC8nzCcaojm8+gLqFcjNbDPAHXn3oHAxVRS2xFTSD4\/KPNrctCqmuWsMqIx6772Gkhym4L4VVevCoOyPaXOPEC8TChwCgT+Peoxbt6FpNVYpJYCWjK9Hjz3mdKikuGiPgEmCbj7PTIn4KIE1BTvjwfo+AFmw5rw7EyEqYUwi1Bc3tjV\/jXozS3JrHgMRECmgzCGtHEg4y2Y2sySlsKx7bNpa5jFEC7EitAxLB46Q4EEWyf9gOCGwW7YuiNCQ5Ip7\/jQSz8bpeWasRNPFMViRLQZPJo8+dV2vjjsiXFBXorOu8WaEmbfvhkLEipj3SOD2oj3oh96hRtbN1ZbNyLX5HEECj8zo3Hj3UUrmMjSLl0sukqoXPEYWsMfY3s9Z5C9p3wsEZcruuVkj1vii8y9Vrb3NwsHRf2mpJqlVhzntAo9yMlXtN80d28slxcMqd87IHAKHhhWz7sjKY8bBZurT8X3npSmq5HUXVU6gTsV5AHmw\/KjnDLBEqJyFmm+0oEzop6+pQ6XQJhLdbiYonCJRPGkT43i3BHXPB6Ts9rhFUt\/G7+9nYVcWS94VrNWloSrd3PatgPnLCqusKpjuu3Q9pxyv8BVb3XBNS3Vn0AAAAASUVORK5CYII=\n name: Complex PDF with Images & Tables\nversion: 0.1.0\nworkflow:\n conversation_variables: []\n environment_variables: []\n features: {}\n graph:\n edges:\n - data:\n isInLoop: false\n sourceType: datasource\n targetType: tool\n id: 1750400203722-source-1751281136356-target\n selected: false\n source: '1750400203722'\n sourceHandle: source\n target: '1751281136356'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: knowledge-index\n id: 1751338398711-source-1750400198569-target\n selected: false\n source: '1751338398711'\n sourceHandle: source\n target: '1750400198569'\n targetHandle: target\n type: custom\n zIndex: 0\n - data:\n isInLoop: false\n sourceType: tool\n targetType: tool\n id: 1751281136356-source-1751338398711-target\n selected: false\n source: '1751281136356'\n sourceHandle: source\n target: '1751338398711'\n targetHandle: target\n type: custom\n zIndex: 0\n nodes:\n - data:\n chunk_structure: hierarchical_model\n embedding_model: jina-embeddings-v2-base-en\n embedding_model_provider: langgenius\/jina\/jina\n index_chunk_variable_selector:\n - '1751338398711'\n - result\n indexing_technique: high_quality\n keyword_number: 10\n retrieval_model:\n reranking_enable: true\n reranking_mode: reranking_model\n reranking_model:\n reranking_model_name: jina-reranker-v1-base-en\n reranking_provider_name: langgenius\/jina\/jina\n score_threshold: 0\n score_threshold_enabled: false\n search_method: hybrid_search\n top_k: 3\n weights: null\n selected: true\n title: Knowledge Base\n type: knowledge-index\n height: 114\n id: '1750400198569'\n position:\n x: 355.92518399555183\n y: 282\n positionAbsolute:\n x: 355.92518399555183\n y: 282\n selected: true\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n datasource_configurations: {}\n datasource_label: File\n datasource_name: upload-file\n datasource_parameters: {}\n fileExtensions:\n - txt\n - markdown\n - mdx\n - pdf\n - html\n - xlsx\n - xls\n - vtt\n - properties\n - doc\n - docx\n - csv\n - eml\n - msg\n - pptx\n - xml\n - epub\n - ppt\n - md\n plugin_id: langgenius\/file\n provider_name: file\n provider_type: local_file\n selected: false\n title: File Upload\n type: datasource\n height: 52\n id: '1750400203722'\n position:\n x: -579\n y: 282\n positionAbsolute:\n x: -579\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n author: TenTen\n desc: ''\n height: 337\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently\n we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data\n Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\":\n File Upload, Online Drive, Online Doc, and Web Crawler. Different types\n of Data Sources have different input and output types. The output of File\n Upload and Online Drive are files, while the output of Online Doc and WebCrawler\n are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n Knowledge Pipeline can have multiple data sources. Each data source can\n be selected more than once with different settings. Each added data source\n is a tab on the add file interface. However, each time the user can only\n select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 358\n height: 337\n id: '1751264451381'\n position:\n x: -990.8091030156684\n y: 282\n positionAbsolute:\n x: -990.8091030156684\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 358\n - data:\n author: TenTen\n desc: ''\n height: 260\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge\n Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n starts with Data Source as the starting node and ends with the knowledge\n base node. The general steps are: import documents from the data source\n \u2192 use extractor to extract document content \u2192 split and clean content into\n structured chunks \u2192 store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n user input variables required by the Knowledge Pipeline node must be predefined\n and managed via the Input Field section located in the top-right corner\n of the orchestration canvas. It determines what input fields the end users\n will see and need to fill in when importing files to the knowledge base\n through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique\n Inputs: Input fields defined here are only available to the selected data\n source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global\n Inputs: These input fields are shared across all subsequent nodes after\n the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For\n more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\/knowledge-pipeline\/knowledge-pipeline-orchestration.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\/knowledge-pipeline\/knowledge-pipeline-orchestration\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 1182\n height: 260\n id: '1751266376760'\n position:\n x: -579\n y: -22.64803881585007\n positionAbsolute:\n x: -579\n y: -22.64803881585007\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 1182\n - data:\n author: TenTen\n desc: ''\n height: 541\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A\n document extractor for large language models (LLMs) like MinerU is a tool\n that preprocesses and converts diverse document types into structured, clean,\n and machine-readable data. This structured data can then be used to train\n or augment LLMs and retrieval-augmented generation (RAG) systems by providing\n them with accurate, well-organized content from varied sources. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"MinerU\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n is an advanced open-source document extractor designed specifically to convert\n complex, unstructured documents\u2014such as PDFs, Word files, and PPTs\u2014into\n high-quality, machine-readable formats like Markdown and JSON. MinerU addresses\n challenges in document parsing such as layout detection, formula recognition,\n and multi-language support, which are critical for generating high-quality\n training corpora for LLMs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 541\n id: '1751266402561'\n position:\n x: -263.7680017647218\n y: 558.328085421591\n positionAbsolute:\n x: -263.7680017647218\n y: 558.328085421591\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 554\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\n addresses the dilemma of context and precision by leveraging a two-tier\n hierarchical approach that effectively balances the trade-off between accurate\n matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here\n is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Query Matching with Child Chunks: Small, focused pieces of information,\n often as concise as a single sentence within a paragraph, are used to match\n the user''s query. These child chunks enable precise and relevant initial\n retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"-\n Contextual Enrichment with Parent Chunks: Larger, encompassing sections\u2014such\n as a paragraph, a section, or even an entire document\u2014that include the matched\n child chunks are then retrieved. These parent chunks provide comprehensive\n context for the Language Model (LLM).\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 554\n id: '1751266447821'\n position:\n x: 42.95253988413964\n y: 366.1915342509804\n positionAbsolute:\n x: 42.95253988413964\n y: 366.1915342509804\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n author: TenTen\n desc: ''\n height: 411\n selected: false\n showAuthor: true\n text: '{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The\n knowledge base provides two indexing methods:\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\",\n each with different retrieval strategies. High-Quality mode uses embeddings\n for vectorization and supports vector, full-text, and hybrid retrieval,\n offering more accurate results but higher resource usage. Economical mode\n uses keyword-based inverted indexing with no token consumption but lower\n accuracy; upgrading to High-Quality is possible, but downgrading requires\n creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"*\n Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A\n Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0only\n support the\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0indexing\n method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}'\n theme: blue\n title: ''\n type: ''\n width: 240\n height: 411\n id: '1751266580099'\n position:\n x: 355.92518399555183\n y: 434.6494699299023\n positionAbsolute:\n x: 355.92518399555183\n y: 434.6494699299023\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom-note\n width: 240\n - data:\n credential_id: fd1cbc33-1481-47ee-9af2-954b53d350e0\n is_team_authorization: false\n output_schema:\n properties:\n full_zip_url:\n description: The zip URL of the complete parsed result\n type: string\n images:\n description: The images extracted from the file\n items:\n type: object\n type: array\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n ja_JP: \u89e3\u6790\u3059\u308b\u30d5\u30a1\u30a4\u30eb(pdf\u3001ppt\u3001pptx\u3001doc\u3001docx\u3001png\u3001jpg\u3001jpeg\u3092\u30b5\u30dd\u30fc\u30c8)\n pt_BR: the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg,\n jpeg)\n zh_Hans: \u7528\u4e8e\u89e3\u6790\u7684\u6587\u4ef6(\u652f\u6301 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)\n label:\n en_US: file\n ja_JP: file\n pt_BR: file\n zh_Hans: file\n llm_description: the file to be parsed (support pdf, ppt, pptx, doc, docx,\n png, jpg, jpeg)\n max: null\n min: null\n name: file\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: file\n - auto_generate: null\n default: auto\n form: form\n human_description:\n en_US: (For local deployment service)Parsing method, can be auto, ocr,\n or txt. Default is auto. If results are not satisfactory, try ocr\n ja_JP: \uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8\u30b5\u30fc\u30d3\u30b9\u7528\uff09\u89e3\u6790\u65b9\u6cd5\u306f\u3001auto\u3001ocr\u3001\u307e\u305f\u306ftxt\u306e\u3044\u305a\u308c\u304b\u3067\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fauto\u3067\u3059\u3002\u7d50\u679c\u304c\u6e80\u8db3\u3067\u304d\u306a\u3044\u5834\u5408\u306f\u3001ocr\u3092\u8a66\u3057\u3066\u304f\u3060\u3055\u3044\n pt_BR: (For local deployment service)Parsing method, can be auto, ocr,\n or txt. Default is auto. If results are not satisfactory, try ocr\n zh_Hans: \uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72\u670d\u52a1\uff09\u89e3\u6790\u65b9\u6cd5\uff0c\u53ef\u4ee5\u662fauto, ocr, \u6216 txt\u3002\u9ed8\u8ba4\u662fauto\u3002\u5982\u679c\u7ed3\u679c\u4e0d\u7406\u60f3\uff0c\u8bf7\u5c1d\u8bd5ocr\n label:\n en_US: parse method\n ja_JP: \u89e3\u6790\u65b9\u6cd5\n pt_BR: parse method\n zh_Hans: \u89e3\u6790\u65b9\u6cd5\n llm_description: Parsing method, can be auto, ocr, or txt. Default is auto.\n If results are not satisfactory, try ocr\n max: null\n min: null\n name: parse_method\n options:\n - label:\n en_US: auto\n ja_JP: auto\n pt_BR: auto\n zh_Hans: auto\n value: auto\n - label:\n en_US: ocr\n ja_JP: ocr\n pt_BR: ocr\n zh_Hans: ocr\n value: ocr\n - label:\n en_US: txt\n ja_JP: txt\n pt_BR: txt\n zh_Hans: txt\n value: txt\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: 1\n form: form\n human_description:\n en_US: (For official API) Whether to enable formula recognition\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09\u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b\n pt_BR: (For official API) Whether to enable formula recognition\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542f\u516c\u5f0f\u8bc6\u522b\n label:\n en_US: Enable formula recognition\n ja_JP: \u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\n pt_BR: Enable formula recognition\n zh_Hans: \u5f00\u542f\u516c\u5f0f\u8bc6\u522b\n llm_description: (For official API) Whether to enable formula recognition\n max: null\n min: null\n name: enable_formula\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: 1\n form: form\n human_description:\n en_US: (For official API) Whether to enable table recognition\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09\u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b\n pt_BR: (For official API) Whether to enable table recognition\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542f\u8868\u683c\u8bc6\u522b\n label:\n en_US: Enable table recognition\n ja_JP: \u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\n pt_BR: Enable table recognition\n zh_Hans: \u5f00\u542f\u8868\u683c\u8bc6\u522b\n llm_description: (For official API) Whether to enable table recognition\n max: null\n min: null\n name: enable_table\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: doclayout_yolo\n form: form\n human_description:\n en_US: '(For official API) Optional values: doclayout_yolo, layoutlmv3,\n default value is doclayout_yolo. doclayout_yolo is a self-developed\n model with better effect'\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09\u30aa\u30d7\u30b7\u30e7\u30f3\u5024\uff1adoclayout_yolo\u3001layoutlmv3\u3001\u30c7\u30d5\u30a9\u30eb\u30c8\u5024\u306f doclayout_yolo\u3002doclayout_yolo\n \u306f\u81ea\u5df1\u958b\u767a\u30e2\u30c7\u30eb\u3067\u3001\u52b9\u679c\u304c\u3088\u308a\u826f\u3044\n pt_BR: '(For official API) Optional values: doclayout_yolo, layoutlmv3,\n default value is doclayout_yolo. doclayout_yolo is a self-developed\n model with better effect'\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u53ef\u9009\u503c\uff1adoclayout_yolo\u3001layoutlmv3\uff0c\u9ed8\u8ba4\u503c\u4e3a doclayout_yolo\u3002doclayout_yolo\n \u4e3a\u81ea\u7814\u6a21\u578b\uff0c\u6548\u679c\u66f4\u597d\n label:\n en_US: Layout model\n ja_JP: \u30ec\u30a4\u30a2\u30a6\u30c8\u691c\u51fa\u30e2\u30c7\u30eb\n pt_BR: Layout model\n zh_Hans: \u5e03\u5c40\u68c0\u6d4b\u6a21\u578b\n llm_description: '(For official API) Optional values: doclayout_yolo, layoutlmv3,\n default value is doclayout_yolo. doclayout_yolo is a self-developed model\n withbetter effect'\n max: null\n min: null\n name: layout_model\n options:\n - label:\n en_US: doclayout_yolo\n ja_JP: doclayout_yolo\n pt_BR: doclayout_yolo\n zh_Hans: doclayout_yolo\n value: doclayout_yolo\n - label:\n en_US: layoutlmv3\n ja_JP: layoutlmv3\n pt_BR: layoutlmv3\n zh_Hans: layoutlmv3\n value: layoutlmv3\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: auto\n form: form\n human_description:\n en_US: '(For official API) Specify document language, default ch, can\n be set to auto, when auto, the model will automatically identify document\n language, other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5'\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u6307\u5b9a\u3057\u307e\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fch\u3067\u3001auto\u306b\u8a2d\u5b9a\u3067\u304d\u307e\u3059\u3002auto\u306e\u5834\u5408\u3001\u30e2\u30c7\u30eb\u306f\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u81ea\u52d5\u7684\u306b\u8b58\u5225\u3057\u307e\u3059\u3002\u4ed6\u306e\u30aa\u30d7\u30b7\u30e7\u30f3\u5024\u30ea\u30b9\u30c8\u306b\u3064\u3044\u3066\u306f\u3001\u6b21\u3092\u53c2\u7167\u3057\u3066\u304f\u3060\u3055\u3044\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5\n pt_BR: '(For official API) Specify document language, default ch, can\n be set to auto, when auto, the model will automatically identify document\n language, other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5'\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u6307\u5b9a\u6587\u6863\u8bed\u8a00\uff0c\u9ed8\u8ba4 ch\uff0c\u53ef\u4ee5\u8bbe\u7f6e\u4e3aauto\uff0c\u5f53\u4e3aauto\u65f6\u6a21\u578b\u4f1a\u81ea\u52a8\u8bc6\u522b\u6587\u6863\u8bed\u8a00\uff0c\u5176\u4ed6\u53ef\u9009\u503c\u5217\u8868\u8be6\u89c1\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5\n label:\n en_US: Document language\n ja_JP: \u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\n pt_BR: Document language\n zh_Hans: \u6587\u6863\u8bed\u8a00\n llm_description: '(For official API) Specify document language, default\n ch, can be set to auto, when auto, the model will automatically identify\n document language, other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5'\n max: null\n min: null\n name: language\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 0\n form: form\n human_description:\n en_US: (For official API) Whether to enable OCR recognition\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b\n pt_BR: (For official API) Whether to enable OCR recognition\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542fOCR\u8bc6\u522b\n label:\n en_US: Enable OCR recognition\n ja_JP: OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\n pt_BR: Enable OCR recognition\n zh_Hans: \u5f00\u542fOCR\u8bc6\u522b\n llm_description: (For official API) Whether to enable OCR recognition\n max: null\n min: null\n name: enable_ocr\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: '[]'\n form: form\n human_description:\n en_US: '(For official API) Example: [\"docx\",\"html\"], markdown, json are\n the default export formats, no need to set, this parameter only supports\n one or more of docx, html, latex'\n ja_JP: \uff08\u516c\u5f0fAPI\u7528\uff09\u4f8b\uff1a[\"docx\",\"html\"]\u3001markdown\u3001json\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f\u3067\u3042\u308a\u3001\u8a2d\u5b9a\u3059\u308b\u5fc5\u8981\u306f\u3042\u308a\u307e\u305b\u3093\u3002\u3053\u306e\u30d1\u30e9\u30e1\u30fc\u30bf\u306f\u3001docx\u3001html\u3001latex\u306e3\u3064\u306e\u5f62\u5f0f\u306e\u3044\u305a\u308c\u304b\u307e\u305f\u306f\u8907\u6570\u306e\u307f\u3092\u30b5\u30dd\u30fc\u30c8\u3057\u307e\u3059\n pt_BR: '(For official API) Example: [\"docx\",\"html\"], markdown, json are\n the default export formats, no need to set, this parameter only supports\n one or more of docx, html, latex'\n zh_Hans: \uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u793a\u4f8b\uff1a[\"docx\",\"html\"],markdown\u3001json\u4e3a\u9ed8\u8ba4\u5bfc\u51fa\u683c\u5f0f\uff0c\u65e0\u987b\u8bbe\u7f6e\uff0c\u8be5\u53c2\u6570\u4ec5\u652f\u6301docx\u3001html\u3001latex\u4e09\u79cd\u683c\u5f0f\u4e2d\u7684\u4e00\u4e2a\u6216\u591a\u4e2a\n label:\n en_US: Extra export formats\n ja_JP: \u8ffd\u52a0\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f\n pt_BR: Extra export formats\n zh_Hans: \u989d\u5916\u5bfc\u51fa\u683c\u5f0f\n llm_description: '(For official API) Example: [\"docx\",\"html\"], markdown,\n json are the default export formats, no need to set, this parameter only\n supports one or more of docx, html, latex'\n max: null\n min: null\n name: extra_formats\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n params:\n enable_formula: ''\n enable_ocr: ''\n enable_table: ''\n extra_formats: ''\n file: ''\n language: ''\n layout_model: ''\n parse_method: ''\n provider_id: langgenius\/mineru\/mineru\n provider_name: langgenius\/mineru\/mineru\n provider_type: builtin\n selected: false\n title: MinerU\n tool_configurations:\n enable_formula:\n type: constant\n value: 1\n enable_ocr:\n type: constant\n value: 0\n enable_table:\n type: constant\n value: 1\n extra_formats:\n type: constant\n value: '[]'\n language:\n type: constant\n value: auto\n layout_model:\n type: constant\n value: doclayout_yolo\n parse_method:\n type: constant\n value: auto\n tool_description: a tool for parsing text, tables, and images, supporting\n multiple formats such as pdf, pptx, docx, etc. supporting multiple languages\n such as English, Chinese, etc.\n tool_label: Parse File\n tool_name: parse-file\n tool_node_version: '2'\n tool_parameters:\n file:\n type: variable\n value:\n - '1750400203722'\n - file\n type: tool\n height: 244\n id: '1751281136356'\n position:\n x: -263.7680017647218\n y: 282\n positionAbsolute:\n x: -263.7680017647218\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n - data:\n is_team_authorization: true\n output_schema:\n properties:\n result:\n description: Parent child chunks result\n items:\n type: object\n type: array\n type: object\n paramSchemas:\n - auto_generate: null\n default: null\n form: llm\n human_description:\n en_US: ''\n ja_JP: ''\n pt_BR: ''\n zh_Hans: ''\n label:\n en_US: Input Content\n ja_JP: Input Content\n pt_BR: Conte\u00fado de Entrada\n zh_Hans: \u8f93\u5165\u6587\u672c\n llm_description: The text you want to chunk.\n max: null\n min: null\n name: input_text\n options: []\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: paragraph\n form: llm\n human_description:\n en_US: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n ja_JP: Split text into paragraphs based on separator and maximum chunk\n length, using split text as parent block or entire document as parent\n block and directly retrieve.\n pt_BR: Dividir texto em par\u00e1grafos com base no separador e no comprimento\n m\u00e1ximo do bloco, usando o texto dividido como bloco pai ou documento\n completo como bloco pai e diretamente recuper\u00e1-lo.\n zh_Hans: \u6839\u636e\u5206\u9694\u7b26\u548c\u6700\u5927\u5757\u957f\u5ea6\u5c06\u6587\u672c\u62c6\u5206\u4e3a\u6bb5\u843d\uff0c\u4f7f\u7528\u62c6\u5206\u6587\u672c\u4f5c\u4e3a\u68c0\u7d22\u7684\u7236\u5757\u6216\u6574\u4e2a\u6587\u6863\u7528\u4f5c\u7236\u5757\u5e76\u76f4\u63a5\u68c0\u7d22\u3002\n label:\n en_US: Parent Mode\n ja_JP: Parent Mode\n pt_BR: Modo Pai\n zh_Hans: \u7236\u5757\u6a21\u5f0f\n llm_description: Split text into paragraphs based on separator and maximum\n chunk length, using split text as parent block or entire document as parent\n block and directly retrieve.\n max: null\n min: null\n name: parent_mode\n options:\n - label:\n en_US: Paragraph\n ja_JP: Paragraph\n pt_BR: Par\u00e1grafo\n zh_Hans: \u6bb5\u843d\n value: paragraph\n - label:\n en_US: Full Document\n ja_JP: Full Document\n pt_BR: Documento Completo\n zh_Hans: \u5168\u6587\n value: full_doc\n placeholder: null\n precision: null\n required: true\n scope: null\n template: null\n type: select\n - auto_generate: null\n default: '\n\n\n '\n form: llm\n human_description:\n en_US: Separator used for chunking\n ja_JP: Separator used for chunking\n pt_BR: Separador usado para divis\u00e3o\n zh_Hans: \u7528\u4e8e\u5206\u5757\u7684\u5206\u9694\u7b26\n label:\n en_US: Parent Delimiter\n ja_JP: Parent Delimiter\n pt_BR: Separador de Pai\n zh_Hans: \u7236\u5757\u5206\u9694\u7b26\n llm_description: The separator used to split chunks\n max: null\n min: null\n name: separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 1024\n form: llm\n human_description:\n en_US: Maximum length for chunking\n ja_JP: Maximum length for chunking\n pt_BR: Comprimento m\u00e1ximo para divis\u00e3o\n zh_Hans: \u7528\u4e8e\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6\n label:\n en_US: Maximum Parent Chunk Length\n ja_JP: Maximum Parent Chunk Length\n pt_BR: Comprimento M\u00e1ximo do Bloco Pai\n zh_Hans: \u6700\u5927\u7236\u5757\u957f\u5ea6\n llm_description: Maximum length allowed per chunk\n max: null\n min: null\n name: max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: '. '\n form: llm\n human_description:\n en_US: Separator used for subchunking\n ja_JP: Separator used for subchunking\n pt_BR: Separador usado para subdivis\u00e3o\n zh_Hans: \u7528\u4e8e\u5b50\u5206\u5757\u7684\u5206\u9694\u7b26\n label:\n en_US: Child Delimiter\n ja_JP: Child Delimiter\n pt_BR: Separador de Subdivis\u00e3o\n zh_Hans: \u5b50\u5206\u5757\u5206\u9694\u7b26\n llm_description: The separator used to split subchunks\n max: null\n min: null\n name: subchunk_separator\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: string\n - auto_generate: null\n default: 512\n form: llm\n human_description:\n en_US: Maximum length for subchunking\n ja_JP: Maximum length for subchunking\n pt_BR: Comprimento m\u00e1ximo para subdivis\u00e3o\n zh_Hans: \u7528\u4e8e\u5b50\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6\n label:\n en_US: Maximum Child Chunk Length\n ja_JP: Maximum Child Chunk Length\n pt_BR: Comprimento M\u00e1ximo de Subdivis\u00e3o\n zh_Hans: \u5b50\u5206\u5757\u6700\u5927\u957f\u5ea6\n llm_description: Maximum length allowed per subchunk\n max: null\n min: null\n name: subchunk_max_length\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: number\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove consecutive spaces, newlines and tabs\n ja_JP: Whether to remove consecutive spaces, newlines and tabs\n pt_BR: Se deve remover espa\u00e7os extras no texto\n zh_Hans: \u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26\n label:\n en_US: Replace consecutive spaces, newlines and tabs\n ja_JP: Replace consecutive spaces, newlines and tabs\n pt_BR: Substituir espa\u00e7os consecutivos, novas linhas e guias\n zh_Hans: \u66ff\u6362\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26\n llm_description: Whether to remove consecutive spaces, newlines and tabs\n max: null\n min: null\n name: remove_extra_spaces\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n - auto_generate: null\n default: 0\n form: llm\n human_description:\n en_US: Whether to remove URLs and emails in the text\n ja_JP: Whether to remove URLs and emails in the text\n pt_BR: Se deve remover URLs e e-mails no texto\n zh_Hans: \u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740\n label:\n en_US: Delete all URLs and email addresses\n ja_JP: Delete all URLs and email addresses\n pt_BR: Remover todas as URLs e e-mails\n zh_Hans: \u5220\u9664\u6240\u6709URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740\n llm_description: Whether to remove URLs and emails in the text\n max: null\n min: null\n name: remove_urls_emails\n options: []\n placeholder: null\n precision: null\n required: false\n scope: null\n template: null\n type: boolean\n params:\n input_text: ''\n max_length: ''\n parent_mode: ''\n remove_extra_spaces: ''\n remove_urls_emails: ''\n separator: ''\n subchunk_max_length: ''\n subchunk_separator: ''\n provider_id: langgenius\/parentchild_chunker\/parentchild_chunker\n provider_name: langgenius\/parentchild_chunker\/parentchild_chunker\n provider_type: builtin\n selected: false\n title: Parent-child Chunker\n tool_configurations: {}\n tool_description: Process documents into parent-child chunk structures\n tool_label: Parent-child Chunker\n tool_name: parentchild_chunker\n tool_node_version: '2'\n tool_parameters:\n input_text:\n type: mixed\n value: '{{#1751281136356.text#}}'\n max_length:\n type: variable\n value:\n - rag\n - shared\n - Maximum_Parent_Length\n parent_mode:\n type: variable\n value:\n - rag\n - shared\n - Parent_Mode\n remove_extra_spaces:\n type: variable\n value:\n - rag\n - shared\n - clean_1\n remove_urls_emails:\n type: variable\n value:\n - rag\n - shared\n - clean_2\n separator:\n type: mixed\n value: '{{#rag.shared.Parent_Delimiter#}}'\n subchunk_max_length:\n type: variable\n value:\n - rag\n - shared\n - Maximum_Child_Length\n subchunk_separator:\n type: mixed\n value: '{{#rag.shared.Child_Delimiter#}}'\n type: tool\n height: 52\n id: '1751338398711'\n position:\n x: 42.95253988413964\n y: 282\n positionAbsolute:\n x: 42.95253988413964\n y: 282\n selected: false\n sourcePosition: right\n targetPosition: left\n type: custom\n width: 242\n viewport:\n x: 628.3302331655243\n y: 120.08894361588159\n zoom: 0.7027501395646496\n rag_pipeline_variables:\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: paragraph\n label: Parent Mode\n max_length: 48\n options:\n - paragraph\n - full_doc\n placeholder: null\n required: true\n tooltips: 'Parent Mode provides two options: paragraph mode splits text into paragraphs\n as parent chunks for retrieval, while full_doc mode uses the entire document\n as a single parent chunk (text beyond 10,000 tokens will be truncated).'\n type: select\n unit: null\n variable: Parent_Mode\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\\n\n label: Parent Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: A delimiter is the character used to separate text. \\n\\n is recommended\n for splitting the original document into large parent chunks. You can also use\n special delimiters defined by yourself.\n type: text-input\n unit: null\n variable: Parent_Delimiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 1024\n label: Maximum Parent Length\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: number\n unit: tokens\n variable: Maximum_Parent_Length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: \\n\n label: Child Delimiter\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: A delimiter is the character used to separate text. \\n is recommended\n for splitting parent chunks into small child chunks. You can also use special\n delimiters defined by yourself.\n type: text-input\n unit: null\n variable: Child_Delimiter\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: 256\n label: Maximum Child Length\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: number\n unit: tokens\n variable: Maximum_Child_Length\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: true\n label: Replace consecutive spaces, newlines and tabs.\n max_length: 48\n options: []\n placeholder: null\n required: true\n tooltips: null\n type: checkbox\n unit: null\n variable: clean_1\n - allow_file_extension: null\n allow_file_upload_methods: null\n allowed_file_types: null\n belong_to_node_id: shared\n default_value: null\n label: Delete all URLs and email addresses.\n max_length: 48\n options: []\n placeholder: null\n required: false\n tooltips: null\n type: checkbox\n unit: null\n variable: clean_2\n", + "graph": { + "edges": [ + { + "data": { + "isInLoop": false, + "sourceType": "datasource", + "targetType": "tool" + }, + "id": "1750400203722-source-1751281136356-target", + "selected": false, + "source": "1750400203722", + "sourceHandle": "source", + "target": "1751281136356", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "knowledge-index" + }, + "id": "1751338398711-source-1750400198569-target", + "selected": false, + "source": "1751338398711", + "sourceHandle": "source", + "target": "1750400198569", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + }, + { + "data": { + "isInLoop": false, + "sourceType": "tool", + "targetType": "tool" + }, + "id": "1751281136356-source-1751338398711-target", + "selected": false, + "source": "1751281136356", + "sourceHandle": "source", + "target": "1751338398711", + "targetHandle": "target", + "type": "custom", + "zIndex": 0 + } + ], + "nodes": [ + { + "data": { + "chunk_structure": "hierarchical_model", + "embedding_model": "jina-embeddings-v2-base-en", + "embedding_model_provider": "langgenius\/jina\/jina", + "index_chunk_variable_selector": [ + "1751338398711", + "result" + ], + "indexing_technique": "high_quality", + "keyword_number": 10, + "retrieval_model": { + "reranking_enable": true, + "reranking_mode": "reranking_model", + "reranking_model": { + "reranking_model_name": "jina-reranker-v1-base-en", + "reranking_provider_name": "langgenius\/jina\/jina" + }, + "score_threshold": 0, + "score_threshold_enabled": false, + "search_method": "hybrid_search", + "top_k": 3, + "weights": null + }, + "selected": true, + "title": "Knowledge Base", + "type": "knowledge-index" + }, + "height": 114, + "id": "1750400198569", + "position": { + "x": 355.92518399555183, + "y": 282 + }, + "positionAbsolute": { + "x": 355.92518399555183, + "y": 282 + }, + "selected": true, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "datasource_configurations": {}, + "datasource_label": "File", + "datasource_name": "upload-file", + "datasource_parameters": {}, + "fileExtensions": [ + "txt", + "markdown", + "mdx", + "pdf", + "html", + "xlsx", + "xls", + "vtt", + "properties", + "doc", + "docx", + "csv", + "eml", + "msg", + "pptx", + "xml", + "epub", + "ppt", + "md" + ], + "plugin_id": "langgenius\/file", + "provider_name": "file", + "provider_type": "local_file", + "selected": false, + "title": "File Upload", + "type": "datasource" + }, + "height": 52, + "id": "1750400203722", + "position": { + "x": -579, + "y": 282 + }, + "positionAbsolute": { + "x": -579, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 337, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Currently we support 4 types of \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Data Sources\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\": File Upload, Online Drive, Online Doc, and Web Crawler. Different types of Data Sources have different input and output types. The output of File Upload and Online Drive are files, while the output of Online Doc and WebCrawler are pages. You can find more Data Sources on our Marketplace.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A Knowledge Pipeline can have multiple data sources. Each data source can be selected more than once with different settings. Each added data source is a tab on the add file interface. However, each time the user can only select one data source to import the file and trigger its subsequent processing.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 358 + }, + "height": 337, + "id": "1751264451381", + "position": { + "x": -990.8091030156684, + "y": 282 + }, + "positionAbsolute": { + "x": -990.8091030156684, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 358 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 260, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A \",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Knowledge Pipeline\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" starts with Data Source as the starting node and ends with the knowledge base node. The general steps are: import documents from the data source \u2192 use extractor to extract document content \u2192 split and clean content into structured chunks \u2192 store in the knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The user input variables required by the Knowledge Pipeline node must be predefined and managed via the Input Field section located in the top-right corner of the orchestration canvas. It determines what input fields the end users will see and need to fill in when importing files to the knowledge base through this pipeline.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Unique Inputs: Input fields defined here are only available to the selected data source and its downstream nodes.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Global Inputs: These input fields are shared across all subsequent nodes after the data source and are typically set during the Process Documents step.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"For more information, see \",\"type\":\"text\",\"version\":1},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\/knowledge-pipeline\/knowledge-pipeline-orchestration.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"link\",\"version\":1,\"rel\":\"noreferrer\",\"target\":null,\"title\":null,\"url\":\"https:\/\/docs.dify.ai\/en\/guides\/knowledge-base\/knowledge-pipeline\/knowledge-pipeline-orchestration\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 1182 + }, + "height": 260, + "id": "1751266376760", + "position": { + "x": -579, + "y": -22.64803881585007 + }, + "positionAbsolute": { + "x": -579, + "y": -22.64803881585007 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 1182 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 541, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"A document extractor for large language models (LLMs) like MinerU is a tool that preprocesses and converts diverse document types into structured, clean, and machine-readable data. This structured data can then be used to train or augment LLMs and retrieval-augmented generation (RAG) systems by providing them with accurate, well-organized content from varied sources. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"MinerU\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" is an advanced open-source document extractor designed specifically to convert complex, unstructured documents\u2014such as PDFs, Word files, and PPTs\u2014into high-quality, machine-readable formats like Markdown and JSON. MinerU addresses challenges in document parsing such as layout detection, formula recognition, and multi-language support, which are critical for generating high-quality training corpora for LLMs.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 541, + "id": "1751266402561", + "position": { + "x": -263.7680017647218, + "y": 558.328085421591 + }, + "positionAbsolute": { + "x": -263.7680017647218, + "y": 558.328085421591 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 554, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\" addresses the dilemma of context and precision by leveraging a two-tier hierarchical approach that effectively balances the trade-off between accurate matching and comprehensive contextual information in RAG systems. \",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Here is the essential mechanism of this structured, two-level information access:\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Query Matching with Child Chunks: Small, focused pieces of information, often as concise as a single sentence within a paragraph, are used to match the user's query. These child chunks enable precise and relevant initial retrieval.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"- Contextual Enrichment with Parent Chunks: Larger, encompassing sections\u2014such as a paragraph, a section, or even an entire document\u2014that include the matched child chunks are then retrieved. These parent chunks provide comprehensive context for the Language Model (LLM).\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 554, + "id": "1751266447821", + "position": { + "x": 42.95253988413964, + "y": 366.1915342509804 + }, + "positionAbsolute": { + "x": 42.95253988413964, + "y": 366.1915342509804 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "author": "TenTen", + "desc": "", + "height": 411, + "selected": false, + "showAuthor": true, + "text": "{\"root\":{\"children\":[{\"children\":[{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"The knowledge base provides two indexing methods:\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Economical\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\", each with different retrieval strategies. High-Quality mode uses embeddings for vectorization and supports vector, full-text, and hybrid retrieval, offering more accurate results but higher resource usage. Economical mode uses keyword-based inverted indexing with no token consumption but lower accuracy; upgrading to High-Quality is possible, but downgrading requires creating a new knowledge base.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[],\"direction\":null,\"format\":\"\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":0,\"textStyle\":\"\"},{\"children\":[{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"* Parent-Child Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0and\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"Q&A Mode\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0only support the\u00a0\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":1,\"mode\":\"normal\",\"style\":\"\",\"text\":\"High-Quality\",\"type\":\"text\",\"version\":1},{\"detail\":0,\"format\":0,\"mode\":\"normal\",\"style\":\"\",\"text\":\"\u00a0indexing method.\",\"type\":\"text\",\"version\":1}],\"direction\":\"ltr\",\"format\":\"start\",\"indent\":0,\"type\":\"paragraph\",\"version\":1,\"textFormat\":1,\"textStyle\":\"\"}],\"direction\":\"ltr\",\"format\":\"\",\"indent\":0,\"type\":\"root\",\"version\":1,\"textFormat\":1}}", + "theme": "blue", + "title": "", + "type": "", + "width": 240 + }, + "height": 411, + "id": "1751266580099", + "position": { + "x": 355.92518399555183, + "y": 434.6494699299023 + }, + "positionAbsolute": { + "x": 355.92518399555183, + "y": 434.6494699299023 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom-note", + "width": 240 + }, + { + "data": { + "credential_id": "fd1cbc33-1481-47ee-9af2-954b53d350e0", + "is_team_authorization": false, + "output_schema": { + "properties": { + "full_zip_url": { + "description": "The zip URL of the complete parsed result", + "type": "string" + }, + "images": { + "description": "The images extracted from the file", + "items": { + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "ja_JP": "\u89e3\u6790\u3059\u308b\u30d5\u30a1\u30a4\u30eb(pdf\u3001ppt\u3001pptx\u3001doc\u3001docx\u3001png\u3001jpg\u3001jpeg\u3092\u30b5\u30dd\u30fc\u30c8)", + "pt_BR": "the file to be parsed(support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "zh_Hans": "\u7528\u4e8e\u89e3\u6790\u7684\u6587\u4ef6(\u652f\u6301 pdf, ppt, pptx, doc, docx, png, jpg, jpeg)" + }, + "label": { + "en_US": "file", + "ja_JP": "file", + "pt_BR": "file", + "zh_Hans": "file" + }, + "llm_description": "the file to be parsed (support pdf, ppt, pptx, doc, docx, png, jpg, jpeg)", + "max": null, + "min": null, + "name": "file", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "file" + }, + { + "auto_generate": null, + "default": "auto", + "form": "form", + "human_description": { + "en_US": "(For local deployment service)Parsing method, can be auto, ocr, or txt. Default is auto. If results are not satisfactory, try ocr", + "ja_JP": "\uff08\u30ed\u30fc\u30ab\u30eb\u30c7\u30d7\u30ed\u30a4\u30e1\u30f3\u30c8\u30b5\u30fc\u30d3\u30b9\u7528\uff09\u89e3\u6790\u65b9\u6cd5\u306f\u3001auto\u3001ocr\u3001\u307e\u305f\u306ftxt\u306e\u3044\u305a\u308c\u304b\u3067\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fauto\u3067\u3059\u3002\u7d50\u679c\u304c\u6e80\u8db3\u3067\u304d\u306a\u3044\u5834\u5408\u306f\u3001ocr\u3092\u8a66\u3057\u3066\u304f\u3060\u3055\u3044", + "pt_BR": "(For local deployment service)Parsing method, can be auto, ocr, or txt. Default is auto. If results are not satisfactory, try ocr", + "zh_Hans": "\uff08\u7528\u4e8e\u672c\u5730\u90e8\u7f72\u670d\u52a1\uff09\u89e3\u6790\u65b9\u6cd5\uff0c\u53ef\u4ee5\u662fauto, ocr, \u6216 txt\u3002\u9ed8\u8ba4\u662fauto\u3002\u5982\u679c\u7ed3\u679c\u4e0d\u7406\u60f3\uff0c\u8bf7\u5c1d\u8bd5ocr" + }, + "label": { + "en_US": "parse method", + "ja_JP": "\u89e3\u6790\u65b9\u6cd5", + "pt_BR": "parse method", + "zh_Hans": "\u89e3\u6790\u65b9\u6cd5" + }, + "llm_description": "Parsing method, can be auto, ocr, or txt. Default is auto. If results are not satisfactory, try ocr", + "max": null, + "min": null, + "name": "parse_method", + "options": [ + { + "label": { + "en_US": "auto", + "ja_JP": "auto", + "pt_BR": "auto", + "zh_Hans": "auto" + }, + "value": "auto" + }, + { + "label": { + "en_US": "ocr", + "ja_JP": "ocr", + "pt_BR": "ocr", + "zh_Hans": "ocr" + }, + "value": "ocr" + }, + { + "label": { + "en_US": "txt", + "ja_JP": "txt", + "pt_BR": "txt", + "zh_Hans": "txt" + }, + "value": "txt" + } + ], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": 1, + "form": "form", + "human_description": { + "en_US": "(For official API) Whether to enable formula recognition", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09\u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b", + "pt_BR": "(For official API) Whether to enable formula recognition", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542f\u516c\u5f0f\u8bc6\u522b" + }, + "label": { + "en_US": "Enable formula recognition", + "ja_JP": "\u6570\u5f0f\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b", + "pt_BR": "Enable formula recognition", + "zh_Hans": "\u5f00\u542f\u516c\u5f0f\u8bc6\u522b" + }, + "llm_description": "(For official API) Whether to enable formula recognition", + "max": null, + "min": null, + "name": "enable_formula", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": 1, + "form": "form", + "human_description": { + "en_US": "(For official API) Whether to enable table recognition", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09\u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b", + "pt_BR": "(For official API) Whether to enable table recognition", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542f\u8868\u683c\u8bc6\u522b" + }, + "label": { + "en_US": "Enable table recognition", + "ja_JP": "\u8868\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b", + "pt_BR": "Enable table recognition", + "zh_Hans": "\u5f00\u542f\u8868\u683c\u8bc6\u522b" + }, + "llm_description": "(For official API) Whether to enable table recognition", + "max": null, + "min": null, + "name": "enable_table", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": "doclayout_yolo", + "form": "form", + "human_description": { + "en_US": "(For official API) Optional values: doclayout_yolo, layoutlmv3, default value is doclayout_yolo. doclayout_yolo is a self-developed model with better effect", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09\u30aa\u30d7\u30b7\u30e7\u30f3\u5024\uff1adoclayout_yolo\u3001layoutlmv3\u3001\u30c7\u30d5\u30a9\u30eb\u30c8\u5024\u306f doclayout_yolo\u3002doclayout_yolo \u306f\u81ea\u5df1\u958b\u767a\u30e2\u30c7\u30eb\u3067\u3001\u52b9\u679c\u304c\u3088\u308a\u826f\u3044", + "pt_BR": "(For official API) Optional values: doclayout_yolo, layoutlmv3, default value is doclayout_yolo. doclayout_yolo is a self-developed model with better effect", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u53ef\u9009\u503c\uff1adoclayout_yolo\u3001layoutlmv3\uff0c\u9ed8\u8ba4\u503c\u4e3a doclayout_yolo\u3002doclayout_yolo \u4e3a\u81ea\u7814\u6a21\u578b\uff0c\u6548\u679c\u66f4\u597d" + }, + "label": { + "en_US": "Layout model", + "ja_JP": "\u30ec\u30a4\u30a2\u30a6\u30c8\u691c\u51fa\u30e2\u30c7\u30eb", + "pt_BR": "Layout model", + "zh_Hans": "\u5e03\u5c40\u68c0\u6d4b\u6a21\u578b" + }, + "llm_description": "(For official API) Optional values: doclayout_yolo, layoutlmv3, default value is doclayout_yolo. doclayout_yolo is a self-developed model withbetter effect", + "max": null, + "min": null, + "name": "layout_model", + "options": [ + { + "label": { + "en_US": "doclayout_yolo", + "ja_JP": "doclayout_yolo", + "pt_BR": "doclayout_yolo", + "zh_Hans": "doclayout_yolo" + }, + "value": "doclayout_yolo" + }, + { + "label": { + "en_US": "layoutlmv3", + "ja_JP": "layoutlmv3", + "pt_BR": "layoutlmv3", + "zh_Hans": "layoutlmv3" + }, + "value": "layoutlmv3" + } + ], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": "auto", + "form": "form", + "human_description": { + "en_US": "(For official API) Specify document language, default ch, can be set to auto, when auto, the model will automatically identify document language, other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u6307\u5b9a\u3057\u307e\u3059\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306fch\u3067\u3001auto\u306b\u8a2d\u5b9a\u3067\u304d\u307e\u3059\u3002auto\u306e\u5834\u5408\u3001\u30e2\u30c7\u30eb\u306f\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e\u3092\u81ea\u52d5\u7684\u306b\u8b58\u5225\u3057\u307e\u3059\u3002\u4ed6\u306e\u30aa\u30d7\u30b7\u30e7\u30f3\u5024\u30ea\u30b9\u30c8\u306b\u3064\u3044\u3066\u306f\u3001\u6b21\u3092\u53c2\u7167\u3057\u3066\u304f\u3060\u3055\u3044\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5", + "pt_BR": "(For official API) Specify document language, default ch, can be set to auto, when auto, the model will automatically identify document language, other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u6307\u5b9a\u6587\u6863\u8bed\u8a00\uff0c\u9ed8\u8ba4 ch\uff0c\u53ef\u4ee5\u8bbe\u7f6e\u4e3aauto\uff0c\u5f53\u4e3aauto\u65f6\u6a21\u578b\u4f1a\u81ea\u52a8\u8bc6\u522b\u6587\u6863\u8bed\u8a00\uff0c\u5176\u4ed6\u53ef\u9009\u503c\u5217\u8868\u8be6\u89c1\uff1ahttps:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5" + }, + "label": { + "en_US": "Document language", + "ja_JP": "\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u8a00\u8a9e", + "pt_BR": "Document language", + "zh_Hans": "\u6587\u6863\u8bed\u8a00" + }, + "llm_description": "(For official API) Specify document language, default ch, can be set to auto, when auto, the model will automatically identify document language, other optional value list see: https:\/\/paddlepaddle.github.io\/PaddleOCR\/latest\/ppocr\/blog\/multi_languages.html#5", + "max": null, + "min": null, + "name": "language", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 0, + "form": "form", + "human_description": { + "en_US": "(For official API) Whether to enable OCR recognition", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b\u304b\u3069\u3046\u304b", + "pt_BR": "(For official API) Whether to enable OCR recognition", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u662f\u5426\u5f00\u542fOCR\u8bc6\u522b" + }, + "label": { + "en_US": "Enable OCR recognition", + "ja_JP": "OCR\u8a8d\u8b58\u3092\u6709\u52b9\u306b\u3059\u308b", + "pt_BR": "Enable OCR recognition", + "zh_Hans": "\u5f00\u542fOCR\u8bc6\u522b" + }, + "llm_description": "(For official API) Whether to enable OCR recognition", + "max": null, + "min": null, + "name": "enable_ocr", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": "[]", + "form": "form", + "human_description": { + "en_US": "(For official API) Example: [\"docx\",\"html\"], markdown, json are the default export formats, no need to set, this parameter only supports one or more of docx, html, latex", + "ja_JP": "\uff08\u516c\u5f0fAPI\u7528\uff09\u4f8b\uff1a[\"docx\",\"html\"]\u3001markdown\u3001json\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f\u3067\u3042\u308a\u3001\u8a2d\u5b9a\u3059\u308b\u5fc5\u8981\u306f\u3042\u308a\u307e\u305b\u3093\u3002\u3053\u306e\u30d1\u30e9\u30e1\u30fc\u30bf\u306f\u3001docx\u3001html\u3001latex\u306e3\u3064\u306e\u5f62\u5f0f\u306e\u3044\u305a\u308c\u304b\u307e\u305f\u306f\u8907\u6570\u306e\u307f\u3092\u30b5\u30dd\u30fc\u30c8\u3057\u307e\u3059", + "pt_BR": "(For official API) Example: [\"docx\",\"html\"], markdown, json are the default export formats, no need to set, this parameter only supports one or more of docx, html, latex", + "zh_Hans": "\uff08\u7528\u4e8e\u5b98\u65b9API\uff09\u793a\u4f8b\uff1a[\"docx\",\"html\"],markdown\u3001json\u4e3a\u9ed8\u8ba4\u5bfc\u51fa\u683c\u5f0f\uff0c\u65e0\u987b\u8bbe\u7f6e\uff0c\u8be5\u53c2\u6570\u4ec5\u652f\u6301docx\u3001html\u3001latex\u4e09\u79cd\u683c\u5f0f\u4e2d\u7684\u4e00\u4e2a\u6216\u591a\u4e2a" + }, + "label": { + "en_US": "Extra export formats", + "ja_JP": "\u8ffd\u52a0\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u5f62\u5f0f", + "pt_BR": "Extra export formats", + "zh_Hans": "\u989d\u5916\u5bfc\u51fa\u683c\u5f0f" + }, + "llm_description": "(For official API) Example: [\"docx\",\"html\"], markdown, json are the default export formats, no need to set, this parameter only supports one or more of docx, html, latex", + "max": null, + "min": null, + "name": "extra_formats", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + } + ], + "params": { + "enable_formula": "", + "enable_ocr": "", + "enable_table": "", + "extra_formats": "", + "file": "", + "language": "", + "layout_model": "", + "parse_method": "" + }, + "provider_id": "langgenius\/mineru\/mineru", + "provider_name": "langgenius\/mineru\/mineru", + "provider_type": "builtin", + "selected": false, + "title": "MinerU", + "tool_configurations": { + "enable_formula": { + "type": "constant", + "value": 1 + }, + "enable_ocr": { + "type": "constant", + "value": 0 + }, + "enable_table": { + "type": "constant", + "value": 1 + }, + "extra_formats": { + "type": "constant", + "value": "[]" + }, + "language": { + "type": "constant", + "value": "auto" + }, + "layout_model": { + "type": "constant", + "value": "doclayout_yolo" + }, + "parse_method": { + "type": "constant", + "value": "auto" + } + }, + "tool_description": "a tool for parsing text, tables, and images, supporting multiple formats such as pdf, pptx, docx, etc. supporting multiple languages such as English, Chinese, etc.", + "tool_label": "Parse File", + "tool_name": "parse-file", + "tool_node_version": "2", + "tool_parameters": { + "file": { + "type": "variable", + "value": [ + "1750400203722", + "file" + ] + } + }, + "type": "tool" + }, + "height": 244, + "id": "1751281136356", + "position": { + "x": -263.7680017647218, + "y": 282 + }, + "positionAbsolute": { + "x": -263.7680017647218, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + }, + { + "data": { + "is_team_authorization": true, + "output_schema": { + "properties": { + "result": { + "description": "Parent child chunks result", + "items": { + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "paramSchemas": [ + { + "auto_generate": null, + "default": null, + "form": "llm", + "human_description": { + "en_US": "", + "ja_JP": "", + "pt_BR": "", + "zh_Hans": "" + }, + "label": { + "en_US": "Input Content", + "ja_JP": "Input Content", + "pt_BR": "Conte\u00fado de Entrada", + "zh_Hans": "\u8f93\u5165\u6587\u672c" + }, + "llm_description": "The text you want to chunk.", + "max": null, + "min": null, + "name": "input_text", + "options": [], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": "paragraph", + "form": "llm", + "human_description": { + "en_US": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "ja_JP": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "pt_BR": "Dividir texto em par\u00e1grafos com base no separador e no comprimento m\u00e1ximo do bloco, usando o texto dividido como bloco pai ou documento completo como bloco pai e diretamente recuper\u00e1-lo.", + "zh_Hans": "\u6839\u636e\u5206\u9694\u7b26\u548c\u6700\u5927\u5757\u957f\u5ea6\u5c06\u6587\u672c\u62c6\u5206\u4e3a\u6bb5\u843d\uff0c\u4f7f\u7528\u62c6\u5206\u6587\u672c\u4f5c\u4e3a\u68c0\u7d22\u7684\u7236\u5757\u6216\u6574\u4e2a\u6587\u6863\u7528\u4f5c\u7236\u5757\u5e76\u76f4\u63a5\u68c0\u7d22\u3002" + }, + "label": { + "en_US": "Parent Mode", + "ja_JP": "Parent Mode", + "pt_BR": "Modo Pai", + "zh_Hans": "\u7236\u5757\u6a21\u5f0f" + }, + "llm_description": "Split text into paragraphs based on separator and maximum chunk length, using split text as parent block or entire document as parent block and directly retrieve.", + "max": null, + "min": null, + "name": "parent_mode", + "options": [ + { + "label": { + "en_US": "Paragraph", + "ja_JP": "Paragraph", + "pt_BR": "Par\u00e1grafo", + "zh_Hans": "\u6bb5\u843d" + }, + "value": "paragraph" + }, + { + "label": { + "en_US": "Full Document", + "ja_JP": "Full Document", + "pt_BR": "Documento Completo", + "zh_Hans": "\u5168\u6587" + }, + "value": "full_doc" + } + ], + "placeholder": null, + "precision": null, + "required": true, + "scope": null, + "template": null, + "type": "select" + }, + { + "auto_generate": null, + "default": "\n\n", + "form": "llm", + "human_description": { + "en_US": "Separator used for chunking", + "ja_JP": "Separator used for chunking", + "pt_BR": "Separador usado para divis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5206\u5757\u7684\u5206\u9694\u7b26" + }, + "label": { + "en_US": "Parent Delimiter", + "ja_JP": "Parent Delimiter", + "pt_BR": "Separador de Pai", + "zh_Hans": "\u7236\u5757\u5206\u9694\u7b26" + }, + "llm_description": "The separator used to split chunks", + "max": null, + "min": null, + "name": "separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 1024, + "form": "llm", + "human_description": { + "en_US": "Maximum length for chunking", + "ja_JP": "Maximum length for chunking", + "pt_BR": "Comprimento m\u00e1ximo para divis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6" + }, + "label": { + "en_US": "Maximum Parent Chunk Length", + "ja_JP": "Maximum Parent Chunk Length", + "pt_BR": "Comprimento M\u00e1ximo do Bloco Pai", + "zh_Hans": "\u6700\u5927\u7236\u5757\u957f\u5ea6" + }, + "llm_description": "Maximum length allowed per chunk", + "max": null, + "min": null, + "name": "max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": ". ", + "form": "llm", + "human_description": { + "en_US": "Separator used for subchunking", + "ja_JP": "Separator used for subchunking", + "pt_BR": "Separador usado para subdivis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5b50\u5206\u5757\u7684\u5206\u9694\u7b26" + }, + "label": { + "en_US": "Child Delimiter", + "ja_JP": "Child Delimiter", + "pt_BR": "Separador de Subdivis\u00e3o", + "zh_Hans": "\u5b50\u5206\u5757\u5206\u9694\u7b26" + }, + "llm_description": "The separator used to split subchunks", + "max": null, + "min": null, + "name": "subchunk_separator", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "string" + }, + { + "auto_generate": null, + "default": 512, + "form": "llm", + "human_description": { + "en_US": "Maximum length for subchunking", + "ja_JP": "Maximum length for subchunking", + "pt_BR": "Comprimento m\u00e1ximo para subdivis\u00e3o", + "zh_Hans": "\u7528\u4e8e\u5b50\u5206\u5757\u7684\u6700\u5927\u957f\u5ea6" + }, + "label": { + "en_US": "Maximum Child Chunk Length", + "ja_JP": "Maximum Child Chunk Length", + "pt_BR": "Comprimento M\u00e1ximo de Subdivis\u00e3o", + "zh_Hans": "\u5b50\u5206\u5757\u6700\u5927\u957f\u5ea6" + }, + "llm_description": "Maximum length allowed per subchunk", + "max": null, + "min": null, + "name": "subchunk_max_length", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "number" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove consecutive spaces, newlines and tabs", + "ja_JP": "Whether to remove consecutive spaces, newlines and tabs", + "pt_BR": "Se deve remover espa\u00e7os extras no texto", + "zh_Hans": "\u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26" + }, + "label": { + "en_US": "Replace consecutive spaces, newlines and tabs", + "ja_JP": "Replace consecutive spaces, newlines and tabs", + "pt_BR": "Substituir espa\u00e7os consecutivos, novas linhas e guias", + "zh_Hans": "\u66ff\u6362\u8fde\u7eed\u7a7a\u683c\u3001\u6362\u884c\u7b26\u548c\u5236\u8868\u7b26" + }, + "llm_description": "Whether to remove consecutive spaces, newlines and tabs", + "max": null, + "min": null, + "name": "remove_extra_spaces", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + }, + { + "auto_generate": null, + "default": 0, + "form": "llm", + "human_description": { + "en_US": "Whether to remove URLs and emails in the text", + "ja_JP": "Whether to remove URLs and emails in the text", + "pt_BR": "Se deve remover URLs e e-mails no texto", + "zh_Hans": "\u662f\u5426\u79fb\u9664\u6587\u672c\u4e2d\u7684URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740" + }, + "label": { + "en_US": "Delete all URLs and email addresses", + "ja_JP": "Delete all URLs and email addresses", + "pt_BR": "Remover todas as URLs e e-mails", + "zh_Hans": "\u5220\u9664\u6240\u6709URL\u548c\u7535\u5b50\u90ae\u4ef6\u5730\u5740" + }, + "llm_description": "Whether to remove URLs and emails in the text", + "max": null, + "min": null, + "name": "remove_urls_emails", + "options": [], + "placeholder": null, + "precision": null, + "required": false, + "scope": null, + "template": null, + "type": "boolean" + } + ], + "params": { + "input_text": "", + "max_length": "", + "parent_mode": "", + "remove_extra_spaces": "", + "remove_urls_emails": "", + "separator": "", + "subchunk_max_length": "", + "subchunk_separator": "" + }, + "provider_id": "langgenius\/parentchild_chunker\/parentchild_chunker", + "provider_name": "langgenius\/parentchild_chunker\/parentchild_chunker", + "provider_type": "builtin", + "selected": false, + "title": "Parent-child Chunker", + "tool_configurations": {}, + "tool_description": "Process documents into parent-child chunk structures", + "tool_label": "Parent-child Chunker", + "tool_name": "parentchild_chunker", + "tool_node_version": "2", + "tool_parameters": { + "input_text": { + "type": "mixed", + "value": "{{#1751281136356.text#}}" + }, + "max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Maximum_Parent_Length" + ] + }, + "parent_mode": { + "type": "variable", + "value": [ + "rag", + "shared", + "Parent_Mode" + ] + }, + "remove_extra_spaces": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_1" + ] + }, + "remove_urls_emails": { + "type": "variable", + "value": [ + "rag", + "shared", + "clean_2" + ] + }, + "separator": { + "type": "mixed", + "value": "{{#rag.shared.Parent_Delimiter#}}" + }, + "subchunk_max_length": { + "type": "variable", + "value": [ + "rag", + "shared", + "Maximum_Child_Length" + ] + }, + "subchunk_separator": { + "type": "mixed", + "value": "{{#rag.shared.Child_Delimiter#}}" + } + }, + "type": "tool" + }, + "height": 52, + "id": "1751338398711", + "position": { + "x": 42.95253988413964, + "y": 282 + }, + "positionAbsolute": { + "x": 42.95253988413964, + "y": 282 + }, + "selected": false, + "sourcePosition": "right", + "targetPosition": "left", + "type": "custom", + "width": 242 + } + ], + "viewport": { + "x": 628.3302331655243, + "y": 120.08894361588159, + "zoom": 0.7027501395646496 + } + }, + "icon_info": { + "icon": "87426868-91d6-4774-a535-5fd4595a77b3", + "icon_background": null, + "icon_type": "image", + "icon_url": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAKAAAACgCAYAAACLz2ctAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAoAAAAACn7BmJAAARwElEQVR4Ae1dvXPcxhVfLMAP0RR1pL7MGVu8G7sXXdszotNYne1x6kgpktZSiiRNIrtMilgqnNZSb4\/lzm4i5i8w1TvDE+UZyZIlnihKOvIAbN5v7\/aIw93xPvBBHPDezBHYBbC7+O2Pb9++\/YAlMiIPHjwoO65btpQqK6VKVKySsqwV9fQpSliy6IcTubhYxrFTrJJqXe+Mz2+I8KgJoeh3IIRBTW1vt+MoXLWWlgRheo\/uqlmWVSVMa67jVJeXl6sHTx7dGb1HurK9uVnybHtNKXFBWAKEW1XCKvcrhb+tCdi+LBeX2ud80o3AaHipDUGkFErdJXJu2J63vliptAncnXr8MakQ8PH9+2tU9Av0omtCCZx3iZSSsLCE49j6iHPE+U+fCEnnCEOmTp\/uehbXzPWuizmNoFaC4CQdFxCE3V9\/bcd4vk8txpLwW\/f6FPZ9RT8c\/fZ9nSdESmGtK1veOvPGG3SerCRGQGg6V8rLxIwPg6QDUWzb1kTDcXrKaROu16v6T550RMuTJzvCHOhEYBS8PM8TIGmj4QrX9ejndiRG5Kj6lvj8zLlzNzsuxBiInYCaeI7zqeWrK8YuA+lmZqbF9PSUcIh0o2irUQCNEZeJTSoqXg0i4d7evial0ZIgopLWzdNvvvl53MDESsBfNrc+sqX6wth0juOIublZMUXHcSUqoOPmO6nPxYkXiFinn9GMIGLcGjEWApLWK7u2\/ZVpauMgniFAnICaNPN8TAIvaMXd3ZcHdqMlbjve1NXFSvSetIxaGU\/u3\/\/Uk\/aPIB+a1rm5Y+LEwnwkrRe1TPx8vAigBVssLYj51+Z0x5Dq+iNXNn58tLV1OWpOYxMQtt7jra0vqFd1HbYe7DsU8tjsTNQy8fMZRQB2PJQLjiQlS4mvwIEoxR2rCdZNrpTfUnd9FVrv2LHZxIiXRJMSBbCsP5sWXvX6nnj1qq5dPOQQ33D86Y\/HaZJH1oAgnyflHZAPfrrSieOJkS\/rlV3k8s1SS3eC6h4cABc82bizvfmgPComIxHQkA+9XPjwoI6bBRg1W74\/Dwig7sEBuNbIDCPFNDoJhyYgky8PlIn\/HUDChQgkHIqAvcg3ijM5\/tfmFLOEALgwLgmHIiANqX0bbHaZfFmq\/myUJUxCV+5\/S4qrNKh0AwnY7GY3OxwLx18baRhtUOZ8PV8IgITHiSOmY0KDE9cGveGhBHy0SY5GJa4gYe5wDIKSrwMB0zHBDCZw5+G9e1cOQ6YvAWH3kX2pnYzw8zVZfVhSfI0RaCIAroAzEJp6cu0w90xfApL6pEkFogSvN49uNIHlv8MjAD8hRsdISq7d+Krfkz0J2Gp6PwKT51pM7pcAxzMC\/RDQY8fNpnjtV5op1eu+ngSUUmnjEeTjprcXbBw3DALoO5imWJA516tX3EVAmt1yDS4XEK816DxMXnwPI9ATATTFmJ5H5lx5X8quDkkXAZXvX0ZK8\/NzPRPkSEZgVAQwKRlCq34+DWvBDgLC9oP2w\/yvKLOYdW78hxFoIQAuQQuSNNcJBZDpIKCx\/bjpDSDEp7EgYLQgjWR8GEywTcBHmz\/r9bls+wXh4fO4EIAWbDmn1x5v3l8z6bYJKKV3GZFTtEyShRFIAoHp5kxq4Ut\/zaTfJqAS8gIiufk10PAxbgRajmloQs01pK+n5KNn4kp7GxEnlwZOYMBtqUl4inlqGeckoywt5MfODbXajp7G7\/jeIrYB0RoQe7UAb+755oR1GX0NOKYlzZ6GGM5pAhIzVxFp074sLIxAkghg7x8I7VezhmPTBrSs8wiwBgQKLEkigLVEEIyM4Njs8iqLAtQNsdt9ElzLhGTJhskEIBNeCGxG9YLegaZpaaXXYlyzCcbqJhZGIEkEYAdCjAaUD2jiKSJ41gtQYEkaAd0RoYkuEOyKK2mMroyA3YrEOQsjkCQCRgs6dbcsaYtc7fizZFM1Jpkxp80IAAHTE7ZsVZbkgikjkptgoMCSBgJGAxL3SmiMmxqwZRymUQDOo9gIGAKCe9L0RgKRxUaH3z5xBExrS5xbaTv+9FSZxLPmDBiBTgSId9YKorLohO4sKofygoBRdp5Si20NmJeX4\/fIPgLG40JEPMEEzH595bqEtF7Ool4wLUWa0F7wr+\/\/JlMVdOrOfzrKY8p3\/C9\/FjMXL3ZcK2rADHrQHtPkiBa+dsOYdrmooCT93s\/\/8U+x9\/33SWczcelzE5xilYGEjY2NFHPMflZMwJTraOdvfxfuTz+lnGt2s3O8bb0URPheA+NxsZeU5\/N1Qqp2d8Wzq38SJ774l3DefrvzYgZDSazJ0V\/r3Hmu3xZTEHgoLuWKNyT0Hj5MOedsZBfo8OqhOCbgEdQLSLhDmrCIJOwg4BFgz1m2EAD5ikpCQwIHX9SGyJjWAydhM5jC5vFoSLhANqH9+uuZf8W4bHppNZd\/xN\/ryDyE2SugIWERm2MmYEb4aEgI27BIwgTMUG2DhDXqmBSJhEzADBEQRfHISV0kEjIBM0ZAQ0KMmBRBmIAZrWWMGWPsOO\/CBMxwDWP2TN5JyATMMAFRNJBw98t\/Z7yU4xePCTg+dqk9Wf\/6a\/Hy1q3U8kszIyZgmmhHyOvlzVu5JCETMAIp0n40jyRkAqbNooj55Y2ETMCIhDiKx0HCV19\/cxRZx54nEzB2SNNJ8MWXX+ZikRMTMB2+JJJLHnyE\/FmkRKhxkGh4nfDBFT4DAqwBmQdHigAT8Ejh58yZgMyBI0WAbcCY4Td7wcScbN\/kJt3GZA3Yt2r5QhoIMAHTQJnz6IsAE7AvNHwhDQSYgGmgzHn0RYAJ2BcavpAGAkzANFDmPPoiwATsCw1fSAOBifcDTrofLI1KznIerAGzXDsFKBsTsACVnOVXZAJmuXYKUDYmYAEqOcuvyATMcu0UoGxMwAJUcpZfkQmY5dopQNkmzg846nw7m77Fge9xzH7wgZhaPT+wSodN35qf1+kibef8eTHz3rsD0+51w7D59Xq2V9yk+UUnjoC9QD8sDhs+4odNfqZWV8U8fTQwjs3AsYsptlDTn96ivVt2iZDT770n5i79Lpb0D3unPF0rVBMMstT+8MdEPpUFQoLkSD8vi8bTIHqhCAhAQRR8KiupHemRPhaN53lLtTiJOfFN8CCbp7FxV9RJM+398EMbN5Bkl3YfxffaBkm\/9P2Hv2gSI2337t0uQmNLNeSD7wSPIv3yGyWNSbp34gk4CGx0PPCD3RfcY8\/Yb7ALxxH5+lmBn+nY7H3\/g04\/qFnRJDtvvSWO\/faTcbIoxDOFaYLnLl\/SnZBgrYI0ccnMxQ9Er68doTnmz7P2R7kwBAQE6KEGpUFNZ5wCLdubhPndYjcqfoUiYPj7vMHmMiqQ5nmQEK6eoKC5hz3I0o1AoQgI53EaArsybFvWY2zu03iHtPIoFAHRIw5KWCMGr0U9n363c2QEznCWbgQKRcB6wBUDKOTZs92IxBRjescmubjtTZPupB9z74YxFQQXDNwiQZm9eDEYjPU8PNznD2kDjjo2POl+w1wTEIa\/+9P\/tH9Oj9kGKAaCTI85gSCQTN\/TsL3JnZDeUE08AUfVGIAB5IC7hOXoESiUDQi4QT4MwYWbyLirIqzxwhox7vwmNb2J14CjAB\/ndKxB+aLpD8qwhJ90my74zsOc556Akmy9GXKJYK5euGc6DEDj3hMefkuyxz1uGbPw3MQTMKsao\/5N54dkZugfgKUbgcLZgN0QxB+DSQ7hYT5niOUA8Zck+yk6\/vZTXUpfedkv7QSUEMQLTvtCkWdoPcqwNmDWX9F\/8iSWIvq1Zzod1oCxwNlMBOTb6THbGlPBWHoj4FhC1JQQJaWUsCwKsYyFwCuy+fARwbD7Ze7Spdxov7GA6fEQuNaSmkOnNQowAQ0kQx4xJb9BEwwwHR\/T8sPEQzJoeln7dQPaQUB7cVGQ7hOytCCk5BY5DNc4Iy2GfMf\/+pdwchMXlidPxl9m3xfSniLWCTHxbpj40YmWIkY80OzyOpDhcGQCDofTwLtAvGOffKKJx8NuA+Fq38AEbEMx2glIBtfKFG3LgVEW5+239DjzaKkU826\/1QlRQtWsx1tbd8gIXFtYmBdTDvOxmJRI960brit2dmiNjCXWudeRLvacWwgBEBBuGKH8tm8mdAsHGYHkEJDkk9FjIgHfTHK5ccqMACHgeb7GgdwwVW6CmRLpI3AwEiIkWIgSeOQcZGEE0kCg3QtW6t6BDRhgZRqF4DyKi0DA3KtJy7eanRAmYHEZkfKb+8YGtKyqVI5VRf6uy\/MBU66HwmbXboI9qyZd160CiYBaLCww\/OLpIOC3+hvurFOVy5VKFdkikn2B6VRA0XMxBFxeXm66YSyhqgCFxuaKjg2\/f8IIuJ4x9dQGstKDv8qyaAM7UW40XDEzM51wEUZLPq41CKPlmp+7E5nPFwEe0wEhp989JKMd0Rb5YxA4YCdCLIxA\/AhgIgKEiKc1YHMkxLLWEelxTxgwsCSIgPG20PqjAwLanreOPKEBuSOSIPqcNLn7mhrQcE7bgIuVSo3mBa6TK2bN9T0xJbM7LzBrNk3WOJVlm9k0v9Td3QDngF2zCcaZUv\/FYX+\/gQMLIxA7Anv1fZ0m+Vo01xA4IKAv1xGxt9e8CecsjECcCLQ1oO\/fNOm2CXi68uY6pkhjRKR9o7mLj4xARASg2PRgB82+OlOp6A4IkmwTUKev1Hc4vnpZ10H+wwjEhUDdtKyW+DyYZgcBnaZqrEEDshYMwsTnURAAl9D7JduveubcuZvBtDoI2OyZqBu4gbVgECY+j4LA7u5L\/Ti5+G6F0+kgIC6SFrxOY8JVsLZe3wvfz2FGYCQEgrbf2crKZ+GHuwgILSh96ypufPmqzo7pMGIcHhoBLPMAh7SEbD+TSBcBceFU5dxt0yPefdFUn+YBPjICwyIAM05PvbLE7bDtZ9LoSUBcpGG539Ohtt9ocFNs0OLj0AjAfNvb1z7lmutN6Ra118N9CagnqvpKd5mhRnnVXC\/4OK4XAsGmV1ni6nJludrrPsT1JSAunq6sXKfJqjfgnMZeHkxCoMJyGALgCLgCzlCv90a\/ptekcSgBcZPt+59h8Bht+fPnL7hTYpDjYxcCIB040hzxUBtnKitXum4KRQwkIHrFru9\/DNeMR9O1nj0ndvM+MiEYOQjyPUMriSl95HD2\/OmPh0FlIAGRCOxBUq3vMwmHgbR493STb+r9w+y+IEJDERAP9CIh24RBKIt5Dg50ar7hyQfEhiYgbg6TkDsmQKW4YjocB83uaOQDciMREA8YEpqOybNnz9lPCGAKJvDzoe5Nh8PzRycfIBuZgHgIJDy9svKOcdG8ePlKYMCZm2Sgk28xPV3UOc7hanlB\/YNhbb4wOmMR0CRyamXlivKFHjGB1xtNMs+oNujk7witt13bERgdI6kJX12Fq6XSWt8xzhtHIiAyPFM5d5MWMr1DY8e3oY4xdoxC8nzCcaojm8+gLqFcjNbDPAHXn3oHAxVRS2xFTSD4\/KPNrctCqmuWsMqIx6772Gkhym4L4VVevCoOyPaXOPEC8TChwCgT+Peoxbt6FpNVYpJYCWjK9Hjz3mdKikuGiPgEmCbj7PTIn4KIE1BTvjwfo+AFmw5rw7EyEqYUwi1Bc3tjV\/jXozS3JrHgMRECmgzCGtHEg4y2Y2sySlsKx7bNpa5jFEC7EitAxLB46Q4EEWyf9gOCGwW7YuiNCQ5Ip7\/jQSz8bpeWasRNPFMViRLQZPJo8+dV2vjjsiXFBXorOu8WaEmbfvhkLEipj3SOD2oj3oh96hRtbN1ZbNyLX5HEECj8zo3Hj3UUrmMjSLl0sukqoXPEYWsMfY3s9Z5C9p3wsEZcruuVkj1vii8y9Vrb3NwsHRf2mpJqlVhzntAo9yMlXtN80d28slxcMqd87IHAKHhhWz7sjKY8bBZurT8X3npSmq5HUXVU6gTsV5AHmw\/KjnDLBEqJyFmm+0oEzop6+pQ6XQJhLdbiYonCJRPGkT43i3BHXPB6Ts9rhFUt\/G7+9nYVcWS94VrNWloSrd3PatgPnLCqusKpjuu3Q9pxyv8BVb3XBNS3Vn0AAAAASUVORK5CYII=" + }, + "id": "629cb5b8-490a-48bc-808b-ffc13085cb4f", + "name": "Complex PDF with Images & Tables" +} + } +} \ No newline at end of file diff --git a/api/contexts/__init__.py b/api/contexts/__init__.py index 2126a06f75..7c16bc231f 100644 --- a/api/contexts/__init__.py +++ b/api/contexts/__init__.py @@ -9,6 +9,7 @@ if TYPE_CHECKING: from core.model_runtime.entities.model_entities import AIModelEntity from core.plugin.entities.plugin_daemon import PluginModelProviderEntity from core.tools.plugin_tool.provider import PluginToolProviderController + from core.trigger.provider import PluginTriggerProviderController """ @@ -41,3 +42,11 @@ datasource_plugin_providers: RecyclableContextVar[dict[str, "DatasourcePluginPro datasource_plugin_providers_lock: RecyclableContextVar[Lock] = RecyclableContextVar( ContextVar("datasource_plugin_providers_lock") ) + +plugin_trigger_providers: RecyclableContextVar[dict[str, "PluginTriggerProviderController"]] = RecyclableContextVar( + ContextVar("plugin_trigger_providers") +) + +plugin_trigger_providers_lock: RecyclableContextVar[Lock] = RecyclableContextVar( + ContextVar("plugin_trigger_providers_lock") +) diff --git a/api/controllers/common/errors.py b/api/controllers/common/errors.py index 6e2ea952fc..252cf3549a 100644 --- a/api/controllers/common/errors.py +++ b/api/controllers/common/errors.py @@ -25,6 +25,12 @@ class UnsupportedFileTypeError(BaseHTTPException): code = 415 +class BlockedFileExtensionError(BaseHTTPException): + error_code = "file_extension_blocked" + description = "The file extension is blocked for security reasons." + code = 400 + + class TooManyFilesError(BaseHTTPException): error_code = "too_many_files" description = "Only one file is allowed." diff --git a/api/controllers/console/__init__.py b/api/controllers/console/__init__.py index 621f5066e4..ad878fc266 100644 --- a/api/controllers/console/__init__.py +++ b/api/controllers/console/__init__.py @@ -66,6 +66,7 @@ from .app import ( workflow_draft_variable, workflow_run, workflow_statistic, + workflow_trigger, ) # Import auth controllers @@ -126,6 +127,7 @@ from .workspace import ( models, plugin, tool_providers, + trigger_providers, workspace, ) @@ -196,6 +198,7 @@ __all__ = [ "statistic", "tags", "tool_providers", + "trigger_providers", "version", "website", "workflow", @@ -203,5 +206,6 @@ __all__ = [ "workflow_draft_variable", "workflow_run", "workflow_statistic", + "workflow_trigger", "workspace", ] diff --git a/api/controllers/console/app/advanced_prompt_template.py b/api/controllers/console/app/advanced_prompt_template.py index 5885d7b447..075345d860 100644 --- a/api/controllers/console/app/advanced_prompt_template.py +++ b/api/controllers/console/app/advanced_prompt_template.py @@ -5,18 +5,20 @@ from controllers.console.wraps import account_initialization_required, setup_req from libs.login import login_required from services.advanced_prompt_template_service import AdvancedPromptTemplateService +parser = ( + reqparse.RequestParser() + .add_argument("app_mode", type=str, required=True, location="args", help="Application mode") + .add_argument("model_mode", type=str, required=True, location="args", help="Model mode") + .add_argument("has_context", type=str, required=False, default="true", location="args", help="Whether has context") + .add_argument("model_name", type=str, required=True, location="args", help="Model name") +) + @console_ns.route("/app/prompt-templates") class AdvancedPromptTemplateList(Resource): @api.doc("get_advanced_prompt_templates") @api.doc(description="Get advanced prompt templates based on app mode and model configuration") - @api.expect( - api.parser() - .add_argument("app_mode", type=str, required=True, location="args", help="Application mode") - .add_argument("model_mode", type=str, required=True, location="args", help="Model mode") - .add_argument("has_context", type=str, default="true", location="args", help="Whether has context") - .add_argument("model_name", type=str, required=True, location="args", help="Model name") - ) + @api.expect(parser) @api.response( 200, "Prompt templates retrieved successfully", fields.List(fields.Raw(description="Prompt template data")) ) @@ -25,13 +27,6 @@ class AdvancedPromptTemplateList(Resource): @login_required @account_initialization_required def get(self): - parser = ( - reqparse.RequestParser() - .add_argument("app_mode", type=str, required=True, location="args") - .add_argument("model_mode", type=str, required=True, location="args") - .add_argument("has_context", type=str, required=False, default="true", location="args") - .add_argument("model_name", type=str, required=True, location="args") - ) args = parser.parse_args() return AdvancedPromptTemplateService.get_prompt(args) diff --git a/api/controllers/console/app/agent.py b/api/controllers/console/app/agent.py index 717263a74d..fde28fdb98 100644 --- a/api/controllers/console/app/agent.py +++ b/api/controllers/console/app/agent.py @@ -8,17 +8,19 @@ from libs.login import login_required from models.model import AppMode from services.agent_service import AgentService +parser = ( + reqparse.RequestParser() + .add_argument("message_id", type=uuid_value, required=True, location="args", help="Message UUID") + .add_argument("conversation_id", type=uuid_value, required=True, location="args", help="Conversation UUID") +) + @console_ns.route("/apps//agent/logs") class AgentLogApi(Resource): @api.doc("get_agent_logs") @api.doc(description="Get agent execution logs for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("message_id", type=str, required=True, location="args", help="Message UUID") - .add_argument("conversation_id", type=str, required=True, location="args", help="Conversation UUID") - ) + @api.expect(parser) @api.response(200, "Agent logs retrieved successfully", fields.List(fields.Raw(description="Agent log entries"))) @api.response(400, "Invalid request parameters") @setup_required @@ -27,12 +29,6 @@ class AgentLogApi(Resource): @get_app_model(mode=[AppMode.AGENT_CHAT]) def get(self, app_model): """Get agent logs""" - parser = ( - reqparse.RequestParser() - .add_argument("message_id", type=uuid_value, required=True, location="args") - .add_argument("conversation_id", type=uuid_value, required=True, location="args") - ) - args = parser.parse_args() return AgentService.get_agent_logs(app_model, args["conversation_id"], args["message_id"]) diff --git a/api/controllers/console/app/annotation.py b/api/controllers/console/app/annotation.py index 932214058a..bc4113b5c7 100644 --- a/api/controllers/console/app/annotation.py +++ b/api/controllers/console/app/annotation.py @@ -16,6 +16,7 @@ from fields.annotation_fields import ( annotation_fields, annotation_hit_history_fields, ) +from libs.helper import uuid_value from libs.login import login_required from services.annotation_service import AppAnnotationService @@ -175,8 +176,10 @@ class AnnotationApi(Resource): api.model( "CreateAnnotationRequest", { - "question": fields.String(required=True, description="Question text"), - "answer": fields.String(required=True, description="Answer text"), + "message_id": fields.String(description="Message ID (optional)"), + "question": fields.String(description="Question text (required when message_id not provided)"), + "answer": fields.String(description="Answer text (use 'answer' or 'content')"), + "content": fields.String(description="Content text (use 'answer' or 'content')"), "annotation_reply": fields.Raw(description="Annotation reply data"), }, ) @@ -193,11 +196,14 @@ class AnnotationApi(Resource): app_id = str(app_id) parser = ( reqparse.RequestParser() - .add_argument("question", required=True, type=str, location="json") - .add_argument("answer", required=True, type=str, location="json") + .add_argument("message_id", required=False, type=uuid_value, location="json") + .add_argument("question", required=False, type=str, location="json") + .add_argument("answer", required=False, type=str, location="json") + .add_argument("content", required=False, type=str, location="json") + .add_argument("annotation_reply", required=False, type=dict, location="json") ) args = parser.parse_args() - annotation = AppAnnotationService.insert_app_annotation_directly(args, app_id) + annotation = AppAnnotationService.up_insert_app_annotation_from_message(args, app_id) return annotation @setup_required @@ -245,6 +251,13 @@ class AnnotationExportApi(Resource): return response, 200 +parser = ( + reqparse.RequestParser() + .add_argument("question", required=True, type=str, location="json") + .add_argument("answer", required=True, type=str, location="json") +) + + @console_ns.route("/apps//annotations/") class AnnotationUpdateDeleteApi(Resource): @api.doc("update_delete_annotation") @@ -253,6 +266,7 @@ class AnnotationUpdateDeleteApi(Resource): @api.response(200, "Annotation updated successfully", annotation_fields) @api.response(204, "Annotation deleted successfully") @api.response(403, "Insufficient permissions") + @api.expect(parser) @setup_required @login_required @account_initialization_required @@ -262,11 +276,6 @@ class AnnotationUpdateDeleteApi(Resource): def post(self, app_id, annotation_id): app_id = str(app_id) annotation_id = str(annotation_id) - parser = ( - reqparse.RequestParser() - .add_argument("question", required=True, type=str, location="json") - .add_argument("answer", required=True, type=str, location="json") - ) args = parser.parse_args() annotation = AppAnnotationService.update_app_annotation_directly(args, app_id, annotation_id) return annotation diff --git a/api/controllers/console/app/app.py b/api/controllers/console/app/app.py index 17505d69b2..0724a6355d 100644 --- a/api/controllers/console/app/app.py +++ b/api/controllers/console/app/app.py @@ -15,11 +15,12 @@ from controllers.console.wraps import ( setup_required, ) from core.ops.ops_trace_manager import OpsTraceManager +from core.workflow.enums import NodeType from extensions.ext_database import db from fields.app_fields import app_detail_fields, app_detail_fields_with_site, app_pagination_fields from libs.login import current_account_with_tenant, login_required from libs.validators import validate_description_length -from models import App +from models import App, Workflow from services.app_dsl_service import AppDslService, ImportMode from services.app_service import AppService from services.enterprise.enterprise_service import EnterpriseService @@ -106,6 +107,35 @@ class AppListApi(Resource): if str(app.id) in res: app.access_mode = res[str(app.id)].access_mode + workflow_capable_app_ids = [ + str(app.id) for app in app_pagination.items if app.mode in {"workflow", "advanced-chat"} + ] + draft_trigger_app_ids: set[str] = set() + if workflow_capable_app_ids: + draft_workflows = ( + db.session.execute( + select(Workflow).where( + Workflow.version == Workflow.VERSION_DRAFT, + Workflow.app_id.in_(workflow_capable_app_ids), + ) + ) + .scalars() + .all() + ) + trigger_node_types = { + NodeType.TRIGGER_WEBHOOK, + NodeType.TRIGGER_SCHEDULE, + NodeType.TRIGGER_PLUGIN, + } + for workflow in draft_workflows: + for _, node_data in workflow.walk_nodes(): + if node_data.get("type") in trigger_node_types: + draft_trigger_app_ids.add(str(workflow.app_id)) + break + + for app in app_pagination.items: + app.has_draft_trigger = str(app.id) in draft_trigger_app_ids + return marshal(app_pagination, app_pagination_fields), 200 @api.doc("create_app") @@ -353,12 +383,15 @@ class AppExportApi(Resource): } +parser = reqparse.RequestParser().add_argument("name", type=str, required=True, location="json", help="Name to check") + + @console_ns.route("/apps//name") class AppNameApi(Resource): @api.doc("check_app_name") @api.doc(description="Check if app name is available") @api.doc(params={"app_id": "Application ID"}) - @api.expect(api.parser().add_argument("name", type=str, required=True, location="args", help="Name to check")) + @api.expect(parser) @api.response(200, "Name availability checked") @setup_required @login_required @@ -367,7 +400,6 @@ class AppNameApi(Resource): @marshal_with(app_detail_fields) @edit_permission_required def post(self, app_model): - parser = reqparse.RequestParser().add_argument("name", type=str, required=True, location="json") args = parser.parse_args() app_service = AppService() diff --git a/api/controllers/console/app/app_import.py b/api/controllers/console/app/app_import.py index d902c129ad..02dbd42515 100644 --- a/api/controllers/console/app/app_import.py +++ b/api/controllers/console/app/app_import.py @@ -1,6 +1,7 @@ from flask_restx import Resource, marshal_with, reqparse from sqlalchemy.orm import Session +from controllers.console import api from controllers.console.app.wraps import get_app_model from controllers.console.wraps import ( account_initialization_required, @@ -18,9 +19,23 @@ from services.feature_service import FeatureService from .. import console_ns +parser = ( + reqparse.RequestParser() + .add_argument("mode", type=str, required=True, location="json") + .add_argument("yaml_content", type=str, location="json") + .add_argument("yaml_url", type=str, location="json") + .add_argument("name", type=str, location="json") + .add_argument("description", type=str, location="json") + .add_argument("icon_type", type=str, location="json") + .add_argument("icon", type=str, location="json") + .add_argument("icon_background", type=str, location="json") + .add_argument("app_id", type=str, location="json") +) + @console_ns.route("/apps/imports") class AppImportApi(Resource): + @api.expect(parser) @setup_required @login_required @account_initialization_required @@ -30,18 +45,6 @@ class AppImportApi(Resource): def post(self): # Check user role first current_user, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("mode", type=str, required=True, location="json") - .add_argument("yaml_content", type=str, location="json") - .add_argument("yaml_url", type=str, location="json") - .add_argument("name", type=str, location="json") - .add_argument("description", type=str, location="json") - .add_argument("icon_type", type=str, location="json") - .add_argument("icon", type=str, location="json") - .add_argument("icon_background", type=str, location="json") - .add_argument("app_id", type=str, location="json") - ) args = parser.parse_args() # Create service with session diff --git a/api/controllers/console/app/conversation.py b/api/controllers/console/app/conversation.py index d5fa70d678..57b6c314f3 100644 --- a/api/controllers/console/app/conversation.py +++ b/api/controllers/console/app/conversation.py @@ -1,7 +1,5 @@ -from datetime import datetime - -import pytz import sqlalchemy as sa +from flask import abort from flask_restx import Resource, marshal_with, reqparse from flask_restx.inputs import int_range from sqlalchemy import func, or_ @@ -19,7 +17,7 @@ from fields.conversation_fields import ( conversation_pagination_fields, conversation_with_summary_pagination_fields, ) -from libs.datetime_utils import naive_utc_now +from libs.datetime_utils import naive_utc_now, parse_time_range from libs.helper import DatetimeString from libs.login import current_account_with_tenant, login_required from models import Conversation, EndUser, Message, MessageAnnotation @@ -90,25 +88,17 @@ class CompletionConversationApi(Resource): account = current_user assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: query = query.where(Conversation.created_at >= start_datetime_utc) - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=59) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: + end_datetime_utc = end_datetime_utc.replace(second=59) query = query.where(Conversation.created_at < end_datetime_utc) # FIXME, the type ignore in this file @@ -270,29 +260,21 @@ class ChatConversationApi(Resource): account = current_user assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: match args["sort_by"]: case "updated_at" | "-updated_at": query = query.where(Conversation.updated_at >= start_datetime_utc) case "created_at" | "-created_at" | _: query = query.where(Conversation.created_at >= start_datetime_utc) - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=59) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: + end_datetime_utc = end_datetime_utc.replace(second=59) match args["sort_by"]: case "updated_at" | "-updated_at": query = query.where(Conversation.updated_at <= end_datetime_utc) diff --git a/api/controllers/console/app/generator.py b/api/controllers/console/app/generator.py index b6ca97ab4f..54a101946c 100644 --- a/api/controllers/console/app/generator.py +++ b/api/controllers/console/app/generator.py @@ -11,6 +11,7 @@ from controllers.console.app.error import ( ) from controllers.console.wraps import account_initialization_required, setup_required from core.errors.error import ModelCurrentlyNotSupportError, ProviderTokenNotInitError, QuotaExceededError +from core.helper.code_executor.code_node_provider import CodeNodeProvider from core.helper.code_executor.javascript.javascript_code_provider import JavascriptCodeProvider from core.helper.code_executor.python3.python3_code_provider import Python3CodeProvider from core.llm_generator.llm_generator import LLMGenerator @@ -206,13 +207,11 @@ class InstructionGenerateApi(Resource): ) args = parser.parse_args() _, current_tenant_id = current_account_with_tenant() - code_template = ( - Python3CodeProvider.get_default_code() - if args["language"] == "python" - else (JavascriptCodeProvider.get_default_code()) - if args["language"] == "javascript" - else "" + providers: list[type[CodeNodeProvider]] = [Python3CodeProvider, JavascriptCodeProvider] + code_provider: type[CodeNodeProvider] | None = next( + (p for p in providers if p.is_accept_language(args["language"])), None ) + code_template = code_provider.get_default_code() if code_provider else "" try: # Generate from nothing for a workflow node if (args["current"] == code_template or args["current"] == "") and args["node_id"] != "": diff --git a/api/controllers/console/app/message.py b/api/controllers/console/app/message.py index 7e0ae370ef..3f66278940 100644 --- a/api/controllers/console/app/message.py +++ b/api/controllers/console/app/message.py @@ -16,7 +16,6 @@ from controllers.console.app.wraps import get_app_model from controllers.console.explore.error import AppSuggestedQuestionsAfterAnswerDisabledError from controllers.console.wraps import ( account_initialization_required, - cloud_edition_billing_resource_check, edit_permission_required, setup_required, ) @@ -24,12 +23,11 @@ from core.app.entities.app_invoke_entities import InvokeFrom from core.errors.error import ModelCurrentlyNotSupportError, ProviderTokenNotInitError, QuotaExceededError from core.model_runtime.errors.invoke import InvokeError from extensions.ext_database import db -from fields.conversation_fields import annotation_fields, message_detail_fields +from fields.conversation_fields import message_detail_fields from libs.helper import uuid_value from libs.infinite_scroll_pagination import InfiniteScrollPagination from libs.login import current_account_with_tenant, login_required from models.model import AppMode, Conversation, Message, MessageAnnotation, MessageFeedback -from services.annotation_service import AppAnnotationService from services.errors.conversation import ConversationNotExistsError from services.errors.message import MessageNotExistsError, SuggestedQuestionsAfterAnswerDisabledError from services.message_service import MessageService @@ -194,45 +192,6 @@ class MessageFeedbackApi(Resource): return {"result": "success"} -@console_ns.route("/apps//annotations") -class MessageAnnotationApi(Resource): - @api.doc("create_message_annotation") - @api.doc(description="Create message annotation") - @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.model( - "MessageAnnotationRequest", - { - "message_id": fields.String(description="Message ID"), - "question": fields.String(required=True, description="Question text"), - "answer": fields.String(required=True, description="Answer text"), - "annotation_reply": fields.Raw(description="Annotation reply"), - }, - ) - ) - @api.response(200, "Annotation created successfully", annotation_fields) - @api.response(403, "Insufficient permissions") - @marshal_with(annotation_fields) - @get_app_model - @setup_required - @login_required - @cloud_edition_billing_resource_check("annotation") - @account_initialization_required - @edit_permission_required - def post(self, app_model): - parser = ( - reqparse.RequestParser() - .add_argument("message_id", required=False, type=uuid_value, location="json") - .add_argument("question", required=True, type=str, location="json") - .add_argument("answer", required=True, type=str, location="json") - .add_argument("annotation_reply", required=False, type=dict, location="json") - ) - args = parser.parse_args() - annotation = AppAnnotationService.up_insert_app_annotation_from_message(args, app_model.id) - - return annotation - - @console_ns.route("/apps//annotations/count") class MessageAnnotationCountApi(Resource): @api.doc("get_annotation_count") diff --git a/api/controllers/console/app/statistic.py b/api/controllers/console/app/statistic.py index 0917a6e53c..37ed3d9e27 100644 --- a/api/controllers/console/app/statistic.py +++ b/api/controllers/console/app/statistic.py @@ -1,9 +1,7 @@ -from datetime import datetime from decimal import Decimal -import pytz import sqlalchemy as sa -from flask import jsonify +from flask import abort, jsonify from flask_restx import Resource, fields, reqparse from controllers.console import api, console_ns @@ -11,6 +9,7 @@ from controllers.console.app.wraps import get_app_model from controllers.console.wraps import account_initialization_required, setup_required from core.app.entities.app_invoke_entities import InvokeFrom from extensions.ext_database import db +from libs.datetime_utils import parse_time_range from libs.helper import DatetimeString from libs.login import current_account_with_tenant, login_required from models import AppMode, Message @@ -56,26 +55,16 @@ WHERE arg_dict = {"tz": account.timezone, "app_id": app_model.id, "invoke_from": InvokeFrom.DEBUGGER} assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: sql_query += " AND created_at >= :start" arg_dict["start"] = start_datetime_utc - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: sql_query += " AND created_at < :end" arg_dict["end"] = end_datetime_utc @@ -91,16 +80,19 @@ WHERE return jsonify({"data": response_data}) +parser = ( + reqparse.RequestParser() + .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args", help="Start date (YYYY-MM-DD HH:MM)") + .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args", help="End date (YYYY-MM-DD HH:MM)") +) + + @console_ns.route("/apps//statistics/daily-conversations") class DailyConversationStatistic(Resource): @api.doc("get_daily_conversation_statistics") @api.doc(description="Get daily conversation statistics for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("start", type=str, location="args", help="Start date (YYYY-MM-DD HH:MM)") - .add_argument("end", type=str, location="args", help="End date (YYYY-MM-DD HH:MM)") - ) + @api.expect(parser) @api.response( 200, "Daily conversation statistics retrieved successfully", @@ -113,15 +105,13 @@ class DailyConversationStatistic(Resource): def get(self, app_model): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - ) args = parser.parse_args() assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc + + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) stmt = ( sa.select( @@ -134,18 +124,10 @@ class DailyConversationStatistic(Resource): .where(Message.app_id == app_model.id, Message.invoke_from != InvokeFrom.DEBUGGER) ) - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + if start_datetime_utc: stmt = stmt.where(Message.created_at >= start_datetime_utc) - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) + if end_datetime_utc: stmt = stmt.where(Message.created_at < end_datetime_utc) stmt = stmt.group_by("date").order_by("date") @@ -164,11 +146,7 @@ class DailyTerminalsStatistic(Resource): @api.doc("get_daily_terminals_statistics") @api.doc(description="Get daily terminal/end-user statistics for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("start", type=str, location="args", help="Start date (YYYY-MM-DD HH:MM)") - .add_argument("end", type=str, location="args", help="End date (YYYY-MM-DD HH:MM)") - ) + @api.expect(parser) @api.response( 200, "Daily terminal statistics retrieved successfully", @@ -181,11 +159,6 @@ class DailyTerminalsStatistic(Resource): def get(self, app_model): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - ) args = parser.parse_args() sql_query = """SELECT @@ -198,26 +171,17 @@ WHERE AND invoke_from != :invoke_from""" arg_dict = {"tz": account.timezone, "app_id": app_model.id, "invoke_from": InvokeFrom.DEBUGGER} assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: sql_query += " AND created_at >= :start" arg_dict["start"] = start_datetime_utc - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: sql_query += " AND created_at < :end" arg_dict["end"] = end_datetime_utc @@ -238,11 +202,7 @@ class DailyTokenCostStatistic(Resource): @api.doc("get_daily_token_cost_statistics") @api.doc(description="Get daily token cost statistics for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("start", type=str, location="args", help="Start date (YYYY-MM-DD HH:MM)") - .add_argument("end", type=str, location="args", help="End date (YYYY-MM-DD HH:MM)") - ) + @api.expect(parser) @api.response( 200, "Daily token cost statistics retrieved successfully", @@ -255,11 +215,6 @@ class DailyTokenCostStatistic(Resource): def get(self, app_model): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - ) args = parser.parse_args() sql_query = """SELECT @@ -273,26 +228,17 @@ WHERE AND invoke_from != :invoke_from""" arg_dict = {"tz": account.timezone, "app_id": app_model.id, "invoke_from": InvokeFrom.DEBUGGER} assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: sql_query += " AND created_at >= :start" arg_dict["start"] = start_datetime_utc - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: sql_query += " AND created_at < :end" arg_dict["end"] = end_datetime_utc @@ -315,11 +261,7 @@ class AverageSessionInteractionStatistic(Resource): @api.doc("get_average_session_interaction_statistics") @api.doc(description="Get average session interaction statistics for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("start", type=str, location="args", help="Start date (YYYY-MM-DD HH:MM)") - .add_argument("end", type=str, location="args", help="End date (YYYY-MM-DD HH:MM)") - ) + @api.expect(parser) @api.response( 200, "Average session interaction statistics retrieved successfully", @@ -332,11 +274,6 @@ class AverageSessionInteractionStatistic(Resource): def get(self, app_model): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - ) args = parser.parse_args() sql_query = """SELECT @@ -357,26 +294,17 @@ FROM AND m.invoke_from != :invoke_from""" arg_dict = {"tz": account.timezone, "app_id": app_model.id, "invoke_from": InvokeFrom.DEBUGGER} assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: sql_query += " AND c.created_at >= :start" arg_dict["start"] = start_datetime_utc - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: sql_query += " AND c.created_at < :end" arg_dict["end"] = end_datetime_utc @@ -408,11 +336,7 @@ class UserSatisfactionRateStatistic(Resource): @api.doc("get_user_satisfaction_rate_statistics") @api.doc(description="Get user satisfaction rate statistics for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("start", type=str, location="args", help="Start date (YYYY-MM-DD HH:MM)") - .add_argument("end", type=str, location="args", help="End date (YYYY-MM-DD HH:MM)") - ) + @api.expect(parser) @api.response( 200, "User satisfaction rate statistics retrieved successfully", @@ -425,11 +349,6 @@ class UserSatisfactionRateStatistic(Resource): def get(self, app_model): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - ) args = parser.parse_args() sql_query = """SELECT @@ -446,26 +365,17 @@ WHERE AND m.invoke_from != :invoke_from""" arg_dict = {"tz": account.timezone, "app_id": app_model.id, "invoke_from": InvokeFrom.DEBUGGER} assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: sql_query += " AND m.created_at >= :start" arg_dict["start"] = start_datetime_utc - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: sql_query += " AND m.created_at < :end" arg_dict["end"] = end_datetime_utc @@ -491,11 +401,7 @@ class AverageResponseTimeStatistic(Resource): @api.doc("get_average_response_time_statistics") @api.doc(description="Get average response time statistics for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("start", type=str, location="args", help="Start date (YYYY-MM-DD HH:MM)") - .add_argument("end", type=str, location="args", help="End date (YYYY-MM-DD HH:MM)") - ) + @api.expect(parser) @api.response( 200, "Average response time statistics retrieved successfully", @@ -508,11 +414,6 @@ class AverageResponseTimeStatistic(Resource): def get(self, app_model): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - ) args = parser.parse_args() sql_query = """SELECT @@ -525,26 +426,17 @@ WHERE AND invoke_from != :invoke_from""" arg_dict = {"tz": account.timezone, "app_id": app_model.id, "invoke_from": InvokeFrom.DEBUGGER} assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: sql_query += " AND created_at >= :start" arg_dict["start"] = start_datetime_utc - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: sql_query += " AND created_at < :end" arg_dict["end"] = end_datetime_utc @@ -565,11 +457,7 @@ class TokensPerSecondStatistic(Resource): @api.doc("get_tokens_per_second_statistics") @api.doc(description="Get tokens per second statistics for an application") @api.doc(params={"app_id": "Application ID"}) - @api.expect( - api.parser() - .add_argument("start", type=str, location="args", help="Start date (YYYY-MM-DD HH:MM)") - .add_argument("end", type=str, location="args", help="End date (YYYY-MM-DD HH:MM)") - ) + @api.expect(parser) @api.response( 200, "Tokens per second statistics retrieved successfully", @@ -581,12 +469,6 @@ class TokensPerSecondStatistic(Resource): @account_initialization_required def get(self, app_model): account, _ = current_account_with_tenant() - - parser = ( - reqparse.RequestParser() - .add_argument("start", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - .add_argument("end", type=DatetimeString("%Y-%m-%d %H:%M"), location="args") - ) args = parser.parse_args() sql_query = """SELECT @@ -602,26 +484,17 @@ WHERE AND invoke_from != :invoke_from""" arg_dict = {"tz": account.timezone, "app_id": app_model.id, "invoke_from": InvokeFrom.DEBUGGER} assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) - - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) + try: + start_datetime_utc, end_datetime_utc = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) + if start_datetime_utc: sql_query += " AND created_at >= :start" arg_dict["start"] = start_datetime_utc - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - + if end_datetime_utc: sql_query += " AND created_at < :end" arg_dict["end"] = end_datetime_utc diff --git a/api/controllers/console/app/workflow.py b/api/controllers/console/app/workflow.py index 56771ed420..31077e371b 100644 --- a/api/controllers/console/app/workflow.py +++ b/api/controllers/console/app/workflow.py @@ -16,9 +16,19 @@ from controllers.console.wraps import account_initialization_required, edit_perm from controllers.web.error import InvokeRateLimitError as InvokeRateLimitHttpError from core.app.app_config.features.file_upload.manager import FileUploadConfigManager from core.app.apps.base_app_queue_manager import AppQueueManager +from core.app.apps.workflow.app_generator import SKIP_PREPARE_USER_INPUTS_KEY from core.app.entities.app_invoke_entities import InvokeFrom from core.file.models import File from core.helper.trace_id_helper import get_external_trace_id +from core.model_runtime.utils.encoders import jsonable_encoder +from core.plugin.impl.exc import PluginInvokeError +from core.trigger.debug.event_selectors import ( + TriggerDebugEvent, + TriggerDebugEventPoller, + create_event_poller, + select_trigger_debug_events, +) +from core.workflow.enums import NodeType from core.workflow.graph_engine.manager import GraphEngineManager from extensions.ext_database import db from factories import file_factory, variable_factory @@ -37,6 +47,7 @@ from services.errors.llm import InvokeRateLimitError from services.workflow_service import DraftWorkflowDeletionError, WorkflowInUseError, WorkflowService logger = logging.getLogger(__name__) +LISTENING_RETRY_IN = 2000 # TODO(QuantumGhost): Refactor existing node run API to handle file parameter parsing @@ -102,7 +113,18 @@ class DraftWorkflowApi(Resource): }, ) ) - @api.response(200, "Draft workflow synced successfully", workflow_fields) + @api.response( + 200, + "Draft workflow synced successfully", + api.model( + "SyncDraftWorkflowResponse", + { + "result": fields.String, + "hash": fields.String, + "updated_at": fields.String, + }, + ), + ) @api.response(400, "Invalid workflow configuration") @api.response(403, "Permission denied") @edit_permission_required @@ -564,6 +586,13 @@ class DraftWorkflowNodeRunApi(Resource): return workflow_node_execution +parser_publish = ( + reqparse.RequestParser() + .add_argument("marked_name", type=str, required=False, default="", location="json") + .add_argument("marked_comment", type=str, required=False, default="", location="json") +) + + @console_ns.route("/apps//workflows/publish") class PublishedWorkflowApi(Resource): @api.doc("get_published_workflow") @@ -588,6 +617,7 @@ class PublishedWorkflowApi(Resource): # return workflow, if not found, return None return workflow + @api.expect(parser_publish) @setup_required @login_required @account_initialization_required @@ -598,12 +628,8 @@ class PublishedWorkflowApi(Resource): Publish workflow """ current_user, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("marked_name", type=str, required=False, default="", location="json") - .add_argument("marked_comment", type=str, required=False, default="", location="json") - ) - args = parser.parse_args() + + args = parser_publish.parse_args() # Validate name and comment length if args.marked_name and len(args.marked_name) > 20: @@ -658,6 +684,9 @@ class DefaultBlockConfigsApi(Resource): return workflow_service.get_default_block_configs() +parser_block = reqparse.RequestParser().add_argument("q", type=str, location="args") + + @console_ns.route("/apps//workflows/default-workflow-block-configs/") class DefaultBlockConfigApi(Resource): @api.doc("get_default_block_config") @@ -665,6 +694,7 @@ class DefaultBlockConfigApi(Resource): @api.doc(params={"app_id": "Application ID", "block_type": "Block type"}) @api.response(200, "Default block configuration retrieved successfully") @api.response(404, "Block type not found") + @api.expect(parser_block) @setup_required @login_required @account_initialization_required @@ -674,8 +704,7 @@ class DefaultBlockConfigApi(Resource): """ Get default block config """ - parser = reqparse.RequestParser().add_argument("q", type=str, location="args") - args = parser.parse_args() + args = parser_block.parse_args() q = args.get("q") @@ -691,8 +720,18 @@ class DefaultBlockConfigApi(Resource): return workflow_service.get_default_block_config(node_type=block_type, filters=filters) +parser_convert = ( + reqparse.RequestParser() + .add_argument("name", type=str, required=False, nullable=True, location="json") + .add_argument("icon_type", type=str, required=False, nullable=True, location="json") + .add_argument("icon", type=str, required=False, nullable=True, location="json") + .add_argument("icon_background", type=str, required=False, nullable=True, location="json") +) + + @console_ns.route("/apps//convert-to-workflow") class ConvertToWorkflowApi(Resource): + @api.expect(parser_convert) @api.doc("convert_to_workflow") @api.doc(description="Convert application to workflow mode") @api.doc(params={"app_id": "Application ID"}) @@ -713,14 +752,7 @@ class ConvertToWorkflowApi(Resource): current_user, _ = current_account_with_tenant() if request.data: - parser = ( - reqparse.RequestParser() - .add_argument("name", type=str, required=False, nullable=True, location="json") - .add_argument("icon_type", type=str, required=False, nullable=True, location="json") - .add_argument("icon", type=str, required=False, nullable=True, location="json") - .add_argument("icon_background", type=str, required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_convert.parse_args() else: args = {} @@ -734,8 +766,18 @@ class ConvertToWorkflowApi(Resource): } +parser_workflows = ( + reqparse.RequestParser() + .add_argument("page", type=inputs.int_range(1, 99999), required=False, default=1, location="args") + .add_argument("limit", type=inputs.int_range(1, 100), required=False, default=10, location="args") + .add_argument("user_id", type=str, required=False, location="args") + .add_argument("named_only", type=inputs.boolean, required=False, default=False, location="args") +) + + @console_ns.route("/apps//workflows") class PublishedAllWorkflowApi(Resource): + @api.expect(parser_workflows) @api.doc("get_all_published_workflows") @api.doc(description="Get all published workflows for an application") @api.doc(params={"app_id": "Application ID"}) @@ -752,16 +794,9 @@ class PublishedAllWorkflowApi(Resource): """ current_user, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("page", type=inputs.int_range(1, 99999), required=False, default=1, location="args") - .add_argument("limit", type=inputs.int_range(1, 100), required=False, default=20, location="args") - .add_argument("user_id", type=str, required=False, location="args") - .add_argument("named_only", type=inputs.boolean, required=False, default=False, location="args") - ) - args = parser.parse_args() - page = int(args.get("page", 1)) - limit = int(args.get("limit", 10)) + args = parser_workflows.parse_args() + page = args["page"] + limit = args["limit"] user_id = args.get("user_id") named_only = args.get("named_only", False) @@ -915,3 +950,234 @@ class DraftWorkflowNodeLastRunApi(Resource): if node_exec is None: raise NotFound("last run not found") return node_exec + + +@console_ns.route("/apps//workflows/draft/trigger/run") +class DraftWorkflowTriggerRunApi(Resource): + """ + Full workflow debug - Polling API for trigger events + Path: /apps//workflows/draft/trigger/run + """ + + @api.doc("poll_draft_workflow_trigger_run") + @api.doc(description="Poll for trigger events and execute full workflow when event arrives") + @api.doc(params={"app_id": "Application ID"}) + @api.expect( + api.model( + "DraftWorkflowTriggerRunRequest", + { + "node_id": fields.String(required=True, description="Node ID"), + }, + ) + ) + @api.response(200, "Trigger event received and workflow executed successfully") + @api.response(403, "Permission denied") + @api.response(500, "Internal server error") + @setup_required + @login_required + @account_initialization_required + @get_app_model(mode=[AppMode.WORKFLOW]) + @edit_permission_required + def post(self, app_model: App): + """ + Poll for trigger events and execute full workflow when event arrives + """ + current_user, _ = current_account_with_tenant() + parser = reqparse.RequestParser() + parser.add_argument("node_id", type=str, required=True, location="json", nullable=False) + args = parser.parse_args() + node_id = args["node_id"] + workflow_service = WorkflowService() + draft_workflow = workflow_service.get_draft_workflow(app_model) + if not draft_workflow: + raise ValueError("Workflow not found") + + poller: TriggerDebugEventPoller = create_event_poller( + draft_workflow=draft_workflow, + tenant_id=app_model.tenant_id, + user_id=current_user.id, + app_id=app_model.id, + node_id=node_id, + ) + event: TriggerDebugEvent | None = None + try: + event = poller.poll() + if not event: + return jsonable_encoder({"status": "waiting", "retry_in": LISTENING_RETRY_IN}) + workflow_args = dict(event.workflow_args) + workflow_args[SKIP_PREPARE_USER_INPUTS_KEY] = True + return helper.compact_generate_response( + AppGenerateService.generate( + app_model=app_model, + user=current_user, + args=workflow_args, + invoke_from=InvokeFrom.DEBUGGER, + streaming=True, + root_node_id=node_id, + ) + ) + except InvokeRateLimitError as ex: + raise InvokeRateLimitHttpError(ex.description) + except PluginInvokeError as e: + return jsonable_encoder({"status": "error", "error": e.to_user_friendly_error()}), 400 + except Exception as e: + logger.exception("Error polling trigger debug event") + raise e + + +@console_ns.route("/apps//workflows/draft/nodes//trigger/run") +class DraftWorkflowTriggerNodeApi(Resource): + """ + Single node debug - Polling API for trigger events + Path: /apps//workflows/draft/nodes//trigger/run + """ + + @api.doc("poll_draft_workflow_trigger_node") + @api.doc(description="Poll for trigger events and execute single node when event arrives") + @api.doc(params={"app_id": "Application ID", "node_id": "Node ID"}) + @api.response(200, "Trigger event received and node executed successfully") + @api.response(403, "Permission denied") + @api.response(500, "Internal server error") + @setup_required + @login_required + @account_initialization_required + @get_app_model(mode=[AppMode.WORKFLOW]) + @edit_permission_required + def post(self, app_model: App, node_id: str): + """ + Poll for trigger events and execute single node when event arrives + """ + current_user, _ = current_account_with_tenant() + + workflow_service = WorkflowService() + draft_workflow = workflow_service.get_draft_workflow(app_model) + if not draft_workflow: + raise ValueError("Workflow not found") + + node_config = draft_workflow.get_node_config_by_id(node_id=node_id) + if not node_config: + raise ValueError("Node data not found for node %s", node_id) + node_type: NodeType = draft_workflow.get_node_type_from_node_config(node_config) + event: TriggerDebugEvent | None = None + # for schedule trigger, when run single node, just execute directly + if node_type == NodeType.TRIGGER_SCHEDULE: + event = TriggerDebugEvent( + workflow_args={}, + node_id=node_id, + ) + # for other trigger types, poll for the event + else: + try: + poller: TriggerDebugEventPoller = create_event_poller( + draft_workflow=draft_workflow, + tenant_id=app_model.tenant_id, + user_id=current_user.id, + app_id=app_model.id, + node_id=node_id, + ) + event = poller.poll() + except PluginInvokeError as e: + return jsonable_encoder({"status": "error", "error": e.to_user_friendly_error()}), 400 + except Exception as e: + logger.exception("Error polling trigger debug event") + raise e + if not event: + return jsonable_encoder({"status": "waiting", "retry_in": LISTENING_RETRY_IN}) + + raw_files = event.workflow_args.get("files") + files = _parse_file(draft_workflow, raw_files if isinstance(raw_files, list) else None) + try: + node_execution = workflow_service.run_draft_workflow_node( + app_model=app_model, + draft_workflow=draft_workflow, + node_id=node_id, + user_inputs=event.workflow_args.get("inputs") or {}, + account=current_user, + query="", + files=files, + ) + return jsonable_encoder(node_execution) + except Exception as e: + logger.exception("Error running draft workflow trigger node") + return jsonable_encoder( + {"status": "error", "error": "An unexpected error occurred while running the node."} + ), 400 + + +@console_ns.route("/apps//workflows/draft/trigger/run-all") +class DraftWorkflowTriggerRunAllApi(Resource): + """ + Full workflow debug - Polling API for trigger events + Path: /apps//workflows/draft/trigger/run-all + """ + + @api.doc("draft_workflow_trigger_run_all") + @api.doc(description="Full workflow debug when the start node is a trigger") + @api.doc(params={"app_id": "Application ID"}) + @api.expect( + api.model( + "DraftWorkflowTriggerRunAllRequest", + { + "node_ids": fields.List(fields.String, required=True, description="Node IDs"), + }, + ) + ) + @api.response(200, "Workflow executed successfully") + @api.response(403, "Permission denied") + @api.response(500, "Internal server error") + @setup_required + @login_required + @account_initialization_required + @get_app_model(mode=[AppMode.WORKFLOW]) + @edit_permission_required + def post(self, app_model: App): + """ + Full workflow debug when the start node is a trigger + """ + current_user, _ = current_account_with_tenant() + + parser = reqparse.RequestParser() + parser.add_argument("node_ids", type=list, required=True, location="json", nullable=False) + args = parser.parse_args() + node_ids = args["node_ids"] + workflow_service = WorkflowService() + draft_workflow = workflow_service.get_draft_workflow(app_model) + if not draft_workflow: + raise ValueError("Workflow not found") + + try: + trigger_debug_event: TriggerDebugEvent | None = select_trigger_debug_events( + draft_workflow=draft_workflow, + app_model=app_model, + user_id=current_user.id, + node_ids=node_ids, + ) + except PluginInvokeError as e: + return jsonable_encoder({"status": "error", "error": e.to_user_friendly_error()}), 400 + except Exception as e: + logger.exception("Error polling trigger debug event") + raise e + if trigger_debug_event is None: + return jsonable_encoder({"status": "waiting", "retry_in": LISTENING_RETRY_IN}) + + try: + workflow_args = dict(trigger_debug_event.workflow_args) + workflow_args[SKIP_PREPARE_USER_INPUTS_KEY] = True + response = AppGenerateService.generate( + app_model=app_model, + user=current_user, + args=workflow_args, + invoke_from=InvokeFrom.DEBUGGER, + streaming=True, + root_node_id=trigger_debug_event.node_id, + ) + return helper.compact_generate_response(response) + except InvokeRateLimitError as ex: + raise InvokeRateLimitHttpError(ex.description) + except Exception: + logger.exception("Error running draft workflow trigger run-all") + return jsonable_encoder( + { + "status": "error", + } + ), 400 diff --git a/api/controllers/console/app/workflow_app_log.py b/api/controllers/console/app/workflow_app_log.py index cbf4e84ff0..d7ecc7c91b 100644 --- a/api/controllers/console/app/workflow_app_log.py +++ b/api/controllers/console/app/workflow_app_log.py @@ -28,6 +28,7 @@ class WorkflowAppLogApi(Resource): "created_at__after": "Filter logs created after this timestamp", "created_by_end_user_session_id": "Filter by end user session ID", "created_by_account": "Filter by account", + "detail": "Whether to return detailed logs", "page": "Page number (1-99999)", "limit": "Number of items per page (1-100)", } @@ -68,6 +69,7 @@ class WorkflowAppLogApi(Resource): required=False, default=None, ) + .add_argument("detail", type=bool, location="args", required=False, default=False) .add_argument("page", type=int_range(1, 99999), default=1, location="args") .add_argument("limit", type=int_range(1, 100), default=20, location="args") ) @@ -92,6 +94,7 @@ class WorkflowAppLogApi(Resource): created_at_after=args.created_at__after, page=args.page, limit=args.limit, + detail=args.detail, created_by_end_user_session_id=args.created_by_end_user_session_id, created_by_account=args.created_by_account, ) diff --git a/api/controllers/console/app/workflow_run.py b/api/controllers/console/app/workflow_run.py index 311aa81279..23c228efbe 100644 --- a/api/controllers/console/app/workflow_run.py +++ b/api/controllers/console/app/workflow_run.py @@ -30,23 +30,25 @@ def _parse_workflow_run_list_args(): Returns: Parsed arguments containing last_id, limit, status, and triggered_from filters """ - parser = reqparse.RequestParser() - parser.add_argument("last_id", type=uuid_value, location="args") - parser.add_argument("limit", type=int_range(1, 100), required=False, default=20, location="args") - parser.add_argument( - "status", - type=str, - choices=WORKFLOW_RUN_STATUS_CHOICES, - location="args", - required=False, - ) - parser.add_argument( - "triggered_from", - type=str, - choices=["debugging", "app-run"], - location="args", - required=False, - help="Filter by trigger source: debugging or app-run", + parser = ( + reqparse.RequestParser() + .add_argument("last_id", type=uuid_value, location="args") + .add_argument("limit", type=int_range(1, 100), required=False, default=20, location="args") + .add_argument( + "status", + type=str, + choices=WORKFLOW_RUN_STATUS_CHOICES, + location="args", + required=False, + ) + .add_argument( + "triggered_from", + type=str, + choices=["debugging", "app-run"], + location="args", + required=False, + help="Filter by trigger source: debugging or app-run", + ) ) return parser.parse_args() @@ -58,28 +60,30 @@ def _parse_workflow_run_count_args(): Returns: Parsed arguments containing status, time_range, and triggered_from filters """ - parser = reqparse.RequestParser() - parser.add_argument( - "status", - type=str, - choices=WORKFLOW_RUN_STATUS_CHOICES, - location="args", - required=False, - ) - parser.add_argument( - "time_range", - type=time_duration, - location="args", - required=False, - help="Time range filter (e.g., 7d, 4h, 30m, 30s)", - ) - parser.add_argument( - "triggered_from", - type=str, - choices=["debugging", "app-run"], - location="args", - required=False, - help="Filter by trigger source: debugging or app-run", + parser = ( + reqparse.RequestParser() + .add_argument( + "status", + type=str, + choices=WORKFLOW_RUN_STATUS_CHOICES, + location="args", + required=False, + ) + .add_argument( + "time_range", + type=time_duration, + location="args", + required=False, + help="Time range filter (e.g., 7d, 4h, 30m, 30s)", + ) + .add_argument( + "triggered_from", + type=str, + choices=["debugging", "app-run"], + location="args", + required=False, + help="Filter by trigger source: debugging or app-run", + ) ) return parser.parse_args() diff --git a/api/controllers/console/app/workflow_statistic.py b/api/controllers/console/app/workflow_statistic.py index bbea04640a..ef5205c1ee 100644 --- a/api/controllers/console/app/workflow_statistic.py +++ b/api/controllers/console/app/workflow_statistic.py @@ -1,23 +1,26 @@ -from datetime import datetime -from decimal import Decimal - -import pytz -import sqlalchemy as sa -from flask import jsonify +from flask import abort, jsonify from flask_restx import Resource, reqparse +from sqlalchemy.orm import sessionmaker from controllers.console import api, console_ns from controllers.console.app.wraps import get_app_model from controllers.console.wraps import account_initialization_required, setup_required from extensions.ext_database import db +from libs.datetime_utils import parse_time_range from libs.helper import DatetimeString from libs.login import current_account_with_tenant, login_required from models.enums import WorkflowRunTriggeredFrom from models.model import AppMode +from repositories.factory import DifyAPIRepositoryFactory @console_ns.route("/apps//workflow/statistics/daily-conversations") class WorkflowDailyRunsStatistic(Resource): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + @api.doc("get_workflow_daily_runs_statistic") @api.doc(description="Get workflow daily runs statistics") @api.doc(params={"app_id": "Application ID"}) @@ -37,57 +40,32 @@ class WorkflowDailyRunsStatistic(Resource): ) args = parser.parse_args() - sql_query = """SELECT - DATE(DATE_TRUNC('day', created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, - COUNT(id) AS runs -FROM - workflow_runs -WHERE - app_id = :app_id - AND triggered_from = :triggered_from""" - arg_dict = { - "tz": account.timezone, - "app_id": app_model.id, - "triggered_from": WorkflowRunTriggeredFrom.APP_RUN, - } assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) + try: + start_date, end_date = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) - - sql_query += " AND created_at >= :start" - arg_dict["start"] = start_datetime_utc - - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - - sql_query += " AND created_at < :end" - arg_dict["end"] = end_datetime_utc - - sql_query += " GROUP BY date ORDER BY date" - - response_data = [] - - with db.engine.begin() as conn: - rs = conn.execute(sa.text(sql_query), arg_dict) - for i in rs: - response_data.append({"date": str(i.date), "runs": i.runs}) + response_data = self._workflow_run_repo.get_daily_runs_statistics( + tenant_id=app_model.tenant_id, + app_id=app_model.id, + triggered_from=WorkflowRunTriggeredFrom.APP_RUN, + start_date=start_date, + end_date=end_date, + timezone=account.timezone, + ) return jsonify({"data": response_data}) @console_ns.route("/apps//workflow/statistics/daily-terminals") class WorkflowDailyTerminalsStatistic(Resource): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + @api.doc("get_workflow_daily_terminals_statistic") @api.doc(description="Get workflow daily terminals statistics") @api.doc(params={"app_id": "Application ID"}) @@ -107,57 +85,32 @@ class WorkflowDailyTerminalsStatistic(Resource): ) args = parser.parse_args() - sql_query = """SELECT - DATE(DATE_TRUNC('day', created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, - COUNT(DISTINCT workflow_runs.created_by) AS terminal_count -FROM - workflow_runs -WHERE - app_id = :app_id - AND triggered_from = :triggered_from""" - arg_dict = { - "tz": account.timezone, - "app_id": app_model.id, - "triggered_from": WorkflowRunTriggeredFrom.APP_RUN, - } assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) + try: + start_date, end_date = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) - - sql_query += " AND created_at >= :start" - arg_dict["start"] = start_datetime_utc - - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - - sql_query += " AND created_at < :end" - arg_dict["end"] = end_datetime_utc - - sql_query += " GROUP BY date ORDER BY date" - - response_data = [] - - with db.engine.begin() as conn: - rs = conn.execute(sa.text(sql_query), arg_dict) - for i in rs: - response_data.append({"date": str(i.date), "terminal_count": i.terminal_count}) + response_data = self._workflow_run_repo.get_daily_terminals_statistics( + tenant_id=app_model.tenant_id, + app_id=app_model.id, + triggered_from=WorkflowRunTriggeredFrom.APP_RUN, + start_date=start_date, + end_date=end_date, + timezone=account.timezone, + ) return jsonify({"data": response_data}) @console_ns.route("/apps//workflow/statistics/token-costs") class WorkflowDailyTokenCostStatistic(Resource): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + @api.doc("get_workflow_daily_token_cost_statistic") @api.doc(description="Get workflow daily token cost statistics") @api.doc(params={"app_id": "Application ID"}) @@ -177,62 +130,32 @@ class WorkflowDailyTokenCostStatistic(Resource): ) args = parser.parse_args() - sql_query = """SELECT - DATE(DATE_TRUNC('day', created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, - SUM(workflow_runs.total_tokens) AS token_count -FROM - workflow_runs -WHERE - app_id = :app_id - AND triggered_from = :triggered_from""" - arg_dict = { - "tz": account.timezone, - "app_id": app_model.id, - "triggered_from": WorkflowRunTriggeredFrom.APP_RUN, - } assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) + try: + start_date, end_date = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) - - sql_query += " AND created_at >= :start" - arg_dict["start"] = start_datetime_utc - - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - - sql_query += " AND created_at < :end" - arg_dict["end"] = end_datetime_utc - - sql_query += " GROUP BY date ORDER BY date" - - response_data = [] - - with db.engine.begin() as conn: - rs = conn.execute(sa.text(sql_query), arg_dict) - for i in rs: - response_data.append( - { - "date": str(i.date), - "token_count": i.token_count, - } - ) + response_data = self._workflow_run_repo.get_daily_token_cost_statistics( + tenant_id=app_model.tenant_id, + app_id=app_model.id, + triggered_from=WorkflowRunTriggeredFrom.APP_RUN, + start_date=start_date, + end_date=end_date, + timezone=account.timezone, + ) return jsonify({"data": response_data}) @console_ns.route("/apps//workflow/statistics/average-app-interactions") class WorkflowAverageAppInteractionStatistic(Resource): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + @api.doc("get_workflow_average_app_interaction_statistic") @api.doc(description="Get workflow average app interaction statistics") @api.doc(params={"app_id": "Application ID"}) @@ -252,67 +175,20 @@ class WorkflowAverageAppInteractionStatistic(Resource): ) args = parser.parse_args() - sql_query = """SELECT - AVG(sub.interactions) AS interactions, - sub.date -FROM - ( - SELECT - DATE(DATE_TRUNC('day', c.created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, - c.created_by, - COUNT(c.id) AS interactions - FROM - workflow_runs c - WHERE - c.app_id = :app_id - AND c.triggered_from = :triggered_from - {{start}} - {{end}} - GROUP BY - date, c.created_by - ) sub -GROUP BY - sub.date""" - arg_dict = { - "tz": account.timezone, - "app_id": app_model.id, - "triggered_from": WorkflowRunTriggeredFrom.APP_RUN, - } assert account.timezone is not None - timezone = pytz.timezone(account.timezone) - utc_timezone = pytz.utc - if args["start"]: - start_datetime = datetime.strptime(args["start"], "%Y-%m-%d %H:%M") - start_datetime = start_datetime.replace(second=0) + try: + start_date, end_date = parse_time_range(args["start"], args["end"], account.timezone) + except ValueError as e: + abort(400, description=str(e)) - start_datetime_timezone = timezone.localize(start_datetime) - start_datetime_utc = start_datetime_timezone.astimezone(utc_timezone) - - sql_query = sql_query.replace("{{start}}", " AND c.created_at >= :start") - arg_dict["start"] = start_datetime_utc - else: - sql_query = sql_query.replace("{{start}}", "") - - if args["end"]: - end_datetime = datetime.strptime(args["end"], "%Y-%m-%d %H:%M") - end_datetime = end_datetime.replace(second=0) - - end_datetime_timezone = timezone.localize(end_datetime) - end_datetime_utc = end_datetime_timezone.astimezone(utc_timezone) - - sql_query = sql_query.replace("{{end}}", " AND c.created_at < :end") - arg_dict["end"] = end_datetime_utc - else: - sql_query = sql_query.replace("{{end}}", "") - - response_data = [] - - with db.engine.begin() as conn: - rs = conn.execute(sa.text(sql_query), arg_dict) - for i in rs: - response_data.append( - {"date": str(i.date), "interactions": float(i.interactions.quantize(Decimal("0.01")))} - ) + response_data = self._workflow_run_repo.get_average_app_interaction_statistics( + tenant_id=app_model.tenant_id, + app_id=app_model.id, + triggered_from=WorkflowRunTriggeredFrom.APP_RUN, + start_date=start_date, + end_date=end_date, + timezone=account.timezone, + ) return jsonify({"data": response_data}) diff --git a/api/controllers/console/app/workflow_trigger.py b/api/controllers/console/app/workflow_trigger.py new file mode 100644 index 0000000000..fd64261525 --- /dev/null +++ b/api/controllers/console/app/workflow_trigger.py @@ -0,0 +1,145 @@ +import logging + +from flask_restx import Resource, marshal_with, reqparse +from sqlalchemy import select +from sqlalchemy.orm import Session +from werkzeug.exceptions import Forbidden, NotFound + +from configs import dify_config +from controllers.console import api +from controllers.console.app.wraps import get_app_model +from controllers.console.wraps import account_initialization_required, setup_required +from extensions.ext_database import db +from fields.workflow_trigger_fields import trigger_fields, triggers_list_fields, webhook_trigger_fields +from libs.login import current_user, login_required +from models.enums import AppTriggerStatus +from models.model import Account, App, AppMode +from models.trigger import AppTrigger, WorkflowWebhookTrigger + +logger = logging.getLogger(__name__) + + +class WebhookTriggerApi(Resource): + """Webhook Trigger API""" + + @setup_required + @login_required + @account_initialization_required + @get_app_model(mode=AppMode.WORKFLOW) + @marshal_with(webhook_trigger_fields) + def get(self, app_model: App): + """Get webhook trigger for a node""" + parser = reqparse.RequestParser() + parser.add_argument("node_id", type=str, required=True, help="Node ID is required") + args = parser.parse_args() + + node_id = str(args["node_id"]) + + with Session(db.engine) as session: + # Get webhook trigger for this app and node + webhook_trigger = ( + session.query(WorkflowWebhookTrigger) + .where( + WorkflowWebhookTrigger.app_id == app_model.id, + WorkflowWebhookTrigger.node_id == node_id, + ) + .first() + ) + + if not webhook_trigger: + raise NotFound("Webhook trigger not found for this node") + + return webhook_trigger + + +class AppTriggersApi(Resource): + """App Triggers list API""" + + @setup_required + @login_required + @account_initialization_required + @get_app_model(mode=AppMode.WORKFLOW) + @marshal_with(triggers_list_fields) + def get(self, app_model: App): + """Get app triggers list""" + assert isinstance(current_user, Account) + assert current_user.current_tenant_id is not None + + with Session(db.engine) as session: + # Get all triggers for this app using select API + triggers = ( + session.execute( + select(AppTrigger) + .where( + AppTrigger.tenant_id == current_user.current_tenant_id, + AppTrigger.app_id == app_model.id, + ) + .order_by(AppTrigger.created_at.desc(), AppTrigger.id.desc()) + ) + .scalars() + .all() + ) + + # Add computed icon field for each trigger + url_prefix = dify_config.CONSOLE_API_URL + "/console/api/workspaces/current/tool-provider/builtin/" + for trigger in triggers: + if trigger.trigger_type == "trigger-plugin": + trigger.icon = url_prefix + trigger.provider_name + "/icon" # type: ignore + else: + trigger.icon = "" # type: ignore + + return {"data": triggers} + + +class AppTriggerEnableApi(Resource): + @setup_required + @login_required + @account_initialization_required + @get_app_model(mode=AppMode.WORKFLOW) + @marshal_with(trigger_fields) + def post(self, app_model: App): + """Update app trigger (enable/disable)""" + parser = reqparse.RequestParser() + parser.add_argument("trigger_id", type=str, required=True, nullable=False, location="json") + parser.add_argument("enable_trigger", type=bool, required=True, nullable=False, location="json") + args = parser.parse_args() + + assert isinstance(current_user, Account) + assert current_user.current_tenant_id is not None + if not current_user.has_edit_permission: + raise Forbidden() + + trigger_id = args["trigger_id"] + + with Session(db.engine) as session: + # Find the trigger using select + trigger = session.execute( + select(AppTrigger).where( + AppTrigger.id == trigger_id, + AppTrigger.tenant_id == current_user.current_tenant_id, + AppTrigger.app_id == app_model.id, + ) + ).scalar_one_or_none() + + if not trigger: + raise NotFound("Trigger not found") + + # Update status based on enable_trigger boolean + trigger.status = AppTriggerStatus.ENABLED if args["enable_trigger"] else AppTriggerStatus.DISABLED + + session.commit() + session.refresh(trigger) + + # Add computed icon field + url_prefix = dify_config.CONSOLE_API_URL + "/console/api/workspaces/current/tool-provider/builtin/" + if trigger.trigger_type == "trigger-plugin": + trigger.icon = url_prefix + trigger.provider_name + "/icon" # type: ignore + else: + trigger.icon = "" # type: ignore + + return trigger + + +api.add_resource(WebhookTriggerApi, "/apps//workflows/triggers/webhook") +api.add_resource(AppTriggersApi, "/apps//triggers") +api.add_resource(AppTriggerEnableApi, "/apps//trigger-enable") diff --git a/api/controllers/console/auth/login.py b/api/controllers/console/auth/login.py index c0a565b5da..77ecd5a5e4 100644 --- a/api/controllers/console/auth/login.py +++ b/api/controllers/console/auth/login.py @@ -29,6 +29,7 @@ from libs.token import ( clear_access_token_from_cookie, clear_csrf_token_from_cookie, clear_refresh_token_from_cookie, + extract_refresh_token, set_access_token_to_cookie, set_csrf_token_to_cookie, set_refresh_token_to_cookie, @@ -270,7 +271,7 @@ class EmailCodeLoginApi(Resource): class RefreshTokenApi(Resource): def post(self): # Get refresh token from cookie instead of request body - refresh_token = request.cookies.get("refresh_token") + refresh_token = extract_refresh_token(request) if not refresh_token: return {"result": "fail", "message": "No refresh token provided"}, 401 diff --git a/api/controllers/console/billing/billing.py b/api/controllers/console/billing/billing.py index 705f5970dd..436d29df83 100644 --- a/api/controllers/console/billing/billing.py +++ b/api/controllers/console/billing/billing.py @@ -2,6 +2,7 @@ from flask_restx import Resource, reqparse from controllers.console import console_ns from controllers.console.wraps import account_initialization_required, only_edition_cloud, setup_required +from enums.cloud_plan import CloudPlan from libs.login import current_account_with_tenant, login_required from services.billing_service import BillingService @@ -16,7 +17,13 @@ class Subscription(Resource): current_user, current_tenant_id = current_account_with_tenant() parser = ( reqparse.RequestParser() - .add_argument("plan", type=str, required=True, location="args", choices=["professional", "team"]) + .add_argument( + "plan", + type=str, + required=True, + location="args", + choices=[CloudPlan.PROFESSIONAL, CloudPlan.TEAM], + ) .add_argument("interval", type=str, required=True, location="args", choices=["month", "year"]) ) args = parser.parse_args() diff --git a/api/controllers/console/datasets/datasets_document.py b/api/controllers/console/datasets/datasets_document.py index 85fd0535c7..f398989d27 100644 --- a/api/controllers/console/datasets/datasets_document.py +++ b/api/controllers/console/datasets/datasets_document.py @@ -746,7 +746,7 @@ class DocumentApi(DocumentResource): "name": document.name, "created_from": document.created_from, "created_by": document.created_by, - "created_at": document.created_at.timestamp(), + "created_at": int(document.created_at.timestamp()), "tokens": document.tokens, "indexing_status": document.indexing_status, "completed_at": int(document.completed_at.timestamp()) if document.completed_at else None, @@ -779,7 +779,7 @@ class DocumentApi(DocumentResource): "name": document.name, "created_from": document.created_from, "created_by": document.created_by, - "created_at": document.created_at.timestamp(), + "created_at": int(document.created_at.timestamp()), "tokens": document.tokens, "indexing_status": document.indexing_status, "completed_at": int(document.completed_at.timestamp()) if document.completed_at else None, diff --git a/api/controllers/console/datasets/rag_pipeline/datasource_auth.py b/api/controllers/console/datasets/rag_pipeline/datasource_auth.py index 2111ee2ecf..f83ee69beb 100644 --- a/api/controllers/console/datasets/rag_pipeline/datasource_auth.py +++ b/api/controllers/console/datasets/rag_pipeline/datasource_auth.py @@ -3,7 +3,7 @@ from flask_restx import Resource, reqparse from werkzeug.exceptions import Forbidden, NotFound from configs import dify_config -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.wraps import account_initialization_required, edit_permission_required, setup_required from core.model_runtime.errors.validate import CredentialsValidateFailedError from core.model_runtime.utils.encoders import jsonable_encoder @@ -121,8 +121,16 @@ class DatasourceOAuthCallback(Resource): return redirect(f"{dify_config.CONSOLE_WEB_URL}/oauth-callback") +parser_datasource = ( + reqparse.RequestParser() + .add_argument("name", type=StrLen(max_length=100), required=False, nullable=True, location="json", default=None) + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") +) + + @console_ns.route("/auth/plugin/datasource/") class DatasourceAuth(Resource): + @api.expect(parser_datasource) @setup_required @login_required @account_initialization_required @@ -130,14 +138,7 @@ class DatasourceAuth(Resource): def post(self, provider_id: str): _, current_tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument( - "name", type=StrLen(max_length=100), required=False, nullable=True, location="json", default=None - ) - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_datasource.parse_args() datasource_provider_id = DatasourceProviderID(provider_id) datasource_provider_service = DatasourceProviderService() @@ -168,8 +169,14 @@ class DatasourceAuth(Resource): return {"result": datasources}, 200 +parser_datasource_delete = reqparse.RequestParser().add_argument( + "credential_id", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/auth/plugin/datasource//delete") class DatasourceAuthDeleteApi(Resource): + @api.expect(parser_datasource_delete) @setup_required @login_required @account_initialization_required @@ -181,10 +188,7 @@ class DatasourceAuthDeleteApi(Resource): plugin_id = datasource_provider_id.plugin_id provider_name = datasource_provider_id.provider_name - parser = reqparse.RequestParser().add_argument( - "credential_id", type=str, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + args = parser_datasource_delete.parse_args() datasource_provider_service = DatasourceProviderService() datasource_provider_service.remove_datasource_credentials( tenant_id=current_tenant_id, @@ -195,8 +199,17 @@ class DatasourceAuthDeleteApi(Resource): return {"result": "success"}, 200 +parser_datasource_update = ( + reqparse.RequestParser() + .add_argument("credentials", type=dict, required=False, nullable=True, location="json") + .add_argument("name", type=StrLen(max_length=100), required=False, nullable=True, location="json") + .add_argument("credential_id", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/auth/plugin/datasource//update") class DatasourceAuthUpdateApi(Resource): + @api.expect(parser_datasource_update) @setup_required @login_required @account_initialization_required @@ -205,13 +218,7 @@ class DatasourceAuthUpdateApi(Resource): _, current_tenant_id = current_account_with_tenant() datasource_provider_id = DatasourceProviderID(provider_id) - parser = ( - reqparse.RequestParser() - .add_argument("credentials", type=dict, required=False, nullable=True, location="json") - .add_argument("name", type=StrLen(max_length=100), required=False, nullable=True, location="json") - .add_argument("credential_id", type=str, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_datasource_update.parse_args() datasource_provider_service = DatasourceProviderService() datasource_provider_service.update_datasource_credentials( @@ -251,8 +258,16 @@ class DatasourceHardCodeAuthListApi(Resource): return {"result": jsonable_encoder(datasources)}, 200 +parser_datasource_custom = ( + reqparse.RequestParser() + .add_argument("client_params", type=dict, required=False, nullable=True, location="json") + .add_argument("enable_oauth_custom_client", type=bool, required=False, nullable=True, location="json") +) + + @console_ns.route("/auth/plugin/datasource//custom-client") class DatasourceAuthOauthCustomClient(Resource): + @api.expect(parser_datasource_custom) @setup_required @login_required @account_initialization_required @@ -260,12 +275,7 @@ class DatasourceAuthOauthCustomClient(Resource): def post(self, provider_id: str): _, current_tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("client_params", type=dict, required=False, nullable=True, location="json") - .add_argument("enable_oauth_custom_client", type=bool, required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_datasource_custom.parse_args() datasource_provider_id = DatasourceProviderID(provider_id) datasource_provider_service = DatasourceProviderService() datasource_provider_service.setup_oauth_custom_client_params( @@ -291,8 +301,12 @@ class DatasourceAuthOauthCustomClient(Resource): return {"result": "success"}, 200 +parser_default = reqparse.RequestParser().add_argument("id", type=str, required=True, nullable=False, location="json") + + @console_ns.route("/auth/plugin/datasource//default") class DatasourceAuthDefaultApi(Resource): + @api.expect(parser_default) @setup_required @login_required @account_initialization_required @@ -300,8 +314,7 @@ class DatasourceAuthDefaultApi(Resource): def post(self, provider_id: str): _, current_tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("id", type=str, required=True, nullable=False, location="json") - args = parser.parse_args() + args = parser_default.parse_args() datasource_provider_id = DatasourceProviderID(provider_id) datasource_provider_service = DatasourceProviderService() datasource_provider_service.set_default_datasource_provider( @@ -312,8 +325,16 @@ class DatasourceAuthDefaultApi(Resource): return {"result": "success"}, 200 +parser_update_name = ( + reqparse.RequestParser() + .add_argument("name", type=StrLen(max_length=100), required=True, nullable=False, location="json") + .add_argument("credential_id", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/auth/plugin/datasource//update-name") class DatasourceUpdateProviderNameApi(Resource): + @api.expect(parser_update_name) @setup_required @login_required @account_initialization_required @@ -321,12 +342,7 @@ class DatasourceUpdateProviderNameApi(Resource): def post(self, provider_id: str): _, current_tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("name", type=StrLen(max_length=100), required=True, nullable=False, location="json") - .add_argument("credential_id", type=str, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_update_name.parse_args() datasource_provider_id = DatasourceProviderID(provider_id) datasource_provider_service = DatasourceProviderService() datasource_provider_service.update_datasource_provider_name( diff --git a/api/controllers/console/datasets/rag_pipeline/datasource_content_preview.py b/api/controllers/console/datasets/rag_pipeline/datasource_content_preview.py index 856e4a1c70..d413def27f 100644 --- a/api/controllers/console/datasets/rag_pipeline/datasource_content_preview.py +++ b/api/controllers/console/datasets/rag_pipeline/datasource_content_preview.py @@ -4,7 +4,7 @@ from flask_restx import ( # type: ignore ) from werkzeug.exceptions import Forbidden -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.datasets.wraps import get_rag_pipeline from controllers.console.wraps import account_initialization_required, setup_required from libs.login import current_user, login_required @@ -12,9 +12,17 @@ from models import Account from models.dataset import Pipeline from services.rag_pipeline.rag_pipeline import RagPipelineService +parser = ( + reqparse.RequestParser() + .add_argument("inputs", type=dict, required=True, nullable=False, location="json") + .add_argument("datasource_type", type=str, required=True, location="json") + .add_argument("credential_id", type=str, required=False, location="json") +) + @console_ns.route("/rag/pipelines//workflows/published/datasource/nodes//preview") class DataSourceContentPreviewApi(Resource): + @api.expect(parser) @setup_required @login_required @account_initialization_required @@ -26,12 +34,6 @@ class DataSourceContentPreviewApi(Resource): if not isinstance(current_user, Account): raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("inputs", type=dict, required=True, nullable=False, location="json") - .add_argument("datasource_type", type=str, required=True, location="json") - .add_argument("credential_id", type=str, required=False, location="json") - ) args = parser.parse_args() inputs = args.get("inputs") diff --git a/api/controllers/console/datasets/rag_pipeline/rag_pipeline_workflow.py b/api/controllers/console/datasets/rag_pipeline/rag_pipeline_workflow.py index 5fe8572dfa..1e77a988bd 100644 --- a/api/controllers/console/datasets/rag_pipeline/rag_pipeline_workflow.py +++ b/api/controllers/console/datasets/rag_pipeline/rag_pipeline_workflow.py @@ -9,7 +9,7 @@ from sqlalchemy.orm import Session from werkzeug.exceptions import Forbidden, InternalServerError, NotFound import services -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.app.error import ( ConversationCompletedError, DraftWorkflowNotExist, @@ -148,8 +148,12 @@ class DraftRagPipelineApi(Resource): } +parser_run = reqparse.RequestParser().add_argument("inputs", type=dict, location="json") + + @console_ns.route("/rag/pipelines//workflows/draft/iteration/nodes//run") class RagPipelineDraftRunIterationNodeApi(Resource): + @api.expect(parser_run) @setup_required @login_required @account_initialization_required @@ -162,8 +166,7 @@ class RagPipelineDraftRunIterationNodeApi(Resource): # The role of the current user in the ta table must be admin, owner, or editor current_user, _ = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("inputs", type=dict, location="json") - args = parser.parse_args() + args = parser_run.parse_args() try: response = PipelineGenerateService.generate_single_iteration( @@ -184,6 +187,7 @@ class RagPipelineDraftRunIterationNodeApi(Resource): @console_ns.route("/rag/pipelines//workflows/draft/loop/nodes//run") class RagPipelineDraftRunLoopNodeApi(Resource): + @api.expect(parser_run) @setup_required @login_required @account_initialization_required @@ -197,8 +201,7 @@ class RagPipelineDraftRunLoopNodeApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = reqparse.RequestParser().add_argument("inputs", type=dict, location="json") - args = parser.parse_args() + args = parser_run.parse_args() try: response = PipelineGenerateService.generate_single_loop( @@ -217,8 +220,18 @@ class RagPipelineDraftRunLoopNodeApi(Resource): raise InternalServerError() +parser_draft_run = ( + reqparse.RequestParser() + .add_argument("inputs", type=dict, required=True, nullable=False, location="json") + .add_argument("datasource_type", type=str, required=True, location="json") + .add_argument("datasource_info_list", type=list, required=True, location="json") + .add_argument("start_node_id", type=str, required=True, location="json") +) + + @console_ns.route("/rag/pipelines//workflows/draft/run") class DraftRagPipelineRunApi(Resource): + @api.expect(parser_draft_run) @setup_required @login_required @account_initialization_required @@ -232,14 +245,7 @@ class DraftRagPipelineRunApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("inputs", type=dict, required=True, nullable=False, location="json") - .add_argument("datasource_type", type=str, required=True, location="json") - .add_argument("datasource_info_list", type=list, required=True, location="json") - .add_argument("start_node_id", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_draft_run.parse_args() try: response = PipelineGenerateService.generate( @@ -255,8 +261,21 @@ class DraftRagPipelineRunApi(Resource): raise InvokeRateLimitHttpError(ex.description) +parser_published_run = ( + reqparse.RequestParser() + .add_argument("inputs", type=dict, required=True, nullable=False, location="json") + .add_argument("datasource_type", type=str, required=True, location="json") + .add_argument("datasource_info_list", type=list, required=True, location="json") + .add_argument("start_node_id", type=str, required=True, location="json") + .add_argument("is_preview", type=bool, required=True, location="json", default=False) + .add_argument("response_mode", type=str, required=True, location="json", default="streaming") + .add_argument("original_document_id", type=str, required=False, location="json") +) + + @console_ns.route("/rag/pipelines//workflows/published/run") class PublishedRagPipelineRunApi(Resource): + @api.expect(parser_published_run) @setup_required @login_required @account_initialization_required @@ -270,17 +289,7 @@ class PublishedRagPipelineRunApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("inputs", type=dict, required=True, nullable=False, location="json") - .add_argument("datasource_type", type=str, required=True, location="json") - .add_argument("datasource_info_list", type=list, required=True, location="json") - .add_argument("start_node_id", type=str, required=True, location="json") - .add_argument("is_preview", type=bool, required=True, location="json", default=False) - .add_argument("response_mode", type=str, required=True, location="json", default="streaming") - .add_argument("original_document_id", type=str, required=False, location="json") - ) - args = parser.parse_args() + args = parser_published_run.parse_args() streaming = args["response_mode"] == "streaming" @@ -381,8 +390,17 @@ class PublishedRagPipelineRunApi(Resource): # # return result # +parser_rag_run = ( + reqparse.RequestParser() + .add_argument("inputs", type=dict, required=True, nullable=False, location="json") + .add_argument("datasource_type", type=str, required=True, location="json") + .add_argument("credential_id", type=str, required=False, location="json") +) + + @console_ns.route("/rag/pipelines//workflows/published/datasource/nodes//run") class RagPipelinePublishedDatasourceNodeRunApi(Resource): + @api.expect(parser_rag_run) @setup_required @login_required @account_initialization_required @@ -396,13 +414,7 @@ class RagPipelinePublishedDatasourceNodeRunApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("inputs", type=dict, required=True, nullable=False, location="json") - .add_argument("datasource_type", type=str, required=True, location="json") - .add_argument("credential_id", type=str, required=False, location="json") - ) - args = parser.parse_args() + args = parser_rag_run.parse_args() inputs = args.get("inputs") if inputs is None: @@ -429,6 +441,7 @@ class RagPipelinePublishedDatasourceNodeRunApi(Resource): @console_ns.route("/rag/pipelines//workflows/draft/datasource/nodes//run") class RagPipelineDraftDatasourceNodeRunApi(Resource): + @api.expect(parser_rag_run) @setup_required @login_required @account_initialization_required @@ -442,13 +455,7 @@ class RagPipelineDraftDatasourceNodeRunApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("inputs", type=dict, required=True, nullable=False, location="json") - .add_argument("datasource_type", type=str, required=True, location="json") - .add_argument("credential_id", type=str, required=False, location="json") - ) - args = parser.parse_args() + args = parser_rag_run.parse_args() inputs = args.get("inputs") if inputs is None: @@ -473,8 +480,14 @@ class RagPipelineDraftDatasourceNodeRunApi(Resource): ) +parser_run_api = reqparse.RequestParser().add_argument( + "inputs", type=dict, required=True, nullable=False, location="json" +) + + @console_ns.route("/rag/pipelines//workflows/draft/nodes//run") class RagPipelineDraftNodeRunApi(Resource): + @api.expect(parser_run_api) @setup_required @login_required @account_initialization_required @@ -489,10 +502,7 @@ class RagPipelineDraftNodeRunApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = reqparse.RequestParser().add_argument( - "inputs", type=dict, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + args = parser_run_api.parse_args() inputs = args.get("inputs") if inputs == None: @@ -607,8 +617,12 @@ class DefaultRagPipelineBlockConfigsApi(Resource): return rag_pipeline_service.get_default_block_configs() +parser_default = reqparse.RequestParser().add_argument("q", type=str, location="args") + + @console_ns.route("/rag/pipelines//workflows/default-workflow-block-configs/") class DefaultRagPipelineBlockConfigApi(Resource): + @api.expect(parser_default) @setup_required @login_required @account_initialization_required @@ -622,8 +636,7 @@ class DefaultRagPipelineBlockConfigApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = reqparse.RequestParser().add_argument("q", type=str, location="args") - args = parser.parse_args() + args = parser_default.parse_args() q = args.get("q") @@ -639,8 +652,18 @@ class DefaultRagPipelineBlockConfigApi(Resource): return rag_pipeline_service.get_default_block_config(node_type=block_type, filters=filters) +parser_wf = ( + reqparse.RequestParser() + .add_argument("page", type=inputs.int_range(1, 99999), required=False, default=1, location="args") + .add_argument("limit", type=inputs.int_range(1, 100), required=False, default=10, location="args") + .add_argument("user_id", type=str, required=False, location="args") + .add_argument("named_only", type=inputs.boolean, required=False, default=False, location="args") +) + + @console_ns.route("/rag/pipelines//workflows") class PublishedAllRagPipelineApi(Resource): + @api.expect(parser_wf) @setup_required @login_required @account_initialization_required @@ -654,16 +677,9 @@ class PublishedAllRagPipelineApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("page", type=inputs.int_range(1, 99999), required=False, default=1, location="args") - .add_argument("limit", type=inputs.int_range(1, 100), required=False, default=20, location="args") - .add_argument("user_id", type=str, required=False, location="args") - .add_argument("named_only", type=inputs.boolean, required=False, default=False, location="args") - ) - args = parser.parse_args() - page = int(args.get("page", 1)) - limit = int(args.get("limit", 10)) + args = parser_wf.parse_args() + page = args["page"] + limit = args["limit"] user_id = args.get("user_id") named_only = args.get("named_only", False) @@ -691,8 +707,16 @@ class PublishedAllRagPipelineApi(Resource): } +parser_wf_id = ( + reqparse.RequestParser() + .add_argument("marked_name", type=str, required=False, location="json") + .add_argument("marked_comment", type=str, required=False, location="json") +) + + @console_ns.route("/rag/pipelines//workflows/") class RagPipelineByIdApi(Resource): + @api.expect(parser_wf_id) @setup_required @login_required @account_initialization_required @@ -707,19 +731,13 @@ class RagPipelineByIdApi(Resource): if not current_user.has_edit_permission: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("marked_name", type=str, required=False, location="json") - .add_argument("marked_comment", type=str, required=False, location="json") - ) - args = parser.parse_args() + args = parser_wf_id.parse_args() # Validate name and comment length if args.marked_name and len(args.marked_name) > 20: raise ValueError("Marked name cannot exceed 20 characters") if args.marked_comment and len(args.marked_comment) > 100: raise ValueError("Marked comment cannot exceed 100 characters") - args = parser.parse_args() # Prepare update data update_data = {} @@ -752,8 +770,12 @@ class RagPipelineByIdApi(Resource): return workflow +parser_parameters = reqparse.RequestParser().add_argument("node_id", type=str, required=True, location="args") + + @console_ns.route("/rag/pipelines//workflows/published/processing/parameters") class PublishedRagPipelineSecondStepApi(Resource): + @api.expect(parser_parameters) @setup_required @login_required @account_initialization_required @@ -763,8 +785,7 @@ class PublishedRagPipelineSecondStepApi(Resource): """ Get second step parameters of rag pipeline """ - parser = reqparse.RequestParser().add_argument("node_id", type=str, required=True, location="args") - args = parser.parse_args() + args = parser_parameters.parse_args() node_id = args.get("node_id") if not node_id: raise ValueError("Node ID is required") @@ -777,6 +798,7 @@ class PublishedRagPipelineSecondStepApi(Resource): @console_ns.route("/rag/pipelines//workflows/published/pre-processing/parameters") class PublishedRagPipelineFirstStepApi(Resource): + @api.expect(parser_parameters) @setup_required @login_required @account_initialization_required @@ -786,8 +808,7 @@ class PublishedRagPipelineFirstStepApi(Resource): """ Get first step parameters of rag pipeline """ - parser = reqparse.RequestParser().add_argument("node_id", type=str, required=True, location="args") - args = parser.parse_args() + args = parser_parameters.parse_args() node_id = args.get("node_id") if not node_id: raise ValueError("Node ID is required") @@ -800,6 +821,7 @@ class PublishedRagPipelineFirstStepApi(Resource): @console_ns.route("/rag/pipelines//workflows/draft/pre-processing/parameters") class DraftRagPipelineFirstStepApi(Resource): + @api.expect(parser_parameters) @setup_required @login_required @account_initialization_required @@ -809,8 +831,7 @@ class DraftRagPipelineFirstStepApi(Resource): """ Get first step parameters of rag pipeline """ - parser = reqparse.RequestParser().add_argument("node_id", type=str, required=True, location="args") - args = parser.parse_args() + args = parser_parameters.parse_args() node_id = args.get("node_id") if not node_id: raise ValueError("Node ID is required") @@ -823,6 +844,7 @@ class DraftRagPipelineFirstStepApi(Resource): @console_ns.route("/rag/pipelines//workflows/draft/processing/parameters") class DraftRagPipelineSecondStepApi(Resource): + @api.expect(parser_parameters) @setup_required @login_required @account_initialization_required @@ -832,8 +854,7 @@ class DraftRagPipelineSecondStepApi(Resource): """ Get second step parameters of rag pipeline """ - parser = reqparse.RequestParser().add_argument("node_id", type=str, required=True, location="args") - args = parser.parse_args() + args = parser_parameters.parse_args() node_id = args.get("node_id") if not node_id: raise ValueError("Node ID is required") @@ -845,8 +866,16 @@ class DraftRagPipelineSecondStepApi(Resource): } +parser_wf_run = ( + reqparse.RequestParser() + .add_argument("last_id", type=uuid_value, location="args") + .add_argument("limit", type=int_range(1, 100), required=False, default=20, location="args") +) + + @console_ns.route("/rag/pipelines//workflow-runs") class RagPipelineWorkflowRunListApi(Resource): + @api.expect(parser_wf_run) @setup_required @login_required @account_initialization_required @@ -856,12 +885,7 @@ class RagPipelineWorkflowRunListApi(Resource): """ Get workflow run list """ - parser = ( - reqparse.RequestParser() - .add_argument("last_id", type=uuid_value, location="args") - .add_argument("limit", type=int_range(1, 100), required=False, default=20, location="args") - ) - args = parser.parse_args() + args = parser_wf_run.parse_args() rag_pipeline_service = RagPipelineService() result = rag_pipeline_service.get_rag_pipeline_paginate_workflow_runs(pipeline=pipeline, args=args) @@ -961,8 +985,18 @@ class RagPipelineTransformApi(Resource): return result +parser_var = ( + reqparse.RequestParser() + .add_argument("datasource_type", type=str, required=True, location="json") + .add_argument("datasource_info", type=dict, required=True, location="json") + .add_argument("start_node_id", type=str, required=True, location="json") + .add_argument("start_node_title", type=str, required=True, location="json") +) + + @console_ns.route("/rag/pipelines//workflows/draft/datasource/variables-inspect") class RagPipelineDatasourceVariableApi(Resource): + @api.expect(parser_var) @setup_required @login_required @account_initialization_required @@ -974,14 +1008,7 @@ class RagPipelineDatasourceVariableApi(Resource): Set datasource variables """ current_user, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("datasource_type", type=str, required=True, location="json") - .add_argument("datasource_info", type=dict, required=True, location="json") - .add_argument("start_node_id", type=str, required=True, location="json") - .add_argument("start_node_title", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_var.parse_args() rag_pipeline_service = RagPipelineService() workflow_node_execution = rag_pipeline_service.set_datasource_variables( diff --git a/api/controllers/console/explore/recommended_app.py b/api/controllers/console/explore/recommended_app.py index 751012757a..11c7a1bc18 100644 --- a/api/controllers/console/explore/recommended_app.py +++ b/api/controllers/console/explore/recommended_app.py @@ -1,7 +1,7 @@ from flask_restx import Resource, fields, marshal_with, reqparse from constants.languages import languages -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.wraps import account_initialization_required from libs.helper import AppIconUrlField from libs.login import current_user, login_required @@ -35,15 +35,18 @@ recommended_app_list_fields = { } +parser_apps = reqparse.RequestParser().add_argument("language", type=str, location="args") + + @console_ns.route("/explore/apps") class RecommendedAppListApi(Resource): + @api.expect(parser_apps) @login_required @account_initialization_required @marshal_with(recommended_app_list_fields) def get(self): # language args - parser = reqparse.RequestParser().add_argument("language", type=str, location="args") - args = parser.parse_args() + args = parser_apps.parse_args() language = args.get("language") if language and language in languages: diff --git a/api/controllers/console/explore/workflow.py b/api/controllers/console/explore/workflow.py index 3022d937b9..125f603a5a 100644 --- a/api/controllers/console/explore/workflow.py +++ b/api/controllers/console/explore/workflow.py @@ -22,7 +22,7 @@ from core.errors.error import ( from core.model_runtime.errors.invoke import InvokeError from core.workflow.graph_engine.manager import GraphEngineManager from libs import helper -from libs.login import current_user as current_user_ +from libs.login import current_account_with_tenant from models.model import AppMode, InstalledApp from services.app_generate_service import AppGenerateService from services.errors.llm import InvokeRateLimitError @@ -31,8 +31,6 @@ from .. import console_ns logger = logging.getLogger(__name__) -current_user = current_user_._get_current_object() # type: ignore - @console_ns.route("/installed-apps//workflows/run") class InstalledAppWorkflowRunApi(InstalledAppResource): @@ -40,6 +38,7 @@ class InstalledAppWorkflowRunApi(InstalledAppResource): """ Run workflow """ + current_user, _ = current_account_with_tenant() app_model = installed_app.app if not app_model: raise NotWorkflowAppError() @@ -53,7 +52,6 @@ class InstalledAppWorkflowRunApi(InstalledAppResource): .add_argument("files", type=list, required=False, location="json") ) args = parser.parse_args() - assert current_user is not None try: response = AppGenerateService.generate( app_model=app_model, user=current_user, args=args, invoke_from=InvokeFrom.EXPLORE, streaming=True @@ -89,7 +87,6 @@ class InstalledAppWorkflowTaskStopApi(InstalledAppResource): app_mode = AppMode.value_of(app_model.mode) if app_mode != AppMode.WORKFLOW: raise NotWorkflowAppError() - assert current_user is not None # Stop using both mechanisms for backward compatibility # Legacy stop flag mechanism (without user check) diff --git a/api/controllers/console/extension.py b/api/controllers/console/extension.py index 4e1a8aeb3e..a1d36def0d 100644 --- a/api/controllers/console/extension.py +++ b/api/controllers/console/extension.py @@ -66,13 +66,7 @@ class APIBasedExtensionAPI(Resource): @account_initialization_required @marshal_with(api_based_extension_fields) def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("name", type=str, required=True, location="json") - .add_argument("api_endpoint", type=str, required=True, location="json") - .add_argument("api_key", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = api.payload _, current_tenant_id = current_account_with_tenant() extension_data = APIBasedExtension( @@ -125,13 +119,7 @@ class APIBasedExtensionDetailAPI(Resource): extension_data_from_db = APIBasedExtensionService.get_with_tenant_id(current_tenant_id, api_based_extension_id) - parser = ( - reqparse.RequestParser() - .add_argument("name", type=str, required=True, location="json") - .add_argument("api_endpoint", type=str, required=True, location="json") - .add_argument("api_key", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = api.payload extension_data_from_db.name = args["name"] extension_data_from_db.api_endpoint = args["api_endpoint"] diff --git a/api/controllers/console/files.py b/api/controllers/console/files.py index 1cd193f7ad..fdd7c2f479 100644 --- a/api/controllers/console/files.py +++ b/api/controllers/console/files.py @@ -8,6 +8,7 @@ import services from configs import dify_config from constants import DOCUMENT_EXTENSIONS from controllers.common.errors import ( + BlockedFileExtensionError, FilenameNotExistsError, FileTooLargeError, NoFileUploadedError, @@ -39,6 +40,7 @@ class FileApi(Resource): return { "file_size_limit": dify_config.UPLOAD_FILE_SIZE_LIMIT, "batch_count_limit": dify_config.UPLOAD_FILE_BATCH_LIMIT, + "file_upload_limit": dify_config.BATCH_UPLOAD_LIMIT, "image_file_size_limit": dify_config.UPLOAD_IMAGE_FILE_SIZE_LIMIT, "video_file_size_limit": dify_config.UPLOAD_VIDEO_FILE_SIZE_LIMIT, "audio_file_size_limit": dify_config.UPLOAD_AUDIO_FILE_SIZE_LIMIT, @@ -82,6 +84,8 @@ class FileApi(Resource): raise FileTooLargeError(file_too_large_error.description) except services.errors.file.UnsupportedFileTypeError: raise UnsupportedFileTypeError() + except services.errors.file.BlockedFileExtensionError as blocked_extension_error: + raise BlockedFileExtensionError(blocked_extension_error.description) return upload_file, 201 diff --git a/api/controllers/console/remote_files.py b/api/controllers/console/remote_files.py index 96c86dc0db..47c7ecde9a 100644 --- a/api/controllers/console/remote_files.py +++ b/api/controllers/console/remote_files.py @@ -10,6 +10,7 @@ from controllers.common.errors import ( RemoteFileUploadError, UnsupportedFileTypeError, ) +from controllers.console import api from core.file import helpers as file_helpers from core.helper import ssrf_proxy from extensions.ext_database import db @@ -36,12 +37,15 @@ class RemoteFileInfoApi(Resource): } +parser_upload = reqparse.RequestParser().add_argument("url", type=str, required=True, help="URL is required") + + @console_ns.route("/remote-files/upload") class RemoteFileUploadApi(Resource): + @api.expect(parser_upload) @marshal_with(file_fields_with_signed_url) def post(self): - parser = reqparse.RequestParser().add_argument("url", type=str, required=True, help="URL is required") - args = parser.parse_args() + args = parser_upload.parse_args() url = args["url"] diff --git a/api/controllers/console/setup.py b/api/controllers/console/setup.py index 1200349e2d..22929c851e 100644 --- a/api/controllers/console/setup.py +++ b/api/controllers/console/setup.py @@ -49,6 +49,7 @@ class SetupApi(Resource): "email": fields.String(required=True, description="Admin email address"), "name": fields.String(required=True, description="Admin name (max 30 characters)"), "password": fields.String(required=True, description="Admin password"), + "language": fields.String(required=False, description="Admin language"), }, ) ) diff --git a/api/controllers/console/tag/tags.py b/api/controllers/console/tag/tags.py index 40ae7fb4d0..ca8259238b 100644 --- a/api/controllers/console/tag/tags.py +++ b/api/controllers/console/tag/tags.py @@ -2,7 +2,7 @@ from flask import request from flask_restx import Resource, marshal_with, reqparse from werkzeug.exceptions import Forbidden -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.wraps import account_initialization_required, setup_required from fields.tag_fields import dataset_tag_fields from libs.login import current_account_with_tenant, login_required @@ -16,6 +16,19 @@ def _validate_name(name): return name +parser_tags = ( + reqparse.RequestParser() + .add_argument( + "name", + nullable=False, + required=True, + help="Name must be between 1 to 50 characters.", + type=_validate_name, + ) + .add_argument("type", type=str, location="json", choices=Tag.TAG_TYPE_LIST, nullable=True, help="Invalid tag type.") +) + + @console_ns.route("/tags") class TagListApi(Resource): @setup_required @@ -30,6 +43,7 @@ class TagListApi(Resource): return tags, 200 + @api.expect(parser_tags) @setup_required @login_required @account_initialization_required @@ -39,20 +53,7 @@ class TagListApi(Resource): if not (current_user.has_edit_permission or current_user.is_dataset_editor): raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument( - "name", - nullable=False, - required=True, - help="Name must be between 1 to 50 characters.", - type=_validate_name, - ) - .add_argument( - "type", type=str, location="json", choices=Tag.TAG_TYPE_LIST, nullable=True, help="Invalid tag type." - ) - ) - args = parser.parse_args() + args = parser_tags.parse_args() tag = TagService.save_tags(args) response = {"id": tag.id, "name": tag.name, "type": tag.type, "binding_count": 0} @@ -60,8 +61,14 @@ class TagListApi(Resource): return response, 200 +parser_tag_id = reqparse.RequestParser().add_argument( + "name", nullable=False, required=True, help="Name must be between 1 to 50 characters.", type=_validate_name +) + + @console_ns.route("/tags/") class TagUpdateDeleteApi(Resource): + @api.expect(parser_tag_id) @setup_required @login_required @account_initialization_required @@ -72,10 +79,7 @@ class TagUpdateDeleteApi(Resource): if not (current_user.has_edit_permission or current_user.is_dataset_editor): raise Forbidden() - parser = reqparse.RequestParser().add_argument( - "name", nullable=False, required=True, help="Name must be between 1 to 50 characters.", type=_validate_name - ) - args = parser.parse_args() + args = parser_tag_id.parse_args() tag = TagService.update_tags(args, tag_id) binding_count = TagService.get_tag_binding_count(tag_id) @@ -99,8 +103,17 @@ class TagUpdateDeleteApi(Resource): return 204 +parser_create = ( + reqparse.RequestParser() + .add_argument("tag_ids", type=list, nullable=False, required=True, location="json", help="Tag IDs is required.") + .add_argument("target_id", type=str, nullable=False, required=True, location="json", help="Target ID is required.") + .add_argument("type", type=str, location="json", choices=Tag.TAG_TYPE_LIST, nullable=True, help="Invalid tag type.") +) + + @console_ns.route("/tag-bindings/create") class TagBindingCreateApi(Resource): + @api.expect(parser_create) @setup_required @login_required @account_initialization_required @@ -110,26 +123,23 @@ class TagBindingCreateApi(Resource): if not (current_user.has_edit_permission or current_user.is_dataset_editor): raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument( - "tag_ids", type=list, nullable=False, required=True, location="json", help="Tag IDs is required." - ) - .add_argument( - "target_id", type=str, nullable=False, required=True, location="json", help="Target ID is required." - ) - .add_argument( - "type", type=str, location="json", choices=Tag.TAG_TYPE_LIST, nullable=True, help="Invalid tag type." - ) - ) - args = parser.parse_args() + args = parser_create.parse_args() TagService.save_tag_binding(args) return {"result": "success"}, 200 +parser_remove = ( + reqparse.RequestParser() + .add_argument("tag_id", type=str, nullable=False, required=True, help="Tag ID is required.") + .add_argument("target_id", type=str, nullable=False, required=True, help="Target ID is required.") + .add_argument("type", type=str, location="json", choices=Tag.TAG_TYPE_LIST, nullable=True, help="Invalid tag type.") +) + + @console_ns.route("/tag-bindings/remove") class TagBindingDeleteApi(Resource): + @api.expect(parser_remove) @setup_required @login_required @account_initialization_required @@ -139,15 +149,7 @@ class TagBindingDeleteApi(Resource): if not (current_user.has_edit_permission or current_user.is_dataset_editor): raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("tag_id", type=str, nullable=False, required=True, help="Tag ID is required.") - .add_argument("target_id", type=str, nullable=False, required=True, help="Target ID is required.") - .add_argument( - "type", type=str, location="json", choices=Tag.TAG_TYPE_LIST, nullable=True, help="Invalid tag type." - ) - ) - args = parser.parse_args() + args = parser_remove.parse_args() TagService.delete_tag_binding(args) return {"result": "success"}, 200 diff --git a/api/controllers/console/version.py b/api/controllers/console/version.py index 417486f59e..104a205fc8 100644 --- a/api/controllers/console/version.py +++ b/api/controllers/console/version.py @@ -11,16 +11,16 @@ from . import api, console_ns logger = logging.getLogger(__name__) +parser = reqparse.RequestParser().add_argument( + "current_version", type=str, required=True, location="args", help="Current application version" +) + @console_ns.route("/version") class VersionApi(Resource): @api.doc("check_version_update") @api.doc(description="Check for application version updates") - @api.expect( - api.parser().add_argument( - "current_version", type=str, required=True, location="args", help="Current application version" - ) - ) + @api.expect(parser) @api.response( 200, "Success", @@ -37,7 +37,6 @@ class VersionApi(Resource): ) def get(self): """Check for application version updates""" - parser = reqparse.RequestParser().add_argument("current_version", type=str, required=True, location="args") args = parser.parse_args() check_update_url = dify_config.CHECK_UPDATE_URL diff --git a/api/controllers/console/workspace/account.py b/api/controllers/console/workspace/account.py index 499a52370f..0833b39f41 100644 --- a/api/controllers/console/workspace/account.py +++ b/api/controllers/console/workspace/account.py @@ -8,7 +8,7 @@ from sqlalchemy.orm import Session from configs import dify_config from constants.languages import supported_language -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.auth.error import ( EmailAlreadyInUseError, EmailChangeLimitError, @@ -43,8 +43,19 @@ from services.billing_service import BillingService from services.errors.account import CurrentPasswordIncorrectError as ServiceCurrentPasswordIncorrectError +def _init_parser(): + parser = reqparse.RequestParser() + if dify_config.EDITION == "CLOUD": + parser.add_argument("invitation_code", type=str, location="json") + parser.add_argument("interface_language", type=supported_language, required=True, location="json").add_argument( + "timezone", type=timezone, required=True, location="json" + ) + return parser + + @console_ns.route("/account/init") class AccountInitApi(Resource): + @api.expect(_init_parser()) @setup_required @login_required def post(self): @@ -53,14 +64,7 @@ class AccountInitApi(Resource): if account.status == "active": raise AccountAlreadyInitedError() - parser = reqparse.RequestParser() - - if dify_config.EDITION == "CLOUD": - parser.add_argument("invitation_code", type=str, location="json") - parser.add_argument("interface_language", type=supported_language, required=True, location="json").add_argument( - "timezone", type=timezone, required=True, location="json" - ) - args = parser.parse_args() + args = _init_parser().parse_args() if dify_config.EDITION == "CLOUD": if not args["invitation_code"]: @@ -106,16 +110,19 @@ class AccountProfileApi(Resource): return current_user +parser_name = reqparse.RequestParser().add_argument("name", type=str, required=True, location="json") + + @console_ns.route("/account/name") class AccountNameApi(Resource): + @api.expect(parser_name) @setup_required @login_required @account_initialization_required @marshal_with(account_fields) def post(self): current_user, _ = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("name", type=str, required=True, location="json") - args = parser.parse_args() + args = parser_name.parse_args() # Validate account name length if len(args["name"]) < 3 or len(args["name"]) > 30: @@ -126,68 +133,80 @@ class AccountNameApi(Resource): return updated_account +parser_avatar = reqparse.RequestParser().add_argument("avatar", type=str, required=True, location="json") + + @console_ns.route("/account/avatar") class AccountAvatarApi(Resource): + @api.expect(parser_avatar) @setup_required @login_required @account_initialization_required @marshal_with(account_fields) def post(self): current_user, _ = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("avatar", type=str, required=True, location="json") - args = parser.parse_args() + args = parser_avatar.parse_args() updated_account = AccountService.update_account(current_user, avatar=args["avatar"]) return updated_account +parser_interface = reqparse.RequestParser().add_argument( + "interface_language", type=supported_language, required=True, location="json" +) + + @console_ns.route("/account/interface-language") class AccountInterfaceLanguageApi(Resource): + @api.expect(parser_interface) @setup_required @login_required @account_initialization_required @marshal_with(account_fields) def post(self): current_user, _ = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument( - "interface_language", type=supported_language, required=True, location="json" - ) - args = parser.parse_args() + args = parser_interface.parse_args() updated_account = AccountService.update_account(current_user, interface_language=args["interface_language"]) return updated_account +parser_theme = reqparse.RequestParser().add_argument( + "interface_theme", type=str, choices=["light", "dark"], required=True, location="json" +) + + @console_ns.route("/account/interface-theme") class AccountInterfaceThemeApi(Resource): + @api.expect(parser_theme) @setup_required @login_required @account_initialization_required @marshal_with(account_fields) def post(self): current_user, _ = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument( - "interface_theme", type=str, choices=["light", "dark"], required=True, location="json" - ) - args = parser.parse_args() + args = parser_theme.parse_args() updated_account = AccountService.update_account(current_user, interface_theme=args["interface_theme"]) return updated_account +parser_timezone = reqparse.RequestParser().add_argument("timezone", type=str, required=True, location="json") + + @console_ns.route("/account/timezone") class AccountTimezoneApi(Resource): + @api.expect(parser_timezone) @setup_required @login_required @account_initialization_required @marshal_with(account_fields) def post(self): current_user, _ = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("timezone", type=str, required=True, location="json") - args = parser.parse_args() + args = parser_timezone.parse_args() # Validate timezone string, e.g. America/New_York, Asia/Shanghai if args["timezone"] not in pytz.all_timezones: @@ -198,21 +217,24 @@ class AccountTimezoneApi(Resource): return updated_account +parser_pw = ( + reqparse.RequestParser() + .add_argument("password", type=str, required=False, location="json") + .add_argument("new_password", type=str, required=True, location="json") + .add_argument("repeat_new_password", type=str, required=True, location="json") +) + + @console_ns.route("/account/password") class AccountPasswordApi(Resource): + @api.expect(parser_pw) @setup_required @login_required @account_initialization_required @marshal_with(account_fields) def post(self): current_user, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("password", type=str, required=False, location="json") - .add_argument("new_password", type=str, required=True, location="json") - .add_argument("repeat_new_password", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_pw.parse_args() if args["new_password"] != args["repeat_new_password"]: raise RepeatPasswordNotMatchError() @@ -294,20 +316,23 @@ class AccountDeleteVerifyApi(Resource): return {"result": "success", "data": token} +parser_delete = ( + reqparse.RequestParser() + .add_argument("token", type=str, required=True, location="json") + .add_argument("code", type=str, required=True, location="json") +) + + @console_ns.route("/account/delete") class AccountDeleteApi(Resource): + @api.expect(parser_delete) @setup_required @login_required @account_initialization_required def post(self): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("token", type=str, required=True, location="json") - .add_argument("code", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_delete.parse_args() if not AccountService.verify_account_deletion_code(args["token"], args["code"]): raise InvalidAccountDeletionCodeError() @@ -317,16 +342,19 @@ class AccountDeleteApi(Resource): return {"result": "success"} +parser_feedback = ( + reqparse.RequestParser() + .add_argument("email", type=str, required=True, location="json") + .add_argument("feedback", type=str, required=True, location="json") +) + + @console_ns.route("/account/delete/feedback") class AccountDeleteUpdateFeedbackApi(Resource): + @api.expect(parser_feedback) @setup_required def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("email", type=str, required=True, location="json") - .add_argument("feedback", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_feedback.parse_args() BillingService.update_account_deletion_feedback(args["email"], args["feedback"]) @@ -351,6 +379,14 @@ class EducationVerifyApi(Resource): return BillingService.EducationIdentity.verify(account.id, account.email) +parser_edu = ( + reqparse.RequestParser() + .add_argument("token", type=str, required=True, location="json") + .add_argument("institution", type=str, required=True, location="json") + .add_argument("role", type=str, required=True, location="json") +) + + @console_ns.route("/account/education") class EducationApi(Resource): status_fields = { @@ -360,6 +396,7 @@ class EducationApi(Resource): "allow_refresh": fields.Boolean, } + @api.expect(parser_edu) @setup_required @login_required @account_initialization_required @@ -368,13 +405,7 @@ class EducationApi(Resource): def post(self): account, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("token", type=str, required=True, location="json") - .add_argument("institution", type=str, required=True, location="json") - .add_argument("role", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_edu.parse_args() return BillingService.EducationIdentity.activate(account, args["token"], args["institution"], args["role"]) @@ -394,6 +425,14 @@ class EducationApi(Resource): return res +parser_autocomplete = ( + reqparse.RequestParser() + .add_argument("keywords", type=str, required=True, location="args") + .add_argument("page", type=int, required=False, location="args", default=0) + .add_argument("limit", type=int, required=False, location="args", default=20) +) + + @console_ns.route("/account/education/autocomplete") class EducationAutoCompleteApi(Resource): data_fields = { @@ -402,6 +441,7 @@ class EducationAutoCompleteApi(Resource): "has_next": fields.Boolean, } + @api.expect(parser_autocomplete) @setup_required @login_required @account_initialization_required @@ -409,33 +449,30 @@ class EducationAutoCompleteApi(Resource): @cloud_edition_billing_enabled @marshal_with(data_fields) def get(self): - parser = ( - reqparse.RequestParser() - .add_argument("keywords", type=str, required=True, location="args") - .add_argument("page", type=int, required=False, location="args", default=0) - .add_argument("limit", type=int, required=False, location="args", default=20) - ) - args = parser.parse_args() + args = parser_autocomplete.parse_args() return BillingService.EducationIdentity.autocomplete(args["keywords"], args["page"], args["limit"]) +parser_change_email = ( + reqparse.RequestParser() + .add_argument("email", type=email, required=True, location="json") + .add_argument("language", type=str, required=False, location="json") + .add_argument("phase", type=str, required=False, location="json") + .add_argument("token", type=str, required=False, location="json") +) + + @console_ns.route("/account/change-email") class ChangeEmailSendEmailApi(Resource): + @api.expect(parser_change_email) @enable_change_email @setup_required @login_required @account_initialization_required def post(self): current_user, _ = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("email", type=email, required=True, location="json") - .add_argument("language", type=str, required=False, location="json") - .add_argument("phase", type=str, required=False, location="json") - .add_argument("token", type=str, required=False, location="json") - ) - args = parser.parse_args() + args = parser_change_email.parse_args() ip_address = extract_remote_ip(request) if AccountService.is_email_send_ip_limit(ip_address): @@ -470,20 +507,23 @@ class ChangeEmailSendEmailApi(Resource): return {"result": "success", "data": token} +parser_validity = ( + reqparse.RequestParser() + .add_argument("email", type=email, required=True, location="json") + .add_argument("code", type=str, required=True, location="json") + .add_argument("token", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/account/change-email/validity") class ChangeEmailCheckApi(Resource): + @api.expect(parser_validity) @enable_change_email @setup_required @login_required @account_initialization_required def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("email", type=email, required=True, location="json") - .add_argument("code", type=str, required=True, location="json") - .add_argument("token", type=str, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_validity.parse_args() user_email = args["email"] @@ -514,20 +554,23 @@ class ChangeEmailCheckApi(Resource): return {"is_valid": True, "email": token_data.get("email"), "token": new_token} +parser_reset = ( + reqparse.RequestParser() + .add_argument("new_email", type=email, required=True, location="json") + .add_argument("token", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/account/change-email/reset") class ChangeEmailResetApi(Resource): + @api.expect(parser_reset) @enable_change_email @setup_required @login_required @account_initialization_required @marshal_with(account_fields) def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("new_email", type=email, required=True, location="json") - .add_argument("token", type=str, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_reset.parse_args() if AccountService.is_account_in_freeze(args["new_email"]): raise AccountInFreezeError() @@ -555,12 +598,15 @@ class ChangeEmailResetApi(Resource): return updated_account +parser_check = reqparse.RequestParser().add_argument("email", type=email, required=True, location="json") + + @console_ns.route("/account/change-email/check-email-unique") class CheckEmailUnique(Resource): + @api.expect(parser_check) @setup_required def post(self): - parser = reqparse.RequestParser().add_argument("email", type=email, required=True, location="json") - args = parser.parse_args() + args = parser_check.parse_args() if AccountService.is_account_in_freeze(args["email"]): raise AccountInFreezeError() if not AccountService.check_email_unique(args["email"]): diff --git a/api/controllers/console/workspace/members.py b/api/controllers/console/workspace/members.py index d66f861799..3ca453f1da 100644 --- a/api/controllers/console/workspace/members.py +++ b/api/controllers/console/workspace/members.py @@ -5,7 +5,7 @@ from flask_restx import Resource, marshal_with, reqparse import services from configs import dify_config -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.auth.error import ( CannotTransferOwnerToSelfError, EmailCodeError, @@ -48,22 +48,25 @@ class MemberListApi(Resource): return {"result": "success", "accounts": members}, 200 +parser_invite = ( + reqparse.RequestParser() + .add_argument("emails", type=list, required=True, location="json") + .add_argument("role", type=str, required=True, default="admin", location="json") + .add_argument("language", type=str, required=False, location="json") +) + + @console_ns.route("/workspaces/current/members/invite-email") class MemberInviteEmailApi(Resource): """Invite a new member by email.""" + @api.expect(parser_invite) @setup_required @login_required @account_initialization_required @cloud_edition_billing_resource_check("members") def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("emails", type=list, required=True, location="json") - .add_argument("role", type=str, required=True, default="admin", location="json") - .add_argument("language", type=str, required=False, location="json") - ) - args = parser.parse_args() + args = parser_invite.parse_args() invitee_emails = args["emails"] invitee_role = args["role"] @@ -143,16 +146,19 @@ class MemberCancelInviteApi(Resource): }, 200 +parser_update = reqparse.RequestParser().add_argument("role", type=str, required=True, location="json") + + @console_ns.route("/workspaces/current/members//update-role") class MemberUpdateRoleApi(Resource): """Update member role.""" + @api.expect(parser_update) @setup_required @login_required @account_initialization_required def put(self, member_id): - parser = reqparse.RequestParser().add_argument("role", type=str, required=True, location="json") - args = parser.parse_args() + args = parser_update.parse_args() new_role = args["role"] if not TenantAccountRole.is_valid_role(new_role): @@ -191,17 +197,20 @@ class DatasetOperatorMemberListApi(Resource): return {"result": "success", "accounts": members}, 200 +parser_send = reqparse.RequestParser().add_argument("language", type=str, required=False, location="json") + + @console_ns.route("/workspaces/current/members/send-owner-transfer-confirm-email") class SendOwnerTransferEmailApi(Resource): """Send owner transfer email.""" + @api.expect(parser_send) @setup_required @login_required @account_initialization_required @is_allow_transfer_owner def post(self): - parser = reqparse.RequestParser().add_argument("language", type=str, required=False, location="json") - args = parser.parse_args() + args = parser_send.parse_args() ip_address = extract_remote_ip(request) if AccountService.is_email_send_ip_limit(ip_address): raise EmailSendIpLimitError() @@ -229,19 +238,22 @@ class SendOwnerTransferEmailApi(Resource): return {"result": "success", "data": token} +parser_owner = ( + reqparse.RequestParser() + .add_argument("code", type=str, required=True, location="json") + .add_argument("token", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/workspaces/current/members/owner-transfer-check") class OwnerTransferCheckApi(Resource): + @api.expect(parser_owner) @setup_required @login_required @account_initialization_required @is_allow_transfer_owner def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("code", type=str, required=True, location="json") - .add_argument("token", type=str, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_owner.parse_args() # check if the current user is the owner of the workspace current_user, _ = current_account_with_tenant() if not current_user.current_tenant: @@ -276,17 +288,20 @@ class OwnerTransferCheckApi(Resource): return {"is_valid": True, "email": token_data.get("email"), "token": new_token} +parser_owner_transfer = reqparse.RequestParser().add_argument( + "token", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/members//owner-transfer") class OwnerTransfer(Resource): + @api.expect(parser_owner_transfer) @setup_required @login_required @account_initialization_required @is_allow_transfer_owner def post(self, member_id): - parser = reqparse.RequestParser().add_argument( - "token", type=str, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + args = parser_owner_transfer.parse_args() # check if the current user is the owner of the workspace current_user, _ = current_account_with_tenant() diff --git a/api/controllers/console/workspace/model_providers.py b/api/controllers/console/workspace/model_providers.py index 04db975fc2..832ec8af0f 100644 --- a/api/controllers/console/workspace/model_providers.py +++ b/api/controllers/console/workspace/model_providers.py @@ -4,7 +4,7 @@ from flask import send_file from flask_restx import Resource, reqparse from werkzeug.exceptions import Forbidden -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.wraps import account_initialization_required, setup_required from core.model_runtime.entities.model_entities import ModelType from core.model_runtime.errors.validate import CredentialsValidateFailedError @@ -14,9 +14,19 @@ from libs.login import current_account_with_tenant, login_required from services.billing_service import BillingService from services.model_provider_service import ModelProviderService +parser_model = reqparse.RequestParser().add_argument( + "model_type", + type=str, + required=False, + nullable=True, + choices=[mt.value for mt in ModelType], + location="args", +) + @console_ns.route("/workspaces/current/model-providers") class ModelProviderListApi(Resource): + @api.expect(parser_model) @setup_required @login_required @account_initialization_required @@ -24,15 +34,7 @@ class ModelProviderListApi(Resource): _, current_tenant_id = current_account_with_tenant() tenant_id = current_tenant_id - parser = reqparse.RequestParser().add_argument( - "model_type", - type=str, - required=False, - nullable=True, - choices=[mt.value for mt in ModelType], - location="args", - ) - args = parser.parse_args() + args = parser_model.parse_args() model_provider_service = ModelProviderService() provider_list = model_provider_service.get_provider_list(tenant_id=tenant_id, model_type=args.get("model_type")) @@ -40,8 +42,30 @@ class ModelProviderListApi(Resource): return jsonable_encoder({"data": provider_list}) +parser_cred = reqparse.RequestParser().add_argument( + "credential_id", type=uuid_value, required=False, nullable=True, location="args" +) +parser_post_cred = ( + reqparse.RequestParser() + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") + .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") +) + +parser_put_cred = ( + reqparse.RequestParser() + .add_argument("credential_id", type=uuid_value, required=True, nullable=False, location="json") + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") + .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") +) + +parser_delete_cred = reqparse.RequestParser().add_argument( + "credential_id", type=uuid_value, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/model-providers//credentials") class ModelProviderCredentialApi(Resource): + @api.expect(parser_cred) @setup_required @login_required @account_initialization_required @@ -49,10 +73,7 @@ class ModelProviderCredentialApi(Resource): _, current_tenant_id = current_account_with_tenant() tenant_id = current_tenant_id # if credential_id is not provided, return current used credential - parser = reqparse.RequestParser().add_argument( - "credential_id", type=uuid_value, required=False, nullable=True, location="args" - ) - args = parser.parse_args() + args = parser_cred.parse_args() model_provider_service = ModelProviderService() credentials = model_provider_service.get_provider_credential( @@ -61,6 +82,7 @@ class ModelProviderCredentialApi(Resource): return {"credentials": credentials} + @api.expect(parser_post_cred) @setup_required @login_required @account_initialization_required @@ -69,12 +91,7 @@ class ModelProviderCredentialApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_post_cred.parse_args() model_provider_service = ModelProviderService() @@ -90,6 +107,7 @@ class ModelProviderCredentialApi(Resource): return {"result": "success"}, 201 + @api.expect(parser_put_cred) @setup_required @login_required @account_initialization_required @@ -98,13 +116,7 @@ class ModelProviderCredentialApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("credential_id", type=uuid_value, required=True, nullable=False, location="json") - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_put_cred.parse_args() model_provider_service = ModelProviderService() @@ -121,6 +133,7 @@ class ModelProviderCredentialApi(Resource): return {"result": "success"} + @api.expect(parser_delete_cred) @setup_required @login_required @account_initialization_required @@ -128,10 +141,8 @@ class ModelProviderCredentialApi(Resource): current_user, current_tenant_id = current_account_with_tenant() if not current_user.is_admin_or_owner: raise Forbidden() - parser = reqparse.RequestParser().add_argument( - "credential_id", type=uuid_value, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + + args = parser_delete_cred.parse_args() model_provider_service = ModelProviderService() model_provider_service.remove_provider_credential( @@ -141,8 +152,14 @@ class ModelProviderCredentialApi(Resource): return {"result": "success"}, 204 +parser_switch = reqparse.RequestParser().add_argument( + "credential_id", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/model-providers//credentials/switch") class ModelProviderCredentialSwitchApi(Resource): + @api.expect(parser_switch) @setup_required @login_required @account_initialization_required @@ -150,10 +167,7 @@ class ModelProviderCredentialSwitchApi(Resource): current_user, current_tenant_id = current_account_with_tenant() if not current_user.is_admin_or_owner: raise Forbidden() - parser = reqparse.RequestParser().add_argument( - "credential_id", type=str, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + args = parser_switch.parse_args() service = ModelProviderService() service.switch_active_provider_credential( @@ -164,17 +178,20 @@ class ModelProviderCredentialSwitchApi(Resource): return {"result": "success"} +parser_validate = reqparse.RequestParser().add_argument( + "credentials", type=dict, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/model-providers//credentials/validate") class ModelProviderValidateApi(Resource): + @api.expect(parser_validate) @setup_required @login_required @account_initialization_required def post(self, provider: str): _, current_tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument( - "credentials", type=dict, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + args = parser_validate.parse_args() tenant_id = current_tenant_id @@ -218,8 +235,19 @@ class ModelProviderIconApi(Resource): return send_file(io.BytesIO(icon), mimetype=mimetype) +parser_preferred = reqparse.RequestParser().add_argument( + "preferred_provider_type", + type=str, + required=True, + nullable=False, + choices=["system", "custom"], + location="json", +) + + @console_ns.route("/workspaces/current/model-providers//preferred-provider-type") class PreferredProviderTypeUpdateApi(Resource): + @api.expect(parser_preferred) @setup_required @login_required @account_initialization_required @@ -230,15 +258,7 @@ class PreferredProviderTypeUpdateApi(Resource): tenant_id = current_tenant_id - parser = reqparse.RequestParser().add_argument( - "preferred_provider_type", - type=str, - required=True, - nullable=False, - choices=["system", "custom"], - location="json", - ) - args = parser.parse_args() + args = parser_preferred.parse_args() model_provider_service = ModelProviderService() model_provider_service.switch_preferred_provider( diff --git a/api/controllers/console/workspace/models.py b/api/controllers/console/workspace/models.py index 5ab958d585..d6aad129a6 100644 --- a/api/controllers/console/workspace/models.py +++ b/api/controllers/console/workspace/models.py @@ -3,7 +3,7 @@ import logging from flask_restx import Resource, reqparse from werkzeug.exceptions import Forbidden -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.wraps import account_initialization_required, setup_required from core.model_runtime.entities.model_entities import ModelType from core.model_runtime.errors.validate import CredentialsValidateFailedError @@ -16,23 +16,29 @@ from services.model_provider_service import ModelProviderService logger = logging.getLogger(__name__) +parser_get_default = reqparse.RequestParser().add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="args", +) +parser_post_default = reqparse.RequestParser().add_argument( + "model_settings", type=list, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/default-model") class DefaultModelApi(Resource): + @api.expect(parser_get_default) @setup_required @login_required @account_initialization_required def get(self): _, tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="args", - ) - args = parser.parse_args() + args = parser_get_default.parse_args() model_provider_service = ModelProviderService() default_model_entity = model_provider_service.get_default_model_of_model_type( @@ -41,6 +47,7 @@ class DefaultModelApi(Resource): return jsonable_encoder({"data": default_model_entity}) + @api.expect(parser_post_default) @setup_required @login_required @account_initialization_required @@ -50,10 +57,7 @@ class DefaultModelApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = reqparse.RequestParser().add_argument( - "model_settings", type=list, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + args = parser_post_default.parse_args() model_provider_service = ModelProviderService() model_settings = args["model_settings"] for model_setting in model_settings: @@ -84,6 +88,35 @@ class DefaultModelApi(Resource): return {"result": "success"} +parser_post_models = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) + .add_argument("load_balancing", type=dict, required=False, nullable=True, location="json") + .add_argument("config_from", type=str, required=False, nullable=True, location="json") + .add_argument("credential_id", type=uuid_value, required=False, nullable=True, location="json") +) +parser_delete_models = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) +) + + @console_ns.route("/workspaces/current/model-providers//models") class ModelProviderModelApi(Resource): @setup_required @@ -97,6 +130,7 @@ class ModelProviderModelApi(Resource): return jsonable_encoder({"data": models}) + @api.expect(parser_post_models) @setup_required @login_required @account_initialization_required @@ -106,23 +140,7 @@ class ModelProviderModelApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - .add_argument("load_balancing", type=dict, required=False, nullable=True, location="json") - .add_argument("config_from", type=str, required=False, nullable=True, location="json") - .add_argument("credential_id", type=uuid_value, required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_post_models.parse_args() if args.get("config_from", "") == "custom-model": if not args.get("credential_id"): @@ -160,6 +178,7 @@ class ModelProviderModelApi(Resource): return {"result": "success"}, 200 + @api.expect(parser_delete_models) @setup_required @login_required @account_initialization_required @@ -169,19 +188,7 @@ class ModelProviderModelApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - ) - args = parser.parse_args() + args = parser_delete_models.parse_args() model_provider_service = ModelProviderService() model_provider_service.remove_model( @@ -191,29 +198,76 @@ class ModelProviderModelApi(Resource): return {"result": "success"}, 204 +parser_get_credentials = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="args") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="args", + ) + .add_argument("config_from", type=str, required=False, nullable=True, location="args") + .add_argument("credential_id", type=uuid_value, required=False, nullable=True, location="args") +) + + +parser_post_cred = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) + .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") +) +parser_put_cred = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) + .add_argument("credential_id", type=uuid_value, required=True, nullable=False, location="json") + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") + .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") +) +parser_delete_cred = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) + .add_argument("credential_id", type=uuid_value, required=True, nullable=False, location="json") +) + + @console_ns.route("/workspaces/current/model-providers//models/credentials") class ModelProviderModelCredentialApi(Resource): + @api.expect(parser_get_credentials) @setup_required @login_required @account_initialization_required def get(self, provider: str): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="args") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="args", - ) - .add_argument("config_from", type=str, required=False, nullable=True, location="args") - .add_argument("credential_id", type=uuid_value, required=False, nullable=True, location="args") - ) - args = parser.parse_args() + args = parser_get_credentials.parse_args() model_provider_service = ModelProviderService() current_credential = model_provider_service.get_model_credential( @@ -257,6 +311,7 @@ class ModelProviderModelCredentialApi(Resource): } ) + @api.expect(parser_post_cred) @setup_required @login_required @account_initialization_required @@ -266,21 +321,7 @@ class ModelProviderModelCredentialApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_post_cred.parse_args() model_provider_service = ModelProviderService() @@ -304,6 +345,7 @@ class ModelProviderModelCredentialApi(Resource): return {"result": "success"}, 201 + @api.expect(parser_put_cred) @setup_required @login_required @account_initialization_required @@ -313,22 +355,7 @@ class ModelProviderModelCredentialApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - .add_argument("credential_id", type=uuid_value, required=True, nullable=False, location="json") - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_put_cred.parse_args() model_provider_service = ModelProviderService() @@ -347,6 +374,7 @@ class ModelProviderModelCredentialApi(Resource): return {"result": "success"} + @api.expect(parser_delete_cred) @setup_required @login_required @account_initialization_required @@ -355,20 +383,7 @@ class ModelProviderModelCredentialApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - .add_argument("credential_id", type=uuid_value, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_delete_cred.parse_args() model_provider_service = ModelProviderService() model_provider_service.remove_model_credential( @@ -382,8 +397,24 @@ class ModelProviderModelCredentialApi(Resource): return {"result": "success"}, 204 +parser_switch = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) + .add_argument("credential_id", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/workspaces/current/model-providers//models/credentials/switch") class ModelProviderModelCredentialSwitchApi(Resource): + @api.expect(parser_switch) @setup_required @login_required @account_initialization_required @@ -392,20 +423,7 @@ class ModelProviderModelCredentialSwitchApi(Resource): if not current_user.is_admin_or_owner: raise Forbidden() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - .add_argument("credential_id", type=str, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_switch.parse_args() service = ModelProviderService() service.add_model_credential_to_model_list( @@ -418,29 +436,32 @@ class ModelProviderModelCredentialSwitchApi(Resource): return {"result": "success"} +parser_model_enable_disable = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) +) + + @console_ns.route( "/workspaces/current/model-providers//models/enable", endpoint="model-provider-model-enable" ) class ModelProviderModelEnableApi(Resource): + @api.expect(parser_model_enable_disable) @setup_required @login_required @account_initialization_required def patch(self, provider: str): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - ) - args = parser.parse_args() + args = parser_model_enable_disable.parse_args() model_provider_service = ModelProviderService() model_provider_service.enable_model( @@ -454,25 +475,14 @@ class ModelProviderModelEnableApi(Resource): "/workspaces/current/model-providers//models/disable", endpoint="model-provider-model-disable" ) class ModelProviderModelDisableApi(Resource): + @api.expect(parser_model_enable_disable) @setup_required @login_required @account_initialization_required def patch(self, provider: str): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - ) - args = parser.parse_args() + args = parser_model_enable_disable.parse_args() model_provider_service = ModelProviderService() model_provider_service.disable_model( @@ -482,28 +492,31 @@ class ModelProviderModelDisableApi(Resource): return {"result": "success"} +parser_validate = ( + reqparse.RequestParser() + .add_argument("model", type=str, required=True, nullable=False, location="json") + .add_argument( + "model_type", + type=str, + required=True, + nullable=False, + choices=[mt.value for mt in ModelType], + location="json", + ) + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") +) + + @console_ns.route("/workspaces/current/model-providers//models/credentials/validate") class ModelProviderModelValidateApi(Resource): + @api.expect(parser_validate) @setup_required @login_required @account_initialization_required def post(self, provider: str): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("model", type=str, required=True, nullable=False, location="json") - .add_argument( - "model_type", - type=str, - required=True, - nullable=False, - choices=[mt.value for mt in ModelType], - location="json", - ) - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_validate.parse_args() model_provider_service = ModelProviderService() @@ -530,16 +543,19 @@ class ModelProviderModelValidateApi(Resource): return response +parser_parameter = reqparse.RequestParser().add_argument( + "model", type=str, required=True, nullable=False, location="args" +) + + @console_ns.route("/workspaces/current/model-providers//models/parameter-rules") class ModelProviderModelParameterRuleApi(Resource): + @api.expect(parser_parameter) @setup_required @login_required @account_initialization_required def get(self, provider: str): - parser = reqparse.RequestParser().add_argument( - "model", type=str, required=True, nullable=False, location="args" - ) - args = parser.parse_args() + args = parser_parameter.parse_args() _, tenant_id = current_account_with_tenant() model_provider_service = ModelProviderService() diff --git a/api/controllers/console/workspace/plugin.py b/api/controllers/console/workspace/plugin.py index e8bc312caf..bb8c02b99a 100644 --- a/api/controllers/console/workspace/plugin.py +++ b/api/controllers/console/workspace/plugin.py @@ -5,7 +5,7 @@ from flask_restx import Resource, reqparse from werkzeug.exceptions import Forbidden from configs import dify_config -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.workspace import plugin_permission_required from controllers.console.wraps import account_initialization_required, setup_required from core.model_runtime.utils.encoders import jsonable_encoder @@ -37,19 +37,22 @@ class PluginDebuggingKeyApi(Resource): raise ValueError(e) +parser_list = ( + reqparse.RequestParser() + .add_argument("page", type=int, required=False, location="args", default=1) + .add_argument("page_size", type=int, required=False, location="args", default=256) +) + + @console_ns.route("/workspaces/current/plugin/list") class PluginListApi(Resource): + @api.expect(parser_list) @setup_required @login_required @account_initialization_required def get(self): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("page", type=int, required=False, location="args", default=1) - .add_argument("page_size", type=int, required=False, location="args", default=256) - ) - args = parser.parse_args() + args = parser_list.parse_args() try: plugins_with_total = PluginService.list_with_total(tenant_id, args["page"], args["page_size"]) except PluginDaemonClientSideError as e: @@ -58,14 +61,17 @@ class PluginListApi(Resource): return jsonable_encoder({"plugins": plugins_with_total.list, "total": plugins_with_total.total}) +parser_latest = reqparse.RequestParser().add_argument("plugin_ids", type=list, required=True, location="json") + + @console_ns.route("/workspaces/current/plugin/list/latest-versions") class PluginListLatestVersionsApi(Resource): + @api.expect(parser_latest) @setup_required @login_required @account_initialization_required def post(self): - req = reqparse.RequestParser().add_argument("plugin_ids", type=list, required=True, location="json") - args = req.parse_args() + args = parser_latest.parse_args() try: versions = PluginService.list_latest_versions(args["plugin_ids"]) @@ -75,16 +81,19 @@ class PluginListLatestVersionsApi(Resource): return jsonable_encoder({"versions": versions}) +parser_ids = reqparse.RequestParser().add_argument("plugin_ids", type=list, required=True, location="json") + + @console_ns.route("/workspaces/current/plugin/list/installations/ids") class PluginListInstallationsFromIdsApi(Resource): + @api.expect(parser_ids) @setup_required @login_required @account_initialization_required def post(self): _, tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("plugin_ids", type=list, required=True, location="json") - args = parser.parse_args() + args = parser_ids.parse_args() try: plugins = PluginService.list_installations_from_ids(tenant_id, args["plugin_ids"]) @@ -94,16 +103,19 @@ class PluginListInstallationsFromIdsApi(Resource): return jsonable_encoder({"plugins": plugins}) +parser_icon = ( + reqparse.RequestParser() + .add_argument("tenant_id", type=str, required=True, location="args") + .add_argument("filename", type=str, required=True, location="args") +) + + @console_ns.route("/workspaces/current/plugin/icon") class PluginIconApi(Resource): + @api.expect(parser_icon) @setup_required def get(self): - req = ( - reqparse.RequestParser() - .add_argument("tenant_id", type=str, required=True, location="args") - .add_argument("filename", type=str, required=True, location="args") - ) - args = req.parse_args() + args = parser_icon.parse_args() try: icon_bytes, mimetype = PluginService.get_asset(args["tenant_id"], args["filename"]) @@ -114,6 +126,25 @@ class PluginIconApi(Resource): return send_file(io.BytesIO(icon_bytes), mimetype=mimetype, max_age=icon_cache_max_age) +@console_ns.route("/workspaces/current/plugin/asset") +class PluginAssetApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self): + req = reqparse.RequestParser() + req.add_argument("plugin_unique_identifier", type=str, required=True, location="args") + req.add_argument("file_name", type=str, required=True, location="args") + args = req.parse_args() + + _, tenant_id = current_account_with_tenant() + try: + binary = PluginService.extract_asset(tenant_id, args["plugin_unique_identifier"], args["file_name"]) + return send_file(io.BytesIO(binary), mimetype="application/octet-stream") + except PluginDaemonClientSideError as e: + raise ValueError(e) + + @console_ns.route("/workspaces/current/plugin/upload/pkg") class PluginUploadFromPkgApi(Resource): @setup_required @@ -138,8 +169,17 @@ class PluginUploadFromPkgApi(Resource): return jsonable_encoder(response) +parser_github = ( + reqparse.RequestParser() + .add_argument("repo", type=str, required=True, location="json") + .add_argument("version", type=str, required=True, location="json") + .add_argument("package", type=str, required=True, location="json") +) + + @console_ns.route("/workspaces/current/plugin/upload/github") class PluginUploadFromGithubApi(Resource): + @api.expect(parser_github) @setup_required @login_required @account_initialization_required @@ -147,13 +187,7 @@ class PluginUploadFromGithubApi(Resource): def post(self): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("repo", type=str, required=True, location="json") - .add_argument("version", type=str, required=True, location="json") - .add_argument("package", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_github.parse_args() try: response = PluginService.upload_pkg_from_github(tenant_id, args["repo"], args["version"], args["package"]) @@ -187,19 +221,21 @@ class PluginUploadFromBundleApi(Resource): return jsonable_encoder(response) +parser_pkg = reqparse.RequestParser().add_argument( + "plugin_unique_identifiers", type=list, required=True, location="json" +) + + @console_ns.route("/workspaces/current/plugin/install/pkg") class PluginInstallFromPkgApi(Resource): + @api.expect(parser_pkg) @setup_required @login_required @account_initialization_required @plugin_permission_required(install_required=True) def post(self): _, tenant_id = current_account_with_tenant() - - parser = reqparse.RequestParser().add_argument( - "plugin_unique_identifiers", type=list, required=True, location="json" - ) - args = parser.parse_args() + args = parser_pkg.parse_args() # check if all plugin_unique_identifiers are valid string for plugin_unique_identifier in args["plugin_unique_identifiers"]: @@ -214,8 +250,18 @@ class PluginInstallFromPkgApi(Resource): return jsonable_encoder(response) +parser_githubapi = ( + reqparse.RequestParser() + .add_argument("repo", type=str, required=True, location="json") + .add_argument("version", type=str, required=True, location="json") + .add_argument("package", type=str, required=True, location="json") + .add_argument("plugin_unique_identifier", type=str, required=True, location="json") +) + + @console_ns.route("/workspaces/current/plugin/install/github") class PluginInstallFromGithubApi(Resource): + @api.expect(parser_githubapi) @setup_required @login_required @account_initialization_required @@ -223,14 +269,7 @@ class PluginInstallFromGithubApi(Resource): def post(self): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("repo", type=str, required=True, location="json") - .add_argument("version", type=str, required=True, location="json") - .add_argument("package", type=str, required=True, location="json") - .add_argument("plugin_unique_identifier", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_githubapi.parse_args() try: response = PluginService.install_from_github( @@ -246,8 +285,14 @@ class PluginInstallFromGithubApi(Resource): return jsonable_encoder(response) +parser_marketplace = reqparse.RequestParser().add_argument( + "plugin_unique_identifiers", type=list, required=True, location="json" +) + + @console_ns.route("/workspaces/current/plugin/install/marketplace") class PluginInstallFromMarketplaceApi(Resource): + @api.expect(parser_marketplace) @setup_required @login_required @account_initialization_required @@ -255,10 +300,7 @@ class PluginInstallFromMarketplaceApi(Resource): def post(self): _, tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument( - "plugin_unique_identifiers", type=list, required=True, location="json" - ) - args = parser.parse_args() + args = parser_marketplace.parse_args() # check if all plugin_unique_identifiers are valid string for plugin_unique_identifier in args["plugin_unique_identifiers"]: @@ -273,19 +315,21 @@ class PluginInstallFromMarketplaceApi(Resource): return jsonable_encoder(response) +parser_pkgapi = reqparse.RequestParser().add_argument( + "plugin_unique_identifier", type=str, required=True, location="args" +) + + @console_ns.route("/workspaces/current/plugin/marketplace/pkg") class PluginFetchMarketplacePkgApi(Resource): + @api.expect(parser_pkgapi) @setup_required @login_required @account_initialization_required @plugin_permission_required(install_required=True) def get(self): _, tenant_id = current_account_with_tenant() - - parser = reqparse.RequestParser().add_argument( - "plugin_unique_identifier", type=str, required=True, location="args" - ) - args = parser.parse_args() + args = parser_pkgapi.parse_args() try: return jsonable_encoder( @@ -300,8 +344,14 @@ class PluginFetchMarketplacePkgApi(Resource): raise ValueError(e) +parser_fetch = reqparse.RequestParser().add_argument( + "plugin_unique_identifier", type=str, required=True, location="args" +) + + @console_ns.route("/workspaces/current/plugin/fetch-manifest") class PluginFetchManifestApi(Resource): + @api.expect(parser_fetch) @setup_required @login_required @account_initialization_required @@ -309,10 +359,7 @@ class PluginFetchManifestApi(Resource): def get(self): _, tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument( - "plugin_unique_identifier", type=str, required=True, location="args" - ) - args = parser.parse_args() + args = parser_fetch.parse_args() try: return jsonable_encoder( @@ -326,8 +373,16 @@ class PluginFetchManifestApi(Resource): raise ValueError(e) +parser_tasks = ( + reqparse.RequestParser() + .add_argument("page", type=int, required=True, location="args") + .add_argument("page_size", type=int, required=True, location="args") +) + + @console_ns.route("/workspaces/current/plugin/tasks") class PluginFetchInstallTasksApi(Resource): + @api.expect(parser_tasks) @setup_required @login_required @account_initialization_required @@ -335,12 +390,7 @@ class PluginFetchInstallTasksApi(Resource): def get(self): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("page", type=int, required=True, location="args") - .add_argument("page_size", type=int, required=True, location="args") - ) - args = parser.parse_args() + args = parser_tasks.parse_args() try: return jsonable_encoder( @@ -410,8 +460,16 @@ class PluginDeleteInstallTaskItemApi(Resource): raise ValueError(e) +parser_marketplace_api = ( + reqparse.RequestParser() + .add_argument("original_plugin_unique_identifier", type=str, required=True, location="json") + .add_argument("new_plugin_unique_identifier", type=str, required=True, location="json") +) + + @console_ns.route("/workspaces/current/plugin/upgrade/marketplace") class PluginUpgradeFromMarketplaceApi(Resource): + @api.expect(parser_marketplace_api) @setup_required @login_required @account_initialization_required @@ -419,12 +477,7 @@ class PluginUpgradeFromMarketplaceApi(Resource): def post(self): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("original_plugin_unique_identifier", type=str, required=True, location="json") - .add_argument("new_plugin_unique_identifier", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_marketplace_api.parse_args() try: return jsonable_encoder( @@ -436,8 +489,19 @@ class PluginUpgradeFromMarketplaceApi(Resource): raise ValueError(e) +parser_github_post = ( + reqparse.RequestParser() + .add_argument("original_plugin_unique_identifier", type=str, required=True, location="json") + .add_argument("new_plugin_unique_identifier", type=str, required=True, location="json") + .add_argument("repo", type=str, required=True, location="json") + .add_argument("version", type=str, required=True, location="json") + .add_argument("package", type=str, required=True, location="json") +) + + @console_ns.route("/workspaces/current/plugin/upgrade/github") class PluginUpgradeFromGithubApi(Resource): + @api.expect(parser_github_post) @setup_required @login_required @account_initialization_required @@ -445,15 +509,7 @@ class PluginUpgradeFromGithubApi(Resource): def post(self): _, tenant_id = current_account_with_tenant() - parser = ( - reqparse.RequestParser() - .add_argument("original_plugin_unique_identifier", type=str, required=True, location="json") - .add_argument("new_plugin_unique_identifier", type=str, required=True, location="json") - .add_argument("repo", type=str, required=True, location="json") - .add_argument("version", type=str, required=True, location="json") - .add_argument("package", type=str, required=True, location="json") - ) - args = parser.parse_args() + args = parser_github_post.parse_args() try: return jsonable_encoder( @@ -470,15 +526,20 @@ class PluginUpgradeFromGithubApi(Resource): raise ValueError(e) +parser_uninstall = reqparse.RequestParser().add_argument( + "plugin_installation_id", type=str, required=True, location="json" +) + + @console_ns.route("/workspaces/current/plugin/uninstall") class PluginUninstallApi(Resource): + @api.expect(parser_uninstall) @setup_required @login_required @account_initialization_required @plugin_permission_required(install_required=True) def post(self): - req = reqparse.RequestParser().add_argument("plugin_installation_id", type=str, required=True, location="json") - args = req.parse_args() + args = parser_uninstall.parse_args() _, tenant_id = current_account_with_tenant() @@ -488,8 +549,16 @@ class PluginUninstallApi(Resource): raise ValueError(e) +parser_change_post = ( + reqparse.RequestParser() + .add_argument("install_permission", type=str, required=True, location="json") + .add_argument("debug_permission", type=str, required=True, location="json") +) + + @console_ns.route("/workspaces/current/plugin/permission/change") class PluginChangePermissionApi(Resource): + @api.expect(parser_change_post) @setup_required @login_required @account_initialization_required @@ -499,12 +568,7 @@ class PluginChangePermissionApi(Resource): if not user.is_admin_or_owner: raise Forbidden() - req = ( - reqparse.RequestParser() - .add_argument("install_permission", type=str, required=True, location="json") - .add_argument("debug_permission", type=str, required=True, location="json") - ) - args = req.parse_args() + args = parser_change_post.parse_args() install_permission = TenantPluginPermission.InstallPermission(args["install_permission"]) debug_permission = TenantPluginPermission.DebugPermission(args["debug_permission"]) @@ -539,8 +603,20 @@ class PluginFetchPermissionApi(Resource): ) +parser_dynamic = ( + reqparse.RequestParser() + .add_argument("plugin_id", type=str, required=True, location="args") + .add_argument("provider", type=str, required=True, location="args") + .add_argument("action", type=str, required=True, location="args") + .add_argument("parameter", type=str, required=True, location="args") + .add_argument("credential_id", type=str, required=False, location="args") + .add_argument("provider_type", type=str, required=True, location="args") +) + + @console_ns.route("/workspaces/current/plugin/parameters/dynamic-options") class PluginFetchDynamicSelectOptionsApi(Resource): + @api.expect(parser_dynamic) @setup_required @login_required @account_initialization_required @@ -552,25 +628,18 @@ class PluginFetchDynamicSelectOptionsApi(Resource): user_id = current_user.id - parser = ( - reqparse.RequestParser() - .add_argument("plugin_id", type=str, required=True, location="args") - .add_argument("provider", type=str, required=True, location="args") - .add_argument("action", type=str, required=True, location="args") - .add_argument("parameter", type=str, required=True, location="args") - .add_argument("provider_type", type=str, required=True, location="args") - ) - args = parser.parse_args() + args = parser_dynamic.parse_args() try: options = PluginParameterService.get_dynamic_select_options( - tenant_id, - user_id, - args["plugin_id"], - args["provider"], - args["action"], - args["parameter"], - args["provider_type"], + tenant_id=tenant_id, + user_id=user_id, + plugin_id=args["plugin_id"], + provider=args["provider"], + action=args["action"], + parameter=args["parameter"], + credential_id=args["credential_id"], + provider_type=args["provider_type"], ) except PluginDaemonClientSideError as e: raise ValueError(e) @@ -578,8 +647,16 @@ class PluginFetchDynamicSelectOptionsApi(Resource): return jsonable_encoder({"options": options}) +parser_change = ( + reqparse.RequestParser() + .add_argument("permission", type=dict, required=True, location="json") + .add_argument("auto_upgrade", type=dict, required=True, location="json") +) + + @console_ns.route("/workspaces/current/plugin/preferences/change") class PluginChangePreferencesApi(Resource): + @api.expect(parser_change) @setup_required @login_required @account_initialization_required @@ -588,12 +665,7 @@ class PluginChangePreferencesApi(Resource): if not user.is_admin_or_owner: raise Forbidden() - req = ( - reqparse.RequestParser() - .add_argument("permission", type=dict, required=True, location="json") - .add_argument("auto_upgrade", type=dict, required=True, location="json") - ) - args = req.parse_args() + args = parser_change.parse_args() permission = args["permission"] @@ -673,8 +745,12 @@ class PluginFetchPreferencesApi(Resource): return jsonable_encoder({"permission": permission_dict, "auto_upgrade": auto_upgrade_dict}) +parser_exclude = reqparse.RequestParser().add_argument("plugin_id", type=str, required=True, location="json") + + @console_ns.route("/workspaces/current/plugin/preferences/autoupgrade/exclude") class PluginAutoUpgradeExcludePluginApi(Resource): + @api.expect(parser_exclude) @setup_required @login_required @account_initialization_required @@ -682,7 +758,26 @@ class PluginAutoUpgradeExcludePluginApi(Resource): # exclude one single plugin _, tenant_id = current_account_with_tenant() - req = reqparse.RequestParser().add_argument("plugin_id", type=str, required=True, location="json") - args = req.parse_args() + args = parser_exclude.parse_args() return jsonable_encoder({"success": PluginAutoUpgradeService.exclude_plugin(tenant_id, args["plugin_id"])}) + + +@console_ns.route("/workspaces/current/plugin/readme") +class PluginReadmeApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self): + _, tenant_id = current_account_with_tenant() + parser = reqparse.RequestParser() + parser.add_argument("plugin_unique_identifier", type=str, required=True, location="args") + parser.add_argument("language", type=str, required=False, location="args") + args = parser.parse_args() + return jsonable_encoder( + { + "readme": PluginService.fetch_plugin_readme( + tenant_id, args["plugin_unique_identifier"], args.get("language", "en-US") + ) + } + ) diff --git a/api/controllers/console/workspace/tool_providers.py b/api/controllers/console/workspace/tool_providers.py index cc50131f0a..1c9d438ca6 100644 --- a/api/controllers/console/workspace/tool_providers.py +++ b/api/controllers/console/workspace/tool_providers.py @@ -6,29 +6,33 @@ from flask_restx import ( Resource, reqparse, ) +from sqlalchemy.orm import Session from werkzeug.exceptions import Forbidden from configs import dify_config -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.wraps import ( account_initialization_required, enterprise_license_required, setup_required, ) +from core.entities.mcp_provider import MCPAuthentication, MCPConfiguration from core.mcp.auth.auth_flow import auth, handle_callback -from core.mcp.auth.auth_provider import OAuthClientProvider -from core.mcp.error import MCPAuthError, MCPError +from core.mcp.error import MCPAuthError, MCPError, MCPRefreshTokenError from core.mcp.mcp_client import MCPClient from core.model_runtime.utils.encoders import jsonable_encoder +from core.plugin.entities.plugin_daemon import CredentialType from core.plugin.impl.oauth import OAuthHandler -from core.tools.entities.tool_entities import CredentialType +from extensions.ext_database import db from libs.helper import StrLen, alphanumeric, uuid_value from libs.login import current_account_with_tenant, login_required from models.provider_ids import ToolProviderID + +# from models.provider_ids import ToolProviderID from services.plugin.oauth_service import OAuthProxyService from services.tools.api_tools_manage_service import ApiToolManageService from services.tools.builtin_tools_manage_service import BuiltinToolManageService -from services.tools.mcp_tools_manage_service import MCPToolManageService +from services.tools.mcp_tools_manage_service import MCPToolManageService, OAuthDataType from services.tools.tool_labels_service import ToolLabelsService from services.tools.tools_manage_service import ToolCommonService from services.tools.tools_transform_service import ToolTransformService @@ -42,12 +46,25 @@ def is_valid_url(url: str) -> bool: try: parsed = urlparse(url) return all([parsed.scheme, parsed.netloc]) and parsed.scheme in ["http", "https"] - except Exception: + except (ValueError, TypeError): + # ValueError: Invalid URL format + # TypeError: url is not a string return False +parser_tool = reqparse.RequestParser().add_argument( + "type", + type=str, + choices=["builtin", "model", "api", "workflow", "mcp"], + required=False, + nullable=True, + location="args", +) + + @console_ns.route("/workspaces/current/tool-providers") class ToolProviderListApi(Resource): + @api.expect(parser_tool) @setup_required @login_required @account_initialization_required @@ -56,15 +73,7 @@ class ToolProviderListApi(Resource): user_id = user.id - req = reqparse.RequestParser().add_argument( - "type", - type=str, - choices=["builtin", "model", "api", "workflow", "mcp"], - required=False, - nullable=True, - location="args", - ) - args = req.parse_args() + args = parser_tool.parse_args() return ToolCommonService.list_tool_providers(user_id, tenant_id, args.get("type", None)) @@ -96,8 +105,14 @@ class ToolBuiltinProviderInfoApi(Resource): return jsonable_encoder(BuiltinToolManageService.get_builtin_tool_provider_info(tenant_id, provider)) +parser_delete = reqparse.RequestParser().add_argument( + "credential_id", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/tool-provider/builtin//delete") class ToolBuiltinProviderDeleteApi(Resource): + @api.expect(parser_delete) @setup_required @login_required @account_initialization_required @@ -106,10 +121,7 @@ class ToolBuiltinProviderDeleteApi(Resource): if not user.is_admin_or_owner: raise Forbidden() - req = reqparse.RequestParser().add_argument( - "credential_id", type=str, required=True, nullable=False, location="json" - ) - args = req.parse_args() + args = parser_delete.parse_args() return BuiltinToolManageService.delete_builtin_tool_provider( tenant_id, @@ -118,8 +130,17 @@ class ToolBuiltinProviderDeleteApi(Resource): ) +parser_add = ( + reqparse.RequestParser() + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") + .add_argument("name", type=StrLen(30), required=False, nullable=False, location="json") + .add_argument("type", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/builtin//add") class ToolBuiltinProviderAddApi(Resource): + @api.expect(parser_add) @setup_required @login_required @account_initialization_required @@ -128,13 +149,7 @@ class ToolBuiltinProviderAddApi(Resource): user_id = user.id - parser = ( - reqparse.RequestParser() - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - .add_argument("name", type=StrLen(30), required=False, nullable=False, location="json") - .add_argument("type", type=str, required=True, nullable=False, location="json") - ) - args = parser.parse_args() + args = parser_add.parse_args() if args["type"] not in CredentialType.values(): raise ValueError(f"Invalid credential type: {args['type']}") @@ -149,8 +164,17 @@ class ToolBuiltinProviderAddApi(Resource): ) +parser_update = ( + reqparse.RequestParser() + .add_argument("credential_id", type=str, required=True, nullable=False, location="json") + .add_argument("credentials", type=dict, required=False, nullable=True, location="json") + .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/builtin//update") class ToolBuiltinProviderUpdateApi(Resource): + @api.expect(parser_update) @setup_required @login_required @account_initialization_required @@ -162,14 +186,7 @@ class ToolBuiltinProviderUpdateApi(Resource): user_id = user.id - parser = ( - reqparse.RequestParser() - .add_argument("credential_id", type=str, required=True, nullable=False, location="json") - .add_argument("credentials", type=dict, required=False, nullable=True, location="json") - .add_argument("name", type=StrLen(30), required=False, nullable=True, location="json") - ) - - args = parser.parse_args() + args = parser_update.parse_args() result = BuiltinToolManageService.update_builtin_tool_provider( user_id=user_id, @@ -207,8 +224,22 @@ class ToolBuiltinProviderIconApi(Resource): return send_file(io.BytesIO(icon_bytes), mimetype=mimetype, max_age=icon_cache_max_age) +parser_api_add = ( + reqparse.RequestParser() + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") + .add_argument("schema_type", type=str, required=True, nullable=False, location="json") + .add_argument("schema", type=str, required=True, nullable=False, location="json") + .add_argument("provider", type=str, required=True, nullable=False, location="json") + .add_argument("icon", type=dict, required=True, nullable=False, location="json") + .add_argument("privacy_policy", type=str, required=False, nullable=True, location="json") + .add_argument("labels", type=list[str], required=False, nullable=True, location="json", default=[]) + .add_argument("custom_disclaimer", type=str, required=False, nullable=True, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/api/add") class ToolApiProviderAddApi(Resource): + @api.expect(parser_api_add) @setup_required @login_required @account_initialization_required @@ -220,19 +251,7 @@ class ToolApiProviderAddApi(Resource): user_id = user.id - parser = ( - reqparse.RequestParser() - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - .add_argument("schema_type", type=str, required=True, nullable=False, location="json") - .add_argument("schema", type=str, required=True, nullable=False, location="json") - .add_argument("provider", type=str, required=True, nullable=False, location="json") - .add_argument("icon", type=dict, required=True, nullable=False, location="json") - .add_argument("privacy_policy", type=str, required=False, nullable=True, location="json") - .add_argument("labels", type=list[str], required=False, nullable=True, location="json", default=[]) - .add_argument("custom_disclaimer", type=str, required=False, nullable=True, location="json") - ) - - args = parser.parse_args() + args = parser_api_add.parse_args() return ApiToolManageService.create_api_tool_provider( user_id, @@ -248,8 +267,12 @@ class ToolApiProviderAddApi(Resource): ) +parser_remote = reqparse.RequestParser().add_argument("url", type=str, required=True, nullable=False, location="args") + + @console_ns.route("/workspaces/current/tool-provider/api/remote") class ToolApiProviderGetRemoteSchemaApi(Resource): + @api.expect(parser_remote) @setup_required @login_required @account_initialization_required @@ -258,9 +281,7 @@ class ToolApiProviderGetRemoteSchemaApi(Resource): user_id = user.id - parser = reqparse.RequestParser().add_argument("url", type=str, required=True, nullable=False, location="args") - - args = parser.parse_args() + args = parser_remote.parse_args() return ApiToolManageService.get_api_tool_provider_remote_schema( user_id, @@ -269,8 +290,14 @@ class ToolApiProviderGetRemoteSchemaApi(Resource): ) +parser_tools = reqparse.RequestParser().add_argument( + "provider", type=str, required=True, nullable=False, location="args" +) + + @console_ns.route("/workspaces/current/tool-provider/api/tools") class ToolApiProviderListToolsApi(Resource): + @api.expect(parser_tools) @setup_required @login_required @account_initialization_required @@ -279,11 +306,7 @@ class ToolApiProviderListToolsApi(Resource): user_id = user.id - parser = reqparse.RequestParser().add_argument( - "provider", type=str, required=True, nullable=False, location="args" - ) - - args = parser.parse_args() + args = parser_tools.parse_args() return jsonable_encoder( ApiToolManageService.list_api_tool_provider_tools( @@ -294,8 +317,23 @@ class ToolApiProviderListToolsApi(Resource): ) +parser_api_update = ( + reqparse.RequestParser() + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") + .add_argument("schema_type", type=str, required=True, nullable=False, location="json") + .add_argument("schema", type=str, required=True, nullable=False, location="json") + .add_argument("provider", type=str, required=True, nullable=False, location="json") + .add_argument("original_provider", type=str, required=True, nullable=False, location="json") + .add_argument("icon", type=dict, required=True, nullable=False, location="json") + .add_argument("privacy_policy", type=str, required=True, nullable=True, location="json") + .add_argument("labels", type=list[str], required=False, nullable=True, location="json") + .add_argument("custom_disclaimer", type=str, required=True, nullable=True, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/api/update") class ToolApiProviderUpdateApi(Resource): + @api.expect(parser_api_update) @setup_required @login_required @account_initialization_required @@ -307,20 +345,7 @@ class ToolApiProviderUpdateApi(Resource): user_id = user.id - parser = ( - reqparse.RequestParser() - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - .add_argument("schema_type", type=str, required=True, nullable=False, location="json") - .add_argument("schema", type=str, required=True, nullable=False, location="json") - .add_argument("provider", type=str, required=True, nullable=False, location="json") - .add_argument("original_provider", type=str, required=True, nullable=False, location="json") - .add_argument("icon", type=dict, required=True, nullable=False, location="json") - .add_argument("privacy_policy", type=str, required=True, nullable=True, location="json") - .add_argument("labels", type=list[str], required=False, nullable=True, location="json") - .add_argument("custom_disclaimer", type=str, required=True, nullable=True, location="json") - ) - - args = parser.parse_args() + args = parser_api_update.parse_args() return ApiToolManageService.update_api_tool_provider( user_id, @@ -337,8 +362,14 @@ class ToolApiProviderUpdateApi(Resource): ) +parser_api_delete = reqparse.RequestParser().add_argument( + "provider", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/tool-provider/api/delete") class ToolApiProviderDeleteApi(Resource): + @api.expect(parser_api_delete) @setup_required @login_required @account_initialization_required @@ -350,11 +381,7 @@ class ToolApiProviderDeleteApi(Resource): user_id = user.id - parser = reqparse.RequestParser().add_argument( - "provider", type=str, required=True, nullable=False, location="json" - ) - - args = parser.parse_args() + args = parser_api_delete.parse_args() return ApiToolManageService.delete_api_tool_provider( user_id, @@ -363,8 +390,12 @@ class ToolApiProviderDeleteApi(Resource): ) +parser_get = reqparse.RequestParser().add_argument("provider", type=str, required=True, nullable=False, location="args") + + @console_ns.route("/workspaces/current/tool-provider/api/get") class ToolApiProviderGetApi(Resource): + @api.expect(parser_get) @setup_required @login_required @account_initialization_required @@ -373,11 +404,7 @@ class ToolApiProviderGetApi(Resource): user_id = user.id - parser = reqparse.RequestParser().add_argument( - "provider", type=str, required=True, nullable=False, location="args" - ) - - args = parser.parse_args() + args = parser_get.parse_args() return ApiToolManageService.get_api_tool_provider( user_id, @@ -401,40 +428,44 @@ class ToolBuiltinProviderCredentialsSchemaApi(Resource): ) +parser_schema = reqparse.RequestParser().add_argument( + "schema", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/tool-provider/api/schema") class ToolApiProviderSchemaApi(Resource): + @api.expect(parser_schema) @setup_required @login_required @account_initialization_required def post(self): - parser = reqparse.RequestParser().add_argument( - "schema", type=str, required=True, nullable=False, location="json" - ) - - args = parser.parse_args() + args = parser_schema.parse_args() return ApiToolManageService.parser_api_schema( schema=args["schema"], ) +parser_pre = ( + reqparse.RequestParser() + .add_argument("tool_name", type=str, required=True, nullable=False, location="json") + .add_argument("provider_name", type=str, required=False, nullable=False, location="json") + .add_argument("credentials", type=dict, required=True, nullable=False, location="json") + .add_argument("parameters", type=dict, required=True, nullable=False, location="json") + .add_argument("schema_type", type=str, required=True, nullable=False, location="json") + .add_argument("schema", type=str, required=True, nullable=False, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/api/test/pre") class ToolApiProviderPreviousTestApi(Resource): + @api.expect(parser_pre) @setup_required @login_required @account_initialization_required def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("tool_name", type=str, required=True, nullable=False, location="json") - .add_argument("provider_name", type=str, required=False, nullable=False, location="json") - .add_argument("credentials", type=dict, required=True, nullable=False, location="json") - .add_argument("parameters", type=dict, required=True, nullable=False, location="json") - .add_argument("schema_type", type=str, required=True, nullable=False, location="json") - .add_argument("schema", type=str, required=True, nullable=False, location="json") - ) - - args = parser.parse_args() + args = parser_pre.parse_args() _, current_tenant_id = current_account_with_tenant() return ApiToolManageService.test_api_tool_preview( current_tenant_id, @@ -447,8 +478,22 @@ class ToolApiProviderPreviousTestApi(Resource): ) +parser_create = ( + reqparse.RequestParser() + .add_argument("workflow_app_id", type=uuid_value, required=True, nullable=False, location="json") + .add_argument("name", type=alphanumeric, required=True, nullable=False, location="json") + .add_argument("label", type=str, required=True, nullable=False, location="json") + .add_argument("description", type=str, required=True, nullable=False, location="json") + .add_argument("icon", type=dict, required=True, nullable=False, location="json") + .add_argument("parameters", type=list[dict], required=True, nullable=False, location="json") + .add_argument("privacy_policy", type=str, required=False, nullable=True, location="json", default="") + .add_argument("labels", type=list[str], required=False, nullable=True, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/workflow/create") class ToolWorkflowProviderCreateApi(Resource): + @api.expect(parser_create) @setup_required @login_required @account_initialization_required @@ -460,19 +505,7 @@ class ToolWorkflowProviderCreateApi(Resource): user_id = user.id - reqparser = ( - reqparse.RequestParser() - .add_argument("workflow_app_id", type=uuid_value, required=True, nullable=False, location="json") - .add_argument("name", type=alphanumeric, required=True, nullable=False, location="json") - .add_argument("label", type=str, required=True, nullable=False, location="json") - .add_argument("description", type=str, required=True, nullable=False, location="json") - .add_argument("icon", type=dict, required=True, nullable=False, location="json") - .add_argument("parameters", type=list[dict], required=True, nullable=False, location="json") - .add_argument("privacy_policy", type=str, required=False, nullable=True, location="json", default="") - .add_argument("labels", type=list[str], required=False, nullable=True, location="json") - ) - - args = reqparser.parse_args() + args = parser_create.parse_args() return WorkflowToolManageService.create_workflow_tool( user_id=user_id, @@ -488,8 +521,22 @@ class ToolWorkflowProviderCreateApi(Resource): ) +parser_workflow_update = ( + reqparse.RequestParser() + .add_argument("workflow_tool_id", type=uuid_value, required=True, nullable=False, location="json") + .add_argument("name", type=alphanumeric, required=True, nullable=False, location="json") + .add_argument("label", type=str, required=True, nullable=False, location="json") + .add_argument("description", type=str, required=True, nullable=False, location="json") + .add_argument("icon", type=dict, required=True, nullable=False, location="json") + .add_argument("parameters", type=list[dict], required=True, nullable=False, location="json") + .add_argument("privacy_policy", type=str, required=False, nullable=True, location="json", default="") + .add_argument("labels", type=list[str], required=False, nullable=True, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/workflow/update") class ToolWorkflowProviderUpdateApi(Resource): + @api.expect(parser_workflow_update) @setup_required @login_required @account_initialization_required @@ -501,19 +548,7 @@ class ToolWorkflowProviderUpdateApi(Resource): user_id = user.id - reqparser = ( - reqparse.RequestParser() - .add_argument("workflow_tool_id", type=uuid_value, required=True, nullable=False, location="json") - .add_argument("name", type=alphanumeric, required=True, nullable=False, location="json") - .add_argument("label", type=str, required=True, nullable=False, location="json") - .add_argument("description", type=str, required=True, nullable=False, location="json") - .add_argument("icon", type=dict, required=True, nullable=False, location="json") - .add_argument("parameters", type=list[dict], required=True, nullable=False, location="json") - .add_argument("privacy_policy", type=str, required=False, nullable=True, location="json", default="") - .add_argument("labels", type=list[str], required=False, nullable=True, location="json") - ) - - args = reqparser.parse_args() + args = parser_workflow_update.parse_args() if not args["workflow_tool_id"]: raise ValueError("incorrect workflow_tool_id") @@ -532,8 +567,14 @@ class ToolWorkflowProviderUpdateApi(Resource): ) +parser_workflow_delete = reqparse.RequestParser().add_argument( + "workflow_tool_id", type=uuid_value, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/tool-provider/workflow/delete") class ToolWorkflowProviderDeleteApi(Resource): + @api.expect(parser_workflow_delete) @setup_required @login_required @account_initialization_required @@ -545,11 +586,7 @@ class ToolWorkflowProviderDeleteApi(Resource): user_id = user.id - reqparser = reqparse.RequestParser().add_argument( - "workflow_tool_id", type=uuid_value, required=True, nullable=False, location="json" - ) - - args = reqparser.parse_args() + args = parser_workflow_delete.parse_args() return WorkflowToolManageService.delete_workflow_tool( user_id, @@ -558,8 +595,16 @@ class ToolWorkflowProviderDeleteApi(Resource): ) +parser_wf_get = ( + reqparse.RequestParser() + .add_argument("workflow_tool_id", type=uuid_value, required=False, nullable=True, location="args") + .add_argument("workflow_app_id", type=uuid_value, required=False, nullable=True, location="args") +) + + @console_ns.route("/workspaces/current/tool-provider/workflow/get") class ToolWorkflowProviderGetApi(Resource): + @api.expect(parser_wf_get) @setup_required @login_required @account_initialization_required @@ -568,13 +613,7 @@ class ToolWorkflowProviderGetApi(Resource): user_id = user.id - parser = ( - reqparse.RequestParser() - .add_argument("workflow_tool_id", type=uuid_value, required=False, nullable=True, location="args") - .add_argument("workflow_app_id", type=uuid_value, required=False, nullable=True, location="args") - ) - - args = parser.parse_args() + args = parser_wf_get.parse_args() if args.get("workflow_tool_id"): tool = WorkflowToolManageService.get_workflow_tool_by_tool_id( @@ -594,8 +633,14 @@ class ToolWorkflowProviderGetApi(Resource): return jsonable_encoder(tool) +parser_wf_tools = reqparse.RequestParser().add_argument( + "workflow_tool_id", type=uuid_value, required=True, nullable=False, location="args" +) + + @console_ns.route("/workspaces/current/tool-provider/workflow/tools") class ToolWorkflowProviderListToolApi(Resource): + @api.expect(parser_wf_tools) @setup_required @login_required @account_initialization_required @@ -604,11 +649,7 @@ class ToolWorkflowProviderListToolApi(Resource): user_id = user.id - parser = reqparse.RequestParser().add_argument( - "workflow_tool_id", type=uuid_value, required=True, nullable=False, location="args" - ) - - args = parser.parse_args() + args = parser_wf_tools.parse_args() return jsonable_encoder( WorkflowToolManageService.list_single_workflow_tools( @@ -784,32 +825,40 @@ class ToolOAuthCallback(Resource): return redirect(f"{dify_config.CONSOLE_WEB_URL}/oauth-callback") +parser_default_cred = reqparse.RequestParser().add_argument( + "id", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/tool-provider/builtin//default-credential") class ToolBuiltinProviderSetDefaultApi(Resource): + @api.expect(parser_default_cred) @setup_required @login_required @account_initialization_required def post(self, provider): current_user, current_tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("id", type=str, required=True, nullable=False, location="json") - args = parser.parse_args() + args = parser_default_cred.parse_args() return BuiltinToolManageService.set_default_provider( tenant_id=current_tenant_id, user_id=current_user.id, provider=provider, id=args["id"] ) +parser_custom = ( + reqparse.RequestParser() + .add_argument("client_params", type=dict, required=False, nullable=True, location="json") + .add_argument("enable_oauth_custom_client", type=bool, required=False, nullable=True, location="json") +) + + @console_ns.route("/workspaces/current/tool-provider/builtin//oauth/custom-client") class ToolOAuthCustomClient(Resource): + @api.expect(parser_custom) @setup_required @login_required @account_initialization_required def post(self, provider): - parser = ( - reqparse.RequestParser() - .add_argument("client_params", type=dict, required=False, nullable=True, location="json") - .add_argument("enable_oauth_custom_client", type=bool, required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_custom.parse_args() user, tenant_id = current_account_with_tenant() @@ -872,141 +921,185 @@ class ToolBuiltinProviderGetCredentialInfoApi(Resource): ) +parser_mcp = ( + reqparse.RequestParser() + .add_argument("server_url", type=str, required=True, nullable=False, location="json") + .add_argument("name", type=str, required=True, nullable=False, location="json") + .add_argument("icon", type=str, required=True, nullable=False, location="json") + .add_argument("icon_type", type=str, required=True, nullable=False, location="json") + .add_argument("icon_background", type=str, required=False, nullable=True, location="json", default="") + .add_argument("server_identifier", type=str, required=True, nullable=False, location="json") + .add_argument("configuration", type=dict, required=False, nullable=True, location="json", default={}) + .add_argument("headers", type=dict, required=False, nullable=True, location="json", default={}) + .add_argument("authentication", type=dict, required=False, nullable=True, location="json", default={}) +) +parser_mcp_put = ( + reqparse.RequestParser() + .add_argument("server_url", type=str, required=True, nullable=False, location="json") + .add_argument("name", type=str, required=True, nullable=False, location="json") + .add_argument("icon", type=str, required=True, nullable=False, location="json") + .add_argument("icon_type", type=str, required=True, nullable=False, location="json") + .add_argument("icon_background", type=str, required=False, nullable=True, location="json") + .add_argument("provider_id", type=str, required=True, nullable=False, location="json") + .add_argument("server_identifier", type=str, required=True, nullable=False, location="json") + .add_argument("configuration", type=dict, required=False, nullable=True, location="json", default={}) + .add_argument("headers", type=dict, required=False, nullable=True, location="json", default={}) + .add_argument("authentication", type=dict, required=False, nullable=True, location="json", default={}) +) +parser_mcp_delete = reqparse.RequestParser().add_argument( + "provider_id", type=str, required=True, nullable=False, location="json" +) + + @console_ns.route("/workspaces/current/tool-provider/mcp") class ToolProviderMCPApi(Resource): + @api.expect(parser_mcp) @setup_required @login_required @account_initialization_required def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("server_url", type=str, required=True, nullable=False, location="json") - .add_argument("name", type=str, required=True, nullable=False, location="json") - .add_argument("icon", type=str, required=True, nullable=False, location="json") - .add_argument("icon_type", type=str, required=True, nullable=False, location="json") - .add_argument("icon_background", type=str, required=False, nullable=True, location="json", default="") - .add_argument("server_identifier", type=str, required=True, nullable=False, location="json") - .add_argument("timeout", type=float, required=False, nullable=False, location="json", default=30) - .add_argument("sse_read_timeout", type=float, required=False, nullable=False, location="json", default=300) - .add_argument("headers", type=dict, required=False, nullable=True, location="json", default={}) - ) - args = parser.parse_args() + args = parser_mcp.parse_args() user, tenant_id = current_account_with_tenant() - if not is_valid_url(args["server_url"]): - raise ValueError("Server URL is not valid.") - return jsonable_encoder( - MCPToolManageService.create_mcp_provider( + + # Parse and validate models + configuration = MCPConfiguration.model_validate(args["configuration"]) + authentication = MCPAuthentication.model_validate(args["authentication"]) if args["authentication"] else None + + # Create provider + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + result = service.create_provider( tenant_id=tenant_id, + user_id=user.id, server_url=args["server_url"], name=args["name"], icon=args["icon"], icon_type=args["icon_type"], icon_background=args["icon_background"], - user_id=user.id, server_identifier=args["server_identifier"], - timeout=args["timeout"], - sse_read_timeout=args["sse_read_timeout"], headers=args["headers"], + configuration=configuration, + authentication=authentication, ) - ) + return jsonable_encoder(result) + @api.expect(parser_mcp_put) @setup_required @login_required @account_initialization_required def put(self): - parser = ( - reqparse.RequestParser() - .add_argument("server_url", type=str, required=True, nullable=False, location="json") - .add_argument("name", type=str, required=True, nullable=False, location="json") - .add_argument("icon", type=str, required=True, nullable=False, location="json") - .add_argument("icon_type", type=str, required=True, nullable=False, location="json") - .add_argument("icon_background", type=str, required=False, nullable=True, location="json") - .add_argument("provider_id", type=str, required=True, nullable=False, location="json") - .add_argument("server_identifier", type=str, required=True, nullable=False, location="json") - .add_argument("timeout", type=float, required=False, nullable=True, location="json") - .add_argument("sse_read_timeout", type=float, required=False, nullable=True, location="json") - .add_argument("headers", type=dict, required=False, nullable=True, location="json") - ) - args = parser.parse_args() - if not is_valid_url(args["server_url"]): - if "[__HIDDEN__]" in args["server_url"]: - pass - else: - raise ValueError("Server URL is not valid.") + args = parser_mcp_put.parse_args() + configuration = MCPConfiguration.model_validate(args["configuration"]) + authentication = MCPAuthentication.model_validate(args["authentication"]) if args["authentication"] else None _, current_tenant_id = current_account_with_tenant() - MCPToolManageService.update_mcp_provider( - tenant_id=current_tenant_id, - provider_id=args["provider_id"], - server_url=args["server_url"], - name=args["name"], - icon=args["icon"], - icon_type=args["icon_type"], - icon_background=args["icon_background"], - server_identifier=args["server_identifier"], - timeout=args.get("timeout"), - sse_read_timeout=args.get("sse_read_timeout"), - headers=args.get("headers"), - ) - return {"result": "success"} + # Step 1: Validate server URL change if needed (includes URL format validation and network operation) + validation_result = None + with Session(db.engine) as session: + service = MCPToolManageService(session=session) + validation_result = service.validate_server_url_change( + tenant_id=current_tenant_id, provider_id=args["provider_id"], new_server_url=args["server_url"] + ) + + # No need to check for errors here, exceptions will be raised directly + + # Step 2: Perform database update in a transaction + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + service.update_provider( + tenant_id=current_tenant_id, + provider_id=args["provider_id"], + server_url=args["server_url"], + name=args["name"], + icon=args["icon"], + icon_type=args["icon_type"], + icon_background=args["icon_background"], + server_identifier=args["server_identifier"], + headers=args["headers"], + configuration=configuration, + authentication=authentication, + validation_result=validation_result, + ) + return {"result": "success"} + + @api.expect(parser_mcp_delete) @setup_required @login_required @account_initialization_required def delete(self): - parser = reqparse.RequestParser().add_argument( - "provider_id", type=str, required=True, nullable=False, location="json" - ) - args = parser.parse_args() + args = parser_mcp_delete.parse_args() _, current_tenant_id = current_account_with_tenant() - MCPToolManageService.delete_mcp_tool(tenant_id=current_tenant_id, provider_id=args["provider_id"]) - return {"result": "success"} + + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + service.delete_provider(tenant_id=current_tenant_id, provider_id=args["provider_id"]) + return {"result": "success"} + + +parser_auth = ( + reqparse.RequestParser() + .add_argument("provider_id", type=str, required=True, nullable=False, location="json") + .add_argument("authorization_code", type=str, required=False, nullable=True, location="json") +) @console_ns.route("/workspaces/current/tool-provider/mcp/auth") class ToolMCPAuthApi(Resource): + @api.expect(parser_auth) @setup_required @login_required @account_initialization_required def post(self): - parser = ( - reqparse.RequestParser() - .add_argument("provider_id", type=str, required=True, nullable=False, location="json") - .add_argument("authorization_code", type=str, required=False, nullable=True, location="json") - ) - args = parser.parse_args() + args = parser_auth.parse_args() provider_id = args["provider_id"] _, tenant_id = current_account_with_tenant() - provider = MCPToolManageService.get_mcp_provider_by_provider_id(provider_id, tenant_id) - if not provider: - raise ValueError("provider not found") - try: - with MCPClient( - provider.decrypted_server_url, - provider_id, - tenant_id, - authed=False, - authorization_code=args["authorization_code"], - for_list=True, - headers=provider.decrypted_headers, - timeout=provider.timeout, - sse_read_timeout=provider.sse_read_timeout, - ): - MCPToolManageService.update_mcp_provider_credentials( - mcp_provider=provider, - credentials=provider.decrypted_credentials, - authed=True, - ) - return {"result": "success"} - except MCPAuthError: - auth_provider = OAuthClientProvider(provider_id, tenant_id, for_list=True) - return auth(auth_provider, provider.decrypted_server_url, args["authorization_code"]) + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + db_provider = service.get_provider(provider_id=provider_id, tenant_id=tenant_id) + if not db_provider: + raise ValueError("provider not found") + + # Convert to entity + provider_entity = db_provider.to_entity() + server_url = provider_entity.decrypt_server_url() + headers = provider_entity.decrypt_authentication() + + # Try to connect without active transaction + try: + # Use MCPClientWithAuthRetry to handle authentication automatically + with MCPClient( + server_url=server_url, + headers=headers, + timeout=provider_entity.timeout, + sse_read_timeout=provider_entity.sse_read_timeout, + ): + # Update credentials in new transaction + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + service.update_provider_credentials( + provider_id=provider_id, + tenant_id=tenant_id, + credentials=provider_entity.credentials, + authed=True, + ) + return {"result": "success"} + except MCPAuthError as e: + try: + auth_result = auth(provider_entity, args.get("authorization_code")) + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + response = service.execute_auth_actions(auth_result) + return response + except MCPRefreshTokenError as e: + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + service.clear_provider_credentials(provider_id=provider_id, tenant_id=tenant_id) + raise ValueError(f"Failed to refresh token, please try to authorize again: {e}") from e except MCPError as e: - MCPToolManageService.update_mcp_provider_credentials( - mcp_provider=provider, - credentials={}, - authed=False, - ) + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + service.clear_provider_credentials(provider_id=provider_id, tenant_id=tenant_id) raise ValueError(f"Failed to connect to MCP server: {e}") from e @@ -1017,8 +1110,10 @@ class ToolMCPDetailApi(Resource): @account_initialization_required def get(self, provider_id): _, tenant_id = current_account_with_tenant() - provider = MCPToolManageService.get_mcp_provider_by_provider_id(provider_id, tenant_id) - return jsonable_encoder(ToolTransformService.mcp_provider_to_user_provider(provider, for_list=True)) + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + provider = service.get_provider(provider_id=provider_id, tenant_id=tenant_id) + return jsonable_encoder(ToolTransformService.mcp_provider_to_user_provider(provider, for_list=True)) @console_ns.route("/workspaces/current/tools/mcp") @@ -1029,9 +1124,12 @@ class ToolMCPListAllApi(Resource): def get(self): _, tenant_id = current_account_with_tenant() - tools = MCPToolManageService.retrieve_mcp_tools(tenant_id=tenant_id) + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + # Skip sensitive data decryption for list view to improve performance + tools = service.list_providers(tenant_id=tenant_id, include_sensitive=False) - return [tool.to_dict() for tool in tools] + return [tool.to_dict() for tool in tools] @console_ns.route("/workspaces/current/tool-provider/mcp/update/") @@ -1041,23 +1139,38 @@ class ToolMCPUpdateApi(Resource): @account_initialization_required def get(self, provider_id): _, tenant_id = current_account_with_tenant() - tools = MCPToolManageService.list_mcp_tool_from_remote_server( - tenant_id=tenant_id, - provider_id=provider_id, - ) - return jsonable_encoder(tools) + with Session(db.engine) as session, session.begin(): + service = MCPToolManageService(session=session) + tools = service.list_provider_tools( + tenant_id=tenant_id, + provider_id=provider_id, + ) + return jsonable_encoder(tools) + + +parser_cb = ( + reqparse.RequestParser() + .add_argument("code", type=str, required=True, nullable=False, location="args") + .add_argument("state", type=str, required=True, nullable=False, location="args") +) @console_ns.route("/mcp/oauth/callback") class ToolMCPCallbackApi(Resource): + @api.expect(parser_cb) def get(self): - parser = ( - reqparse.RequestParser() - .add_argument("code", type=str, required=True, nullable=False, location="args") - .add_argument("state", type=str, required=True, nullable=False, location="args") - ) - args = parser.parse_args() + args = parser_cb.parse_args() state_key = args["state"] authorization_code = args["code"] - handle_callback(state_key, authorization_code) + + # Create service instance for handle_callback + with Session(db.engine) as session, session.begin(): + mcp_service = MCPToolManageService(session=session) + # handle_callback now returns state data and tokens + state_data, tokens = handle_callback(state_key, authorization_code) + # Save tokens using the service layer + mcp_service.save_oauth_data( + state_data.provider_id, state_data.tenant_id, tokens.model_dump(), OAuthDataType.TOKENS + ) + return redirect(f"{dify_config.CONSOLE_WEB_URL}/oauth-callback") diff --git a/api/controllers/console/workspace/trigger_providers.py b/api/controllers/console/workspace/trigger_providers.py new file mode 100644 index 0000000000..bbbbe12fb0 --- /dev/null +++ b/api/controllers/console/workspace/trigger_providers.py @@ -0,0 +1,592 @@ +import logging + +from flask import make_response, redirect, request +from flask_restx import Resource, reqparse +from sqlalchemy.orm import Session +from werkzeug.exceptions import BadRequest, Forbidden + +from configs import dify_config +from controllers.console import api +from controllers.console.wraps import account_initialization_required, setup_required +from controllers.web.error import NotFoundError +from core.model_runtime.utils.encoders import jsonable_encoder +from core.plugin.entities.plugin_daemon import CredentialType +from core.plugin.impl.oauth import OAuthHandler +from core.trigger.entities.entities import SubscriptionBuilderUpdater +from core.trigger.trigger_manager import TriggerManager +from extensions.ext_database import db +from libs.login import current_user, login_required +from models.account import Account +from models.provider_ids import TriggerProviderID +from services.plugin.oauth_service import OAuthProxyService +from services.trigger.trigger_provider_service import TriggerProviderService +from services.trigger.trigger_subscription_builder_service import TriggerSubscriptionBuilderService +from services.trigger.trigger_subscription_operator_service import TriggerSubscriptionOperatorService + +logger = logging.getLogger(__name__) + + +class TriggerProviderIconApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self, provider): + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + + return TriggerManager.get_trigger_plugin_icon(tenant_id=user.current_tenant_id, provider_id=provider) + + +class TriggerProviderListApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self): + """List all trigger providers for the current tenant""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + return jsonable_encoder(TriggerProviderService.list_trigger_providers(user.current_tenant_id)) + + +class TriggerProviderInfoApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self, provider): + """Get info for a trigger provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + return jsonable_encoder( + TriggerProviderService.get_trigger_provider(user.current_tenant_id, TriggerProviderID(provider)) + ) + + +class TriggerSubscriptionListApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self, provider): + """List all trigger subscriptions for the current tenant's provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + try: + return jsonable_encoder( + TriggerProviderService.list_trigger_provider_subscriptions( + tenant_id=user.current_tenant_id, provider_id=TriggerProviderID(provider) + ) + ) + except ValueError as e: + return jsonable_encoder({"error": str(e)}), 404 + except Exception as e: + logger.exception("Error listing trigger providers", exc_info=e) + raise + + +class TriggerSubscriptionBuilderCreateApi(Resource): + @setup_required + @login_required + @account_initialization_required + def post(self, provider): + """Add a new subscription instance for a trigger provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + parser = reqparse.RequestParser() + parser.add_argument("credential_type", type=str, required=False, nullable=True, location="json") + args = parser.parse_args() + + try: + credential_type = CredentialType.of(args.get("credential_type") or CredentialType.UNAUTHORIZED.value) + subscription_builder = TriggerSubscriptionBuilderService.create_trigger_subscription_builder( + tenant_id=user.current_tenant_id, + user_id=user.id, + provider_id=TriggerProviderID(provider), + credential_type=credential_type, + ) + return jsonable_encoder({"subscription_builder": subscription_builder}) + except Exception as e: + logger.exception("Error adding provider credential", exc_info=e) + raise + + +class TriggerSubscriptionBuilderGetApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self, provider, subscription_builder_id): + """Get a subscription instance for a trigger provider""" + return jsonable_encoder( + TriggerSubscriptionBuilderService.get_subscription_builder_by_id(subscription_builder_id) + ) + + +class TriggerSubscriptionBuilderVerifyApi(Resource): + @setup_required + @login_required + @account_initialization_required + def post(self, provider, subscription_builder_id): + """Verify a subscription instance for a trigger provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + parser = reqparse.RequestParser() + # The credentials of the subscription builder + parser.add_argument("credentials", type=dict, required=False, nullable=True, location="json") + args = parser.parse_args() + + try: + # Use atomic update_and_verify to prevent race conditions + return TriggerSubscriptionBuilderService.update_and_verify_builder( + tenant_id=user.current_tenant_id, + user_id=user.id, + provider_id=TriggerProviderID(provider), + subscription_builder_id=subscription_builder_id, + subscription_builder_updater=SubscriptionBuilderUpdater( + credentials=args.get("credentials", None), + ), + ) + except Exception as e: + logger.exception("Error verifying provider credential", exc_info=e) + raise ValueError(str(e)) from e + + +class TriggerSubscriptionBuilderUpdateApi(Resource): + @setup_required + @login_required + @account_initialization_required + def post(self, provider, subscription_builder_id): + """Update a subscription instance for a trigger provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + + parser = reqparse.RequestParser() + # The name of the subscription builder + parser.add_argument("name", type=str, required=False, nullable=True, location="json") + # The parameters of the subscription builder + parser.add_argument("parameters", type=dict, required=False, nullable=True, location="json") + # The properties of the subscription builder + parser.add_argument("properties", type=dict, required=False, nullable=True, location="json") + # The credentials of the subscription builder + parser.add_argument("credentials", type=dict, required=False, nullable=True, location="json") + args = parser.parse_args() + try: + return jsonable_encoder( + TriggerSubscriptionBuilderService.update_trigger_subscription_builder( + tenant_id=user.current_tenant_id, + provider_id=TriggerProviderID(provider), + subscription_builder_id=subscription_builder_id, + subscription_builder_updater=SubscriptionBuilderUpdater( + name=args.get("name", None), + parameters=args.get("parameters", None), + properties=args.get("properties", None), + credentials=args.get("credentials", None), + ), + ) + ) + except Exception as e: + logger.exception("Error updating provider credential", exc_info=e) + raise + + +class TriggerSubscriptionBuilderLogsApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self, provider, subscription_builder_id): + """Get the request logs for a subscription instance for a trigger provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + + try: + logs = TriggerSubscriptionBuilderService.list_logs(subscription_builder_id) + return jsonable_encoder({"logs": [log.model_dump(mode="json") for log in logs]}) + except Exception as e: + logger.exception("Error getting request logs for subscription builder", exc_info=e) + raise + + +class TriggerSubscriptionBuilderBuildApi(Resource): + @setup_required + @login_required + @account_initialization_required + def post(self, provider, subscription_builder_id): + """Build a subscription instance for a trigger provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + parser = reqparse.RequestParser() + # The name of the subscription builder + parser.add_argument("name", type=str, required=False, nullable=True, location="json") + # The parameters of the subscription builder + parser.add_argument("parameters", type=dict, required=False, nullable=True, location="json") + # The properties of the subscription builder + parser.add_argument("properties", type=dict, required=False, nullable=True, location="json") + # The credentials of the subscription builder + parser.add_argument("credentials", type=dict, required=False, nullable=True, location="json") + args = parser.parse_args() + try: + # Use atomic update_and_build to prevent race conditions + TriggerSubscriptionBuilderService.update_and_build_builder( + tenant_id=user.current_tenant_id, + user_id=user.id, + provider_id=TriggerProviderID(provider), + subscription_builder_id=subscription_builder_id, + subscription_builder_updater=SubscriptionBuilderUpdater( + name=args.get("name", None), + parameters=args.get("parameters", None), + properties=args.get("properties", None), + ), + ) + return 200 + except Exception as e: + logger.exception("Error building provider credential", exc_info=e) + raise ValueError(str(e)) from e + + +class TriggerSubscriptionDeleteApi(Resource): + @setup_required + @login_required + @account_initialization_required + def post(self, subscription_id: str): + """Delete a subscription instance""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + try: + with Session(db.engine) as session: + # Delete trigger provider subscription + TriggerProviderService.delete_trigger_provider( + session=session, + tenant_id=user.current_tenant_id, + subscription_id=subscription_id, + ) + # Delete plugin triggers + TriggerSubscriptionOperatorService.delete_plugin_trigger_by_subscription( + session=session, + tenant_id=user.current_tenant_id, + subscription_id=subscription_id, + ) + session.commit() + return {"result": "success"} + except ValueError as e: + raise BadRequest(str(e)) + except Exception as e: + logger.exception("Error deleting provider credential", exc_info=e) + raise + + +class TriggerOAuthAuthorizeApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self, provider): + """Initiate OAuth authorization flow for a trigger provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + + try: + provider_id = TriggerProviderID(provider) + plugin_id = provider_id.plugin_id + provider_name = provider_id.provider_name + tenant_id = user.current_tenant_id + + # Get OAuth client configuration + oauth_client_params = TriggerProviderService.get_oauth_client( + tenant_id=tenant_id, + provider_id=provider_id, + ) + + if oauth_client_params is None: + raise NotFoundError("No OAuth client configuration found for this trigger provider") + + # Create subscription builder + subscription_builder = TriggerSubscriptionBuilderService.create_trigger_subscription_builder( + tenant_id=tenant_id, + user_id=user.id, + provider_id=provider_id, + credential_type=CredentialType.OAUTH2, + ) + + # Create OAuth handler and proxy context + oauth_handler = OAuthHandler() + context_id = OAuthProxyService.create_proxy_context( + user_id=user.id, + tenant_id=tenant_id, + plugin_id=plugin_id, + provider=provider_name, + extra_data={ + "subscription_builder_id": subscription_builder.id, + }, + ) + + # Build redirect URI for callback + redirect_uri = f"{dify_config.CONSOLE_API_URL}/console/api/oauth/plugin/{provider}/trigger/callback" + + # Get authorization URL + authorization_url_response = oauth_handler.get_authorization_url( + tenant_id=tenant_id, + user_id=user.id, + plugin_id=plugin_id, + provider=provider_name, + redirect_uri=redirect_uri, + system_credentials=oauth_client_params, + ) + + # Create response with cookie + response = make_response( + jsonable_encoder( + { + "authorization_url": authorization_url_response.authorization_url, + "subscription_builder_id": subscription_builder.id, + "subscription_builder": subscription_builder, + } + ) + ) + response.set_cookie( + "context_id", + context_id, + httponly=True, + samesite="Lax", + max_age=OAuthProxyService.__MAX_AGE__, + ) + + return response + + except Exception as e: + logger.exception("Error initiating OAuth flow", exc_info=e) + raise + + +class TriggerOAuthCallbackApi(Resource): + @setup_required + def get(self, provider): + """Handle OAuth callback for trigger provider""" + context_id = request.cookies.get("context_id") + if not context_id: + raise Forbidden("context_id not found") + + # Use and validate proxy context + context = OAuthProxyService.use_proxy_context(context_id) + if context is None: + raise Forbidden("Invalid context_id") + + # Parse provider ID + provider_id = TriggerProviderID(provider) + plugin_id = provider_id.plugin_id + provider_name = provider_id.provider_name + user_id = context.get("user_id") + tenant_id = context.get("tenant_id") + subscription_builder_id = context.get("subscription_builder_id") + + # Get OAuth client configuration + oauth_client_params = TriggerProviderService.get_oauth_client( + tenant_id=tenant_id, + provider_id=provider_id, + ) + + if oauth_client_params is None: + raise Forbidden("No OAuth client configuration found for this trigger provider") + + # Get OAuth credentials from callback + oauth_handler = OAuthHandler() + redirect_uri = f"{dify_config.CONSOLE_API_URL}/console/api/oauth/plugin/{provider}/trigger/callback" + + credentials_response = oauth_handler.get_credentials( + tenant_id=tenant_id, + user_id=user_id, + plugin_id=plugin_id, + provider=provider_name, + redirect_uri=redirect_uri, + system_credentials=oauth_client_params, + request=request, + ) + + credentials = credentials_response.credentials + expires_at = credentials_response.expires_at + + if not credentials: + raise ValueError("Failed to get OAuth credentials from the provider.") + + # Update subscription builder + TriggerSubscriptionBuilderService.update_trigger_subscription_builder( + tenant_id=tenant_id, + provider_id=provider_id, + subscription_builder_id=subscription_builder_id, + subscription_builder_updater=SubscriptionBuilderUpdater( + credentials=credentials, + credential_expires_at=expires_at, + ), + ) + # Redirect to OAuth callback page + return redirect(f"{dify_config.CONSOLE_WEB_URL}/oauth-callback") + + +class TriggerOAuthClientManageApi(Resource): + @setup_required + @login_required + @account_initialization_required + def get(self, provider): + """Get OAuth client configuration for a provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + try: + provider_id = TriggerProviderID(provider) + + # Get custom OAuth client params if exists + custom_params = TriggerProviderService.get_custom_oauth_client_params( + tenant_id=user.current_tenant_id, + provider_id=provider_id, + ) + + # Check if custom client is enabled + is_custom_enabled = TriggerProviderService.is_oauth_custom_client_enabled( + tenant_id=user.current_tenant_id, + provider_id=provider_id, + ) + system_client_exists = TriggerProviderService.is_oauth_system_client_exists( + tenant_id=user.current_tenant_id, + provider_id=provider_id, + ) + provider_controller = TriggerManager.get_trigger_provider(user.current_tenant_id, provider_id) + redirect_uri = f"{dify_config.CONSOLE_API_URL}/console/api/oauth/plugin/{provider}/trigger/callback" + return jsonable_encoder( + { + "configured": bool(custom_params or system_client_exists), + "system_configured": system_client_exists, + "custom_configured": bool(custom_params), + "oauth_client_schema": provider_controller.get_oauth_client_schema(), + "custom_enabled": is_custom_enabled, + "redirect_uri": redirect_uri, + "params": custom_params or {}, + } + ) + + except Exception as e: + logger.exception("Error getting OAuth client", exc_info=e) + raise + + @setup_required + @login_required + @account_initialization_required + def post(self, provider): + """Configure custom OAuth client for a provider""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + parser = reqparse.RequestParser() + parser.add_argument("client_params", type=dict, required=False, nullable=True, location="json") + parser.add_argument("enabled", type=bool, required=False, nullable=True, location="json") + args = parser.parse_args() + + try: + provider_id = TriggerProviderID(provider) + return TriggerProviderService.save_custom_oauth_client_params( + tenant_id=user.current_tenant_id, + provider_id=provider_id, + client_params=args.get("client_params"), + enabled=args.get("enabled"), + ) + + except ValueError as e: + raise BadRequest(str(e)) + except Exception as e: + logger.exception("Error configuring OAuth client", exc_info=e) + raise + + @setup_required + @login_required + @account_initialization_required + def delete(self, provider): + """Remove custom OAuth client configuration""" + user = current_user + assert isinstance(user, Account) + assert user.current_tenant_id is not None + if not user.is_admin_or_owner: + raise Forbidden() + + try: + provider_id = TriggerProviderID(provider) + + return TriggerProviderService.delete_custom_oauth_client_params( + tenant_id=user.current_tenant_id, + provider_id=provider_id, + ) + except ValueError as e: + raise BadRequest(str(e)) + except Exception as e: + logger.exception("Error removing OAuth client", exc_info=e) + raise + + +# Trigger Subscription +api.add_resource(TriggerProviderIconApi, "/workspaces/current/trigger-provider//icon") +api.add_resource(TriggerProviderListApi, "/workspaces/current/triggers") +api.add_resource(TriggerProviderInfoApi, "/workspaces/current/trigger-provider//info") +api.add_resource(TriggerSubscriptionListApi, "/workspaces/current/trigger-provider//subscriptions/list") +api.add_resource( + TriggerSubscriptionDeleteApi, + "/workspaces/current/trigger-provider//subscriptions/delete", +) + +# Trigger Subscription Builder +api.add_resource( + TriggerSubscriptionBuilderCreateApi, + "/workspaces/current/trigger-provider//subscriptions/builder/create", +) +api.add_resource( + TriggerSubscriptionBuilderGetApi, + "/workspaces/current/trigger-provider//subscriptions/builder/", +) +api.add_resource( + TriggerSubscriptionBuilderUpdateApi, + "/workspaces/current/trigger-provider//subscriptions/builder/update/", +) +api.add_resource( + TriggerSubscriptionBuilderVerifyApi, + "/workspaces/current/trigger-provider//subscriptions/builder/verify/", +) +api.add_resource( + TriggerSubscriptionBuilderBuildApi, + "/workspaces/current/trigger-provider//subscriptions/builder/build/", +) +api.add_resource( + TriggerSubscriptionBuilderLogsApi, + "/workspaces/current/trigger-provider//subscriptions/builder/logs/", +) + + +# OAuth +api.add_resource( + TriggerOAuthAuthorizeApi, "/workspaces/current/trigger-provider//subscriptions/oauth/authorize" +) +api.add_resource(TriggerOAuthCallbackApi, "/oauth/plugin//trigger/callback") +api.add_resource(TriggerOAuthClientManageApi, "/workspaces/current/trigger-provider//oauth/client") diff --git a/api/controllers/console/workspace/workspace.py b/api/controllers/console/workspace/workspace.py index f9856df9ea..f10c30db2e 100644 --- a/api/controllers/console/workspace/workspace.py +++ b/api/controllers/console/workspace/workspace.py @@ -13,7 +13,7 @@ from controllers.common.errors import ( TooManyFilesError, UnsupportedFileTypeError, ) -from controllers.console import console_ns +from controllers.console import api, console_ns from controllers.console.admin import admin_required from controllers.console.error import AccountNotLinkTenantError from controllers.console.wraps import ( @@ -21,6 +21,7 @@ from controllers.console.wraps import ( cloud_edition_billing_resource_check, setup_required, ) +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from libs.helper import TimestampField from libs.login import current_account_with_tenant, login_required @@ -83,7 +84,7 @@ class TenantListApi(Resource): "name": tenant.name, "status": tenant.status, "created_at": tenant.created_at, - "plan": features.billing.subscription.plan if features.billing.enabled else "sandbox", + "plan": features.billing.subscription.plan if features.billing.enabled else CloudPlan.SANDBOX, "current": tenant.id == current_tenant_id if current_tenant_id else False, } @@ -149,15 +150,18 @@ class TenantApi(Resource): return WorkspaceService.get_tenant_info(tenant), 200 +parser_switch = reqparse.RequestParser().add_argument("tenant_id", type=str, required=True, location="json") + + @console_ns.route("/workspaces/switch") class SwitchWorkspaceApi(Resource): + @api.expect(parser_switch) @setup_required @login_required @account_initialization_required def post(self): current_user, _ = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("tenant_id", type=str, required=True, location="json") - args = parser.parse_args() + args = parser_switch.parse_args() # check if tenant_id is valid, 403 if not try: @@ -241,16 +245,19 @@ class WebappLogoWorkspaceApi(Resource): return {"id": upload_file.id}, 201 +parser_info = reqparse.RequestParser().add_argument("name", type=str, required=True, location="json") + + @console_ns.route("/workspaces/info") class WorkspaceInfoApi(Resource): + @api.expect(parser_info) @setup_required @login_required @account_initialization_required # Change workspace name def post(self): _, current_tenant_id = current_account_with_tenant() - parser = reqparse.RequestParser().add_argument("name", type=str, required=True, location="json") - args = parser.parse_args() + args = parser_info.parse_args() if not current_tenant_id: raise ValueError("No current tenant") diff --git a/api/controllers/console/wraps.py b/api/controllers/console/wraps.py index 8572a6dc9b..9b485544db 100644 --- a/api/controllers/console/wraps.py +++ b/api/controllers/console/wraps.py @@ -10,6 +10,7 @@ from flask import abort, request from configs import dify_config from controllers.console.workspace.error import AccountNotInitializedError +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from extensions.ext_redis import redis_client from libs.login import current_account_with_tenant @@ -133,7 +134,7 @@ def cloud_edition_billing_knowledge_limit_check(resource: str): features = FeatureService.get_features(current_tenant_id) if features.billing.enabled: if resource == "add_segment": - if features.billing.subscription.plan == "sandbox": + if features.billing.subscription.plan == CloudPlan.SANDBOX: abort( 403, "To unlock this feature and elevate your Dify experience, please upgrade to a paid plan.", diff --git a/api/controllers/files/image_preview.py b/api/controllers/files/image_preview.py index 3db82456d5..d320855f29 100644 --- a/api/controllers/files/image_preview.py +++ b/api/controllers/files/image_preview.py @@ -14,10 +14,25 @@ from services.file_service import FileService @files_ns.route("//image-preview") class ImagePreviewApi(Resource): - """ - Deprecated - """ + """Deprecated endpoint for retrieving image previews.""" + @files_ns.doc("get_image_preview") + @files_ns.doc(description="Retrieve a signed image preview for a file") + @files_ns.doc( + params={ + "file_id": "ID of the file to preview", + "timestamp": "Unix timestamp used in the signature", + "nonce": "Random string used in the signature", + "sign": "HMAC signature verifying the request", + } + ) + @files_ns.doc( + responses={ + 200: "Image preview returned successfully", + 400: "Missing or invalid signature parameters", + 415: "Unsupported file type", + } + ) def get(self, file_id): file_id = str(file_id) @@ -43,6 +58,25 @@ class ImagePreviewApi(Resource): @files_ns.route("//file-preview") class FilePreviewApi(Resource): + @files_ns.doc("get_file_preview") + @files_ns.doc(description="Download a file preview or attachment using signed parameters") + @files_ns.doc( + params={ + "file_id": "ID of the file to preview", + "timestamp": "Unix timestamp used in the signature", + "nonce": "Random string used in the signature", + "sign": "HMAC signature verifying the request", + "as_attachment": "Whether to download the file as an attachment", + } + ) + @files_ns.doc( + responses={ + 200: "File stream returned successfully", + 400: "Missing or invalid signature parameters", + 404: "File not found", + 415: "Unsupported file type", + } + ) def get(self, file_id): file_id = str(file_id) @@ -101,6 +135,20 @@ class FilePreviewApi(Resource): @files_ns.route("/workspaces//webapp-logo") class WorkspaceWebappLogoApi(Resource): + @files_ns.doc("get_workspace_webapp_logo") + @files_ns.doc(description="Fetch the custom webapp logo for a workspace") + @files_ns.doc( + params={ + "workspace_id": "Workspace identifier", + } + ) + @files_ns.doc( + responses={ + 200: "Logo returned successfully", + 404: "Webapp logo not configured", + 415: "Unsupported file type", + } + ) def get(self, workspace_id): workspace_id = str(workspace_id) diff --git a/api/controllers/files/tool_files.py b/api/controllers/files/tool_files.py index dec5a4a1b2..ecaeb85821 100644 --- a/api/controllers/files/tool_files.py +++ b/api/controllers/files/tool_files.py @@ -13,6 +13,26 @@ from extensions.ext_database import db as global_db @files_ns.route("/tools/.") class ToolFileApi(Resource): + @files_ns.doc("get_tool_file") + @files_ns.doc(description="Download a tool file by ID using signed parameters") + @files_ns.doc( + params={ + "file_id": "Tool file identifier", + "extension": "Expected file extension", + "timestamp": "Unix timestamp used in the signature", + "nonce": "Random string used in the signature", + "sign": "HMAC signature verifying the request", + "as_attachment": "Whether to download the file as an attachment", + } + ) + @files_ns.doc( + responses={ + 200: "Tool file stream returned successfully", + 403: "Forbidden - invalid signature", + 404: "File not found", + 415: "Unsupported file type", + } + ) def get(self, file_id, extension): file_id = str(file_id) diff --git a/api/controllers/service_api/dataset/document.py b/api/controllers/service_api/dataset/document.py index 893cd7c923..358605e8a8 100644 --- a/api/controllers/service_api/dataset/document.py +++ b/api/controllers/service_api/dataset/document.py @@ -592,7 +592,7 @@ class DocumentApi(DatasetApiResource): "name": document.name, "created_from": document.created_from, "created_by": document.created_by, - "created_at": document.created_at.timestamp(), + "created_at": int(document.created_at.timestamp()), "tokens": document.tokens, "indexing_status": document.indexing_status, "completed_at": int(document.completed_at.timestamp()) if document.completed_at else None, @@ -625,7 +625,7 @@ class DocumentApi(DatasetApiResource): "name": document.name, "created_from": document.created_from, "created_by": document.created_by, - "created_at": document.created_at.timestamp(), + "created_at": int(document.created_at.timestamp()), "tokens": document.tokens, "indexing_status": document.indexing_status, "completed_at": int(document.completed_at.timestamp()) if document.completed_at else None, diff --git a/api/controllers/service_api/dataset/segment.py b/api/controllers/service_api/dataset/segment.py index 81abd19fed..9ca500b044 100644 --- a/api/controllers/service_api/dataset/segment.py +++ b/api/controllers/service_api/dataset/segment.py @@ -2,6 +2,7 @@ from flask import request from flask_restx import marshal, reqparse from werkzeug.exceptions import NotFound +from configs import dify_config from controllers.service_api import service_api_ns from controllers.service_api.app.error import ProviderNotInitializeError from controllers.service_api.wraps import ( @@ -107,6 +108,10 @@ class SegmentApi(DatasetApiResource): # validate args args = segment_create_parser.parse_args() if args["segments"] is not None: + segments_limit = dify_config.DATASET_MAX_SEGMENTS_PER_REQUEST + if segments_limit > 0 and len(args["segments"]) > segments_limit: + raise ValueError(f"Exceeded maximum segments limit of {segments_limit}.") + for args_item in args["segments"]: SegmentService.segment_create_args_validate(args_item, document) segments = SegmentService.multi_create_segment(args["segments"], document, dataset) diff --git a/api/controllers/service_api/wraps.py b/api/controllers/service_api/wraps.py index 638ab528f3..c07e18c686 100644 --- a/api/controllers/service_api/wraps.py +++ b/api/controllers/service_api/wraps.py @@ -13,13 +13,15 @@ from sqlalchemy import select, update from sqlalchemy.orm import Session from werkzeug.exceptions import Forbidden, NotFound, Unauthorized +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from extensions.ext_redis import redis_client from libs.datetime_utils import naive_utc_now from libs.login import current_user from models import Account, Tenant, TenantAccountJoin, TenantStatus from models.dataset import Dataset, RateLimitLog -from models.model import ApiToken, App, DefaultEndUserSessionID, EndUser +from models.model import ApiToken, App +from services.end_user_service import EndUserService from services.feature_service import FeatureService P = ParamSpec("P") @@ -66,6 +68,7 @@ def validate_app_token(view: Callable[P, R] | None = None, *, fetch_user_arg: Fe kwargs["app_model"] = app_model + # If caller needs end-user context, attach EndUser to current_user if fetch_user_arg: if fetch_user_arg.fetch_from == WhereisUserArg.QUERY: user_id = request.args.get("user") @@ -74,7 +77,6 @@ def validate_app_token(view: Callable[P, R] | None = None, *, fetch_user_arg: Fe elif fetch_user_arg.fetch_from == WhereisUserArg.FORM: user_id = request.form.get("user") else: - # use default-user user_id = None if not user_id and fetch_user_arg.required: @@ -83,12 +85,34 @@ def validate_app_token(view: Callable[P, R] | None = None, *, fetch_user_arg: Fe if user_id: user_id = str(user_id) - end_user = create_or_update_end_user_for_user_id(app_model, user_id) + end_user = EndUserService.get_or_create_end_user(app_model, user_id) kwargs["end_user"] = end_user # Set EndUser as current logged-in user for flask_login.current_user current_app.login_manager._update_request_context_with_user(end_user) # type: ignore user_logged_in.send(current_app._get_current_object(), user=end_user) # type: ignore + else: + # For service API without end-user context, ensure an Account is logged in + # so services relying on current_account_with_tenant() work correctly. + tenant_owner_info = ( + db.session.query(Tenant, Account) + .join(TenantAccountJoin, Tenant.id == TenantAccountJoin.tenant_id) + .join(Account, TenantAccountJoin.account_id == Account.id) + .where( + Tenant.id == app_model.tenant_id, + TenantAccountJoin.role == "owner", + Tenant.status == TenantStatus.NORMAL, + ) + .one_or_none() + ) + + if tenant_owner_info: + tenant_model, account = tenant_owner_info + account.current_tenant = tenant_model + current_app.login_manager._update_request_context_with_user(account) # type: ignore + user_logged_in.send(current_app._get_current_object(), user=current_user) # type: ignore + else: + raise Unauthorized("Tenant owner account not found or tenant is not active.") return view_func(*args, **kwargs) @@ -138,7 +162,7 @@ def cloud_edition_billing_knowledge_limit_check(resource: str, api_token_type: s features = FeatureService.get_features(api_token.tenant_id) if features.billing.enabled: if resource == "add_segment": - if features.billing.subscription.plan == "sandbox": + if features.billing.subscription.plan == CloudPlan.SANDBOX: raise Forbidden( "To unlock this feature and elevate your Dify experience, please upgrade to a paid plan." ) @@ -308,39 +332,6 @@ def validate_and_get_api_token(scope: str | None = None): return api_token -def create_or_update_end_user_for_user_id(app_model: App, user_id: str | None = None) -> EndUser: - """ - Create or update session terminal based on user ID. - """ - if not user_id: - user_id = DefaultEndUserSessionID.DEFAULT_SESSION_ID - - with Session(db.engine, expire_on_commit=False) as session: - end_user = ( - session.query(EndUser) - .where( - EndUser.tenant_id == app_model.tenant_id, - EndUser.app_id == app_model.id, - EndUser.session_id == user_id, - EndUser.type == "service_api", - ) - .first() - ) - - if end_user is None: - end_user = EndUser( - tenant_id=app_model.tenant_id, - app_id=app_model.id, - type="service_api", - is_anonymous=user_id == DefaultEndUserSessionID.DEFAULT_SESSION_ID, - session_id=user_id, - ) - session.add(end_user) - session.commit() - - return end_user - - class DatasetApiResource(Resource): method_decorators = [validate_dataset_token] diff --git a/api/controllers/trigger/__init__.py b/api/controllers/trigger/__init__.py new file mode 100644 index 0000000000..4f584dc4f6 --- /dev/null +++ b/api/controllers/trigger/__init__.py @@ -0,0 +1,12 @@ +from flask import Blueprint + +# Create trigger blueprint +bp = Blueprint("trigger", __name__, url_prefix="/triggers") + +# Import routes after blueprint creation to avoid circular imports +from . import trigger, webhook + +__all__ = [ + "trigger", + "webhook", +] diff --git a/api/controllers/trigger/trigger.py b/api/controllers/trigger/trigger.py new file mode 100644 index 0000000000..e69b22d880 --- /dev/null +++ b/api/controllers/trigger/trigger.py @@ -0,0 +1,43 @@ +import logging +import re + +from flask import jsonify, request +from werkzeug.exceptions import NotFound + +from controllers.trigger import bp +from services.trigger.trigger_service import TriggerService +from services.trigger.trigger_subscription_builder_service import TriggerSubscriptionBuilderService + +logger = logging.getLogger(__name__) + +UUID_PATTERN = r"^[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$" +UUID_MATCHER = re.compile(UUID_PATTERN) + + +@bp.route("/plugin/", methods=["GET", "POST", "PUT", "PATCH", "DELETE", "HEAD", "OPTIONS"]) +def trigger_endpoint(endpoint_id: str): + """ + Handle endpoint trigger calls. + """ + # endpoint_id must be UUID + if not UUID_MATCHER.match(endpoint_id): + raise NotFound("Invalid endpoint ID") + handling_chain = [ + TriggerService.process_endpoint, + TriggerSubscriptionBuilderService.process_builder_validation_endpoint, + ] + response = None + try: + for handler in handling_chain: + response = handler(endpoint_id, request) + if response: + break + if not response: + logger.error("Endpoint not found for {endpoint_id}") + return jsonify({"error": "Endpoint not found"}), 404 + return response + except ValueError as e: + return jsonify({"error": "Endpoint processing failed", "message": str(e)}), 400 + except Exception: + logger.exception("Webhook processing failed for {endpoint_id}") + return jsonify({"error": "Internal server error"}), 500 diff --git a/api/controllers/trigger/webhook.py b/api/controllers/trigger/webhook.py new file mode 100644 index 0000000000..cec5c3d8ae --- /dev/null +++ b/api/controllers/trigger/webhook.py @@ -0,0 +1,105 @@ +import logging +import time + +from flask import jsonify +from werkzeug.exceptions import NotFound, RequestEntityTooLarge + +from controllers.trigger import bp +from core.trigger.debug.event_bus import TriggerDebugEventBus +from core.trigger.debug.events import WebhookDebugEvent, build_webhook_pool_key +from services.trigger.webhook_service import WebhookService + +logger = logging.getLogger(__name__) + + +def _prepare_webhook_execution(webhook_id: str, is_debug: bool = False): + """Fetch trigger context, extract request data, and validate payload using unified processing. + + Args: + webhook_id: The webhook ID to process + is_debug: If True, skip status validation for debug mode + """ + webhook_trigger, workflow, node_config = WebhookService.get_webhook_trigger_and_workflow( + webhook_id, is_debug=is_debug + ) + + try: + # Use new unified extraction and validation + webhook_data = WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + return webhook_trigger, workflow, node_config, webhook_data, None + except ValueError as e: + # Fall back to raw extraction for error reporting + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + return webhook_trigger, workflow, node_config, webhook_data, str(e) + + +@bp.route("/webhook/", methods=["GET", "POST", "PUT", "PATCH", "DELETE", "HEAD", "OPTIONS"]) +def handle_webhook(webhook_id: str): + """ + Handle webhook trigger calls. + + This endpoint receives webhook calls and processes them according to the + configured webhook trigger settings. + """ + try: + webhook_trigger, workflow, node_config, webhook_data, error = _prepare_webhook_execution(webhook_id) + if error: + return jsonify({"error": "Bad Request", "message": error}), 400 + + # Process webhook call (send to Celery) + WebhookService.trigger_workflow_execution(webhook_trigger, webhook_data, workflow) + + # Return configured response + response_data, status_code = WebhookService.generate_webhook_response(node_config) + return jsonify(response_data), status_code + + except ValueError as e: + raise NotFound(str(e)) + except RequestEntityTooLarge: + raise + except Exception as e: + logger.exception("Webhook processing failed for %s", webhook_id) + return jsonify({"error": "Internal server error", "message": str(e)}), 500 + + +@bp.route("/webhook-debug/", methods=["GET", "POST", "PUT", "PATCH", "DELETE", "HEAD", "OPTIONS"]) +def handle_webhook_debug(webhook_id: str): + """Handle webhook debug calls without triggering production workflow execution.""" + try: + webhook_trigger, _, node_config, webhook_data, error = _prepare_webhook_execution(webhook_id, is_debug=True) + if error: + return jsonify({"error": "Bad Request", "message": error}), 400 + + workflow_inputs = WebhookService.build_workflow_inputs(webhook_data) + + # Generate pool key and dispatch debug event + pool_key: str = build_webhook_pool_key( + tenant_id=webhook_trigger.tenant_id, + app_id=webhook_trigger.app_id, + node_id=webhook_trigger.node_id, + ) + event = WebhookDebugEvent( + request_id=f"webhook_debug_{webhook_trigger.webhook_id}_{int(time.time() * 1000)}", + timestamp=int(time.time()), + node_id=webhook_trigger.node_id, + payload={ + "inputs": workflow_inputs, + "webhook_data": webhook_data, + "method": webhook_data.get("method"), + }, + ) + TriggerDebugEventBus.dispatch( + tenant_id=webhook_trigger.tenant_id, + event=event, + pool_key=pool_key, + ) + response_data, status_code = WebhookService.generate_webhook_response(node_config) + return jsonify(response_data), status_code + + except ValueError as e: + raise NotFound(str(e)) + except RequestEntityTooLarge: + raise + except Exception as e: + logger.exception("Webhook debug processing failed for %s", webhook_id) + return jsonify({"error": "Internal server error", "message": "An internal error has occurred."}), 500 diff --git a/api/controllers/web/audio.py b/api/controllers/web/audio.py index 3103851088..b9fef48c4d 100644 --- a/api/controllers/web/audio.py +++ b/api/controllers/web/audio.py @@ -88,12 +88,6 @@ class AudioApi(WebApiResource): @web_ns.route("/text-to-audio") class TextApi(WebApiResource): - text_to_audio_response_fields = { - "audio_url": fields.String, - "duration": fields.Float, - } - - @marshal_with(text_to_audio_response_fields) @web_ns.doc("Text to Audio") @web_ns.doc(description="Convert text to audio using text-to-speech service.") @web_ns.doc( diff --git a/api/controllers/web/login.py b/api/controllers/web/login.py index f213fd8c90..244ef47982 100644 --- a/api/controllers/web/login.py +++ b/api/controllers/web/login.py @@ -17,8 +17,8 @@ from libs.helper import email from libs.passport import PassportService from libs.password import valid_password from libs.token import ( - clear_access_token_from_cookie, - extract_access_token, + clear_webapp_access_token_from_cookie, + extract_webapp_access_token, ) from services.account_service import AccountService from services.app_service import AppService @@ -81,7 +81,7 @@ class LoginStatusApi(Resource): ) def get(self): app_code = request.args.get("app_code") - token = extract_access_token(request) + token = extract_webapp_access_token(request) if not app_code: return { "logged_in": bool(token), @@ -128,7 +128,7 @@ class LogoutApi(Resource): response = make_response({"result": "success"}) # enterprise SSO sets same site to None in https deployment # so we need to logout by calling api - clear_access_token_from_cookie(response, samesite="None") + clear_webapp_access_token_from_cookie(response, samesite="None") return response diff --git a/api/controllers/web/passport.py b/api/controllers/web/passport.py index a344777783..6a2e0b65fb 100644 --- a/api/controllers/web/passport.py +++ b/api/controllers/web/passport.py @@ -12,7 +12,7 @@ from controllers.web import web_ns from controllers.web.error import WebAppAuthRequiredError from extensions.ext_database import db from libs.passport import PassportService -from libs.token import extract_access_token +from libs.token import extract_webapp_access_token from models.model import App, EndUser, Site from services.feature_service import FeatureService from services.webapp_auth_service import WebAppAuthService, WebAppAuthType @@ -35,7 +35,7 @@ class PassportResource(Resource): system_features = FeatureService.get_system_features() app_code = request.headers.get(HEADER_NAME_APP_CODE) user_id = request.args.get("user_id") - access_token = extract_access_token(request) + access_token = extract_webapp_access_token(request) if app_code is None: raise Unauthorized("X-App-Code header is missing.") if system_features.webapp_auth.enabled: diff --git a/api/core/app/apps/advanced_chat/app_runner.py b/api/core/app/apps/advanced_chat/app_runner.py index 587c663482..c029e00553 100644 --- a/api/core/app/apps/advanced_chat/app_runner.py +++ b/api/core/app/apps/advanced_chat/app_runner.py @@ -1,6 +1,6 @@ import logging import time -from collections.abc import Mapping +from collections.abc import Mapping, Sequence from typing import Any, cast from sqlalchemy import select @@ -25,6 +25,7 @@ from core.moderation.input_moderation import InputModeration from core.variables.variables import VariableUnion from core.workflow.enums import WorkflowType from core.workflow.graph_engine.command_channels.redis_channel import RedisChannel +from core.workflow.graph_engine.layers.base import GraphEngineLayer from core.workflow.graph_engine.layers.persistence import PersistenceWorkflowInfo, WorkflowPersistenceLayer from core.workflow.repositories.workflow_execution_repository import WorkflowExecutionRepository from core.workflow.repositories.workflow_node_execution_repository import WorkflowNodeExecutionRepository @@ -61,11 +62,13 @@ class AdvancedChatAppRunner(WorkflowBasedAppRunner): app: App, workflow_execution_repository: WorkflowExecutionRepository, workflow_node_execution_repository: WorkflowNodeExecutionRepository, + graph_engine_layers: Sequence[GraphEngineLayer] = (), ): super().__init__( queue_manager=queue_manager, variable_loader=variable_loader, app_id=application_generate_entity.app_config.app_id, + graph_engine_layers=graph_engine_layers, ) self.application_generate_entity = application_generate_entity self.conversation = conversation @@ -195,6 +198,8 @@ class AdvancedChatAppRunner(WorkflowBasedAppRunner): ) workflow_entry.graph_engine.layer(persistence_layer) + for layer in self._graph_engine_layers: + workflow_entry.graph_engine.layer(layer) generator = workflow_entry.run() diff --git a/api/core/app/apps/advanced_chat/generate_task_pipeline.py b/api/core/app/apps/advanced_chat/generate_task_pipeline.py index 8c0102d9bd..01c377956b 100644 --- a/api/core/app/apps/advanced_chat/generate_task_pipeline.py +++ b/api/core/app/apps/advanced_chat/generate_task_pipeline.py @@ -1,3 +1,4 @@ +import json import logging import re import time @@ -60,6 +61,7 @@ from core.app.task_pipeline.based_generate_task_pipeline import BasedGenerateTas from core.app.task_pipeline.message_cycle_manager import MessageCycleManager from core.base.tts import AppGeneratorTTSPublisher, AudioTrunk from core.model_runtime.entities.llm_entities import LLMUsage +from core.model_runtime.utils.encoders import jsonable_encoder from core.ops.ops_trace_manager import TraceQueueManager from core.workflow.enums import WorkflowExecutionStatus from core.workflow.nodes import NodeType @@ -391,6 +393,14 @@ class AdvancedChatAppGenerateTaskPipeline(GraphRuntimeStateSupport): if should_direct_answer: return + current_time = time.perf_counter() + if self._task_state.first_token_time is None and delta_text.strip(): + self._task_state.first_token_time = current_time + self._task_state.is_streaming_response = True + + if delta_text.strip(): + self._task_state.last_token_time = current_time + # Only publish tts message at text chunk streaming if tts_publisher and queue_message: tts_publisher.publish(queue_message) @@ -772,7 +782,33 @@ class AdvancedChatAppGenerateTaskPipeline(GraphRuntimeStateSupport): message.answer = answer_text message.updated_at = naive_utc_now() message.provider_response_latency = time.perf_counter() - self._base_task_pipeline.start_at - message.message_metadata = self._task_state.metadata.model_dump_json() + + # Set usage first before dumping metadata + if graph_runtime_state and graph_runtime_state.llm_usage: + usage = graph_runtime_state.llm_usage + message.message_tokens = usage.prompt_tokens + message.message_unit_price = usage.prompt_unit_price + message.message_price_unit = usage.prompt_price_unit + message.answer_tokens = usage.completion_tokens + message.answer_unit_price = usage.completion_unit_price + message.answer_price_unit = usage.completion_price_unit + message.total_price = usage.total_price + message.currency = usage.currency + self._task_state.metadata.usage = usage + else: + usage = LLMUsage.empty_usage() + self._task_state.metadata.usage = usage + + # Add streaming metrics to usage if available + if self._task_state.is_streaming_response and self._task_state.first_token_time: + start_time = self._base_task_pipeline.start_at + first_token_time = self._task_state.first_token_time + last_token_time = self._task_state.last_token_time or first_token_time + usage.time_to_first_token = round(first_token_time - start_time, 3) + usage.time_to_generate = round(last_token_time - first_token_time, 3) + + metadata = self._task_state.metadata.model_dump() + message.message_metadata = json.dumps(jsonable_encoder(metadata)) message_files = [ MessageFile( message_id=message.id, @@ -790,20 +826,6 @@ class AdvancedChatAppGenerateTaskPipeline(GraphRuntimeStateSupport): ] session.add_all(message_files) - if graph_runtime_state and graph_runtime_state.llm_usage: - usage = graph_runtime_state.llm_usage - message.message_tokens = usage.prompt_tokens - message.message_unit_price = usage.prompt_unit_price - message.message_price_unit = usage.prompt_price_unit - message.answer_tokens = usage.completion_tokens - message.answer_unit_price = usage.completion_unit_price - message.answer_price_unit = usage.completion_price_unit - message.total_price = usage.total_price - message.currency = usage.currency - self._task_state.metadata.usage = usage - else: - self._task_state.metadata.usage = LLMUsage.empty_usage() - def _seed_graph_runtime_state_from_queue_manager(self) -> None: """Bootstrap the cached runtime state from the queue manager when present.""" candidate = self._base_task_pipeline.queue_manager.graph_runtime_state diff --git a/api/core/app/apps/agent_chat/app_runner.py b/api/core/app/apps/agent_chat/app_runner.py index 759398b556..2760466a3b 100644 --- a/api/core/app/apps/agent_chat/app_runner.py +++ b/api/core/app/apps/agent_chat/app_runner.py @@ -144,7 +144,7 @@ class AgentChatAppRunner(AppRunner): prompt_template_entity=app_config.prompt_template, inputs=dict(inputs), files=list(files), - query=query or "", + query=query, memory=memory, ) @@ -172,7 +172,7 @@ class AgentChatAppRunner(AppRunner): prompt_template_entity=app_config.prompt_template, inputs=dict(inputs), files=list(files), - query=query or "", + query=query, memory=memory, ) diff --git a/api/core/app/apps/base_app_runner.py b/api/core/app/apps/base_app_runner.py index 61ac040c05..9a9832dd4a 100644 --- a/api/core/app/apps/base_app_runner.py +++ b/api/core/app/apps/base_app_runner.py @@ -79,7 +79,7 @@ class AppRunner: prompt_template_entity: PromptTemplateEntity, inputs: Mapping[str, str], files: Sequence["File"], - query: str | None = None, + query: str = "", context: str | None = None, memory: TokenBufferMemory | None = None, image_detail_config: ImagePromptMessageContent.DETAIL | None = None, @@ -105,7 +105,7 @@ class AppRunner: app_mode=AppMode.value_of(app_record.mode), prompt_template_entity=prompt_template_entity, inputs=inputs, - query=query or "", + query=query, files=files, context=context, memory=memory, diff --git a/api/core/app/apps/common/workflow_response_converter.py b/api/core/app/apps/common/workflow_response_converter.py index 2c9ce5b56d..14795a430c 100644 --- a/api/core/app/apps/common/workflow_response_converter.py +++ b/api/core/app/apps/common/workflow_response_converter.py @@ -4,7 +4,7 @@ from dataclasses import dataclass from datetime import datetime from typing import Any, NewType, Union -from core.app.entities.app_invoke_entities import AdvancedChatAppGenerateEntity, WorkflowAppGenerateEntity +from core.app.entities.app_invoke_entities import AdvancedChatAppGenerateEntity, InvokeFrom, WorkflowAppGenerateEntity from core.app.entities.queue_entities import ( QueueAgentLogEvent, QueueIterationCompletedEvent, @@ -37,6 +37,7 @@ from core.file import FILE_MODEL_IDENTITY, File from core.plugin.impl.datasource import PluginDatasourceManager from core.tools.entities.tool_entities import ToolProviderType from core.tools.tool_manager import ToolManager +from core.trigger.trigger_manager import TriggerManager from core.variables.segments import ArrayFileSegment, FileSegment, Segment from core.workflow.enums import ( NodeType, @@ -51,7 +52,7 @@ from core.workflow.workflow_entry import WorkflowEntry from core.workflow.workflow_type_encoder import WorkflowRuntimeTypeConverter from libs.datetime_utils import naive_utc_now from models import Account, EndUser -from services.variable_truncator import VariableTruncator +from services.variable_truncator import BaseTruncator, DummyVariableTruncator, VariableTruncator NodeExecutionId = NewType("NodeExecutionId", str) @@ -70,6 +71,8 @@ class _NodeSnapshot: class WorkflowResponseConverter: + _truncator: BaseTruncator + def __init__( self, *, @@ -81,7 +84,13 @@ class WorkflowResponseConverter: self._user = user self._system_variables = system_variables self._workflow_inputs = self._prepare_workflow_inputs() - self._truncator = VariableTruncator.default() + + # Disable truncation for SERVICE_API calls to keep backward compatibility. + if application_generate_entity.invoke_from == InvokeFrom.SERVICE_API: + self._truncator = DummyVariableTruncator() + else: + self._truncator = VariableTruncator.default() + self._node_snapshots: dict[NodeExecutionId, _NodeSnapshot] = {} self._workflow_execution_id: str | None = None self._workflow_started_at: datetime | None = None @@ -295,6 +304,11 @@ class WorkflowResponseConverter: response.data.extras["icon"] = provider_entity.declaration.identity.generate_datasource_icon_url( self._application_generate_entity.app_config.tenant_id ) + elif event.node_type == NodeType.TRIGGER_PLUGIN: + response.data.extras["icon"] = TriggerManager.get_trigger_plugin_icon( + self._application_generate_entity.app_config.tenant_id, + event.provider_id, + ) return response diff --git a/api/core/app/apps/message_based_app_generator.py b/api/core/app/apps/message_based_app_generator.py index 7a51b8f3a5..53e67fd578 100644 --- a/api/core/app/apps/message_based_app_generator.py +++ b/api/core/app/apps/message_based_app_generator.py @@ -190,7 +190,7 @@ class MessageBasedAppGenerator(BaseAppGenerator): override_model_configs=json.dumps(override_model_configs) if override_model_configs else None, conversation_id=conversation.id, inputs=application_generate_entity.inputs, - query=application_generate_entity.query or "", + query=application_generate_entity.query, message="", message_tokens=0, message_unit_price=0, diff --git a/api/core/app/apps/pipeline/pipeline_generator.py b/api/core/app/apps/pipeline/pipeline_generator.py index f8bfbce37a..a1390ad0be 100644 --- a/api/core/app/apps/pipeline/pipeline_generator.py +++ b/api/core/app/apps/pipeline/pipeline_generator.py @@ -41,18 +41,14 @@ from core.workflow.repositories.workflow_execution_repository import WorkflowExe from core.workflow.repositories.workflow_node_execution_repository import WorkflowNodeExecutionRepository from core.workflow.variable_loader import DUMMY_VARIABLE_LOADER, VariableLoader from extensions.ext_database import db -from extensions.ext_redis import redis_client from libs.flask_utils import preserve_flask_contexts from models import Account, EndUser, Workflow, WorkflowNodeExecutionTriggeredFrom from models.dataset import Document, DocumentPipelineExecutionLog, Pipeline from models.enums import WorkflowRunTriggeredFrom from models.model import AppMode from services.datasource_provider_service import DatasourceProviderService -from services.feature_service import FeatureService -from services.file_service import FileService +from services.rag_pipeline.rag_pipeline_task_proxy import RagPipelineTaskProxy from services.workflow_draft_variable_service import DraftVarLoader, WorkflowDraftVariableService -from tasks.rag_pipeline.priority_rag_pipeline_run_task import priority_rag_pipeline_run_task -from tasks.rag_pipeline.rag_pipeline_run_task import rag_pipeline_run_task logger = logging.getLogger(__name__) @@ -248,34 +244,7 @@ class PipelineGenerator(BaseAppGenerator): ) if rag_pipeline_invoke_entities: - # store the rag_pipeline_invoke_entities to object storage - text = [item.model_dump() for item in rag_pipeline_invoke_entities] - name = "rag_pipeline_invoke_entities.json" - # Convert list to proper JSON string - json_text = json.dumps(text) - upload_file = FileService(db.engine).upload_text(json_text, name, user.id, dataset.tenant_id) - features = FeatureService.get_features(dataset.tenant_id) - if features.billing.enabled and features.billing.subscription.plan == "sandbox": - tenant_pipeline_task_key = f"tenant_pipeline_task:{dataset.tenant_id}" - tenant_self_pipeline_task_queue = f"tenant_self_pipeline_task_queue:{dataset.tenant_id}" - - if redis_client.get(tenant_pipeline_task_key): - # Add to waiting queue using List operations (lpush) - redis_client.lpush(tenant_self_pipeline_task_queue, upload_file.id) - else: - # Set flag and execute task - redis_client.set(tenant_pipeline_task_key, 1, ex=60 * 60) - rag_pipeline_run_task.delay( # type: ignore - rag_pipeline_invoke_entities_file_id=upload_file.id, - tenant_id=dataset.tenant_id, - ) - - else: - priority_rag_pipeline_run_task.delay( # type: ignore - rag_pipeline_invoke_entities_file_id=upload_file.id, - tenant_id=dataset.tenant_id, - ) - + RagPipelineTaskProxy(dataset.tenant_id, user.id, rag_pipeline_invoke_entities).delay() # return batch, dataset, documents return { "batch": batch, diff --git a/api/core/app/apps/workflow/app_generator.py b/api/core/app/apps/workflow/app_generator.py index f22ef5431e..be331b92a8 100644 --- a/api/core/app/apps/workflow/app_generator.py +++ b/api/core/app/apps/workflow/app_generator.py @@ -27,6 +27,7 @@ from core.helper.trace_id_helper import extract_external_trace_id_from_args from core.model_runtime.errors.invoke import InvokeAuthorizationError from core.ops.ops_trace_manager import TraceQueueManager from core.repositories import DifyCoreRepositoryFactory +from core.workflow.graph_engine.layers.base import GraphEngineLayer from core.workflow.repositories.draft_variable_repository import DraftVariableSaverFactory from core.workflow.repositories.workflow_execution_repository import WorkflowExecutionRepository from core.workflow.repositories.workflow_node_execution_repository import WorkflowNodeExecutionRepository @@ -38,10 +39,16 @@ from models import Account, App, EndUser, Workflow, WorkflowNodeExecutionTrigger from models.enums import WorkflowRunTriggeredFrom from services.workflow_draft_variable_service import DraftVarLoader, WorkflowDraftVariableService +SKIP_PREPARE_USER_INPUTS_KEY = "_skip_prepare_user_inputs" + logger = logging.getLogger(__name__) class WorkflowAppGenerator(BaseAppGenerator): + @staticmethod + def _should_prepare_user_inputs(args: Mapping[str, Any]) -> bool: + return not bool(args.get(SKIP_PREPARE_USER_INPUTS_KEY)) + @overload def generate( self, @@ -53,7 +60,10 @@ class WorkflowAppGenerator(BaseAppGenerator): invoke_from: InvokeFrom, streaming: Literal[True], call_depth: int, - ) -> Generator[Mapping | str, None, None]: ... + triggered_from: WorkflowRunTriggeredFrom | None = None, + root_node_id: str | None = None, + graph_engine_layers: Sequence[GraphEngineLayer] = (), + ) -> Generator[Mapping[str, Any] | str, None, None]: ... @overload def generate( @@ -66,6 +76,9 @@ class WorkflowAppGenerator(BaseAppGenerator): invoke_from: InvokeFrom, streaming: Literal[False], call_depth: int, + triggered_from: WorkflowRunTriggeredFrom | None = None, + root_node_id: str | None = None, + graph_engine_layers: Sequence[GraphEngineLayer] = (), ) -> Mapping[str, Any]: ... @overload @@ -79,7 +92,10 @@ class WorkflowAppGenerator(BaseAppGenerator): invoke_from: InvokeFrom, streaming: bool, call_depth: int, - ) -> Union[Mapping[str, Any], Generator[Mapping | str, None, None]]: ... + triggered_from: WorkflowRunTriggeredFrom | None = None, + root_node_id: str | None = None, + graph_engine_layers: Sequence[GraphEngineLayer] = (), + ) -> Union[Mapping[str, Any], Generator[Mapping[str, Any] | str, None, None]]: ... def generate( self, @@ -91,7 +107,10 @@ class WorkflowAppGenerator(BaseAppGenerator): invoke_from: InvokeFrom, streaming: bool = True, call_depth: int = 0, - ) -> Union[Mapping[str, Any], Generator[Mapping | str, None, None]]: + triggered_from: WorkflowRunTriggeredFrom | None = None, + root_node_id: str | None = None, + graph_engine_layers: Sequence[GraphEngineLayer] = (), + ) -> Union[Mapping[str, Any], Generator[Mapping[str, Any] | str, None, None]]: files: Sequence[Mapping[str, Any]] = args.get("files") or [] # parse files @@ -126,17 +145,20 @@ class WorkflowAppGenerator(BaseAppGenerator): **extract_external_trace_id_from_args(args), } workflow_run_id = str(uuid.uuid4()) + # for trigger debug run, not prepare user inputs + if self._should_prepare_user_inputs(args): + inputs = self._prepare_user_inputs( + user_inputs=inputs, + variables=app_config.variables, + tenant_id=app_model.tenant_id, + strict_type_validation=True if invoke_from == InvokeFrom.SERVICE_API else False, + ) # init application generate entity application_generate_entity = WorkflowAppGenerateEntity( task_id=str(uuid.uuid4()), app_config=app_config, file_upload_config=file_extra_config, - inputs=self._prepare_user_inputs( - user_inputs=inputs, - variables=app_config.variables, - tenant_id=app_model.tenant_id, - strict_type_validation=True if invoke_from == InvokeFrom.SERVICE_API else False, - ), + inputs=inputs, files=list(system_files), user_id=user.id, stream=streaming, @@ -155,7 +177,10 @@ class WorkflowAppGenerator(BaseAppGenerator): # Create session factory session_factory = sessionmaker(bind=db.engine, expire_on_commit=False) # Create workflow execution(aka workflow run) repository - if invoke_from == InvokeFrom.DEBUGGER: + if triggered_from is not None: + # Use explicitly provided triggered_from (for async triggers) + workflow_triggered_from = triggered_from + elif invoke_from == InvokeFrom.DEBUGGER: workflow_triggered_from = WorkflowRunTriggeredFrom.DEBUGGING else: workflow_triggered_from = WorkflowRunTriggeredFrom.APP_RUN @@ -182,8 +207,16 @@ class WorkflowAppGenerator(BaseAppGenerator): workflow_execution_repository=workflow_execution_repository, workflow_node_execution_repository=workflow_node_execution_repository, streaming=streaming, + root_node_id=root_node_id, + graph_engine_layers=graph_engine_layers, ) + def resume(self, *, workflow_run_id: str) -> None: + """ + @TBD + """ + pass + def _generate( self, *, @@ -196,6 +229,8 @@ class WorkflowAppGenerator(BaseAppGenerator): workflow_node_execution_repository: WorkflowNodeExecutionRepository, streaming: bool = True, variable_loader: VariableLoader = DUMMY_VARIABLE_LOADER, + root_node_id: str | None = None, + graph_engine_layers: Sequence[GraphEngineLayer] = (), ) -> Union[Mapping[str, Any], Generator[str | Mapping[str, Any], None, None]]: """ Generate App response. @@ -231,8 +266,10 @@ class WorkflowAppGenerator(BaseAppGenerator): "queue_manager": queue_manager, "context": context, "variable_loader": variable_loader, + "root_node_id": root_node_id, "workflow_execution_repository": workflow_execution_repository, "workflow_node_execution_repository": workflow_node_execution_repository, + "graph_engine_layers": graph_engine_layers, }, ) @@ -426,6 +463,8 @@ class WorkflowAppGenerator(BaseAppGenerator): variable_loader: VariableLoader, workflow_execution_repository: WorkflowExecutionRepository, workflow_node_execution_repository: WorkflowNodeExecutionRepository, + root_node_id: str | None = None, + graph_engine_layers: Sequence[GraphEngineLayer] = (), ) -> None: """ Generate worker in a new thread. @@ -469,6 +508,8 @@ class WorkflowAppGenerator(BaseAppGenerator): system_user_id=system_user_id, workflow_execution_repository=workflow_execution_repository, workflow_node_execution_repository=workflow_node_execution_repository, + root_node_id=root_node_id, + graph_engine_layers=graph_engine_layers, ) try: diff --git a/api/core/app/apps/workflow/app_runner.py b/api/core/app/apps/workflow/app_runner.py index 3c9bf176b5..d8460df390 100644 --- a/api/core/app/apps/workflow/app_runner.py +++ b/api/core/app/apps/workflow/app_runner.py @@ -1,5 +1,6 @@ import logging import time +from collections.abc import Sequence from typing import cast from core.app.apps.base_app_queue_manager import AppQueueManager @@ -8,6 +9,7 @@ from core.app.apps.workflow_app_runner import WorkflowBasedAppRunner from core.app.entities.app_invoke_entities import InvokeFrom, WorkflowAppGenerateEntity from core.workflow.enums import WorkflowType from core.workflow.graph_engine.command_channels.redis_channel import RedisChannel +from core.workflow.graph_engine.layers.base import GraphEngineLayer from core.workflow.graph_engine.layers.persistence import PersistenceWorkflowInfo, WorkflowPersistenceLayer from core.workflow.repositories.workflow_execution_repository import WorkflowExecutionRepository from core.workflow.repositories.workflow_node_execution_repository import WorkflowNodeExecutionRepository @@ -16,6 +18,7 @@ from core.workflow.system_variable import SystemVariable from core.workflow.variable_loader import VariableLoader from core.workflow.workflow_entry import WorkflowEntry from extensions.ext_redis import redis_client +from libs.datetime_utils import naive_utc_now from models.enums import UserFrom from models.workflow import Workflow @@ -35,17 +38,21 @@ class WorkflowAppRunner(WorkflowBasedAppRunner): variable_loader: VariableLoader, workflow: Workflow, system_user_id: str, + root_node_id: str | None = None, workflow_execution_repository: WorkflowExecutionRepository, workflow_node_execution_repository: WorkflowNodeExecutionRepository, + graph_engine_layers: Sequence[GraphEngineLayer] = (), ): super().__init__( queue_manager=queue_manager, variable_loader=variable_loader, app_id=application_generate_entity.app_config.app_id, + graph_engine_layers=graph_engine_layers, ) self.application_generate_entity = application_generate_entity self._workflow = workflow self._sys_user_id = system_user_id + self._root_node_id = root_node_id self._workflow_execution_repository = workflow_execution_repository self._workflow_node_execution_repository = workflow_node_execution_repository @@ -60,6 +67,7 @@ class WorkflowAppRunner(WorkflowBasedAppRunner): files=self.application_generate_entity.files, user_id=self._sys_user_id, app_id=app_config.app_id, + timestamp=int(naive_utc_now().timestamp()), workflow_id=app_config.workflow_id, workflow_execution_id=self.application_generate_entity.workflow_execution_id, ) @@ -92,6 +100,7 @@ class WorkflowAppRunner(WorkflowBasedAppRunner): workflow_id=self._workflow.id, tenant_id=self._workflow.tenant_id, user_id=self.application_generate_entity.user_id, + root_node_id=self._root_node_id, ) # RUN WORKFLOW @@ -135,6 +144,8 @@ class WorkflowAppRunner(WorkflowBasedAppRunner): ) workflow_entry.graph_engine.layer(persistence_layer) + for layer in self._graph_engine_layers: + workflow_entry.graph_engine.layer(layer) generator = workflow_entry.run() diff --git a/api/core/app/apps/workflow_app_runner.py b/api/core/app/apps/workflow_app_runner.py index 5e2bd17f8c..0e125b3538 100644 --- a/api/core/app/apps/workflow_app_runner.py +++ b/api/core/app/apps/workflow_app_runner.py @@ -1,5 +1,5 @@ import time -from collections.abc import Mapping +from collections.abc import Mapping, Sequence from typing import Any, cast from core.app.apps.base_app_queue_manager import AppQueueManager, PublishFrom @@ -27,6 +27,7 @@ from core.app.entities.queue_entities import ( ) from core.workflow.entities import GraphInitParams from core.workflow.graph import Graph +from core.workflow.graph_engine.layers.base import GraphEngineLayer from core.workflow.graph_events import ( GraphEngineEvent, GraphRunFailedEvent, @@ -69,10 +70,12 @@ class WorkflowBasedAppRunner: queue_manager: AppQueueManager, variable_loader: VariableLoader = DUMMY_VARIABLE_LOADER, app_id: str, + graph_engine_layers: Sequence[GraphEngineLayer] = (), ): self._queue_manager = queue_manager self._variable_loader = variable_loader self._app_id = app_id + self._graph_engine_layers = graph_engine_layers def _init_graph( self, @@ -81,6 +84,7 @@ class WorkflowBasedAppRunner: workflow_id: str = "", tenant_id: str = "", user_id: str = "", + root_node_id: str | None = None, ) -> Graph: """ Init graph @@ -114,7 +118,7 @@ class WorkflowBasedAppRunner: ) # init graph - graph = Graph.init(graph_config=graph_config, node_factory=node_factory) + graph = Graph.init(graph_config=graph_config, node_factory=node_factory, root_node_id=root_node_id) if not graph: raise ValueError("graph not found in workflow") diff --git a/api/core/app/entities/app_invoke_entities.py b/api/core/app/entities/app_invoke_entities.py index a5ed0f8fa3..5143dbf1e8 100644 --- a/api/core/app/entities/app_invoke_entities.py +++ b/api/core/app/entities/app_invoke_entities.py @@ -32,6 +32,10 @@ class InvokeFrom(StrEnum): # https://docs.dify.ai/en/guides/application-publishing/launch-your-webapp-quickly/README WEB_APP = "web-app" + # TRIGGER indicates that this invocation is from a trigger. + # this is used for plugin trigger and webhook trigger. + TRIGGER = "trigger" + # EXPLORE indicates that this invocation is from # the workflow (or chatflow) explore page. EXPLORE = "explore" @@ -40,6 +44,9 @@ class InvokeFrom(StrEnum): DEBUGGER = "debugger" PUBLISHED = "published" + # VALIDATION indicates that this invocation is from validation. + VALIDATION = "validation" + @classmethod def value_of(cls, value: str): """ @@ -65,6 +72,8 @@ class InvokeFrom(StrEnum): return "dev" elif self == InvokeFrom.EXPLORE: return "explore_app" + elif self == InvokeFrom.TRIGGER: + return "trigger" elif self == InvokeFrom.SERVICE_API: return "api" @@ -104,6 +113,11 @@ class AppGenerateEntity(BaseModel): inputs: Mapping[str, Any] files: Sequence[File] + + # Unique identifier of the user initiating the execution. + # This corresponds to `Account.id` for platform users or `EndUser.id` for end users. + # + # Note: The `user_id` field does not indicate whether the user is a platform user or an end user. user_id: str # extras @@ -129,7 +143,7 @@ class EasyUIBasedAppGenerateEntity(AppGenerateEntity): app_config: EasyUIBasedAppConfig = None # type: ignore model_conf: ModelConfigWithCredentialsEntity - query: str | None = None + query: str = "" # pydantic configs model_config = ConfigDict(protected_namespaces=()) diff --git a/api/core/app/entities/task_entities.py b/api/core/app/entities/task_entities.py index 72a92add04..79a5e657b3 100644 --- a/api/core/app/entities/task_entities.py +++ b/api/core/app/entities/task_entities.py @@ -48,6 +48,9 @@ class WorkflowTaskState(TaskState): """ answer: str = "" + first_token_time: float | None = None + last_token_time: float | None = None + is_streaming_response: bool = False class StreamEvent(StrEnum): diff --git a/api/core/app/layers/pause_state_persist_layer.py b/api/core/app/layers/pause_state_persist_layer.py new file mode 100644 index 0000000000..412eb98dd4 --- /dev/null +++ b/api/core/app/layers/pause_state_persist_layer.py @@ -0,0 +1,133 @@ +from typing import Annotated, Literal, Self, TypeAlias + +from pydantic import BaseModel, Field +from sqlalchemy import Engine +from sqlalchemy.orm import Session, sessionmaker + +from core.app.entities.app_invoke_entities import AdvancedChatAppGenerateEntity, WorkflowAppGenerateEntity +from core.workflow.graph_engine.layers.base import GraphEngineLayer +from core.workflow.graph_events.base import GraphEngineEvent +from core.workflow.graph_events.graph import GraphRunPausedEvent +from models.model import AppMode +from repositories.api_workflow_run_repository import APIWorkflowRunRepository +from repositories.factory import DifyAPIRepositoryFactory + + +# Wrapper types for `WorkflowAppGenerateEntity` and +# `AdvancedChatAppGenerateEntity`. These wrappers enable type discrimination +# and correct reconstruction of the entity field during (de)serialization. +class _WorkflowGenerateEntityWrapper(BaseModel): + type: Literal[AppMode.WORKFLOW] = AppMode.WORKFLOW + entity: WorkflowAppGenerateEntity + + +class _AdvancedChatAppGenerateEntityWrapper(BaseModel): + type: Literal[AppMode.ADVANCED_CHAT] = AppMode.ADVANCED_CHAT + entity: AdvancedChatAppGenerateEntity + + +_GenerateEntityUnion: TypeAlias = Annotated[ + _WorkflowGenerateEntityWrapper | _AdvancedChatAppGenerateEntityWrapper, + Field(discriminator="type"), +] + + +class WorkflowResumptionContext(BaseModel): + """WorkflowResumptionContext captures all state necessary for resumption.""" + + version: Literal["1"] = "1" + + # Only workflow / chatflow could be paused. + generate_entity: _GenerateEntityUnion + serialized_graph_runtime_state: str + + def dumps(self) -> str: + return self.model_dump_json() + + @classmethod + def loads(cls, value: str) -> Self: + return cls.model_validate_json(value) + + def get_generate_entity(self) -> WorkflowAppGenerateEntity | AdvancedChatAppGenerateEntity: + return self.generate_entity.entity + + +class PauseStatePersistenceLayer(GraphEngineLayer): + def __init__( + self, + session_factory: Engine | sessionmaker[Session], + generate_entity: WorkflowAppGenerateEntity | AdvancedChatAppGenerateEntity, + state_owner_user_id: str, + ): + """Create a PauseStatePersistenceLayer. + + The `state_owner_user_id` is used when creating state file for pause. + It generally should id of the creator of workflow. + """ + if isinstance(session_factory, Engine): + session_factory = sessionmaker(session_factory) + self._session_maker = session_factory + self._state_owner_user_id = state_owner_user_id + self._generate_entity = generate_entity + + def _get_repo(self) -> APIWorkflowRunRepository: + return DifyAPIRepositoryFactory.create_api_workflow_run_repository(self._session_maker) + + def on_graph_start(self) -> None: + """ + Called when graph execution starts. + + This is called after the engine has been initialized but before any nodes + are executed. Layers can use this to set up resources or log start information. + """ + pass + + def on_event(self, event: GraphEngineEvent) -> None: + """ + Called for every event emitted by the engine. + + This method receives all events generated during graph execution, including: + - Graph lifecycle events (start, success, failure) + - Node execution events (start, success, failure, retry) + - Stream events for response nodes + - Container events (iteration, loop) + + Args: + event: The event emitted by the engine + """ + if not isinstance(event, GraphRunPausedEvent): + return + + assert self.graph_runtime_state is not None + + entity_wrapper: _GenerateEntityUnion + if isinstance(self._generate_entity, WorkflowAppGenerateEntity): + entity_wrapper = _WorkflowGenerateEntityWrapper(entity=self._generate_entity) + else: + entity_wrapper = _AdvancedChatAppGenerateEntityWrapper(entity=self._generate_entity) + + state = WorkflowResumptionContext( + serialized_graph_runtime_state=self.graph_runtime_state.dumps(), + generate_entity=entity_wrapper, + ) + + workflow_run_id: str | None = self.graph_runtime_state.system_variable.workflow_execution_id + assert workflow_run_id is not None + repo = self._get_repo() + repo.create_workflow_pause( + workflow_run_id=workflow_run_id, + state_owner_user_id=self._state_owner_user_id, + state=state.dumps(), + ) + + def on_graph_end(self, error: Exception | None) -> None: + """ + Called when graph execution ends. + + This is called after all nodes have been executed or when execution is + aborted. Layers can use this to clean up resources or log final state. + + Args: + error: The exception that caused execution to fail, or None if successful + """ + pass diff --git a/api/core/app/layers/suspend_layer.py b/api/core/app/layers/suspend_layer.py new file mode 100644 index 0000000000..0a107de012 --- /dev/null +++ b/api/core/app/layers/suspend_layer.py @@ -0,0 +1,21 @@ +from core.workflow.graph_engine.layers.base import GraphEngineLayer +from core.workflow.graph_events.base import GraphEngineEvent +from core.workflow.graph_events.graph import GraphRunPausedEvent + + +class SuspendLayer(GraphEngineLayer): + """ """ + + def on_graph_start(self): + pass + + def on_event(self, event: GraphEngineEvent): + """ + Handle the paused event, stash runtime state into storage and wait for resume. + """ + if isinstance(event, GraphRunPausedEvent): + pass + + def on_graph_end(self, error: Exception | None): + """ """ + pass diff --git a/api/core/app/layers/timeslice_layer.py b/api/core/app/layers/timeslice_layer.py new file mode 100644 index 0000000000..f82397deca --- /dev/null +++ b/api/core/app/layers/timeslice_layer.py @@ -0,0 +1,88 @@ +import logging +import uuid +from typing import ClassVar + +from apscheduler.schedulers.background import BackgroundScheduler # type: ignore + +from core.workflow.graph_engine.entities.commands import CommandType, GraphEngineCommand +from core.workflow.graph_engine.layers.base import GraphEngineLayer +from core.workflow.graph_events.base import GraphEngineEvent +from services.workflow.entities import WorkflowScheduleCFSPlanEntity +from services.workflow.scheduler import CFSPlanScheduler, SchedulerCommand + +logger = logging.getLogger(__name__) + + +class TimeSliceLayer(GraphEngineLayer): + """ + CFS plan scheduler to control the timeslice of the workflow. + """ + + scheduler: ClassVar[BackgroundScheduler] = BackgroundScheduler() + + def __init__(self, cfs_plan_scheduler: CFSPlanScheduler) -> None: + """ + CFS plan scheduler allows to control the timeslice of the workflow. + """ + + if not TimeSliceLayer.scheduler.running: + TimeSliceLayer.scheduler.start() + + super().__init__() + self.cfs_plan_scheduler = cfs_plan_scheduler + self.stopped = False + self.schedule_id = "" + + def _checker_job(self, schedule_id: str): + """ + Check if the workflow need to be suspended. + """ + try: + if self.stopped: + self.scheduler.remove_job(schedule_id) + return + + if self.cfs_plan_scheduler.can_schedule() == SchedulerCommand.RESOURCE_LIMIT_REACHED: + # remove the job + self.scheduler.remove_job(schedule_id) + + if not self.command_channel: + logger.exception("No command channel to stop the workflow") + return + + # send command to pause the workflow + self.command_channel.send_command( + GraphEngineCommand( + command_type=CommandType.PAUSE, + payload={ + "reason": SchedulerCommand.RESOURCE_LIMIT_REACHED, + }, + ) + ) + + except Exception: + logger.exception("scheduler error during check if the workflow need to be suspended") + + def on_graph_start(self): + """ + Start timer to check if the workflow need to be suspended. + """ + + if self.cfs_plan_scheduler.plan.schedule_strategy == WorkflowScheduleCFSPlanEntity.Strategy.TimeSlice: + self.schedule_id = uuid.uuid4().hex + + self.scheduler.add_job( + lambda: self._checker_job(self.schedule_id), + "interval", + seconds=self.cfs_plan_scheduler.plan.granularity, + id=self.schedule_id, + ) + + def on_event(self, event: GraphEngineEvent): + pass + + def on_graph_end(self, error: Exception | None) -> None: + self.stopped = True + # remove the scheduler + if self.schedule_id: + self.scheduler.remove_job(self.schedule_id) diff --git a/api/core/app/layers/trigger_post_layer.py b/api/core/app/layers/trigger_post_layer.py new file mode 100644 index 0000000000..fe1a46a945 --- /dev/null +++ b/api/core/app/layers/trigger_post_layer.py @@ -0,0 +1,88 @@ +import logging +from datetime import UTC, datetime +from typing import Any, ClassVar + +from pydantic import TypeAdapter +from sqlalchemy.orm import Session, sessionmaker + +from core.workflow.graph_engine.layers.base import GraphEngineLayer +from core.workflow.graph_events.base import GraphEngineEvent +from core.workflow.graph_events.graph import GraphRunFailedEvent, GraphRunPausedEvent, GraphRunSucceededEvent +from models.enums import WorkflowTriggerStatus +from repositories.sqlalchemy_workflow_trigger_log_repository import SQLAlchemyWorkflowTriggerLogRepository +from tasks.workflow_cfs_scheduler.cfs_scheduler import AsyncWorkflowCFSPlanEntity + +logger = logging.getLogger(__name__) + + +class TriggerPostLayer(GraphEngineLayer): + """ + Trigger post layer. + """ + + _STATUS_MAP: ClassVar[dict[type[GraphEngineEvent], WorkflowTriggerStatus]] = { + GraphRunSucceededEvent: WorkflowTriggerStatus.SUCCEEDED, + GraphRunFailedEvent: WorkflowTriggerStatus.FAILED, + GraphRunPausedEvent: WorkflowTriggerStatus.PAUSED, + } + + def __init__( + self, + cfs_plan_scheduler_entity: AsyncWorkflowCFSPlanEntity, + start_time: datetime, + trigger_log_id: str, + session_maker: sessionmaker[Session], + ): + self.trigger_log_id = trigger_log_id + self.start_time = start_time + self.cfs_plan_scheduler_entity = cfs_plan_scheduler_entity + self.session_maker = session_maker + + def on_graph_start(self): + pass + + def on_event(self, event: GraphEngineEvent): + """ + Update trigger log with success or failure. + """ + if isinstance(event, tuple(self._STATUS_MAP.keys())): + with self.session_maker() as session: + repo = SQLAlchemyWorkflowTriggerLogRepository(session) + trigger_log = repo.get_by_id(self.trigger_log_id) + if not trigger_log: + logger.exception("Trigger log not found: %s", self.trigger_log_id) + return + + # Calculate elapsed time + elapsed_time = (datetime.now(UTC) - self.start_time).total_seconds() + + # Extract relevant data from result + if not self.graph_runtime_state: + logger.exception("Graph runtime state is not set") + return + + outputs = self.graph_runtime_state.outputs + + # BASICLY, workflow_execution_id is the same as workflow_run_id + workflow_run_id = self.graph_runtime_state.system_variable.workflow_execution_id + assert workflow_run_id, "Workflow run id is not set" + + total_tokens = self.graph_runtime_state.total_tokens + + # Update trigger log with success + trigger_log.status = self._STATUS_MAP[type(event)] + trigger_log.workflow_run_id = workflow_run_id + trigger_log.outputs = TypeAdapter(dict[str, Any]).dump_json(outputs).decode() + + if trigger_log.elapsed_time is None: + trigger_log.elapsed_time = elapsed_time + else: + trigger_log.elapsed_time += elapsed_time + + trigger_log.total_tokens = total_tokens + trigger_log.finished_at = datetime.now(UTC) + repo.update(trigger_log) + session.commit() + + def on_graph_end(self, error: Exception | None) -> None: + pass diff --git a/api/core/app/task_pipeline/easy_ui_based_generate_task_pipeline.py b/api/core/app/task_pipeline/easy_ui_based_generate_task_pipeline.py index 67abb569e3..da2ebac3bd 100644 --- a/api/core/app/task_pipeline/easy_ui_based_generate_task_pipeline.py +++ b/api/core/app/task_pipeline/easy_ui_based_generate_task_pipeline.py @@ -121,7 +121,7 @@ class EasyUIBasedGenerateTaskPipeline(BasedGenerateTaskPipeline): if self._application_generate_entity.app_config.app_mode != AppMode.COMPLETION: # start generate conversation name thread self._conversation_name_generate_thread = self._message_cycle_manager.generate_conversation_name( - conversation_id=self._conversation_id, query=self._application_generate_entity.query or "" + conversation_id=self._conversation_id, query=self._application_generate_entity.query ) generator = self._wrapper_process_stream_response(trace_manager=self._application_generate_entity.trace_manager) diff --git a/api/core/app/task_pipeline/message_cycle_manager.py b/api/core/app/task_pipeline/message_cycle_manager.py index 7a384e5c92..e7daeb4a32 100644 --- a/api/core/app/task_pipeline/message_cycle_manager.py +++ b/api/core/app/task_pipeline/message_cycle_manager.py @@ -140,7 +140,27 @@ class MessageCycleManager: if not self._application_generate_entity.app_config.additional_features: raise ValueError("Additional features not found") if self._application_generate_entity.app_config.additional_features.show_retrieve_source: - self._task_state.metadata.retriever_resources = event.retriever_resources + merged_resources = [r for r in self._task_state.metadata.retriever_resources or [] if r] + existing_ids = {(r.dataset_id, r.document_id) for r in merged_resources if r.dataset_id and r.document_id} + + # Add new unique resources from the event + for resource in event.retriever_resources or []: + if not resource: + continue + + is_duplicate = ( + resource.dataset_id + and resource.document_id + and (resource.dataset_id, resource.document_id) in existing_ids + ) + + if not is_duplicate: + merged_resources.append(resource) + + for i, resource in enumerate(merged_resources, 1): + resource.position = i + + self._task_state.metadata.retriever_resources = merged_resources def message_file_to_stream_response(self, event: QueueMessageFileEvent) -> MessageFileStreamResponse | None: """ diff --git a/api/core/entities/document_task.py b/api/core/entities/document_task.py new file mode 100644 index 0000000000..27ab5c84f7 --- /dev/null +++ b/api/core/entities/document_task.py @@ -0,0 +1,15 @@ +from collections.abc import Sequence +from dataclasses import dataclass + + +@dataclass +class DocumentTask: + """Document task entity for document indexing operations. + + This class represents a document indexing task that can be queued + and processed by the document indexing system. + """ + + tenant_id: str + dataset_id: str + document_ids: Sequence[str] diff --git a/api/core/entities/mcp_provider.py b/api/core/entities/mcp_provider.py new file mode 100644 index 0000000000..7484cea04a --- /dev/null +++ b/api/core/entities/mcp_provider.py @@ -0,0 +1,329 @@ +import json +from datetime import datetime +from enum import StrEnum +from typing import TYPE_CHECKING, Any +from urllib.parse import urlparse + +from pydantic import BaseModel + +from configs import dify_config +from core.entities.provider_entities import BasicProviderConfig +from core.file import helpers as file_helpers +from core.helper import encrypter +from core.helper.provider_cache import NoOpProviderCredentialCache +from core.mcp.types import OAuthClientInformation, OAuthClientMetadata, OAuthTokens +from core.tools.entities.common_entities import I18nObject +from core.tools.entities.tool_entities import ToolProviderType + +if TYPE_CHECKING: + from models.tools import MCPToolProvider + +# Constants +CLIENT_NAME = "Dify" +CLIENT_URI = "https://github.com/langgenius/dify" +DEFAULT_TOKEN_TYPE = "Bearer" +DEFAULT_EXPIRES_IN = 3600 +MASK_CHAR = "*" +MIN_UNMASK_LENGTH = 6 + + +class MCPSupportGrantType(StrEnum): + """The supported grant types for MCP""" + + AUTHORIZATION_CODE = "authorization_code" + CLIENT_CREDENTIALS = "client_credentials" + REFRESH_TOKEN = "refresh_token" + + +class MCPAuthentication(BaseModel): + client_id: str + client_secret: str | None = None + + +class MCPConfiguration(BaseModel): + timeout: float = 30 + sse_read_timeout: float = 300 + + +class MCPProviderEntity(BaseModel): + """MCP Provider domain entity for business logic operations""" + + # Basic identification + id: str + provider_id: str # server_identifier + name: str + tenant_id: str + user_id: str + + # Server connection info + server_url: str # encrypted URL + headers: dict[str, str] # encrypted headers + timeout: float + sse_read_timeout: float + + # Authentication related + authed: bool + credentials: dict[str, Any] # encrypted credentials + code_verifier: str | None = None # for OAuth + + # Tools and display info + tools: list[dict[str, Any]] # parsed tools list + icon: str | dict[str, str] # parsed icon + + # Timestamps + created_at: datetime + updated_at: datetime + + @classmethod + def from_db_model(cls, db_provider: "MCPToolProvider") -> "MCPProviderEntity": + """Create entity from database model with decryption""" + + return cls( + id=db_provider.id, + provider_id=db_provider.server_identifier, + name=db_provider.name, + tenant_id=db_provider.tenant_id, + user_id=db_provider.user_id, + server_url=db_provider.server_url, + headers=db_provider.headers, + timeout=db_provider.timeout, + sse_read_timeout=db_provider.sse_read_timeout, + authed=db_provider.authed, + credentials=db_provider.credentials, + tools=db_provider.tool_dict, + icon=db_provider.icon or "", + created_at=db_provider.created_at, + updated_at=db_provider.updated_at, + ) + + @property + def redirect_url(self) -> str: + """OAuth redirect URL""" + return dify_config.CONSOLE_API_URL + "/console/api/mcp/oauth/callback" + + @property + def client_metadata(self) -> OAuthClientMetadata: + """Metadata about this OAuth client.""" + # Get grant type from credentials + credentials = self.decrypt_credentials() + + # Try to get grant_type from different locations + grant_type = credentials.get("grant_type", MCPSupportGrantType.AUTHORIZATION_CODE) + + # For nested structure, check if client_information has grant_types + if "client_information" in credentials and isinstance(credentials["client_information"], dict): + client_info = credentials["client_information"] + # If grant_types is specified in client_information, use it to determine grant_type + if "grant_types" in client_info and isinstance(client_info["grant_types"], list): + if "client_credentials" in client_info["grant_types"]: + grant_type = MCPSupportGrantType.CLIENT_CREDENTIALS + elif "authorization_code" in client_info["grant_types"]: + grant_type = MCPSupportGrantType.AUTHORIZATION_CODE + + # Configure based on grant type + is_client_credentials = grant_type == MCPSupportGrantType.CLIENT_CREDENTIALS + + grant_types = ["refresh_token"] + grant_types.append("client_credentials" if is_client_credentials else "authorization_code") + + response_types = [] if is_client_credentials else ["code"] + redirect_uris = [] if is_client_credentials else [self.redirect_url] + + return OAuthClientMetadata( + redirect_uris=redirect_uris, + token_endpoint_auth_method="none", + grant_types=grant_types, + response_types=response_types, + client_name=CLIENT_NAME, + client_uri=CLIENT_URI, + ) + + @property + def provider_icon(self) -> dict[str, str] | str: + """Get provider icon, handling both dict and string formats""" + if isinstance(self.icon, dict): + return self.icon + try: + return json.loads(self.icon) + except (json.JSONDecodeError, TypeError): + # If not JSON, assume it's a file path + return file_helpers.get_signed_file_url(self.icon) + + def to_api_response(self, user_name: str | None = None, include_sensitive: bool = True) -> dict[str, Any]: + """Convert to API response format + + Args: + user_name: User name to display + include_sensitive: If False, skip expensive decryption operations (for list view optimization) + """ + response = { + "id": self.id, + "author": user_name or "Anonymous", + "name": self.name, + "icon": self.provider_icon, + "type": ToolProviderType.MCP.value, + "is_team_authorization": self.authed, + "server_url": self.masked_server_url(), + "server_identifier": self.provider_id, + "updated_at": int(self.updated_at.timestamp()), + "label": I18nObject(en_US=self.name, zh_Hans=self.name).to_dict(), + "description": I18nObject(en_US="", zh_Hans="").to_dict(), + } + + # Add configuration + response["configuration"] = { + "timeout": str(self.timeout), + "sse_read_timeout": str(self.sse_read_timeout), + } + + # Skip expensive operations when sensitive data is not needed (e.g., list view) + if not include_sensitive: + response["masked_headers"] = {} + response["is_dynamic_registration"] = True + else: + # Add masked headers + response["masked_headers"] = self.masked_headers() + + # Add authentication info if available + masked_creds = self.masked_credentials() + if masked_creds: + response["authentication"] = masked_creds + response["is_dynamic_registration"] = self.credentials.get("client_information", {}).get( + "is_dynamic_registration", True + ) + + return response + + def retrieve_client_information(self) -> OAuthClientInformation | None: + """OAuth client information if available""" + credentials = self.decrypt_credentials() + if not credentials: + return None + + # Check if we have nested client_information structure + if "client_information" not in credentials: + return None + client_info_data = credentials["client_information"] + if isinstance(client_info_data, dict): + if "encrypted_client_secret" in client_info_data: + client_info_data["client_secret"] = encrypter.decrypt_token( + self.tenant_id, client_info_data["encrypted_client_secret"] + ) + return OAuthClientInformation.model_validate(client_info_data) + return None + + def retrieve_tokens(self) -> OAuthTokens | None: + """OAuth tokens if available""" + if not self.credentials: + return None + credentials = self.decrypt_credentials() + return OAuthTokens( + access_token=credentials.get("access_token", ""), + token_type=credentials.get("token_type", DEFAULT_TOKEN_TYPE), + expires_in=int(credentials.get("expires_in", str(DEFAULT_EXPIRES_IN)) or DEFAULT_EXPIRES_IN), + refresh_token=credentials.get("refresh_token", ""), + ) + + def masked_server_url(self) -> str: + """Masked server URL for display""" + parsed = urlparse(self.decrypt_server_url()) + if parsed.path and parsed.path != "/": + masked = parsed._replace(path="/******") + return masked.geturl() + return parsed.geturl() + + def _mask_value(self, value: str) -> str: + """Mask a sensitive value for display""" + if len(value) > MIN_UNMASK_LENGTH: + return value[:2] + MASK_CHAR * (len(value) - 4) + value[-2:] + else: + return MASK_CHAR * len(value) + + def masked_headers(self) -> dict[str, str]: + """Masked headers for display""" + return {key: self._mask_value(value) for key, value in self.decrypt_headers().items()} + + def masked_credentials(self) -> dict[str, str]: + """Masked credentials for display""" + credentials = self.decrypt_credentials() + if not credentials: + return {} + + masked = {} + + if "client_information" not in credentials or not isinstance(credentials["client_information"], dict): + return {} + client_info = credentials["client_information"] + # Mask sensitive fields from nested structure + if client_info.get("client_id"): + masked["client_id"] = self._mask_value(client_info["client_id"]) + if client_info.get("encrypted_client_secret"): + masked["client_secret"] = self._mask_value( + encrypter.decrypt_token(self.tenant_id, client_info["encrypted_client_secret"]) + ) + if client_info.get("client_secret"): + masked["client_secret"] = self._mask_value(client_info["client_secret"]) + return masked + + def decrypt_server_url(self) -> str: + """Decrypt server URL""" + return encrypter.decrypt_token(self.tenant_id, self.server_url) + + def _decrypt_dict(self, data: dict[str, Any]) -> dict[str, Any]: + """Generic method to decrypt dictionary fields""" + from core.tools.utils.encryption import create_provider_encrypter + + if not data: + return {} + + # Only decrypt fields that are actually encrypted + # For nested structures, client_information is not encrypted as a whole + encrypted_fields = [] + for key, value in data.items(): + # Skip nested objects - they are not encrypted + if isinstance(value, dict): + continue + # Only process string values that might be encrypted + if isinstance(value, str) and value: + encrypted_fields.append(key) + + if not encrypted_fields: + return data + + # Create dynamic config only for encrypted fields + config = [BasicProviderConfig(type=BasicProviderConfig.Type.SECRET_INPUT, name=key) for key in encrypted_fields] + + encrypter_instance, _ = create_provider_encrypter( + tenant_id=self.tenant_id, + config=config, + cache=NoOpProviderCredentialCache(), + ) + + # Decrypt only the encrypted fields + decrypted_data = encrypter_instance.decrypt({k: data[k] for k in encrypted_fields}) + + # Merge decrypted data with original data (preserving non-encrypted fields) + result = data.copy() + result.update(decrypted_data) + + return result + + def decrypt_headers(self) -> dict[str, Any]: + """Decrypt headers""" + return self._decrypt_dict(self.headers) + + def decrypt_credentials(self) -> dict[str, Any]: + """Decrypt credentials""" + return self._decrypt_dict(self.credentials) + + def decrypt_authentication(self) -> dict[str, Any]: + """Decrypt authentication""" + # Option 1: if headers is provided, use it and don't need to get token + headers = self.decrypt_headers() + + # Option 2: Add OAuth token if authed and no headers provided + if not self.headers and self.authed: + token = self.retrieve_tokens() + if token: + headers["Authorization"] = f"{token.token_type.capitalize()} {token.access_token}" + return headers diff --git a/api/core/entities/parameter_entities.py b/api/core/entities/parameter_entities.py index 0afb51edce..b61c4ad4bb 100644 --- a/api/core/entities/parameter_entities.py +++ b/api/core/entities/parameter_entities.py @@ -14,6 +14,7 @@ class CommonParameterType(StrEnum): APP_SELECTOR = "app-selector" MODEL_SELECTOR = "model-selector" TOOLS_SELECTOR = "array[tools]" + CHECKBOX = "checkbox" ANY = auto() # Dynamic select parameter diff --git a/api/core/entities/provider_configuration.py b/api/core/entities/provider_configuration.py index b10838f8c9..56c133e598 100644 --- a/api/core/entities/provider_configuration.py +++ b/api/core/entities/provider_configuration.py @@ -1533,6 +1533,9 @@ class ProviderConfiguration(BaseModel): # Return composite sort key: (model_type value, model position index) return (model.model_type.value, position_index) + # Deduplicate + provider_models = list({(m.model, m.model_type, m.fetch_from): m for m in provider_models}.values()) + # Sort using the composite sort key return sorted(provider_models, key=get_sort_key) diff --git a/api/core/entities/provider_entities.py b/api/core/entities/provider_entities.py index 0496959ce2..8a8067332d 100644 --- a/api/core/entities/provider_entities.py +++ b/api/core/entities/provider_entities.py @@ -107,7 +107,7 @@ class CustomModelConfiguration(BaseModel): model: str model_type: ModelType - credentials: dict | None = None + credentials: dict | None current_credential_id: str | None = None current_credential_name: str | None = None available_model_credentials: list[CredentialConfiguration] = [] @@ -207,6 +207,7 @@ class ProviderConfig(BasicProviderConfig): required: bool = False default: Union[int, str, float, bool] | None = None options: list[Option] | None = None + multiple: bool | None = False label: I18nObject | None = None help: I18nObject | None = None url: str | None = None diff --git a/api/core/file/models.py b/api/core/file/models.py index 7089b7ce7a..d149205d77 100644 --- a/api/core/file/models.py +++ b/api/core/file/models.py @@ -74,6 +74,10 @@ class File(BaseModel): storage_key: str | None = None, dify_model_identity: str | None = FILE_MODEL_IDENTITY, url: str | None = None, + # Legacy compatibility fields - explicitly handle known extra fields + tool_file_id: str | None = None, + upload_file_id: str | None = None, + datasource_file_id: str | None = None, ): super().__init__( id=id, diff --git a/api/core/helper/code_executor/javascript/javascript_transformer.py b/api/core/helper/code_executor/javascript/javascript_transformer.py index 62489cdf29..e28f027a3a 100644 --- a/api/core/helper/code_executor/javascript/javascript_transformer.py +++ b/api/core/helper/code_executor/javascript/javascript_transformer.py @@ -6,10 +6,7 @@ from core.helper.code_executor.template_transformer import TemplateTransformer class NodeJsTemplateTransformer(TemplateTransformer): @classmethod def get_runner_script(cls) -> str: - runner_script = dedent( - f""" - // declare main function - {cls._code_placeholder} + runner_script = dedent(f""" {cls._code_placeholder} // decode and prepare input object var inputs_obj = JSON.parse(Buffer.from('{cls._inputs_placeholder}', 'base64').toString('utf-8')) @@ -21,6 +18,5 @@ class NodeJsTemplateTransformer(TemplateTransformer): var output_json = JSON.stringify(output_obj) var result = `<>${{output_json}}<>` console.log(result) - """ - ) + """) return runner_script diff --git a/api/core/helper/code_executor/python3/python3_transformer.py b/api/core/helper/code_executor/python3/python3_transformer.py index 836fd273ae..ee866eeb81 100644 --- a/api/core/helper/code_executor/python3/python3_transformer.py +++ b/api/core/helper/code_executor/python3/python3_transformer.py @@ -6,9 +6,7 @@ from core.helper.code_executor.template_transformer import TemplateTransformer class Python3TemplateTransformer(TemplateTransformer): @classmethod def get_runner_script(cls) -> str: - runner_script = dedent(f""" - # declare main function - {cls._code_placeholder} + runner_script = dedent(f""" {cls._code_placeholder} import json from base64 import b64decode diff --git a/api/core/helper/marketplace.py b/api/core/helper/marketplace.py index bddb864a95..b2286d39ed 100644 --- a/api/core/helper/marketplace.py +++ b/api/core/helper/marketplace.py @@ -29,6 +29,18 @@ def batch_fetch_plugin_manifests(plugin_ids: list[str]) -> Sequence[MarketplaceP return [MarketplacePluginDeclaration.model_validate(plugin) for plugin in response.json()["data"]["plugins"]] +def batch_fetch_plugin_by_ids(plugin_ids: list[str]) -> list[dict]: + if not plugin_ids: + return [] + + url = str(marketplace_api_url / "api/v1/plugins/batch") + response = httpx.post(url, json={"plugin_ids": plugin_ids}, headers={"X-Dify-Version": dify_config.project.version}) + response.raise_for_status() + + data = response.json() + return data.get("data", {}).get("plugins", []) + + def batch_fetch_plugin_manifests_ignore_deserialization_error( plugin_ids: list[str], ) -> Sequence[MarketplacePluginDeclaration]: diff --git a/api/core/helper/name_generator.py b/api/core/helper/name_generator.py index 4e19e3946f..b5f9299d9f 100644 --- a/api/core/helper/name_generator.py +++ b/api/core/helper/name_generator.py @@ -3,7 +3,7 @@ import re from collections.abc import Sequence from typing import Any -from core.tools.entities.tool_entities import CredentialType +from core.plugin.entities.plugin_daemon import CredentialType logger = logging.getLogger(__name__) diff --git a/api/core/helper/provider_encryption.py b/api/core/helper/provider_encryption.py new file mode 100644 index 0000000000..8484a28c05 --- /dev/null +++ b/api/core/helper/provider_encryption.py @@ -0,0 +1,129 @@ +import contextlib +from collections.abc import Mapping +from copy import deepcopy +from typing import Any, Protocol + +from core.entities.provider_entities import BasicProviderConfig +from core.helper import encrypter + + +class ProviderConfigCache(Protocol): + """ + Interface for provider configuration cache operations + """ + + def get(self) -> dict[str, Any] | None: + """Get cached provider configuration""" + ... + + def set(self, config: dict[str, Any]) -> None: + """Cache provider configuration""" + ... + + def delete(self) -> None: + """Delete cached provider configuration""" + ... + + +class ProviderConfigEncrypter: + tenant_id: str + config: list[BasicProviderConfig] + provider_config_cache: ProviderConfigCache + + def __init__( + self, + tenant_id: str, + config: list[BasicProviderConfig], + provider_config_cache: ProviderConfigCache, + ): + self.tenant_id = tenant_id + self.config = config + self.provider_config_cache = provider_config_cache + + def _deep_copy(self, data: Mapping[str, Any]) -> Mapping[str, Any]: + """ + deep copy data + """ + return deepcopy(data) + + def encrypt(self, data: Mapping[str, Any]) -> Mapping[str, Any]: + """ + encrypt tool credentials with tenant id + + return a deep copy of credentials with encrypted values + """ + data = dict(self._deep_copy(data)) + + # get fields need to be decrypted + fields = dict[str, BasicProviderConfig]() + for credential in self.config: + fields[credential.name] = credential + + for field_name, field in fields.items(): + if field.type == BasicProviderConfig.Type.SECRET_INPUT: + if field_name in data: + encrypted = encrypter.encrypt_token(self.tenant_id, data[field_name] or "") + data[field_name] = encrypted + + return data + + def mask_credentials(self, data: Mapping[str, Any]) -> Mapping[str, Any]: + """ + mask credentials + + return a deep copy of credentials with masked values + """ + data = dict(self._deep_copy(data)) + + # get fields need to be decrypted + fields = dict[str, BasicProviderConfig]() + for credential in self.config: + fields[credential.name] = credential + + for field_name, field in fields.items(): + if field.type == BasicProviderConfig.Type.SECRET_INPUT: + if field_name in data: + if len(data[field_name]) > 6: + data[field_name] = ( + data[field_name][:2] + "*" * (len(data[field_name]) - 4) + data[field_name][-2:] + ) + else: + data[field_name] = "*" * len(data[field_name]) + + return data + + def mask_plugin_credentials(self, data: Mapping[str, Any]) -> Mapping[str, Any]: + return self.mask_credentials(data) + + def decrypt(self, data: Mapping[str, Any]) -> Mapping[str, Any]: + """ + decrypt tool credentials with tenant id + + return a deep copy of credentials with decrypted values + """ + cached_credentials = self.provider_config_cache.get() + if cached_credentials: + return cached_credentials + + data = dict(self._deep_copy(data)) + # get fields need to be decrypted + fields = dict[str, BasicProviderConfig]() + for credential in self.config: + fields[credential.name] = credential + + for field_name, field in fields.items(): + if field.type == BasicProviderConfig.Type.SECRET_INPUT: + if field_name in data: + with contextlib.suppress(Exception): + # if the value is None or empty string, skip decrypt + if not data[field_name]: + continue + + data[field_name] = encrypter.decrypt_token(self.tenant_id, data[field_name]) + + self.provider_config_cache.set(dict(data)) + return data + + +def create_provider_encrypter(tenant_id: str, config: list[BasicProviderConfig], cache: ProviderConfigCache): + return ProviderConfigEncrypter(tenant_id=tenant_id, config=config, provider_config_cache=cache), cache diff --git a/api/core/indexing_runner.py b/api/core/indexing_runner.py index c430fba0b9..36b38b7b45 100644 --- a/api/core/indexing_runner.py +++ b/api/core/indexing_runner.py @@ -415,7 +415,6 @@ class IndexingRunner: document_id=dataset_document.id, after_indexing_status="splitting", extra_update_params={ - DatasetDocument.word_count: sum(len(text_doc.page_content) for text_doc in text_docs), DatasetDocument.parsing_completed_at: naive_utc_now(), }, ) @@ -755,6 +754,7 @@ class IndexingRunner: extra_update_params={ DatasetDocument.cleaning_completed_at: cur_time, DatasetDocument.splitting_completed_at: cur_time, + DatasetDocument.word_count: sum(len(doc.page_content) for doc in documents), }, ) diff --git a/api/core/mcp/auth/auth_flow.py b/api/core/mcp/auth/auth_flow.py index 7d938a8a7d..951c22f6dd 100644 --- a/api/core/mcp/auth/auth_flow.py +++ b/api/core/mcp/auth/auth_flow.py @@ -6,11 +6,15 @@ import secrets import urllib.parse from urllib.parse import urljoin, urlparse -import httpx -from pydantic import BaseModel, ValidationError +from httpx import ConnectError, HTTPStatusError, RequestError +from pydantic import ValidationError -from core.mcp.auth.auth_provider import OAuthClientProvider +from core.entities.mcp_provider import MCPProviderEntity, MCPSupportGrantType +from core.helper import ssrf_proxy +from core.mcp.entities import AuthAction, AuthActionType, AuthResult, OAuthCallbackState +from core.mcp.error import MCPRefreshTokenError from core.mcp.types import ( + LATEST_PROTOCOL_VERSION, OAuthClientInformation, OAuthClientInformationFull, OAuthClientMetadata, @@ -19,21 +23,10 @@ from core.mcp.types import ( ) from extensions.ext_redis import redis_client -LATEST_PROTOCOL_VERSION = "1.0" OAUTH_STATE_EXPIRY_SECONDS = 5 * 60 # 5 minutes expiry OAUTH_STATE_REDIS_KEY_PREFIX = "oauth_state:" -class OAuthCallbackState(BaseModel): - provider_id: str - tenant_id: str - server_url: str - metadata: OAuthMetadata | None = None - client_information: OAuthClientInformation - code_verifier: str - redirect_uri: str - - def generate_pkce_challenge() -> tuple[str, str]: """Generate PKCE challenge and verifier.""" code_verifier = base64.urlsafe_b64encode(os.urandom(40)).decode("utf-8") @@ -80,8 +73,13 @@ def _retrieve_redis_state(state_key: str) -> OAuthCallbackState: raise ValueError(f"Invalid state parameter: {str(e)}") -def handle_callback(state_key: str, authorization_code: str) -> OAuthCallbackState: - """Handle the callback from the OAuth provider.""" +def handle_callback(state_key: str, authorization_code: str) -> tuple[OAuthCallbackState, OAuthTokens]: + """ + Handle the callback from the OAuth provider. + + Returns: + A tuple of (callback_state, tokens) that can be used by the caller to save data. + """ # Retrieve state data from Redis (state is automatically deleted after retrieval) full_state_data = _retrieve_redis_state(state_key) @@ -93,30 +91,32 @@ def handle_callback(state_key: str, authorization_code: str) -> OAuthCallbackSta full_state_data.code_verifier, full_state_data.redirect_uri, ) - provider = OAuthClientProvider(full_state_data.provider_id, full_state_data.tenant_id, for_list=True) - provider.save_tokens(tokens) - return full_state_data + + return full_state_data, tokens def check_support_resource_discovery(server_url: str) -> tuple[bool, str]: """Check if the server supports OAuth 2.0 Resource Discovery.""" - b_scheme, b_netloc, b_path, _, b_query, b_fragment = urlparse(server_url, "", True) - url_for_resource_discovery = f"{b_scheme}://{b_netloc}/.well-known/oauth-protected-resource{b_path}" + b_scheme, b_netloc, _, _, b_query, b_fragment = urlparse(server_url, "", True) + url_for_resource_discovery = f"{b_scheme}://{b_netloc}/.well-known/oauth-protected-resource" if b_query: url_for_resource_discovery += f"?{b_query}" if b_fragment: url_for_resource_discovery += f"#{b_fragment}" try: headers = {"MCP-Protocol-Version": LATEST_PROTOCOL_VERSION, "User-Agent": "Dify"} - response = httpx.get(url_for_resource_discovery, headers=headers) + response = ssrf_proxy.get(url_for_resource_discovery, headers=headers) if 200 <= response.status_code < 300: body = response.json() - if "authorization_server_url" in body: + # Support both singular and plural forms + if body.get("authorization_servers"): + return True, body["authorization_servers"][0] + elif body.get("authorization_server_url"): return True, body["authorization_server_url"][0] else: return False, "" return False, "" - except httpx.RequestError: + except RequestError: # Not support resource discovery, fall back to well-known OAuth metadata return False, "" @@ -126,27 +126,37 @@ def discover_oauth_metadata(server_url: str, protocol_version: str | None = None # First check if the server supports OAuth 2.0 Resource Discovery support_resource_discovery, oauth_discovery_url = check_support_resource_discovery(server_url) if support_resource_discovery: - url = oauth_discovery_url + # The oauth_discovery_url is the authorization server base URL + # Try OpenID Connect discovery first (more common), then OAuth 2.0 + urls_to_try = [ + urljoin(oauth_discovery_url + "/", ".well-known/oauth-authorization-server"), + urljoin(oauth_discovery_url + "/", ".well-known/openid-configuration"), + ] else: - url = urljoin(server_url, "/.well-known/oauth-authorization-server") + urls_to_try = [urljoin(server_url, "/.well-known/oauth-authorization-server")] - try: - headers = {"MCP-Protocol-Version": protocol_version or LATEST_PROTOCOL_VERSION} - response = httpx.get(url, headers=headers) - if response.status_code == 404: - return None - if not response.is_success: - raise ValueError(f"HTTP {response.status_code} trying to load well-known OAuth metadata") - return OAuthMetadata.model_validate(response.json()) - except httpx.RequestError as e: - if isinstance(e, httpx.ConnectError): - response = httpx.get(url) + headers = {"MCP-Protocol-Version": protocol_version or LATEST_PROTOCOL_VERSION} + + for url in urls_to_try: + try: + response = ssrf_proxy.get(url, headers=headers) if response.status_code == 404: - return None + continue if not response.is_success: - raise ValueError(f"HTTP {response.status_code} trying to load well-known OAuth metadata") + response.raise_for_status() return OAuthMetadata.model_validate(response.json()) - raise + except (RequestError, HTTPStatusError) as e: + if isinstance(e, ConnectError): + response = ssrf_proxy.get(url) + if response.status_code == 404: + continue # Try next URL + if not response.is_success: + raise ValueError(f"HTTP {response.status_code} trying to load well-known OAuth metadata") + return OAuthMetadata.model_validate(response.json()) + # For other errors, try next URL + continue + + return None # No metadata found def start_authorization( @@ -213,7 +223,7 @@ def exchange_authorization( redirect_uri: str, ) -> OAuthTokens: """Exchanges an authorization code for an access token.""" - grant_type = "authorization_code" + grant_type = MCPSupportGrantType.AUTHORIZATION_CODE.value if metadata: token_url = metadata.token_endpoint @@ -233,7 +243,7 @@ def exchange_authorization( if client_information.client_secret: params["client_secret"] = client_information.client_secret - response = httpx.post(token_url, data=params) + response = ssrf_proxy.post(token_url, data=params) if not response.is_success: raise ValueError(f"Token exchange failed: HTTP {response.status_code}") return OAuthTokens.model_validate(response.json()) @@ -246,7 +256,7 @@ def refresh_authorization( refresh_token: str, ) -> OAuthTokens: """Exchange a refresh token for an updated access token.""" - grant_type = "refresh_token" + grant_type = MCPSupportGrantType.REFRESH_TOKEN.value if metadata: token_url = metadata.token_endpoint @@ -263,10 +273,55 @@ def refresh_authorization( if client_information.client_secret: params["client_secret"] = client_information.client_secret - - response = httpx.post(token_url, data=params) + try: + response = ssrf_proxy.post(token_url, data=params) + except ssrf_proxy.MaxRetriesExceededError as e: + raise MCPRefreshTokenError(e) from e if not response.is_success: - raise ValueError(f"Token refresh failed: HTTP {response.status_code}") + raise MCPRefreshTokenError(response.text) + return OAuthTokens.model_validate(response.json()) + + +def client_credentials_flow( + server_url: str, + metadata: OAuthMetadata | None, + client_information: OAuthClientInformation, + scope: str | None = None, +) -> OAuthTokens: + """Execute Client Credentials Flow to get access token.""" + grant_type = MCPSupportGrantType.CLIENT_CREDENTIALS.value + + if metadata: + token_url = metadata.token_endpoint + if metadata.grant_types_supported and grant_type not in metadata.grant_types_supported: + raise ValueError(f"Incompatible auth server: does not support grant type {grant_type}") + else: + token_url = urljoin(server_url, "/token") + + # Support both Basic Auth and body parameters for client authentication + headers = {"Content-Type": "application/x-www-form-urlencoded"} + data = {"grant_type": grant_type} + + if scope: + data["scope"] = scope + + # If client_secret is provided, use Basic Auth (preferred method) + if client_information.client_secret: + credentials = f"{client_information.client_id}:{client_information.client_secret}" + encoded_credentials = base64.b64encode(credentials.encode()).decode() + headers["Authorization"] = f"Basic {encoded_credentials}" + else: + # Fall back to including credentials in the body + data["client_id"] = client_information.client_id + if client_information.client_secret: + data["client_secret"] = client_information.client_secret + + response = ssrf_proxy.post(token_url, headers=headers, data=data) + if not response.is_success: + raise ValueError( + f"Client credentials token request failed: HTTP {response.status_code}, Response: {response.text}" + ) + return OAuthTokens.model_validate(response.json()) @@ -283,7 +338,7 @@ def register_client( else: registration_url = urljoin(server_url, "/register") - response = httpx.post( + response = ssrf_proxy.post( registration_url, json=client_metadata.model_dump(), headers={"Content-Type": "application/json"}, @@ -294,28 +349,111 @@ def register_client( def auth( - provider: OAuthClientProvider, - server_url: str, + provider: MCPProviderEntity, authorization_code: str | None = None, state_param: str | None = None, - for_list: bool = False, -) -> dict[str, str]: - """Orchestrates the full auth flow with a server using secure Redis state storage.""" - metadata = discover_oauth_metadata(server_url) +) -> AuthResult: + """ + Orchestrates the full auth flow with a server using secure Redis state storage. + + This function performs only network operations and returns actions that need + to be performed by the caller (such as saving data to database). + + Args: + provider: The MCP provider entity + authorization_code: Optional authorization code from OAuth callback + state_param: Optional state parameter from OAuth callback + + Returns: + AuthResult containing actions to be performed and response data + """ + actions: list[AuthAction] = [] + server_url = provider.decrypt_server_url() + server_metadata = discover_oauth_metadata(server_url) + client_metadata = provider.client_metadata + provider_id = provider.id + tenant_id = provider.tenant_id + client_information = provider.retrieve_client_information() + redirect_url = provider.redirect_url + + # Determine grant type based on server metadata + if not server_metadata: + raise ValueError("Failed to discover OAuth metadata from server") + + supported_grant_types = server_metadata.grant_types_supported or [] + + # Convert to lowercase for comparison + supported_grant_types_lower = [gt.lower() for gt in supported_grant_types] + + # Determine which grant type to use + effective_grant_type = None + if MCPSupportGrantType.AUTHORIZATION_CODE.value in supported_grant_types_lower: + effective_grant_type = MCPSupportGrantType.AUTHORIZATION_CODE.value + else: + effective_grant_type = MCPSupportGrantType.CLIENT_CREDENTIALS.value + + # Get stored credentials + credentials = provider.decrypt_credentials() - # Handle client registration if needed - client_information = provider.client_information() if not client_information: if authorization_code is not None: raise ValueError("Existing OAuth client information is required when exchanging an authorization code") + + # For client credentials flow, we don't need to register client dynamically + if effective_grant_type == MCPSupportGrantType.CLIENT_CREDENTIALS.value: + # Client should provide client_id and client_secret directly + raise ValueError("Client credentials flow requires client_id and client_secret to be provided") + try: - full_information = register_client(server_url, metadata, provider.client_metadata) - except httpx.RequestError as e: + full_information = register_client(server_url, server_metadata, client_metadata) + except RequestError as e: raise ValueError(f"Could not register OAuth client: {e}") - provider.save_client_information(full_information) + + # Return action to save client information + actions.append( + AuthAction( + action_type=AuthActionType.SAVE_CLIENT_INFO, + data={"client_information": full_information.model_dump()}, + provider_id=provider_id, + tenant_id=tenant_id, + ) + ) + client_information = full_information - # Exchange authorization code for tokens + # Handle client credentials flow + if effective_grant_type == MCPSupportGrantType.CLIENT_CREDENTIALS.value: + # Direct token request without user interaction + try: + scope = credentials.get("scope") + tokens = client_credentials_flow( + server_url, + server_metadata, + client_information, + scope, + ) + + # Return action to save tokens and grant type + token_data = tokens.model_dump() + token_data["grant_type"] = MCPSupportGrantType.CLIENT_CREDENTIALS.value + + actions.append( + AuthAction( + action_type=AuthActionType.SAVE_TOKENS, + data=token_data, + provider_id=provider_id, + tenant_id=tenant_id, + ) + ) + + return AuthResult(actions=actions, response={"result": "success"}) + except (RequestError, ValueError, KeyError) as e: + # RequestError: HTTP request failed + # ValueError: Invalid response data + # KeyError: Missing required fields in response + raise ValueError(f"Client credentials flow failed: {e}") + + # Exchange authorization code for tokens (Authorization Code flow) if authorization_code is not None: if not state_param: raise ValueError("State parameter is required when exchanging authorization code") @@ -335,35 +473,69 @@ def auth( tokens = exchange_authorization( server_url, - metadata, + server_metadata, client_information, authorization_code, code_verifier, redirect_uri, ) - provider.save_tokens(tokens) - return {"result": "success"} - provider_tokens = provider.tokens() + # Return action to save tokens + actions.append( + AuthAction( + action_type=AuthActionType.SAVE_TOKENS, + data=tokens.model_dump(), + provider_id=provider_id, + tenant_id=tenant_id, + ) + ) + + return AuthResult(actions=actions, response={"result": "success"}) + + provider_tokens = provider.retrieve_tokens() # Handle token refresh or new authorization if provider_tokens and provider_tokens.refresh_token: try: - new_tokens = refresh_authorization(server_url, metadata, client_information, provider_tokens.refresh_token) - provider.save_tokens(new_tokens) - return {"result": "success"} - except Exception as e: + new_tokens = refresh_authorization( + server_url, server_metadata, client_information, provider_tokens.refresh_token + ) + + # Return action to save new tokens + actions.append( + AuthAction( + action_type=AuthActionType.SAVE_TOKENS, + data=new_tokens.model_dump(), + provider_id=provider_id, + tenant_id=tenant_id, + ) + ) + + return AuthResult(actions=actions, response={"result": "success"}) + except (RequestError, ValueError, KeyError) as e: + # RequestError: HTTP request failed + # ValueError: Invalid response data + # KeyError: Missing required fields in response raise ValueError(f"Could not refresh OAuth tokens: {e}") - # Start new authorization flow + # Start new authorization flow (only for authorization code flow) authorization_url, code_verifier = start_authorization( server_url, - metadata, + server_metadata, client_information, - provider.redirect_url, - provider.mcp_provider.id, - provider.mcp_provider.tenant_id, + redirect_url, + provider_id, + tenant_id, ) - provider.save_code_verifier(code_verifier) - return {"authorization_url": authorization_url} + # Return action to save code verifier + actions.append( + AuthAction( + action_type=AuthActionType.SAVE_CODE_VERIFIER, + data={"code_verifier": code_verifier}, + provider_id=provider_id, + tenant_id=tenant_id, + ) + ) + + return AuthResult(actions=actions, response={"authorization_url": authorization_url}) diff --git a/api/core/mcp/auth/auth_provider.py b/api/core/mcp/auth/auth_provider.py deleted file mode 100644 index 3a550eb1b6..0000000000 --- a/api/core/mcp/auth/auth_provider.py +++ /dev/null @@ -1,77 +0,0 @@ -from configs import dify_config -from core.mcp.types import ( - OAuthClientInformation, - OAuthClientInformationFull, - OAuthClientMetadata, - OAuthTokens, -) -from models.tools import MCPToolProvider -from services.tools.mcp_tools_manage_service import MCPToolManageService - - -class OAuthClientProvider: - mcp_provider: MCPToolProvider - - def __init__(self, provider_id: str, tenant_id: str, for_list: bool = False): - if for_list: - self.mcp_provider = MCPToolManageService.get_mcp_provider_by_provider_id(provider_id, tenant_id) - else: - self.mcp_provider = MCPToolManageService.get_mcp_provider_by_server_identifier(provider_id, tenant_id) - - @property - def redirect_url(self) -> str: - """The URL to redirect the user agent to after authorization.""" - return dify_config.CONSOLE_API_URL + "/console/api/mcp/oauth/callback" - - @property - def client_metadata(self) -> OAuthClientMetadata: - """Metadata about this OAuth client.""" - return OAuthClientMetadata( - redirect_uris=[self.redirect_url], - token_endpoint_auth_method="none", - grant_types=["authorization_code", "refresh_token"], - response_types=["code"], - client_name="Dify", - client_uri="https://github.com/langgenius/dify", - ) - - def client_information(self) -> OAuthClientInformation | None: - """Loads information about this OAuth client.""" - client_information = self.mcp_provider.decrypted_credentials.get("client_information", {}) - if not client_information: - return None - return OAuthClientInformation.model_validate(client_information) - - def save_client_information(self, client_information: OAuthClientInformationFull): - """Saves client information after dynamic registration.""" - MCPToolManageService.update_mcp_provider_credentials( - self.mcp_provider, - {"client_information": client_information.model_dump()}, - ) - - def tokens(self) -> OAuthTokens | None: - """Loads any existing OAuth tokens for the current session.""" - credentials = self.mcp_provider.decrypted_credentials - if not credentials: - return None - return OAuthTokens( - access_token=credentials.get("access_token", ""), - token_type=credentials.get("token_type", "Bearer"), - expires_in=int(credentials.get("expires_in", "3600") or 3600), - refresh_token=credentials.get("refresh_token", ""), - ) - - def save_tokens(self, tokens: OAuthTokens): - """Stores new OAuth tokens for the current session.""" - # update mcp provider credentials - token_dict = tokens.model_dump() - MCPToolManageService.update_mcp_provider_credentials(self.mcp_provider, token_dict, authed=True) - - def save_code_verifier(self, code_verifier: str): - """Saves a PKCE code verifier for the current session.""" - MCPToolManageService.update_mcp_provider_credentials(self.mcp_provider, {"code_verifier": code_verifier}) - - def code_verifier(self) -> str: - """Loads the PKCE code verifier for the current session.""" - # get code verifier from mcp provider credentials - return str(self.mcp_provider.decrypted_credentials.get("code_verifier", "")) diff --git a/api/core/mcp/auth_client.py b/api/core/mcp/auth_client.py new file mode 100644 index 0000000000..942c8d3c23 --- /dev/null +++ b/api/core/mcp/auth_client.py @@ -0,0 +1,191 @@ +""" +MCP Client with Authentication Retry Support + +This module provides an enhanced MCPClient that automatically handles +authentication failures and retries operations after refreshing tokens. +""" + +import logging +from collections.abc import Callable +from typing import Any + +from sqlalchemy.orm import Session + +from core.entities.mcp_provider import MCPProviderEntity +from core.mcp.error import MCPAuthError +from core.mcp.mcp_client import MCPClient +from core.mcp.types import CallToolResult, Tool +from extensions.ext_database import db + +logger = logging.getLogger(__name__) + + +class MCPClientWithAuthRetry(MCPClient): + """ + An enhanced MCPClient that provides automatic authentication retry. + + This class extends MCPClient and intercepts MCPAuthError exceptions + to refresh authentication before retrying failed operations. + + Note: This class uses lazy session creation - database sessions are only + created when authentication retry is actually needed, not on every request. + """ + + def __init__( + self, + server_url: str, + headers: dict[str, str] | None = None, + timeout: float | None = None, + sse_read_timeout: float | None = None, + provider_entity: MCPProviderEntity | None = None, + authorization_code: str | None = None, + by_server_id: bool = False, + ): + """ + Initialize the MCP client with auth retry capability. + + Args: + server_url: The MCP server URL + headers: Optional headers for requests + timeout: Request timeout + sse_read_timeout: SSE read timeout + provider_entity: Provider entity for authentication + authorization_code: Optional authorization code for initial auth + by_server_id: Whether to look up provider by server ID + """ + super().__init__(server_url, headers, timeout, sse_read_timeout) + + self.provider_entity = provider_entity + self.authorization_code = authorization_code + self.by_server_id = by_server_id + self._has_retried = False + + def _handle_auth_error(self, error: MCPAuthError) -> None: + """ + Handle authentication error by refreshing tokens. + + This method creates a short-lived database session only when authentication + retry is needed, minimizing database connection hold time. + + Args: + error: The authentication error + + Raises: + MCPAuthError: If authentication fails or max retries reached + """ + if not self.provider_entity: + raise error + if self._has_retried: + raise error + + self._has_retried = True + + try: + # Create a temporary session only for auth retry + # This session is short-lived and only exists during the auth operation + + from services.tools.mcp_tools_manage_service import MCPToolManageService + + with Session(db.engine) as session, session.begin(): + mcp_service = MCPToolManageService(session=session) + + # Perform authentication using the service's auth method + mcp_service.auth_with_actions(self.provider_entity, self.authorization_code) + + # Retrieve new tokens + self.provider_entity = mcp_service.get_provider_entity( + self.provider_entity.id, self.provider_entity.tenant_id, by_server_id=self.by_server_id + ) + + # Session is closed here, before we update headers + token = self.provider_entity.retrieve_tokens() + if not token: + raise MCPAuthError("Authentication failed - no token received") + + # Update headers with new token + self.headers["Authorization"] = f"{token.token_type.capitalize()} {token.access_token}" + + # Clear authorization code after first use + self.authorization_code = None + + except MCPAuthError: + # Re-raise MCPAuthError as is + raise + except Exception as e: + # Catch all exceptions during auth retry + logger.exception("Authentication retry failed") + raise MCPAuthError(f"Authentication retry failed: {e}") from e + + def _execute_with_retry(self, func: Callable[..., Any], *args, **kwargs) -> Any: + """ + Execute a function with authentication retry logic. + + Args: + func: The function to execute + *args: Positional arguments for the function + **kwargs: Keyword arguments for the function + + Returns: + The result of the function call + + Raises: + MCPAuthError: If authentication fails after retries + Any other exceptions from the function + """ + try: + return func(*args, **kwargs) + except MCPAuthError as e: + self._handle_auth_error(e) + + # Re-initialize the connection with new headers + if self._initialized: + # Clean up existing connection + self._exit_stack.close() + self._session = None + self._initialized = False + + # Re-initialize with new headers + self._initialize() + self._initialized = True + + return func(*args, **kwargs) + finally: + # Reset retry flag after operation completes + self._has_retried = False + + def __enter__(self): + """Enter the context manager with retry support.""" + + def initialize_with_retry(): + super(MCPClientWithAuthRetry, self).__enter__() + return self + + return self._execute_with_retry(initialize_with_retry) + + def list_tools(self) -> list[Tool]: + """ + List available tools from the MCP server with auth retry. + + Returns: + List of available tools + + Raises: + MCPAuthError: If authentication fails after retries + """ + return self._execute_with_retry(super().list_tools) + + def invoke_tool(self, tool_name: str, tool_args: dict[str, Any]) -> CallToolResult: + """ + Invoke a tool on the MCP server with auth retry. + + Args: + tool_name: Name of the tool to invoke + tool_args: Arguments for the tool + + Returns: + Result of the tool invocation + + Raises: + MCPAuthError: If authentication fails after retries + """ + return self._execute_with_retry(super().invoke_tool, tool_name, tool_args) diff --git a/api/core/workflow/nodes/enums.py b/api/core/mcp/auth_client_comparison.md similarity index 100% rename from api/core/workflow/nodes/enums.py rename to api/core/mcp/auth_client_comparison.md diff --git a/api/core/mcp/client/sse_client.py b/api/core/mcp/client/sse_client.py index 6db22a09e0..2d5e3dd263 100644 --- a/api/core/mcp/client/sse_client.py +++ b/api/core/mcp/client/sse_client.py @@ -46,7 +46,7 @@ class SSETransport: url: str, headers: dict[str, Any] | None = None, timeout: float = 5.0, - sse_read_timeout: float = 5 * 60, + sse_read_timeout: float = 1 * 60, ): """Initialize the SSE transport. @@ -255,7 +255,7 @@ def sse_client( url: str, headers: dict[str, Any] | None = None, timeout: float = 5.0, - sse_read_timeout: float = 5 * 60, + sse_read_timeout: float = 1 * 60, ) -> Generator[tuple[ReadQueue, WriteQueue], None, None]: """ Client transport for SSE. @@ -276,31 +276,34 @@ def sse_client( read_queue: ReadQueue | None = None write_queue: WriteQueue | None = None - with ThreadPoolExecutor() as executor: - try: - with create_ssrf_proxy_mcp_http_client(headers=transport.headers) as client: - with ssrf_proxy_sse_connect( - url, timeout=httpx.Timeout(timeout, read=sse_read_timeout), client=client - ) as event_source: - event_source.response.raise_for_status() + executor = ThreadPoolExecutor() + try: + with create_ssrf_proxy_mcp_http_client(headers=transport.headers) as client: + with ssrf_proxy_sse_connect( + url, timeout=httpx.Timeout(timeout, read=sse_read_timeout), client=client + ) as event_source: + event_source.response.raise_for_status() - read_queue, write_queue = transport.connect(executor, client, event_source) + read_queue, write_queue = transport.connect(executor, client, event_source) - yield read_queue, write_queue + yield read_queue, write_queue - except httpx.HTTPStatusError as exc: - if exc.response.status_code == 401: - raise MCPAuthError() - raise MCPConnectionError() - except Exception: - logger.exception("Error connecting to SSE endpoint") - raise - finally: - # Clean up queues - if read_queue: - read_queue.put(None) - if write_queue: - write_queue.put(None) + except httpx.HTTPStatusError as exc: + if exc.response.status_code == 401: + raise MCPAuthError() + raise MCPConnectionError() + except Exception: + logger.exception("Error connecting to SSE endpoint") + raise + finally: + # Clean up queues + if read_queue: + read_queue.put(None) + if write_queue: + write_queue.put(None) + + # Shutdown executor without waiting to prevent hanging + executor.shutdown(wait=False) def send_message(http_client: httpx.Client, endpoint_url: str, session_message: SessionMessage): diff --git a/api/core/mcp/client/streamable_client.py b/api/core/mcp/client/streamable_client.py index 7eafa79837..805c16c838 100644 --- a/api/core/mcp/client/streamable_client.py +++ b/api/core/mcp/client/streamable_client.py @@ -138,6 +138,10 @@ class StreamableHTTPTransport: ) -> bool: """Handle an SSE event, returning True if the response is complete.""" if sse.event == "message": + # ping event send by server will be recognized as a message event with empty data by httpx-sse's SSEDecoder + if not sse.data.strip(): + return False + try: message = JSONRPCMessage.model_validate_json(sse.data) logger.debug("SSE message: %s", message) @@ -434,45 +438,48 @@ def streamablehttp_client( server_to_client_queue: ServerToClientQueue = queue.Queue() # For messages FROM server TO client client_to_server_queue: ClientToServerQueue = queue.Queue() # For messages FROM client TO server - with ThreadPoolExecutor(max_workers=2) as executor: - try: - with create_ssrf_proxy_mcp_http_client( - headers=transport.request_headers, - timeout=httpx.Timeout(transport.timeout, read=transport.sse_read_timeout), - ) as client: - # Define callbacks that need access to thread pool - def start_get_stream(): - """Start a worker thread to handle server-initiated messages.""" - executor.submit(transport.handle_get_stream, client, server_to_client_queue) + executor = ThreadPoolExecutor(max_workers=2) + try: + with create_ssrf_proxy_mcp_http_client( + headers=transport.request_headers, + timeout=httpx.Timeout(transport.timeout, read=transport.sse_read_timeout), + ) as client: + # Define callbacks that need access to thread pool + def start_get_stream(): + """Start a worker thread to handle server-initiated messages.""" + executor.submit(transport.handle_get_stream, client, server_to_client_queue) - # Start the post_writer worker thread - executor.submit( - transport.post_writer, - client, - client_to_server_queue, # Queue for messages FROM client TO server - server_to_client_queue, # Queue for messages FROM server TO client - start_get_stream, - ) + # Start the post_writer worker thread + executor.submit( + transport.post_writer, + client, + client_to_server_queue, # Queue for messages FROM client TO server + server_to_client_queue, # Queue for messages FROM server TO client + start_get_stream, + ) - try: - yield ( - server_to_client_queue, # Queue for receiving messages FROM server - client_to_server_queue, # Queue for sending messages TO server - transport.get_session_id, - ) - finally: - if transport.session_id and terminate_on_close: - transport.terminate_session(client) - - # Signal threads to stop - client_to_server_queue.put(None) - finally: - # Clear any remaining items and add None sentinel to unblock any waiting threads try: - while not client_to_server_queue.empty(): - client_to_server_queue.get_nowait() - except queue.Empty: - pass + yield ( + server_to_client_queue, # Queue for receiving messages FROM server + client_to_server_queue, # Queue for sending messages TO server + transport.get_session_id, + ) + finally: + if transport.session_id and terminate_on_close: + transport.terminate_session(client) - client_to_server_queue.put(None) - server_to_client_queue.put(None) + # Signal threads to stop + client_to_server_queue.put(None) + finally: + # Clear any remaining items and add None sentinel to unblock any waiting threads + try: + while not client_to_server_queue.empty(): + client_to_server_queue.get_nowait() + except queue.Empty: + pass + + client_to_server_queue.put(None) + server_to_client_queue.put(None) + + # Shutdown executor without waiting to prevent hanging + executor.shutdown(wait=False) diff --git a/api/core/mcp/entities.py b/api/core/mcp/entities.py index 7553c10a2e..08823daab1 100644 --- a/api/core/mcp/entities.py +++ b/api/core/mcp/entities.py @@ -1,10 +1,13 @@ from dataclasses import dataclass +from enum import StrEnum from typing import Any, Generic, TypeVar -from core.mcp.session.base_session import BaseSession -from core.mcp.types import LATEST_PROTOCOL_VERSION, RequestId, RequestParams +from pydantic import BaseModel -SUPPORTED_PROTOCOL_VERSIONS: list[str] = ["2024-11-05", LATEST_PROTOCOL_VERSION] +from core.mcp.session.base_session import BaseSession +from core.mcp.types import LATEST_PROTOCOL_VERSION, OAuthClientInformation, OAuthMetadata, RequestId, RequestParams + +SUPPORTED_PROTOCOL_VERSIONS: list[str] = ["2024-11-05", "2025-03-26", LATEST_PROTOCOL_VERSION] SessionT = TypeVar("SessionT", bound=BaseSession[Any, Any, Any, Any, Any]) @@ -17,3 +20,41 @@ class RequestContext(Generic[SessionT, LifespanContextT]): meta: RequestParams.Meta | None session: SessionT lifespan_context: LifespanContextT + + +class AuthActionType(StrEnum): + """Types of actions that can be performed during auth flow.""" + + SAVE_CLIENT_INFO = "save_client_info" + SAVE_TOKENS = "save_tokens" + SAVE_CODE_VERIFIER = "save_code_verifier" + START_AUTHORIZATION = "start_authorization" + SUCCESS = "success" + + +class AuthAction(BaseModel): + """Represents an action that needs to be performed as a result of auth flow.""" + + action_type: AuthActionType + data: dict[str, Any] + provider_id: str | None = None + tenant_id: str | None = None + + +class AuthResult(BaseModel): + """Result of auth function containing actions to be performed and response data.""" + + actions: list[AuthAction] + response: dict[str, str] + + +class OAuthCallbackState(BaseModel): + """State data stored in Redis during OAuth callback flow.""" + + provider_id: str + tenant_id: str + server_url: str + metadata: OAuthMetadata | None = None + client_information: OAuthClientInformation + code_verifier: str + redirect_uri: str diff --git a/api/core/mcp/error.py b/api/core/mcp/error.py index 92ea7bde09..d4fb8b7674 100644 --- a/api/core/mcp/error.py +++ b/api/core/mcp/error.py @@ -8,3 +8,7 @@ class MCPConnectionError(MCPError): class MCPAuthError(MCPConnectionError): pass + + +class MCPRefreshTokenError(MCPError): + pass diff --git a/api/core/mcp/mcp_client.py b/api/core/mcp/mcp_client.py index 86ec2c4db9..b0e0dab9be 100644 --- a/api/core/mcp/mcp_client.py +++ b/api/core/mcp/mcp_client.py @@ -7,9 +7,9 @@ from urllib.parse import urlparse from core.mcp.client.sse_client import sse_client from core.mcp.client.streamable_client import streamablehttp_client -from core.mcp.error import MCPAuthError, MCPConnectionError +from core.mcp.error import MCPConnectionError from core.mcp.session.client_session import ClientSession -from core.mcp.types import Tool +from core.mcp.types import CallToolResult, Tool logger = logging.getLogger(__name__) @@ -18,40 +18,18 @@ class MCPClient: def __init__( self, server_url: str, - provider_id: str, - tenant_id: str, - authed: bool = True, - authorization_code: str | None = None, - for_list: bool = False, headers: dict[str, str] | None = None, timeout: float | None = None, sse_read_timeout: float | None = None, ): - # Initialize info - self.provider_id = provider_id - self.tenant_id = tenant_id - self.client_type = "streamable" self.server_url = server_url self.headers = headers or {} self.timeout = timeout self.sse_read_timeout = sse_read_timeout - # Authentication info - self.authed = authed - self.authorization_code = authorization_code - if authed: - from core.mcp.auth.auth_provider import OAuthClientProvider - - self.provider = OAuthClientProvider(self.provider_id, self.tenant_id, for_list=for_list) - self.token = self.provider.tokens() - # Initialize session and client objects self._session: ClientSession | None = None - self._streams_context: AbstractContextManager[Any] | None = None - self._session_context: ClientSession | None = None self._exit_stack = ExitStack() - - # Whether the client has been initialized self._initialized = False def __enter__(self): @@ -85,61 +63,42 @@ class MCPClient: logger.debug("MCP connection failed with 'sse', falling back to 'mcp' method.") self.connect_server(streamablehttp_client, "mcp") - def connect_server( - self, client_factory: Callable[..., AbstractContextManager[Any]], method_name: str, first_try: bool = True - ): - from core.mcp.auth.auth_flow import auth + def connect_server(self, client_factory: Callable[..., AbstractContextManager[Any]], method_name: str) -> None: + """ + Connect to the MCP server using streamable http or sse. + Default to streamable http. + Args: + client_factory: The client factory to use(streamablehttp_client or sse_client). + method_name: The method name to use(mcp or sse). + """ + streams_context = client_factory( + url=self.server_url, + headers=self.headers, + timeout=self.timeout, + sse_read_timeout=self.sse_read_timeout, + ) - try: - headers = ( - {"Authorization": f"{self.token.token_type.capitalize()} {self.token.access_token}"} - if self.authed and self.token - else self.headers - ) - self._streams_context = client_factory( - url=self.server_url, - headers=headers, - timeout=self.timeout, - sse_read_timeout=self.sse_read_timeout, - ) - if not self._streams_context: - raise MCPConnectionError("Failed to create connection context") + # Use exit_stack to manage context managers properly + if method_name == "mcp": + read_stream, write_stream, _ = self._exit_stack.enter_context(streams_context) + streams = (read_stream, write_stream) + else: # sse_client + streams = self._exit_stack.enter_context(streams_context) - # Use exit_stack to manage context managers properly - if method_name == "mcp": - read_stream, write_stream, _ = self._exit_stack.enter_context(self._streams_context) - streams = (read_stream, write_stream) - else: # sse_client - streams = self._exit_stack.enter_context(self._streams_context) - - self._session_context = ClientSession(*streams) - self._session = self._exit_stack.enter_context(self._session_context) - self._session.initialize() - return - - except MCPAuthError: - if not self.authed: - raise - try: - auth(self.provider, self.server_url, self.authorization_code) - except Exception as e: - raise ValueError(f"Failed to authenticate: {e}") - self.token = self.provider.tokens() - if first_try: - return self.connect_server(client_factory, method_name, first_try=False) + session_context = ClientSession(*streams) + self._session = self._exit_stack.enter_context(session_context) + self._session.initialize() def list_tools(self) -> list[Tool]: - """Connect to an MCP server running with SSE transport""" - # List available tools to verify connection - if not self._initialized or not self._session: + """List available tools from the MCP server""" + if not self._session: raise ValueError("Session not initialized.") response = self._session.list_tools() - tools = response.tools - return tools + return response.tools - def invoke_tool(self, tool_name: str, tool_args: dict): + def invoke_tool(self, tool_name: str, tool_args: dict[str, Any]) -> CallToolResult: """Call a tool""" - if not self._initialized or not self._session: + if not self._session: raise ValueError("Session not initialized.") return self._session.call_tool(tool_name, tool_args) @@ -153,6 +112,4 @@ class MCPClient: raise ValueError(f"Error during cleanup: {e}") finally: self._session = None - self._session_context = None - self._streams_context = None self._initialized = False diff --git a/api/core/mcp/session/base_session.py b/api/core/mcp/session/base_session.py index 653b3773c0..3dcd166ea2 100644 --- a/api/core/mcp/session/base_session.py +++ b/api/core/mcp/session/base_session.py @@ -201,11 +201,14 @@ class BaseSession( self._receiver_future.result(timeout=5.0) # Wait up to 5 seconds except TimeoutError: # If the receiver loop is still running after timeout, we'll force shutdown - pass + # Cancel the future to interrupt the receiver loop + self._receiver_future.cancel() # Shutdown the executor if self._executor: - self._executor.shutdown(wait=True) + # Use non-blocking shutdown to prevent hanging + # The receiver thread should have already exited due to the None message in the queue + self._executor.shutdown(wait=False) def send_request( self, diff --git a/api/core/mcp/session/client_session.py b/api/core/mcp/session/client_session.py index fa1d309134..d684fe0dd7 100644 --- a/api/core/mcp/session/client_session.py +++ b/api/core/mcp/session/client_session.py @@ -109,12 +109,16 @@ class ClientSession( self._message_handler = message_handler or _default_message_handler def initialize(self) -> types.InitializeResult: - sampling = types.SamplingCapability() - roots = types.RootsCapability( - # TODO: Should this be based on whether we - # _will_ send notifications, or only whether - # they're supported? - listChanged=True, + # Only set capabilities if non-default callbacks are provided + # This prevents servers from attempting callbacks when we don't actually support them + sampling = types.SamplingCapability() if self._sampling_callback is not _default_sampling_callback else None + roots = ( + types.RootsCapability( + # Only enable listChanged if we have a custom callback + listChanged=True, + ) + if self._list_roots_callback is not _default_list_roots_callback + else None ) result = self.send_request( @@ -284,7 +288,7 @@ class ClientSession( def complete( self, - ref: types.ResourceReference | types.PromptReference, + ref: types.ResourceTemplateReference | types.PromptReference, argument: dict[str, str], ) -> types.CompleteResult: """Send a completion/complete request.""" diff --git a/api/core/mcp/types.py b/api/core/mcp/types.py index c7a046b585..fd2062d2e1 100644 --- a/api/core/mcp/types.py +++ b/api/core/mcp/types.py @@ -1,13 +1,6 @@ from collections.abc import Callable from dataclasses import dataclass -from typing import ( - Annotated, - Any, - Generic, - Literal, - TypeAlias, - TypeVar, -) +from typing import Annotated, Any, Generic, Literal, TypeAlias, TypeVar from pydantic import BaseModel, ConfigDict, Field, FileUrl, RootModel from pydantic.networks import AnyUrl, UrlConstraints @@ -33,6 +26,7 @@ for reference. LATEST_PROTOCOL_VERSION = "2025-03-26" # Server support 2024-11-05 to allow claude to use. SERVER_LATEST_PROTOCOL_VERSION = "2024-11-05" +DEFAULT_NEGOTIATED_VERSION = "2025-03-26" ProgressToken = str | int Cursor = str Role = Literal["user", "assistant"] @@ -55,14 +49,22 @@ class RequestParams(BaseModel): meta: Meta | None = Field(alias="_meta", default=None) +class PaginatedRequestParams(RequestParams): + cursor: Cursor | None = None + """ + An opaque token representing the current pagination position. + If provided, the server should return results starting after this cursor. + """ + + class NotificationParams(BaseModel): class Meta(BaseModel): model_config = ConfigDict(extra="allow") meta: Meta | None = Field(alias="_meta", default=None) """ - This parameter name is reserved by MCP to allow clients and servers to attach - additional metadata to their notifications. + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. """ @@ -79,12 +81,11 @@ class Request(BaseModel, Generic[RequestParamsT, MethodT]): model_config = ConfigDict(extra="allow") -class PaginatedRequest(Request[RequestParamsT, MethodT]): - cursor: Cursor | None = None - """ - An opaque token representing the current pagination position. - If provided, the server should return results starting after this cursor. - """ +class PaginatedRequest(Request[PaginatedRequestParams | None, MethodT], Generic[MethodT]): + """Base class for paginated requests, + matching the schema's PaginatedRequest interface.""" + + params: PaginatedRequestParams | None = None class Notification(BaseModel, Generic[NotificationParamsT, MethodT]): @@ -98,13 +99,12 @@ class Notification(BaseModel, Generic[NotificationParamsT, MethodT]): class Result(BaseModel): """Base class for JSON-RPC results.""" - model_config = ConfigDict(extra="allow") - meta: dict[str, Any] | None = Field(alias="_meta", default=None) """ - This result property is reserved by the protocol to allow clients and servers to - attach additional metadata to their responses. + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. """ + model_config = ConfigDict(extra="allow") class PaginatedResult(Result): @@ -186,10 +186,26 @@ class EmptyResult(Result): """A response that indicates success but carries no data.""" -class Implementation(BaseModel): - """Describes the name and version of an MCP implementation.""" +class BaseMetadata(BaseModel): + """Base class for entities with name and optional title fields.""" name: str + """The programmatic name of the entity.""" + + title: str | None = None + """ + Intended for UI and end-user contexts — optimized to be human-readable and easily understood, + even by those unfamiliar with domain-specific terminology. + + If not provided, the name should be used for display (except for Tool, + where `annotations.title` should be given precedence over using `name`, + if present). + """ + + +class Implementation(BaseMetadata): + """Describes the name and version of an MCP implementation.""" + version: str model_config = ConfigDict(extra="allow") @@ -203,7 +219,7 @@ class RootsCapability(BaseModel): class SamplingCapability(BaseModel): - """Capability for logging operations.""" + """Capability for sampling operations.""" model_config = ConfigDict(extra="allow") @@ -252,6 +268,12 @@ class LoggingCapability(BaseModel): model_config = ConfigDict(extra="allow") +class CompletionsCapability(BaseModel): + """Capability for completions operations.""" + + model_config = ConfigDict(extra="allow") + + class ServerCapabilities(BaseModel): """Capabilities that a server may support.""" @@ -265,6 +287,8 @@ class ServerCapabilities(BaseModel): """Present if the server offers any resources to read.""" tools: ToolsCapability | None = None """Present if the server offers any tools to call.""" + completions: CompletionsCapability | None = None + """Present if the server offers autocompletion suggestions for prompts and resources.""" model_config = ConfigDict(extra="allow") @@ -284,7 +308,7 @@ class InitializeRequest(Request[InitializeRequestParams, Literal["initialize"]]) to begin initialization. """ - method: Literal["initialize"] + method: Literal["initialize"] = "initialize" params: InitializeRequestParams @@ -305,7 +329,7 @@ class InitializedNotification(Notification[NotificationParams | None, Literal["n finished. """ - method: Literal["notifications/initialized"] + method: Literal["notifications/initialized"] = "notifications/initialized" params: NotificationParams | None = None @@ -315,7 +339,7 @@ class PingRequest(Request[RequestParams | None, Literal["ping"]]): still alive. """ - method: Literal["ping"] + method: Literal["ping"] = "ping" params: RequestParams | None = None @@ -334,6 +358,11 @@ class ProgressNotificationParams(NotificationParams): """ total: float | None = None """Total number of items to process (or total progress required), if known.""" + message: str | None = None + """ + Message related to progress. This should provide relevant human readable + progress information. + """ model_config = ConfigDict(extra="allow") @@ -343,15 +372,14 @@ class ProgressNotification(Notification[ProgressNotificationParams, Literal["not long-running request. """ - method: Literal["notifications/progress"] + method: Literal["notifications/progress"] = "notifications/progress" params: ProgressNotificationParams -class ListResourcesRequest(PaginatedRequest[RequestParams | None, Literal["resources/list"]]): +class ListResourcesRequest(PaginatedRequest[Literal["resources/list"]]): """Sent from the client to request a list of resources the server has.""" - method: Literal["resources/list"] - params: RequestParams | None = None + method: Literal["resources/list"] = "resources/list" class Annotations(BaseModel): @@ -360,13 +388,11 @@ class Annotations(BaseModel): model_config = ConfigDict(extra="allow") -class Resource(BaseModel): +class Resource(BaseMetadata): """A known resource that the server is capable of reading.""" uri: Annotated[AnyUrl, UrlConstraints(host_required=False)] """The URI of this resource.""" - name: str - """A human-readable name for this resource.""" description: str | None = None """A description of what this resource represents.""" mimeType: str | None = None @@ -379,10 +405,15 @@ class Resource(BaseModel): This can be used by Hosts to display file sizes and estimate context window usage. """ annotations: Annotations | None = None + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") -class ResourceTemplate(BaseModel): +class ResourceTemplate(BaseMetadata): """A template description for resources available on the server.""" uriTemplate: str @@ -390,8 +421,6 @@ class ResourceTemplate(BaseModel): A URI template (according to RFC 6570) that can be used to construct resource URIs. """ - name: str - """A human-readable name for the type of resource this template refers to.""" description: str | None = None """A human-readable description of what this template is for.""" mimeType: str | None = None @@ -400,6 +429,11 @@ class ResourceTemplate(BaseModel): included if all resources matching this template have the same type. """ annotations: Annotations | None = None + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") @@ -409,11 +443,10 @@ class ListResourcesResult(PaginatedResult): resources: list[Resource] -class ListResourceTemplatesRequest(PaginatedRequest[RequestParams | None, Literal["resources/templates/list"]]): +class ListResourceTemplatesRequest(PaginatedRequest[Literal["resources/templates/list"]]): """Sent from the client to request a list of resource templates the server has.""" - method: Literal["resources/templates/list"] - params: RequestParams | None = None + method: Literal["resources/templates/list"] = "resources/templates/list" class ListResourceTemplatesResult(PaginatedResult): @@ -436,7 +469,7 @@ class ReadResourceRequestParams(RequestParams): class ReadResourceRequest(Request[ReadResourceRequestParams, Literal["resources/read"]]): """Sent from the client to the server, to read a specific resource URI.""" - method: Literal["resources/read"] + method: Literal["resources/read"] = "resources/read" params: ReadResourceRequestParams @@ -447,6 +480,11 @@ class ResourceContents(BaseModel): """The URI of this resource.""" mimeType: str | None = None """The MIME type of this resource, if known.""" + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") @@ -481,7 +519,7 @@ class ResourceListChangedNotification( of resources it can read from has changed. """ - method: Literal["notifications/resources/list_changed"] + method: Literal["notifications/resources/list_changed"] = "notifications/resources/list_changed" params: NotificationParams | None = None @@ -502,7 +540,7 @@ class SubscribeRequest(Request[SubscribeRequestParams, Literal["resources/subscr whenever a particular resource changes. """ - method: Literal["resources/subscribe"] + method: Literal["resources/subscribe"] = "resources/subscribe" params: SubscribeRequestParams @@ -520,7 +558,7 @@ class UnsubscribeRequest(Request[UnsubscribeRequestParams, Literal["resources/un the server. """ - method: Literal["resources/unsubscribe"] + method: Literal["resources/unsubscribe"] = "resources/unsubscribe" params: UnsubscribeRequestParams @@ -543,15 +581,14 @@ class ResourceUpdatedNotification( changed and may need to be read again. """ - method: Literal["notifications/resources/updated"] + method: Literal["notifications/resources/updated"] = "notifications/resources/updated" params: ResourceUpdatedNotificationParams -class ListPromptsRequest(PaginatedRequest[RequestParams | None, Literal["prompts/list"]]): +class ListPromptsRequest(PaginatedRequest[Literal["prompts/list"]]): """Sent from the client to request a list of prompts and prompt templates.""" - method: Literal["prompts/list"] - params: RequestParams | None = None + method: Literal["prompts/list"] = "prompts/list" class PromptArgument(BaseModel): @@ -566,15 +603,18 @@ class PromptArgument(BaseModel): model_config = ConfigDict(extra="allow") -class Prompt(BaseModel): +class Prompt(BaseMetadata): """A prompt or prompt template that the server offers.""" - name: str - """The name of the prompt or prompt template.""" description: str | None = None """An optional description of what this prompt provides.""" arguments: list[PromptArgument] | None = None """A list of arguments to use for templating the prompt.""" + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") @@ -597,7 +637,7 @@ class GetPromptRequestParams(RequestParams): class GetPromptRequest(Request[GetPromptRequestParams, Literal["prompts/get"]]): """Used by the client to get a prompt provided by the server.""" - method: Literal["prompts/get"] + method: Literal["prompts/get"] = "prompts/get" params: GetPromptRequestParams @@ -608,6 +648,11 @@ class TextContent(BaseModel): text: str """The text content of the message.""" annotations: Annotations | None = None + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") @@ -623,6 +668,31 @@ class ImageContent(BaseModel): image types. """ annotations: Annotations | None = None + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ + model_config = ConfigDict(extra="allow") + + +class AudioContent(BaseModel): + """Audio content for a message.""" + + type: Literal["audio"] + data: str + """The base64-encoded audio data.""" + mimeType: str + """ + The MIME type of the audio. Different providers may support different + audio types. + """ + annotations: Annotations | None = None + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") @@ -630,7 +700,7 @@ class SamplingMessage(BaseModel): """Describes a message issued to or received from an LLM API.""" role: Role - content: TextContent | ImageContent + content: TextContent | ImageContent | AudioContent model_config = ConfigDict(extra="allow") @@ -645,14 +715,36 @@ class EmbeddedResource(BaseModel): type: Literal["resource"] resource: TextResourceContents | BlobResourceContents annotations: Annotations | None = None + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") +class ResourceLink(Resource): + """ + A resource that the server is capable of reading, included in a prompt or tool call result. + + Note: resource links returned by tools are not guaranteed to appear in the results of `resources/list` requests. + """ + + type: Literal["resource_link"] + + +ContentBlock = TextContent | ImageContent | AudioContent | ResourceLink | EmbeddedResource +"""A content block that can be used in prompts and tool results.""" + +Content: TypeAlias = ContentBlock +# """DEPRECATED: Content is deprecated, you should use ContentBlock directly.""" + + class PromptMessage(BaseModel): """Describes a message returned as part of a prompt.""" role: Role - content: TextContent | ImageContent | EmbeddedResource + content: ContentBlock model_config = ConfigDict(extra="allow") @@ -672,15 +764,14 @@ class PromptListChangedNotification( of prompts it offers has changed. """ - method: Literal["notifications/prompts/list_changed"] + method: Literal["notifications/prompts/list_changed"] = "notifications/prompts/list_changed" params: NotificationParams | None = None -class ListToolsRequest(PaginatedRequest[RequestParams | None, Literal["tools/list"]]): +class ListToolsRequest(PaginatedRequest[Literal["tools/list"]]): """Sent from the client to request a list of tools the server has.""" - method: Literal["tools/list"] - params: RequestParams | None = None + method: Literal["tools/list"] = "tools/list" class ToolAnnotations(BaseModel): @@ -731,17 +822,25 @@ class ToolAnnotations(BaseModel): model_config = ConfigDict(extra="allow") -class Tool(BaseModel): +class Tool(BaseMetadata): """Definition for a tool the client can call.""" - name: str - """The name of the tool.""" description: str | None = None """A human-readable description of the tool.""" inputSchema: dict[str, Any] """A JSON Schema object defining the expected parameters for the tool.""" + outputSchema: dict[str, Any] | None = None + """ + An optional JSON Schema object defining the structure of the tool's output + returned in the structuredContent field of a CallToolResult. + """ annotations: ToolAnnotations | None = None """Optional additional tool information.""" + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") @@ -762,14 +861,16 @@ class CallToolRequestParams(RequestParams): class CallToolRequest(Request[CallToolRequestParams, Literal["tools/call"]]): """Used by the client to invoke a tool provided by the server.""" - method: Literal["tools/call"] + method: Literal["tools/call"] = "tools/call" params: CallToolRequestParams class CallToolResult(Result): """The server's response to a tool call.""" - content: list[TextContent | ImageContent | EmbeddedResource] + content: list[ContentBlock] + structuredContent: dict[str, Any] | None = None + """An optional JSON object that represents the structured result of the tool call.""" isError: bool = False @@ -779,7 +880,7 @@ class ToolListChangedNotification(Notification[NotificationParams | None, Litera of tools it offers has changed. """ - method: Literal["notifications/tools/list_changed"] + method: Literal["notifications/tools/list_changed"] = "notifications/tools/list_changed" params: NotificationParams | None = None @@ -797,7 +898,7 @@ class SetLevelRequestParams(RequestParams): class SetLevelRequest(Request[SetLevelRequestParams, Literal["logging/setLevel"]]): """A request from the client to the server, to enable or adjust logging.""" - method: Literal["logging/setLevel"] + method: Literal["logging/setLevel"] = "logging/setLevel" params: SetLevelRequestParams @@ -808,7 +909,7 @@ class LoggingMessageNotificationParams(NotificationParams): """The severity of this log message.""" logger: str | None = None """An optional name of the logger issuing this message.""" - data: Any = None + data: Any """ The data to be logged, such as a string message or an object. Any JSON serializable type is allowed here. @@ -819,7 +920,7 @@ class LoggingMessageNotificationParams(NotificationParams): class LoggingMessageNotification(Notification[LoggingMessageNotificationParams, Literal["notifications/message"]]): """Notification of a log message passed from server to client.""" - method: Literal["notifications/message"] + method: Literal["notifications/message"] = "notifications/message" params: LoggingMessageNotificationParams @@ -914,7 +1015,7 @@ class CreateMessageRequestParams(RequestParams): class CreateMessageRequest(Request[CreateMessageRequestParams, Literal["sampling/createMessage"]]): """A request from the server to sample an LLM via the client.""" - method: Literal["sampling/createMessage"] + method: Literal["sampling/createMessage"] = "sampling/createMessage" params: CreateMessageRequestParams @@ -925,14 +1026,14 @@ class CreateMessageResult(Result): """The client's response to a sampling/create_message request from the server.""" role: Role - content: TextContent | ImageContent + content: TextContent | ImageContent | AudioContent model: str """The name of the model that generated the message.""" stopReason: StopReason | None = None """The reason why sampling stopped, if known.""" -class ResourceReference(BaseModel): +class ResourceTemplateReference(BaseModel): """A reference to a resource or resource template definition.""" type: Literal["ref/resource"] @@ -960,18 +1061,28 @@ class CompletionArgument(BaseModel): model_config = ConfigDict(extra="allow") +class CompletionContext(BaseModel): + """Additional, optional context for completions.""" + + arguments: dict[str, str] | None = None + """Previously-resolved variables in a URI template or prompt.""" + model_config = ConfigDict(extra="allow") + + class CompleteRequestParams(RequestParams): """Parameters for completion requests.""" - ref: ResourceReference | PromptReference + ref: ResourceTemplateReference | PromptReference argument: CompletionArgument + context: CompletionContext | None = None + """Additional, optional context for completions""" model_config = ConfigDict(extra="allow") class CompleteRequest(Request[CompleteRequestParams, Literal["completion/complete"]]): """A request from the client to the server, to ask for completion options.""" - method: Literal["completion/complete"] + method: Literal["completion/complete"] = "completion/complete" params: CompleteRequestParams @@ -1010,7 +1121,7 @@ class ListRootsRequest(Request[RequestParams | None, Literal["roots/list"]]): structure or access specific locations that the client has permission to read from. """ - method: Literal["roots/list"] + method: Literal["roots/list"] = "roots/list" params: RequestParams | None = None @@ -1029,6 +1140,11 @@ class Root(BaseModel): identifier for the root, which may be useful for display purposes or for referencing the root in other parts of the application. """ + meta: dict[str, Any] | None = Field(alias="_meta", default=None) + """ + See [MCP specification](https://github.com/modelcontextprotocol/modelcontextprotocol/blob/47339c03c143bb4ec01a26e721a1b8fe66634ebe/docs/specification/draft/basic/index.mdx#general-fields) + for notes on _meta usage. + """ model_config = ConfigDict(extra="allow") @@ -1054,7 +1170,7 @@ class RootsListChangedNotification( using the ListRootsRequest. """ - method: Literal["notifications/roots/list_changed"] + method: Literal["notifications/roots/list_changed"] = "notifications/roots/list_changed" params: NotificationParams | None = None @@ -1074,7 +1190,7 @@ class CancelledNotification(Notification[CancelledNotificationParams, Literal["n previously-issued request. """ - method: Literal["notifications/cancelled"] + method: Literal["notifications/cancelled"] = "notifications/cancelled" params: CancelledNotificationParams diff --git a/api/core/memory/token_buffer_memory.py b/api/core/memory/token_buffer_memory.py index 35af742f2a..3ebbb60f85 100644 --- a/api/core/memory/token_buffer_memory.py +++ b/api/core/memory/token_buffer_memory.py @@ -1,6 +1,7 @@ from collections.abc import Sequence from sqlalchemy import select +from sqlalchemy.orm import sessionmaker from core.app.app_config.features.file_upload.manager import FileUploadConfigManager from core.file import file_manager @@ -18,7 +19,9 @@ from core.prompt.utils.extract_thread_messages import extract_thread_messages from extensions.ext_database import db from factories import file_factory from models.model import AppMode, Conversation, Message, MessageFile -from models.workflow import Workflow, WorkflowRun +from models.workflow import Workflow +from repositories.api_workflow_run_repository import APIWorkflowRunRepository +from repositories.factory import DifyAPIRepositoryFactory class TokenBufferMemory: @@ -29,6 +32,14 @@ class TokenBufferMemory: ): self.conversation = conversation self.model_instance = model_instance + self._workflow_run_repo: APIWorkflowRunRepository | None = None + + @property + def workflow_run_repo(self) -> APIWorkflowRunRepository: + if self._workflow_run_repo is None: + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + return self._workflow_run_repo def _build_prompt_message_with_files( self, @@ -50,7 +61,16 @@ class TokenBufferMemory: if self.conversation.mode in {AppMode.AGENT_CHAT, AppMode.COMPLETION, AppMode.CHAT}: file_extra_config = FileUploadConfigManager.convert(self.conversation.model_config) elif self.conversation.mode in {AppMode.ADVANCED_CHAT, AppMode.WORKFLOW}: - workflow_run = db.session.scalar(select(WorkflowRun).where(WorkflowRun.id == message.workflow_run_id)) + app = self.conversation.app + if not app: + raise ValueError("App not found for conversation") + + if not message.workflow_run_id: + raise ValueError("Workflow run ID not found") + + workflow_run = self.workflow_run_repo.get_workflow_run_by_id( + tenant_id=app.tenant_id, app_id=app.id, run_id=message.workflow_run_id + ) if not workflow_run: raise ValueError(f"Workflow run not found: {message.workflow_run_id}") workflow = db.session.scalar(select(Workflow).where(Workflow.id == workflow_run.workflow_id)) diff --git a/api/core/model_runtime/entities/llm_entities.py b/api/core/model_runtime/entities/llm_entities.py index 17f6000d93..2c7c421eed 100644 --- a/api/core/model_runtime/entities/llm_entities.py +++ b/api/core/model_runtime/entities/llm_entities.py @@ -38,6 +38,8 @@ class LLMUsageMetadata(TypedDict, total=False): prompt_price: Union[float, str] completion_price: Union[float, str] latency: float + time_to_first_token: float + time_to_generate: float class LLMUsage(ModelUsage): @@ -57,6 +59,8 @@ class LLMUsage(ModelUsage): total_price: Decimal currency: str latency: float + time_to_first_token: float | None = None + time_to_generate: float | None = None @classmethod def empty_usage(cls): @@ -73,6 +77,8 @@ class LLMUsage(ModelUsage): total_price=Decimal("0.0"), currency="USD", latency=0.0, + time_to_first_token=None, + time_to_generate=None, ) @classmethod @@ -108,6 +114,8 @@ class LLMUsage(ModelUsage): prompt_price=Decimal(str(metadata.get("prompt_price", 0))), completion_price=Decimal(str(metadata.get("completion_price", 0))), latency=metadata.get("latency", 0.0), + time_to_first_token=metadata.get("time_to_first_token"), + time_to_generate=metadata.get("time_to_generate"), ) def plus(self, other: LLMUsage) -> LLMUsage: @@ -133,6 +141,8 @@ class LLMUsage(ModelUsage): total_price=self.total_price + other.total_price, currency=other.currency, latency=self.latency + other.latency, + time_to_first_token=other.time_to_first_token, + time_to_generate=other.time_to_generate, ) def __add__(self, other: LLMUsage) -> LLMUsage: diff --git a/api/core/moderation/openai_moderation/openai_moderation.py b/api/core/moderation/openai_moderation/openai_moderation.py index 74ef6f7ceb..5cab4841f5 100644 --- a/api/core/moderation/openai_moderation/openai_moderation.py +++ b/api/core/moderation/openai_moderation/openai_moderation.py @@ -52,7 +52,7 @@ class OpenAIModeration(Moderation): text = "\n".join(str(inputs.values())) model_manager = ModelManager() model_instance = model_manager.get_model_instance( - tenant_id=self.tenant_id, provider="openai", model_type=ModelType.MODERATION, model="text-moderation-stable" + tenant_id=self.tenant_id, provider="openai", model_type=ModelType.MODERATION, model="omni-moderation-latest" ) openai_moderation = model_instance.invoke_moderation(text=text) diff --git a/api/core/ops/arize_phoenix_trace/arize_phoenix_trace.py b/api/core/ops/arize_phoenix_trace/arize_phoenix_trace.py index 03d2d75372..347992fa0d 100644 --- a/api/core/ops/arize_phoenix_trace/arize_phoenix_trace.py +++ b/api/core/ops/arize_phoenix_trace/arize_phoenix_trace.py @@ -1,21 +1,22 @@ -import hashlib import json import logging import os +import traceback from datetime import datetime, timedelta from typing import Any, Union, cast from urllib.parse import urlparse -from openinference.semconv.trace import OpenInferenceSpanKindValues, SpanAttributes -from opentelemetry import trace +from openinference.semconv.trace import OpenInferenceMimeTypeValues, OpenInferenceSpanKindValues, SpanAttributes from opentelemetry.exporter.otlp.proto.grpc.trace_exporter import OTLPSpanExporter as GrpcOTLPSpanExporter from opentelemetry.exporter.otlp.proto.http.trace_exporter import OTLPSpanExporter as HttpOTLPSpanExporter from opentelemetry.sdk import trace as trace_sdk from opentelemetry.sdk.resources import Resource from opentelemetry.sdk.trace.export import SimpleSpanProcessor -from opentelemetry.sdk.trace.id_generator import RandomIdGenerator -from opentelemetry.trace import SpanContext, TraceFlags, TraceState -from sqlalchemy import select +from opentelemetry.semconv.trace import SpanAttributes as OTELSpanAttributes +from opentelemetry.trace import Span, Status, StatusCode, set_span_in_context, use_span +from opentelemetry.trace.propagation.tracecontext import TraceContextTextMapPropagator +from opentelemetry.util.types import AttributeValue +from sqlalchemy.orm import sessionmaker from core.ops.base_trace_instance import BaseTraceInstance from core.ops.entities.config_entity import ArizeConfig, PhoenixConfig @@ -30,9 +31,10 @@ from core.ops.entities.trace_entity import ( TraceTaskName, WorkflowTraceInfo, ) +from core.repositories import DifyCoreRepositoryFactory from extensions.ext_database import db from models.model import EndUser, MessageFile -from models.workflow import WorkflowNodeExecutionModel +from models.workflow import WorkflowNodeExecutionTriggeredFrom logger = logging.getLogger(__name__) @@ -99,22 +101,45 @@ def datetime_to_nanos(dt: datetime | None) -> int: return int(dt.timestamp() * 1_000_000_000) -def string_to_trace_id128(string: str | None) -> int: - """ - Convert any input string into a stable 128-bit integer trace ID. +def error_to_string(error: Exception | str | None) -> str: + """Convert an error to a string with traceback information.""" + error_message = "Empty Stack Trace" + if error: + if isinstance(error, Exception): + string_stacktrace = "".join(traceback.format_exception(error)) + error_message = f"{error.__class__.__name__}: {error}\n\n{string_stacktrace}" + else: + error_message = str(error) + return error_message - This uses SHA-256 hashing and takes the first 16 bytes (128 bits) of the digest. - It's suitable for generating consistent, unique identifiers from strings. - """ - if string is None: - string = "" - hash_object = hashlib.sha256(string.encode()) - # Take the first 16 bytes (128 bits) of the hash digest - digest = hash_object.digest()[:16] +def set_span_status(current_span: Span, error: Exception | str | None = None): + """Set the status of the current span based on the presence of an error.""" + if error: + error_string = error_to_string(error) + current_span.set_status(Status(StatusCode.ERROR, error_string)) - # Convert to a 128-bit integer - return int.from_bytes(digest, byteorder="big") + if isinstance(error, Exception): + current_span.record_exception(error) + else: + exception_type = error.__class__.__name__ + exception_message = str(error) + if not exception_message: + exception_message = repr(error) + attributes: dict[str, AttributeValue] = { + OTELSpanAttributes.EXCEPTION_TYPE: exception_type, + OTELSpanAttributes.EXCEPTION_MESSAGE: exception_message, + OTELSpanAttributes.EXCEPTION_ESCAPED: False, + OTELSpanAttributes.EXCEPTION_STACKTRACE: error_string, + } + current_span.add_event(name="exception", attributes=attributes) + else: + current_span.set_status(Status(StatusCode.OK)) + + +def safe_json_dumps(obj: Any) -> str: + """A convenience wrapper around `json.dumps` that ensures that any object can be safely encoded.""" + return json.dumps(obj, default=str, ensure_ascii=False) class ArizePhoenixDataTrace(BaseTraceInstance): @@ -131,9 +156,12 @@ class ArizePhoenixDataTrace(BaseTraceInstance): self.tracer, self.processor = setup_tracer(arize_phoenix_config) self.project = arize_phoenix_config.project self.file_base_url = os.getenv("FILES_URL", "http://127.0.0.1:5001") + self.propagator = TraceContextTextMapPropagator() + self.dify_trace_ids: set[str] = set() def trace(self, trace_info: BaseTraceInfo): - logger.info("[Arize/Phoenix] Trace: %s", trace_info) + logger.info("[Arize/Phoenix] Trace Entity Info: %s", trace_info) + logger.info("[Arize/Phoenix] Trace Entity Type: %s", type(trace_info)) try: if isinstance(trace_info, WorkflowTraceInfo): self.workflow_trace(trace_info) @@ -151,7 +179,7 @@ class ArizePhoenixDataTrace(BaseTraceInstance): self.generate_name_trace(trace_info) except Exception as e: - logger.error("[Arize/Phoenix] Error in the trace: %s", str(e), exc_info=True) + logger.error("[Arize/Phoenix] Trace Entity Error: %s", str(e), exc_info=True) raise def workflow_trace(self, trace_info: WorkflowTraceInfo): @@ -166,15 +194,9 @@ class ArizePhoenixDataTrace(BaseTraceInstance): } workflow_metadata.update(trace_info.metadata) - trace_id = string_to_trace_id128(trace_info.trace_id or trace_info.workflow_run_id) - span_id = RandomIdGenerator().generate_span_id() - context = SpanContext( - trace_id=trace_id, - span_id=span_id, - is_remote=False, - trace_flags=TraceFlags(TraceFlags.SAMPLED), - trace_state=TraceState(), - ) + dify_trace_id = trace_info.trace_id or trace_info.message_id or trace_info.workflow_run_id + self.ensure_root_span(dify_trace_id) + root_span_context = self.propagator.extract(carrier=self.carrier) workflow_span = self.tracer.start_span( name=TraceTaskName.WORKFLOW_TRACE.value, @@ -186,31 +208,58 @@ class ArizePhoenixDataTrace(BaseTraceInstance): SpanAttributes.SESSION_ID: trace_info.conversation_id or "", }, start_time=datetime_to_nanos(trace_info.start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(context)), + context=root_span_context, + ) + + # Through workflow_run_id, get all_nodes_execution using repository + session_factory = sessionmaker(bind=db.engine) + + # Find the app's creator account + app_id = trace_info.metadata.get("app_id") + if not app_id: + raise ValueError("No app_id found in trace_info metadata") + + service_account = self.get_service_account_with_tenant(app_id) + + workflow_node_execution_repository = DifyCoreRepositoryFactory.create_workflow_node_execution_repository( + session_factory=session_factory, + user=service_account, + app_id=app_id, + triggered_from=WorkflowNodeExecutionTriggeredFrom.WORKFLOW_RUN, + ) + + # Get all executions for this workflow run + workflow_node_executions = workflow_node_execution_repository.get_by_workflow_run( + workflow_run_id=trace_info.workflow_run_id ) try: - # Process workflow nodes - for node_execution in self._get_workflow_nodes(trace_info.workflow_run_id): + for node_execution in workflow_node_executions: + tenant_id = trace_info.tenant_id # Use from trace_info instead + app_id = trace_info.metadata.get("app_id") # Use from trace_info instead + inputs_value = node_execution.inputs or {} + outputs_value = node_execution.outputs or {} + created_at = node_execution.created_at or datetime.now() elapsed_time = node_execution.elapsed_time finished_at = created_at + timedelta(seconds=elapsed_time) - process_data = json.loads(node_execution.process_data) if node_execution.process_data else {} + process_data = node_execution.process_data or {} + execution_metadata = node_execution.metadata or {} + node_metadata = {str(k): v for k, v in execution_metadata.items()} - node_metadata = { - "node_id": node_execution.id, - "node_type": node_execution.node_type, - "node_status": node_execution.status, - "tenant_id": node_execution.tenant_id, - "app_id": node_execution.app_id, - "app_name": node_execution.title, - "status": node_execution.status, - "level": "ERROR" if node_execution.status != "succeeded" else "DEFAULT", - } - - if node_execution.execution_metadata: - node_metadata.update(json.loads(node_execution.execution_metadata)) + node_metadata.update( + { + "node_id": node_execution.id, + "node_type": node_execution.node_type, + "node_status": node_execution.status, + "tenant_id": tenant_id, + "app_id": app_id, + "app_name": node_execution.title, + "status": node_execution.status, + "level": "ERROR" if node_execution.status == "failed" else "DEFAULT", + } + ) # Determine the correct span kind based on node type span_kind = OpenInferenceSpanKindValues.CHAIN @@ -223,8 +272,9 @@ class ArizePhoenixDataTrace(BaseTraceInstance): if model: node_metadata["ls_model_name"] = model - outputs = json.loads(node_execution.outputs).get("usage", {}) if "outputs" in node_execution else {} - usage_data = process_data.get("usage", {}) if "usage" in process_data else outputs.get("usage", {}) + usage_data = ( + process_data.get("usage", {}) if "usage" in process_data else outputs_value.get("usage", {}) + ) if usage_data: node_metadata["total_tokens"] = usage_data.get("total_tokens", 0) node_metadata["prompt_tokens"] = usage_data.get("prompt_tokens", 0) @@ -236,17 +286,20 @@ class ArizePhoenixDataTrace(BaseTraceInstance): else: span_kind = OpenInferenceSpanKindValues.CHAIN + workflow_span_context = set_span_in_context(workflow_span) node_span = self.tracer.start_span( name=node_execution.node_type, attributes={ - SpanAttributes.INPUT_VALUE: node_execution.inputs or "{}", - SpanAttributes.OUTPUT_VALUE: node_execution.outputs or "{}", + SpanAttributes.INPUT_VALUE: safe_json_dumps(inputs_value), + SpanAttributes.INPUT_MIME_TYPE: OpenInferenceMimeTypeValues.JSON.value, + SpanAttributes.OUTPUT_VALUE: safe_json_dumps(outputs_value), + SpanAttributes.OUTPUT_MIME_TYPE: OpenInferenceMimeTypeValues.JSON.value, SpanAttributes.OPENINFERENCE_SPAN_KIND: span_kind.value, - SpanAttributes.METADATA: json.dumps(node_metadata, ensure_ascii=False), + SpanAttributes.METADATA: safe_json_dumps(node_metadata), SpanAttributes.SESSION_ID: trace_info.conversation_id or "", }, start_time=datetime_to_nanos(created_at), - context=trace.set_span_in_context(trace.NonRecordingSpan(context)), + context=workflow_span_context, ) try: @@ -260,11 +313,8 @@ class ArizePhoenixDataTrace(BaseTraceInstance): llm_attributes[SpanAttributes.LLM_PROVIDER] = provider if model: llm_attributes[SpanAttributes.LLM_MODEL_NAME] = model - outputs = ( - json.loads(node_execution.outputs).get("usage", {}) if "outputs" in node_execution else {} - ) usage_data = ( - process_data.get("usage", {}) if "usage" in process_data else outputs.get("usage", {}) + process_data.get("usage", {}) if "usage" in process_data else outputs_value.get("usage", {}) ) if usage_data: llm_attributes[SpanAttributes.LLM_TOKEN_COUNT_TOTAL] = usage_data.get("total_tokens", 0) @@ -275,8 +325,16 @@ class ArizePhoenixDataTrace(BaseTraceInstance): llm_attributes.update(self._construct_llm_attributes(process_data.get("prompts", []))) node_span.set_attributes(llm_attributes) finally: + if node_execution.status == "failed": + set_span_status(node_span, node_execution.error) + else: + set_span_status(node_span) node_span.end(end_time=datetime_to_nanos(finished_at)) finally: + if trace_info.error: + set_span_status(workflow_span, trace_info.error) + else: + set_span_status(workflow_span) workflow_span.end(end_time=datetime_to_nanos(trace_info.end_time)) def message_trace(self, trace_info: MessageTraceInfo): @@ -322,34 +380,18 @@ class ArizePhoenixDataTrace(BaseTraceInstance): SpanAttributes.SESSION_ID: trace_info.message_data.conversation_id, } - trace_id = string_to_trace_id128(trace_info.trace_id or trace_info.message_id) - message_span_id = RandomIdGenerator().generate_span_id() - span_context = SpanContext( - trace_id=trace_id, - span_id=message_span_id, - is_remote=False, - trace_flags=TraceFlags(TraceFlags.SAMPLED), - trace_state=TraceState(), - ) + dify_trace_id = trace_info.trace_id or trace_info.message_id + self.ensure_root_span(dify_trace_id) + root_span_context = self.propagator.extract(carrier=self.carrier) message_span = self.tracer.start_span( name=TraceTaskName.MESSAGE_TRACE.value, attributes=attributes, start_time=datetime_to_nanos(trace_info.start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(span_context)), + context=root_span_context, ) try: - if trace_info.error: - message_span.add_event( - "exception", - attributes={ - "exception.message": trace_info.error, - "exception.type": "Error", - "exception.stacktrace": trace_info.error, - }, - ) - # Convert outputs to string based on type if isinstance(trace_info.outputs, dict | list): outputs_str = json.dumps(trace_info.outputs, ensure_ascii=False) @@ -383,26 +425,26 @@ class ArizePhoenixDataTrace(BaseTraceInstance): if model_params := metadata_dict.get("model_parameters"): llm_attributes[SpanAttributes.LLM_INVOCATION_PARAMETERS] = json.dumps(model_params) + message_span_context = set_span_in_context(message_span) llm_span = self.tracer.start_span( name="llm", attributes=llm_attributes, start_time=datetime_to_nanos(trace_info.start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(span_context)), + context=message_span_context, ) try: - if trace_info.error: - llm_span.add_event( - "exception", - attributes={ - "exception.message": trace_info.error, - "exception.type": "Error", - "exception.stacktrace": trace_info.error, - }, - ) + if trace_info.message_data.error: + set_span_status(llm_span, trace_info.message_data.error) + else: + set_span_status(llm_span) finally: llm_span.end(end_time=datetime_to_nanos(trace_info.end_time)) finally: + if trace_info.error: + set_span_status(message_span, trace_info.error) + else: + set_span_status(message_span) message_span.end(end_time=datetime_to_nanos(trace_info.end_time)) def moderation_trace(self, trace_info: ModerationTraceInfo): @@ -418,15 +460,9 @@ class ArizePhoenixDataTrace(BaseTraceInstance): } metadata.update(trace_info.metadata) - trace_id = string_to_trace_id128(trace_info.message_id) - span_id = RandomIdGenerator().generate_span_id() - context = SpanContext( - trace_id=trace_id, - span_id=span_id, - is_remote=False, - trace_flags=TraceFlags(TraceFlags.SAMPLED), - trace_state=TraceState(), - ) + dify_trace_id = trace_info.trace_id or trace_info.message_id + self.ensure_root_span(dify_trace_id) + root_span_context = self.propagator.extract(carrier=self.carrier) span = self.tracer.start_span( name=TraceTaskName.MODERATION_TRACE.value, @@ -445,19 +481,14 @@ class ArizePhoenixDataTrace(BaseTraceInstance): SpanAttributes.METADATA: json.dumps(metadata, ensure_ascii=False), }, start_time=datetime_to_nanos(trace_info.start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(context)), + context=root_span_context, ) try: if trace_info.message_data.error: - span.add_event( - "exception", - attributes={ - "exception.message": trace_info.message_data.error, - "exception.type": "Error", - "exception.stacktrace": trace_info.message_data.error, - }, - ) + set_span_status(span, trace_info.message_data.error) + else: + set_span_status(span) finally: span.end(end_time=datetime_to_nanos(trace_info.end_time)) @@ -480,15 +511,9 @@ class ArizePhoenixDataTrace(BaseTraceInstance): } metadata.update(trace_info.metadata) - trace_id = string_to_trace_id128(trace_info.message_id) - span_id = RandomIdGenerator().generate_span_id() - context = SpanContext( - trace_id=trace_id, - span_id=span_id, - is_remote=False, - trace_flags=TraceFlags(TraceFlags.SAMPLED), - trace_state=TraceState(), - ) + dify_trace_id = trace_info.trace_id or trace_info.message_id + self.ensure_root_span(dify_trace_id) + root_span_context = self.propagator.extract(carrier=self.carrier) span = self.tracer.start_span( name=TraceTaskName.SUGGESTED_QUESTION_TRACE.value, @@ -499,19 +524,14 @@ class ArizePhoenixDataTrace(BaseTraceInstance): SpanAttributes.METADATA: json.dumps(metadata, ensure_ascii=False), }, start_time=datetime_to_nanos(start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(context)), + context=root_span_context, ) try: if trace_info.error: - span.add_event( - "exception", - attributes={ - "exception.message": trace_info.error, - "exception.type": "Error", - "exception.stacktrace": trace_info.error, - }, - ) + set_span_status(span, trace_info.error) + else: + set_span_status(span) finally: span.end(end_time=datetime_to_nanos(end_time)) @@ -533,15 +553,9 @@ class ArizePhoenixDataTrace(BaseTraceInstance): } metadata.update(trace_info.metadata) - trace_id = string_to_trace_id128(trace_info.message_id) - span_id = RandomIdGenerator().generate_span_id() - context = SpanContext( - trace_id=trace_id, - span_id=span_id, - is_remote=False, - trace_flags=TraceFlags(TraceFlags.SAMPLED), - trace_state=TraceState(), - ) + dify_trace_id = trace_info.trace_id or trace_info.message_id + self.ensure_root_span(dify_trace_id) + root_span_context = self.propagator.extract(carrier=self.carrier) span = self.tracer.start_span( name=TraceTaskName.DATASET_RETRIEVAL_TRACE.value, @@ -554,19 +568,14 @@ class ArizePhoenixDataTrace(BaseTraceInstance): "end_time": end_time.isoformat() if end_time else "", }, start_time=datetime_to_nanos(start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(context)), + context=root_span_context, ) try: if trace_info.message_data.error: - span.add_event( - "exception", - attributes={ - "exception.message": trace_info.message_data.error, - "exception.type": "Error", - "exception.stacktrace": trace_info.message_data.error, - }, - ) + set_span_status(span, trace_info.message_data.error) + else: + set_span_status(span) finally: span.end(end_time=datetime_to_nanos(end_time)) @@ -580,20 +589,9 @@ class ArizePhoenixDataTrace(BaseTraceInstance): "tool_config": json.dumps(trace_info.tool_config, ensure_ascii=False), } - trace_id = string_to_trace_id128(trace_info.message_id) - tool_span_id = RandomIdGenerator().generate_span_id() - logger.info("[Arize/Phoenix] Creating tool trace with trace_id: %s, span_id: %s", trace_id, tool_span_id) - - # Create span context with the same trace_id as the parent - # todo: Create with the appropriate parent span context, so that the tool span is - # a child of the appropriate span (e.g. message span) - span_context = SpanContext( - trace_id=trace_id, - span_id=tool_span_id, - is_remote=False, - trace_flags=TraceFlags(TraceFlags.SAMPLED), - trace_state=TraceState(), - ) + dify_trace_id = trace_info.trace_id or trace_info.message_id + self.ensure_root_span(dify_trace_id) + root_span_context = self.propagator.extract(carrier=self.carrier) tool_params_str = ( json.dumps(trace_info.tool_parameters, ensure_ascii=False) @@ -612,19 +610,14 @@ class ArizePhoenixDataTrace(BaseTraceInstance): SpanAttributes.TOOL_PARAMETERS: tool_params_str, }, start_time=datetime_to_nanos(trace_info.start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(span_context)), + context=root_span_context, ) try: if trace_info.error: - span.add_event( - "exception", - attributes={ - "exception.message": trace_info.error, - "exception.type": "Error", - "exception.stacktrace": trace_info.error, - }, - ) + set_span_status(span, trace_info.error) + else: + set_span_status(span) finally: span.end(end_time=datetime_to_nanos(trace_info.end_time)) @@ -641,15 +634,9 @@ class ArizePhoenixDataTrace(BaseTraceInstance): } metadata.update(trace_info.metadata) - trace_id = string_to_trace_id128(trace_info.message_id) - span_id = RandomIdGenerator().generate_span_id() - context = SpanContext( - trace_id=trace_id, - span_id=span_id, - is_remote=False, - trace_flags=TraceFlags(TraceFlags.SAMPLED), - trace_state=TraceState(), - ) + dify_trace_id = trace_info.trace_id or trace_info.message_id or trace_info.conversation_id + self.ensure_root_span(dify_trace_id) + root_span_context = self.propagator.extract(carrier=self.carrier) span = self.tracer.start_span( name=TraceTaskName.GENERATE_NAME_TRACE.value, @@ -663,22 +650,34 @@ class ArizePhoenixDataTrace(BaseTraceInstance): "end_time": trace_info.end_time.isoformat() if trace_info.end_time else "", }, start_time=datetime_to_nanos(trace_info.start_time), - context=trace.set_span_in_context(trace.NonRecordingSpan(context)), + context=root_span_context, ) try: if trace_info.message_data.error: - span.add_event( - "exception", - attributes={ - "exception.message": trace_info.message_data.error, - "exception.type": "Error", - "exception.stacktrace": trace_info.message_data.error, - }, - ) + set_span_status(span, trace_info.message_data.error) + else: + set_span_status(span) finally: span.end(end_time=datetime_to_nanos(trace_info.end_time)) + def ensure_root_span(self, dify_trace_id: str | None): + """Ensure a unique root span exists for the given Dify trace ID.""" + if str(dify_trace_id) not in self.dify_trace_ids: + self.carrier: dict[str, str] = {} + + root_span = self.tracer.start_span(name="Dify") + root_span.set_attribute(SpanAttributes.OPENINFERENCE_SPAN_KIND, OpenInferenceSpanKindValues.CHAIN.value) + root_span.set_attribute("dify_project_name", str(self.project)) + root_span.set_attribute("dify_trace_id", str(dify_trace_id)) + + with use_span(root_span, end_on_exit=False): + self.propagator.inject(carrier=self.carrier) + + set_span_status(root_span) + root_span.end() + self.dify_trace_ids.add(str(dify_trace_id)) + def api_check(self): try: with self.tracer.start_span("api_check") as span: @@ -698,26 +697,6 @@ class ArizePhoenixDataTrace(BaseTraceInstance): logger.info("[Arize/Phoenix] Get run url failed: %s", str(e), exc_info=True) raise ValueError(f"[Arize/Phoenix] Get run url failed: {str(e)}") - def _get_workflow_nodes(self, workflow_run_id: str): - """Helper method to get workflow nodes""" - workflow_nodes = db.session.scalars( - select( - WorkflowNodeExecutionModel.id, - WorkflowNodeExecutionModel.tenant_id, - WorkflowNodeExecutionModel.app_id, - WorkflowNodeExecutionModel.title, - WorkflowNodeExecutionModel.node_type, - WorkflowNodeExecutionModel.status, - WorkflowNodeExecutionModel.inputs, - WorkflowNodeExecutionModel.outputs, - WorkflowNodeExecutionModel.created_at, - WorkflowNodeExecutionModel.elapsed_time, - WorkflowNodeExecutionModel.process_data, - WorkflowNodeExecutionModel.execution_metadata, - ).where(WorkflowNodeExecutionModel.workflow_run_id == workflow_run_id) - ).all() - return workflow_nodes - def _construct_llm_attributes(self, prompts: dict | list | str | None) -> dict[str, str]: """Helper method to construct LLM attributes with passed prompts.""" attributes = {} diff --git a/api/core/ops/entities/trace_entity.py b/api/core/ops/entities/trace_entity.py index 5b81c09a2d..50a2cdea63 100644 --- a/api/core/ops/entities/trace_entity.py +++ b/api/core/ops/entities/trace_entity.py @@ -62,6 +62,9 @@ class MessageTraceInfo(BaseTraceInfo): file_list: Union[str, dict[str, Any], list] | None = None message_file_data: Any | None = None conversation_mode: str + gen_ai_server_time_to_first_token: float | None = None + llm_streaming_time_to_generate: float | None = None + is_streaming_request: bool = False class ModerationTraceInfo(BaseTraceInfo): diff --git a/api/core/ops/ops_trace_manager.py b/api/core/ops/ops_trace_manager.py index 7db9b076d2..5bb539b7dc 100644 --- a/api/core/ops/ops_trace_manager.py +++ b/api/core/ops/ops_trace_manager.py @@ -12,9 +12,9 @@ from uuid import UUID, uuid4 from cachetools import LRUCache from flask import current_app from sqlalchemy import select -from sqlalchemy.orm import Session +from sqlalchemy.orm import Session, sessionmaker -from core.helper.encrypter import decrypt_token, encrypt_token, obfuscated_token +from core.helper.encrypter import batch_decrypt_token, encrypt_token, obfuscated_token from core.ops.entities.config_entity import ( OPS_FILE_PATH, TracingProviderEnum, @@ -34,7 +34,8 @@ from core.ops.utils import get_message_data from extensions.ext_database import db from extensions.ext_storage import storage from models.model import App, AppModelConfig, Conversation, Message, MessageFile, TraceAppConfig -from models.workflow import WorkflowAppLog, WorkflowRun +from models.workflow import WorkflowAppLog +from repositories.factory import DifyAPIRepositoryFactory from tasks.ops_trace_task import process_trace_tasks if TYPE_CHECKING: @@ -140,6 +141,8 @@ provider_config_map = OpsTraceProviderConfigMap() class OpsTraceManager: ops_trace_instances_cache: LRUCache = LRUCache(maxsize=128) + decrypted_configs_cache: LRUCache = LRUCache(maxsize=128) + _decryption_cache_lock = threading.RLock() @classmethod def encrypt_tracing_config( @@ -160,7 +163,7 @@ class OpsTraceManager: provider_config_map[tracing_provider]["other_keys"], ) - new_config = {} + new_config: dict[str, Any] = {} # Encrypt necessary keys for key in secret_keys: if key in tracing_config: @@ -190,20 +193,41 @@ class OpsTraceManager: :param tracing_config: tracing config :return: """ - config_class, secret_keys, other_keys = ( - provider_config_map[tracing_provider]["config_class"], - provider_config_map[tracing_provider]["secret_keys"], - provider_config_map[tracing_provider]["other_keys"], + config_json = json.dumps(tracing_config, sort_keys=True) + decrypted_config_key = ( + tenant_id, + tracing_provider, + config_json, ) - new_config = {} - for key in secret_keys: - if key in tracing_config: - new_config[key] = decrypt_token(tenant_id, tracing_config[key]) - for key in other_keys: - new_config[key] = tracing_config.get(key, "") + # First check without lock for performance + cached_config = cls.decrypted_configs_cache.get(decrypted_config_key) + if cached_config is not None: + return dict(cached_config) - return config_class(**new_config).model_dump() + with cls._decryption_cache_lock: + # Second check (double-checked locking) to prevent race conditions + cached_config = cls.decrypted_configs_cache.get(decrypted_config_key) + if cached_config is not None: + return dict(cached_config) + + config_class, secret_keys, other_keys = ( + provider_config_map[tracing_provider]["config_class"], + provider_config_map[tracing_provider]["secret_keys"], + provider_config_map[tracing_provider]["other_keys"], + ) + new_config: dict[str, Any] = {} + keys_to_decrypt = [key for key in secret_keys if key in tracing_config] + if keys_to_decrypt: + decrypted_values = batch_decrypt_token(tenant_id, [tracing_config[key] for key in keys_to_decrypt]) + new_config.update(zip(keys_to_decrypt, decrypted_values)) + + for key in other_keys: + new_config[key] = tracing_config.get(key, "") + + decrypted_config = config_class(**new_config).model_dump() + cls.decrypted_configs_cache[decrypted_config_key] = decrypted_config + return dict(decrypted_config) @classmethod def obfuscated_decrypt_token(cls, tracing_provider: str, decrypt_tracing_config: dict): @@ -218,7 +242,7 @@ class OpsTraceManager: provider_config_map[tracing_provider]["secret_keys"], provider_config_map[tracing_provider]["other_keys"], ) - new_config = {} + new_config: dict[str, Any] = {} for key in secret_keys: if key in decrypt_tracing_config: new_config[key] = obfuscated_token(decrypt_tracing_config[key]) @@ -419,6 +443,18 @@ class OpsTraceManager: class TraceTask: + _workflow_run_repo = None + _repo_lock = threading.Lock() + + @classmethod + def _get_workflow_run_repo(cls): + if cls._workflow_run_repo is None: + with cls._repo_lock: + if cls._workflow_run_repo is None: + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + cls._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + return cls._workflow_run_repo + def __init__( self, trace_type: Any, @@ -486,27 +522,27 @@ class TraceTask: if not workflow_run_id: return {} + workflow_run_repo = self._get_workflow_run_repo() + workflow_run = workflow_run_repo.get_workflow_run_by_id_without_tenant(run_id=workflow_run_id) + if not workflow_run: + raise ValueError("Workflow run not found") + + workflow_id = workflow_run.workflow_id + tenant_id = workflow_run.tenant_id + workflow_run_id = workflow_run.id + workflow_run_elapsed_time = workflow_run.elapsed_time + workflow_run_status = workflow_run.status + workflow_run_inputs = workflow_run.inputs_dict + workflow_run_outputs = workflow_run.outputs_dict + workflow_run_version = workflow_run.version + error = workflow_run.error or "" + + total_tokens = workflow_run.total_tokens + + file_list = workflow_run_inputs.get("sys.file") or [] + query = workflow_run_inputs.get("query") or workflow_run_inputs.get("sys.query") or "" + with Session(db.engine) as session: - workflow_run_stmt = select(WorkflowRun).where(WorkflowRun.id == workflow_run_id) - workflow_run = session.scalars(workflow_run_stmt).first() - if not workflow_run: - raise ValueError("Workflow run not found") - - workflow_id = workflow_run.workflow_id - tenant_id = workflow_run.tenant_id - workflow_run_id = workflow_run.id - workflow_run_elapsed_time = workflow_run.elapsed_time - workflow_run_status = workflow_run.status - workflow_run_inputs = workflow_run.inputs_dict - workflow_run_outputs = workflow_run.outputs_dict - workflow_run_version = workflow_run.version - error = workflow_run.error or "" - - total_tokens = workflow_run.total_tokens - - file_list = workflow_run_inputs.get("sys.file") or [] - query = workflow_run_inputs.get("query") or workflow_run_inputs.get("sys.query") or "" - # get workflow_app_log_id workflow_app_log_data_stmt = select(WorkflowAppLog.id).where( WorkflowAppLog.tenant_id == tenant_id, @@ -523,43 +559,43 @@ class TraceTask: ) message_id = session.scalar(message_data_stmt) - metadata = { - "workflow_id": workflow_id, - "conversation_id": conversation_id, - "workflow_run_id": workflow_run_id, - "tenant_id": tenant_id, - "elapsed_time": workflow_run_elapsed_time, - "status": workflow_run_status, - "version": workflow_run_version, - "total_tokens": total_tokens, - "file_list": file_list, - "triggered_from": workflow_run.triggered_from, - "user_id": user_id, - "app_id": workflow_run.app_id, - } + metadata = { + "workflow_id": workflow_id, + "conversation_id": conversation_id, + "workflow_run_id": workflow_run_id, + "tenant_id": tenant_id, + "elapsed_time": workflow_run_elapsed_time, + "status": workflow_run_status, + "version": workflow_run_version, + "total_tokens": total_tokens, + "file_list": file_list, + "triggered_from": workflow_run.triggered_from, + "user_id": user_id, + "app_id": workflow_run.app_id, + } - workflow_trace_info = WorkflowTraceInfo( - trace_id=self.trace_id, - workflow_data=workflow_run.to_dict(), - conversation_id=conversation_id, - workflow_id=workflow_id, - tenant_id=tenant_id, - workflow_run_id=workflow_run_id, - workflow_run_elapsed_time=workflow_run_elapsed_time, - workflow_run_status=workflow_run_status, - workflow_run_inputs=workflow_run_inputs, - workflow_run_outputs=workflow_run_outputs, - workflow_run_version=workflow_run_version, - error=error, - total_tokens=total_tokens, - file_list=file_list, - query=query, - metadata=metadata, - workflow_app_log_id=workflow_app_log_id, - message_id=message_id, - start_time=workflow_run.created_at, - end_time=workflow_run.finished_at, - ) + workflow_trace_info = WorkflowTraceInfo( + trace_id=self.trace_id, + workflow_data=workflow_run.to_dict(), + conversation_id=conversation_id, + workflow_id=workflow_id, + tenant_id=tenant_id, + workflow_run_id=workflow_run_id, + workflow_run_elapsed_time=workflow_run_elapsed_time, + workflow_run_status=workflow_run_status, + workflow_run_inputs=workflow_run_inputs, + workflow_run_outputs=workflow_run_outputs, + workflow_run_version=workflow_run_version, + error=error, + total_tokens=total_tokens, + file_list=file_list, + query=query, + metadata=metadata, + workflow_app_log_id=workflow_app_log_id, + message_id=message_id, + start_time=workflow_run.created_at, + end_time=workflow_run.finished_at, + ) return workflow_trace_info def message_trace(self, message_id: str | None): @@ -583,6 +619,8 @@ class TraceTask: file_url = f"{self.file_base_url}/{message_file_data.url}" if message_file_data else "" file_list.append(file_url) + streaming_metrics = self._extract_streaming_metrics(message_data) + metadata = { "conversation_id": message_data.conversation_id, "ls_provider": message_data.model_provider, @@ -615,6 +653,9 @@ class TraceTask: metadata=metadata, message_file_data=message_file_data, conversation_mode=conversation_mode, + gen_ai_server_time_to_first_token=streaming_metrics.get("gen_ai_server_time_to_first_token"), + llm_streaming_time_to_generate=streaming_metrics.get("llm_streaming_time_to_generate"), + is_streaming_request=streaming_metrics.get("is_streaming_request", False), ) return message_trace_info @@ -840,6 +881,24 @@ class TraceTask: return generate_name_trace_info + def _extract_streaming_metrics(self, message_data) -> dict: + if not message_data.message_metadata: + return {} + + try: + metadata = json.loads(message_data.message_metadata) + usage = metadata.get("usage", {}) + time_to_first_token = usage.get("time_to_first_token") + time_to_generate = usage.get("time_to_generate") + + return { + "gen_ai_server_time_to_first_token": time_to_first_token, + "llm_streaming_time_to_generate": time_to_generate, + "is_streaming_request": time_to_first_token is not None, + } + except (json.JSONDecodeError, AttributeError): + return {} + trace_manager_timer: threading.Timer | None = None trace_manager_queue: queue.Queue = queue.Queue() diff --git a/api/core/ops/tencent_trace/client.py b/api/core/ops/tencent_trace/client.py index 270732aa02..bf1ab5e7e6 100644 --- a/api/core/ops/tencent_trace/client.py +++ b/api/core/ops/tencent_trace/client.py @@ -5,12 +5,18 @@ Tencent APM Trace Client - handles network operations, metrics, and API communic from __future__ import annotations import importlib +import json import logging import os import socket from typing import TYPE_CHECKING from urllib.parse import urlparse +try: + from importlib.metadata import version +except ImportError: + from importlib_metadata import version # type: ignore[import-not-found] + if TYPE_CHECKING: from opentelemetry.metrics import Meter from opentelemetry.metrics._internal.instrument import Histogram @@ -27,12 +33,27 @@ from opentelemetry.util.types import AttributeValue from configs import dify_config -from .entities.tencent_semconv import LLM_OPERATION_DURATION +from .entities.semconv import ( + GEN_AI_SERVER_TIME_TO_FIRST_TOKEN, + GEN_AI_STREAMING_TIME_TO_GENERATE, + GEN_AI_TOKEN_USAGE, + GEN_AI_TRACE_DURATION, + LLM_OPERATION_DURATION, +) from .entities.tencent_trace_entity import SpanData logger = logging.getLogger(__name__) +def _get_opentelemetry_sdk_version() -> str: + """Get OpenTelemetry SDK version dynamically.""" + try: + return version("opentelemetry-sdk") + except Exception: + logger.debug("Failed to get opentelemetry-sdk version, using default") + return "1.27.0" # fallback version + + class TencentTraceClient: """Tencent APM trace client using OpenTelemetry OTLP exporter""" @@ -57,6 +78,9 @@ class TencentTraceClient: ResourceAttributes.SERVICE_VERSION: f"dify-{dify_config.project.version}-{dify_config.COMMIT_SHA}", ResourceAttributes.DEPLOYMENT_ENVIRONMENT: f"{dify_config.DEPLOY_ENV}-{dify_config.EDITION}", ResourceAttributes.HOST_NAME: socket.gethostname(), + ResourceAttributes.TELEMETRY_SDK_LANGUAGE: "python", + ResourceAttributes.TELEMETRY_SDK_NAME: "opentelemetry", + ResourceAttributes.TELEMETRY_SDK_VERSION: _get_opentelemetry_sdk_version(), } ) # Prepare gRPC endpoint/metadata @@ -80,18 +104,23 @@ class TencentTraceClient: ) self.tracer_provider.add_span_processor(self.span_processor) - self.tracer = self.tracer_provider.get_tracer("dify.tencent_apm") + # use dify api version as tracer version + self.tracer = self.tracer_provider.get_tracer("dify-sdk", dify_config.project.version) # Store span contexts for parent-child relationships self.span_contexts: dict[int, trace_api.SpanContext] = {} self.meter: Meter | None = None + self.meter_provider: MeterProvider | None = None self.hist_llm_duration: Histogram | None = None + self.hist_token_usage: Histogram | None = None + self.hist_time_to_first_token: Histogram | None = None + self.hist_time_to_generate: Histogram | None = None + self.hist_trace_duration: Histogram | None = None self.metric_reader: MetricReader | None = None # Metrics exporter and instruments try: - from opentelemetry import metrics from opentelemetry.sdk.metrics import Histogram, MeterProvider from opentelemetry.sdk.metrics.export import AggregationTemporality, PeriodicExportingMetricReader @@ -99,7 +128,7 @@ class TencentTraceClient: use_http_protobuf = protocol in {"http/protobuf", "http-protobuf"} use_http_json = protocol in {"http/json", "http-json"} - # Set preferred temporality for histograms to DELTA + # Tencent APM works best with delta aggregation temporality preferred_temporality: dict[type, AggregationTemporality] = {Histogram: AggregationTemporality.DELTA} def _create_metric_exporter(exporter_cls, **kwargs): @@ -174,23 +203,66 @@ class TencentTraceClient: ) if metric_reader is not None: + # Use instance-level MeterProvider instead of global to support config changes + # without worker restart. Each TencentTraceClient manages its own MeterProvider. provider = MeterProvider(resource=self.resource, metric_readers=[metric_reader]) - metrics.set_meter_provider(provider) - self.meter = metrics.get_meter("dify-sdk", dify_config.project.version) + self.meter_provider = provider + self.meter = provider.get_meter("dify-sdk", dify_config.project.version) + + # LLM operation duration histogram self.hist_llm_duration = self.meter.create_histogram( name=LLM_OPERATION_DURATION, unit="s", description="LLM operation duration (seconds)", ) + + # Token usage histogram with exponential buckets + self.hist_token_usage = self.meter.create_histogram( + name=GEN_AI_TOKEN_USAGE, + unit="token", + description="Number of tokens used in prompt and completions", + ) + + # Time to first token histogram + self.hist_time_to_first_token = self.meter.create_histogram( + name=GEN_AI_SERVER_TIME_TO_FIRST_TOKEN, + unit="s", + description="Time to first token for streaming LLM responses (seconds)", + ) + + # Time to generate histogram + self.hist_time_to_generate = self.meter.create_histogram( + name=GEN_AI_STREAMING_TIME_TO_GENERATE, + unit="s", + description="Total time to generate streaming LLM responses (seconds)", + ) + + # Trace duration histogram + self.hist_trace_duration = self.meter.create_histogram( + name=GEN_AI_TRACE_DURATION, + unit="s", + description="End-to-end GenAI trace duration (seconds)", + ) + self.metric_reader = metric_reader else: self.meter = None + self.meter_provider = None self.hist_llm_duration = None + self.hist_token_usage = None + self.hist_time_to_first_token = None + self.hist_time_to_generate = None + self.hist_trace_duration = None self.metric_reader = None except Exception: logger.exception("[Tencent APM] Metrics initialization failed; metrics disabled") self.meter = None + self.meter_provider = None self.hist_llm_duration = None + self.hist_token_usage = None + self.hist_time_to_first_token = None + self.hist_time_to_generate = None + self.hist_trace_duration = None self.metric_reader = None def add_span(self, span_data: SpanData) -> None: @@ -212,10 +284,158 @@ class TencentTraceClient: if attributes: for k, v in attributes.items(): attrs[k] = str(v) if not isinstance(v, (str, int, float, bool)) else v # type: ignore[assignment] + + logger.info( + "[Tencent Metrics] Metric: %s | Value: %.4f | Attributes: %s", + LLM_OPERATION_DURATION, + latency_seconds, + json.dumps(attrs, ensure_ascii=False), + ) + self.hist_llm_duration.record(latency_seconds, attrs) # type: ignore[attr-defined] except Exception: logger.debug("[Tencent APM] Failed to record LLM duration", exc_info=True) + def record_token_usage( + self, + token_count: int, + token_type: str, + operation_name: str, + request_model: str, + response_model: str, + server_address: str, + provider: str, + ) -> None: + """Record token usage histogram. + + Args: + token_count: Number of tokens used + token_type: "input" or "output" + operation_name: Operation name (e.g., "chat") + request_model: Model used in request + response_model: Model used in response + server_address: Server address + provider: Model provider name + """ + try: + if not hasattr(self, "hist_token_usage") or self.hist_token_usage is None: + return + + attributes = { + "gen_ai.operation.name": operation_name, + "gen_ai.request.model": request_model, + "gen_ai.response.model": response_model, + "gen_ai.system": provider, + "gen_ai.token.type": token_type, + "server.address": server_address, + } + + logger.info( + "[Tencent Metrics] Metric: %s | Value: %d | Attributes: %s", + GEN_AI_TOKEN_USAGE, + token_count, + json.dumps(attributes, ensure_ascii=False), + ) + + self.hist_token_usage.record(token_count, attributes) # type: ignore[attr-defined] + except Exception: + logger.debug("[Tencent APM] Failed to record token usage", exc_info=True) + + def record_time_to_first_token( + self, ttft_seconds: float, provider: str, model: str, operation_name: str = "chat" + ) -> None: + """Record time to first token histogram. + + Args: + ttft_seconds: Time to first token in seconds + provider: Model provider name + model: Model name + operation_name: Operation name (default: "chat") + """ + try: + if not hasattr(self, "hist_time_to_first_token") or self.hist_time_to_first_token is None: + return + + attributes = { + "gen_ai.operation.name": operation_name, + "gen_ai.system": provider, + "gen_ai.request.model": model, + "gen_ai.response.model": model, + "stream": "true", + } + + logger.info( + "[Tencent Metrics] Metric: %s | Value: %.4f | Attributes: %s", + GEN_AI_SERVER_TIME_TO_FIRST_TOKEN, + ttft_seconds, + json.dumps(attributes, ensure_ascii=False), + ) + + self.hist_time_to_first_token.record(ttft_seconds, attributes) # type: ignore[attr-defined] + except Exception: + logger.debug("[Tencent APM] Failed to record time to first token", exc_info=True) + + def record_time_to_generate( + self, ttg_seconds: float, provider: str, model: str, operation_name: str = "chat" + ) -> None: + """Record time to generate histogram. + + Args: + ttg_seconds: Time to generate in seconds + provider: Model provider name + model: Model name + operation_name: Operation name (default: "chat") + """ + try: + if not hasattr(self, "hist_time_to_generate") or self.hist_time_to_generate is None: + return + + attributes = { + "gen_ai.operation.name": operation_name, + "gen_ai.system": provider, + "gen_ai.request.model": model, + "gen_ai.response.model": model, + "stream": "true", + } + + logger.info( + "[Tencent Metrics] Metric: %s | Value: %.4f | Attributes: %s", + GEN_AI_STREAMING_TIME_TO_GENERATE, + ttg_seconds, + json.dumps(attributes, ensure_ascii=False), + ) + + self.hist_time_to_generate.record(ttg_seconds, attributes) # type: ignore[attr-defined] + except Exception: + logger.debug("[Tencent APM] Failed to record time to generate", exc_info=True) + + def record_trace_duration(self, duration_seconds: float, attributes: dict[str, str] | None = None) -> None: + """Record end-to-end trace duration histogram in seconds. + + Args: + duration_seconds: Trace duration in seconds + attributes: Optional attributes (e.g., conversation_mode, app_id) + """ + try: + if not hasattr(self, "hist_trace_duration") or self.hist_trace_duration is None: + return + + attrs: dict[str, str] = {} + if attributes: + for k, v in attributes.items(): + attrs[k] = str(v) if not isinstance(v, (str, int, float, bool)) else v # type: ignore[assignment] + + logger.info( + "[Tencent Metrics] Metric: %s | Value: %.4f | Attributes: %s", + GEN_AI_TRACE_DURATION, + duration_seconds, + json.dumps(attrs, ensure_ascii=False), + ) + + self.hist_trace_duration.record(duration_seconds, attrs) # type: ignore[attr-defined] + except Exception: + logger.debug("[Tencent APM] Failed to record trace duration", exc_info=True) + def _create_and_export_span(self, span_data: SpanData) -> None: """Create span using OpenTelemetry Tracer API""" try: @@ -296,11 +516,19 @@ class TencentTraceClient: if self.tracer_provider: self.tracer_provider.shutdown() + + # Shutdown instance-level meter provider + if self.meter_provider is not None: + try: + self.meter_provider.shutdown() # type: ignore[attr-defined] + except Exception: + logger.debug("[Tencent APM] Error shutting down meter provider", exc_info=True) + if self.metric_reader is not None: try: self.metric_reader.shutdown() # type: ignore[attr-defined] except Exception: - pass + logger.debug("[Tencent APM] Error shutting down metric reader", exc_info=True) except Exception: logger.exception("[Tencent APM] Error during client shutdown") diff --git a/api/core/ops/tencent_trace/entities/tencent_semconv.py b/api/core/ops/tencent_trace/entities/semconv.py similarity index 69% rename from api/core/ops/tencent_trace/entities/tencent_semconv.py rename to api/core/ops/tencent_trace/entities/semconv.py index 5ea6eeacef..cd2dbade8b 100644 --- a/api/core/ops/tencent_trace/entities/tencent_semconv.py +++ b/api/core/ops/tencent_trace/entities/semconv.py @@ -47,6 +47,9 @@ GEN_AI_COMPLETION = "gen_ai.completion" GEN_AI_RESPONSE_FINISH_REASON = "gen_ai.response.finish_reason" +# Streaming Span Attributes +GEN_AI_IS_STREAMING_REQUEST = "llm.is_streaming" # Same as OpenLLMetry semconv + # Tool TOOL_NAME = "tool.name" @@ -62,6 +65,19 @@ INSTRUMENTATION_LANGUAGE = "python" # Metrics LLM_OPERATION_DURATION = "gen_ai.client.operation.duration" +GEN_AI_TOKEN_USAGE = "gen_ai.client.token.usage" +GEN_AI_SERVER_TIME_TO_FIRST_TOKEN = "gen_ai.server.time_to_first_token" +GEN_AI_STREAMING_TIME_TO_GENERATE = "gen_ai.streaming.time_to_generate" +# The LLM trace duration which is exclusive to tencent apm +GEN_AI_TRACE_DURATION = "gen_ai.trace.duration" + +# Token Usage Attributes +GEN_AI_OPERATION_NAME = "gen_ai.operation.name" +GEN_AI_REQUEST_MODEL = "gen_ai.request.model" +GEN_AI_RESPONSE_MODEL = "gen_ai.response.model" +GEN_AI_SYSTEM = "gen_ai.system" +GEN_AI_TOKEN_TYPE = "gen_ai.token.type" +SERVER_ADDRESS = "server.address" class GenAISpanKind(Enum): diff --git a/api/core/ops/tencent_trace/span_builder.py b/api/core/ops/tencent_trace/span_builder.py index 5ba592290d..26e8779e3e 100644 --- a/api/core/ops/tencent_trace/span_builder.py +++ b/api/core/ops/tencent_trace/span_builder.py @@ -14,10 +14,11 @@ from core.ops.entities.trace_entity import ( ToolTraceInfo, WorkflowTraceInfo, ) -from core.ops.tencent_trace.entities.tencent_semconv import ( +from core.ops.tencent_trace.entities.semconv import ( GEN_AI_COMPLETION, GEN_AI_FRAMEWORK, GEN_AI_IS_ENTRY, + GEN_AI_IS_STREAMING_REQUEST, GEN_AI_MODEL_NAME, GEN_AI_PROMPT, GEN_AI_PROVIDER, @@ -156,6 +157,25 @@ class TencentSpanBuilder: outputs = node_execution.outputs or {} usage_data = process_data.get("usage", {}) if "usage" in process_data else outputs.get("usage", {}) + attributes = { + GEN_AI_SESSION_ID: trace_info.metadata.get("conversation_id", ""), + GEN_AI_SPAN_KIND: GenAISpanKind.GENERATION.value, + GEN_AI_FRAMEWORK: "dify", + GEN_AI_MODEL_NAME: process_data.get("model_name", ""), + GEN_AI_PROVIDER: process_data.get("model_provider", ""), + GEN_AI_USAGE_INPUT_TOKENS: str(usage_data.get("prompt_tokens", 0)), + GEN_AI_USAGE_OUTPUT_TOKENS: str(usage_data.get("completion_tokens", 0)), + GEN_AI_USAGE_TOTAL_TOKENS: str(usage_data.get("total_tokens", 0)), + GEN_AI_PROMPT: json.dumps(process_data.get("prompts", []), ensure_ascii=False), + GEN_AI_COMPLETION: str(outputs.get("text", "")), + GEN_AI_RESPONSE_FINISH_REASON: outputs.get("finish_reason", ""), + INPUT_VALUE: json.dumps(process_data.get("prompts", []), ensure_ascii=False), + OUTPUT_VALUE: str(outputs.get("text", "")), + } + + if usage_data.get("time_to_first_token") is not None: + attributes[GEN_AI_IS_STREAMING_REQUEST] = "true" + return SpanData( trace_id=trace_id, parent_span_id=workflow_span_id, @@ -163,21 +183,7 @@ class TencentSpanBuilder: name="GENERATION", start_time=TencentSpanBuilder._get_time_nanoseconds(node_execution.created_at), end_time=TencentSpanBuilder._get_time_nanoseconds(node_execution.finished_at), - attributes={ - GEN_AI_SESSION_ID: trace_info.metadata.get("conversation_id", ""), - GEN_AI_SPAN_KIND: GenAISpanKind.GENERATION.value, - GEN_AI_FRAMEWORK: "dify", - GEN_AI_MODEL_NAME: process_data.get("model_name", ""), - GEN_AI_PROVIDER: process_data.get("model_provider", ""), - GEN_AI_USAGE_INPUT_TOKENS: str(usage_data.get("prompt_tokens", 0)), - GEN_AI_USAGE_OUTPUT_TOKENS: str(usage_data.get("completion_tokens", 0)), - GEN_AI_USAGE_TOTAL_TOKENS: str(usage_data.get("total_tokens", 0)), - GEN_AI_PROMPT: json.dumps(process_data.get("prompts", []), ensure_ascii=False), - GEN_AI_COMPLETION: str(outputs.get("text", "")), - GEN_AI_RESPONSE_FINISH_REASON: outputs.get("finish_reason", ""), - INPUT_VALUE: json.dumps(process_data.get("prompts", []), ensure_ascii=False), - OUTPUT_VALUE: str(outputs.get("text", "")), - }, + attributes=attributes, status=TencentSpanBuilder._get_workflow_node_status(node_execution), ) @@ -191,6 +197,19 @@ class TencentSpanBuilder: if trace_info.error: status = Status(StatusCode.ERROR, trace_info.error) + attributes = { + GEN_AI_SESSION_ID: trace_info.metadata.get("conversation_id", ""), + GEN_AI_USER_ID: str(user_id), + GEN_AI_SPAN_KIND: GenAISpanKind.WORKFLOW.value, + GEN_AI_FRAMEWORK: "dify", + GEN_AI_IS_ENTRY: "true", + INPUT_VALUE: str(trace_info.inputs or ""), + OUTPUT_VALUE: str(trace_info.outputs or ""), + } + + if trace_info.is_streaming_request: + attributes[GEN_AI_IS_STREAMING_REQUEST] = "true" + return SpanData( trace_id=trace_id, parent_span_id=None, @@ -198,15 +217,7 @@ class TencentSpanBuilder: name="message", start_time=TencentSpanBuilder._get_time_nanoseconds(trace_info.start_time), end_time=TencentSpanBuilder._get_time_nanoseconds(trace_info.end_time), - attributes={ - GEN_AI_SESSION_ID: trace_info.metadata.get("conversation_id", ""), - GEN_AI_USER_ID: str(user_id), - GEN_AI_SPAN_KIND: GenAISpanKind.WORKFLOW.value, - GEN_AI_FRAMEWORK: "dify", - GEN_AI_IS_ENTRY: "true", - INPUT_VALUE: str(trace_info.inputs or ""), - OUTPUT_VALUE: str(trace_info.outputs or ""), - }, + attributes=attributes, status=status, links=links, ) diff --git a/api/core/ops/tencent_trace/tencent_trace.py b/api/core/ops/tencent_trace/tencent_trace.py index 5ef1c61b24..9b3df86e16 100644 --- a/api/core/ops/tencent_trace/tencent_trace.py +++ b/api/core/ops/tencent_trace/tencent_trace.py @@ -90,6 +90,9 @@ class TencentDataTrace(BaseTraceInstance): self._process_workflow_nodes(trace_info, trace_id) + # Record trace duration for entry span + self._record_workflow_trace_duration(trace_info) + except Exception: logger.exception("[Tencent APM] Failed to process workflow trace") @@ -107,6 +110,11 @@ class TencentDataTrace(BaseTraceInstance): self.trace_client.add_span(message_span) + self._record_message_llm_metrics(trace_info) + + # Record trace duration for entry span + self._record_message_trace_duration(trace_info) + except Exception: logger.exception("[Tencent APM] Failed to process message trace") @@ -290,24 +298,219 @@ class TencentDataTrace(BaseTraceInstance): def _record_llm_metrics(self, node_execution: WorkflowNodeExecution) -> None: """Record LLM performance metrics""" try: - if not hasattr(self.trace_client, "record_llm_duration"): - return - process_data = node_execution.process_data or {} - usage = process_data.get("usage", {}) - latency_s = float(usage.get("latency", 0.0)) + outputs = node_execution.outputs or {} + usage = process_data.get("usage", {}) if "usage" in process_data else outputs.get("usage", {}) - if latency_s > 0: - attributes = { - "provider": process_data.get("model_provider", ""), - "model": process_data.get("model_name", ""), - "span_kind": "GENERATION", - } - self.trace_client.record_llm_duration(latency_s, attributes) + model_provider = process_data.get("model_provider", "unknown") + model_name = process_data.get("model_name", "unknown") + model_mode = process_data.get("model_mode", "chat") + + # Record LLM duration + if hasattr(self.trace_client, "record_llm_duration"): + latency_s = float(usage.get("latency", 0.0)) + + if latency_s > 0: + # Determine if streaming from usage metrics + is_streaming = usage.get("time_to_first_token") is not None + + attributes = { + "gen_ai.system": model_provider, + "gen_ai.response.model": model_name, + "gen_ai.operation.name": model_mode, + "stream": "true" if is_streaming else "false", + } + self.trace_client.record_llm_duration(latency_s, attributes) + + # Record streaming metrics from usage + time_to_first_token = usage.get("time_to_first_token") + if time_to_first_token is not None and hasattr(self.trace_client, "record_time_to_first_token"): + ttft_seconds = float(time_to_first_token) + if ttft_seconds > 0: + self.trace_client.record_time_to_first_token( + ttft_seconds=ttft_seconds, provider=model_provider, model=model_name, operation_name=model_mode + ) + + time_to_generate = usage.get("time_to_generate") + if time_to_generate is not None and hasattr(self.trace_client, "record_time_to_generate"): + ttg_seconds = float(time_to_generate) + if ttg_seconds > 0: + self.trace_client.record_time_to_generate( + ttg_seconds=ttg_seconds, provider=model_provider, model=model_name, operation_name=model_mode + ) + + # Record token usage + if hasattr(self.trace_client, "record_token_usage"): + # Extract token counts + input_tokens = int(usage.get("prompt_tokens", 0)) + output_tokens = int(usage.get("completion_tokens", 0)) + + if input_tokens > 0 or output_tokens > 0: + server_address = f"{model_provider}" + + # Record input tokens + if input_tokens > 0: + self.trace_client.record_token_usage( + token_count=input_tokens, + token_type="input", + operation_name=model_mode, + request_model=model_name, + response_model=model_name, + server_address=server_address, + provider=model_provider, + ) + + # Record output tokens + if output_tokens > 0: + self.trace_client.record_token_usage( + token_count=output_tokens, + token_type="output", + operation_name=model_mode, + request_model=model_name, + response_model=model_name, + server_address=server_address, + provider=model_provider, + ) except Exception: logger.debug("[Tencent APM] Failed to record LLM metrics") + def _record_message_llm_metrics(self, trace_info: MessageTraceInfo) -> None: + """Record LLM metrics for message traces""" + try: + trace_metadata = trace_info.metadata or {} + message_data = trace_info.message_data or {} + provider_latency = 0.0 + if isinstance(message_data, dict): + provider_latency = float(message_data.get("provider_response_latency", 0.0) or 0.0) + else: + provider_latency = float(getattr(message_data, "provider_response_latency", 0.0) or 0.0) + + model_provider = trace_metadata.get("ls_provider") or ( + message_data.get("model_provider", "") if isinstance(message_data, dict) else "" + ) + model_name = trace_metadata.get("ls_model_name") or ( + message_data.get("model_id", "") if isinstance(message_data, dict) else "" + ) + + # Record LLM duration + if provider_latency > 0 and hasattr(self.trace_client, "record_llm_duration"): + is_streaming = trace_info.is_streaming_request + + duration_attributes = { + "gen_ai.system": model_provider, + "gen_ai.response.model": model_name, + "gen_ai.operation.name": "chat", # Message traces are always chat + "stream": "true" if is_streaming else "false", + } + self.trace_client.record_llm_duration(provider_latency, duration_attributes) + + # Record streaming metrics for message traces + if trace_info.is_streaming_request: + # Record time to first token + if trace_info.gen_ai_server_time_to_first_token is not None and hasattr( + self.trace_client, "record_time_to_first_token" + ): + ttft_seconds = float(trace_info.gen_ai_server_time_to_first_token) + if ttft_seconds > 0: + self.trace_client.record_time_to_first_token( + ttft_seconds=ttft_seconds, provider=str(model_provider or ""), model=str(model_name or "") + ) + + # Record time to generate + if trace_info.llm_streaming_time_to_generate is not None and hasattr( + self.trace_client, "record_time_to_generate" + ): + ttg_seconds = float(trace_info.llm_streaming_time_to_generate) + if ttg_seconds > 0: + self.trace_client.record_time_to_generate( + ttg_seconds=ttg_seconds, provider=str(model_provider or ""), model=str(model_name or "") + ) + + # Record token usage + if hasattr(self.trace_client, "record_token_usage"): + input_tokens = int(trace_info.message_tokens or 0) + output_tokens = int(trace_info.answer_tokens or 0) + + if input_tokens > 0: + self.trace_client.record_token_usage( + token_count=input_tokens, + token_type="input", + operation_name="chat", + request_model=str(model_name or ""), + response_model=str(model_name or ""), + server_address=str(model_provider or ""), + provider=str(model_provider or ""), + ) + + if output_tokens > 0: + self.trace_client.record_token_usage( + token_count=output_tokens, + token_type="output", + operation_name="chat", + request_model=str(model_name or ""), + response_model=str(model_name or ""), + server_address=str(model_provider or ""), + provider=str(model_provider or ""), + ) + + except Exception: + logger.debug("[Tencent APM] Failed to record message LLM metrics") + + def _record_workflow_trace_duration(self, trace_info: WorkflowTraceInfo) -> None: + """Record end-to-end workflow trace duration.""" + try: + if not hasattr(self.trace_client, "record_trace_duration"): + return + + # Calculate duration from start_time and end_time to match span duration + if trace_info.start_time and trace_info.end_time: + duration_s = (trace_info.end_time - trace_info.start_time).total_seconds() + else: + # Fallback to workflow_run_elapsed_time if timestamps not available + duration_s = float(trace_info.workflow_run_elapsed_time) + + if duration_s > 0: + attributes = { + "conversation_mode": "workflow", + "workflow_status": trace_info.workflow_run_status, + } + + # Add conversation_id if available + if trace_info.conversation_id: + attributes["has_conversation"] = "true" + else: + attributes["has_conversation"] = "false" + + self.trace_client.record_trace_duration(duration_s, attributes) + + except Exception: + logger.debug("[Tencent APM] Failed to record workflow trace duration") + + def _record_message_trace_duration(self, trace_info: MessageTraceInfo) -> None: + """Record end-to-end message trace duration.""" + try: + if not hasattr(self.trace_client, "record_trace_duration"): + return + + # Calculate duration from start_time and end_time + if trace_info.start_time and trace_info.end_time: + duration = (trace_info.end_time - trace_info.start_time).total_seconds() + + if duration > 0: + attributes = { + "conversation_mode": trace_info.conversation_mode, + } + + # Add streaming flag if available + if hasattr(trace_info, "is_streaming_request"): + attributes["stream"] = "true" if trace_info.is_streaming_request else "false" + + self.trace_client.record_trace_duration(duration, attributes) + + except Exception: + logger.debug("[Tencent APM] Failed to record message trace duration") + def __del__(self): """Ensure proper cleanup on garbage collection.""" try: diff --git a/api/core/plugin/backwards_invocation/app.py b/api/core/plugin/backwards_invocation/app.py index 32ac132e1e..32e8ef385c 100644 --- a/api/core/plugin/backwards_invocation/app.py +++ b/api/core/plugin/backwards_invocation/app.py @@ -4,7 +4,6 @@ from typing import Union from sqlalchemy import select from sqlalchemy.orm import Session -from controllers.service_api.wraps import create_or_update_end_user_for_user_id from core.app.app_config.common.parameters_mapping import get_parameters_from_feature_dict from core.app.apps.advanced_chat.app_generator import AdvancedChatAppGenerator from core.app.apps.agent_chat.app_generator import AgentChatAppGenerator @@ -16,6 +15,7 @@ from core.plugin.backwards_invocation.base import BaseBackwardsInvocation from extensions.ext_database import db from models import Account from models.model import App, AppMode, EndUser +from services.end_user_service import EndUserService class PluginAppBackwardsInvocation(BaseBackwardsInvocation): @@ -64,7 +64,7 @@ class PluginAppBackwardsInvocation(BaseBackwardsInvocation): """ app = cls._get_app(app_id, tenant_id) if not user_id: - user = create_or_update_end_user_for_user_id(app) + user = EndUserService.get_or_create_end_user(app) else: user = cls._get_user(user_id) diff --git a/api/core/plugin/entities/parameters.py b/api/core/plugin/entities/parameters.py index 1e7f8e4c86..88a3a7bd43 100644 --- a/api/core/plugin/entities/parameters.py +++ b/api/core/plugin/entities/parameters.py @@ -39,7 +39,7 @@ class PluginParameterType(StrEnum): TOOLS_SELECTOR = CommonParameterType.TOOLS_SELECTOR ANY = CommonParameterType.ANY DYNAMIC_SELECT = CommonParameterType.DYNAMIC_SELECT - + CHECKBOX = CommonParameterType.CHECKBOX # deprecated, should not use. SYSTEM_FILES = CommonParameterType.SYSTEM_FILES @@ -94,6 +94,7 @@ def as_normal_type(typ: StrEnum): if typ.value in { PluginParameterType.SECRET_INPUT, PluginParameterType.SELECT, + PluginParameterType.CHECKBOX, }: return "string" return typ.value @@ -102,7 +103,13 @@ def as_normal_type(typ: StrEnum): def cast_parameter_value(typ: StrEnum, value: Any, /): try: match typ.value: - case PluginParameterType.STRING | PluginParameterType.SECRET_INPUT | PluginParameterType.SELECT: + case ( + PluginParameterType.STRING + | PluginParameterType.SECRET_INPUT + | PluginParameterType.SELECT + | PluginParameterType.CHECKBOX + | PluginParameterType.DYNAMIC_SELECT + ): if value is None: return "" else: diff --git a/api/core/plugin/entities/plugin.py b/api/core/plugin/entities/plugin.py index f32b356937..9e1a9edf82 100644 --- a/api/core/plugin/entities/plugin.py +++ b/api/core/plugin/entities/plugin.py @@ -13,6 +13,7 @@ from core.plugin.entities.base import BasePluginEntity from core.plugin.entities.endpoint import EndpointProviderDeclaration from core.tools.entities.common_entities import I18nObject from core.tools.entities.tool_entities import ToolProviderEntity +from core.trigger.entities.entities import TriggerProviderEntity class PluginInstallationSource(StrEnum): @@ -63,6 +64,7 @@ class PluginCategory(StrEnum): Extension = auto() AgentStrategy = "agent-strategy" Datasource = "datasource" + Trigger = "trigger" class PluginDeclaration(BaseModel): @@ -71,6 +73,7 @@ class PluginDeclaration(BaseModel): models: list[str] | None = Field(default_factory=list[str]) endpoints: list[str] | None = Field(default_factory=list[str]) datasources: list[str] | None = Field(default_factory=list[str]) + triggers: list[str] | None = Field(default_factory=list[str]) class Meta(BaseModel): minimum_dify_version: str | None = Field(default=None) @@ -106,6 +109,7 @@ class PluginDeclaration(BaseModel): endpoint: EndpointProviderDeclaration | None = None agent_strategy: AgentStrategyProviderEntity | None = None datasource: DatasourceProviderEntity | None = None + trigger: TriggerProviderEntity | None = None meta: Meta @field_validator("version") @@ -129,6 +133,8 @@ class PluginDeclaration(BaseModel): values["category"] = PluginCategory.Datasource elif values.get("agent_strategy"): values["category"] = PluginCategory.AgentStrategy + elif values.get("trigger"): + values["category"] = PluginCategory.Trigger else: values["category"] = PluginCategory.Extension return values diff --git a/api/core/plugin/entities/plugin_daemon.py b/api/core/plugin/entities/plugin_daemon.py index f15acc16f9..3b83121357 100644 --- a/api/core/plugin/entities/plugin_daemon.py +++ b/api/core/plugin/entities/plugin_daemon.py @@ -1,3 +1,4 @@ +import enum from collections.abc import Mapping, Sequence from datetime import datetime from enum import StrEnum @@ -14,6 +15,7 @@ from core.plugin.entities.parameters import PluginParameterOption from core.plugin.entities.plugin import PluginDeclaration, PluginEntity from core.tools.entities.common_entities import I18nObject from core.tools.entities.tool_entities import ToolProviderEntityWithPlugin +from core.trigger.entities.entities import TriggerProviderEntity T = TypeVar("T", bound=(BaseModel | dict | list | bool | str)) @@ -205,3 +207,53 @@ class PluginListResponse(BaseModel): class PluginDynamicSelectOptionsResponse(BaseModel): options: Sequence[PluginParameterOption] = Field(description="The options of the dynamic select.") + + +class PluginTriggerProviderEntity(BaseModel): + provider: str + plugin_unique_identifier: str + plugin_id: str + declaration: TriggerProviderEntity + + +class CredentialType(enum.StrEnum): + API_KEY = "api-key" + OAUTH2 = "oauth2" + UNAUTHORIZED = "unauthorized" + + def get_name(self): + if self == CredentialType.API_KEY: + return "API KEY" + elif self == CredentialType.OAUTH2: + return "AUTH" + elif self == CredentialType.UNAUTHORIZED: + return "UNAUTHORIZED" + else: + return self.value.replace("-", " ").upper() + + def is_editable(self): + return self == CredentialType.API_KEY + + def is_validate_allowed(self): + return self == CredentialType.API_KEY + + @classmethod + def values(cls): + return [item.value for item in cls] + + @classmethod + def of(cls, credential_type: str) -> "CredentialType": + type_name = credential_type.lower() + if type_name in {"api-key", "api_key"}: + return cls.API_KEY + elif type_name in {"oauth2", "oauth"}: + return cls.OAUTH2 + elif type_name == "unauthorized": + return cls.UNAUTHORIZED + else: + raise ValueError(f"Invalid credential type: {credential_type}") + + +class PluginReadmeResponse(BaseModel): + content: str = Field(description="The readme of the plugin.") + language: str = Field(description="The language of the readme.") diff --git a/api/core/plugin/entities/request.py b/api/core/plugin/entities/request.py index d5df85730b..73d3b8c89c 100644 --- a/api/core/plugin/entities/request.py +++ b/api/core/plugin/entities/request.py @@ -1,5 +1,9 @@ +import binascii +import json +from collections.abc import Mapping from typing import Any, Literal +from flask import Response from pydantic import BaseModel, ConfigDict, Field, field_validator from core.entities.provider_entities import BasicProviderConfig @@ -13,6 +17,7 @@ from core.model_runtime.entities.message_entities import ( UserPromptMessage, ) from core.model_runtime.entities.model_entities import ModelType +from core.plugin.utils.http_parser import deserialize_response from core.workflow.nodes.parameter_extractor.entities import ( ModelConfig as ParameterExtractorModelConfig, ) @@ -237,3 +242,43 @@ class RequestFetchAppInfo(BaseModel): """ app_id: str + + +class TriggerInvokeEventResponse(BaseModel): + variables: Mapping[str, Any] = Field(default_factory=dict) + cancelled: bool = Field(default=False) + + model_config = ConfigDict(protected_namespaces=(), arbitrary_types_allowed=True) + + @field_validator("variables", mode="before") + @classmethod + def convert_variables(cls, v): + if isinstance(v, str): + return json.loads(v) + else: + return v + + +class TriggerSubscriptionResponse(BaseModel): + subscription: dict[str, Any] + + +class TriggerValidateProviderCredentialsResponse(BaseModel): + result: bool + + +class TriggerDispatchResponse(BaseModel): + user_id: str + events: list[str] + response: Response + payload: Mapping[str, Any] = Field(default_factory=dict) + + model_config = ConfigDict(protected_namespaces=(), arbitrary_types_allowed=True) + + @field_validator("response", mode="before") + @classmethod + def convert_response(cls, v: str): + try: + return deserialize_response(binascii.unhexlify(v.encode())) + except Exception as e: + raise ValueError("Failed to deserialize response from hex string") from e diff --git a/api/core/plugin/impl/asset.py b/api/core/plugin/impl/asset.py index b9bfe2d2cf..2798e736a9 100644 --- a/api/core/plugin/impl/asset.py +++ b/api/core/plugin/impl/asset.py @@ -10,3 +10,13 @@ class PluginAssetManager(BasePluginClient): if response.status_code != 200: raise ValueError(f"can not found asset {id}") return response.content + + def extract_asset(self, tenant_id: str, plugin_unique_identifier: str, filename: str) -> bytes: + response = self._request( + method="GET", + path=f"plugin/{tenant_id}/extract-asset/", + params={"plugin_unique_identifier": plugin_unique_identifier, "file_path": filename}, + ) + if response.status_code != 200: + raise ValueError(f"can not found asset {plugin_unique_identifier}, {str(response.status_code)}") + return response.content diff --git a/api/core/plugin/impl/base.py b/api/core/plugin/impl/base.py index e9dc58eec8..a1c84bd5d9 100644 --- a/api/core/plugin/impl/base.py +++ b/api/core/plugin/impl/base.py @@ -29,6 +29,12 @@ from core.plugin.impl.exc import ( PluginPermissionDeniedError, PluginUniqueIdentifierError, ) +from core.trigger.errors import ( + EventIgnoreError, + TriggerInvokeError, + TriggerPluginInvokeError, + TriggerProviderCredentialValidationError, +) plugin_daemon_inner_api_baseurl = URL(str(dify_config.PLUGIN_DAEMON_URL)) _plugin_daemon_timeout_config = cast( @@ -43,7 +49,7 @@ elif isinstance(_plugin_daemon_timeout_config, httpx.Timeout): else: plugin_daemon_request_timeout = httpx.Timeout(_plugin_daemon_timeout_config) -T = TypeVar("T", bound=(BaseModel | dict | list | bool | str)) +T = TypeVar("T", bound=(BaseModel | dict[str, Any] | list[Any] | bool | str)) logger = logging.getLogger(__name__) @@ -53,10 +59,10 @@ class BasePluginClient: self, method: str, path: str, - headers: dict | None = None, - data: bytes | dict | str | None = None, - params: dict | None = None, - files: dict | None = None, + headers: dict[str, str] | None = None, + data: bytes | dict[str, Any] | str | None = None, + params: dict[str, Any] | None = None, + files: dict[str, Any] | None = None, ) -> httpx.Response: """ Make a request to the plugin daemon inner API. @@ -87,17 +93,17 @@ class BasePluginClient: def _prepare_request( self, path: str, - headers: dict | None, - data: bytes | dict | str | None, - params: dict | None, - files: dict | None, - ) -> tuple[str, dict, bytes | dict | str | None, dict | None, dict | None]: + headers: dict[str, str] | None, + data: bytes | dict[str, Any] | str | None, + params: dict[str, Any] | None, + files: dict[str, Any] | None, + ) -> tuple[str, dict[str, str], bytes | dict[str, Any] | str | None, dict[str, Any] | None, dict[str, Any] | None]: url = plugin_daemon_inner_api_baseurl / path prepared_headers = dict(headers or {}) prepared_headers["X-Api-Key"] = dify_config.PLUGIN_DAEMON_KEY prepared_headers.setdefault("Accept-Encoding", "gzip, deflate, br") - prepared_data: bytes | dict | str | None = ( + prepared_data: bytes | dict[str, Any] | str | None = ( data if isinstance(data, (bytes, str, dict)) or data is None else None ) if isinstance(data, dict): @@ -112,10 +118,10 @@ class BasePluginClient: self, method: str, path: str, - params: dict | None = None, - headers: dict | None = None, - data: bytes | dict | None = None, - files: dict | None = None, + params: dict[str, Any] | None = None, + headers: dict[str, str] | None = None, + data: bytes | dict[str, Any] | None = None, + files: dict[str, Any] | None = None, ) -> Generator[str, None, None]: """ Make a stream request to the plugin daemon inner API @@ -138,7 +144,7 @@ class BasePluginClient: try: with httpx.stream(**stream_kwargs) as response: for raw_line in response.iter_lines(): - if raw_line is None: + if not raw_line: continue line = raw_line.decode("utf-8") if isinstance(raw_line, bytes) else raw_line line = line.strip() @@ -155,10 +161,10 @@ class BasePluginClient: method: str, path: str, type_: type[T], - headers: dict | None = None, - data: bytes | dict | None = None, - params: dict | None = None, - files: dict | None = None, + headers: dict[str, str] | None = None, + data: bytes | dict[str, Any] | None = None, + params: dict[str, Any] | None = None, + files: dict[str, Any] | None = None, ) -> Generator[T, None, None]: """ Make a stream request to the plugin daemon inner API and yield the response as a model. @@ -171,10 +177,10 @@ class BasePluginClient: method: str, path: str, type_: type[T], - headers: dict | None = None, + headers: dict[str, str] | None = None, data: bytes | None = None, - params: dict | None = None, - files: dict | None = None, + params: dict[str, Any] | None = None, + files: dict[str, Any] | None = None, ) -> T: """ Make a request to the plugin daemon inner API and return the response as a model. @@ -187,11 +193,11 @@ class BasePluginClient: method: str, path: str, type_: type[T], - headers: dict | None = None, - data: bytes | dict | None = None, - params: dict | None = None, - files: dict | None = None, - transformer: Callable[[dict], dict] | None = None, + headers: dict[str, str] | None = None, + data: bytes | dict[str, Any] | None = None, + params: dict[str, Any] | None = None, + files: dict[str, Any] | None = None, + transformer: Callable[[dict[str, Any]], dict[str, Any]] | None = None, ) -> T: """ Make a request to the plugin daemon inner API and return the response as a model. @@ -239,10 +245,10 @@ class BasePluginClient: method: str, path: str, type_: type[T], - headers: dict | None = None, - data: bytes | dict | None = None, - params: dict | None = None, - files: dict | None = None, + headers: dict[str, str] | None = None, + data: bytes | dict[str, Any] | None = None, + params: dict[str, Any] | None = None, + files: dict[str, Any] | None = None, ) -> Generator[T, None, None]: """ Make a stream request to the plugin daemon inner API and yield the response as a model. @@ -302,6 +308,14 @@ class BasePluginClient: raise CredentialsValidateFailedError(error_object.get("message")) case EndpointSetupFailedError.__name__: raise EndpointSetupFailedError(error_object.get("message")) + case TriggerProviderCredentialValidationError.__name__: + raise TriggerProviderCredentialValidationError(error_object.get("message")) + case TriggerPluginInvokeError.__name__: + raise TriggerPluginInvokeError(description=error_object.get("description")) + case TriggerInvokeError.__name__: + raise TriggerInvokeError(error_object.get("message")) + case EventIgnoreError.__name__: + raise EventIgnoreError(description=error_object.get("description")) case _: raise PluginInvokeError(description=message) case PluginDaemonInternalServerError.__name__: diff --git a/api/core/plugin/impl/dynamic_select.py b/api/core/plugin/impl/dynamic_select.py index 24839849b9..0a580a2978 100644 --- a/api/core/plugin/impl/dynamic_select.py +++ b/api/core/plugin/impl/dynamic_select.py @@ -15,6 +15,7 @@ class DynamicSelectClient(BasePluginClient): provider: str, action: str, credentials: Mapping[str, Any], + credential_type: str, parameter: str, ) -> PluginDynamicSelectOptionsResponse: """ @@ -29,6 +30,7 @@ class DynamicSelectClient(BasePluginClient): "data": { "provider": GenericProviderID(provider).provider_name, "credentials": credentials, + "credential_type": credential_type, "provider_action": action, "parameter": parameter, }, diff --git a/api/core/plugin/impl/exc.py b/api/core/plugin/impl/exc.py index e28a324217..4cabdc1732 100644 --- a/api/core/plugin/impl/exc.py +++ b/api/core/plugin/impl/exc.py @@ -58,6 +58,20 @@ class PluginInvokeError(PluginDaemonClientSideError, ValueError): except Exception: return self.description + def to_user_friendly_error(self, plugin_name: str = "currently running plugin") -> str: + """ + Convert the error to a user-friendly error message. + + :param plugin_name: The name of the plugin that caused the error. + :return: A user-friendly error message. + """ + return ( + f"An error occurred in the {plugin_name}, " + f"please contact the author of {plugin_name} for help, " + f"error type: {self.get_error_type()}, " + f"error details: {self.get_error_message()}" + ) + class PluginUniqueIdentifierError(PluginDaemonClientSideError): description: str = "Unique Identifier Error" diff --git a/api/core/plugin/impl/plugin.py b/api/core/plugin/impl/plugin.py index 18b5fa8af6..0bbb62af93 100644 --- a/api/core/plugin/impl/plugin.py +++ b/api/core/plugin/impl/plugin.py @@ -1,5 +1,7 @@ from collections.abc import Sequence +from requests import HTTPError + from core.plugin.entities.bundle import PluginBundleDependency from core.plugin.entities.plugin import ( MissingPluginDependency, @@ -13,12 +15,35 @@ from core.plugin.entities.plugin_daemon import ( PluginInstallTask, PluginInstallTaskStartResponse, PluginListResponse, + PluginReadmeResponse, ) from core.plugin.impl.base import BasePluginClient from models.provider_ids import GenericProviderID class PluginInstaller(BasePluginClient): + def fetch_plugin_readme(self, tenant_id: str, plugin_unique_identifier: str, language: str) -> str: + """ + Fetch plugin readme + """ + try: + response = self._request_with_plugin_daemon_response( + "GET", + f"plugin/{tenant_id}/management/fetch/readme", + PluginReadmeResponse, + params={ + "tenant_id": tenant_id, + "plugin_unique_identifier": plugin_unique_identifier, + "language": language, + }, + ) + return response.content + except HTTPError as e: + message = e.args[0] + if "404" in message: + return "" + raise e + def fetch_plugin_by_identifier( self, tenant_id: str, diff --git a/api/core/plugin/impl/tool.py b/api/core/plugin/impl/tool.py index bc4de38099..6fa5136b42 100644 --- a/api/core/plugin/impl/tool.py +++ b/api/core/plugin/impl/tool.py @@ -3,14 +3,12 @@ from typing import Any from pydantic import BaseModel -from core.plugin.entities.plugin_daemon import ( - PluginBasicBooleanResponse, - PluginToolProviderEntity, -) +# from core.plugin.entities.plugin import GenericProviderID, ToolProviderID +from core.plugin.entities.plugin_daemon import CredentialType, PluginBasicBooleanResponse, PluginToolProviderEntity from core.plugin.impl.base import BasePluginClient from core.plugin.utils.chunk_merger import merge_blob_chunks from core.schemas.resolver import resolve_dify_schema_refs -from core.tools.entities.tool_entities import CredentialType, ToolInvokeMessage, ToolParameter +from core.tools.entities.tool_entities import ToolInvokeMessage, ToolParameter from models.provider_ids import GenericProviderID, ToolProviderID diff --git a/api/core/plugin/impl/trigger.py b/api/core/plugin/impl/trigger.py new file mode 100644 index 0000000000..611ce74907 --- /dev/null +++ b/api/core/plugin/impl/trigger.py @@ -0,0 +1,305 @@ +import binascii +from collections.abc import Generator, Mapping +from typing import Any + +from flask import Request + +from core.plugin.entities.plugin_daemon import CredentialType, PluginTriggerProviderEntity +from core.plugin.entities.request import ( + TriggerDispatchResponse, + TriggerInvokeEventResponse, + TriggerSubscriptionResponse, + TriggerValidateProviderCredentialsResponse, +) +from core.plugin.impl.base import BasePluginClient +from core.plugin.utils.http_parser import serialize_request +from core.trigger.entities.entities import Subscription +from models.provider_ids import TriggerProviderID + + +class PluginTriggerClient(BasePluginClient): + def fetch_trigger_providers(self, tenant_id: str) -> list[PluginTriggerProviderEntity]: + """ + Fetch trigger providers for the given tenant. + """ + + def transformer(json_response: dict[str, Any]) -> dict[str, Any]: + for provider in json_response.get("data", []): + declaration = provider.get("declaration", {}) or {} + provider_id = provider.get("plugin_id") + "/" + provider.get("provider") + for event in declaration.get("events", []): + event["identity"]["provider"] = provider_id + + return json_response + + response: list[PluginTriggerProviderEntity] = self._request_with_plugin_daemon_response( + method="GET", + path=f"plugin/{tenant_id}/management/triggers", + type_=list[PluginTriggerProviderEntity], + params={"page": 1, "page_size": 256}, + transformer=transformer, + ) + + for provider in response: + provider.declaration.identity.name = f"{provider.plugin_id}/{provider.declaration.identity.name}" + + # override the provider name for each trigger to plugin_id/provider_name + for event in provider.declaration.events: + event.identity.provider = provider.declaration.identity.name + + return response + + def fetch_trigger_provider(self, tenant_id: str, provider_id: TriggerProviderID) -> PluginTriggerProviderEntity: + """ + Fetch trigger provider for the given tenant and plugin. + """ + + def transformer(json_response: dict[str, Any]) -> dict[str, Any]: + data = json_response.get("data") + if data: + for event in data.get("declaration", {}).get("events", []): + event["identity"]["provider"] = str(provider_id) + + return json_response + + response: PluginTriggerProviderEntity = self._request_with_plugin_daemon_response( + method="GET", + path=f"plugin/{tenant_id}/management/trigger", + type_=PluginTriggerProviderEntity, + params={"provider": provider_id.provider_name, "plugin_id": provider_id.plugin_id}, + transformer=transformer, + ) + + response.declaration.identity.name = str(provider_id) + + # override the provider name for each trigger to plugin_id/provider_name + for event in response.declaration.events: + event.identity.provider = str(provider_id) + + return response + + def invoke_trigger_event( + self, + tenant_id: str, + user_id: str, + provider: str, + event_name: str, + credentials: Mapping[str, str], + credential_type: CredentialType, + request: Request, + parameters: Mapping[str, Any], + subscription: Subscription, + payload: Mapping[str, Any], + ) -> TriggerInvokeEventResponse: + """ + Invoke a trigger with the given parameters. + """ + provider_id = TriggerProviderID(provider) + response: Generator[TriggerInvokeEventResponse, None, None] = self._request_with_plugin_daemon_response_stream( + method="POST", + path=f"plugin/{tenant_id}/dispatch/trigger/invoke_event", + type_=TriggerInvokeEventResponse, + data={ + "user_id": user_id, + "data": { + "provider": provider_id.provider_name, + "event": event_name, + "credentials": credentials, + "credential_type": credential_type, + "subscription": subscription.model_dump(), + "raw_http_request": binascii.hexlify(serialize_request(request)).decode(), + "parameters": parameters, + "payload": payload, + }, + }, + headers={ + "X-Plugin-ID": provider_id.plugin_id, + "Content-Type": "application/json", + }, + ) + + for resp in response: + return resp + + raise ValueError("No response received from plugin daemon for invoke trigger") + + def validate_provider_credentials( + self, tenant_id: str, user_id: str, provider: str, credentials: Mapping[str, str] + ) -> bool: + """ + Validate the credentials of the trigger provider. + """ + provider_id = TriggerProviderID(provider) + response: Generator[TriggerValidateProviderCredentialsResponse, None, None] = ( + self._request_with_plugin_daemon_response_stream( + method="POST", + path=f"plugin/{tenant_id}/dispatch/trigger/validate_credentials", + type_=TriggerValidateProviderCredentialsResponse, + data={ + "user_id": user_id, + "data": { + "provider": provider_id.provider_name, + "credentials": credentials, + }, + }, + headers={ + "X-Plugin-ID": provider_id.plugin_id, + "Content-Type": "application/json", + }, + ) + ) + + for resp in response: + return resp.result + + raise ValueError("No response received from plugin daemon for validate provider credentials") + + def dispatch_event( + self, + tenant_id: str, + provider: str, + subscription: Mapping[str, Any], + request: Request, + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> TriggerDispatchResponse: + """ + Dispatch an event to triggers. + """ + provider_id = TriggerProviderID(provider) + response = self._request_with_plugin_daemon_response_stream( + method="POST", + path=f"plugin/{tenant_id}/dispatch/trigger/dispatch_event", + type_=TriggerDispatchResponse, + data={ + "data": { + "provider": provider_id.provider_name, + "subscription": subscription, + "credentials": credentials, + "credential_type": credential_type, + "raw_http_request": binascii.hexlify(serialize_request(request)).decode(), + }, + }, + headers={ + "X-Plugin-ID": provider_id.plugin_id, + "Content-Type": "application/json", + }, + ) + + for resp in response: + return resp + + raise ValueError("No response received from plugin daemon for dispatch event") + + def subscribe( + self, + tenant_id: str, + user_id: str, + provider: str, + credentials: Mapping[str, str], + credential_type: CredentialType, + endpoint: str, + parameters: Mapping[str, Any], + ) -> TriggerSubscriptionResponse: + """ + Subscribe to a trigger. + """ + provider_id = TriggerProviderID(provider) + response: Generator[TriggerSubscriptionResponse, None, None] = self._request_with_plugin_daemon_response_stream( + method="POST", + path=f"plugin/{tenant_id}/dispatch/trigger/subscribe", + type_=TriggerSubscriptionResponse, + data={ + "user_id": user_id, + "data": { + "provider": provider_id.provider_name, + "credentials": credentials, + "credential_type": credential_type, + "endpoint": endpoint, + "parameters": parameters, + }, + }, + headers={ + "X-Plugin-ID": provider_id.plugin_id, + "Content-Type": "application/json", + }, + ) + + for resp in response: + return resp + + raise ValueError("No response received from plugin daemon for subscribe") + + def unsubscribe( + self, + tenant_id: str, + user_id: str, + provider: str, + subscription: Subscription, + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> TriggerSubscriptionResponse: + """ + Unsubscribe from a trigger. + """ + provider_id = TriggerProviderID(provider) + response: Generator[TriggerSubscriptionResponse, None, None] = self._request_with_plugin_daemon_response_stream( + method="POST", + path=f"plugin/{tenant_id}/dispatch/trigger/unsubscribe", + type_=TriggerSubscriptionResponse, + data={ + "user_id": user_id, + "data": { + "provider": provider_id.provider_name, + "subscription": subscription.model_dump(), + "credentials": credentials, + "credential_type": credential_type, + }, + }, + headers={ + "X-Plugin-ID": provider_id.plugin_id, + "Content-Type": "application/json", + }, + ) + + for resp in response: + return resp + + raise ValueError("No response received from plugin daemon for unsubscribe") + + def refresh( + self, + tenant_id: str, + user_id: str, + provider: str, + subscription: Subscription, + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> TriggerSubscriptionResponse: + """ + Refresh a trigger subscription. + """ + provider_id = TriggerProviderID(provider) + response: Generator[TriggerSubscriptionResponse, None, None] = self._request_with_plugin_daemon_response_stream( + method="POST", + path=f"plugin/{tenant_id}/dispatch/trigger/refresh", + type_=TriggerSubscriptionResponse, + data={ + "user_id": user_id, + "data": { + "provider": provider_id.provider_name, + "subscription": subscription.model_dump(), + "credentials": credentials, + "credential_type": credential_type, + }, + }, + headers={ + "X-Plugin-ID": provider_id.plugin_id, + "Content-Type": "application/json", + }, + ) + + for resp in response: + return resp + + raise ValueError("No response received from plugin daemon for refresh") diff --git a/api/core/plugin/utils/http_parser.py b/api/core/plugin/utils/http_parser.py new file mode 100644 index 0000000000..ce943929be --- /dev/null +++ b/api/core/plugin/utils/http_parser.py @@ -0,0 +1,163 @@ +from io import BytesIO + +from flask import Request, Response +from werkzeug.datastructures import Headers + + +def serialize_request(request: Request) -> bytes: + method = request.method + path = request.full_path.rstrip("?") + raw = f"{method} {path} HTTP/1.1\r\n".encode() + + for name, value in request.headers.items(): + raw += f"{name}: {value}\r\n".encode() + + raw += b"\r\n" + + body = request.get_data(as_text=False) + if body: + raw += body + + return raw + + +def deserialize_request(raw_data: bytes) -> Request: + header_end = raw_data.find(b"\r\n\r\n") + if header_end == -1: + header_end = raw_data.find(b"\n\n") + if header_end == -1: + header_data = raw_data + body = b"" + else: + header_data = raw_data[:header_end] + body = raw_data[header_end + 2 :] + else: + header_data = raw_data[:header_end] + body = raw_data[header_end + 4 :] + + lines = header_data.split(b"\r\n") + if len(lines) == 1 and b"\n" in lines[0]: + lines = header_data.split(b"\n") + + if not lines or not lines[0]: + raise ValueError("Empty HTTP request") + + request_line = lines[0].decode("utf-8", errors="ignore") + parts = request_line.split(" ", 2) + if len(parts) < 2: + raise ValueError(f"Invalid request line: {request_line}") + + method = parts[0] + full_path = parts[1] + protocol = parts[2] if len(parts) > 2 else "HTTP/1.1" + + if "?" in full_path: + path, query_string = full_path.split("?", 1) + else: + path = full_path + query_string = "" + + headers = Headers() + for line in lines[1:]: + if not line: + continue + line_str = line.decode("utf-8", errors="ignore") + if ":" not in line_str: + continue + name, value = line_str.split(":", 1) + headers.add(name, value.strip()) + + host = headers.get("Host", "localhost") + if ":" in host: + server_name, server_port = host.rsplit(":", 1) + else: + server_name = host + server_port = "80" + + environ = { + "REQUEST_METHOD": method, + "PATH_INFO": path, + "QUERY_STRING": query_string, + "SERVER_NAME": server_name, + "SERVER_PORT": server_port, + "SERVER_PROTOCOL": protocol, + "wsgi.input": BytesIO(body), + "wsgi.url_scheme": "http", + } + + if "Content-Type" in headers: + content_type = headers.get("Content-Type") + if content_type is not None: + environ["CONTENT_TYPE"] = content_type + + if "Content-Length" in headers: + content_length = headers.get("Content-Length") + if content_length is not None: + environ["CONTENT_LENGTH"] = content_length + elif body: + environ["CONTENT_LENGTH"] = str(len(body)) + + for name, value in headers.items(): + if name.upper() in ("CONTENT-TYPE", "CONTENT-LENGTH"): + continue + env_name = f"HTTP_{name.upper().replace('-', '_')}" + environ[env_name] = value + + return Request(environ) + + +def serialize_response(response: Response) -> bytes: + raw = f"HTTP/1.1 {response.status}\r\n".encode() + + for name, value in response.headers.items(): + raw += f"{name}: {value}\r\n".encode() + + raw += b"\r\n" + + body = response.get_data(as_text=False) + if body: + raw += body + + return raw + + +def deserialize_response(raw_data: bytes) -> Response: + header_end = raw_data.find(b"\r\n\r\n") + if header_end == -1: + header_end = raw_data.find(b"\n\n") + if header_end == -1: + header_data = raw_data + body = b"" + else: + header_data = raw_data[:header_end] + body = raw_data[header_end + 2 :] + else: + header_data = raw_data[:header_end] + body = raw_data[header_end + 4 :] + + lines = header_data.split(b"\r\n") + if len(lines) == 1 and b"\n" in lines[0]: + lines = header_data.split(b"\n") + + if not lines or not lines[0]: + raise ValueError("Empty HTTP response") + + status_line = lines[0].decode("utf-8", errors="ignore") + parts = status_line.split(" ", 2) + if len(parts) < 2: + raise ValueError(f"Invalid status line: {status_line}") + + status_code = int(parts[1]) + + response = Response(response=body, status=status_code) + + for line in lines[1:]: + if not line: + continue + line_str = line.decode("utf-8", errors="ignore") + if ":" not in line_str: + continue + name, value = line_str.split(":", 1) + response.headers[name] = value.strip() + + return response diff --git a/api/core/rag/datasource/vdb/elasticsearch/elasticsearch_vector.py b/api/core/rag/datasource/vdb/elasticsearch/elasticsearch_vector.py index 0ff8c915e6..1470713b88 100644 --- a/api/core/rag/datasource/vdb/elasticsearch/elasticsearch_vector.py +++ b/api/core/rag/datasource/vdb/elasticsearch/elasticsearch_vector.py @@ -147,7 +147,8 @@ class ElasticSearchVector(BaseVector): def _get_version(self) -> str: info = self._client.info() - return cast(str, info["version"]["number"]) + # remove any suffix like "-SNAPSHOT" from the version string + return cast(str, info["version"]["number"]).split("-")[0] def _check_version(self): if parse_version(self._version) < parse_version("8.0.0"): diff --git a/api/core/rag/datasource/vdb/opensearch/opensearch_vector.py b/api/core/rag/datasource/vdb/opensearch/opensearch_vector.py index 80ffdadd96..2f77776807 100644 --- a/api/core/rag/datasource/vdb/opensearch/opensearch_vector.py +++ b/api/core/rag/datasource/vdb/opensearch/opensearch_vector.py @@ -161,7 +161,7 @@ class OpenSearchVector(BaseVector): logger.exception("Error deleting document: %s", error) def delete(self): - self._client.indices.delete(index=self._collection_name.lower()) + self._client.indices.delete(index=self._collection_name.lower(), ignore_unavailable=True) def text_exists(self, id: str) -> bool: try: diff --git a/api/core/rag/datasource/vdb/weaviate/weaviate_vector.py b/api/core/rag/datasource/vdb/weaviate/weaviate_vector.py index d2d8fcf964..591de01669 100644 --- a/api/core/rag/datasource/vdb/weaviate/weaviate_vector.py +++ b/api/core/rag/datasource/vdb/weaviate/weaviate_vector.py @@ -39,11 +39,13 @@ class WeaviateConfig(BaseModel): Attributes: endpoint: Weaviate server endpoint URL + grpc_endpoint: Optional Weaviate gRPC server endpoint URL api_key: Optional API key for authentication batch_size: Number of objects to batch per insert operation """ endpoint: str + grpc_endpoint: str | None = None api_key: str | None = None batch_size: int = 100 @@ -88,9 +90,22 @@ class WeaviateVector(BaseVector): http_secure = p.scheme == "https" http_port = p.port or (443 if http_secure else 80) - grpc_host = host - grpc_secure = http_secure - grpc_port = 443 if grpc_secure else 50051 + # Parse gRPC configuration + if config.grpc_endpoint: + # Urls without scheme won't be parsed correctly in some python versions, + # see https://bugs.python.org/issue27657 + grpc_endpoint_with_scheme = ( + config.grpc_endpoint if "://" in config.grpc_endpoint else f"grpc://{config.grpc_endpoint}" + ) + grpc_p = urlparse(grpc_endpoint_with_scheme) + grpc_host = grpc_p.hostname or "localhost" + grpc_port = grpc_p.port or (443 if grpc_p.scheme == "grpcs" else 50051) + grpc_secure = grpc_p.scheme == "grpcs" + else: + # Infer from HTTP endpoint as fallback + grpc_host = host + grpc_secure = http_secure + grpc_port = 443 if grpc_secure else 50051 client = weaviate.connect_to_custom( http_host=host, @@ -100,6 +115,7 @@ class WeaviateVector(BaseVector): grpc_port=grpc_port, grpc_secure=grpc_secure, auth_credentials=Auth.api_key(config.api_key) if config.api_key else None, + skip_init_checks=True, # Skip PyPI version check to avoid unnecessary HTTP requests ) if not client.is_ready(): @@ -431,6 +447,7 @@ class WeaviateVectorFactory(AbstractVectorFactory): collection_name=collection_name, config=WeaviateConfig( endpoint=dify_config.WEAVIATE_ENDPOINT or "", + grpc_endpoint=dify_config.WEAVIATE_GRPC_ENDPOINT or "", api_key=dify_config.WEAVIATE_API_KEY, batch_size=dify_config.WEAVIATE_BATCH_SIZE, ), diff --git a/api/core/rag/extractor/word_extractor.py b/api/core/rag/extractor/word_extractor.py index 1a9704688a..c7a5568866 100644 --- a/api/core/rag/extractor/word_extractor.py +++ b/api/core/rag/extractor/word_extractor.py @@ -152,13 +152,15 @@ class WordExtractor(BaseExtractor): # Initialize a row, all of which are empty by default row_cells = [""] * total_cols col_index = 0 - for cell in row.cells: + while col_index < len(row.cells): # make sure the col_index is not out of range - while col_index < total_cols and row_cells[col_index] != "": + while col_index < len(row.cells) and row_cells[col_index] != "": col_index += 1 # if col_index is out of range the loop is jumped - if col_index >= total_cols: + if col_index >= len(row.cells): break + # get the correct cell + cell = row.cells[col_index] cell_content = self._parse_cell(cell, image_map).strip() cell_colspan = cell.grid_span or 1 for i in range(cell_colspan): diff --git a/api/core/rag/pipeline/__init__.py b/api/core/rag/pipeline/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/core/rag/pipeline/queue.py b/api/core/rag/pipeline/queue.py new file mode 100644 index 0000000000..7472598a7f --- /dev/null +++ b/api/core/rag/pipeline/queue.py @@ -0,0 +1,82 @@ +import json +from collections.abc import Sequence +from typing import Any + +from pydantic import BaseModel, ValidationError + +from extensions.ext_redis import redis_client + +_DEFAULT_TASK_TTL = 60 * 60 # 1 hour + + +class TaskWrapper(BaseModel): + data: Any + + def serialize(self) -> str: + return self.model_dump_json() + + @classmethod + def deserialize(cls, serialized_data: str) -> "TaskWrapper": + return cls.model_validate_json(serialized_data) + + +class TenantIsolatedTaskQueue: + """ + Simple queue for tenant isolated tasks, used for rag related tenant tasks isolation. + It uses Redis list to store tasks, and Redis key to store task waiting flag. + Support tasks that can be serialized by json. + """ + + def __init__(self, tenant_id: str, unique_key: str): + self._tenant_id = tenant_id + self._unique_key = unique_key + self._queue = f"tenant_self_{unique_key}_task_queue:{tenant_id}" + self._task_key = f"tenant_{unique_key}_task:{tenant_id}" + + def get_task_key(self): + return redis_client.get(self._task_key) + + def set_task_waiting_time(self, ttl: int = _DEFAULT_TASK_TTL): + redis_client.setex(self._task_key, ttl, 1) + + def delete_task_key(self): + redis_client.delete(self._task_key) + + def push_tasks(self, tasks: Sequence[Any]): + serialized_tasks = [] + for task in tasks: + # Store str list directly, maintaining full compatibility for pipeline scenarios + if isinstance(task, str): + serialized_tasks.append(task) + else: + # Use TaskWrapper to do JSON serialization for non-string tasks + wrapper = TaskWrapper(data=task) + serialized_data = wrapper.serialize() + serialized_tasks.append(serialized_data) + + if not serialized_tasks: + return + + redis_client.lpush(self._queue, *serialized_tasks) + + def pull_tasks(self, count: int = 1) -> Sequence[Any]: + if count <= 0: + return [] + + tasks = [] + for _ in range(count): + serialized_task = redis_client.rpop(self._queue) + if not serialized_task: + break + + if isinstance(serialized_task, bytes): + serialized_task = serialized_task.decode("utf-8") + + try: + wrapper = TaskWrapper.deserialize(serialized_task) + tasks.append(wrapper.data) + except (json.JSONDecodeError, ValidationError, TypeError, ValueError): + # Fall back to raw string for legacy format or invalid JSON + tasks.append(serialized_task) + + return tasks diff --git a/api/core/tools/__base/tool.py b/api/core/tools/__base/tool.py index 6e0462c530..8ca4eabb7a 100644 --- a/api/core/tools/__base/tool.py +++ b/api/core/tools/__base/tool.py @@ -210,10 +210,24 @@ class Tool(ABC): meta=meta, ) - def create_json_message(self, object: dict) -> ToolInvokeMessage: + def create_json_message(self, object: dict, suppress_output: bool = False) -> ToolInvokeMessage: """ create a json message """ return ToolInvokeMessage( - type=ToolInvokeMessage.MessageType.JSON, message=ToolInvokeMessage.JsonMessage(json_object=object) + type=ToolInvokeMessage.MessageType.JSON, + message=ToolInvokeMessage.JsonMessage(json_object=object, suppress_output=suppress_output), + ) + + def create_variable_message( + self, variable_name: str, variable_value: Any, stream: bool = False + ) -> ToolInvokeMessage: + """ + create a variable message + """ + return ToolInvokeMessage( + type=ToolInvokeMessage.MessageType.VARIABLE, + message=ToolInvokeMessage.VariableMessage( + variable_name=variable_name, variable_value=variable_value, stream=stream + ), ) diff --git a/api/core/tools/__base/tool_runtime.py b/api/core/tools/__base/tool_runtime.py index 09bc817c01..961d13f90a 100644 --- a/api/core/tools/__base/tool_runtime.py +++ b/api/core/tools/__base/tool_runtime.py @@ -3,7 +3,8 @@ from typing import Any from pydantic import BaseModel, Field from core.app.entities.app_invoke_entities import InvokeFrom -from core.tools.entities.tool_entities import CredentialType, ToolInvokeFrom +from core.plugin.entities.plugin_daemon import CredentialType +from core.tools.entities.tool_entities import ToolInvokeFrom class ToolRuntime(BaseModel): diff --git a/api/core/tools/builtin_tool/provider.py b/api/core/tools/builtin_tool/provider.py index a391136a5c..50105bd707 100644 --- a/api/core/tools/builtin_tool/provider.py +++ b/api/core/tools/builtin_tool/provider.py @@ -4,11 +4,11 @@ from typing import Any from core.entities.provider_entities import ProviderConfig from core.helper.module_import_helper import load_single_subclass_from_source +from core.plugin.entities.plugin_daemon import CredentialType from core.tools.__base.tool_provider import ToolProviderController from core.tools.__base.tool_runtime import ToolRuntime from core.tools.builtin_tool.tool import BuiltinTool from core.tools.entities.tool_entities import ( - CredentialType, OAuthSchema, ToolEntity, ToolProviderEntity, diff --git a/api/core/tools/entities/api_entities.py b/api/core/tools/entities/api_entities.py index de6bf01ae9..807d0245d1 100644 --- a/api/core/tools/entities/api_entities.py +++ b/api/core/tools/entities/api_entities.py @@ -4,10 +4,12 @@ from typing import Any, Literal from pydantic import BaseModel, Field, field_validator +from core.entities.mcp_provider import MCPAuthentication, MCPConfiguration from core.model_runtime.utils.encoders import jsonable_encoder +from core.plugin.entities.plugin_daemon import CredentialType from core.tools.__base.tool import ToolParameter from core.tools.entities.common_entities import I18nObject -from core.tools.entities.tool_entities import CredentialType, ToolProviderType +from core.tools.entities.tool_entities import ToolProviderType class ToolApiEntity(BaseModel): @@ -44,10 +46,14 @@ class ToolProviderApiEntity(BaseModel): server_url: str | None = Field(default="", description="The server url of the tool") updated_at: int = Field(default_factory=lambda: int(datetime.now().timestamp())) server_identifier: str | None = Field(default="", description="The server identifier of the MCP tool") - timeout: float | None = Field(default=30.0, description="The timeout of the MCP tool") - sse_read_timeout: float | None = Field(default=300.0, description="The SSE read timeout of the MCP tool") + masked_headers: dict[str, str] | None = Field(default=None, description="The masked headers of the MCP tool") original_headers: dict[str, str] | None = Field(default=None, description="The original headers of the MCP tool") + authentication: MCPAuthentication | None = Field(default=None, description="The OAuth config of the MCP tool") + is_dynamic_registration: bool = Field(default=True, description="Whether the MCP tool is dynamically registered") + configuration: MCPConfiguration | None = Field( + default=None, description="The timeout and sse_read_timeout of the MCP tool" + ) @field_validator("tools", mode="before") @classmethod @@ -70,8 +76,15 @@ class ToolProviderApiEntity(BaseModel): if self.type == ToolProviderType.MCP: optional_fields.update(self.optional_field("updated_at", self.updated_at)) optional_fields.update(self.optional_field("server_identifier", self.server_identifier)) - optional_fields.update(self.optional_field("timeout", self.timeout)) - optional_fields.update(self.optional_field("sse_read_timeout", self.sse_read_timeout)) + optional_fields.update( + self.optional_field( + "configuration", self.configuration.model_dump() if self.configuration else MCPConfiguration() + ) + ) + optional_fields.update( + self.optional_field("authentication", self.authentication.model_dump() if self.authentication else None) + ) + optional_fields.update(self.optional_field("is_dynamic_registration", self.is_dynamic_registration)) optional_fields.update(self.optional_field("masked_headers", self.masked_headers)) optional_fields.update(self.optional_field("original_headers", self.original_headers)) return { diff --git a/api/core/tools/entities/tool_entities.py b/api/core/tools/entities/tool_entities.py index 15a4f0aafd..353f3a646a 100644 --- a/api/core/tools/entities/tool_entities.py +++ b/api/core/tools/entities/tool_entities.py @@ -129,6 +129,7 @@ class ToolInvokeMessage(BaseModel): class JsonMessage(BaseModel): json_object: dict + suppress_output: bool = Field(default=False, description="Whether to suppress JSON output in result string") class BlobMessage(BaseModel): blob: bytes @@ -267,6 +268,7 @@ class ToolParameter(PluginParameter): SECRET_INPUT = PluginParameterType.SECRET_INPUT FILE = PluginParameterType.FILE FILES = PluginParameterType.FILES + CHECKBOX = PluginParameterType.CHECKBOX APP_SELECTOR = PluginParameterType.APP_SELECTOR MODEL_SELECTOR = PluginParameterType.MODEL_SELECTOR ANY = PluginParameterType.ANY @@ -488,36 +490,3 @@ class ToolSelector(BaseModel): def to_plugin_parameter(self) -> dict[str, Any]: return self.model_dump() - - -class CredentialType(StrEnum): - API_KEY = "api-key" - OAUTH2 = auto() - - def get_name(self): - if self == CredentialType.API_KEY: - return "API KEY" - elif self == CredentialType.OAUTH2: - return "AUTH" - else: - return self.value.replace("-", " ").upper() - - def is_editable(self): - return self == CredentialType.API_KEY - - def is_validate_allowed(self): - return self == CredentialType.API_KEY - - @classmethod - def values(cls): - return [item.value for item in cls] - - @classmethod - def of(cls, credential_type: str) -> "CredentialType": - type_name = credential_type.lower() - if type_name in {"api-key", "api_key"}: - return cls.API_KEY - elif type_name in {"oauth2", "oauth"}: - return cls.OAUTH2 - else: - raise ValueError(f"Invalid credential type: {credential_type}") diff --git a/api/core/tools/mcp_tool/provider.py b/api/core/tools/mcp_tool/provider.py index f0e4dba9c3..557211c8c8 100644 --- a/api/core/tools/mcp_tool/provider.py +++ b/api/core/tools/mcp_tool/provider.py @@ -1,6 +1,6 @@ -import json from typing import Any, Self +from core.entities.mcp_provider import MCPProviderEntity from core.mcp.types import Tool as RemoteMCPTool from core.tools.__base.tool_provider import ToolProviderController from core.tools.__base.tool_runtime import ToolRuntime @@ -52,18 +52,25 @@ class MCPToolProviderController(ToolProviderController): """ from db provider """ - tools = [] - tools_data = json.loads(db_provider.tools) - remote_mcp_tools = [RemoteMCPTool.model_validate(tool) for tool in tools_data] - user = db_provider.load_user() + # Convert to entity first + provider_entity = db_provider.to_entity() + return cls.from_entity(provider_entity) + + @classmethod + def from_entity(cls, entity: MCPProviderEntity) -> Self: + """ + create a MCPToolProviderController from a MCPProviderEntity + """ + remote_mcp_tools = [RemoteMCPTool(**tool) for tool in entity.tools] + tools = [ ToolEntity( identity=ToolIdentity( - author=user.name if user else "Anonymous", + author="Anonymous", # Tool level author is not stored name=remote_mcp_tool.name, label=I18nObject(en_US=remote_mcp_tool.name, zh_Hans=remote_mcp_tool.name), - provider=db_provider.server_identifier, - icon=db_provider.icon, + provider=entity.provider_id, + icon=entity.icon if isinstance(entity.icon, str) else "", ), parameters=ToolTransformService.convert_mcp_schema_to_parameter(remote_mcp_tool.inputSchema), description=ToolDescription( @@ -72,31 +79,32 @@ class MCPToolProviderController(ToolProviderController): ), llm=remote_mcp_tool.description or "", ), + output_schema=remote_mcp_tool.outputSchema or {}, has_runtime_parameters=len(remote_mcp_tool.inputSchema) > 0, ) for remote_mcp_tool in remote_mcp_tools ] - if not db_provider.icon: + if not entity.icon: raise ValueError("Database provider icon is required") return cls( entity=ToolProviderEntityWithPlugin( identity=ToolProviderIdentity( - author=user.name if user else "Anonymous", - name=db_provider.name, - label=I18nObject(en_US=db_provider.name, zh_Hans=db_provider.name), + author="Anonymous", # Provider level author is not stored in entity + name=entity.name, + label=I18nObject(en_US=entity.name, zh_Hans=entity.name), description=I18nObject(en_US="", zh_Hans=""), - icon=db_provider.icon, + icon=entity.icon if isinstance(entity.icon, str) else "", ), plugin_id=None, credentials_schema=[], tools=tools, ), - provider_id=db_provider.server_identifier or "", - tenant_id=db_provider.tenant_id or "", - server_url=db_provider.decrypted_server_url, - headers=db_provider.decrypted_headers or {}, - timeout=db_provider.timeout, - sse_read_timeout=db_provider.sse_read_timeout, + provider_id=entity.provider_id, + tenant_id=entity.tenant_id, + server_url=entity.server_url, + headers=entity.headers, + timeout=entity.timeout, + sse_read_timeout=entity.sse_read_timeout, ) def _validate_credentials(self, user_id: str, credentials: dict[str, Any]): diff --git a/api/core/tools/mcp_tool/tool.py b/api/core/tools/mcp_tool/tool.py index 976d4dc942..fbaf31ad09 100644 --- a/api/core/tools/mcp_tool/tool.py +++ b/api/core/tools/mcp_tool/tool.py @@ -1,14 +1,18 @@ import base64 import json +import logging from collections.abc import Generator from typing import Any -from core.mcp.error import MCPAuthError, MCPConnectionError -from core.mcp.mcp_client import MCPClient -from core.mcp.types import ImageContent, TextContent +from core.mcp.auth_client import MCPClientWithAuthRetry +from core.mcp.error import MCPConnectionError +from core.mcp.types import AudioContent, CallToolResult, ImageContent, TextContent from core.tools.__base.tool import Tool from core.tools.__base.tool_runtime import ToolRuntime from core.tools.entities.tool_entities import ToolEntity, ToolInvokeMessage, ToolProviderType +from core.tools.errors import ToolInvokeError + +logger = logging.getLogger(__name__) class MCPTool(Tool): @@ -44,40 +48,37 @@ class MCPTool(Tool): app_id: str | None = None, message_id: str | None = None, ) -> Generator[ToolInvokeMessage, None, None]: - from core.tools.errors import ToolInvokeError - - try: - with MCPClient( - self.server_url, - self.provider_id, - self.tenant_id, - authed=True, - headers=self.headers, - timeout=self.timeout, - sse_read_timeout=self.sse_read_timeout, - ) as mcp_client: - tool_parameters = self._handle_none_parameter(tool_parameters) - result = mcp_client.invoke_tool(tool_name=self.entity.identity.name, tool_args=tool_parameters) - except MCPAuthError as e: - raise ToolInvokeError("Please auth the tool first") from e - except MCPConnectionError as e: - raise ToolInvokeError(f"Failed to connect to MCP server: {e}") from e - except Exception as e: - raise ToolInvokeError(f"Failed to invoke tool: {e}") from e - + result = self.invoke_remote_mcp_tool(tool_parameters) + # handle dify tool output for content in result.content: if isinstance(content, TextContent): yield from self._process_text_content(content) elif isinstance(content, ImageContent): yield self._process_image_content(content) + elif isinstance(content, AudioContent): + yield self._process_audio_content(content) + else: + logger.warning("Unsupported content type=%s", type(content)) + + # handle MCP structured output + if self.entity.output_schema and result.structuredContent: + for k, v in result.structuredContent.items(): + yield self.create_variable_message(k, v) def _process_text_content(self, content: TextContent) -> Generator[ToolInvokeMessage, None, None]: """Process text content and yield appropriate messages.""" - try: - content_json = json.loads(content.text) - yield from self._process_json_content(content_json) - except json.JSONDecodeError: - yield self.create_text_message(content.text) + # Check if content looks like JSON before attempting to parse + text = content.text.strip() + if text and text[0] in ("{", "[") and text[-1] in ("}", "]"): + try: + content_json = json.loads(text) + yield from self._process_json_content(content_json) + return + except json.JSONDecodeError: + pass + + # If not JSON or parsing failed, treat as plain text + yield self.create_text_message(content.text) def _process_json_content(self, content_json: Any) -> Generator[ToolInvokeMessage, None, None]: """Process JSON content based on its type.""" @@ -104,6 +105,10 @@ class MCPTool(Tool): """Process image content and return a blob message.""" return self.create_blob_message(blob=base64.b64decode(content.data), meta={"mime_type": content.mimeType}) + def _process_audio_content(self, content: AudioContent) -> ToolInvokeMessage: + """Process audio content and return a blob message.""" + return self.create_blob_message(blob=base64.b64decode(content.data), meta={"mime_type": content.mimeType}) + def fork_tool_runtime(self, runtime: ToolRuntime) -> "MCPTool": return MCPTool( entity=self.entity, @@ -126,3 +131,44 @@ class MCPTool(Tool): for key, value in parameter.items() if value is not None and not (isinstance(value, str) and value.strip() == "") } + + def invoke_remote_mcp_tool(self, tool_parameters: dict[str, Any]) -> CallToolResult: + headers = self.headers.copy() if self.headers else {} + tool_parameters = self._handle_none_parameter(tool_parameters) + + from sqlalchemy.orm import Session + + from extensions.ext_database import db + from services.tools.mcp_tools_manage_service import MCPToolManageService + + # Step 1: Load provider entity and credentials in a short-lived session + # This minimizes database connection hold time + with Session(db.engine, expire_on_commit=False) as session: + mcp_service = MCPToolManageService(session=session) + provider_entity = mcp_service.get_provider_entity(self.provider_id, self.tenant_id, by_server_id=True) + + # Decrypt and prepare all credentials before closing session + server_url = provider_entity.decrypt_server_url() + headers = provider_entity.decrypt_headers() + + # Try to get existing token and add to headers + if not headers: + tokens = provider_entity.retrieve_tokens() + if tokens and tokens.access_token: + headers["Authorization"] = f"{tokens.token_type.capitalize()} {tokens.access_token}" + + # Step 2: Session is now closed, perform network operations without holding database connection + # MCPClientWithAuthRetry will create a new session lazily only if auth retry is needed + try: + with MCPClientWithAuthRetry( + server_url=server_url, + headers=headers, + timeout=self.timeout, + sse_read_timeout=self.sse_read_timeout, + provider_entity=provider_entity, + ) as mcp_client: + return mcp_client.invoke_tool(tool_name=self.entity.identity.name, tool_args=tool_parameters) + except MCPConnectionError as e: + raise ToolInvokeError(f"Failed to connect to MCP server: {e}") from e + except Exception as e: + raise ToolInvokeError(f"Failed to invoke tool: {e}") from e diff --git a/api/core/tools/tool_engine.py b/api/core/tools/tool_engine.py index 9fb6062770..13fd579e20 100644 --- a/api/core/tools/tool_engine.py +++ b/api/core/tools/tool_engine.py @@ -228,29 +228,41 @@ class ToolEngine: """ Handle tool response """ - result = "" + parts: list[str] = [] + json_parts: list[str] = [] + for response in tool_response: if response.type == ToolInvokeMessage.MessageType.TEXT: - result += cast(ToolInvokeMessage.TextMessage, response.message).text + parts.append(cast(ToolInvokeMessage.TextMessage, response.message).text) elif response.type == ToolInvokeMessage.MessageType.LINK: - result += ( + parts.append( f"result link: {cast(ToolInvokeMessage.TextMessage, response.message).text}." + " please tell user to check it." ) elif response.type in {ToolInvokeMessage.MessageType.IMAGE_LINK, ToolInvokeMessage.MessageType.IMAGE}: - result += ( + parts.append( "image has been created and sent to user already, " + "you do not need to create it, just tell the user to check it now." ) elif response.type == ToolInvokeMessage.MessageType.JSON: - result += json.dumps( - safe_json_value(cast(ToolInvokeMessage.JsonMessage, response.message).json_object), - ensure_ascii=False, + json_message = cast(ToolInvokeMessage.JsonMessage, response.message) + if json_message.suppress_output: + continue + json_parts.append( + json.dumps( + safe_json_value(cast(ToolInvokeMessage.JsonMessage, response.message).json_object), + ensure_ascii=False, + ) ) else: - result += str(response.message) + parts.append(str(response.message)) - return result + # Add JSON parts, avoiding duplicates from text parts. + if json_parts: + existing_parts = set(parts) + parts.extend(p for p in json_parts if p not in existing_parts) + + return "".join(parts) @staticmethod def _extract_tool_response_binary_and_text( diff --git a/api/core/tools/tool_manager.py b/api/core/tools/tool_manager.py index 5c414915f4..daf3772d30 100644 --- a/api/core/tools/tool_manager.py +++ b/api/core/tools/tool_manager.py @@ -8,23 +8,38 @@ from threading import Lock from typing import TYPE_CHECKING, Any, Literal, Optional, Union, cast import sqlalchemy as sa -from pydantic import TypeAdapter from sqlalchemy import select from sqlalchemy.orm import Session from yarl import URL import contexts +from core.helper.provider_cache import ToolProviderCredentialsCache +from core.plugin.impl.tool import PluginToolManager +from core.tools.__base.tool_provider import ToolProviderController +from core.tools.__base.tool_runtime import ToolRuntime +from core.tools.mcp_tool.provider import MCPToolProviderController +from core.tools.mcp_tool.tool import MCPTool +from core.tools.plugin_tool.provider import PluginToolProviderController +from core.tools.plugin_tool.tool import PluginTool +from core.tools.utils.uuid_utils import is_valid_uuid +from core.tools.workflow_as_tool.provider import WorkflowToolProviderController +from core.workflow.runtime.variable_pool import VariablePool +from extensions.ext_database import db +from models.provider_ids import ToolProviderID +from services.enterprise.plugin_manager_service import PluginCredentialType +from services.tools.mcp_tools_manage_service import MCPToolManageService + +if TYPE_CHECKING: + from core.workflow.nodes.tool.entities import ToolEntity + from configs import dify_config from core.agent.entities import AgentToolEntity from core.app.entities.app_invoke_entities import InvokeFrom from core.helper.module_import_helper import load_single_subclass_from_source from core.helper.position_helper import is_filtered -from core.helper.provider_cache import ToolProviderCredentialsCache from core.model_runtime.utils.encoders import jsonable_encoder -from core.plugin.impl.tool import PluginToolManager +from core.plugin.entities.plugin_daemon import CredentialType from core.tools.__base.tool import Tool -from core.tools.__base.tool_provider import ToolProviderController -from core.tools.__base.tool_runtime import ToolRuntime from core.tools.builtin_tool.provider import BuiltinToolProviderController from core.tools.builtin_tool.providers._positions import BuiltinToolProviderSort from core.tools.builtin_tool.tool import BuiltinTool @@ -34,27 +49,16 @@ from core.tools.entities.api_entities import ToolProviderApiEntity, ToolProvider from core.tools.entities.common_entities import I18nObject from core.tools.entities.tool_entities import ( ApiProviderAuthType, - CredentialType, ToolInvokeFrom, ToolParameter, ToolProviderType, ) from core.tools.errors import ToolProviderNotFoundError -from core.tools.mcp_tool.provider import MCPToolProviderController -from core.tools.mcp_tool.tool import MCPTool -from core.tools.plugin_tool.provider import PluginToolProviderController -from core.tools.plugin_tool.tool import PluginTool from core.tools.tool_label_manager import ToolLabelManager from core.tools.utils.configuration import ToolParameterConfigurationManager from core.tools.utils.encryption import create_provider_encrypter, create_tool_provider_encrypter -from core.tools.utils.uuid_utils import is_valid_uuid -from core.tools.workflow_as_tool.provider import WorkflowToolProviderController from core.tools.workflow_as_tool.tool import WorkflowTool -from extensions.ext_database import db -from models.provider_ids import ToolProviderID -from models.tools import ApiToolProvider, BuiltinToolProvider, MCPToolProvider, WorkflowToolProvider -from services.enterprise.plugin_manager_service import PluginCredentialType -from services.tools.mcp_tools_manage_service import MCPToolManageService +from models.tools import ApiToolProvider, BuiltinToolProvider, WorkflowToolProvider from services.tools.tools_transform_service import ToolTransformService if TYPE_CHECKING: @@ -284,10 +288,8 @@ class ToolManager: credentials=decrypted_credentials, ) # update the credentials - builtin_provider.encrypted_credentials = ( - TypeAdapter(dict[str, Any]) - .dump_json(encrypter.encrypt(dict(refreshed_credentials.credentials))) - .decode("utf-8") + builtin_provider.encrypted_credentials = json.dumps( + encrypter.encrypt(refreshed_credentials.credentials) ) builtin_provider.expires_at = refreshed_credentials.expires_at db.session.commit() @@ -317,7 +319,7 @@ class ToolManager: return api_provider.get_tool(tool_name).fork_tool_runtime( runtime=ToolRuntime( tenant_id=tenant_id, - credentials=encrypter.decrypt(credentials), + credentials=dict(encrypter.decrypt(credentials)), invoke_from=invoke_from, tool_invoke_from=tool_invoke_from, ) @@ -719,7 +721,9 @@ class ToolManager: ) result_providers[f"workflow_provider.{user_provider.name}"] = user_provider if "mcp" in filters: - mcp_providers = MCPToolManageService.retrieve_mcp_tools(tenant_id, for_list=True) + with Session(db.engine) as session: + mcp_service = MCPToolManageService(session=session) + mcp_providers = mcp_service.list_providers(tenant_id=tenant_id, for_list=True) for mcp_provider in mcp_providers: result_providers[f"mcp_provider.{mcp_provider.name}"] = mcp_provider @@ -774,17 +778,12 @@ class ToolManager: :return: the provider controller, the credentials """ - provider: MCPToolProvider | None = ( - db.session.query(MCPToolProvider) - .where( - MCPToolProvider.server_identifier == provider_id, - MCPToolProvider.tenant_id == tenant_id, - ) - .first() - ) - - if provider is None: - raise ToolProviderNotFoundError(f"mcp provider {provider_id} not found") + with Session(db.engine) as session: + mcp_service = MCPToolManageService(session=session) + try: + provider = mcp_service.get_provider(server_identifier=provider_id, tenant_id=tenant_id) + except ValueError: + raise ToolProviderNotFoundError(f"mcp provider {provider_id} not found") controller = MCPToolProviderController.from_db(provider) @@ -831,7 +830,7 @@ class ToolManager: controller=controller, ) - masked_credentials = encrypter.mask_tool_credentials(encrypter.decrypt(credentials)) + masked_credentials = encrypter.mask_plugin_credentials(encrypter.decrypt(credentials)) try: icon = json.loads(provider_obj.icon) @@ -922,16 +921,15 @@ class ToolManager: @classmethod def generate_mcp_tool_icon_url(cls, tenant_id: str, provider_id: str) -> Mapping[str, str] | str: try: - mcp_provider: MCPToolProvider | None = ( - db.session.query(MCPToolProvider) - .where(MCPToolProvider.tenant_id == tenant_id, MCPToolProvider.server_identifier == provider_id) - .first() - ) - - if mcp_provider is None: - raise ToolProviderNotFoundError(f"mcp provider {provider_id} not found") - - return mcp_provider.provider_icon + with Session(db.engine) as session: + mcp_service = MCPToolManageService(session=session) + try: + mcp_provider = mcp_service.get_provider_entity( + provider_id=provider_id, tenant_id=tenant_id, by_server_id=True + ) + return mcp_provider.provider_icon + except ValueError: + raise ToolProviderNotFoundError(f"mcp provider {provider_id} not found") except Exception: return {"background": "#252525", "content": "\ud83d\ude01"} diff --git a/api/core/tools/utils/encryption.py b/api/core/tools/utils/encryption.py index 6ea033b2b6..3b6af302db 100644 --- a/api/core/tools/utils/encryption.py +++ b/api/core/tools/utils/encryption.py @@ -1,137 +1,24 @@ -import contextlib -from copy import deepcopy -from typing import Any, Protocol +# Import generic components from provider_encryption module +from core.helper.provider_encryption import ( + ProviderConfigCache, + ProviderConfigEncrypter, + create_provider_encrypter, +) -from core.entities.provider_entities import BasicProviderConfig -from core.helper import encrypter +# Re-export for backward compatibility +__all__ = [ + "ProviderConfigCache", + "ProviderConfigEncrypter", + "create_provider_encrypter", + "create_tool_provider_encrypter", +] + +# Tool-specific imports from core.helper.provider_cache import SingletonProviderCredentialsCache from core.tools.__base.tool_provider import ToolProviderController -class ProviderConfigCache(Protocol): - """ - Interface for provider configuration cache operations - """ - - def get(self) -> dict | None: - """Get cached provider configuration""" - ... - - def set(self, config: dict[str, Any]): - """Cache provider configuration""" - ... - - def delete(self): - """Delete cached provider configuration""" - ... - - -class ProviderConfigEncrypter: - tenant_id: str - config: list[BasicProviderConfig] - provider_config_cache: ProviderConfigCache - - def __init__( - self, - tenant_id: str, - config: list[BasicProviderConfig], - provider_config_cache: ProviderConfigCache, - ): - self.tenant_id = tenant_id - self.config = config - self.provider_config_cache = provider_config_cache - - def _deep_copy(self, data: dict[str, str]) -> dict[str, str]: - """ - deep copy data - """ - return deepcopy(data) - - def encrypt(self, data: dict[str, str]) -> dict[str, str]: - """ - encrypt tool credentials with tenant id - - return a deep copy of credentials with encrypted values - """ - data = self._deep_copy(data) - - # get fields need to be decrypted - fields = dict[str, BasicProviderConfig]() - for credential in self.config: - fields[credential.name] = credential - - for field_name, field in fields.items(): - if field.type == BasicProviderConfig.Type.SECRET_INPUT: - if field_name in data: - encrypted = encrypter.encrypt_token(self.tenant_id, data[field_name] or "") - data[field_name] = encrypted - - return data - - def mask_tool_credentials(self, data: dict[str, Any]) -> dict[str, Any]: - """ - mask tool credentials - - return a deep copy of credentials with masked values - """ - data = self._deep_copy(data) - - # get fields need to be decrypted - fields = dict[str, BasicProviderConfig]() - for credential in self.config: - fields[credential.name] = credential - - for field_name, field in fields.items(): - if field.type == BasicProviderConfig.Type.SECRET_INPUT: - if field_name in data: - if len(data[field_name]) > 6: - data[field_name] = ( - data[field_name][:2] + "*" * (len(data[field_name]) - 4) + data[field_name][-2:] - ) - else: - data[field_name] = "*" * len(data[field_name]) - - return data - - def decrypt(self, data: dict[str, str]) -> dict[str, Any]: - """ - decrypt tool credentials with tenant id - - return a deep copy of credentials with decrypted values - """ - cached_credentials = self.provider_config_cache.get() - if cached_credentials: - return cached_credentials - - data = self._deep_copy(data) - # get fields need to be decrypted - fields = dict[str, BasicProviderConfig]() - for credential in self.config: - fields[credential.name] = credential - - for field_name, field in fields.items(): - if field.type == BasicProviderConfig.Type.SECRET_INPUT: - if field_name in data: - with contextlib.suppress(Exception): - # if the value is None or empty string, skip decrypt - if not data[field_name]: - continue - - data[field_name] = encrypter.decrypt_token(self.tenant_id, data[field_name]) - - self.provider_config_cache.set(data) - return data - - -def create_provider_encrypter( - tenant_id: str, config: list[BasicProviderConfig], cache: ProviderConfigCache -) -> tuple[ProviderConfigEncrypter, ProviderConfigCache]: - return ProviderConfigEncrypter(tenant_id=tenant_id, config=config, provider_config_cache=cache), cache - - -def create_tool_provider_encrypter( - tenant_id: str, controller: ToolProviderController -) -> tuple[ProviderConfigEncrypter, ProviderConfigCache]: +def create_tool_provider_encrypter(tenant_id: str, controller: ToolProviderController): cache = SingletonProviderCredentialsCache( tenant_id=tenant_id, provider_type=controller.provider_type.value, diff --git a/api/core/tools/workflow_as_tool/provider.py b/api/core/tools/workflow_as_tool/provider.py index d7afbc7389..c8e91413cd 100644 --- a/api/core/tools/workflow_as_tool/provider.py +++ b/api/core/tools/workflow_as_tool/provider.py @@ -31,6 +31,7 @@ VARIABLE_TO_PARAMETER_TYPE_MAPPING = { VariableEntityType.PARAGRAPH: ToolParameter.ToolParameterType.STRING, VariableEntityType.SELECT: ToolParameter.ToolParameterType.SELECT, VariableEntityType.NUMBER: ToolParameter.ToolParameterType.NUMBER, + VariableEntityType.CHECKBOX: ToolParameter.ToolParameterType.BOOLEAN, VariableEntityType.FILE: ToolParameter.ToolParameterType.FILE, VariableEntityType.FILE_LIST: ToolParameter.ToolParameterType.FILES, } diff --git a/api/core/tools/workflow_as_tool/tool.py b/api/core/tools/workflow_as_tool/tool.py index 2cd46647a0..5703c19c88 100644 --- a/api/core/tools/workflow_as_tool/tool.py +++ b/api/core/tools/workflow_as_tool/tool.py @@ -117,7 +117,7 @@ class WorkflowTool(Tool): self._latest_usage = self._derive_usage_from_result(data) yield self.create_text_message(json.dumps(outputs, ensure_ascii=False)) - yield self.create_json_message(outputs) + yield self.create_json_message(outputs, suppress_output=True) @property def latest_usage(self) -> LLMUsage: diff --git a/api/core/trigger/__init__.py b/api/core/trigger/__init__.py new file mode 100644 index 0000000000..1e5b8bb445 --- /dev/null +++ b/api/core/trigger/__init__.py @@ -0,0 +1 @@ +# Core trigger module initialization diff --git a/api/core/trigger/debug/event_bus.py b/api/core/trigger/debug/event_bus.py new file mode 100644 index 0000000000..9d10e1a0e0 --- /dev/null +++ b/api/core/trigger/debug/event_bus.py @@ -0,0 +1,124 @@ +import hashlib +import logging +from typing import TypeVar + +from redis import RedisError + +from core.trigger.debug.events import BaseDebugEvent +from extensions.ext_redis import redis_client + +logger = logging.getLogger(__name__) + +TRIGGER_DEBUG_EVENT_TTL = 300 + +TTriggerDebugEvent = TypeVar("TTriggerDebugEvent", bound="BaseDebugEvent") + + +class TriggerDebugEventBus: + """ + Unified Redis-based trigger debug service with polling support. + + Uses {tenant_id} hash tags for Redis Cluster compatibility. + Supports multiple event types through a generic dispatch/poll interface. + """ + + # LUA_SELECT: Atomic poll or register for event + # KEYS[1] = trigger_debug_inbox:{tenant_id}:{address_id} + # KEYS[2] = trigger_debug_waiting_pool:{tenant_id}:... + # ARGV[1] = address_id + LUA_SELECT = ( + "local v=redis.call('GET',KEYS[1]);" + "if v then redis.call('DEL',KEYS[1]);return v end;" + "redis.call('SADD',KEYS[2],ARGV[1]);" + f"redis.call('EXPIRE',KEYS[2],{TRIGGER_DEBUG_EVENT_TTL});" + "return false" + ) + + # LUA_DISPATCH: Dispatch event to all waiting addresses + # KEYS[1] = trigger_debug_waiting_pool:{tenant_id}:... + # ARGV[1] = tenant_id + # ARGV[2] = event_json + LUA_DISPATCH = ( + "local a=redis.call('SMEMBERS',KEYS[1]);" + "if #a==0 then return 0 end;" + "redis.call('DEL',KEYS[1]);" + "for i=1,#a do " + f"redis.call('SET','trigger_debug_inbox:'..ARGV[1]..':'..a[i],ARGV[2],'EX',{TRIGGER_DEBUG_EVENT_TTL});" + "end;" + "return #a" + ) + + @classmethod + def dispatch( + cls, + tenant_id: str, + event: BaseDebugEvent, + pool_key: str, + ) -> int: + """ + Dispatch event to all waiting addresses in the pool. + + Args: + tenant_id: Tenant ID for hash tag + event: Event object to dispatch + pool_key: Pool key (generate using build_{?}_pool_key(...)) + + Returns: + Number of addresses the event was dispatched to + """ + event_data = event.model_dump_json() + try: + result = redis_client.eval( + cls.LUA_DISPATCH, + 1, + pool_key, + tenant_id, + event_data, + ) + return int(result) + except RedisError: + logger.exception("Failed to dispatch event to pool: %s", pool_key) + return 0 + + @classmethod + def poll( + cls, + event_type: type[TTriggerDebugEvent], + pool_key: str, + tenant_id: str, + user_id: str, + app_id: str, + node_id: str, + ) -> TTriggerDebugEvent | None: + """ + Poll for an event or register to the waiting pool. + + If an event is available in the inbox, return it immediately. + Otherwise, register the address to the waiting pool for future dispatch. + + Args: + event_class: Event class for deserialization and type safety + pool_key: Pool key (generate using build_{?}_pool_key(...)) + tenant_id: Tenant ID + user_id: User ID for address calculation + app_id: App ID for address calculation + node_id: Node ID for address calculation + + Returns: + Event object if available, None otherwise + """ + address_id: str = hashlib.sha256(f"{user_id}|{app_id}|{node_id}".encode()).hexdigest() + address: str = f"trigger_debug_inbox:{tenant_id}:{address_id}" + + try: + event_data = redis_client.eval( + cls.LUA_SELECT, + 2, + address, + pool_key, + address_id, + ) + return event_type.model_validate_json(json_data=event_data) if event_data else None + except RedisError: + logger.exception("Failed to poll event from pool: %s", pool_key) + return None diff --git a/api/core/trigger/debug/event_selectors.py b/api/core/trigger/debug/event_selectors.py new file mode 100644 index 0000000000..bd1ff4ebfe --- /dev/null +++ b/api/core/trigger/debug/event_selectors.py @@ -0,0 +1,243 @@ +"""Trigger debug service supporting plugin and webhook debugging in draft workflows.""" + +import hashlib +import logging +import time +from abc import ABC, abstractmethod +from collections.abc import Mapping +from datetime import datetime +from typing import Any + +from pydantic import BaseModel + +from core.plugin.entities.request import TriggerInvokeEventResponse +from core.trigger.debug.event_bus import TriggerDebugEventBus +from core.trigger.debug.events import ( + PluginTriggerDebugEvent, + ScheduleDebugEvent, + WebhookDebugEvent, + build_plugin_pool_key, + build_webhook_pool_key, +) +from core.workflow.enums import NodeType +from core.workflow.nodes.trigger_plugin.entities import TriggerEventNodeData +from core.workflow.nodes.trigger_schedule.entities import ScheduleConfig +from extensions.ext_redis import redis_client +from libs.datetime_utils import ensure_naive_utc, naive_utc_now +from libs.schedule_utils import calculate_next_run_at +from models.model import App +from models.provider_ids import TriggerProviderID +from models.workflow import Workflow + +logger = logging.getLogger(__name__) + + +class TriggerDebugEvent(BaseModel): + workflow_args: Mapping[str, Any] + node_id: str + + +class TriggerDebugEventPoller(ABC): + app_id: str + user_id: str + tenant_id: str + node_config: Mapping[str, Any] + node_id: str + + def __init__(self, tenant_id: str, user_id: str, app_id: str, node_config: Mapping[str, Any], node_id: str): + self.tenant_id = tenant_id + self.user_id = user_id + self.app_id = app_id + self.node_config = node_config + self.node_id = node_id + + @abstractmethod + def poll(self) -> TriggerDebugEvent | None: + raise NotImplementedError + + +class PluginTriggerDebugEventPoller(TriggerDebugEventPoller): + def poll(self) -> TriggerDebugEvent | None: + from services.trigger.trigger_service import TriggerService + + plugin_trigger_data = TriggerEventNodeData.model_validate(self.node_config.get("data", {})) + provider_id = TriggerProviderID(plugin_trigger_data.provider_id) + pool_key: str = build_plugin_pool_key( + name=plugin_trigger_data.event_name, + provider_id=str(provider_id), + tenant_id=self.tenant_id, + subscription_id=plugin_trigger_data.subscription_id, + ) + plugin_trigger_event: PluginTriggerDebugEvent | None = TriggerDebugEventBus.poll( + event_type=PluginTriggerDebugEvent, + pool_key=pool_key, + tenant_id=self.tenant_id, + user_id=self.user_id, + app_id=self.app_id, + node_id=self.node_id, + ) + if not plugin_trigger_event: + return None + trigger_event_response: TriggerInvokeEventResponse = TriggerService.invoke_trigger_event( + event=plugin_trigger_event, + user_id=plugin_trigger_event.user_id, + tenant_id=self.tenant_id, + node_config=self.node_config, + ) + + if trigger_event_response.cancelled: + return None + + return TriggerDebugEvent( + workflow_args={ + "inputs": trigger_event_response.variables, + "files": [], + }, + node_id=self.node_id, + ) + + +class WebhookTriggerDebugEventPoller(TriggerDebugEventPoller): + def poll(self) -> TriggerDebugEvent | None: + pool_key = build_webhook_pool_key( + tenant_id=self.tenant_id, + app_id=self.app_id, + node_id=self.node_id, + ) + webhook_event: WebhookDebugEvent | None = TriggerDebugEventBus.poll( + event_type=WebhookDebugEvent, + pool_key=pool_key, + tenant_id=self.tenant_id, + user_id=self.user_id, + app_id=self.app_id, + node_id=self.node_id, + ) + if not webhook_event: + return None + + from services.trigger.webhook_service import WebhookService + + payload = webhook_event.payload or {} + workflow_inputs = payload.get("inputs") + if workflow_inputs is None: + webhook_data = payload.get("webhook_data", {}) + workflow_inputs = WebhookService.build_workflow_inputs(webhook_data) + + workflow_args: Mapping[str, Any] = { + "inputs": workflow_inputs or {}, + "files": [], + } + return TriggerDebugEvent(workflow_args=workflow_args, node_id=self.node_id) + + +class ScheduleTriggerDebugEventPoller(TriggerDebugEventPoller): + """ + Poller for schedule trigger debug events. + + This poller will simulate the schedule trigger event by creating a schedule debug runtime cache + and calculating the next run at. + """ + + RUNTIME_CACHE_TTL = 60 * 5 + + class ScheduleDebugRuntime(BaseModel): + cache_key: str + timezone: str + cron_expression: str + next_run_at: datetime + + def schedule_debug_runtime_key(self, cron_hash: str) -> str: + return f"schedule_debug_runtime:{self.tenant_id}:{self.user_id}:{self.app_id}:{self.node_id}:{cron_hash}" + + def get_or_create_schedule_debug_runtime(self): + from services.trigger.schedule_service import ScheduleService + + schedule_config: ScheduleConfig = ScheduleService.to_schedule_config(self.node_config) + cron_hash = hashlib.sha256(schedule_config.cron_expression.encode()).hexdigest() + cache_key = self.schedule_debug_runtime_key(cron_hash) + runtime_cache = redis_client.get(cache_key) + if runtime_cache is None: + schedule_debug_runtime = self.ScheduleDebugRuntime( + cron_expression=schedule_config.cron_expression, + timezone=schedule_config.timezone, + cache_key=cache_key, + next_run_at=ensure_naive_utc( + calculate_next_run_at(schedule_config.cron_expression, schedule_config.timezone) + ), + ) + redis_client.setex( + name=self.schedule_debug_runtime_key(cron_hash), + time=self.RUNTIME_CACHE_TTL, + value=schedule_debug_runtime.model_dump_json(), + ) + return schedule_debug_runtime + else: + redis_client.expire(cache_key, self.RUNTIME_CACHE_TTL) + runtime = self.ScheduleDebugRuntime.model_validate_json(runtime_cache) + runtime.next_run_at = ensure_naive_utc(runtime.next_run_at) + return runtime + + def create_schedule_event(self, schedule_debug_runtime: ScheduleDebugRuntime) -> ScheduleDebugEvent: + redis_client.delete(schedule_debug_runtime.cache_key) + return ScheduleDebugEvent( + timestamp=int(time.time()), + node_id=self.node_id, + inputs={}, + ) + + def poll(self) -> TriggerDebugEvent | None: + schedule_debug_runtime = self.get_or_create_schedule_debug_runtime() + if schedule_debug_runtime.next_run_at > naive_utc_now(): + return None + + schedule_event: ScheduleDebugEvent = self.create_schedule_event(schedule_debug_runtime) + workflow_args: Mapping[str, Any] = { + "inputs": schedule_event.inputs or {}, + "files": [], + } + return TriggerDebugEvent(workflow_args=workflow_args, node_id=self.node_id) + + +def create_event_poller( + draft_workflow: Workflow, tenant_id: str, user_id: str, app_id: str, node_id: str +) -> TriggerDebugEventPoller: + node_config = draft_workflow.get_node_config_by_id(node_id=node_id) + if not node_config: + raise ValueError("Node data not found for node %s", node_id) + node_type = draft_workflow.get_node_type_from_node_config(node_config) + match node_type: + case NodeType.TRIGGER_PLUGIN: + return PluginTriggerDebugEventPoller( + tenant_id=tenant_id, user_id=user_id, app_id=app_id, node_config=node_config, node_id=node_id + ) + case NodeType.TRIGGER_WEBHOOK: + return WebhookTriggerDebugEventPoller( + tenant_id=tenant_id, user_id=user_id, app_id=app_id, node_config=node_config, node_id=node_id + ) + case NodeType.TRIGGER_SCHEDULE: + return ScheduleTriggerDebugEventPoller( + tenant_id=tenant_id, user_id=user_id, app_id=app_id, node_config=node_config, node_id=node_id + ) + case _: + raise ValueError("unable to create event poller for node type %s", node_type) + + +def select_trigger_debug_events( + draft_workflow: Workflow, app_model: App, user_id: str, node_ids: list[str] +) -> TriggerDebugEvent | None: + event: TriggerDebugEvent | None = None + for node_id in node_ids: + node_config = draft_workflow.get_node_config_by_id(node_id=node_id) + if not node_config: + raise ValueError("Node data not found for node %s", node_id) + poller: TriggerDebugEventPoller = create_event_poller( + draft_workflow=draft_workflow, + tenant_id=app_model.tenant_id, + user_id=user_id, + app_id=app_model.id, + node_id=node_id, + ) + event = poller.poll() + if event is not None: + return event + return None diff --git a/api/core/trigger/debug/events.py b/api/core/trigger/debug/events.py new file mode 100644 index 0000000000..9f7bab5e49 --- /dev/null +++ b/api/core/trigger/debug/events.py @@ -0,0 +1,67 @@ +from collections.abc import Mapping +from enum import StrEnum +from typing import Any + +from pydantic import BaseModel, Field + + +class TriggerDebugPoolKey(StrEnum): + """Trigger debug pool key.""" + + SCHEDULE = "schedule_trigger_debug_waiting_pool" + WEBHOOK = "webhook_trigger_debug_waiting_pool" + PLUGIN = "plugin_trigger_debug_waiting_pool" + + +class BaseDebugEvent(BaseModel): + """Base class for all debug events.""" + + timestamp: int + + +class ScheduleDebugEvent(BaseDebugEvent): + """Debug event for schedule triggers.""" + + node_id: str + inputs: Mapping[str, Any] + + +class WebhookDebugEvent(BaseDebugEvent): + """Debug event for webhook triggers.""" + + request_id: str + node_id: str + payload: dict[str, Any] = Field(default_factory=dict) + + +def build_webhook_pool_key(tenant_id: str, app_id: str, node_id: str) -> str: + """Generate pool key for webhook events. + + Args: + tenant_id: Tenant ID + app_id: App ID + node_id: Node ID + """ + return f"{TriggerDebugPoolKey.WEBHOOK}:{tenant_id}:{app_id}:{node_id}" + + +class PluginTriggerDebugEvent(BaseDebugEvent): + """Debug event for plugin triggers.""" + + name: str + user_id: str = Field(description="This is end user id, only for trigger the event. no related with account user id") + request_id: str + subscription_id: str + provider_id: str + + +def build_plugin_pool_key(tenant_id: str, provider_id: str, subscription_id: str, name: str) -> str: + """Generate pool key for plugin trigger events. + + Args: + name: Event name + tenant_id: Tenant ID + provider_id: Provider ID + subscription_id: Subscription ID + """ + return f"{TriggerDebugPoolKey.PLUGIN}:{tenant_id}:{str(provider_id)}:{subscription_id}:{name}" diff --git a/api/core/trigger/entities/api_entities.py b/api/core/trigger/entities/api_entities.py new file mode 100644 index 0000000000..ad7c816144 --- /dev/null +++ b/api/core/trigger/entities/api_entities.py @@ -0,0 +1,76 @@ +from collections.abc import Mapping +from typing import Any + +from pydantic import BaseModel, Field + +from core.entities.provider_entities import ProviderConfig +from core.plugin.entities.plugin_daemon import CredentialType +from core.tools.entities.common_entities import I18nObject +from core.trigger.entities.entities import ( + EventIdentity, + EventParameter, + SubscriptionConstructor, + TriggerCreationMethod, +) + + +class TriggerProviderSubscriptionApiEntity(BaseModel): + id: str = Field(description="The unique id of the subscription") + name: str = Field(description="The name of the subscription") + provider: str = Field(description="The provider id of the subscription") + credential_type: CredentialType = Field(description="The type of the credential") + credentials: dict[str, Any] = Field(description="The credentials of the subscription") + endpoint: str = Field(description="The endpoint of the subscription") + parameters: dict[str, Any] = Field(description="The parameters of the subscription") + properties: dict[str, Any] = Field(description="The properties of the subscription") + workflows_in_use: int = Field(description="The number of workflows using this subscription") + + +class EventApiEntity(BaseModel): + name: str = Field(description="The name of the trigger") + identity: EventIdentity = Field(description="The identity of the trigger") + description: I18nObject = Field(description="The description of the trigger") + parameters: list[EventParameter] = Field(description="The parameters of the trigger") + output_schema: Mapping[str, Any] | None = Field(description="The output schema of the trigger") + + +class TriggerProviderApiEntity(BaseModel): + author: str = Field(..., description="The author of the trigger provider") + name: str = Field(..., description="The name of the trigger provider") + label: I18nObject = Field(..., description="The label of the trigger provider") + description: I18nObject = Field(..., description="The description of the trigger provider") + icon: str | None = Field(default=None, description="The icon of the trigger provider") + icon_dark: str | None = Field(default=None, description="The dark icon of the trigger provider") + tags: list[str] = Field(default_factory=list, description="The tags of the trigger provider") + + plugin_id: str | None = Field(default="", description="The plugin id of the tool") + plugin_unique_identifier: str | None = Field(default="", description="The unique identifier of the tool") + + supported_creation_methods: list[TriggerCreationMethod] = Field( + default_factory=list, + description="Supported creation methods for the trigger provider. like 'OAUTH', 'APIKEY', 'MANUAL'.", + ) + + subscription_constructor: SubscriptionConstructor | None = Field( + default=None, description="The subscription constructor of the trigger provider" + ) + + subscription_schema: list[ProviderConfig] = Field( + default_factory=list, + description="The subscription schema of the trigger provider", + ) + events: list[EventApiEntity] = Field(description="The events of the trigger provider") + + +class SubscriptionBuilderApiEntity(BaseModel): + id: str = Field(description="The id of the subscription builder") + name: str = Field(description="The name of the subscription builder") + provider: str = Field(description="The provider id of the subscription builder") + endpoint: str = Field(description="The endpoint id of the subscription builder") + parameters: Mapping[str, Any] = Field(description="The parameters of the subscription builder") + properties: Mapping[str, Any] = Field(description="The properties of the subscription builder") + credentials: Mapping[str, str] = Field(description="The credentials of the subscription builder") + credential_type: CredentialType = Field(description="The credential type of the subscription builder") + + +__all__ = ["EventApiEntity", "TriggerProviderApiEntity", "TriggerProviderSubscriptionApiEntity"] diff --git a/api/core/trigger/entities/entities.py b/api/core/trigger/entities/entities.py new file mode 100644 index 0000000000..49e24fe8b8 --- /dev/null +++ b/api/core/trigger/entities/entities.py @@ -0,0 +1,288 @@ +from collections.abc import Mapping +from datetime import datetime +from enum import StrEnum +from typing import Any, Union + +from pydantic import BaseModel, ConfigDict, Field, ValidationInfo, field_validator + +from core.entities.provider_entities import ProviderConfig +from core.plugin.entities.parameters import ( + PluginParameterAutoGenerate, + PluginParameterOption, + PluginParameterTemplate, + PluginParameterType, +) +from core.tools.entities.common_entities import I18nObject + + +class EventParameterType(StrEnum): + """The type of the parameter""" + + STRING = PluginParameterType.STRING + NUMBER = PluginParameterType.NUMBER + BOOLEAN = PluginParameterType.BOOLEAN + SELECT = PluginParameterType.SELECT + FILE = PluginParameterType.FILE + FILES = PluginParameterType.FILES + MODEL_SELECTOR = PluginParameterType.MODEL_SELECTOR + APP_SELECTOR = PluginParameterType.APP_SELECTOR + OBJECT = PluginParameterType.OBJECT + ARRAY = PluginParameterType.ARRAY + DYNAMIC_SELECT = PluginParameterType.DYNAMIC_SELECT + CHECKBOX = PluginParameterType.CHECKBOX + + +class EventParameter(BaseModel): + """ + The parameter of the event + """ + + name: str = Field(..., description="The name of the parameter") + label: I18nObject = Field(..., description="The label presented to the user") + type: EventParameterType = Field(..., description="The type of the parameter") + auto_generate: PluginParameterAutoGenerate | None = Field( + default=None, description="The auto generate of the parameter" + ) + template: PluginParameterTemplate | None = Field(default=None, description="The template of the parameter") + scope: str | None = None + required: bool | None = False + multiple: bool | None = Field( + default=False, + description="Whether the parameter is multiple select, only valid for select or dynamic-select type", + ) + default: Union[int, float, str, list[Any], None] = None + min: Union[float, int, None] = None + max: Union[float, int, None] = None + precision: int | None = None + options: list[PluginParameterOption] | None = None + description: I18nObject | None = None + + +class TriggerProviderIdentity(BaseModel): + """ + The identity of the trigger provider + """ + + author: str = Field(..., description="The author of the trigger provider") + name: str = Field(..., description="The name of the trigger provider") + label: I18nObject = Field(..., description="The label of the trigger provider") + description: I18nObject = Field(..., description="The description of the trigger provider") + icon: str | None = Field(default=None, description="The icon of the trigger provider") + icon_dark: str | None = Field(default=None, description="The dark icon of the trigger provider") + tags: list[str] = Field(default_factory=list, description="The tags of the trigger provider") + + +class EventIdentity(BaseModel): + """ + The identity of the event + """ + + author: str = Field(..., description="The author of the event") + name: str = Field(..., description="The name of the event") + label: I18nObject = Field(..., description="The label of the event") + provider: str | None = Field(default=None, description="The provider of the event") + + +class EventEntity(BaseModel): + """ + The configuration of an event + """ + + identity: EventIdentity = Field(..., description="The identity of the event") + parameters: list[EventParameter] = Field( + default_factory=list[EventParameter], description="The parameters of the event" + ) + description: I18nObject = Field(..., description="The description of the event") + output_schema: Mapping[str, Any] | None = Field( + default=None, description="The output schema that this event produces" + ) + + @field_validator("parameters", mode="before") + @classmethod + def set_parameters(cls, v, validation_info: ValidationInfo) -> list[EventParameter]: + return v or [] + + +class OAuthSchema(BaseModel): + client_schema: list[ProviderConfig] = Field(default_factory=list, description="The schema of the OAuth client") + credentials_schema: list[ProviderConfig] = Field( + default_factory=list, description="The schema of the OAuth credentials" + ) + + +class SubscriptionConstructor(BaseModel): + """ + The subscription constructor of the trigger provider + """ + + parameters: list[EventParameter] = Field( + default_factory=list, description="The parameters schema of the subscription constructor" + ) + + credentials_schema: list[ProviderConfig] = Field( + default_factory=list, + description="The credentials schema of the subscription constructor", + ) + + oauth_schema: OAuthSchema | None = Field( + default=None, + description="The OAuth schema of the subscription constructor if OAuth is supported", + ) + + def get_default_parameters(self) -> Mapping[str, Any]: + """Get the default parameters from the parameters schema""" + if not self.parameters: + return {} + return {param.name: param.default for param in self.parameters if param.default} + + +class TriggerProviderEntity(BaseModel): + """ + The configuration of a trigger provider + """ + + identity: TriggerProviderIdentity = Field(..., description="The identity of the trigger provider") + subscription_schema: list[ProviderConfig] = Field( + default_factory=list, + description="The configuration schema stored in the subscription entity", + ) + subscription_constructor: SubscriptionConstructor | None = Field( + default=None, + description="The subscription constructor of the trigger provider", + ) + events: list[EventEntity] = Field(default_factory=list, description="The events of the trigger provider") + + +class Subscription(BaseModel): + """ + Result of a successful trigger subscription operation. + + Contains all information needed to manage the subscription lifecycle. + """ + + expires_at: int = Field( + ..., description="The timestamp when the subscription will expire, this for refresh the subscription" + ) + + endpoint: str = Field(..., description="The webhook endpoint URL allocated by Dify for receiving events") + parameters: Mapping[str, Any] = Field( + default_factory=dict, description="The parameters of the subscription constructor" + ) + properties: Mapping[str, Any] = Field( + ..., description="Subscription data containing all properties and provider-specific information" + ) + + +class UnsubscribeResult(BaseModel): + """ + Result of a trigger unsubscription operation. + + Provides detailed information about the unsubscription attempt, + including success status and error details if failed. + """ + + success: bool = Field(..., description="Whether the unsubscription was successful") + + message: str | None = Field( + None, + description="Human-readable message about the operation result. " + "Success message for successful operations, " + "detailed error information for failures.", + ) + + +class RequestLog(BaseModel): + id: str = Field(..., description="The id of the request log") + endpoint: str = Field(..., description="The endpoint of the request log") + request: dict[str, Any] = Field(..., description="The request of the request log") + response: dict[str, Any] = Field(..., description="The response of the request log") + created_at: datetime = Field(..., description="The created at of the request log") + + +class SubscriptionBuilder(BaseModel): + id: str = Field(..., description="The id of the subscription builder") + name: str | None = Field(default=None, description="The name of the subscription builder") + tenant_id: str = Field(..., description="The tenant id of the subscription builder") + user_id: str = Field(..., description="The user id of the subscription builder") + provider_id: str = Field(..., description="The provider id of the subscription builder") + endpoint_id: str = Field(..., description="The endpoint id of the subscription builder") + parameters: Mapping[str, Any] = Field(..., description="The parameters of the subscription builder") + properties: Mapping[str, Any] = Field(..., description="The properties of the subscription builder") + credentials: Mapping[str, Any] = Field(..., description="The credentials of the subscription builder") + credential_type: str | None = Field(default=None, description="The credential type of the subscription builder") + credential_expires_at: int | None = Field( + default=None, description="The credential expires at of the subscription builder" + ) + expires_at: int = Field(..., description="The expires at of the subscription builder") + + def to_subscription(self) -> Subscription: + return Subscription( + expires_at=self.expires_at, + endpoint=self.endpoint_id, + properties=self.properties, + ) + + +class SubscriptionBuilderUpdater(BaseModel): + name: str | None = Field(default=None, description="The name of the subscription builder") + parameters: Mapping[str, Any] | None = Field(default=None, description="The parameters of the subscription builder") + properties: Mapping[str, Any] | None = Field(default=None, description="The properties of the subscription builder") + credentials: Mapping[str, Any] | None = Field( + default=None, description="The credentials of the subscription builder" + ) + credential_type: str | None = Field(default=None, description="The credential type of the subscription builder") + credential_expires_at: int | None = Field( + default=None, description="The credential expires at of the subscription builder" + ) + expires_at: int | None = Field(default=None, description="The expires at of the subscription builder") + + def update(self, subscription_builder: SubscriptionBuilder) -> None: + if self.name is not None: + subscription_builder.name = self.name + if self.parameters is not None: + subscription_builder.parameters = self.parameters + if self.properties is not None: + subscription_builder.properties = self.properties + if self.credentials is not None: + subscription_builder.credentials = self.credentials + if self.credential_type is not None: + subscription_builder.credential_type = self.credential_type + if self.credential_expires_at is not None: + subscription_builder.credential_expires_at = self.credential_expires_at + if self.expires_at is not None: + subscription_builder.expires_at = self.expires_at + + +class TriggerEventData(BaseModel): + """Event data dispatched to trigger sessions.""" + + subscription_id: str + events: list[str] + request_id: str + timestamp: float + + model_config = ConfigDict(arbitrary_types_allowed=True) + + +class TriggerCreationMethod(StrEnum): + OAUTH = "OAUTH" + APIKEY = "APIKEY" + MANUAL = "MANUAL" + + +# Export all entities +__all__: list[str] = [ + "EventEntity", + "EventIdentity", + "EventParameter", + "EventParameterType", + "OAuthSchema", + "RequestLog", + "Subscription", + "SubscriptionBuilder", + "TriggerCreationMethod", + "TriggerEventData", + "TriggerProviderEntity", + "TriggerProviderIdentity", + "UnsubscribeResult", +] diff --git a/api/core/trigger/errors.py b/api/core/trigger/errors.py new file mode 100644 index 0000000000..4edb1def22 --- /dev/null +++ b/api/core/trigger/errors.py @@ -0,0 +1,19 @@ +from core.plugin.impl.exc import PluginInvokeError + + +class TriggerProviderCredentialValidationError(ValueError): + pass + + +class TriggerPluginInvokeError(PluginInvokeError): + pass + + +class TriggerInvokeError(PluginInvokeError): + pass + + +class EventIgnoreError(TriggerInvokeError): + """ + Trigger event ignore error + """ diff --git a/api/core/trigger/provider.py b/api/core/trigger/provider.py new file mode 100644 index 0000000000..10fa31fdfa --- /dev/null +++ b/api/core/trigger/provider.py @@ -0,0 +1,421 @@ +""" +Trigger Provider Controller for managing trigger providers +""" + +import logging +from collections.abc import Mapping +from typing import Any + +from flask import Request + +from core.entities.provider_entities import BasicProviderConfig +from core.plugin.entities.plugin_daemon import CredentialType +from core.plugin.entities.request import ( + TriggerDispatchResponse, + TriggerInvokeEventResponse, + TriggerSubscriptionResponse, +) +from core.plugin.impl.trigger import PluginTriggerClient +from core.trigger.entities.api_entities import EventApiEntity, TriggerProviderApiEntity +from core.trigger.entities.entities import ( + EventEntity, + EventParameter, + ProviderConfig, + Subscription, + SubscriptionConstructor, + TriggerCreationMethod, + TriggerProviderEntity, + TriggerProviderIdentity, + UnsubscribeResult, +) +from core.trigger.errors import TriggerProviderCredentialValidationError +from models.provider_ids import TriggerProviderID +from services.plugin.plugin_service import PluginService + +logger = logging.getLogger(__name__) + + +class PluginTriggerProviderController: + """ + Controller for plugin trigger providers + """ + + def __init__( + self, + entity: TriggerProviderEntity, + plugin_id: str, + plugin_unique_identifier: str, + provider_id: TriggerProviderID, + tenant_id: str, + ): + """ + Initialize plugin trigger provider controller + + :param entity: Trigger provider entity + :param plugin_id: Plugin ID + :param plugin_unique_identifier: Plugin unique identifier + :param provider_id: Provider ID + :param tenant_id: Tenant ID + """ + self.entity = entity + self.tenant_id = tenant_id + self.plugin_id = plugin_id + self.provider_id = provider_id + self.plugin_unique_identifier = plugin_unique_identifier + + def get_provider_id(self) -> TriggerProviderID: + """ + Get provider ID + """ + return self.provider_id + + def to_api_entity(self) -> TriggerProviderApiEntity: + """ + Convert to API entity + """ + icon = ( + PluginService.get_plugin_icon_url(self.tenant_id, self.entity.identity.icon) + if self.entity.identity.icon + else None + ) + icon_dark = ( + PluginService.get_plugin_icon_url(self.tenant_id, self.entity.identity.icon_dark) + if self.entity.identity.icon_dark + else None + ) + subscription_constructor = self.entity.subscription_constructor + supported_creation_methods = [TriggerCreationMethod.MANUAL] + if subscription_constructor and subscription_constructor.oauth_schema: + supported_creation_methods.append(TriggerCreationMethod.OAUTH) + if subscription_constructor and subscription_constructor.credentials_schema: + supported_creation_methods.append(TriggerCreationMethod.APIKEY) + return TriggerProviderApiEntity( + author=self.entity.identity.author, + name=self.entity.identity.name, + label=self.entity.identity.label, + description=self.entity.identity.description, + icon=icon, + icon_dark=icon_dark, + tags=self.entity.identity.tags, + plugin_id=self.plugin_id, + plugin_unique_identifier=self.plugin_unique_identifier, + subscription_constructor=subscription_constructor, + subscription_schema=self.entity.subscription_schema, + supported_creation_methods=supported_creation_methods, + events=[ + EventApiEntity( + name=event.identity.name, + identity=event.identity, + description=event.description, + parameters=event.parameters, + output_schema=event.output_schema, + ) + for event in self.entity.events + ], + ) + + @property + def identity(self) -> TriggerProviderIdentity: + """Get provider identity""" + return self.entity.identity + + def get_events(self) -> list[EventEntity]: + """ + Get all events for this provider + + :return: List of event entities + """ + return self.entity.events + + def get_event(self, event_name: str) -> EventEntity | None: + """ + Get a specific event by name + + :param event_name: Event name + :return: Event entity or None + """ + for event in self.entity.events: + if event.identity.name == event_name: + return event + return None + + def get_subscription_default_properties(self) -> Mapping[str, Any]: + """ + Get default properties for this provider + + :return: Default properties + """ + return {prop.name: prop.default for prop in self.entity.subscription_schema if prop.default} + + def get_subscription_constructor(self) -> SubscriptionConstructor | None: + """ + Get subscription constructor for this provider + + :return: Subscription constructor + """ + return self.entity.subscription_constructor + + def validate_credentials(self, user_id: str, credentials: Mapping[str, str]) -> None: + """ + Validate credentials against schema + + :param credentials: Credentials to validate + :return: Validation response + """ + # First validate against schema + subscription_constructor: SubscriptionConstructor | None = self.entity.subscription_constructor + if not subscription_constructor: + raise ValueError("Subscription constructor not found") + for config in subscription_constructor.credentials_schema or []: + if config.required and config.name not in credentials: + raise TriggerProviderCredentialValidationError(f"Missing required credential field: {config.name}") + + # Then validate with the plugin daemon + manager = PluginTriggerClient() + provider_id = self.get_provider_id() + response = manager.validate_provider_credentials( + tenant_id=self.tenant_id, + user_id=user_id, + provider=str(provider_id), + credentials=credentials, + ) + if not response: + raise TriggerProviderCredentialValidationError( + "Invalid credentials", + ) + + def get_supported_credential_types(self) -> list[CredentialType]: + """ + Get supported credential types for this provider. + + :return: List of supported credential types + """ + types: list[CredentialType] = [] + subscription_constructor = self.entity.subscription_constructor + if subscription_constructor and subscription_constructor.oauth_schema: + types.append(CredentialType.OAUTH2) + if subscription_constructor and subscription_constructor.credentials_schema: + types.append(CredentialType.API_KEY) + return types + + def get_credentials_schema(self, credential_type: CredentialType | str) -> list[ProviderConfig]: + """ + Get credentials schema by credential type + + :param credential_type: The type of credential (oauth or api_key) + :return: List of provider config schemas + """ + subscription_constructor = self.entity.subscription_constructor + if not subscription_constructor: + return [] + credential_type = CredentialType.of(credential_type) + if credential_type == CredentialType.OAUTH2: + return ( + subscription_constructor.oauth_schema.credentials_schema.copy() + if subscription_constructor and subscription_constructor.oauth_schema + else [] + ) + if credential_type == CredentialType.API_KEY: + return ( + subscription_constructor.credentials_schema.copy() or [] + if subscription_constructor and subscription_constructor.credentials_schema + else [] + ) + if credential_type == CredentialType.UNAUTHORIZED: + return [] + raise ValueError(f"Invalid credential type: {credential_type}") + + def get_credential_schema_config(self, credential_type: CredentialType | str) -> list[BasicProviderConfig]: + """ + Get credential schema config by credential type + """ + return [x.to_basic_provider_config() for x in self.get_credentials_schema(credential_type)] + + def get_oauth_client_schema(self) -> list[ProviderConfig]: + """ + Get OAuth client schema for this provider + + :return: List of OAuth client config schemas + """ + subscription_constructor = self.entity.subscription_constructor + return ( + subscription_constructor.oauth_schema.client_schema.copy() + if subscription_constructor and subscription_constructor.oauth_schema + else [] + ) + + def get_properties_schema(self) -> list[BasicProviderConfig]: + """ + Get properties schema for this provider + + :return: List of properties config schemas + """ + return ( + [x.to_basic_provider_config() for x in self.entity.subscription_schema.copy()] + if self.entity.subscription_schema + else [] + ) + + def get_event_parameters(self, event_name: str) -> Mapping[str, EventParameter]: + """ + Get event parameters for this provider + """ + event = self.get_event(event_name) + if not event: + return {} + return {parameter.name: parameter for parameter in event.parameters} + + def dispatch( + self, + request: Request, + subscription: Subscription, + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> TriggerDispatchResponse: + """ + Dispatch a trigger through plugin runtime + + :param user_id: User ID + :param request: Flask request object + :param subscription: Subscription + :param credentials: Provider credentials + :param credential_type: Credential type + :return: Dispatch response with triggers and raw HTTP response + """ + manager = PluginTriggerClient() + provider_id: TriggerProviderID = self.get_provider_id() + + response: TriggerDispatchResponse = manager.dispatch_event( + tenant_id=self.tenant_id, + provider=str(provider_id), + subscription=subscription.model_dump(), + request=request, + credentials=credentials, + credential_type=credential_type, + ) + return response + + def invoke_trigger_event( + self, + user_id: str, + event_name: str, + parameters: Mapping[str, Any], + credentials: Mapping[str, str], + credential_type: CredentialType, + subscription: Subscription, + request: Request, + payload: Mapping[str, Any], + ) -> TriggerInvokeEventResponse: + """ + Execute a trigger through plugin runtime + + :param user_id: User ID + :param event_name: Event name + :param parameters: Trigger parameters + :param credentials: Provider credentials + :param credential_type: Credential type + :param request: Request + :param payload: Payload + :return: Trigger execution result + """ + manager = PluginTriggerClient() + provider_id: TriggerProviderID = self.get_provider_id() + + return manager.invoke_trigger_event( + tenant_id=self.tenant_id, + user_id=user_id, + provider=str(provider_id), + event_name=event_name, + credentials=credentials, + credential_type=credential_type, + request=request, + parameters=parameters, + subscription=subscription, + payload=payload, + ) + + def subscribe_trigger( + self, + user_id: str, + endpoint: str, + parameters: Mapping[str, Any], + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> Subscription: + """ + Subscribe to a trigger through plugin runtime + + :param user_id: User ID + :param endpoint: Subscription endpoint + :param subscription_params: Subscription parameters + :param credentials: Provider credentials + :param credential_type: Credential type + :return: Subscription result + """ + manager = PluginTriggerClient() + provider_id: TriggerProviderID = self.get_provider_id() + + response: TriggerSubscriptionResponse = manager.subscribe( + tenant_id=self.tenant_id, + user_id=user_id, + provider=str(provider_id), + endpoint=endpoint, + parameters=parameters, + credentials=credentials, + credential_type=credential_type, + ) + + return Subscription.model_validate(response.subscription) + + def unsubscribe_trigger( + self, user_id: str, subscription: Subscription, credentials: Mapping[str, str], credential_type: CredentialType + ) -> UnsubscribeResult: + """ + Unsubscribe from a trigger through plugin runtime + + :param user_id: User ID + :param subscription: Subscription metadata + :param credentials: Provider credentials + :param credential_type: Credential type + :return: Unsubscribe result + """ + manager = PluginTriggerClient() + provider_id: TriggerProviderID = self.get_provider_id() + + response: TriggerSubscriptionResponse = manager.unsubscribe( + tenant_id=self.tenant_id, + user_id=user_id, + provider=str(provider_id), + subscription=subscription, + credentials=credentials, + credential_type=credential_type, + ) + + return UnsubscribeResult.model_validate(response.subscription) + + def refresh_trigger( + self, subscription: Subscription, credentials: Mapping[str, str], credential_type: CredentialType + ) -> Subscription: + """ + Refresh a trigger subscription through plugin runtime + + :param subscription: Subscription metadata + :param credentials: Provider credentials + :return: Refreshed subscription result + """ + manager = PluginTriggerClient() + provider_id: TriggerProviderID = self.get_provider_id() + + response: TriggerSubscriptionResponse = manager.refresh( + tenant_id=self.tenant_id, + user_id="system", # System refresh + provider=str(provider_id), + subscription=subscription, + credentials=credentials, + credential_type=credential_type, + ) + + return Subscription.model_validate(response.subscription) + + +__all__ = ["PluginTriggerProviderController"] diff --git a/api/core/trigger/trigger_manager.py b/api/core/trigger/trigger_manager.py new file mode 100644 index 0000000000..0ef968b265 --- /dev/null +++ b/api/core/trigger/trigger_manager.py @@ -0,0 +1,285 @@ +""" +Trigger Manager for loading and managing trigger providers and triggers +""" + +import logging +from collections.abc import Mapping +from threading import Lock +from typing import Any + +from flask import Request + +import contexts +from configs import dify_config +from core.plugin.entities.plugin_daemon import CredentialType, PluginTriggerProviderEntity +from core.plugin.entities.request import TriggerInvokeEventResponse +from core.plugin.impl.exc import PluginDaemonError, PluginNotFoundError +from core.plugin.impl.trigger import PluginTriggerClient +from core.trigger.entities.entities import ( + EventEntity, + Subscription, + UnsubscribeResult, +) +from core.trigger.errors import EventIgnoreError +from core.trigger.provider import PluginTriggerProviderController +from models.provider_ids import TriggerProviderID + +logger = logging.getLogger(__name__) + + +class TriggerManager: + """ + Manager for trigger providers and triggers + """ + + @classmethod + def get_trigger_plugin_icon(cls, tenant_id: str, provider_id: str) -> str: + """ + Get the icon of a trigger plugin + """ + manager = PluginTriggerClient() + provider: PluginTriggerProviderEntity = manager.fetch_trigger_provider( + tenant_id=tenant_id, provider_id=TriggerProviderID(provider_id) + ) + filename = provider.declaration.identity.icon + base_url = f"{dify_config.CONSOLE_API_URL}/console/api/workspaces/current/plugin/icon" + return f"{base_url}?tenant_id={tenant_id}&filename={filename}" + + @classmethod + def list_plugin_trigger_providers(cls, tenant_id: str) -> list[PluginTriggerProviderController]: + """ + List all plugin trigger providers for a tenant + + :param tenant_id: Tenant ID + :return: List of trigger provider controllers + """ + manager = PluginTriggerClient() + provider_entities = manager.fetch_trigger_providers(tenant_id) + + controllers: list[PluginTriggerProviderController] = [] + for provider in provider_entities: + try: + controller = PluginTriggerProviderController( + entity=provider.declaration, + plugin_id=provider.plugin_id, + plugin_unique_identifier=provider.plugin_unique_identifier, + provider_id=TriggerProviderID(provider.provider), + tenant_id=tenant_id, + ) + controllers.append(controller) + except Exception: + logger.exception("Failed to load trigger provider %s", provider.plugin_id) + continue + + return controllers + + @classmethod + def get_trigger_provider(cls, tenant_id: str, provider_id: TriggerProviderID) -> PluginTriggerProviderController: + """ + Get a specific plugin trigger provider + + :param tenant_id: Tenant ID + :param provider_id: Provider ID + :return: Trigger provider controller or None + """ + # check if context is set + try: + contexts.plugin_trigger_providers.get() + except LookupError: + contexts.plugin_trigger_providers.set({}) + contexts.plugin_trigger_providers_lock.set(Lock()) + + plugin_trigger_providers = contexts.plugin_trigger_providers.get() + provider_id_str = str(provider_id) + if provider_id_str in plugin_trigger_providers: + return plugin_trigger_providers[provider_id_str] + + with contexts.plugin_trigger_providers_lock.get(): + # double check + plugin_trigger_providers = contexts.plugin_trigger_providers.get() + if provider_id_str in plugin_trigger_providers: + return plugin_trigger_providers[provider_id_str] + + try: + manager = PluginTriggerClient() + provider = manager.fetch_trigger_provider(tenant_id, provider_id) + + if not provider: + raise ValueError(f"Trigger provider {provider_id} not found") + + controller = PluginTriggerProviderController( + entity=provider.declaration, + plugin_id=provider.plugin_id, + plugin_unique_identifier=provider.plugin_unique_identifier, + provider_id=provider_id, + tenant_id=tenant_id, + ) + plugin_trigger_providers[provider_id_str] = controller + return controller + except PluginNotFoundError as e: + raise ValueError(f"Trigger provider {provider_id} not found") from e + except PluginDaemonError as e: + raise e + except Exception as e: + logger.exception("Failed to load trigger provider") + raise e + + @classmethod + def list_all_trigger_providers(cls, tenant_id: str) -> list[PluginTriggerProviderController]: + """ + List all trigger providers (plugin) + + :param tenant_id: Tenant ID + :return: List of all trigger provider controllers + """ + return cls.list_plugin_trigger_providers(tenant_id) + + @classmethod + def list_triggers_by_provider(cls, tenant_id: str, provider_id: TriggerProviderID) -> list[EventEntity]: + """ + List all triggers for a specific provider + + :param tenant_id: Tenant ID + :param provider_id: Provider ID + :return: List of trigger entities + """ + provider = cls.get_trigger_provider(tenant_id, provider_id) + return provider.get_events() + + @classmethod + def invoke_trigger_event( + cls, + tenant_id: str, + user_id: str, + provider_id: TriggerProviderID, + event_name: str, + parameters: Mapping[str, Any], + credentials: Mapping[str, str], + credential_type: CredentialType, + subscription: Subscription, + request: Request, + payload: Mapping[str, Any], + ) -> TriggerInvokeEventResponse: + """ + Execute a trigger + + :param tenant_id: Tenant ID + :param user_id: User ID + :param provider_id: Provider ID + :param event_name: Event name + :param parameters: Trigger parameters + :param credentials: Provider credentials + :param credential_type: Credential type + :param subscription: Subscription + :param request: Request + :param payload: Payload + :return: Trigger execution result + """ + provider: PluginTriggerProviderController = cls.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + try: + return provider.invoke_trigger_event( + user_id=user_id, + event_name=event_name, + parameters=parameters, + credentials=credentials, + credential_type=credential_type, + subscription=subscription, + request=request, + payload=payload, + ) + except EventIgnoreError: + return TriggerInvokeEventResponse(variables={}, cancelled=True) + except Exception as e: + raise e + + @classmethod + def subscribe_trigger( + cls, + tenant_id: str, + user_id: str, + provider_id: TriggerProviderID, + endpoint: str, + parameters: Mapping[str, Any], + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> Subscription: + """ + Subscribe to a trigger (e.g., register webhook) + + :param tenant_id: Tenant ID + :param user_id: User ID + :param provider_id: Provider ID + :param endpoint: Subscription endpoint + :param parameters: Subscription parameters + :param credentials: Provider credentials + :param credential_type: Credential type + :return: Subscription result + """ + provider: PluginTriggerProviderController = cls.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + return provider.subscribe_trigger( + user_id=user_id, + endpoint=endpoint, + parameters=parameters, + credentials=credentials, + credential_type=credential_type, + ) + + @classmethod + def unsubscribe_trigger( + cls, + tenant_id: str, + user_id: str, + provider_id: TriggerProviderID, + subscription: Subscription, + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> UnsubscribeResult: + """ + Unsubscribe from a trigger + + :param tenant_id: Tenant ID + :param user_id: User ID + :param provider_id: Provider ID + :param subscription: Subscription metadata from subscribe operation + :param credentials: Provider credentials + :param credential_type: Credential type + :return: Unsubscription result + """ + provider: PluginTriggerProviderController = cls.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + return provider.unsubscribe_trigger( + user_id=user_id, + subscription=subscription, + credentials=credentials, + credential_type=credential_type, + ) + + @classmethod + def refresh_trigger( + cls, + tenant_id: str, + provider_id: TriggerProviderID, + subscription: Subscription, + credentials: Mapping[str, str], + credential_type: CredentialType, + ) -> Subscription: + """ + Refresh a trigger subscription + + :param tenant_id: Tenant ID + :param provider_id: Provider ID + :param subscription: Subscription metadata from subscribe operation + :param credentials: Provider credentials + :param credential_type: Credential type + :return: Refreshed subscription result + """ + + # TODO you should update the subscription using the return value of the refresh_trigger + return cls.get_trigger_provider(tenant_id=tenant_id, provider_id=provider_id).refresh_trigger( + subscription=subscription, credentials=credentials, credential_type=credential_type + ) diff --git a/api/core/trigger/utils/encryption.py b/api/core/trigger/utils/encryption.py new file mode 100644 index 0000000000..026a65aa23 --- /dev/null +++ b/api/core/trigger/utils/encryption.py @@ -0,0 +1,145 @@ +from collections.abc import Mapping +from typing import Union + +from core.entities.provider_entities import BasicProviderConfig, ProviderConfig +from core.helper.provider_cache import ProviderCredentialsCache +from core.helper.provider_encryption import ProviderConfigCache, ProviderConfigEncrypter, create_provider_encrypter +from core.plugin.entities.plugin_daemon import CredentialType +from core.trigger.entities.api_entities import TriggerProviderSubscriptionApiEntity +from core.trigger.provider import PluginTriggerProviderController +from models.trigger import TriggerSubscription + + +class TriggerProviderCredentialsCache(ProviderCredentialsCache): + """Cache for trigger provider credentials""" + + def __init__(self, tenant_id: str, provider_id: str, credential_id: str): + super().__init__(tenant_id=tenant_id, provider_id=provider_id, credential_id=credential_id) + + def _generate_cache_key(self, **kwargs) -> str: + tenant_id = kwargs["tenant_id"] + provider_id = kwargs["provider_id"] + credential_id = kwargs["credential_id"] + return f"trigger_credentials:tenant_id:{tenant_id}:provider_id:{provider_id}:credential_id:{credential_id}" + + +class TriggerProviderOAuthClientParamsCache(ProviderCredentialsCache): + """Cache for trigger provider OAuth client""" + + def __init__(self, tenant_id: str, provider_id: str): + super().__init__(tenant_id=tenant_id, provider_id=provider_id) + + def _generate_cache_key(self, **kwargs) -> str: + tenant_id = kwargs["tenant_id"] + provider_id = kwargs["provider_id"] + return f"trigger_oauth_client:tenant_id:{tenant_id}:provider_id:{provider_id}" + + +class TriggerProviderPropertiesCache(ProviderCredentialsCache): + """Cache for trigger provider properties""" + + def __init__(self, tenant_id: str, provider_id: str, subscription_id: str): + super().__init__(tenant_id=tenant_id, provider_id=provider_id, subscription_id=subscription_id) + + def _generate_cache_key(self, **kwargs) -> str: + tenant_id = kwargs["tenant_id"] + provider_id = kwargs["provider_id"] + subscription_id = kwargs["subscription_id"] + return f"trigger_properties:tenant_id:{tenant_id}:provider_id:{provider_id}:subscription_id:{subscription_id}" + + +def create_trigger_provider_encrypter_for_subscription( + tenant_id: str, + controller: PluginTriggerProviderController, + subscription: Union[TriggerSubscription, TriggerProviderSubscriptionApiEntity], +) -> tuple[ProviderConfigEncrypter, ProviderConfigCache]: + cache = TriggerProviderCredentialsCache( + tenant_id=tenant_id, + provider_id=str(controller.get_provider_id()), + credential_id=subscription.id, + ) + encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=controller.get_credential_schema_config(subscription.credential_type), + cache=cache, + ) + return encrypter, cache + + +def delete_cache_for_subscription(tenant_id: str, provider_id: str, subscription_id: str): + cache = TriggerProviderCredentialsCache( + tenant_id=tenant_id, + provider_id=provider_id, + credential_id=subscription_id, + ) + cache.delete() + + +def create_trigger_provider_encrypter_for_properties( + tenant_id: str, + controller: PluginTriggerProviderController, + subscription: Union[TriggerSubscription, TriggerProviderSubscriptionApiEntity], +) -> tuple[ProviderConfigEncrypter, ProviderConfigCache]: + cache = TriggerProviderPropertiesCache( + tenant_id=tenant_id, + provider_id=str(controller.get_provider_id()), + subscription_id=subscription.id, + ) + encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=controller.get_properties_schema(), + cache=cache, + ) + return encrypter, cache + + +def create_trigger_provider_encrypter( + tenant_id: str, controller: PluginTriggerProviderController, credential_id: str, credential_type: CredentialType +) -> tuple[ProviderConfigEncrypter, ProviderConfigCache]: + cache = TriggerProviderCredentialsCache( + tenant_id=tenant_id, + provider_id=str(controller.get_provider_id()), + credential_id=credential_id, + ) + encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=controller.get_credential_schema_config(credential_type), + cache=cache, + ) + return encrypter, cache + + +def create_trigger_provider_oauth_encrypter( + tenant_id: str, controller: PluginTriggerProviderController +) -> tuple[ProviderConfigEncrypter, ProviderConfigCache]: + cache = TriggerProviderOAuthClientParamsCache( + tenant_id=tenant_id, + provider_id=str(controller.get_provider_id()), + ) + encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=[x.to_basic_provider_config() for x in controller.get_oauth_client_schema()], + cache=cache, + ) + return encrypter, cache + + +def masked_credentials( + schemas: list[ProviderConfig], + credentials: Mapping[str, str], +) -> Mapping[str, str]: + masked_credentials = {} + configs = {x.name: x.to_basic_provider_config() for x in schemas} + for key, value in credentials.items(): + config = configs.get(key) + if not config: + masked_credentials[key] = value + continue + if config.type == BasicProviderConfig.Type.SECRET_INPUT: + if len(value) <= 4: + masked_credentials[key] = "*" * len(value) + else: + masked_credentials[key] = value[:2] + "*" * (len(value) - 4) + value[-2:] + else: + masked_credentials[key] = value + return masked_credentials diff --git a/api/core/trigger/utils/endpoint.py b/api/core/trigger/utils/endpoint.py new file mode 100644 index 0000000000..b282d62d58 --- /dev/null +++ b/api/core/trigger/utils/endpoint.py @@ -0,0 +1,24 @@ +from yarl import URL + +from configs import dify_config + +""" +Basic URL for thirdparty trigger services +""" +base_url = URL(dify_config.TRIGGER_URL) + + +def generate_plugin_trigger_endpoint_url(endpoint_id: str) -> str: + """ + Generate url for plugin trigger endpoint url + """ + + return str(base_url / "triggers" / "plugin" / endpoint_id) + + +def generate_webhook_trigger_endpoint(webhook_id: str, debug: bool = False) -> str: + """ + Generate url for webhook trigger endpoint url + """ + + return str(base_url / "triggers" / ("webhook-debug" if debug else "webhook") / webhook_id) diff --git a/api/core/trigger/utils/locks.py b/api/core/trigger/utils/locks.py new file mode 100644 index 0000000000..46833396e0 --- /dev/null +++ b/api/core/trigger/utils/locks.py @@ -0,0 +1,12 @@ +from collections.abc import Sequence +from itertools import starmap + + +def build_trigger_refresh_lock_key(tenant_id: str, subscription_id: str) -> str: + """Build the Redis lock key for trigger subscription refresh in-flight protection.""" + return f"trigger_provider_refresh_lock:{tenant_id}_{subscription_id}" + + +def build_trigger_refresh_lock_keys(pairs: Sequence[tuple[str, str]]) -> list[str]: + """Build Redis lock keys for a sequence of (tenant_id, subscription_id) pairs.""" + return list(starmap(build_trigger_refresh_lock_key, pairs)) diff --git a/api/core/variables/types.py b/api/core/variables/types.py index a2e12e742b..b537ff7180 100644 --- a/api/core/variables/types.py +++ b/api/core/variables/types.py @@ -202,6 +202,35 @@ class SegmentType(StrEnum): raise ValueError(f"element_type is only supported by array type, got {self}") return _ARRAY_ELEMENT_TYPES_MAPPING.get(self) + @staticmethod + def get_zero_value(t: "SegmentType"): + # Lazy import to avoid circular dependency + from factories import variable_factory + + match t: + case ( + SegmentType.ARRAY_OBJECT + | SegmentType.ARRAY_ANY + | SegmentType.ARRAY_STRING + | SegmentType.ARRAY_NUMBER + | SegmentType.ARRAY_BOOLEAN + ): + return variable_factory.build_segment_with_type(t, []) + case SegmentType.OBJECT: + return variable_factory.build_segment({}) + case SegmentType.STRING: + return variable_factory.build_segment("") + case SegmentType.INTEGER: + return variable_factory.build_segment(0) + case SegmentType.FLOAT: + return variable_factory.build_segment(0.0) + case SegmentType.NUMBER: + return variable_factory.build_segment(0) + case SegmentType.BOOLEAN: + return variable_factory.build_segment(False) + case _: + raise ValueError(f"unsupported variable type: {t}") + _ARRAY_ELEMENT_TYPES_MAPPING: Mapping[SegmentType, SegmentType] = { # ARRAY_ANY does not have corresponding element type. diff --git a/api/core/workflow/entities/__init__.py b/api/core/workflow/entities/__init__.py index 185f0ad620..f4ce9052e0 100644 --- a/api/core/workflow/entities/__init__.py +++ b/api/core/workflow/entities/__init__.py @@ -4,6 +4,7 @@ from .agent import AgentNodeStrategyInit from .graph_init_params import GraphInitParams from .workflow_execution import WorkflowExecution from .workflow_node_execution import WorkflowNodeExecution +from .workflow_pause import WorkflowPauseEntity __all__ = [ "AgentNodeStrategyInit", @@ -12,4 +13,5 @@ __all__ = [ "VariablePool", "WorkflowExecution", "WorkflowNodeExecution", + "WorkflowPauseEntity", ] diff --git a/api/core/workflow/entities/pause_reason.py b/api/core/workflow/entities/pause_reason.py new file mode 100644 index 0000000000..16ad3d639d --- /dev/null +++ b/api/core/workflow/entities/pause_reason.py @@ -0,0 +1,49 @@ +from enum import StrEnum, auto +from typing import Annotated, Any, ClassVar, TypeAlias + +from pydantic import BaseModel, Discriminator, Tag + + +class _PauseReasonType(StrEnum): + HUMAN_INPUT_REQUIRED = auto() + SCHEDULED_PAUSE = auto() + + +class _PauseReasonBase(BaseModel): + TYPE: ClassVar[_PauseReasonType] + + +class HumanInputRequired(_PauseReasonBase): + TYPE = _PauseReasonType.HUMAN_INPUT_REQUIRED + + +class SchedulingPause(_PauseReasonBase): + TYPE = _PauseReasonType.SCHEDULED_PAUSE + + message: str + + +def _get_pause_reason_discriminator(v: Any) -> _PauseReasonType | None: + if isinstance(v, _PauseReasonBase): + return v.TYPE + elif isinstance(v, dict): + reason_type_str = v.get("TYPE") + if reason_type_str is None: + return None + try: + reason_type = _PauseReasonType(reason_type_str) + except ValueError: + return None + return reason_type + else: + # return None if the discriminator value isn't found + return None + + +PauseReason: TypeAlias = Annotated[ + ( + Annotated[HumanInputRequired, Tag(_PauseReasonType.HUMAN_INPUT_REQUIRED)] + | Annotated[SchedulingPause, Tag(_PauseReasonType.SCHEDULED_PAUSE)] + ), + Discriminator(_get_pause_reason_discriminator), +] diff --git a/api/core/workflow/entities/workflow_pause.py b/api/core/workflow/entities/workflow_pause.py new file mode 100644 index 0000000000..2f31c1ff53 --- /dev/null +++ b/api/core/workflow/entities/workflow_pause.py @@ -0,0 +1,61 @@ +""" +Domain entities for workflow pause management. + +This module contains the domain model for workflow pause, which is used +by the core workflow module. These models are independent of the storage mechanism +and don't contain implementation details like tenant_id, app_id, etc. +""" + +from abc import ABC, abstractmethod +from datetime import datetime + + +class WorkflowPauseEntity(ABC): + """ + Abstract base class for workflow pause entities. + + This domain model represents a paused workflow execution state, + without implementation details like tenant_id, app_id, etc. + It provides the interface for managing workflow pause/resume operations + and state persistence through file storage. + + The `WorkflowPauseEntity` is never reused. If a workflow execution pauses multiple times, + it will generate multiple `WorkflowPauseEntity` records. + """ + + @property + @abstractmethod + def id(self) -> str: + """The identifier of current WorkflowPauseEntity""" + pass + + @property + @abstractmethod + def workflow_execution_id(self) -> str: + """The identifier of the workflow execution record the pause associated with. + Correspond to `WorkflowExecution.id`. + """ + + @abstractmethod + def get_state(self) -> bytes: + """ + Retrieve the serialized workflow state from storage. + + This method should load and return the workflow execution state + that was saved when the workflow was paused. The state contains + all necessary information to resume the workflow execution. + + Returns: + bytes: The serialized workflow state containing + execution context, variable values, node states, etc. + + """ + ... + + @property + @abstractmethod + def resumed_at(self) -> datetime | None: + """`resumed_at` return the resumption time of the current pause, or `None` if + the pause is not resumed yet. + """ + pass diff --git a/api/core/workflow/enums.py b/api/core/workflow/enums.py index 83b9281e51..cf12d5ec1f 100644 --- a/api/core/workflow/enums.py +++ b/api/core/workflow/enums.py @@ -22,6 +22,7 @@ class SystemVariableKey(StrEnum): APP_ID = "app_id" WORKFLOW_ID = "workflow_id" WORKFLOW_EXECUTION_ID = "workflow_run_id" + TIMESTAMP = "timestamp" # RAG Pipeline DOCUMENT_ID = "document_id" ORIGINAL_DOCUMENT_ID = "original_document_id" @@ -58,8 +59,31 @@ class NodeType(StrEnum): DOCUMENT_EXTRACTOR = "document-extractor" LIST_OPERATOR = "list-operator" AGENT = "agent" + TRIGGER_WEBHOOK = "trigger-webhook" + TRIGGER_SCHEDULE = "trigger-schedule" + TRIGGER_PLUGIN = "trigger-plugin" HUMAN_INPUT = "human-input" + @property + def is_trigger_node(self) -> bool: + """Check if this node type is a trigger node.""" + return self in [ + NodeType.TRIGGER_WEBHOOK, + NodeType.TRIGGER_SCHEDULE, + NodeType.TRIGGER_PLUGIN, + ] + + @property + def is_start_node(self) -> bool: + """Check if this node type can serve as a workflow entry point.""" + return self in [ + NodeType.START, + NodeType.DATASOURCE, + NodeType.TRIGGER_WEBHOOK, + NodeType.TRIGGER_SCHEDULE, + NodeType.TRIGGER_PLUGIN, + ] + class NodeExecutionType(StrEnum): """Node execution type classification.""" @@ -92,13 +116,111 @@ class WorkflowType(StrEnum): class WorkflowExecutionStatus(StrEnum): + # State diagram for the workflw status: + # (@) means start, (*) means end + # + # ┌------------------>------------------------->------------------->--------------┐ + # | | + # | ┌-----------------------<--------------------┐ | + # ^ | | | + # | | ^ | + # | V | | + # ┌-----------┐ ┌-----------------------┐ ┌-----------┐ V + # | Scheduled |------->| Running |---------------------->| paused | | + # └-----------┘ └-----------------------┘ └-----------┘ | + # | | | | | | | + # | | | | | | | + # ^ | | | V V | + # | | | | | ┌---------┐ | + # (@) | | | └------------------------>| Stopped |<----┘ + # | | | └---------┘ + # | | | | + # | | V V + # | | ┌-----------┐ | + # | | | Succeeded |------------->--------------┤ + # | | └-----------┘ | + # | V V + # | +--------┐ | + # | | Failed |---------------------->----------------┤ + # | └--------┘ | + # V V + # ┌---------------------┐ | + # | Partially Succeeded |---------------------->-----------------┘--------> (*) + # └---------------------┘ + # + # Mermaid diagram: + # + # --- + # title: State diagram for Workflow run state + # --- + # stateDiagram-v2 + # scheduled: Scheduled + # running: Running + # succeeded: Succeeded + # failed: Failed + # partial_succeeded: Partial Succeeded + # paused: Paused + # stopped: Stopped + # + # [*] --> scheduled: + # scheduled --> running: Start Execution + # running --> paused: Human input required + # paused --> running: human input added + # paused --> stopped: User stops execution + # running --> succeeded: Execution finishes without any error + # running --> failed: Execution finishes with errors + # running --> stopped: User stops execution + # running --> partial_succeeded: some execution occurred and handled during execution + # + # scheduled --> stopped: User stops execution + # + # succeeded --> [*] + # failed --> [*] + # partial_succeeded --> [*] + # stopped --> [*] + + # `SCHEDULED` means that the workflow is scheduled to run, but has not + # started running yet. (maybe due to possible worker saturation.) + # + # This enum value is currently unused. + SCHEDULED = "scheduled" + + # `RUNNING` means the workflow is exeuting. RUNNING = "running" + + # `SUCCEEDED` means the execution of workflow succeed without any error. SUCCEEDED = "succeeded" + + # `FAILED` means the execution of workflow failed without some errors. FAILED = "failed" + + # `STOPPED` means the execution of workflow was stopped, either manually + # by the user, or automatically by the Dify application (E.G. the moderation + # mechanism.) STOPPED = "stopped" + + # `PARTIAL_SUCCEEDED` indicates that some errors occurred during the workflow + # execution, but they were successfully handled (e.g., by using an error + # strategy such as "fail branch" or "default value"). PARTIAL_SUCCEEDED = "partial-succeeded" + + # `PAUSED` indicates that the workflow execution is temporarily paused + # (e.g., awaiting human input) and is expected to resume later. PAUSED = "paused" + def is_ended(self) -> bool: + return self in _END_STATE + + +_END_STATE = frozenset( + [ + WorkflowExecutionStatus.SUCCEEDED, + WorkflowExecutionStatus.FAILED, + WorkflowExecutionStatus.PARTIAL_SUCCEEDED, + WorkflowExecutionStatus.STOPPED, + ] +) + class WorkflowNodeExecutionMetadataKey(StrEnum): """ @@ -110,6 +232,7 @@ class WorkflowNodeExecutionMetadataKey(StrEnum): CURRENCY = "currency" TOOL_INFO = "tool_info" AGENT_LOG = "agent_log" + TRIGGER_INFO = "trigger_info" ITERATION_ID = "iteration_id" ITERATION_INDEX = "iteration_index" LOOP_ID = "loop_id" diff --git a/api/core/workflow/graph/graph.py b/api/core/workflow/graph/graph.py index d04724425c..ba5a01fc94 100644 --- a/api/core/workflow/graph/graph.py +++ b/api/core/workflow/graph/graph.py @@ -117,7 +117,7 @@ class Graph: node_type = node_data.get("type") if not isinstance(node_type, str): continue - if node_type in [NodeType.START, NodeType.DATASOURCE]: + if NodeType(node_type).is_start_node: start_node_id = nid break diff --git a/api/core/workflow/graph/validation.py b/api/core/workflow/graph/validation.py index 87aa7db2e4..41b4fdfa60 100644 --- a/api/core/workflow/graph/validation.py +++ b/api/core/workflow/graph/validation.py @@ -114,9 +114,45 @@ class GraphValidator: raise GraphValidationError(issues) +@dataclass(frozen=True, slots=True) +class _TriggerStartExclusivityValidator: + """Ensures trigger nodes do not coexist with UserInput (start) nodes.""" + + conflict_code: str = "TRIGGER_START_NODE_CONFLICT" + + def validate(self, graph: Graph) -> Sequence[GraphValidationIssue]: + start_node_id: str | None = None + trigger_node_ids: list[str] = [] + + for node in graph.nodes.values(): + node_type = getattr(node, "node_type", None) + if not isinstance(node_type, NodeType): + continue + + if node_type == NodeType.START: + start_node_id = node.id + elif node_type.is_trigger_node: + trigger_node_ids.append(node.id) + + if start_node_id and trigger_node_ids: + trigger_list = ", ".join(trigger_node_ids) + return [ + GraphValidationIssue( + code=self.conflict_code, + message=( + f"UserInput (start) node '{start_node_id}' cannot coexist with trigger nodes: {trigger_list}." + ), + node_id=start_node_id, + ) + ] + + return [] + + _DEFAULT_RULES: tuple[GraphValidationRule, ...] = ( _EdgeEndpointValidator(), _RootNodeValidator(), + _TriggerStartExclusivityValidator(), ) diff --git a/api/core/workflow/graph_engine/command_processing/command_handlers.py b/api/core/workflow/graph_engine/command_processing/command_handlers.py index c26c98c496..e9f109c88c 100644 --- a/api/core/workflow/graph_engine/command_processing/command_handlers.py +++ b/api/core/workflow/graph_engine/command_processing/command_handlers.py @@ -3,6 +3,8 @@ from typing import final from typing_extensions import override +from core.workflow.entities.pause_reason import SchedulingPause + from ..domain.graph_execution import GraphExecution from ..entities.commands import AbortCommand, GraphEngineCommand, PauseCommand from .command_processor import CommandHandler @@ -25,4 +27,7 @@ class PauseCommandHandler(CommandHandler): def handle(self, command: GraphEngineCommand, execution: GraphExecution) -> None: assert isinstance(command, PauseCommand) logger.debug("Pausing workflow %s: %s", execution.workflow_id, command.reason) - execution.pause(command.reason) + # Convert string reason to PauseReason if needed + reason = command.reason + pause_reason = SchedulingPause(message=reason) + execution.pause(pause_reason) diff --git a/api/core/workflow/graph_engine/domain/graph_execution.py b/api/core/workflow/graph_engine/domain/graph_execution.py index 6482c927d6..3d587d6691 100644 --- a/api/core/workflow/graph_engine/domain/graph_execution.py +++ b/api/core/workflow/graph_engine/domain/graph_execution.py @@ -8,6 +8,7 @@ from typing import Literal from pydantic import BaseModel, Field +from core.workflow.entities.pause_reason import PauseReason from core.workflow.enums import NodeState from .node_execution import NodeExecution @@ -41,7 +42,7 @@ class GraphExecutionState(BaseModel): completed: bool = Field(default=False) aborted: bool = Field(default=False) paused: bool = Field(default=False) - pause_reason: str | None = Field(default=None) + pause_reason: PauseReason | None = Field(default=None) error: GraphExecutionErrorState | None = Field(default=None) exceptions_count: int = Field(default=0) node_executions: list[NodeExecutionState] = Field(default_factory=list[NodeExecutionState]) @@ -106,7 +107,7 @@ class GraphExecution: completed: bool = False aborted: bool = False paused: bool = False - pause_reason: str | None = None + pause_reason: PauseReason | None = None error: Exception | None = None node_executions: dict[str, NodeExecution] = field(default_factory=dict[str, NodeExecution]) exceptions_count: int = 0 @@ -130,7 +131,7 @@ class GraphExecution: self.aborted = True self.error = RuntimeError(f"Aborted: {reason}") - def pause(self, reason: str | None = None) -> None: + def pause(self, reason: PauseReason) -> None: """Pause the graph execution without marking it complete.""" if self.completed: raise RuntimeError("Cannot pause execution that has completed") diff --git a/api/core/workflow/graph_engine/entities/commands.py b/api/core/workflow/graph_engine/entities/commands.py index 6070ed8812..0d51b2b716 100644 --- a/api/core/workflow/graph_engine/entities/commands.py +++ b/api/core/workflow/graph_engine/entities/commands.py @@ -36,4 +36,4 @@ class PauseCommand(GraphEngineCommand): """Command to pause a running workflow execution.""" command_type: CommandType = Field(default=CommandType.PAUSE, description="Type of command") - reason: str | None = Field(default=None, description="Optional reason for pause") + reason: str = Field(default="unknown reason", description="reason for pause") diff --git a/api/core/workflow/graph_engine/event_management/event_handlers.py b/api/core/workflow/graph_engine/event_management/event_handlers.py index fe99d3ad50..5b0f56e59d 100644 --- a/api/core/workflow/graph_engine/event_management/event_handlers.py +++ b/api/core/workflow/graph_engine/event_management/event_handlers.py @@ -24,6 +24,7 @@ from core.workflow.graph_events import ( NodeRunLoopStartedEvent, NodeRunLoopSucceededEvent, NodeRunPauseRequestedEvent, + NodeRunRetrieverResourceEvent, NodeRunRetryEvent, NodeRunStartedEvent, NodeRunStreamChunkEvent, @@ -112,6 +113,7 @@ class EventHandler: @_dispatch.register(NodeRunLoopSucceededEvent) @_dispatch.register(NodeRunLoopFailedEvent) @_dispatch.register(NodeRunAgentLogEvent) + @_dispatch.register(NodeRunRetrieverResourceEvent) def _(self, event: GraphNodeEventBase) -> None: self._event_collector.collect(event) @@ -208,7 +210,7 @@ class EventHandler: def _(self, event: NodeRunPauseRequestedEvent) -> None: """Handle pause requests emitted by nodes.""" - pause_reason = event.reason or "Awaiting human input" + pause_reason = event.reason self._graph_execution.pause(pause_reason) self._state_manager.finish_execution(event.node_id) if event.node_id in self._graph.nodes: diff --git a/api/core/workflow/graph_engine/graph_engine.py b/api/core/workflow/graph_engine/graph_engine.py index dd2ca3f93b..7071a1f33a 100644 --- a/api/core/workflow/graph_engine/graph_engine.py +++ b/api/core/workflow/graph_engine/graph_engine.py @@ -247,8 +247,11 @@ class GraphEngine: # Handle completion if self._graph_execution.is_paused: + pause_reason = self._graph_execution.pause_reason + assert pause_reason is not None, "pause_reason should not be None when execution is paused." + # Ensure we have a valid PauseReason for the event paused_event = GraphRunPausedEvent( - reason=self._graph_execution.pause_reason, + reason=pause_reason, outputs=self._graph_runtime_state.outputs, ) self._event_manager.notify_layers(paused_event) diff --git a/api/core/workflow/graph_engine/layers/persistence.py b/api/core/workflow/graph_engine/layers/persistence.py index ecd8e12ca5..b70f36ec9e 100644 --- a/api/core/workflow/graph_engine/layers/persistence.py +++ b/api/core/workflow/graph_engine/layers/persistence.py @@ -216,7 +216,6 @@ class WorkflowPersistenceLayer(GraphEngineLayer): def _handle_graph_run_paused(self, event: GraphRunPausedEvent) -> None: execution = self._get_workflow_execution() execution.status = WorkflowExecutionStatus.PAUSED - execution.error_message = event.reason or "Workflow execution paused" execution.outputs = event.outputs self._populate_completion_statistics(execution, update_finished=False) @@ -296,7 +295,7 @@ class WorkflowPersistenceLayer(GraphEngineLayer): domain_execution, event.node_run_result, WorkflowNodeExecutionStatus.PAUSED, - error=event.reason, + error="", update_outputs=False, ) diff --git a/api/core/workflow/graph_engine/worker.py b/api/core/workflow/graph_engine/worker.py index 42c9b936dd..73e59ee298 100644 --- a/api/core/workflow/graph_engine/worker.py +++ b/api/core/workflow/graph_engine/worker.py @@ -16,7 +16,6 @@ from uuid import uuid4 from flask import Flask from typing_extensions import override -from core.workflow.enums import NodeType from core.workflow.graph import Graph from core.workflow.graph_events import GraphNodeEventBase, NodeRunFailedEvent from core.workflow.nodes.base.node import Node @@ -108,8 +107,8 @@ class Worker(threading.Thread): except Exception as e: error_event = NodeRunFailedEvent( id=str(uuid4()), - node_id="unknown", - node_type=NodeType.CODE, + node_id=node.id, + node_type=node.node_type, in_iteration_id=None, error=str(e), start_at=datetime.now(), diff --git a/api/core/workflow/graph_events/graph.py b/api/core/workflow/graph_events/graph.py index 0da962aa1c..9faafc3173 100644 --- a/api/core/workflow/graph_events/graph.py +++ b/api/core/workflow/graph_events/graph.py @@ -1,5 +1,6 @@ from pydantic import Field +from core.workflow.entities.pause_reason import PauseReason from core.workflow.graph_events import BaseGraphEvent @@ -44,7 +45,8 @@ class GraphRunAbortedEvent(BaseGraphEvent): class GraphRunPausedEvent(BaseGraphEvent): """Event emitted when a graph run is paused by user command.""" - reason: str | None = Field(default=None, description="reason for pause") + # reason: str | None = Field(default=None, description="reason for pause") + reason: PauseReason = Field(..., description="reason for pause") outputs: dict[str, object] = Field( default_factory=dict, description="Outputs available to the client while the run is paused.", diff --git a/api/core/workflow/graph_events/node.py b/api/core/workflow/graph_events/node.py index b880df60d1..f225798d41 100644 --- a/api/core/workflow/graph_events/node.py +++ b/api/core/workflow/graph_events/node.py @@ -5,6 +5,7 @@ from pydantic import Field from core.rag.entities.citation_metadata import RetrievalSourceMetadata from core.workflow.entities import AgentNodeStrategyInit +from core.workflow.entities.pause_reason import PauseReason from .base import GraphNodeEventBase @@ -54,4 +55,4 @@ class NodeRunRetryEvent(NodeRunStartedEvent): class NodeRunPauseRequestedEvent(GraphNodeEventBase): - reason: str | None = Field(default=None, description="Optional pause reason") + reason: PauseReason = Field(..., description="pause reason") diff --git a/api/core/workflow/node_events/node.py b/api/core/workflow/node_events/node.py index 4fd5684436..ebf93f2fc2 100644 --- a/api/core/workflow/node_events/node.py +++ b/api/core/workflow/node_events/node.py @@ -5,6 +5,7 @@ from pydantic import Field from core.model_runtime.entities.llm_entities import LLMUsage from core.rag.entities.citation_metadata import RetrievalSourceMetadata +from core.workflow.entities.pause_reason import PauseReason from core.workflow.node_events import NodeRunResult from .base import NodeEventBase @@ -43,4 +44,4 @@ class StreamCompletedEvent(NodeEventBase): class PauseRequestedEvent(NodeEventBase): - reason: str | None = Field(default=None, description="Optional pause reason") + reason: PauseReason = Field(..., description="pause reason") diff --git a/api/core/workflow/nodes/base/node.py b/api/core/workflow/nodes/base/node.py index 7f8c1eddff..eda030699a 100644 --- a/api/core/workflow/nodes/base/node.py +++ b/api/core/workflow/nodes/base/node.py @@ -126,6 +126,12 @@ class Node: start_event.provider_id = f"{plugin_id}/{provider_name}" start_event.provider_type = getattr(self.get_base_node_data(), "provider_type", "") + from core.workflow.nodes.trigger_plugin.trigger_event_node import TriggerEventNode + + if isinstance(self, TriggerEventNode): + start_event.provider_id = getattr(self.get_base_node_data(), "provider_id", "") + start_event.provider_type = getattr(self.get_base_node_data(), "provider_type", "") + from typing import cast from core.workflow.nodes.agent.agent_node import AgentNode diff --git a/api/core/workflow/nodes/document_extractor/node.py b/api/core/workflow/nodes/document_extractor/node.py index cd5f50aaab..12cd7e2bd9 100644 --- a/api/core/workflow/nodes/document_extractor/node.py +++ b/api/core/workflow/nodes/document_extractor/node.py @@ -171,6 +171,7 @@ def _extract_text_by_file_extension(*, file_content: bytes, file_extension: str) ".txt" | ".markdown" | ".md" + | ".mdx" | ".html" | ".htm" | ".xml" diff --git a/api/core/workflow/nodes/http_request/node.py b/api/core/workflow/nodes/http_request/node.py index 55dec3fb08..152d3cc562 100644 --- a/api/core/workflow/nodes/http_request/node.py +++ b/api/core/workflow/nodes/http_request/node.py @@ -104,7 +104,7 @@ class HttpRequestNode(Node): status=WorkflowNodeExecutionStatus.FAILED, outputs={ "status_code": response.status_code, - "body": response.text if not files else "", + "body": response.text if not files.value else "", "headers": response.headers, "files": files, }, diff --git a/api/core/workflow/nodes/human_input/human_input_node.py b/api/core/workflow/nodes/human_input/human_input_node.py index e49f9a8c81..2d6d9760af 100644 --- a/api/core/workflow/nodes/human_input/human_input_node.py +++ b/api/core/workflow/nodes/human_input/human_input_node.py @@ -1,6 +1,7 @@ from collections.abc import Mapping from typing import Any +from core.workflow.entities.pause_reason import HumanInputRequired from core.workflow.enums import ErrorStrategy, NodeExecutionType, NodeType, WorkflowNodeExecutionStatus from core.workflow.node_events import NodeRunResult, PauseRequestedEvent from core.workflow.nodes.base.entities import BaseNodeData, RetryConfig @@ -64,7 +65,7 @@ class HumanInputNode(Node): return self._pause_generator() def _pause_generator(self): - yield PauseRequestedEvent(reason=self._node_data.pause_reason) + yield PauseRequestedEvent(reason=HumanInputRequired()) def _is_completion_ready(self) -> bool: """Determine whether all required inputs are satisfied.""" diff --git a/api/core/workflow/nodes/iteration/entities.py b/api/core/workflow/nodes/iteration/entities.py index ed4ab2c11c..63a41ec755 100644 --- a/api/core/workflow/nodes/iteration/entities.py +++ b/api/core/workflow/nodes/iteration/entities.py @@ -23,6 +23,7 @@ class IterationNodeData(BaseIterationNodeData): is_parallel: bool = False # open the parallel mode or not parallel_nums: int = 10 # the numbers of parallel error_handle_mode: ErrorHandleMode = ErrorHandleMode.TERMINATED # how to handle the error + flatten_output: bool = True # whether to flatten the output array if all elements are lists class IterationStartNodeData(BaseNodeData): diff --git a/api/core/workflow/nodes/iteration/iteration_node.py b/api/core/workflow/nodes/iteration/iteration_node.py index 3a3a2290be..ce83352dcb 100644 --- a/api/core/workflow/nodes/iteration/iteration_node.py +++ b/api/core/workflow/nodes/iteration/iteration_node.py @@ -98,6 +98,7 @@ class IterationNode(LLMUsageTrackingMixin, Node): "is_parallel": False, "parallel_nums": 10, "error_handle_mode": ErrorHandleMode.TERMINATED, + "flatten_output": True, }, } @@ -411,7 +412,14 @@ class IterationNode(LLMUsageTrackingMixin, Node): """ Flatten the outputs list if all elements are lists. This maintains backward compatibility with version 1.8.1 behavior. + + If flatten_output is False, returns outputs as-is (nested structure). + If flatten_output is True (default), flattens the list if all elements are lists. """ + # If flatten_output is disabled, return outputs as-is + if not self._node_data.flatten_output: + return outputs + if not outputs: return outputs diff --git a/api/core/workflow/nodes/llm/node.py b/api/core/workflow/nodes/llm/node.py index 1644f683bf..06c9beaed2 100644 --- a/api/core/workflow/nodes/llm/node.py +++ b/api/core/workflow/nodes/llm/node.py @@ -3,6 +3,7 @@ import io import json import logging import re +import time from collections.abc import Generator, Mapping, Sequence from typing import TYPE_CHECKING, Any, Literal @@ -384,6 +385,8 @@ class LLMNode(Node): output_schema = LLMNode.fetch_structured_output_schema( structured_output=structured_output or {}, ) + request_start_time = time.perf_counter() + invoke_result = invoke_llm_with_structured_output( provider=model_instance.provider, model_schema=model_schema, @@ -396,6 +399,8 @@ class LLMNode(Node): user=user_id, ) else: + request_start_time = time.perf_counter() + invoke_result = model_instance.invoke_llm( prompt_messages=list(prompt_messages), model_parameters=node_data_model.completion_params, @@ -411,6 +416,7 @@ class LLMNode(Node): node_id=node_id, node_type=node_type, reasoning_format=reasoning_format, + request_start_time=request_start_time, ) @staticmethod @@ -422,14 +428,20 @@ class LLMNode(Node): node_id: str, node_type: NodeType, reasoning_format: Literal["separated", "tagged"] = "tagged", + request_start_time: float | None = None, ) -> Generator[NodeEventBase | LLMStructuredOutput, None, None]: # For blocking mode if isinstance(invoke_result, LLMResult): + duration = None + if request_start_time is not None: + duration = time.perf_counter() - request_start_time + invoke_result.usage.latency = round(duration, 3) event = LLMNode.handle_blocking_result( invoke_result=invoke_result, saver=file_saver, file_outputs=file_outputs, reasoning_format=reasoning_format, + request_latency=duration, ) yield event return @@ -441,6 +453,12 @@ class LLMNode(Node): usage = LLMUsage.empty_usage() finish_reason = None full_text_buffer = io.StringIO() + + # Initialize streaming metrics tracking + start_time = request_start_time if request_start_time is not None else time.perf_counter() + first_token_time = None + has_content = False + collected_structured_output = None # Collect structured_output from streaming chunks # Consume the invoke result and handle generator exception try: @@ -457,6 +475,11 @@ class LLMNode(Node): file_saver=file_saver, file_outputs=file_outputs, ): + # Detect first token for TTFT calculation + if text_part and not has_content: + first_token_time = time.perf_counter() + has_content = True + full_text_buffer.write(text_part) yield StreamChunkEvent( selector=[node_id, "text"], @@ -489,6 +512,16 @@ class LLMNode(Node): # Extract clean text and reasoning from tags clean_text, reasoning_content = LLMNode._split_reasoning(full_text, reasoning_format) + # Calculate streaming metrics + end_time = time.perf_counter() + total_duration = end_time - start_time + usage.latency = round(total_duration, 3) + if has_content and first_token_time: + gen_ai_server_time_to_first_token = first_token_time - start_time + llm_streaming_time_to_generate = end_time - first_token_time + usage.time_to_first_token = round(gen_ai_server_time_to_first_token, 3) + usage.time_to_generate = round(llm_streaming_time_to_generate, 3) + yield ModelInvokeCompletedEvent( # Use clean_text for separated mode, full_text for tagged mode text=clean_text if reasoning_format == "separated" else full_text, @@ -1068,6 +1101,7 @@ class LLMNode(Node): saver: LLMFileSaver, file_outputs: list["File"], reasoning_format: Literal["separated", "tagged"] = "tagged", + request_latency: float | None = None, ) -> ModelInvokeCompletedEvent: buffer = io.StringIO() for text_part in LLMNode._save_multimodal_output_and_convert_result_to_markdown( @@ -1088,7 +1122,7 @@ class LLMNode(Node): # Extract clean text and reasoning from tags clean_text, reasoning_content = LLMNode._split_reasoning(full_text, reasoning_format) - return ModelInvokeCompletedEvent( + event = ModelInvokeCompletedEvent( # Use clean_text for separated mode, full_text for tagged mode text=clean_text if reasoning_format == "separated" else full_text, usage=invoke_result.usage, @@ -1098,6 +1132,9 @@ class LLMNode(Node): # Pass structured output if enabled structured_output=getattr(invoke_result, "structured_output", None), ) + if request_latency is not None: + event.usage.latency = round(request_latency, 3) + return event @staticmethod def save_multimodal_image_output( diff --git a/api/core/workflow/nodes/node_mapping.py b/api/core/workflow/nodes/node_mapping.py index 3ee28802f1..b926645f18 100644 --- a/api/core/workflow/nodes/node_mapping.py +++ b/api/core/workflow/nodes/node_mapping.py @@ -22,6 +22,9 @@ from core.workflow.nodes.question_classifier import QuestionClassifierNode from core.workflow.nodes.start import StartNode from core.workflow.nodes.template_transform import TemplateTransformNode from core.workflow.nodes.tool import ToolNode +from core.workflow.nodes.trigger_plugin import TriggerEventNode +from core.workflow.nodes.trigger_schedule import TriggerScheduleNode +from core.workflow.nodes.trigger_webhook import TriggerWebhookNode from core.workflow.nodes.variable_aggregator import VariableAggregatorNode from core.workflow.nodes.variable_assigner.v1 import VariableAssignerNode as VariableAssignerNodeV1 from core.workflow.nodes.variable_assigner.v2 import VariableAssignerNode as VariableAssignerNodeV2 @@ -147,4 +150,16 @@ NODE_TYPE_CLASSES_MAPPING: Mapping[NodeType, Mapping[str, type[Node]]] = { LATEST_VERSION: KnowledgeIndexNode, "1": KnowledgeIndexNode, }, + NodeType.TRIGGER_WEBHOOK: { + LATEST_VERSION: TriggerWebhookNode, + "1": TriggerWebhookNode, + }, + NodeType.TRIGGER_PLUGIN: { + LATEST_VERSION: TriggerEventNode, + "1": TriggerEventNode, + }, + NodeType.TRIGGER_SCHEDULE: { + LATEST_VERSION: TriggerScheduleNode, + "1": TriggerScheduleNode, + }, } diff --git a/api/core/workflow/nodes/question_classifier/question_classifier_node.py b/api/core/workflow/nodes/question_classifier/question_classifier_node.py index 3f37fc481b..948a1cead7 100644 --- a/api/core/workflow/nodes/question_classifier/question_classifier_node.py +++ b/api/core/workflow/nodes/question_classifier/question_classifier_node.py @@ -193,15 +193,19 @@ class QuestionClassifierNode(Node): finish_reason = event.finish_reason break - category_name = node_data.classes[0].name - category_id = node_data.classes[0].id + rendered_classes = [ + c.model_copy(update={"name": variable_pool.convert_template(c.name).text}) for c in node_data.classes + ] + + category_name = rendered_classes[0].name + category_id = rendered_classes[0].id if "" in result_text: result_text = re.sub(r"]*>[\s\S]*?", "", result_text, flags=re.IGNORECASE) result_text_json = parse_and_check_json_markdown(result_text, []) # result_text_json = json.loads(result_text.strip('```JSON\n')) if "category_name" in result_text_json and "category_id" in result_text_json: category_id_result = result_text_json["category_id"] - classes = node_data.classes + classes = rendered_classes classes_map = {class_.id: class_.name for class_ in classes} category_ids = [_class.id for _class in classes] if category_id_result in category_ids: diff --git a/api/core/workflow/nodes/tool/tool_node.py b/api/core/workflow/nodes/tool/tool_node.py index 69ab6f0718..799ad9b92f 100644 --- a/api/core/workflow/nodes/tool/tool_node.py +++ b/api/core/workflow/nodes/tool/tool_node.py @@ -164,10 +164,7 @@ class ToolNode(Node): status=WorkflowNodeExecutionStatus.FAILED, inputs=parameters_for_log, metadata={WorkflowNodeExecutionMetadataKey.TOOL_INFO: tool_info}, - error="An error occurred in the plugin, " - f"please contact the author of {node_data.provider_name} for help, " - f"error type: {e.get_error_type()}, " - f"error details: {e.get_error_message()}", + error=e.to_user_friendly_error(plugin_name=node_data.provider_name), error_type=type(e).__name__, ) ) diff --git a/api/core/workflow/nodes/trigger_plugin/__init__.py b/api/core/workflow/nodes/trigger_plugin/__init__.py new file mode 100644 index 0000000000..0f700fbcf9 --- /dev/null +++ b/api/core/workflow/nodes/trigger_plugin/__init__.py @@ -0,0 +1,3 @@ +from .trigger_event_node import TriggerEventNode + +__all__ = ["TriggerEventNode"] diff --git a/api/core/workflow/nodes/trigger_plugin/entities.py b/api/core/workflow/nodes/trigger_plugin/entities.py new file mode 100644 index 0000000000..6c53acee4f --- /dev/null +++ b/api/core/workflow/nodes/trigger_plugin/entities.py @@ -0,0 +1,77 @@ +from collections.abc import Mapping +from typing import Any, Literal, Union + +from pydantic import BaseModel, Field, ValidationInfo, field_validator + +from core.trigger.entities.entities import EventParameter +from core.workflow.nodes.base.entities import BaseNodeData +from core.workflow.nodes.trigger_plugin.exc import TriggerEventParameterError + + +class TriggerEventNodeData(BaseNodeData): + """Plugin trigger node data""" + + class TriggerEventInput(BaseModel): + value: Union[Any, list[str]] + type: Literal["mixed", "variable", "constant"] + + @field_validator("type", mode="before") + @classmethod + def check_type(cls, value, validation_info: ValidationInfo): + type = value + value = validation_info.data.get("value") + + if value is None: + return type + + if type == "mixed" and not isinstance(value, str): + raise ValueError("value must be a string") + + if type == "variable": + if not isinstance(value, list): + raise ValueError("value must be a list") + for val in value: + if not isinstance(val, str): + raise ValueError("value must be a list of strings") + + if type == "constant" and not isinstance(value, str | int | float | bool | dict | list): + raise ValueError("value must be a string, int, float, bool or dict") + return type + + title: str + desc: str | None = None + plugin_id: str = Field(..., description="Plugin ID") + provider_id: str = Field(..., description="Provider ID") + event_name: str = Field(..., description="Event name") + subscription_id: str = Field(..., description="Subscription ID") + plugin_unique_identifier: str = Field(..., description="Plugin unique identifier") + event_parameters: Mapping[str, TriggerEventInput] = Field(default_factory=dict, description="Trigger parameters") + + def resolve_parameters( + self, + *, + parameter_schemas: Mapping[str, EventParameter], + ) -> Mapping[str, Any]: + """ + Generate parameters based on the given plugin trigger parameters. + + Args: + parameter_schemas (Mapping[str, EventParameter]): The mapping of parameter schemas. + + Returns: + Mapping[str, Any]: A dictionary containing the generated parameters. + + """ + result: dict[str, Any] = {} + for parameter_name in self.event_parameters: + parameter: EventParameter | None = parameter_schemas.get(parameter_name) + if not parameter: + result[parameter_name] = None + continue + event_input = self.event_parameters[parameter_name] + + # trigger node only supports constant input + if event_input.type != "constant": + raise TriggerEventParameterError(f"Unknown plugin trigger input type '{event_input.type}'") + result[parameter_name] = event_input.value + return result diff --git a/api/core/workflow/nodes/trigger_plugin/exc.py b/api/core/workflow/nodes/trigger_plugin/exc.py new file mode 100644 index 0000000000..ba884b325c --- /dev/null +++ b/api/core/workflow/nodes/trigger_plugin/exc.py @@ -0,0 +1,10 @@ +class TriggerEventNodeError(ValueError): + """Base exception for plugin trigger node errors.""" + + pass + + +class TriggerEventParameterError(TriggerEventNodeError): + """Exception raised for errors in plugin trigger parameters.""" + + pass diff --git a/api/core/workflow/nodes/trigger_plugin/trigger_event_node.py b/api/core/workflow/nodes/trigger_plugin/trigger_event_node.py new file mode 100644 index 0000000000..c4c2ff87db --- /dev/null +++ b/api/core/workflow/nodes/trigger_plugin/trigger_event_node.py @@ -0,0 +1,89 @@ +from collections.abc import Mapping +from typing import Any + +from core.workflow.constants import SYSTEM_VARIABLE_NODE_ID +from core.workflow.entities.workflow_node_execution import WorkflowNodeExecutionMetadataKey, WorkflowNodeExecutionStatus +from core.workflow.enums import ErrorStrategy, NodeExecutionType, NodeType +from core.workflow.node_events import NodeRunResult +from core.workflow.nodes.base.entities import BaseNodeData, RetryConfig +from core.workflow.nodes.base.node import Node + +from .entities import TriggerEventNodeData + + +class TriggerEventNode(Node): + node_type = NodeType.TRIGGER_PLUGIN + execution_type = NodeExecutionType.ROOT + + _node_data: TriggerEventNodeData + + def init_node_data(self, data: Mapping[str, Any]) -> None: + self._node_data = TriggerEventNodeData.model_validate(data) + + def _get_error_strategy(self) -> ErrorStrategy | None: + return self._node_data.error_strategy + + def _get_retry_config(self) -> RetryConfig: + return self._node_data.retry_config + + def _get_title(self) -> str: + return self._node_data.title + + def _get_description(self) -> str | None: + return self._node_data.desc + + def _get_default_value_dict(self) -> dict[str, Any]: + return self._node_data.default_value_dict + + def get_base_node_data(self) -> BaseNodeData: + return self._node_data + + @classmethod + def get_default_config(cls, filters: Mapping[str, object] | None = None) -> Mapping[str, object]: + return { + "type": "plugin", + "config": { + "title": "", + "plugin_id": "", + "provider_id": "", + "event_name": "", + "subscription_id": "", + "plugin_unique_identifier": "", + "event_parameters": {}, + }, + } + + @classmethod + def version(cls) -> str: + return "1" + + def _run(self) -> NodeRunResult: + """ + Run the plugin trigger node. + + This node invokes the trigger to convert request data into events + and makes them available to downstream nodes. + """ + + # Get trigger data passed when workflow was triggered + metadata = { + WorkflowNodeExecutionMetadataKey.TRIGGER_INFO: { + "provider_id": self._node_data.provider_id, + "event_name": self._node_data.event_name, + "plugin_unique_identifier": self._node_data.plugin_unique_identifier, + }, + } + node_inputs = dict(self.graph_runtime_state.variable_pool.user_inputs) + system_inputs = self.graph_runtime_state.variable_pool.system_variables.to_dict() + + # TODO: System variables should be directly accessible, no need for special handling + # Set system variables as node outputs. + for var in system_inputs: + node_inputs[SYSTEM_VARIABLE_NODE_ID + "." + var] = system_inputs[var] + outputs = dict(node_inputs) + return NodeRunResult( + status=WorkflowNodeExecutionStatus.SUCCEEDED, + inputs=node_inputs, + outputs=outputs, + metadata=metadata, + ) diff --git a/api/core/workflow/nodes/trigger_schedule/__init__.py b/api/core/workflow/nodes/trigger_schedule/__init__.py new file mode 100644 index 0000000000..6773bae502 --- /dev/null +++ b/api/core/workflow/nodes/trigger_schedule/__init__.py @@ -0,0 +1,3 @@ +from core.workflow.nodes.trigger_schedule.trigger_schedule_node import TriggerScheduleNode + +__all__ = ["TriggerScheduleNode"] diff --git a/api/core/workflow/nodes/trigger_schedule/entities.py b/api/core/workflow/nodes/trigger_schedule/entities.py new file mode 100644 index 0000000000..a515d02d55 --- /dev/null +++ b/api/core/workflow/nodes/trigger_schedule/entities.py @@ -0,0 +1,49 @@ +from typing import Literal, Union + +from pydantic import BaseModel, Field + +from core.workflow.nodes.base import BaseNodeData + + +class TriggerScheduleNodeData(BaseNodeData): + """ + Trigger Schedule Node Data + """ + + mode: str = Field(default="visual", description="Schedule mode: visual or cron") + frequency: str | None = Field(default=None, description="Frequency for visual mode: hourly, daily, weekly, monthly") + cron_expression: str | None = Field(default=None, description="Cron expression for cron mode") + visual_config: dict | None = Field(default=None, description="Visual configuration details") + timezone: str = Field(default="UTC", description="Timezone for schedule execution") + + +class ScheduleConfig(BaseModel): + node_id: str + cron_expression: str + timezone: str = "UTC" + + +class SchedulePlanUpdate(BaseModel): + node_id: str | None = None + cron_expression: str | None = None + timezone: str | None = None + + +class VisualConfig(BaseModel): + """Visual configuration for schedule trigger""" + + # For hourly frequency + on_minute: int | None = Field(default=0, ge=0, le=59, description="Minute of the hour (0-59)") + + # For daily, weekly, monthly frequencies + time: str | None = Field(default="12:00 AM", description="Time in 12-hour format (e.g., '2:30 PM')") + + # For weekly frequency + weekdays: list[Literal["sun", "mon", "tue", "wed", "thu", "fri", "sat"]] | None = Field( + default=None, description="List of weekdays to run on" + ) + + # For monthly frequency + monthly_days: list[Union[int, Literal["last"]]] | None = Field( + default=None, description="Days of month to run on (1-31 or 'last')" + ) diff --git a/api/core/workflow/nodes/trigger_schedule/exc.py b/api/core/workflow/nodes/trigger_schedule/exc.py new file mode 100644 index 0000000000..2f99880ff1 --- /dev/null +++ b/api/core/workflow/nodes/trigger_schedule/exc.py @@ -0,0 +1,31 @@ +from core.workflow.nodes.base.exc import BaseNodeError + + +class ScheduleNodeError(BaseNodeError): + """Base schedule node error.""" + + pass + + +class ScheduleNotFoundError(ScheduleNodeError): + """Schedule not found error.""" + + pass + + +class ScheduleConfigError(ScheduleNodeError): + """Schedule configuration error.""" + + pass + + +class ScheduleExecutionError(ScheduleNodeError): + """Schedule execution error.""" + + pass + + +class TenantOwnerNotFoundError(ScheduleExecutionError): + """Tenant owner not found error for schedule execution.""" + + pass diff --git a/api/core/workflow/nodes/trigger_schedule/trigger_schedule_node.py b/api/core/workflow/nodes/trigger_schedule/trigger_schedule_node.py new file mode 100644 index 0000000000..98a841d1be --- /dev/null +++ b/api/core/workflow/nodes/trigger_schedule/trigger_schedule_node.py @@ -0,0 +1,69 @@ +from collections.abc import Mapping +from typing import Any + +from core.workflow.constants import SYSTEM_VARIABLE_NODE_ID +from core.workflow.entities.workflow_node_execution import WorkflowNodeExecutionStatus +from core.workflow.enums import ErrorStrategy, NodeExecutionType, NodeType +from core.workflow.node_events import NodeRunResult +from core.workflow.nodes.base.entities import BaseNodeData, RetryConfig +from core.workflow.nodes.base.node import Node +from core.workflow.nodes.trigger_schedule.entities import TriggerScheduleNodeData + + +class TriggerScheduleNode(Node): + node_type = NodeType.TRIGGER_SCHEDULE + execution_type = NodeExecutionType.ROOT + + _node_data: TriggerScheduleNodeData + + def init_node_data(self, data: Mapping[str, Any]) -> None: + self._node_data = TriggerScheduleNodeData(**data) + + def _get_error_strategy(self) -> ErrorStrategy | None: + return self._node_data.error_strategy + + def _get_retry_config(self) -> RetryConfig: + return self._node_data.retry_config + + def _get_title(self) -> str: + return self._node_data.title + + def _get_description(self) -> str | None: + return self._node_data.desc + + def _get_default_value_dict(self) -> dict[str, Any]: + return self._node_data.default_value_dict + + def get_base_node_data(self) -> BaseNodeData: + return self._node_data + + @classmethod + def version(cls) -> str: + return "1" + + @classmethod + def get_default_config(cls, filters: Mapping[str, object] | None = None) -> Mapping[str, object]: + return { + "type": "trigger-schedule", + "config": { + "mode": "visual", + "frequency": "daily", + "visual_config": {"time": "12:00 AM", "on_minute": 0, "weekdays": ["sun"], "monthly_days": [1]}, + "timezone": "UTC", + }, + } + + def _run(self) -> NodeRunResult: + node_inputs = dict(self.graph_runtime_state.variable_pool.user_inputs) + system_inputs = self.graph_runtime_state.variable_pool.system_variables.to_dict() + + # TODO: System variables should be directly accessible, no need for special handling + # Set system variables as node outputs. + for var in system_inputs: + node_inputs[SYSTEM_VARIABLE_NODE_ID + "." + var] = system_inputs[var] + outputs = dict(node_inputs) + return NodeRunResult( + status=WorkflowNodeExecutionStatus.SUCCEEDED, + inputs=node_inputs, + outputs=outputs, + ) diff --git a/api/core/workflow/nodes/trigger_webhook/__init__.py b/api/core/workflow/nodes/trigger_webhook/__init__.py new file mode 100644 index 0000000000..e41d290f6d --- /dev/null +++ b/api/core/workflow/nodes/trigger_webhook/__init__.py @@ -0,0 +1,3 @@ +from .node import TriggerWebhookNode + +__all__ = ["TriggerWebhookNode"] diff --git a/api/core/workflow/nodes/trigger_webhook/entities.py b/api/core/workflow/nodes/trigger_webhook/entities.py new file mode 100644 index 0000000000..1011e60b43 --- /dev/null +++ b/api/core/workflow/nodes/trigger_webhook/entities.py @@ -0,0 +1,79 @@ +from collections.abc import Sequence +from enum import StrEnum +from typing import Literal + +from pydantic import BaseModel, Field, field_validator + +from core.workflow.nodes.base import BaseNodeData + + +class Method(StrEnum): + GET = "get" + POST = "post" + HEAD = "head" + PATCH = "patch" + PUT = "put" + DELETE = "delete" + + +class ContentType(StrEnum): + JSON = "application/json" + FORM_DATA = "multipart/form-data" + FORM_URLENCODED = "application/x-www-form-urlencoded" + TEXT = "text/plain" + BINARY = "application/octet-stream" + + +class WebhookParameter(BaseModel): + """Parameter definition for headers, query params, or body.""" + + name: str + required: bool = False + + +class WebhookBodyParameter(BaseModel): + """Body parameter with type information.""" + + name: str + type: Literal[ + "string", + "number", + "boolean", + "object", + "array[string]", + "array[number]", + "array[boolean]", + "array[object]", + "file", + ] = "string" + required: bool = False + + +class WebhookData(BaseNodeData): + """ + Webhook Node Data. + """ + + class SyncMode(StrEnum): + SYNC = "async" # only support + + method: Method = Method.GET + content_type: ContentType = Field(default=ContentType.JSON) + headers: Sequence[WebhookParameter] = Field(default_factory=list) + params: Sequence[WebhookParameter] = Field(default_factory=list) # query parameters + body: Sequence[WebhookBodyParameter] = Field(default_factory=list) + + @field_validator("method", mode="before") + @classmethod + def normalize_method(cls, v) -> str: + """Normalize HTTP method to lowercase to support both uppercase and lowercase input.""" + if isinstance(v, str): + return v.lower() + return v + + status_code: int = 200 # Expected status code for response + response_body: str = "" # Template for response body + + # Webhook specific fields (not from client data, set internally) + webhook_id: str | None = None # Set when webhook trigger is created + timeout: int = 30 # Timeout in seconds to wait for webhook response diff --git a/api/core/workflow/nodes/trigger_webhook/exc.py b/api/core/workflow/nodes/trigger_webhook/exc.py new file mode 100644 index 0000000000..dc2239c287 --- /dev/null +++ b/api/core/workflow/nodes/trigger_webhook/exc.py @@ -0,0 +1,25 @@ +from core.workflow.nodes.base.exc import BaseNodeError + + +class WebhookNodeError(BaseNodeError): + """Base webhook node error.""" + + pass + + +class WebhookTimeoutError(WebhookNodeError): + """Webhook timeout error.""" + + pass + + +class WebhookNotFoundError(WebhookNodeError): + """Webhook not found error.""" + + pass + + +class WebhookConfigError(WebhookNodeError): + """Webhook configuration error.""" + + pass diff --git a/api/core/workflow/nodes/trigger_webhook/node.py b/api/core/workflow/nodes/trigger_webhook/node.py new file mode 100644 index 0000000000..15009f90d0 --- /dev/null +++ b/api/core/workflow/nodes/trigger_webhook/node.py @@ -0,0 +1,148 @@ +from collections.abc import Mapping +from typing import Any + +from core.workflow.constants import SYSTEM_VARIABLE_NODE_ID +from core.workflow.entities.workflow_node_execution import WorkflowNodeExecutionStatus +from core.workflow.enums import ErrorStrategy, NodeExecutionType, NodeType +from core.workflow.node_events import NodeRunResult +from core.workflow.nodes.base.entities import BaseNodeData, RetryConfig +from core.workflow.nodes.base.node import Node + +from .entities import ContentType, WebhookData + + +class TriggerWebhookNode(Node): + node_type = NodeType.TRIGGER_WEBHOOK + execution_type = NodeExecutionType.ROOT + + _node_data: WebhookData + + def init_node_data(self, data: Mapping[str, Any]) -> None: + self._node_data = WebhookData.model_validate(data) + + def _get_error_strategy(self) -> ErrorStrategy | None: + return self._node_data.error_strategy + + def _get_retry_config(self) -> RetryConfig: + return self._node_data.retry_config + + def _get_title(self) -> str: + return self._node_data.title + + def _get_description(self) -> str | None: + return self._node_data.desc + + def _get_default_value_dict(self) -> dict[str, Any]: + return self._node_data.default_value_dict + + def get_base_node_data(self) -> BaseNodeData: + return self._node_data + + @classmethod + def get_default_config(cls, filters: Mapping[str, object] | None = None) -> Mapping[str, object]: + return { + "type": "webhook", + "config": { + "method": "get", + "content_type": "application/json", + "headers": [], + "params": [], + "body": [], + "async_mode": True, + "status_code": 200, + "response_body": "", + "timeout": 30, + }, + } + + @classmethod + def version(cls) -> str: + return "1" + + def _run(self) -> NodeRunResult: + """ + Run the webhook node. + + Like the start node, this simply takes the webhook data from the variable pool + and makes it available to downstream nodes. The actual webhook handling + happens in the trigger controller. + """ + # Get webhook data from variable pool (injected by Celery task) + webhook_inputs = dict(self.graph_runtime_state.variable_pool.user_inputs) + + # Extract webhook-specific outputs based on node configuration + outputs = self._extract_configured_outputs(webhook_inputs) + system_inputs = self.graph_runtime_state.variable_pool.system_variables.to_dict() + + # TODO: System variables should be directly accessible, no need for special handling + # Set system variables as node outputs. + for var in system_inputs: + outputs[SYSTEM_VARIABLE_NODE_ID + "." + var] = system_inputs[var] + return NodeRunResult( + status=WorkflowNodeExecutionStatus.SUCCEEDED, + inputs=webhook_inputs, + outputs=outputs, + ) + + def _extract_configured_outputs(self, webhook_inputs: dict[str, Any]) -> dict[str, Any]: + """Extract outputs based on node configuration from webhook inputs.""" + outputs = {} + + # Get the raw webhook data (should be injected by Celery task) + webhook_data = webhook_inputs.get("webhook_data", {}) + + def _to_sanitized(name: str) -> str: + return name.replace("-", "_") + + def _get_normalized(mapping: dict[str, Any], key: str) -> Any: + if not isinstance(mapping, dict): + return None + if key in mapping: + return mapping[key] + alternate = key.replace("-", "_") if "-" in key else key.replace("_", "-") + if alternate in mapping: + return mapping[alternate] + return None + + # Extract configured headers (case-insensitive) + webhook_headers = webhook_data.get("headers", {}) + webhook_headers_lower = {k.lower(): v for k, v in webhook_headers.items()} + + for header in self._node_data.headers: + header_name = header.name + value = _get_normalized(webhook_headers, header_name) + if value is None: + value = _get_normalized(webhook_headers_lower, header_name.lower()) + sanitized_name = _to_sanitized(header_name) + outputs[sanitized_name] = value + + # Extract configured query parameters + for param in self._node_data.params: + param_name = param.name + outputs[param_name] = webhook_data.get("query_params", {}).get(param_name) + + # Extract configured body parameters + for body_param in self._node_data.body: + param_name = body_param.name + param_type = body_param.type + + if self._node_data.content_type == ContentType.TEXT: + # For text/plain, the entire body is a single string parameter + outputs[param_name] = str(webhook_data.get("body", {}).get("raw", "")) + continue + elif self._node_data.content_type == ContentType.BINARY: + outputs[param_name] = webhook_data.get("body", {}).get("raw", b"") + continue + + if param_type == "file": + # Get File object (already processed by webhook controller) + file_obj = webhook_data.get("files", {}).get(param_name) + outputs[param_name] = file_obj + else: + # Get regular body parameter + outputs[param_name] = webhook_data.get("body", {}).get(param_name) + + # Include raw webhook data for debugging/advanced use + outputs["_webhook_raw"] = webhook_data + + return outputs diff --git a/api/core/workflow/nodes/variable_assigner/v1/node.py b/api/core/workflow/nodes/variable_assigner/v1/node.py index 8cd267c4a7..3a0793f092 100644 --- a/api/core/workflow/nodes/variable_assigner/v1/node.py +++ b/api/core/workflow/nodes/variable_assigner/v1/node.py @@ -2,7 +2,6 @@ from collections.abc import Callable, Mapping, Sequence from typing import TYPE_CHECKING, Any, TypeAlias from core.variables import SegmentType, Variable -from core.variables.segments import BooleanSegment from core.workflow.constants import CONVERSATION_VARIABLE_NODE_ID from core.workflow.conversation_variable_updater import ConversationVariableUpdater from core.workflow.entities import GraphInitParams @@ -12,7 +11,6 @@ from core.workflow.nodes.base.entities import BaseNodeData, RetryConfig from core.workflow.nodes.base.node import Node from core.workflow.nodes.variable_assigner.common import helpers as common_helpers from core.workflow.nodes.variable_assigner.common.exc import VariableOperatorNodeError -from factories import variable_factory from ..common.impl import conversation_variable_updater_factory from .node_data import VariableAssignerData, WriteMode @@ -116,7 +114,7 @@ class VariableAssignerNode(Node): updated_variable = original_variable.model_copy(update={"value": updated_value}) case WriteMode.CLEAR: - income_value = get_zero_value(original_variable.value_type) + income_value = SegmentType.get_zero_value(original_variable.value_type) updated_variable = original_variable.model_copy(update={"value": income_value.to_object()}) # Over write the variable. @@ -143,24 +141,3 @@ class VariableAssignerNode(Node): process_data=common_helpers.set_updated_variables({}, updated_variables), outputs={}, ) - - -def get_zero_value(t: SegmentType): - # TODO(QuantumGhost): this should be a method of `SegmentType`. - match t: - case SegmentType.ARRAY_OBJECT | SegmentType.ARRAY_STRING | SegmentType.ARRAY_NUMBER | SegmentType.ARRAY_BOOLEAN: - return variable_factory.build_segment_with_type(t, []) - case SegmentType.OBJECT: - return variable_factory.build_segment({}) - case SegmentType.STRING: - return variable_factory.build_segment("") - case SegmentType.INTEGER: - return variable_factory.build_segment(0) - case SegmentType.FLOAT: - return variable_factory.build_segment(0.0) - case SegmentType.NUMBER: - return variable_factory.build_segment(0) - case SegmentType.BOOLEAN: - return BooleanSegment(value=False) - case _: - raise VariableOperatorNodeError(f"unsupported variable type: {t}") diff --git a/api/core/workflow/nodes/variable_assigner/v2/constants.py b/api/core/workflow/nodes/variable_assigner/v2/constants.py deleted file mode 100644 index 1a4b81c39c..0000000000 --- a/api/core/workflow/nodes/variable_assigner/v2/constants.py +++ /dev/null @@ -1,14 +0,0 @@ -from core.variables import SegmentType - -# Note: This mapping is duplicated with `get_zero_value`. Consider refactoring to avoid redundancy. -EMPTY_VALUE_MAPPING = { - SegmentType.STRING: "", - SegmentType.NUMBER: 0, - SegmentType.BOOLEAN: False, - SegmentType.OBJECT: {}, - SegmentType.ARRAY_ANY: [], - SegmentType.ARRAY_STRING: [], - SegmentType.ARRAY_NUMBER: [], - SegmentType.ARRAY_OBJECT: [], - SegmentType.ARRAY_BOOLEAN: [], -} diff --git a/api/core/workflow/nodes/variable_assigner/v2/node.py b/api/core/workflow/nodes/variable_assigner/v2/node.py index a89055fd66..f15924d78f 100644 --- a/api/core/workflow/nodes/variable_assigner/v2/node.py +++ b/api/core/workflow/nodes/variable_assigner/v2/node.py @@ -16,7 +16,6 @@ from core.workflow.nodes.variable_assigner.common.exc import VariableOperatorNod from core.workflow.nodes.variable_assigner.common.impl import conversation_variable_updater_factory from . import helpers -from .constants import EMPTY_VALUE_MAPPING from .entities import VariableAssignerNodeData, VariableOperationItem from .enums import InputType, Operation from .exc import ( @@ -249,7 +248,7 @@ class VariableAssignerNode(Node): case Operation.OVER_WRITE: return value case Operation.CLEAR: - return EMPTY_VALUE_MAPPING[variable.value_type] + return SegmentType.get_zero_value(variable.value_type).to_object() case Operation.APPEND: return variable.value + [value] case Operation.EXTEND: diff --git a/api/core/workflow/runtime/graph_runtime_state.py b/api/core/workflow/runtime/graph_runtime_state.py index 486718dc62..4c322c6aa6 100644 --- a/api/core/workflow/runtime/graph_runtime_state.py +++ b/api/core/workflow/runtime/graph_runtime_state.py @@ -5,6 +5,7 @@ import json from collections.abc import Mapping, Sequence from collections.abc import Mapping as TypingMapping from copy import deepcopy +from dataclasses import dataclass from typing import Any, Protocol from pydantic.json import pydantic_encoder @@ -106,6 +107,23 @@ class GraphProtocol(Protocol): def get_outgoing_edges(self, node_id: str) -> Sequence[object]: ... +@dataclass(slots=True) +class _GraphRuntimeStateSnapshot: + """Immutable view of a serialized runtime state snapshot.""" + + start_at: float + total_tokens: int + node_run_steps: int + llm_usage: LLMUsage + outputs: dict[str, Any] + variable_pool: VariablePool + has_variable_pool: bool + ready_queue_dump: str | None + graph_execution_dump: str | None + response_coordinator_dump: str | None + paused_nodes: tuple[str, ...] + + class GraphRuntimeState: """Mutable runtime state shared across graph execution components.""" @@ -293,69 +311,28 @@ class GraphRuntimeState: return json.dumps(snapshot, default=pydantic_encoder) - def loads(self, data: str | Mapping[str, Any]) -> None: + @classmethod + def from_snapshot(cls, data: str | Mapping[str, Any]) -> GraphRuntimeState: """Restore runtime state from a serialized snapshot.""" - payload: dict[str, Any] - if isinstance(data, str): - payload = json.loads(data) - else: - payload = dict(data) + snapshot = cls._parse_snapshot_payload(data) - version = payload.get("version") - if version != "1.0": - raise ValueError(f"Unsupported GraphRuntimeState snapshot version: {version}") + state = cls( + variable_pool=snapshot.variable_pool, + start_at=snapshot.start_at, + total_tokens=snapshot.total_tokens, + llm_usage=snapshot.llm_usage, + outputs=snapshot.outputs, + node_run_steps=snapshot.node_run_steps, + ) + state._apply_snapshot(snapshot) + return state - self._start_at = float(payload.get("start_at", 0.0)) - total_tokens = int(payload.get("total_tokens", 0)) - if total_tokens < 0: - raise ValueError("total_tokens must be non-negative") - self._total_tokens = total_tokens + def loads(self, data: str | Mapping[str, Any]) -> None: + """Restore runtime state from a serialized snapshot (legacy API).""" - node_run_steps = int(payload.get("node_run_steps", 0)) - if node_run_steps < 0: - raise ValueError("node_run_steps must be non-negative") - self._node_run_steps = node_run_steps - - llm_usage_payload = payload.get("llm_usage", {}) - self._llm_usage = LLMUsage.model_validate(llm_usage_payload) - - self._outputs = deepcopy(payload.get("outputs", {})) - - variable_pool_payload = payload.get("variable_pool") - if variable_pool_payload is not None: - self._variable_pool = VariablePool.model_validate(variable_pool_payload) - - ready_queue_payload = payload.get("ready_queue") - if ready_queue_payload is not None: - self._ready_queue = self._build_ready_queue() - self._ready_queue.loads(ready_queue_payload) - else: - self._ready_queue = None - - graph_execution_payload = payload.get("graph_execution") - self._graph_execution = None - self._pending_graph_execution_workflow_id = None - if graph_execution_payload is not None: - try: - execution_payload = json.loads(graph_execution_payload) - self._pending_graph_execution_workflow_id = execution_payload.get("workflow_id") - except (json.JSONDecodeError, TypeError, AttributeError): - self._pending_graph_execution_workflow_id = None - self.graph_execution.loads(graph_execution_payload) - - response_payload = payload.get("response_coordinator") - if response_payload is not None: - if self._graph is not None: - self.response_coordinator.loads(response_payload) - else: - self._pending_response_coordinator_dump = response_payload - else: - self._pending_response_coordinator_dump = None - self._response_coordinator = None - - paused_nodes_payload = payload.get("paused_nodes", []) - self._paused_nodes = set(map(str, paused_nodes_payload)) + snapshot = self._parse_snapshot_payload(data) + self._apply_snapshot(snapshot) def register_paused_node(self, node_id: str) -> None: """Record a node that should resume when execution is continued.""" @@ -391,3 +368,106 @@ class GraphRuntimeState: module = importlib.import_module("core.workflow.graph_engine.response_coordinator") coordinator_cls = module.ResponseStreamCoordinator return coordinator_cls(variable_pool=self.variable_pool, graph=graph) + + # ------------------------------------------------------------------ + # Snapshot helpers + # ------------------------------------------------------------------ + @classmethod + def _parse_snapshot_payload(cls, data: str | Mapping[str, Any]) -> _GraphRuntimeStateSnapshot: + payload: dict[str, Any] + if isinstance(data, str): + payload = json.loads(data) + else: + payload = dict(data) + + version = payload.get("version") + if version != "1.0": + raise ValueError(f"Unsupported GraphRuntimeState snapshot version: {version}") + + start_at = float(payload.get("start_at", 0.0)) + + total_tokens = int(payload.get("total_tokens", 0)) + if total_tokens < 0: + raise ValueError("total_tokens must be non-negative") + + node_run_steps = int(payload.get("node_run_steps", 0)) + if node_run_steps < 0: + raise ValueError("node_run_steps must be non-negative") + + llm_usage_payload = payload.get("llm_usage", {}) + llm_usage = LLMUsage.model_validate(llm_usage_payload) + + outputs_payload = deepcopy(payload.get("outputs", {})) + + variable_pool_payload = payload.get("variable_pool") + has_variable_pool = variable_pool_payload is not None + variable_pool = VariablePool.model_validate(variable_pool_payload) if has_variable_pool else VariablePool() + + ready_queue_payload = payload.get("ready_queue") + graph_execution_payload = payload.get("graph_execution") + response_payload = payload.get("response_coordinator") + paused_nodes_payload = payload.get("paused_nodes", []) + + return _GraphRuntimeStateSnapshot( + start_at=start_at, + total_tokens=total_tokens, + node_run_steps=node_run_steps, + llm_usage=llm_usage, + outputs=outputs_payload, + variable_pool=variable_pool, + has_variable_pool=has_variable_pool, + ready_queue_dump=ready_queue_payload, + graph_execution_dump=graph_execution_payload, + response_coordinator_dump=response_payload, + paused_nodes=tuple(map(str, paused_nodes_payload)), + ) + + def _apply_snapshot(self, snapshot: _GraphRuntimeStateSnapshot) -> None: + self._start_at = snapshot.start_at + self._total_tokens = snapshot.total_tokens + self._node_run_steps = snapshot.node_run_steps + self._llm_usage = snapshot.llm_usage.model_copy() + self._outputs = deepcopy(snapshot.outputs) + if snapshot.has_variable_pool or self._variable_pool is None: + self._variable_pool = snapshot.variable_pool + + self._restore_ready_queue(snapshot.ready_queue_dump) + self._restore_graph_execution(snapshot.graph_execution_dump) + self._restore_response_coordinator(snapshot.response_coordinator_dump) + self._paused_nodes = set(snapshot.paused_nodes) + + def _restore_ready_queue(self, payload: str | None) -> None: + if payload is not None: + self._ready_queue = self._build_ready_queue() + self._ready_queue.loads(payload) + else: + self._ready_queue = None + + def _restore_graph_execution(self, payload: str | None) -> None: + self._graph_execution = None + self._pending_graph_execution_workflow_id = None + + if payload is None: + return + + try: + execution_payload = json.loads(payload) + self._pending_graph_execution_workflow_id = execution_payload.get("workflow_id") + except (json.JSONDecodeError, TypeError, AttributeError): + self._pending_graph_execution_workflow_id = None + + self.graph_execution.loads(payload) + + def _restore_response_coordinator(self, payload: str | None) -> None: + if payload is None: + self._pending_response_coordinator_dump = None + self._response_coordinator = None + return + + if self._graph is not None: + self.response_coordinator.loads(payload) + self._pending_response_coordinator_dump = None + return + + self._pending_response_coordinator_dump = payload + self._response_coordinator = None diff --git a/api/core/workflow/runtime/graph_runtime_state_protocol.py b/api/core/workflow/runtime/graph_runtime_state_protocol.py index 40835a936f..5e0878e873 100644 --- a/api/core/workflow/runtime/graph_runtime_state_protocol.py +++ b/api/core/workflow/runtime/graph_runtime_state_protocol.py @@ -3,6 +3,7 @@ from typing import Any, Protocol from core.model_runtime.entities.llm_entities import LLMUsage from core.variables.segments import Segment +from core.workflow.system_variable import SystemVariableReadOnlyView class ReadOnlyVariablePool(Protocol): @@ -30,6 +31,9 @@ class ReadOnlyGraphRuntimeState(Protocol): All methods return defensive copies to ensure immutability. """ + @property + def system_variable(self) -> SystemVariableReadOnlyView: ... + @property def variable_pool(self) -> ReadOnlyVariablePool: """Get read-only access to the variable pool.""" diff --git a/api/core/workflow/runtime/read_only_wrappers.py b/api/core/workflow/runtime/read_only_wrappers.py index 664c365295..8539727fd6 100644 --- a/api/core/workflow/runtime/read_only_wrappers.py +++ b/api/core/workflow/runtime/read_only_wrappers.py @@ -6,6 +6,7 @@ from typing import Any from core.model_runtime.entities.llm_entities import LLMUsage from core.variables.segments import Segment +from core.workflow.system_variable import SystemVariableReadOnlyView from .graph_runtime_state import GraphRuntimeState from .variable_pool import VariablePool @@ -42,6 +43,10 @@ class ReadOnlyGraphRuntimeStateWrapper: self._state = state self._variable_pool_wrapper = ReadOnlyVariablePoolWrapper(state.variable_pool) + @property + def system_variable(self) -> SystemVariableReadOnlyView: + return self._state.variable_pool.system_variables.as_view() + @property def variable_pool(self) -> ReadOnlyVariablePoolWrapper: return self._variable_pool_wrapper diff --git a/api/core/workflow/runtime/variable_pool.py b/api/core/workflow/runtime/variable_pool.py index d41a20dfd7..7fbaec9e70 100644 --- a/api/core/workflow/runtime/variable_pool.py +++ b/api/core/workflow/runtime/variable_pool.py @@ -153,7 +153,11 @@ class VariablePool(BaseModel): return None node_id, name = self._selector_to_keys(selector) - segment: Segment | None = self.variable_dictionary[node_id].get(name) + node_map = self.variable_dictionary.get(node_id) + if node_map is None: + return None + + segment: Segment | None = node_map.get(name) if segment is None: return None diff --git a/api/core/workflow/system_variable.py b/api/core/workflow/system_variable.py index 6716e745cd..ad925912a4 100644 --- a/api/core/workflow/system_variable.py +++ b/api/core/workflow/system_variable.py @@ -1,4 +1,5 @@ from collections.abc import Mapping, Sequence +from types import MappingProxyType from typing import Any from pydantic import AliasChoices, BaseModel, ConfigDict, Field, model_validator @@ -28,6 +29,8 @@ class SystemVariable(BaseModel): app_id: str | None = None workflow_id: str | None = None + timestamp: int | None = None + files: Sequence[File] = Field(default_factory=list) # NOTE: The `workflow_execution_id` field was previously named `workflow_run_id`. @@ -107,4 +110,105 @@ class SystemVariable(BaseModel): d[SystemVariableKey.DATASOURCE_INFO] = self.datasource_info if self.invoke_from is not None: d[SystemVariableKey.INVOKE_FROM] = self.invoke_from + if self.timestamp is not None: + d[SystemVariableKey.TIMESTAMP] = self.timestamp return d + + def as_view(self) -> "SystemVariableReadOnlyView": + return SystemVariableReadOnlyView(self) + + +class SystemVariableReadOnlyView: + """ + A read-only view of a SystemVariable that implements the ReadOnlySystemVariable protocol. + + This class wraps a SystemVariable instance and provides read-only access to all its fields. + It always reads the latest data from the wrapped instance and prevents any write operations. + """ + + def __init__(self, system_variable: SystemVariable) -> None: + """ + Initialize the read-only view with a SystemVariable instance. + + Args: + system_variable: The SystemVariable instance to wrap + """ + self._system_variable = system_variable + + @property + def user_id(self) -> str | None: + return self._system_variable.user_id + + @property + def app_id(self) -> str | None: + return self._system_variable.app_id + + @property + def workflow_id(self) -> str | None: + return self._system_variable.workflow_id + + @property + def workflow_execution_id(self) -> str | None: + return self._system_variable.workflow_execution_id + + @property + def query(self) -> str | None: + return self._system_variable.query + + @property + def conversation_id(self) -> str | None: + return self._system_variable.conversation_id + + @property + def dialogue_count(self) -> int | None: + return self._system_variable.dialogue_count + + @property + def document_id(self) -> str | None: + return self._system_variable.document_id + + @property + def original_document_id(self) -> str | None: + return self._system_variable.original_document_id + + @property + def dataset_id(self) -> str | None: + return self._system_variable.dataset_id + + @property + def batch(self) -> str | None: + return self._system_variable.batch + + @property + def datasource_type(self) -> str | None: + return self._system_variable.datasource_type + + @property + def invoke_from(self) -> str | None: + return self._system_variable.invoke_from + + @property + def files(self) -> Sequence[File]: + """ + Get a copy of the files from the wrapped SystemVariable. + + Returns: + A defensive copy of the files sequence to prevent modification + """ + return tuple(self._system_variable.files) # Convert to immutable tuple + + @property + def datasource_info(self) -> Mapping[str, Any] | None: + """ + Get a copy of the datasource info from the wrapped SystemVariable. + + Returns: + A view of the datasource info mapping to prevent modification + """ + if self._system_variable.datasource_info is None: + return None + return MappingProxyType(self._system_variable.datasource_info) + + def __repr__(self) -> str: + """Return a string representation of the read-only view.""" + return f"SystemVariableReadOnlyView(system_variable={self._system_variable!r})" diff --git a/api/docker/entrypoint.sh b/api/docker/entrypoint.sh index 421d72a3a9..6313085e64 100755 --- a/api/docker/entrypoint.sh +++ b/api/docker/entrypoint.sh @@ -30,10 +30,42 @@ if [[ "${MODE}" == "worker" ]]; then CONCURRENCY_OPTION="-c ${CELERY_WORKER_AMOUNT:-1}" fi - exec celery -A celery_entrypoint.celery worker -P ${CELERY_WORKER_CLASS:-gevent} $CONCURRENCY_OPTION \ + # Configure queues based on edition if not explicitly set + if [[ -z "${CELERY_QUEUES}" ]]; then + if [[ "${EDITION}" == "CLOUD" ]]; then + # Cloud edition: separate queues for dataset and trigger tasks + DEFAULT_QUEUES="dataset,priority_dataset,priority_pipeline,pipeline,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,workflow_professional,workflow_team,workflow_sandbox,schedule_poller,schedule_executor,triggered_workflow_dispatcher,trigger_refresh_executor" + else + # Community edition (SELF_HOSTED): dataset, pipeline and workflow have separate queues + DEFAULT_QUEUES="dataset,priority_dataset,priority_pipeline,pipeline,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,workflow,schedule_poller,schedule_executor,triggered_workflow_dispatcher,trigger_refresh_executor" + fi + else + DEFAULT_QUEUES="${CELERY_QUEUES}" + fi + + # Support for Kubernetes deployment with specific queue workers + # Environment variables that can be set: + # - CELERY_WORKER_QUEUES: Comma-separated list of queues (overrides CELERY_QUEUES) + # - CELERY_WORKER_CONCURRENCY: Number of worker processes (overrides CELERY_WORKER_AMOUNT) + # - CELERY_WORKER_POOL: Pool implementation (overrides CELERY_WORKER_CLASS) + + if [[ -n "${CELERY_WORKER_QUEUES}" ]]; then + DEFAULT_QUEUES="${CELERY_WORKER_QUEUES}" + echo "Using CELERY_WORKER_QUEUES: ${DEFAULT_QUEUES}" + fi + + if [[ -n "${CELERY_WORKER_CONCURRENCY}" ]]; then + CONCURRENCY_OPTION="-c ${CELERY_WORKER_CONCURRENCY}" + echo "Using CELERY_WORKER_CONCURRENCY: ${CELERY_WORKER_CONCURRENCY}" + fi + + WORKER_POOL="${CELERY_WORKER_POOL:-${CELERY_WORKER_CLASS:-gevent}}" + echo "Starting Celery worker with queues: ${DEFAULT_QUEUES}" + + exec celery -A celery_entrypoint.celery worker -P ${WORKER_POOL} $CONCURRENCY_OPTION \ --max-tasks-per-child ${MAX_TASKS_PER_CHILD:-50} --loglevel ${LOG_LEVEL:-INFO} \ - -Q ${CELERY_QUEUES:-dataset,pipeline,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation} \ - --prefetch-multiplier=1 + -Q ${DEFAULT_QUEUES} \ + --prefetch-multiplier=${CELERY_PREFETCH_MULTIPLIER:-1} elif [[ "${MODE}" == "beat" ]]; then exec celery -A app.celery beat --loglevel ${LOG_LEVEL:-INFO} diff --git a/api/enums/__init__.py b/api/enums/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/enums/cloud_plan.py b/api/enums/cloud_plan.py new file mode 100644 index 0000000000..927cff5471 --- /dev/null +++ b/api/enums/cloud_plan.py @@ -0,0 +1,15 @@ +from enum import StrEnum, auto + + +class CloudPlan(StrEnum): + """ + Enum representing user plan types in the cloud platform. + + SANDBOX: Free/default plan with limited features + PROFESSIONAL: Professional paid plan + TEAM: Team collaboration paid plan + """ + + SANDBOX = auto() + PROFESSIONAL = auto() + TEAM = auto() diff --git a/api/events/event_handlers/__init__.py b/api/events/event_handlers/__init__.py index d714747e59..c79764983b 100644 --- a/api/events/event_handlers/__init__.py +++ b/api/events/event_handlers/__init__.py @@ -6,12 +6,18 @@ from .create_site_record_when_app_created import handle as handle_create_site_re from .delete_tool_parameters_cache_when_sync_draft_workflow import ( handle as handle_delete_tool_parameters_cache_when_sync_draft_workflow, ) +from .sync_plugin_trigger_when_app_created import handle as handle_sync_plugin_trigger_when_app_created +from .sync_webhook_when_app_created import handle as handle_sync_webhook_when_app_created +from .sync_workflow_schedule_when_app_published import handle as handle_sync_workflow_schedule_when_app_published from .update_app_dataset_join_when_app_model_config_updated import ( handle as handle_update_app_dataset_join_when_app_model_config_updated, ) from .update_app_dataset_join_when_app_published_workflow_updated import ( handle as handle_update_app_dataset_join_when_app_published_workflow_updated, ) +from .update_app_triggers_when_app_published_workflow_updated import ( + handle as handle_update_app_triggers_when_app_published_workflow_updated, +) # Consolidated handler replaces both deduct_quota_when_message_created and # update_provider_last_used_at_when_message_created @@ -24,7 +30,11 @@ __all__ = [ "handle_create_installed_app_when_app_created", "handle_create_site_record_when_app_created", "handle_delete_tool_parameters_cache_when_sync_draft_workflow", + "handle_sync_plugin_trigger_when_app_created", + "handle_sync_webhook_when_app_created", + "handle_sync_workflow_schedule_when_app_published", "handle_update_app_dataset_join_when_app_model_config_updated", "handle_update_app_dataset_join_when_app_published_workflow_updated", + "handle_update_app_triggers_when_app_published_workflow_updated", "handle_update_provider_when_message_created", ] diff --git a/api/events/event_handlers/sync_plugin_trigger_when_app_created.py b/api/events/event_handlers/sync_plugin_trigger_when_app_created.py new file mode 100644 index 0000000000..68be37dfdb --- /dev/null +++ b/api/events/event_handlers/sync_plugin_trigger_when_app_created.py @@ -0,0 +1,22 @@ +import logging + +from events.app_event import app_draft_workflow_was_synced +from models.model import App, AppMode +from models.workflow import Workflow +from services.trigger.trigger_service import TriggerService + +logger = logging.getLogger(__name__) + + +@app_draft_workflow_was_synced.connect +def handle(sender, synced_draft_workflow: Workflow, **kwargs): + """ + While creating a workflow or updating a workflow, we may need to sync + its plugin trigger relationships in DB. + """ + app: App = sender + if app.mode != AppMode.WORKFLOW.value: + # only handle workflow app, chatflow is not supported yet + return + + TriggerService.sync_plugin_trigger_relationships(app, synced_draft_workflow) diff --git a/api/events/event_handlers/sync_webhook_when_app_created.py b/api/events/event_handlers/sync_webhook_when_app_created.py new file mode 100644 index 0000000000..481561faa2 --- /dev/null +++ b/api/events/event_handlers/sync_webhook_when_app_created.py @@ -0,0 +1,22 @@ +import logging + +from events.app_event import app_draft_workflow_was_synced +from models.model import App, AppMode +from models.workflow import Workflow +from services.trigger.webhook_service import WebhookService + +logger = logging.getLogger(__name__) + + +@app_draft_workflow_was_synced.connect +def handle(sender, synced_draft_workflow: Workflow, **kwargs): + """ + While creating a workflow or updating a workflow, we may need to sync + its webhook relationships in DB. + """ + app: App = sender + if app.mode != AppMode.WORKFLOW.value: + # only handle workflow app, chatflow is not supported yet + return + + WebhookService.sync_webhook_relationships(app, synced_draft_workflow) diff --git a/api/events/event_handlers/sync_workflow_schedule_when_app_published.py b/api/events/event_handlers/sync_workflow_schedule_when_app_published.py new file mode 100644 index 0000000000..168513fc04 --- /dev/null +++ b/api/events/event_handlers/sync_workflow_schedule_when_app_published.py @@ -0,0 +1,86 @@ +import logging +from typing import cast + +from sqlalchemy import select +from sqlalchemy.orm import Session + +from core.workflow.nodes.trigger_schedule.entities import SchedulePlanUpdate +from events.app_event import app_published_workflow_was_updated +from extensions.ext_database import db +from models import AppMode, Workflow, WorkflowSchedulePlan +from services.trigger.schedule_service import ScheduleService + +logger = logging.getLogger(__name__) + + +@app_published_workflow_was_updated.connect +def handle(sender, **kwargs): + """ + Handle app published workflow update event to sync workflow_schedule_plans table. + + When a workflow is published, this handler will: + 1. Extract schedule trigger nodes from the workflow graph + 2. Compare with existing workflow_schedule_plans records + 3. Create/update/delete schedule plans as needed + """ + app = sender + if app.mode != AppMode.WORKFLOW.value: + return + + published_workflow = kwargs.get("published_workflow") + published_workflow = cast(Workflow, published_workflow) + + sync_schedule_from_workflow(tenant_id=app.tenant_id, app_id=app.id, workflow=published_workflow) + + +def sync_schedule_from_workflow(tenant_id: str, app_id: str, workflow: Workflow) -> WorkflowSchedulePlan | None: + """ + Sync schedule plan from workflow graph configuration. + + Args: + tenant_id: Tenant ID + app_id: App ID + workflow: Published workflow instance + + Returns: + Updated or created WorkflowSchedulePlan, or None if no schedule node + """ + with Session(db.engine) as session: + schedule_config = ScheduleService.extract_schedule_config(workflow) + + existing_plan = session.scalar( + select(WorkflowSchedulePlan).where( + WorkflowSchedulePlan.tenant_id == tenant_id, + WorkflowSchedulePlan.app_id == app_id, + ) + ) + + if not schedule_config: + if existing_plan: + logger.info("No schedule node in workflow for app %s, removing schedule plan", app_id) + ScheduleService.delete_schedule(session=session, schedule_id=existing_plan.id) + session.commit() + return None + + if existing_plan: + updates = SchedulePlanUpdate( + node_id=schedule_config.node_id, + cron_expression=schedule_config.cron_expression, + timezone=schedule_config.timezone, + ) + updated_plan = ScheduleService.update_schedule( + session=session, + schedule_id=existing_plan.id, + updates=updates, + ) + session.commit() + return updated_plan + else: + new_plan = ScheduleService.create_schedule( + session=session, + tenant_id=tenant_id, + app_id=app_id, + config=schedule_config, + ) + session.commit() + return new_plan diff --git a/api/events/event_handlers/update_app_triggers_when_app_published_workflow_updated.py b/api/events/event_handlers/update_app_triggers_when_app_published_workflow_updated.py new file mode 100644 index 0000000000..430514ada2 --- /dev/null +++ b/api/events/event_handlers/update_app_triggers_when_app_published_workflow_updated.py @@ -0,0 +1,114 @@ +from typing import cast + +from sqlalchemy import select +from sqlalchemy.orm import Session + +from core.workflow.nodes import NodeType +from events.app_event import app_published_workflow_was_updated +from extensions.ext_database import db +from models import AppMode +from models.enums import AppTriggerStatus +from models.trigger import AppTrigger +from models.workflow import Workflow + + +@app_published_workflow_was_updated.connect +def handle(sender, **kwargs): + """ + Handle app published workflow update event to sync app_triggers table. + + When a workflow is published, this handler will: + 1. Extract trigger nodes from the workflow graph + 2. Compare with existing app_triggers records + 3. Add new triggers and remove obsolete ones + """ + app = sender + if app.mode != AppMode.WORKFLOW.value: + return + + published_workflow = kwargs.get("published_workflow") + published_workflow = cast(Workflow, published_workflow) + # Extract trigger info from workflow + trigger_infos = get_trigger_infos_from_workflow(published_workflow) + + with Session(db.engine) as session: + # Get existing app triggers + existing_triggers = ( + session.execute( + select(AppTrigger).where(AppTrigger.tenant_id == app.tenant_id, AppTrigger.app_id == app.id) + ) + .scalars() + .all() + ) + + # Convert existing triggers to dict for easy lookup + existing_triggers_map = {trigger.node_id: trigger for trigger in existing_triggers} + + # Get current and new node IDs + existing_node_ids = set(existing_triggers_map.keys()) + new_node_ids = {info["node_id"] for info in trigger_infos} + + # Calculate changes + added_node_ids = new_node_ids - existing_node_ids + removed_node_ids = existing_node_ids - new_node_ids + + # Remove obsolete triggers + for node_id in removed_node_ids: + session.delete(existing_triggers_map[node_id]) + + for trigger_info in trigger_infos: + node_id = trigger_info["node_id"] + + if node_id in added_node_ids: + # Create new trigger + app_trigger = AppTrigger( + tenant_id=app.tenant_id, + app_id=app.id, + trigger_type=trigger_info["node_type"], + title=trigger_info["node_title"], + node_id=node_id, + provider_name=trigger_info.get("node_provider_name", ""), + status=AppTriggerStatus.ENABLED, + ) + session.add(app_trigger) + elif node_id in existing_node_ids: + # Update existing trigger if needed + existing_trigger = existing_triggers_map[node_id] + new_title = trigger_info["node_title"] + if new_title and existing_trigger.title != new_title: + existing_trigger.title = new_title + session.add(existing_trigger) + + session.commit() + + +def get_trigger_infos_from_workflow(published_workflow: Workflow) -> list[dict]: + """ + Extract trigger node information from the workflow graph. + + Returns: + List of trigger info dictionaries containing: + - node_type: The type of the trigger node ('trigger-webhook', 'trigger-schedule', 'trigger-plugin') + - node_id: The node ID in the workflow + - node_title: The title of the node + - node_provider_name: The name of the node's provider, only for plugin + """ + graph = published_workflow.graph_dict + if not graph: + return [] + + nodes = graph.get("nodes", []) + trigger_types = {NodeType.TRIGGER_WEBHOOK.value, NodeType.TRIGGER_SCHEDULE.value, NodeType.TRIGGER_PLUGIN.value} + + trigger_infos = [ + { + "node_type": node.get("data", {}).get("type"), + "node_id": node.get("id"), + "node_title": node.get("data", {}).get("title"), + "node_provider_name": node.get("data", {}).get("provider_name"), + } + for node in nodes + if node.get("data", {}).get("type") in trigger_types + ] + + return trigger_infos diff --git a/api/extensions/ext_blueprints.py b/api/extensions/ext_blueprints.py index 82f0542b35..44b50e42ee 100644 --- a/api/extensions/ext_blueprints.py +++ b/api/extensions/ext_blueprints.py @@ -18,6 +18,7 @@ def init_app(app: DifyApp): from controllers.inner_api import bp as inner_api_bp from controllers.mcp import bp as mcp_bp from controllers.service_api import bp as service_api_bp + from controllers.trigger import bp as trigger_bp from controllers.web import bp as web_bp CORS( @@ -56,3 +57,11 @@ def init_app(app: DifyApp): app.register_blueprint(inner_api_bp) app.register_blueprint(mcp_bp) + + # Register trigger blueprint with CORS for webhook calls + CORS( + trigger_bp, + allow_headers=["Content-Type", "Authorization", "X-App-Code"], + methods=["GET", "PUT", "POST", "DELETE", "OPTIONS", "PATCH", "HEAD"], + ) + app.register_blueprint(trigger_bp) diff --git a/api/extensions/ext_celery.py b/api/extensions/ext_celery.py index 6d7d81ed87..5cf4984709 100644 --- a/api/extensions/ext_celery.py +++ b/api/extensions/ext_celery.py @@ -96,7 +96,10 @@ def init_app(app: DifyApp) -> Celery: celery_app.set_default() app.extensions["celery"] = celery_app - imports = [] + imports = [ + "tasks.async_workflow_tasks", # trigger workers + "tasks.trigger_processing_tasks", # async trigger processing + ] day = dify_config.CELERY_BEAT_SCHEDULER_TIME # if you add a new task, please add the switch to CeleryScheduleTasksConfig @@ -157,6 +160,18 @@ def init_app(app: DifyApp) -> Celery: "task": "schedule.clean_workflow_runlogs_precise.clean_workflow_runlogs_precise", "schedule": crontab(minute="0", hour="2"), } + if dify_config.ENABLE_WORKFLOW_SCHEDULE_POLLER_TASK: + imports.append("schedule.workflow_schedule_task") + beat_schedule["workflow_schedule_task"] = { + "task": "schedule.workflow_schedule_task.poll_workflow_schedules", + "schedule": timedelta(minutes=dify_config.WORKFLOW_SCHEDULE_POLLER_INTERVAL), + } + if dify_config.ENABLE_TRIGGER_PROVIDER_REFRESH_TASK: + imports.append("schedule.trigger_provider_refresh_task") + beat_schedule["trigger_provider_refresh"] = { + "task": "schedule.trigger_provider_refresh_task.trigger_provider_refresh", + "schedule": timedelta(minutes=dify_config.TRIGGER_PROVIDER_REFRESH_INTERVAL), + } celery_app.conf.update(beat_schedule=beat_schedule, imports=imports) return celery_app diff --git a/api/extensions/ext_commands.py b/api/extensions/ext_commands.py index 79dcdda6e3..71a63168a5 100644 --- a/api/extensions/ext_commands.py +++ b/api/extensions/ext_commands.py @@ -23,6 +23,7 @@ def init_app(app: DifyApp): reset_password, setup_datasource_oauth_client, setup_system_tool_oauth_client, + setup_system_trigger_oauth_client, transform_datasource_credentials, upgrade_db, vdb_migrate, @@ -47,6 +48,7 @@ def init_app(app: DifyApp): clear_orphaned_file_records, remove_orphaned_files_on_storage, setup_system_tool_oauth_client, + setup_system_trigger_oauth_client, cleanup_orphaned_draft_variables, migrate_oss, setup_datasource_oauth_client, diff --git a/api/extensions/ext_login.py b/api/extensions/ext_login.py index ed4fe332c1..74299956c0 100644 --- a/api/extensions/ext_login.py +++ b/api/extensions/ext_login.py @@ -6,10 +6,11 @@ from flask_login import user_loaded_from_request, user_logged_in from werkzeug.exceptions import NotFound, Unauthorized from configs import dify_config +from constants import HEADER_NAME_APP_CODE from dify_app import DifyApp from extensions.ext_database import db from libs.passport import PassportService -from libs.token import extract_access_token +from libs.token import extract_access_token, extract_webapp_passport from models import Account, Tenant, TenantAccountJoin from models.model import AppMCPServer, EndUser from services.account_service import AccountService @@ -61,14 +62,30 @@ def load_user_from_request(request_from_flask_login): logged_in_account = AccountService.load_logged_in_account(account_id=user_id) return logged_in_account elif request.blueprint == "web": - decoded = PassportService().verify(auth_token) - end_user_id = decoded.get("end_user_id") - if not end_user_id: - raise Unauthorized("Invalid Authorization token.") - end_user = db.session.query(EndUser).where(EndUser.id == decoded["end_user_id"]).first() - if not end_user: - raise NotFound("End user not found.") - return end_user + app_code = request.headers.get(HEADER_NAME_APP_CODE) + webapp_token = extract_webapp_passport(app_code, request) if app_code else None + + if webapp_token: + decoded = PassportService().verify(webapp_token) + end_user_id = decoded.get("end_user_id") + if not end_user_id: + raise Unauthorized("Invalid Authorization token.") + end_user = db.session.query(EndUser).where(EndUser.id == end_user_id).first() + if not end_user: + raise NotFound("End user not found.") + return end_user + else: + if not auth_token: + raise Unauthorized("Invalid Authorization token.") + decoded = PassportService().verify(auth_token) + end_user_id = decoded.get("end_user_id") + if end_user_id: + end_user = db.session.query(EndUser).where(EndUser.id == end_user_id).first() + if not end_user: + raise NotFound("End user not found.") + return end_user + else: + raise Unauthorized("Invalid Authorization token for web API.") elif request.blueprint == "mcp": server_code = request.view_args.get("server_code") if request.view_args else None if not server_code: diff --git a/api/extensions/ext_storage.py b/api/extensions/ext_storage.py index 2960cde242..a609f13dbc 100644 --- a/api/extensions/ext_storage.py +++ b/api/extensions/ext_storage.py @@ -85,7 +85,7 @@ class Storage: case _: raise ValueError(f"unsupported storage type {storage_type}") - def save(self, filename, data): + def save(self, filename: str, data: bytes): self.storage_runner.save(filename, data) @overload diff --git a/api/extensions/storage/base_storage.py b/api/extensions/storage/base_storage.py index 0393206e54..8ddedb24ae 100644 --- a/api/extensions/storage/base_storage.py +++ b/api/extensions/storage/base_storage.py @@ -8,7 +8,7 @@ class BaseStorage(ABC): """Interface for file storage.""" @abstractmethod - def save(self, filename, data): + def save(self, filename: str, data: bytes): raise NotImplementedError @abstractmethod diff --git a/api/extensions/storage/google_cloud_storage.py b/api/extensions/storage/google_cloud_storage.py index d352996518..7f59252f2f 100644 --- a/api/extensions/storage/google_cloud_storage.py +++ b/api/extensions/storage/google_cloud_storage.py @@ -3,7 +3,7 @@ import io import json from collections.abc import Generator -from google.cloud import storage as google_cloud_storage +from google.cloud import storage as google_cloud_storage # type: ignore from configs import dify_config from extensions.storage.base_storage import BaseStorage diff --git a/api/fields/app_fields.py b/api/fields/app_fields.py index 1f14d663b8..7191933eed 100644 --- a/api/fields/app_fields.py +++ b/api/fields/app_fields.py @@ -116,6 +116,7 @@ app_partial_fields = { "access_mode": fields.String, "create_user_name": fields.String, "author_name": fields.String, + "has_draft_trigger": fields.Boolean, } diff --git a/api/fields/workflow_app_log_fields.py b/api/fields/workflow_app_log_fields.py index 243efd817c..4cbdf6f0ca 100644 --- a/api/fields/workflow_app_log_fields.py +++ b/api/fields/workflow_app_log_fields.py @@ -8,6 +8,7 @@ from libs.helper import TimestampField workflow_app_log_partial_fields = { "id": fields.String, "workflow_run": fields.Nested(workflow_run_for_log_fields, attribute="workflow_run", allow_null=True), + "details": fields.Raw(attribute="details"), "created_from": fields.String, "created_by_role": fields.String, "created_by_account": fields.Nested(simple_account_fields, attribute="created_by_account", allow_null=True), diff --git a/api/fields/workflow_run_fields.py b/api/fields/workflow_run_fields.py index 79594beeed..821ce62ecc 100644 --- a/api/fields/workflow_run_fields.py +++ b/api/fields/workflow_run_fields.py @@ -8,6 +8,7 @@ workflow_run_for_log_fields = { "id": fields.String, "version": fields.String, "status": fields.String, + "triggered_from": fields.String, "error": fields.String, "elapsed_time": fields.Float, "total_tokens": fields.Integer, diff --git a/api/fields/workflow_trigger_fields.py b/api/fields/workflow_trigger_fields.py new file mode 100644 index 0000000000..ce51d1833a --- /dev/null +++ b/api/fields/workflow_trigger_fields.py @@ -0,0 +1,25 @@ +from flask_restx import fields + +trigger_fields = { + "id": fields.String, + "trigger_type": fields.String, + "title": fields.String, + "node_id": fields.String, + "provider_name": fields.String, + "icon": fields.String, + "status": fields.String, + "created_at": fields.DateTime(dt_format="iso8601"), + "updated_at": fields.DateTime(dt_format="iso8601"), +} + +triggers_list_fields = {"data": fields.List(fields.Nested(trigger_fields))} + + +webhook_trigger_fields = { + "id": fields.String, + "webhook_id": fields.String, + "webhook_url": fields.String, + "webhook_debug_url": fields.String, + "node_id": fields.String, + "created_at": fields.DateTime(dt_format="iso8601"), +} diff --git a/api/gunicorn.conf.py b/api/gunicorn.conf.py index 943ee100ca..da75d25ba6 100644 --- a/api/gunicorn.conf.py +++ b/api/gunicorn.conf.py @@ -2,6 +2,19 @@ import psycogreen.gevent as pscycogreen_gevent # type: ignore from gevent import events as gevent_events from grpc.experimental import gevent as grpc_gevent # type: ignore +# WARNING: This module is loaded very early in the Gunicorn worker lifecycle, +# before gevent's monkey-patching is applied. Importing modules at the top level here can +# interfere with gevent's ability to properly patch the standard library, +# potentially causing subtle and difficult-to-diagnose bugs. +# +# To ensure correct behavior, defer any initialization or imports that depend on monkey-patching +# to the `post_patch` hook below, or use a gevent_events subscriber as shown. +# +# For further context, see: https://github.com/langgenius/dify/issues/26689 +# +# Note: The `post_fork` hook is also executed before monkey-patching, +# so moving imports there does not resolve this issue. + # NOTE(QuantumGhost): here we cannot use post_fork to patch gRPC, as # grpc_gevent.init_gevent must be called after patching stdlib. # Gunicorn calls `post_init` before applying monkey patch. @@ -11,7 +24,7 @@ from grpc.experimental import gevent as grpc_gevent # type: ignore # ref: # - https://github.com/grpc/grpc/blob/62533ea13879d6ee95c6fda11ec0826ca822c9dd/src/python/grpcio/grpc/experimental/gevent.py # - https://github.com/gevent/gevent/issues/2060#issuecomment-3016768668 -# - https://github.com/benoitc/gunicorn/blob/master/gunicorn/arbiter.py#L607-L613 +# - https://github.com/benoitc/gunicorn/blob/23.0.0/gunicorn/arbiter.py#L605-L609 def post_patch(event): diff --git a/api/libs/broadcast_channel/channel.py b/api/libs/broadcast_channel/channel.py new file mode 100644 index 0000000000..5bbf0c79a3 --- /dev/null +++ b/api/libs/broadcast_channel/channel.py @@ -0,0 +1,134 @@ +""" +Broadcast channel for Pub/Sub messaging. +""" + +import types +from abc import abstractmethod +from collections.abc import Iterator +from contextlib import AbstractContextManager +from typing import Protocol, Self + + +class Subscription(AbstractContextManager["Subscription"], Protocol): + """A subscription to a topic that provides an iterator over received messages. + The subscription can be used as a context manager and will automatically + close when exiting the context. + + Note: `Subscription` instances are not thread-safe. Each thread should create its own + subscription. + """ + + @abstractmethod + def __iter__(self) -> Iterator[bytes]: + """`__iter__` returns an iterator used to consume the message from this subscription. + + If the caller did not enter the context, `__iter__` may lazily perform the setup before + yielding messages; otherwise `__enter__` handles it.” + + If the subscription is closed, then the returned iterator exits without + raising any error. + """ + ... + + @abstractmethod + def close(self) -> None: + """close closes the subscription, releases any resources associated with it.""" + ... + + def __enter__(self) -> Self: + """`__enter__` does the setup logic of the subscription (if any), and return itself.""" + return self + + def __exit__( + self, + exc_type: type[BaseException] | None, + exc_value: BaseException | None, + traceback: types.TracebackType | None, + ) -> bool | None: + self.close() + return None + + @abstractmethod + def receive(self, timeout: float | None = 0.1) -> bytes | None: + """Receive the next message from the broadcast channel. + + If `timeout` is specified, this method returns `None` if no message is + received within the given period. If `timeout` is `None`, the call blocks + until a message is received. + + Calling receive with `timeout=None` is highly discouraged, as it is impossible to + cancel a blocking subscription. + + :param timeout: timeout for receive message, in seconds. + + Returns: + bytes: The received message as a byte string, or + None: If the timeout expires before a message is received. + + Raises: + SubscriptionClosed: If the subscription has already been closed. + """ + ... + + +class Producer(Protocol): + """Producer is an interface for message publishing. It is already bound to a specific topic. + + `Producer` implementations must be thread-safe and support concurrent use by multiple threads. + """ + + @abstractmethod + def publish(self, payload: bytes) -> None: + """Publish a message to the bounded topic.""" + ... + + +class Subscriber(Protocol): + """Subscriber is an interface for subscription creation. It is already bound to a specific topic. + + `Subscriber` implementations must be thread-safe and support concurrent use by multiple threads. + """ + + @abstractmethod + def subscribe(self) -> Subscription: + pass + + +class Topic(Producer, Subscriber, Protocol): + """A named channel for publishing and subscribing to messages. + + Topics provide both read and write access. For restricted access, + use as_producer() for write-only view or as_subscriber() for read-only view. + + `Topic` implementations must be thread-safe and support concurrent use by multiple threads. + """ + + @abstractmethod + def as_producer(self) -> Producer: + """as_producer creates a write-only view for this topic.""" + ... + + @abstractmethod + def as_subscriber(self) -> Subscriber: + """as_subscriber create a read-only view for this topic.""" + ... + + +class BroadcastChannel(Protocol): + """A broadcasting channel is a channel supporting broadcasting semantics. + + Each channel is identified by a topic, different topics are isolated and do not affect each other. + + There can be multiple subscriptions to a specific topic. When a publisher publishes a message to + a specific topic, all subscription should receive the published message. + + There are no restriction for the persistence of messages. Once a subscription is created, it + should receive all subsequent messages published. + + `BroadcastChannel` implementations must be thread-safe and support concurrent use by multiple threads. + """ + + @abstractmethod + def topic(self, topic: str) -> "Topic": + """topic returns a `Topic` instance for the given topic name.""" + ... diff --git a/api/libs/broadcast_channel/exc.py b/api/libs/broadcast_channel/exc.py new file mode 100644 index 0000000000..ab958c94ed --- /dev/null +++ b/api/libs/broadcast_channel/exc.py @@ -0,0 +1,12 @@ +class BroadcastChannelError(Exception): + """`BroadcastChannelError` is the base class for all exceptions related + to `BroadcastChannel`.""" + + pass + + +class SubscriptionClosedError(BroadcastChannelError): + """SubscriptionClosedError means that the subscription has been closed and + methods for consuming messages should not be called.""" + + pass diff --git a/api/libs/broadcast_channel/redis/__init__.py b/api/libs/broadcast_channel/redis/__init__.py new file mode 100644 index 0000000000..138fef5c5f --- /dev/null +++ b/api/libs/broadcast_channel/redis/__init__.py @@ -0,0 +1,3 @@ +from .channel import BroadcastChannel + +__all__ = ["BroadcastChannel"] diff --git a/api/libs/broadcast_channel/redis/channel.py b/api/libs/broadcast_channel/redis/channel.py new file mode 100644 index 0000000000..e6b32345be --- /dev/null +++ b/api/libs/broadcast_channel/redis/channel.py @@ -0,0 +1,200 @@ +import logging +import queue +import threading +import types +from collections.abc import Generator, Iterator +from typing import Self + +from libs.broadcast_channel.channel import Producer, Subscriber, Subscription +from libs.broadcast_channel.exc import SubscriptionClosedError +from redis import Redis +from redis.client import PubSub + +_logger = logging.getLogger(__name__) + + +class BroadcastChannel: + """ + Redis Pub/Sub based broadcast channel implementation. + + Provides "at most once" delivery semantics for messages published to channels. + Uses Redis PUBLISH/SUBSCRIBE commands for real-time message delivery. + + The `redis_client` used to construct BroadcastChannel should have `decode_responses` set to `False`. + """ + + def __init__( + self, + redis_client: Redis, + ): + self._client = redis_client + + def topic(self, topic: str) -> "Topic": + return Topic(self._client, topic) + + +class Topic: + def __init__(self, redis_client: Redis, topic: str): + self._client = redis_client + self._topic = topic + + def as_producer(self) -> Producer: + return self + + def publish(self, payload: bytes) -> None: + self._client.publish(self._topic, payload) + + def as_subscriber(self) -> Subscriber: + return self + + def subscribe(self) -> Subscription: + return _RedisSubscription( + pubsub=self._client.pubsub(), + topic=self._topic, + ) + + +class _RedisSubscription(Subscription): + def __init__( + self, + pubsub: PubSub, + topic: str, + ): + # The _pubsub is None only if the subscription is closed. + self._pubsub: PubSub | None = pubsub + self._topic = topic + self._closed = threading.Event() + self._queue: queue.Queue[bytes] = queue.Queue(maxsize=1024) + self._dropped_count = 0 + self._listener_thread: threading.Thread | None = None + self._start_lock = threading.Lock() + self._started = False + + def _start_if_needed(self) -> None: + with self._start_lock: + if self._started: + return + if self._closed.is_set(): + raise SubscriptionClosedError("The Redis subscription is closed") + if self._pubsub is None: + raise SubscriptionClosedError("The Redis subscription has been cleaned up") + + self._pubsub.subscribe(self._topic) + _logger.debug("Subscribed to channel %s", self._topic) + + self._listener_thread = threading.Thread( + target=self._listen, + name=f"redis-broadcast-{self._topic}", + daemon=True, + ) + self._listener_thread.start() + self._started = True + + def _listen(self) -> None: + pubsub = self._pubsub + assert pubsub is not None, "PubSub should not be None while starting listening." + while not self._closed.is_set(): + raw_message = pubsub.get_message(ignore_subscribe_messages=True, timeout=0.1) + + if raw_message is None: + continue + + if raw_message.get("type") != "message": + continue + + channel_field = raw_message.get("channel") + if isinstance(channel_field, bytes): + channel_name = channel_field.decode("utf-8") + elif isinstance(channel_field, str): + channel_name = channel_field + else: + channel_name = str(channel_field) + + if channel_name != self._topic: + _logger.warning("Ignoring message from unexpected channel %s", channel_name) + continue + + payload_bytes: bytes | None = raw_message.get("data") + if not isinstance(payload_bytes, bytes): + _logger.error("Received invalid data from channel %s, type=%s", self._topic, type(payload_bytes)) + continue + + self._enqueue_message(payload_bytes) + + _logger.debug("Listener thread stopped for channel %s", self._topic) + pubsub.unsubscribe(self._topic) + pubsub.close() + _logger.debug("PubSub closed for topic %s", self._topic) + self._pubsub = None + + def _enqueue_message(self, payload: bytes) -> None: + while not self._closed.is_set(): + try: + self._queue.put_nowait(payload) + return + except queue.Full: + try: + self._queue.get_nowait() + self._dropped_count += 1 + _logger.debug( + "Dropped message from Redis subscription, topic=%s, total_dropped=%d", + self._topic, + self._dropped_count, + ) + except queue.Empty: + continue + return + + def _message_iterator(self) -> Generator[bytes, None, None]: + while not self._closed.is_set(): + try: + item = self._queue.get(timeout=0.1) + except queue.Empty: + continue + + yield item + + def __iter__(self) -> Iterator[bytes]: + if self._closed.is_set(): + raise SubscriptionClosedError("The Redis subscription is closed") + self._start_if_needed() + return iter(self._message_iterator()) + + def receive(self, timeout: float | None = None) -> bytes | None: + if self._closed.is_set(): + raise SubscriptionClosedError("The Redis subscription is closed") + self._start_if_needed() + + try: + item = self._queue.get(timeout=timeout) + except queue.Empty: + return None + + return item + + def __enter__(self) -> Self: + self._start_if_needed() + return self + + def __exit__( + self, + exc_type: type[BaseException] | None, + exc_value: BaseException | None, + traceback: types.TracebackType | None, + ) -> bool | None: + self.close() + return None + + def close(self) -> None: + if self._closed.is_set(): + return + + self._closed.set() + # NOTE: PubSub is not thread-safe. More specifically, the `PubSub.close` method and the `PubSub.get_message` + # method should NOT be called concurrently. + # + # Due to the restriction above, the PubSub cleanup logic happens inside the consumer thread. + listener = self._listener_thread + if listener is not None: + listener.join(timeout=1.0) + self._listener_thread = None diff --git a/api/libs/datetime_utils.py b/api/libs/datetime_utils.py index e576a34629..c08578981b 100644 --- a/api/libs/datetime_utils.py +++ b/api/libs/datetime_utils.py @@ -2,6 +2,8 @@ import abc import datetime from typing import Protocol +import pytz + class _NowFunction(Protocol): @abc.abstractmethod @@ -20,3 +22,62 @@ def naive_utc_now() -> datetime.datetime: representing current UTC time. """ return _now_func(datetime.UTC).replace(tzinfo=None) + + +def ensure_naive_utc(dt: datetime.datetime) -> datetime.datetime: + """Return the datetime as naive UTC (tzinfo=None). + + If the input is timezone-aware, convert to UTC and drop the tzinfo. + Assumes naive datetimes are already expressed in UTC. + """ + if dt.tzinfo is None: + return dt + return dt.astimezone(datetime.UTC).replace(tzinfo=None) + + +def parse_time_range( + start: str | None, end: str | None, tzname: str +) -> tuple[datetime.datetime | None, datetime.datetime | None]: + """ + Parse time range strings and convert to UTC datetime objects. + Handles DST ambiguity and non-existent times gracefully. + + Args: + start: Start time string (YYYY-MM-DD HH:MM) + end: End time string (YYYY-MM-DD HH:MM) + tzname: Timezone name + + Returns: + tuple: (start_datetime_utc, end_datetime_utc) + + Raises: + ValueError: When time range is invalid or start > end + """ + tz = pytz.timezone(tzname) + utc = pytz.utc + + def _parse(time_str: str | None, label: str) -> datetime.datetime | None: + if not time_str: + return None + + try: + dt = datetime.datetime.strptime(time_str, "%Y-%m-%d %H:%M").replace(second=0) + except ValueError as e: + raise ValueError(f"Invalid {label} time format: {e}") + + try: + return tz.localize(dt, is_dst=None).astimezone(utc) + except pytz.AmbiguousTimeError: + return tz.localize(dt, is_dst=False).astimezone(utc) + except pytz.NonExistentTimeError: + dt += datetime.timedelta(hours=1) + return tz.localize(dt, is_dst=None).astimezone(utc) + + start_dt = _parse(start, "start") + end_dt = _parse(end, "end") + + # Range validation + if start_dt and end_dt and start_dt > end_dt: + raise ValueError("start must be earlier than or equal to end") + + return start_dt, end_dt diff --git a/api/libs/external_api.py b/api/libs/external_api.py index 1a4fde960c..61a90ee4a9 100644 --- a/api/libs/external_api.py +++ b/api/libs/external_api.py @@ -9,9 +9,8 @@ from werkzeug.exceptions import HTTPException from werkzeug.http import HTTP_STATUS_CODES from configs import dify_config -from constants import COOKIE_NAME_ACCESS_TOKEN, COOKIE_NAME_CSRF_TOKEN, COOKIE_NAME_REFRESH_TOKEN from core.errors.error import AppInvokeQuotaExceededError -from libs.token import is_secure +from libs.token import build_force_logout_cookie_headers def http_status_message(code): @@ -73,15 +72,7 @@ def register_external_error_handlers(api: Api): error_code = getattr(e, "error_code", None) if error_code == "unauthorized_and_force_logout": # Add Set-Cookie headers to clear auth cookies - - secure = is_secure() - # response is not accessible, so we need to do it ugly - common_part = "Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT; HttpOnly" - headers["Set-Cookie"] = [ - f'{COOKIE_NAME_ACCESS_TOKEN}=""; {common_part}{"; Secure" if secure else ""}; SameSite=Lax', - f'{COOKIE_NAME_CSRF_TOKEN}=""; {common_part}{"; Secure" if secure else ""}; SameSite=Lax', - f'{COOKIE_NAME_REFRESH_TOKEN}=""; {common_part}{"; Secure" if secure else ""}; SameSite=Lax', - ] + headers["Set-Cookie"] = build_force_logout_cookie_headers() return data, status_code, headers _ = handle_http_exception diff --git a/api/libs/schedule_utils.py b/api/libs/schedule_utils.py new file mode 100644 index 0000000000..1ab5f499e9 --- /dev/null +++ b/api/libs/schedule_utils.py @@ -0,0 +1,108 @@ +from datetime import UTC, datetime + +import pytz +from croniter import croniter + + +def calculate_next_run_at( + cron_expression: str, + timezone: str, + base_time: datetime | None = None, +) -> datetime: + """ + Calculate the next run time for a cron expression in a specific timezone. + + Args: + cron_expression: Standard 5-field cron expression or predefined expression + timezone: Timezone string (e.g., 'UTC', 'America/New_York') + base_time: Base time to calculate from (defaults to current UTC time) + + Returns: + Next run time in UTC + + Note: + Supports enhanced cron syntax including: + - Month abbreviations: JAN, FEB, MAR-JUN, JAN,JUN,DEC + - Day abbreviations: MON, TUE, MON-FRI, SUN,WED,FRI + - Predefined expressions: @daily, @weekly, @monthly, @yearly, @hourly + - Special characters: ? wildcard, L (last day), Sunday as 7 + - Standard 5-field format only (minute hour day month dayOfWeek) + """ + # Validate cron expression format to match frontend behavior + parts = cron_expression.strip().split() + + # Support both 5-field format and predefined expressions (matching frontend) + if len(parts) != 5 and not cron_expression.startswith("@"): + raise ValueError( + f"Cron expression must have exactly 5 fields or be a predefined expression " + f"(@daily, @weekly, etc.). Got {len(parts)} fields: '{cron_expression}'" + ) + + tz = pytz.timezone(timezone) + + if base_time is None: + base_time = datetime.now(UTC) + + base_time_tz = base_time.astimezone(tz) + cron = croniter(cron_expression, base_time_tz) + next_run_tz = cron.get_next(datetime) + next_run_utc = next_run_tz.astimezone(UTC) + + return next_run_utc + + +def convert_12h_to_24h(time_str: str) -> tuple[int, int]: + """ + Parse 12-hour time format to 24-hour format for cron compatibility. + + Args: + time_str: Time string in format "HH:MM AM/PM" (e.g., "12:30 PM") + + Returns: + Tuple of (hour, minute) in 24-hour format + + Raises: + ValueError: If time string format is invalid or values are out of range + + Examples: + - "12:00 AM" -> (0, 0) # Midnight + - "12:00 PM" -> (12, 0) # Noon + - "1:30 PM" -> (13, 30) + - "11:59 PM" -> (23, 59) + """ + if not time_str or not time_str.strip(): + raise ValueError("Time string cannot be empty") + + parts = time_str.strip().split() + if len(parts) != 2: + raise ValueError(f"Invalid time format: '{time_str}'. Expected 'HH:MM AM/PM'") + + time_part, period = parts + period = period.upper() + + if period not in ["AM", "PM"]: + raise ValueError(f"Invalid period: '{period}'. Must be 'AM' or 'PM'") + + time_parts = time_part.split(":") + if len(time_parts) != 2: + raise ValueError(f"Invalid time format: '{time_part}'. Expected 'HH:MM'") + + try: + hour = int(time_parts[0]) + minute = int(time_parts[1]) + except ValueError as e: + raise ValueError(f"Invalid time values: {e}") + + if hour < 1 or hour > 12: + raise ValueError(f"Invalid hour: {hour}. Must be between 1 and 12") + + if minute < 0 or minute > 59: + raise ValueError(f"Invalid minute: {minute}. Must be between 0 and 59") + + # Handle 12-hour to 24-hour edge cases + if period == "PM" and hour != 12: + hour += 12 + elif period == "AM" and hour == 12: + hour = 0 + + return hour, minute diff --git a/api/libs/token.py b/api/libs/token.py index 4be25696e7..098ff958da 100644 --- a/api/libs/token.py +++ b/api/libs/token.py @@ -12,6 +12,7 @@ from constants import ( COOKIE_NAME_CSRF_TOKEN, COOKIE_NAME_PASSPORT, COOKIE_NAME_REFRESH_TOKEN, + COOKIE_NAME_WEBAPP_ACCESS_TOKEN, HEADER_NAME_CSRF_TOKEN, HEADER_NAME_PASSPORT, ) @@ -29,17 +30,28 @@ def is_secure() -> bool: return dify_config.CONSOLE_WEB_URL.startswith("https") and dify_config.CONSOLE_API_URL.startswith("https") +def _cookie_domain() -> str | None: + """ + Returns the normalized cookie domain. + + Leading dots are stripped from the configured domain. Historically, a leading dot + indicated that a cookie should be sent to all subdomains, but modern browsers treat + 'example.com' and '.example.com' identically. This normalization ensures consistent + behavior and avoids confusion. + """ + domain = dify_config.COOKIE_DOMAIN.strip() + domain = domain.removeprefix(".") + return domain or None + + def _real_cookie_name(cookie_name: str) -> str: - if is_secure(): + if is_secure() and _cookie_domain() is None: return "__Host-" + cookie_name else: return cookie_name def _try_extract_from_header(request: Request) -> str | None: - """ - Try to extract access token from header - """ auth_header = request.headers.get("Authorization") if auth_header: if " " not in auth_header: @@ -54,40 +66,30 @@ def _try_extract_from_header(request: Request) -> str | None: return None +def extract_refresh_token(request: Request) -> str | None: + return request.cookies.get(_real_cookie_name(COOKIE_NAME_REFRESH_TOKEN)) + + def extract_csrf_token(request: Request) -> str | None: - """ - Try to extract CSRF token from header or cookie. - """ return request.headers.get(HEADER_NAME_CSRF_TOKEN) def extract_csrf_token_from_cookie(request: Request) -> str | None: - """ - Try to extract CSRF token from cookie. - """ return request.cookies.get(_real_cookie_name(COOKIE_NAME_CSRF_TOKEN)) def extract_access_token(request: Request) -> str | None: - """ - Try to extract access token from cookie, header or params. - - Access token is either for console session or webapp passport exchange. - """ - def _try_extract_from_cookie(request: Request) -> str | None: return request.cookies.get(_real_cookie_name(COOKIE_NAME_ACCESS_TOKEN)) return _try_extract_from_cookie(request) or _try_extract_from_header(request) +def extract_webapp_access_token(request: Request) -> str | None: + return request.cookies.get(_real_cookie_name(COOKIE_NAME_WEBAPP_ACCESS_TOKEN)) or _try_extract_from_header(request) + + def extract_webapp_passport(app_code: str, request: Request) -> str | None: - """ - Try to extract app token from header or params. - - Webapp access token (part of passport) is only used for webapp session. - """ - def _try_extract_passport_token_from_cookie(request: Request) -> str | None: return request.cookies.get(_real_cookie_name(COOKIE_NAME_PASSPORT + "-" + app_code)) @@ -103,6 +105,7 @@ def set_access_token_to_cookie(request: Request, response: Response, token: str, _real_cookie_name(COOKIE_NAME_ACCESS_TOKEN), value=token, httponly=True, + domain=_cookie_domain(), secure=is_secure(), samesite=samesite, max_age=int(dify_config.ACCESS_TOKEN_EXPIRE_MINUTES * 60), @@ -115,6 +118,7 @@ def set_refresh_token_to_cookie(request: Request, response: Response, token: str _real_cookie_name(COOKIE_NAME_REFRESH_TOKEN), value=token, httponly=True, + domain=_cookie_domain(), secure=is_secure(), samesite="Lax", max_age=int(60 * 60 * 24 * dify_config.REFRESH_TOKEN_EXPIRE_DAYS), @@ -127,6 +131,7 @@ def set_csrf_token_to_cookie(request: Request, response: Response, token: str): _real_cookie_name(COOKIE_NAME_CSRF_TOKEN), value=token, httponly=False, + domain=_cookie_domain(), secure=is_secure(), samesite="Lax", max_age=int(60 * dify_config.ACCESS_TOKEN_EXPIRE_MINUTES), @@ -145,6 +150,7 @@ def _clear_cookie( "", expires=0, path="/", + domain=_cookie_domain(), secure=is_secure(), httponly=http_only, samesite=samesite, @@ -155,6 +161,10 @@ def clear_access_token_from_cookie(response: Response, samesite: str = "Lax"): _clear_cookie(response, COOKIE_NAME_ACCESS_TOKEN, samesite) +def clear_webapp_access_token_from_cookie(response: Response, samesite: str = "Lax"): + _clear_cookie(response, COOKIE_NAME_WEBAPP_ACCESS_TOKEN, samesite) + + def clear_refresh_token_from_cookie(response: Response): _clear_cookie(response, COOKIE_NAME_REFRESH_TOKEN) @@ -163,6 +173,19 @@ def clear_csrf_token_from_cookie(response: Response): _clear_cookie(response, COOKIE_NAME_CSRF_TOKEN, http_only=False) +def build_force_logout_cookie_headers() -> list[str]: + """ + Generate Set-Cookie header values that clear all auth-related cookies. + This mirrors the behavior of the standard cookie clearing helpers while + allowing callers that do not have a Response instance to reuse the logic. + """ + response = Response() + clear_access_token_from_cookie(response) + clear_csrf_token_from_cookie(response) + clear_refresh_token_from_cookie(response) + return response.headers.getlist("Set-Cookie") + + def check_csrf_token(request: Request, user_id: str): # some apis are sent by beacon, so we need to bypass csrf token check # since these APIs are post, they are already protected by SameSite: Lax, so csrf is not required. diff --git a/api/migrations/versions/2025_10_22_1611-03f8dcbc611e_add_workflowpause_model.py b/api/migrations/versions/2025_10_22_1611-03f8dcbc611e_add_workflowpause_model.py new file mode 100644 index 0000000000..1ab4202674 --- /dev/null +++ b/api/migrations/versions/2025_10_22_1611-03f8dcbc611e_add_workflowpause_model.py @@ -0,0 +1,41 @@ +"""add WorkflowPause model + +Revision ID: 03f8dcbc611e +Revises: ae662b25d9bc +Create Date: 2025-10-22 16:11:31.805407 + +""" + +from alembic import op +import models as models +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = "03f8dcbc611e" +down_revision = "ae662b25d9bc" +branch_labels = None +depends_on = None + +def upgrade(): + # ### commands auto generated by Alembic - please adjust! ### + op.create_table( + "workflow_pauses", + sa.Column("workflow_id", models.types.StringUUID(), nullable=False), + sa.Column("workflow_run_id", models.types.StringUUID(), nullable=False), + sa.Column("resumed_at", sa.DateTime(), nullable=True), + sa.Column("state_object_key", sa.String(length=255), nullable=False), + sa.Column("id", models.types.StringUUID(), server_default=sa.text("uuidv7()"), nullable=False), + sa.Column("created_at", sa.DateTime(), server_default=sa.text("CURRENT_TIMESTAMP"), nullable=False), + sa.Column("updated_at", sa.DateTime(), server_default=sa.text("CURRENT_TIMESTAMP"), nullable=False), + sa.PrimaryKeyConstraint("id", name=op.f("workflow_pauses_pkey")), + sa.UniqueConstraint("workflow_run_id", name=op.f("workflow_pauses_workflow_run_id_key")), + ) + + # ### end Alembic commands ### + + +def downgrade(): + # ### commands auto generated by Alembic - please adjust! ### + op.drop_table("workflow_pauses") + # ### end Alembic commands ### diff --git a/api/migrations/versions/2025_10_30_1518-669ffd70119c_introduce_trigger.py b/api/migrations/versions/2025_10_30_1518-669ffd70119c_introduce_trigger.py new file mode 100644 index 0000000000..c03d64b234 --- /dev/null +++ b/api/migrations/versions/2025_10_30_1518-669ffd70119c_introduce_trigger.py @@ -0,0 +1,235 @@ +"""introduce_trigger + +Revision ID: 669ffd70119c +Revises: 03f8dcbc611e +Create Date: 2025-10-30 15:18:49.549156 + +""" +from alembic import op +import models as models +import sqlalchemy as sa + +from models.enums import AppTriggerStatus, AppTriggerType + + +# revision identifiers, used by Alembic. +revision = '669ffd70119c' +down_revision = '03f8dcbc611e' +branch_labels = None +depends_on = None + + +def upgrade(): + # ### commands auto generated by Alembic - please adjust! ### + op.create_table('app_triggers', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuidv7()'), nullable=False), + sa.Column('tenant_id', models.types.StringUUID(), nullable=False), + sa.Column('app_id', models.types.StringUUID(), nullable=False), + sa.Column('node_id', sa.String(length=64), nullable=False), + sa.Column('trigger_type', models.types.EnumText(AppTriggerType, length=50), nullable=False), + sa.Column('title', sa.String(length=255), nullable=False), + sa.Column('provider_name', sa.String(length=255), server_default='', nullable=True), + sa.Column('status', models.types.EnumText(AppTriggerStatus, length=50), nullable=False), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('updated_at', sa.DateTime(), nullable=False), + sa.PrimaryKeyConstraint('id', name='app_trigger_pkey') + ) + with op.batch_alter_table('app_triggers', schema=None) as batch_op: + batch_op.create_index('app_trigger_tenant_app_idx', ['tenant_id', 'app_id'], unique=False) + + op.create_table('trigger_oauth_system_clients', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuid_generate_v4()'), nullable=False), + sa.Column('plugin_id', sa.String(length=512), nullable=False), + sa.Column('provider', sa.String(length=255), nullable=False), + sa.Column('encrypted_oauth_params', sa.Text(), nullable=False), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('updated_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.PrimaryKeyConstraint('id', name='trigger_oauth_system_client_pkey'), + sa.UniqueConstraint('plugin_id', 'provider', name='trigger_oauth_system_client_plugin_id_provider_idx') + ) + op.create_table('trigger_oauth_tenant_clients', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuid_generate_v4()'), nullable=False), + sa.Column('tenant_id', models.types.StringUUID(), nullable=False), + sa.Column('plugin_id', sa.String(length=512), nullable=False), + sa.Column('provider', sa.String(length=255), nullable=False), + sa.Column('enabled', sa.Boolean(), server_default=sa.text('true'), nullable=False), + sa.Column('encrypted_oauth_params', sa.Text(), nullable=False), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('updated_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.PrimaryKeyConstraint('id', name='trigger_oauth_tenant_client_pkey'), + sa.UniqueConstraint('tenant_id', 'plugin_id', 'provider', name='unique_trigger_oauth_tenant_client') + ) + op.create_table('trigger_subscriptions', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuid_generate_v4()'), nullable=False), + sa.Column('name', sa.String(length=255), nullable=False, comment='Subscription instance name'), + sa.Column('tenant_id', models.types.StringUUID(), nullable=False), + sa.Column('user_id', models.types.StringUUID(), nullable=False), + sa.Column('provider_id', sa.String(length=255), nullable=False, comment='Provider identifier (e.g., plugin_id/provider_name)'), + sa.Column('endpoint_id', sa.String(length=255), nullable=False, comment='Subscription endpoint'), + sa.Column('parameters', sa.JSON(), nullable=False, comment='Subscription parameters JSON'), + sa.Column('properties', sa.JSON(), nullable=False, comment='Subscription properties JSON'), + sa.Column('credentials', sa.JSON(), nullable=False, comment='Subscription credentials JSON'), + sa.Column('credential_type', sa.String(length=50), nullable=False, comment='oauth or api_key'), + sa.Column('credential_expires_at', sa.Integer(), nullable=False, comment='OAuth token expiration timestamp, -1 for never'), + sa.Column('expires_at', sa.Integer(), nullable=False, comment='Subscription instance expiration timestamp, -1 for never'), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('updated_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.PrimaryKeyConstraint('id', name='trigger_provider_pkey'), + sa.UniqueConstraint('tenant_id', 'provider_id', 'name', name='unique_trigger_provider') + ) + with op.batch_alter_table('trigger_subscriptions', schema=None) as batch_op: + batch_op.create_index('idx_trigger_providers_endpoint', ['endpoint_id'], unique=True) + batch_op.create_index('idx_trigger_providers_tenant_endpoint', ['tenant_id', 'endpoint_id'], unique=False) + batch_op.create_index('idx_trigger_providers_tenant_provider', ['tenant_id', 'provider_id'], unique=False) + + op.create_table('workflow_plugin_triggers', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuid_generate_v4()'), nullable=False), + sa.Column('app_id', models.types.StringUUID(), nullable=False), + sa.Column('node_id', sa.String(length=64), nullable=False), + sa.Column('tenant_id', models.types.StringUUID(), nullable=False), + sa.Column('provider_id', sa.String(length=512), nullable=False), + sa.Column('event_name', sa.String(length=255), nullable=False), + sa.Column('subscription_id', sa.String(length=255), nullable=False), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('updated_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.PrimaryKeyConstraint('id', name='workflow_plugin_trigger_pkey'), + sa.UniqueConstraint('app_id', 'node_id', name='uniq_app_node_subscription') + ) + with op.batch_alter_table('workflow_plugin_triggers', schema=None) as batch_op: + batch_op.create_index('workflow_plugin_trigger_tenant_subscription_idx', ['tenant_id', 'subscription_id', 'event_name'], unique=False) + + op.create_table('workflow_schedule_plans', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuidv7()'), nullable=False), + sa.Column('app_id', models.types.StringUUID(), nullable=False), + sa.Column('node_id', sa.String(length=64), nullable=False), + sa.Column('tenant_id', models.types.StringUUID(), nullable=False), + sa.Column('cron_expression', sa.String(length=255), nullable=False), + sa.Column('timezone', sa.String(length=64), nullable=False), + sa.Column('next_run_at', sa.DateTime(), nullable=True), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('updated_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.PrimaryKeyConstraint('id', name='workflow_schedule_plan_pkey'), + sa.UniqueConstraint('app_id', 'node_id', name='uniq_app_node') + ) + with op.batch_alter_table('workflow_schedule_plans', schema=None) as batch_op: + batch_op.create_index('workflow_schedule_plan_next_idx', ['next_run_at'], unique=False) + + op.create_table('workflow_trigger_logs', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuidv7()'), nullable=False), + sa.Column('tenant_id', models.types.StringUUID(), nullable=False), + sa.Column('app_id', models.types.StringUUID(), nullable=False), + sa.Column('workflow_id', models.types.StringUUID(), nullable=False), + sa.Column('workflow_run_id', models.types.StringUUID(), nullable=True), + sa.Column('root_node_id', sa.String(length=255), nullable=True), + sa.Column('trigger_metadata', sa.Text(), nullable=False), + sa.Column('trigger_type', models.types.EnumText(AppTriggerType, length=50), nullable=False), + sa.Column('trigger_data', sa.Text(), nullable=False), + sa.Column('inputs', sa.Text(), nullable=False), + sa.Column('outputs', sa.Text(), nullable=True), + sa.Column('status', models.types.EnumText(AppTriggerStatus, length=50), nullable=False), + sa.Column('error', sa.Text(), nullable=True), + sa.Column('queue_name', sa.String(length=100), nullable=False), + sa.Column('celery_task_id', sa.String(length=255), nullable=True), + sa.Column('retry_count', sa.Integer(), nullable=False), + sa.Column('elapsed_time', sa.Float(), nullable=True), + sa.Column('total_tokens', sa.Integer(), nullable=True), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('created_by_role', sa.String(length=255), nullable=False), + sa.Column('created_by', sa.String(length=255), nullable=False), + sa.Column('triggered_at', sa.DateTime(), nullable=True), + sa.Column('finished_at', sa.DateTime(), nullable=True), + sa.PrimaryKeyConstraint('id', name='workflow_trigger_log_pkey') + ) + with op.batch_alter_table('workflow_trigger_logs', schema=None) as batch_op: + batch_op.create_index('workflow_trigger_log_created_at_idx', ['created_at'], unique=False) + batch_op.create_index('workflow_trigger_log_status_idx', ['status'], unique=False) + batch_op.create_index('workflow_trigger_log_tenant_app_idx', ['tenant_id', 'app_id'], unique=False) + batch_op.create_index('workflow_trigger_log_workflow_id_idx', ['workflow_id'], unique=False) + batch_op.create_index('workflow_trigger_log_workflow_run_idx', ['workflow_run_id'], unique=False) + + op.create_table('workflow_webhook_triggers', + sa.Column('id', models.types.StringUUID(), server_default=sa.text('uuidv7()'), nullable=False), + sa.Column('app_id', models.types.StringUUID(), nullable=False), + sa.Column('node_id', sa.String(length=64), nullable=False), + sa.Column('tenant_id', models.types.StringUUID(), nullable=False), + sa.Column('webhook_id', sa.String(length=24), nullable=False), + sa.Column('created_by', models.types.StringUUID(), nullable=False), + sa.Column('created_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.Column('updated_at', sa.DateTime(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False), + sa.PrimaryKeyConstraint('id', name='workflow_webhook_trigger_pkey'), + sa.UniqueConstraint('app_id', 'node_id', name='uniq_node'), + sa.UniqueConstraint('webhook_id', name='uniq_webhook_id') + ) + with op.batch_alter_table('workflow_webhook_triggers', schema=None) as batch_op: + batch_op.create_index('workflow_webhook_trigger_tenant_idx', ['tenant_id'], unique=False) + + with op.batch_alter_table('celery_taskmeta', schema=None) as batch_op: + batch_op.alter_column('task_id', + existing_type=sa.VARCHAR(length=155), + nullable=False) + batch_op.alter_column('status', + existing_type=sa.VARCHAR(length=50), + nullable=False) + + with op.batch_alter_table('celery_tasksetmeta', schema=None) as batch_op: + batch_op.alter_column('taskset_id', + existing_type=sa.VARCHAR(length=155), + nullable=False) + + with op.batch_alter_table('providers', schema=None) as batch_op: + batch_op.drop_column('credential_status') + + # ### end Alembic commands ### + + +def downgrade(): + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table('providers', schema=None) as batch_op: + batch_op.add_column(sa.Column('credential_status', sa.VARCHAR(length=20), server_default=sa.text("'active'::character varying"), autoincrement=False, nullable=True)) + + with op.batch_alter_table('celery_tasksetmeta', schema=None) as batch_op: + batch_op.alter_column('taskset_id', + existing_type=sa.VARCHAR(length=155), + nullable=True) + + with op.batch_alter_table('celery_taskmeta', schema=None) as batch_op: + batch_op.alter_column('status', + existing_type=sa.VARCHAR(length=50), + nullable=True) + batch_op.alter_column('task_id', + existing_type=sa.VARCHAR(length=155), + nullable=True) + + with op.batch_alter_table('workflow_webhook_triggers', schema=None) as batch_op: + batch_op.drop_index('workflow_webhook_trigger_tenant_idx') + + op.drop_table('workflow_webhook_triggers') + with op.batch_alter_table('workflow_trigger_logs', schema=None) as batch_op: + batch_op.drop_index('workflow_trigger_log_workflow_run_idx') + batch_op.drop_index('workflow_trigger_log_workflow_id_idx') + batch_op.drop_index('workflow_trigger_log_tenant_app_idx') + batch_op.drop_index('workflow_trigger_log_status_idx') + batch_op.drop_index('workflow_trigger_log_created_at_idx') + + op.drop_table('workflow_trigger_logs') + with op.batch_alter_table('workflow_schedule_plans', schema=None) as batch_op: + batch_op.drop_index('workflow_schedule_plan_next_idx') + + op.drop_table('workflow_schedule_plans') + with op.batch_alter_table('workflow_plugin_triggers', schema=None) as batch_op: + batch_op.drop_index('workflow_plugin_trigger_tenant_subscription_idx') + + op.drop_table('workflow_plugin_triggers') + with op.batch_alter_table('trigger_subscriptions', schema=None) as batch_op: + batch_op.drop_index('idx_trigger_providers_tenant_provider') + batch_op.drop_index('idx_trigger_providers_tenant_endpoint') + batch_op.drop_index('idx_trigger_providers_endpoint') + + op.drop_table('trigger_subscriptions') + op.drop_table('trigger_oauth_tenant_clients') + op.drop_table('trigger_oauth_system_clients') + with op.batch_alter_table('app_triggers', schema=None) as batch_op: + batch_op.drop_index('app_trigger_tenant_app_idx') + + op.drop_table('app_triggers') + # ### end Alembic commands ### diff --git a/api/migrations/versions/fecff1c3da27_remove_extra_tracing_app_config_table .py b/api/migrations/versions/fecff1c3da27_remove_extra_tracing_app_config_table.py similarity index 100% rename from api/migrations/versions/fecff1c3da27_remove_extra_tracing_app_config_table .py rename to api/migrations/versions/fecff1c3da27_remove_extra_tracing_app_config_table.py diff --git a/api/models/__init__.py b/api/models/__init__.py index 779484283f..906bc3198e 100644 --- a/api/models/__init__.py +++ b/api/models/__init__.py @@ -26,7 +26,14 @@ from .dataset import ( TidbAuthBinding, Whitelist, ) -from .enums import CreatorUserRole, UserFrom, WorkflowRunTriggeredFrom +from .enums import ( + AppTriggerStatus, + AppTriggerType, + CreatorUserRole, + UserFrom, + WorkflowRunTriggeredFrom, + WorkflowTriggerStatus, +) from .model import ( ApiRequest, ApiToken, @@ -79,6 +86,13 @@ from .tools import ( ToolModelInvoke, WorkflowToolProvider, ) +from .trigger import ( + AppTrigger, + TriggerOAuthSystemClient, + TriggerOAuthTenantClient, + TriggerSubscription, + WorkflowSchedulePlan, +) from .web import PinnedConversation, SavedMessage from .workflow import ( ConversationVariable, @@ -88,6 +102,7 @@ from .workflow import ( WorkflowNodeExecutionModel, WorkflowNodeExecutionOffload, WorkflowNodeExecutionTriggeredFrom, + WorkflowPause, WorkflowRun, WorkflowType, ) @@ -105,9 +120,12 @@ __all__ = [ "AppAnnotationHitHistory", "AppAnnotationSetting", "AppDatasetJoin", - "AppMCPServer", # Added + "AppMCPServer", "AppMode", "AppModelConfig", + "AppTrigger", + "AppTriggerStatus", + "AppTriggerType", "BuiltinToolProvider", "CeleryTask", "CeleryTaskSet", @@ -168,6 +186,9 @@ __all__ = [ "ToolLabelBinding", "ToolModelInvoke", "TraceAppConfig", + "TriggerOAuthSystemClient", + "TriggerOAuthTenantClient", + "TriggerSubscription", "UploadFile", "UserFrom", "Whitelist", @@ -177,8 +198,11 @@ __all__ = [ "WorkflowNodeExecutionModel", "WorkflowNodeExecutionOffload", "WorkflowNodeExecutionTriggeredFrom", + "WorkflowPause", "WorkflowRun", "WorkflowRunTriggeredFrom", + "WorkflowSchedulePlan", "WorkflowToolProvider", + "WorkflowTriggerStatus", "WorkflowType", ] diff --git a/api/models/account.py b/api/models/account.py index 400a2c6362..dc3f2094fd 100644 --- a/api/models/account.py +++ b/api/models/account.py @@ -110,7 +110,7 @@ class Account(UserMixin, TypeBase): DateTime, server_default=func.current_timestamp(), nullable=False, init=False ) updated_at: Mapped[datetime] = mapped_column( - DateTime, server_default=func.current_timestamp(), nullable=False, init=False + DateTime, server_default=func.current_timestamp(), nullable=False, init=False, onupdate=func.current_timestamp() ) role: TenantAccountRole | None = field(default=None, init=False) @@ -250,7 +250,9 @@ class Tenant(TypeBase): created_at: Mapped[datetime] = mapped_column( DateTime, server_default=func.current_timestamp(), nullable=False, init=False ) - updated_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.current_timestamp(), init=False) + updated_at: Mapped[datetime] = mapped_column( + DateTime, server_default=func.current_timestamp(), init=False, onupdate=func.current_timestamp() + ) def get_accounts(self) -> list[Account]: return list( @@ -289,7 +291,7 @@ class TenantAccountJoin(TypeBase): DateTime, server_default=func.current_timestamp(), nullable=False, init=False ) updated_at: Mapped[datetime] = mapped_column( - DateTime, server_default=func.current_timestamp(), nullable=False, init=False + DateTime, server_default=func.current_timestamp(), nullable=False, init=False, onupdate=func.current_timestamp() ) @@ -310,7 +312,7 @@ class AccountIntegrate(TypeBase): DateTime, server_default=func.current_timestamp(), nullable=False, init=False ) updated_at: Mapped[datetime] = mapped_column( - DateTime, server_default=func.current_timestamp(), nullable=False, init=False + DateTime, server_default=func.current_timestamp(), nullable=False, init=False, onupdate=func.current_timestamp() ) @@ -396,5 +398,5 @@ class TenantPluginAutoUpgradeStrategy(TypeBase): DateTime, nullable=False, server_default=func.current_timestamp(), init=False ) updated_at: Mapped[datetime] = mapped_column( - DateTime, nullable=False, server_default=func.current_timestamp(), init=False + DateTime, nullable=False, server_default=func.current_timestamp(), init=False, onupdate=func.current_timestamp() ) diff --git a/api/models/base.py b/api/models/base.py index 76848825fe..3660068035 100644 --- a/api/models/base.py +++ b/api/models/base.py @@ -1,6 +1,12 @@ -from sqlalchemy.orm import DeclarativeBase, MappedAsDataclass +from datetime import datetime +from sqlalchemy import DateTime, func, text +from sqlalchemy.orm import DeclarativeBase, Mapped, MappedAsDataclass, mapped_column + +from libs.datetime_utils import naive_utc_now +from libs.uuid_utils import uuidv7 from models.engine import metadata +from models.types import StringUUID class Base(DeclarativeBase): @@ -13,3 +19,34 @@ class TypeBase(MappedAsDataclass, DeclarativeBase): """ metadata = metadata + + +class DefaultFieldsMixin: + id: Mapped[str] = mapped_column( + StringUUID, + primary_key=True, + # NOTE: The default and server_default serve as fallback mechanisms. + # The application can generate the `id` before saving to optimize + # the insertion process (especially for interdependent models) + # and reduce database roundtrips. + default=uuidv7, + server_default=text("uuidv7()"), + ) + + created_at: Mapped[datetime] = mapped_column( + DateTime, + nullable=False, + default=naive_utc_now, + server_default=func.current_timestamp(), + ) + + updated_at: Mapped[datetime] = mapped_column( + __name_pos=DateTime, + nullable=False, + default=naive_utc_now, + server_default=func.current_timestamp(), + onupdate=func.current_timestamp(), + ) + + def __repr__(self) -> str: + return f"<{self.__class__.__name__}(id={self.id})>" diff --git a/api/models/dataset.py b/api/models/dataset.py index 4a9e2688b8..33d396aeb9 100644 --- a/api/models/dataset.py +++ b/api/models/dataset.py @@ -21,6 +21,7 @@ from configs import dify_config from core.rag.index_processor.constant.built_in_field import BuiltInField, MetadataDataSource from core.rag.retrieval.retrieval_methods import RetrievalMethod from extensions.ext_storage import storage +from models.base import TypeBase from services.entities.knowledge_entities.knowledge_entities import ParentMode, Rule from .account import Account @@ -61,18 +62,20 @@ class Dataset(Base): created_by = mapped_column(StringUUID, nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - embedding_model = mapped_column(db.String(255), nullable=True) - embedding_model_provider = mapped_column(db.String(255), nullable=True) - keyword_number = mapped_column(sa.Integer, nullable=True, server_default=db.text("10")) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) + embedding_model = mapped_column(sa.String(255), nullable=True) + embedding_model_provider = mapped_column(sa.String(255), nullable=True) + keyword_number = mapped_column(sa.Integer, nullable=True, server_default=sa.text("10")) collection_binding_id = mapped_column(StringUUID, nullable=True) retrieval_model = mapped_column(JSONB, nullable=True) - built_in_field_enabled = mapped_column(sa.Boolean, nullable=False, server_default=db.text("false")) + built_in_field_enabled = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("false")) icon_info = mapped_column(JSONB, nullable=True) - runtime_mode = mapped_column(db.String(255), nullable=True, server_default=db.text("'general'::character varying")) + runtime_mode = mapped_column(sa.String(255), nullable=True, server_default=sa.text("'general'::character varying")) pipeline_id = mapped_column(StringUUID, nullable=True) - chunk_structure = mapped_column(db.String(255), nullable=True) - enable_api = mapped_column(sa.Boolean, nullable=False, server_default=db.text("true")) + chunk_structure = mapped_column(sa.String(255), nullable=True) + enable_api = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("true")) @property def total_documents(self): @@ -399,7 +402,9 @@ class Document(Base): archived_reason = mapped_column(String(255), nullable=True) archived_by = mapped_column(StringUUID, nullable=True) archived_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) doc_type = mapped_column(String(40), nullable=True) doc_metadata = mapped_column(JSONB, nullable=True) doc_form = mapped_column(String(255), nullable=False, server_default=sa.text("'text_model'::character varying")) @@ -716,7 +721,9 @@ class DocumentSegment(Base): created_by = mapped_column(StringUUID, nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) indexing_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) completed_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) error = mapped_column(sa.Text, nullable=True) @@ -881,7 +888,7 @@ class ChildChunk(Base): ) updated_by = mapped_column(StringUUID, nullable=True) updated_at: Mapped[datetime] = mapped_column( - DateTime, nullable=False, server_default=sa.text("CURRENT_TIMESTAMP(0)") + DateTime, nullable=False, server_default=sa.text("CURRENT_TIMESTAMP(0)"), onupdate=func.current_timestamp() ) indexing_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) completed_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) @@ -900,17 +907,21 @@ class ChildChunk(Base): return db.session.query(DocumentSegment).where(DocumentSegment.id == self.segment_id).first() -class AppDatasetJoin(Base): +class AppDatasetJoin(TypeBase): __tablename__ = "app_dataset_joins" __table_args__ = ( sa.PrimaryKeyConstraint("id", name="app_dataset_join_pkey"), sa.Index("app_dataset_join_app_dataset_idx", "dataset_id", "app_id"), ) - id = mapped_column(StringUUID, primary_key=True, nullable=False, server_default=sa.text("uuid_generate_v4()")) - app_id = mapped_column(StringUUID, nullable=False) - dataset_id = mapped_column(StringUUID, nullable=False) - created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=sa.func.current_timestamp()) + id: Mapped[str] = mapped_column( + StringUUID, primary_key=True, nullable=False, server_default=sa.text("uuid_generate_v4()"), init=False + ) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + dataset_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + created_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=sa.func.current_timestamp(), init=False + ) @property def app(self): @@ -1036,8 +1047,8 @@ class TidbAuthBinding(Base): tenant_id = mapped_column(StringUUID, nullable=True) cluster_id: Mapped[str] = mapped_column(String(255), nullable=False) cluster_name: Mapped[str] = mapped_column(String(255), nullable=False) - active: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=db.text("false")) - status = mapped_column(String(255), nullable=False, server_default=db.text("'CREATING'::character varying")) + active: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("false")) + status = mapped_column(String(255), nullable=False, server_default=sa.text("'CREATING'::character varying")) account: Mapped[str] = mapped_column(String(255), nullable=False) password: Mapped[str] = mapped_column(String(255), nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) @@ -1088,7 +1099,9 @@ class ExternalKnowledgeApis(Base): created_by = mapped_column(StringUUID, nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) def to_dict(self) -> dict[str, Any]: return { @@ -1141,7 +1154,9 @@ class ExternalKnowledgeBindings(Base): created_by = mapped_column(StringUUID, nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class DatasetAutoDisableLog(Base): @@ -1197,7 +1212,7 @@ class DatasetMetadata(Base): DateTime, nullable=False, server_default=sa.text("CURRENT_TIMESTAMP(0)") ) updated_at: Mapped[datetime] = mapped_column( - DateTime, nullable=False, server_default=sa.text("CURRENT_TIMESTAMP(0)") + DateTime, nullable=False, server_default=sa.text("CURRENT_TIMESTAMP(0)"), onupdate=func.current_timestamp() ) created_by = mapped_column(StringUUID, nullable=False) updated_by = mapped_column(StringUUID, nullable=True) @@ -1224,44 +1239,48 @@ class DatasetMetadataBinding(Base): class PipelineBuiltInTemplate(Base): # type: ignore[name-defined] __tablename__ = "pipeline_built_in_templates" - __table_args__ = (db.PrimaryKeyConstraint("id", name="pipeline_built_in_template_pkey"),) + __table_args__ = (sa.PrimaryKeyConstraint("id", name="pipeline_built_in_template_pkey"),) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) - name = mapped_column(db.String(255), nullable=False) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) + name = mapped_column(sa.String(255), nullable=False) description = mapped_column(sa.Text, nullable=False) - chunk_structure = mapped_column(db.String(255), nullable=False) + chunk_structure = mapped_column(sa.String(255), nullable=False) icon = mapped_column(sa.JSON, nullable=False) yaml_content = mapped_column(sa.Text, nullable=False) - copyright = mapped_column(db.String(255), nullable=False) - privacy_policy = mapped_column(db.String(255), nullable=False) + copyright = mapped_column(sa.String(255), nullable=False) + privacy_policy = mapped_column(sa.String(255), nullable=False) position = mapped_column(sa.Integer, nullable=False) install_count = mapped_column(sa.Integer, nullable=False, default=0) - language = mapped_column(db.String(255), nullable=False) + language = mapped_column(sa.String(255), nullable=False) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class PipelineCustomizedTemplate(Base): # type: ignore[name-defined] __tablename__ = "pipeline_customized_templates" __table_args__ = ( - db.PrimaryKeyConstraint("id", name="pipeline_customized_template_pkey"), - db.Index("pipeline_customized_template_tenant_idx", "tenant_id"), + sa.PrimaryKeyConstraint("id", name="pipeline_customized_template_pkey"), + sa.Index("pipeline_customized_template_tenant_idx", "tenant_id"), ) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) tenant_id = mapped_column(StringUUID, nullable=False) - name = mapped_column(db.String(255), nullable=False) + name = mapped_column(sa.String(255), nullable=False) description = mapped_column(sa.Text, nullable=False) - chunk_structure = mapped_column(db.String(255), nullable=False) + chunk_structure = mapped_column(sa.String(255), nullable=False) icon = mapped_column(sa.JSON, nullable=False) position = mapped_column(sa.Integer, nullable=False) yaml_content = mapped_column(sa.Text, nullable=False) install_count = mapped_column(sa.Integer, nullable=False, default=0) - language = mapped_column(db.String(255), nullable=False) + language = mapped_column(sa.String(255), nullable=False) created_by = mapped_column(StringUUID, nullable=False) updated_by = mapped_column(StringUUID, nullable=True) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) @property def created_user_name(self): @@ -1273,19 +1292,21 @@ class PipelineCustomizedTemplate(Base): # type: ignore[name-defined] class Pipeline(Base): # type: ignore[name-defined] __tablename__ = "pipelines" - __table_args__ = (db.PrimaryKeyConstraint("id", name="pipeline_pkey"),) + __table_args__ = (sa.PrimaryKeyConstraint("id", name="pipeline_pkey"),) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) - name = mapped_column(db.String(255), nullable=False) - description = mapped_column(sa.Text, nullable=False, server_default=db.text("''::character varying")) + name = mapped_column(sa.String(255), nullable=False) + description = mapped_column(sa.Text, nullable=False, server_default=sa.text("''::character varying")) workflow_id = mapped_column(StringUUID, nullable=True) - is_public = mapped_column(sa.Boolean, nullable=False, server_default=db.text("false")) - is_published = mapped_column(sa.Boolean, nullable=False, server_default=db.text("false")) + is_public = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("false")) + is_published = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("false")) created_by = mapped_column(StringUUID, nullable=True) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) def retrieve_dataset(self, session: Session): return session.query(Dataset).where(Dataset.pipeline_id == self.id).first() @@ -1294,16 +1315,16 @@ class Pipeline(Base): # type: ignore[name-defined] class DocumentPipelineExecutionLog(Base): __tablename__ = "document_pipeline_execution_logs" __table_args__ = ( - db.PrimaryKeyConstraint("id", name="document_pipeline_execution_log_pkey"), - db.Index("document_pipeline_execution_logs_document_id_idx", "document_id"), + sa.PrimaryKeyConstraint("id", name="document_pipeline_execution_log_pkey"), + sa.Index("document_pipeline_execution_logs_document_id_idx", "document_id"), ) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) pipeline_id = mapped_column(StringUUID, nullable=False) document_id = mapped_column(StringUUID, nullable=False) - datasource_type = mapped_column(db.String(255), nullable=False) + datasource_type = mapped_column(sa.String(255), nullable=False) datasource_info = mapped_column(sa.Text, nullable=False) - datasource_node_id = mapped_column(db.String(255), nullable=False) + datasource_node_id = mapped_column(sa.String(255), nullable=False) input_data = mapped_column(sa.JSON, nullable=False) created_by = mapped_column(StringUUID, nullable=True) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) @@ -1311,12 +1332,14 @@ class DocumentPipelineExecutionLog(Base): class PipelineRecommendedPlugin(Base): __tablename__ = "pipeline_recommended_plugins" - __table_args__ = (db.PrimaryKeyConstraint("id", name="pipeline_recommended_plugin_pkey"),) + __table_args__ = (sa.PrimaryKeyConstraint("id", name="pipeline_recommended_plugin_pkey"),) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) plugin_id = mapped_column(sa.Text, nullable=False) provider_name = mapped_column(sa.Text, nullable=False) position = mapped_column(sa.Integer, nullable=False, default=0) active = mapped_column(sa.Boolean, nullable=False, default=True) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) diff --git a/api/models/enums.py b/api/models/enums.py index 0be7567c80..d06d0d5ebc 100644 --- a/api/models/enums.py +++ b/api/models/enums.py @@ -1,5 +1,7 @@ from enum import StrEnum +from core.workflow.enums import NodeType + class CreatorUserRole(StrEnum): ACCOUNT = "account" @@ -13,9 +15,12 @@ class UserFrom(StrEnum): class WorkflowRunTriggeredFrom(StrEnum): DEBUGGING = "debugging" - APP_RUN = "app-run" + APP_RUN = "app-run" # webapp / service api RAG_PIPELINE_RUN = "rag-pipeline-run" RAG_PIPELINE_DEBUGGING = "rag-pipeline-debugging" + WEBHOOK = "webhook" + SCHEDULE = "schedule" + PLUGIN = "plugin" class DraftVariableType(StrEnum): @@ -38,3 +43,35 @@ class ExecutionOffLoadType(StrEnum): INPUTS = "inputs" PROCESS_DATA = "process_data" OUTPUTS = "outputs" + + +class WorkflowTriggerStatus(StrEnum): + """Workflow Trigger Execution Status""" + + PENDING = "pending" + QUEUED = "queued" + RUNNING = "running" + SUCCEEDED = "succeeded" + PAUSED = "paused" + FAILED = "failed" + RATE_LIMITED = "rate_limited" + RETRYING = "retrying" + + +class AppTriggerStatus(StrEnum): + """App Trigger Status Enum""" + + ENABLED = "enabled" + DISABLED = "disabled" + UNAUTHORIZED = "unauthorized" + + +class AppTriggerType(StrEnum): + """App Trigger Type Enum""" + + TRIGGER_WEBHOOK = NodeType.TRIGGER_WEBHOOK.value + TRIGGER_SCHEDULE = NodeType.TRIGGER_SCHEDULE.value + TRIGGER_PLUGIN = NodeType.TRIGGER_PLUGIN.value + + # for backward compatibility + UNKNOWN = "unknown" diff --git a/api/models/model.py b/api/models/model.py index 8a8574e2fe..f698b79d32 100644 --- a/api/models/model.py +++ b/api/models/model.py @@ -3,6 +3,7 @@ import re import uuid from collections.abc import Mapping from datetime import datetime +from decimal import Decimal from enum import StrEnum, auto from typing import TYPE_CHECKING, Any, Literal, Optional, cast @@ -94,7 +95,9 @@ class App(Base): created_by = mapped_column(StringUUID, nullable=True) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) use_icon_as_answer_icon: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("false")) @property @@ -313,7 +316,9 @@ class AppModelConfig(Base): created_by = mapped_column(StringUUID, nullable=True) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) opening_statement = mapped_column(sa.Text) suggested_questions = mapped_column(sa.Text) suggested_questions_after_answer = mapped_column(sa.Text) @@ -544,7 +549,9 @@ class RecommendedApp(Base): install_count: Mapped[int] = mapped_column(sa.Integer, nullable=False, default=0) language = mapped_column(String(255), nullable=False, server_default=sa.text("'en-US'::character varying")) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) @property def app(self) -> App | None: @@ -643,7 +650,9 @@ class Conversation(Base): read_account_id = mapped_column(StringUUID) dialogue_count: Mapped[int] = mapped_column(default=0) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) messages = db.relationship("Message", backref="conversation", lazy="select", passive_deletes="all") message_annotations = db.relationship( @@ -914,34 +923,42 @@ class Message(Base): ) id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuid_generate_v4()")) - app_id = mapped_column(StringUUID, nullable=False) - model_provider = mapped_column(String(255), nullable=True) - model_id = mapped_column(String(255), nullable=True) - override_model_configs = mapped_column(sa.Text) - conversation_id = mapped_column(StringUUID, sa.ForeignKey("conversations.id"), nullable=False) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + model_provider: Mapped[str | None] = mapped_column(String(255), nullable=True) + model_id: Mapped[str | None] = mapped_column(String(255), nullable=True) + override_model_configs: Mapped[str | None] = mapped_column(sa.Text) + conversation_id: Mapped[str] = mapped_column(StringUUID, sa.ForeignKey("conversations.id"), nullable=False) _inputs: Mapped[dict[str, Any]] = mapped_column("inputs", sa.JSON) query: Mapped[str] = mapped_column(sa.Text, nullable=False) - message = mapped_column(sa.JSON, nullable=False) + message: Mapped[dict[str, Any]] = mapped_column(sa.JSON, nullable=False) message_tokens: Mapped[int] = mapped_column(sa.Integer, nullable=False, server_default=sa.text("0")) - message_unit_price = mapped_column(sa.Numeric(10, 4), nullable=False) - message_price_unit = mapped_column(sa.Numeric(10, 7), nullable=False, server_default=sa.text("0.001")) + message_unit_price: Mapped[Decimal] = mapped_column(sa.Numeric(10, 4), nullable=False) + message_price_unit: Mapped[Decimal] = mapped_column( + sa.Numeric(10, 7), nullable=False, server_default=sa.text("0.001") + ) answer: Mapped[str] = mapped_column(sa.Text, nullable=False) answer_tokens: Mapped[int] = mapped_column(sa.Integer, nullable=False, server_default=sa.text("0")) - answer_unit_price = mapped_column(sa.Numeric(10, 4), nullable=False) - answer_price_unit = mapped_column(sa.Numeric(10, 7), nullable=False, server_default=sa.text("0.001")) - parent_message_id = mapped_column(StringUUID, nullable=True) - provider_response_latency = mapped_column(sa.Float, nullable=False, server_default=sa.text("0")) - total_price = mapped_column(sa.Numeric(10, 7)) + answer_unit_price: Mapped[Decimal] = mapped_column(sa.Numeric(10, 4), nullable=False) + answer_price_unit: Mapped[Decimal] = mapped_column( + sa.Numeric(10, 7), nullable=False, server_default=sa.text("0.001") + ) + parent_message_id: Mapped[str | None] = mapped_column(StringUUID, nullable=True) + provider_response_latency: Mapped[float] = mapped_column(sa.Float, nullable=False, server_default=sa.text("0")) + total_price: Mapped[Decimal | None] = mapped_column(sa.Numeric(10, 7)) currency: Mapped[str] = mapped_column(String(255), nullable=False) - status = mapped_column(String(255), nullable=False, server_default=sa.text("'normal'::character varying")) - error = mapped_column(sa.Text) - message_metadata = mapped_column(sa.Text) + status: Mapped[str] = mapped_column( + String(255), nullable=False, server_default=sa.text("'normal'::character varying") + ) + error: Mapped[str | None] = mapped_column(sa.Text) + message_metadata: Mapped[str | None] = mapped_column(sa.Text) invoke_from: Mapped[str | None] = mapped_column(String(255), nullable=True) from_source: Mapped[str] = mapped_column(String(255), nullable=False) from_end_user_id: Mapped[str | None] = mapped_column(StringUUID) from_account_id: Mapped[str | None] = mapped_column(StringUUID) created_at: Mapped[datetime] = mapped_column(sa.DateTime, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) agent_based: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("false")) workflow_run_id: Mapped[str | None] = mapped_column(StringUUID) app_mode: Mapped[str | None] = mapped_column(String(255), nullable=True) @@ -1212,9 +1229,13 @@ class Message(Base): @property def workflow_run(self): if self.workflow_run_id: - from .workflow import WorkflowRun + from sqlalchemy.orm import sessionmaker - return db.session.query(WorkflowRun).where(WorkflowRun.id == self.workflow_run_id).first() + from repositories.factory import DifyAPIRepositoryFactory + + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + return repo.get_workflow_run_by_id_without_tenant(run_id=self.workflow_run_id) return None @@ -1275,20 +1296,22 @@ class MessageFeedback(Base): sa.Index("message_feedback_conversation_idx", "conversation_id", "from_source", "rating"), ) - id = mapped_column(StringUUID, server_default=sa.text("uuid_generate_v4()")) - app_id = mapped_column(StringUUID, nullable=False) - conversation_id = mapped_column(StringUUID, nullable=False) - message_id = mapped_column(StringUUID, nullable=False) + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuid_generate_v4()")) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + conversation_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + message_id: Mapped[str] = mapped_column(StringUUID, nullable=False) rating: Mapped[str] = mapped_column(String(255), nullable=False) - content = mapped_column(sa.Text) + content: Mapped[str | None] = mapped_column(sa.Text) from_source: Mapped[str] = mapped_column(String(255), nullable=False) - from_end_user_id = mapped_column(StringUUID) - from_account_id = mapped_column(StringUUID) - created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + from_end_user_id: Mapped[str | None] = mapped_column(StringUUID) + from_account_id: Mapped[str | None] = mapped_column(StringUUID) + created_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) @property - def from_account(self): + def from_account(self) -> Account | None: account = db.session.query(Account).where(Account.id == self.from_account_id).first() return account @@ -1367,7 +1390,9 @@ class MessageAnnotation(Base): hit_count: Mapped[int] = mapped_column(sa.Integer, nullable=False, server_default=sa.text("0")) account_id = mapped_column(StringUUID, nullable=False) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) @property def account(self): @@ -1432,7 +1457,9 @@ class AppAnnotationSetting(Base): created_user_id = mapped_column(StringUUID, nullable=False) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) updated_user_id = mapped_column(StringUUID, nullable=False) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) @property def collection_binding_detail(self): @@ -1460,7 +1487,9 @@ class OperationLog(Base): content = mapped_column(sa.JSON) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) created_ip: Mapped[str] = mapped_column(String(255), nullable=False) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class DefaultEndUserSessionID(StrEnum): @@ -1499,7 +1528,9 @@ class EndUser(Base, UserMixin): session_id: Mapped[str] = mapped_column() created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class AppMCPServer(Base): @@ -1519,7 +1550,9 @@ class AppMCPServer(Base): parameters = mapped_column(sa.Text, nullable=False) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) @staticmethod def generate_server_code(n: int) -> str: @@ -1565,7 +1598,9 @@ class Site(Base): created_by = mapped_column(StringUUID, nullable=True) created_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) updated_by = mapped_column(StringUUID, nullable=True) - updated_at = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) code = mapped_column(String(255)) @property diff --git a/api/models/oauth.py b/api/models/oauth.py index ef23780dc8..e705b3d189 100644 --- a/api/models/oauth.py +++ b/api/models/oauth.py @@ -1,62 +1,66 @@ from datetime import datetime import sqlalchemy as sa +from sqlalchemy import func from sqlalchemy.dialects.postgresql import JSONB from sqlalchemy.orm import Mapped, mapped_column from .base import Base -from .engine import db from .types import StringUUID class DatasourceOauthParamConfig(Base): # type: ignore[name-defined] __tablename__ = "datasource_oauth_params" __table_args__ = ( - db.PrimaryKeyConstraint("id", name="datasource_oauth_config_pkey"), - db.UniqueConstraint("plugin_id", "provider", name="datasource_oauth_config_datasource_id_provider_idx"), + sa.PrimaryKeyConstraint("id", name="datasource_oauth_config_pkey"), + sa.UniqueConstraint("plugin_id", "provider", name="datasource_oauth_config_datasource_id_provider_idx"), ) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) - plugin_id: Mapped[str] = mapped_column(db.String(255), nullable=False) - provider: Mapped[str] = mapped_column(db.String(255), nullable=False) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) + plugin_id: Mapped[str] = mapped_column(sa.String(255), nullable=False) + provider: Mapped[str] = mapped_column(sa.String(255), nullable=False) system_credentials: Mapped[dict] = mapped_column(JSONB, nullable=False) class DatasourceProvider(Base): __tablename__ = "datasource_providers" __table_args__ = ( - db.PrimaryKeyConstraint("id", name="datasource_provider_pkey"), - db.UniqueConstraint("tenant_id", "plugin_id", "provider", "name", name="datasource_provider_unique_name"), - db.Index("datasource_provider_auth_type_provider_idx", "tenant_id", "plugin_id", "provider"), + sa.PrimaryKeyConstraint("id", name="datasource_provider_pkey"), + sa.UniqueConstraint("tenant_id", "plugin_id", "provider", "name", name="datasource_provider_unique_name"), + sa.Index("datasource_provider_auth_type_provider_idx", "tenant_id", "plugin_id", "provider"), ) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) tenant_id = mapped_column(StringUUID, nullable=False) - name: Mapped[str] = mapped_column(db.String(255), nullable=False) - provider: Mapped[str] = mapped_column(db.String(255), nullable=False) - plugin_id: Mapped[str] = mapped_column(db.String(255), nullable=False) - auth_type: Mapped[str] = mapped_column(db.String(255), nullable=False) + name: Mapped[str] = mapped_column(sa.String(255), nullable=False) + provider: Mapped[str] = mapped_column(sa.String(255), nullable=False) + plugin_id: Mapped[str] = mapped_column(sa.String(255), nullable=False) + auth_type: Mapped[str] = mapped_column(sa.String(255), nullable=False) encrypted_credentials: Mapped[dict] = mapped_column(JSONB, nullable=False) avatar_url: Mapped[str] = mapped_column(sa.Text, nullable=True, default="default") - is_default: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=db.text("false")) + is_default: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("false")) expires_at: Mapped[int] = mapped_column(sa.Integer, nullable=False, server_default="-1") - created_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, default=datetime.now) - updated_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, default=datetime.now) + created_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class DatasourceOauthTenantParamConfig(Base): __tablename__ = "datasource_oauth_tenant_params" __table_args__ = ( - db.PrimaryKeyConstraint("id", name="datasource_oauth_tenant_config_pkey"), - db.UniqueConstraint("tenant_id", "plugin_id", "provider", name="datasource_oauth_tenant_config_unique"), + sa.PrimaryKeyConstraint("id", name="datasource_oauth_tenant_config_pkey"), + sa.UniqueConstraint("tenant_id", "plugin_id", "provider", name="datasource_oauth_tenant_config_unique"), ) - id = mapped_column(StringUUID, server_default=db.text("uuidv7()")) + id = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) tenant_id = mapped_column(StringUUID, nullable=False) - provider: Mapped[str] = mapped_column(db.String(255), nullable=False) - plugin_id: Mapped[str] = mapped_column(db.String(255), nullable=False) + provider: Mapped[str] = mapped_column(sa.String(255), nullable=False) + plugin_id: Mapped[str] = mapped_column(sa.String(255), nullable=False) client_params: Mapped[dict] = mapped_column(JSONB, nullable=False, default={}) enabled: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, default=False) - created_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, default=datetime.now) - updated_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, default=datetime.now) + created_at: Mapped[datetime] = mapped_column(sa.DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + sa.DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) diff --git a/api/models/provider.py b/api/models/provider.py index f6852d49f4..4de17a7fd5 100644 --- a/api/models/provider.py +++ b/api/models/provider.py @@ -6,7 +6,7 @@ import sqlalchemy as sa from sqlalchemy import DateTime, String, func, text from sqlalchemy.orm import Mapped, mapped_column -from .base import Base +from .base import Base, TypeBase from .engine import db from .types import StringUUID @@ -41,7 +41,7 @@ class ProviderQuotaType(StrEnum): raise ValueError(f"No matching enum found for value '{value}'") -class Provider(Base): +class Provider(TypeBase): """ Provider model representing the API providers and their configurations. """ @@ -55,24 +55,28 @@ class Provider(Base): ), ) - id: Mapped[str] = mapped_column(StringUUID, server_default=text("uuid_generate_v4()")) + id: Mapped[str] = mapped_column(StringUUID, primary_key=True, server_default=text("uuidv7()"), init=False) tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) provider_name: Mapped[str] = mapped_column(String(255), nullable=False) provider_type: Mapped[str] = mapped_column( - String(40), nullable=False, server_default=text("'custom'::character varying") + String(40), nullable=False, server_default=text("'custom'::character varying"), default="custom" ) - is_valid: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=text("false")) - last_used: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) - credential_id: Mapped[str | None] = mapped_column(StringUUID, nullable=True) + is_valid: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=text("false"), default=False) + last_used: Mapped[datetime | None] = mapped_column(DateTime, nullable=True, init=False) + credential_id: Mapped[str | None] = mapped_column(StringUUID, nullable=True, default=None) quota_type: Mapped[str | None] = mapped_column( - String(40), nullable=True, server_default=text("''::character varying") + String(40), nullable=True, server_default=text("''::character varying"), default="" ) - quota_limit: Mapped[int | None] = mapped_column(sa.BigInteger, nullable=True) - quota_used: Mapped[int | None] = mapped_column(sa.BigInteger, default=0) + quota_limit: Mapped[int | None] = mapped_column(sa.BigInteger, nullable=True, default=None) + quota_used: Mapped[int] = mapped_column(sa.BigInteger, nullable=False, default=0) - created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + created_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), init=False + ) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp(), init=False + ) def __repr__(self): return ( @@ -135,7 +139,9 @@ class ProviderModel(Base): credential_id: Mapped[str | None] = mapped_column(StringUUID, nullable=True) is_valid: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=text("false")) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) @cached_property def credential(self): @@ -170,7 +176,9 @@ class TenantDefaultModel(Base): model_name: Mapped[str] = mapped_column(String(255), nullable=False) model_type: Mapped[str] = mapped_column(String(40), nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class TenantPreferredModelProvider(Base): @@ -185,7 +193,9 @@ class TenantPreferredModelProvider(Base): provider_name: Mapped[str] = mapped_column(String(255), nullable=False) preferred_provider_type: Mapped[str] = mapped_column(String(40), nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class ProviderOrder(Base): @@ -212,7 +222,9 @@ class ProviderOrder(Base): pay_failed_at: Mapped[datetime | None] = mapped_column(DateTime) refunded_at: Mapped[datetime | None] = mapped_column(DateTime) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class ProviderModelSetting(Base): @@ -234,7 +246,9 @@ class ProviderModelSetting(Base): enabled: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=text("true")) load_balancing_enabled: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=text("false")) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class LoadBalancingModelConfig(Base): @@ -259,7 +273,9 @@ class LoadBalancingModelConfig(Base): credential_source_type: Mapped[str | None] = mapped_column(String(40), nullable=True) enabled: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=text("true")) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class ProviderCredential(Base): @@ -279,7 +295,9 @@ class ProviderCredential(Base): credential_name: Mapped[str] = mapped_column(String(255), nullable=False) encrypted_config: Mapped[str] = mapped_column(sa.Text, nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) class ProviderModelCredential(Base): @@ -307,4 +325,6 @@ class ProviderModelCredential(Base): credential_name: Mapped[str] = mapped_column(String(255), nullable=False) encrypted_config: Mapped[str] = mapped_column(sa.Text, nullable=False) created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) - updated_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) diff --git a/api/models/provider_ids.py b/api/models/provider_ids.py index 98dc67f2f3..0be6a3dc98 100644 --- a/api/models/provider_ids.py +++ b/api/models/provider_ids.py @@ -57,3 +57,8 @@ class ToolProviderID(GenericProviderID): class DatasourceProviderID(GenericProviderID): def __init__(self, value: str, is_hardcoded: bool = False) -> None: super().__init__(value, is_hardcoded) + + +class TriggerProviderID(GenericProviderID): + def __init__(self, value: str, is_hardcoded: bool = False) -> None: + super().__init__(value, is_hardcoded) diff --git a/api/models/tools.py b/api/models/tools.py index 4e2976ce1b..12acc149b1 100644 --- a/api/models/tools.py +++ b/api/models/tools.py @@ -1,16 +1,13 @@ import json -from collections.abc import Mapping from datetime import datetime from decimal import Decimal from typing import TYPE_CHECKING, Any, cast -from urllib.parse import urlparse import sqlalchemy as sa from deprecated import deprecated from sqlalchemy import ForeignKey, String, func from sqlalchemy.orm import Mapped, mapped_column -from core.helper import encrypter from core.tools.entities.common_entities import I18nObject from core.tools.entities.tool_bundle import ApiToolBundle from core.tools.entities.tool_entities import ApiProviderSchemaType, WorkflowToolParameterConfiguration @@ -21,7 +18,7 @@ from .model import Account, App, Tenant from .types import StringUUID if TYPE_CHECKING: - from core.mcp.types import Tool as MCPTool + from core.entities.mcp_provider import MCPProviderEntity from core.tools.entities.common_entities import I18nObject from core.tools.entities.tool_bundle import ApiToolBundle from core.tools.entities.tool_entities import ApiProviderSchemaType, WorkflowToolParameterConfiguration @@ -331,126 +328,36 @@ class MCPToolProvider(TypeBase): def load_user(self) -> Account | None: return db.session.query(Account).where(Account.id == self.user_id).first() - @property - def tenant(self) -> Tenant | None: - return db.session.query(Tenant).where(Tenant.id == self.tenant_id).first() - @property def credentials(self) -> dict[str, Any]: if not self.encrypted_credentials: return {} try: - return cast(dict[str, Any], json.loads(self.encrypted_credentials)) or {} - except json.JSONDecodeError: - return {} - - @property - def mcp_tools(self) -> list["MCPTool"]: - from core.mcp.types import Tool as MCPTool - - return [MCPTool.model_validate(tool) for tool in json.loads(self.tools)] - - @property - def provider_icon(self) -> Mapping[str, str] | str: - from core.file import helpers as file_helpers - - assert self.icon - try: - return json.loads(self.icon) - except json.JSONDecodeError: - return file_helpers.get_signed_file_url(self.icon) - - @property - def decrypted_server_url(self) -> str: - return encrypter.decrypt_token(self.tenant_id, self.server_url) - - @property - def decrypted_headers(self) -> dict[str, Any]: - """Get decrypted headers for MCP server requests.""" - from core.entities.provider_entities import BasicProviderConfig - from core.helper.provider_cache import NoOpProviderCredentialCache - from core.tools.utils.encryption import create_provider_encrypter - - try: - if not self.encrypted_headers: - return {} - - headers_data = json.loads(self.encrypted_headers) - - # Create dynamic config for all headers as SECRET_INPUT - config = [BasicProviderConfig(type=BasicProviderConfig.Type.SECRET_INPUT, name=key) for key in headers_data] - - encrypter_instance, _ = create_provider_encrypter( - tenant_id=self.tenant_id, - config=config, - cache=NoOpProviderCredentialCache(), - ) - - result = encrypter_instance.decrypt(headers_data) - return result + return json.loads(self.encrypted_credentials) except Exception: return {} @property - def masked_headers(self) -> dict[str, Any]: - """Get masked headers for frontend display.""" - from core.entities.provider_entities import BasicProviderConfig - from core.helper.provider_cache import NoOpProviderCredentialCache - from core.tools.utils.encryption import create_provider_encrypter - + def headers(self) -> dict[str, Any]: + if self.encrypted_headers is None: + return {} try: - if not self.encrypted_headers: - return {} - - headers_data = json.loads(self.encrypted_headers) - - # Create dynamic config for all headers as SECRET_INPUT - config = [BasicProviderConfig(type=BasicProviderConfig.Type.SECRET_INPUT, name=key) for key in headers_data] - - encrypter_instance, _ = create_provider_encrypter( - tenant_id=self.tenant_id, - config=config, - cache=NoOpProviderCredentialCache(), - ) - - # First decrypt, then mask - decrypted_headers = encrypter_instance.decrypt(headers_data) - result = encrypter_instance.mask_tool_credentials(decrypted_headers) - return result + return json.loads(self.encrypted_headers) except Exception: return {} @property - def masked_server_url(self) -> str: - def mask_url(url: str, mask_char: str = "*") -> str: - """ - mask the url to a simple string - """ - parsed = urlparse(url) - base_url = f"{parsed.scheme}://{parsed.netloc}" + def tool_dict(self) -> list[dict[str, Any]]: + try: + return json.loads(self.tools) if self.tools else [] + except (json.JSONDecodeError, TypeError): + return [] - if parsed.path and parsed.path != "/": - return f"{base_url}/{mask_char * 6}" - else: - return base_url + def to_entity(self) -> "MCPProviderEntity": + """Convert to domain entity""" + from core.entities.mcp_provider import MCPProviderEntity - return mask_url(self.decrypted_server_url) - - @property - def decrypted_credentials(self) -> dict[str, Any]: - from core.helper.provider_cache import NoOpProviderCredentialCache - from core.tools.mcp_tool.provider import MCPToolProviderController - from core.tools.utils.encryption import create_provider_encrypter - - provider_controller = MCPToolProviderController.from_db(self) - - encrypter, _ = create_provider_encrypter( - tenant_id=self.tenant_id, - config=[x.to_basic_provider_config() for x in provider_controller.get_credentials_schema()], - cache=NoOpProviderCredentialCache(), - ) - - return encrypter.decrypt(self.credentials) + return MCPProviderEntity.from_db_model(self) class ToolModelInvoke(TypeBase): diff --git a/api/models/trigger.py b/api/models/trigger.py new file mode 100644 index 0000000000..c2b66ace46 --- /dev/null +++ b/api/models/trigger.py @@ -0,0 +1,456 @@ +import json +import time +from collections.abc import Mapping +from datetime import datetime +from functools import cached_property +from typing import Any, cast + +import sqlalchemy as sa +from sqlalchemy import DateTime, Index, Integer, String, UniqueConstraint, func +from sqlalchemy.orm import Mapped, mapped_column + +from core.plugin.entities.plugin_daemon import CredentialType +from core.trigger.entities.api_entities import TriggerProviderSubscriptionApiEntity +from core.trigger.entities.entities import Subscription +from core.trigger.utils.endpoint import generate_plugin_trigger_endpoint_url, generate_webhook_trigger_endpoint +from libs.datetime_utils import naive_utc_now +from models.base import Base +from models.engine import db +from models.enums import AppTriggerStatus, AppTriggerType, CreatorUserRole, WorkflowTriggerStatus +from models.model import Account +from models.types import EnumText, StringUUID + + +class TriggerSubscription(Base): + """ + Trigger provider model for managing credentials + Supports multiple credential instances per provider + """ + + __tablename__ = "trigger_subscriptions" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="trigger_provider_pkey"), + Index("idx_trigger_providers_tenant_provider", "tenant_id", "provider_id"), + # Primary index for O(1) lookup by endpoint + Index("idx_trigger_providers_endpoint", "endpoint_id", unique=True), + # Composite index for tenant-specific queries (optional, kept for compatibility) + Index("idx_trigger_providers_tenant_endpoint", "tenant_id", "endpoint_id"), + UniqueConstraint("tenant_id", "provider_id", "name", name="unique_trigger_provider"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuid_generate_v4()")) + name: Mapped[str] = mapped_column(String(255), nullable=False, comment="Subscription instance name") + tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + user_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + provider_id: Mapped[str] = mapped_column( + String(255), nullable=False, comment="Provider identifier (e.g., plugin_id/provider_name)" + ) + endpoint_id: Mapped[str] = mapped_column(String(255), nullable=False, comment="Subscription endpoint") + parameters: Mapped[dict[str, Any]] = mapped_column(sa.JSON, nullable=False, comment="Subscription parameters JSON") + properties: Mapped[dict[str, Any]] = mapped_column(sa.JSON, nullable=False, comment="Subscription properties JSON") + + credentials: Mapped[dict[str, Any]] = mapped_column( + sa.JSON, nullable=False, comment="Subscription credentials JSON" + ) + credential_type: Mapped[str] = mapped_column(String(50), nullable=False, comment="oauth or api_key") + credential_expires_at: Mapped[int] = mapped_column( + Integer, default=-1, comment="OAuth token expiration timestamp, -1 for never" + ) + expires_at: Mapped[int] = mapped_column( + Integer, default=-1, comment="Subscription instance expiration timestamp, -1 for never" + ) + + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, + nullable=False, + server_default=func.current_timestamp(), + server_onupdate=func.current_timestamp(), + ) + + def is_credential_expired(self) -> bool: + """Check if credential is expired""" + if self.credential_expires_at == -1: + return False + # Check if token expires in next 3 minutes + return (self.credential_expires_at - 180) < int(time.time()) + + def to_entity(self) -> Subscription: + return Subscription( + expires_at=self.expires_at, + endpoint=generate_plugin_trigger_endpoint_url(self.endpoint_id), + parameters=self.parameters, + properties=self.properties, + ) + + def to_api_entity(self) -> TriggerProviderSubscriptionApiEntity: + return TriggerProviderSubscriptionApiEntity( + id=self.id, + name=self.name, + provider=self.provider_id, + endpoint=generate_plugin_trigger_endpoint_url(self.endpoint_id), + parameters=self.parameters, + properties=self.properties, + credential_type=CredentialType(self.credential_type), + credentials=self.credentials, + workflows_in_use=-1, + ) + + +# system level trigger oauth client params +class TriggerOAuthSystemClient(Base): + __tablename__ = "trigger_oauth_system_clients" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="trigger_oauth_system_client_pkey"), + sa.UniqueConstraint("plugin_id", "provider", name="trigger_oauth_system_client_plugin_id_provider_idx"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuid_generate_v4()")) + plugin_id: Mapped[str] = mapped_column(String(512), nullable=False) + provider: Mapped[str] = mapped_column(String(255), nullable=False) + # oauth params of the trigger provider + encrypted_oauth_params: Mapped[str] = mapped_column(sa.Text, nullable=False) + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, + nullable=False, + server_default=func.current_timestamp(), + server_onupdate=func.current_timestamp(), + ) + + +# tenant level trigger oauth client params (client_id, client_secret, etc.) +class TriggerOAuthTenantClient(Base): + __tablename__ = "trigger_oauth_tenant_clients" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="trigger_oauth_tenant_client_pkey"), + sa.UniqueConstraint("tenant_id", "plugin_id", "provider", name="unique_trigger_oauth_tenant_client"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuid_generate_v4()")) + # tenant id + tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + plugin_id: Mapped[str] = mapped_column(String(512), nullable=False) + provider: Mapped[str] = mapped_column(String(255), nullable=False) + enabled: Mapped[bool] = mapped_column(sa.Boolean, nullable=False, server_default=sa.text("true")) + # oauth params of the trigger provider + encrypted_oauth_params: Mapped[str] = mapped_column(sa.Text, nullable=False) + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, + nullable=False, + server_default=func.current_timestamp(), + server_onupdate=func.current_timestamp(), + ) + + @property + def oauth_params(self) -> Mapping[str, Any]: + return cast(Mapping[str, Any], json.loads(self.encrypted_oauth_params or "{}")) + + +class WorkflowTriggerLog(Base): + """ + Workflow Trigger Log + + Track async trigger workflow runs with re-invocation capability + + Attributes: + - id (uuid) Trigger Log ID (used as workflow_trigger_log_id) + - tenant_id (uuid) Workspace ID + - app_id (uuid) App ID + - workflow_id (uuid) Workflow ID + - workflow_run_id (uuid) Optional - Associated workflow run ID when execution starts + - root_node_id (string) Optional - Custom starting node ID for workflow execution + - trigger_metadata (text) Optional - Trigger metadata (JSON) + - trigger_type (string) Type of trigger: webhook, schedule, plugin + - trigger_data (text) Full trigger data including inputs (JSON) + - inputs (text) Input parameters (JSON) + - outputs (text) Optional - Output content (JSON) + - status (string) Execution status + - error (text) Optional - Error message if failed + - queue_name (string) Celery queue used + - celery_task_id (string) Optional - Celery task ID for tracking + - retry_count (int) Number of retry attempts + - elapsed_time (float) Optional - Time consumption in seconds + - total_tokens (int) Optional - Total tokens used + - created_by_role (string) Creator role: account, end_user + - created_by (string) Creator ID + - created_at (timestamp) Creation time + - triggered_at (timestamp) Optional - When actually triggered + - finished_at (timestamp) Optional - Completion time + """ + + __tablename__ = "workflow_trigger_logs" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="workflow_trigger_log_pkey"), + sa.Index("workflow_trigger_log_tenant_app_idx", "tenant_id", "app_id"), + sa.Index("workflow_trigger_log_status_idx", "status"), + sa.Index("workflow_trigger_log_created_at_idx", "created_at"), + sa.Index("workflow_trigger_log_workflow_run_idx", "workflow_run_id"), + sa.Index("workflow_trigger_log_workflow_id_idx", "workflow_id"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) + tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + workflow_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + workflow_run_id: Mapped[str | None] = mapped_column(StringUUID, nullable=True) + root_node_id: Mapped[str | None] = mapped_column(String(255), nullable=True) + trigger_metadata: Mapped[str] = mapped_column(sa.Text, nullable=False) + trigger_type: Mapped[str] = mapped_column(EnumText(AppTriggerType, length=50), nullable=False) + trigger_data: Mapped[str] = mapped_column(sa.Text, nullable=False) # Full TriggerData as JSON + inputs: Mapped[str] = mapped_column(sa.Text, nullable=False) # Just inputs for easy viewing + outputs: Mapped[str | None] = mapped_column(sa.Text, nullable=True) + + status: Mapped[str] = mapped_column( + EnumText(WorkflowTriggerStatus, length=50), nullable=False, default=WorkflowTriggerStatus.PENDING + ) + error: Mapped[str | None] = mapped_column(sa.Text, nullable=True) + + queue_name: Mapped[str] = mapped_column(String(100), nullable=False) + celery_task_id: Mapped[str | None] = mapped_column(String(255), nullable=True) + retry_count: Mapped[int] = mapped_column(sa.Integer, nullable=False, default=0) + + elapsed_time: Mapped[float | None] = mapped_column(sa.Float, nullable=True) + total_tokens: Mapped[int | None] = mapped_column(sa.Integer, nullable=True) + + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + created_by_role: Mapped[str] = mapped_column(String(255), nullable=False) + created_by: Mapped[str] = mapped_column(String(255), nullable=False) + + triggered_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) + finished_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) + + @property + def created_by_account(self): + created_by_role = CreatorUserRole(self.created_by_role) + return db.session.get(Account, self.created_by) if created_by_role == CreatorUserRole.ACCOUNT else None + + @property + def created_by_end_user(self): + from models.model import EndUser + + created_by_role = CreatorUserRole(self.created_by_role) + return db.session.get(EndUser, self.created_by) if created_by_role == CreatorUserRole.END_USER else None + + def to_dict(self) -> dict[str, Any]: + """Convert to dictionary for API responses""" + return { + "id": self.id, + "tenant_id": self.tenant_id, + "app_id": self.app_id, + "workflow_id": self.workflow_id, + "workflow_run_id": self.workflow_run_id, + "root_node_id": self.root_node_id, + "trigger_metadata": json.loads(self.trigger_metadata) if self.trigger_metadata else None, + "trigger_type": self.trigger_type, + "trigger_data": json.loads(self.trigger_data), + "inputs": json.loads(self.inputs), + "outputs": json.loads(self.outputs) if self.outputs else None, + "status": self.status, + "error": self.error, + "queue_name": self.queue_name, + "celery_task_id": self.celery_task_id, + "retry_count": self.retry_count, + "elapsed_time": self.elapsed_time, + "total_tokens": self.total_tokens, + "created_by_role": self.created_by_role, + "created_by": self.created_by, + "created_at": self.created_at.isoformat() if self.created_at else None, + "triggered_at": self.triggered_at.isoformat() if self.triggered_at else None, + "finished_at": self.finished_at.isoformat() if self.finished_at else None, + } + + +class WorkflowWebhookTrigger(Base): + """ + Workflow Webhook Trigger + + Attributes: + - id (uuid) Primary key + - app_id (uuid) App ID to bind to a specific app + - node_id (varchar) Node ID which node in the workflow + - tenant_id (uuid) Workspace ID + - webhook_id (varchar) Webhook ID for URL: https://api.dify.ai/triggers/webhook/:webhook_id + - created_by (varchar) User ID of the creator + - created_at (timestamp) Creation time + - updated_at (timestamp) Last update time + """ + + __tablename__ = "workflow_webhook_triggers" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="workflow_webhook_trigger_pkey"), + sa.Index("workflow_webhook_trigger_tenant_idx", "tenant_id"), + sa.UniqueConstraint("app_id", "node_id", name="uniq_node"), + sa.UniqueConstraint("webhook_id", name="uniq_webhook_id"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + node_id: Mapped[str] = mapped_column(String(64), nullable=False) + tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + webhook_id: Mapped[str] = mapped_column(String(24), nullable=False) + created_by: Mapped[str] = mapped_column(StringUUID, nullable=False) + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, + nullable=False, + server_default=func.current_timestamp(), + server_onupdate=func.current_timestamp(), + ) + + @cached_property + def webhook_url(self): + """ + Generated webhook url + """ + return generate_webhook_trigger_endpoint(self.webhook_id) + + @cached_property + def webhook_debug_url(self): + """ + Generated debug webhook url + """ + return generate_webhook_trigger_endpoint(self.webhook_id, True) + + +class WorkflowPluginTrigger(Base): + """ + Workflow Plugin Trigger + + Maps plugin triggers to workflow nodes, similar to WorkflowWebhookTrigger + + Attributes: + - id (uuid) Primary key + - app_id (uuid) App ID to bind to a specific app + - node_id (varchar) Node ID which node in the workflow + - tenant_id (uuid) Workspace ID + - provider_id (varchar) Plugin provider ID + - event_name (varchar) trigger name + - subscription_id (varchar) Subscription ID + - created_at (timestamp) Creation time + - updated_at (timestamp) Last update time + """ + + __tablename__ = "workflow_plugin_triggers" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="workflow_plugin_trigger_pkey"), + sa.Index("workflow_plugin_trigger_tenant_subscription_idx", "tenant_id", "subscription_id", "event_name"), + sa.UniqueConstraint("app_id", "node_id", name="uniq_app_node_subscription"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuid_generate_v4()")) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + node_id: Mapped[str] = mapped_column(String(64), nullable=False) + tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + provider_id: Mapped[str] = mapped_column(String(512), nullable=False) + event_name: Mapped[str] = mapped_column(String(255), nullable=False) + subscription_id: Mapped[str] = mapped_column(String(255), nullable=False) + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, + nullable=False, + server_default=func.current_timestamp(), + server_onupdate=func.current_timestamp(), + ) + + +class AppTrigger(Base): + """ + App Trigger + + Manages multiple triggers for an app with enable/disable and authorization states. + + Attributes: + - id (uuid) Primary key + - tenant_id (uuid) Workspace ID + - app_id (uuid) App ID + - trigger_type (string) Type: webhook, schedule, plugin + - title (string) Trigger title + + - status (string) Status: enabled, disabled, unauthorized, error + - node_id (string) Optional workflow node ID + - created_at (timestamp) Creation time + - updated_at (timestamp) Last update time + """ + + __tablename__ = "app_triggers" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="app_trigger_pkey"), + sa.Index("app_trigger_tenant_app_idx", "tenant_id", "app_id"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) + tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + node_id: Mapped[str | None] = mapped_column(String(64), nullable=False) + trigger_type: Mapped[str] = mapped_column(EnumText(AppTriggerType, length=50), nullable=False) + title: Mapped[str] = mapped_column(String(255), nullable=False) + provider_name: Mapped[str] = mapped_column(String(255), server_default="", nullable=True) + status: Mapped[str] = mapped_column( + EnumText(AppTriggerStatus, length=50), nullable=False, default=AppTriggerStatus.ENABLED + ) + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, + nullable=False, + default=naive_utc_now(), + server_onupdate=func.current_timestamp(), + ) + + +class WorkflowSchedulePlan(Base): + """ + Workflow Schedule Configuration + + Store schedule configurations for time-based workflow triggers. + Uses cron expressions with timezone support for flexible scheduling. + + Attributes: + - id (uuid) Primary key + - app_id (uuid) App ID to bind to a specific app + - node_id (varchar) Starting node ID for workflow execution + - tenant_id (uuid) Workspace ID for multi-tenancy + - cron_expression (varchar) Cron expression defining schedule pattern + - timezone (varchar) Timezone for cron evaluation (e.g., 'Asia/Shanghai') + - next_run_at (timestamp) Next scheduled execution time + - created_at (timestamp) Creation timestamp + - updated_at (timestamp) Last update timestamp + """ + + __tablename__ = "workflow_schedule_plans" + __table_args__ = ( + sa.PrimaryKeyConstraint("id", name="workflow_schedule_plan_pkey"), + sa.UniqueConstraint("app_id", "node_id", name="uniq_app_node"), + sa.Index("workflow_schedule_plan_next_idx", "next_run_at"), + ) + + id: Mapped[str] = mapped_column(StringUUID, server_default=sa.text("uuidv7()")) + app_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + node_id: Mapped[str] = mapped_column(String(64), nullable=False) + tenant_id: Mapped[str] = mapped_column(StringUUID, nullable=False) + + # Schedule configuration + cron_expression: Mapped[str] = mapped_column(String(255), nullable=False) + timezone: Mapped[str] = mapped_column(String(64), nullable=False) + + # Schedule control + next_run_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) + created_at: Mapped[datetime] = mapped_column(DateTime, nullable=False, server_default=func.current_timestamp()) + updated_at: Mapped[datetime] = mapped_column( + DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + ) + + def to_dict(self) -> dict[str, Any]: + """Convert to dictionary representation""" + return { + "id": self.id, + "app_id": self.app_id, + "node_id": self.node_id, + "tenant_id": self.tenant_id, + "cron_expression": self.cron_expression, + "timezone": self.timezone, + "next_run_at": self.next_run_at.isoformat() if self.next_run_at else None, + "created_at": self.created_at.isoformat(), + "updated_at": self.updated_at.isoformat(), + } diff --git a/api/models/workflow.py b/api/models/workflow.py index b898f02612..4eff16dda2 100644 --- a/api/models/workflow.py +++ b/api/models/workflow.py @@ -1,6 +1,6 @@ import json import logging -from collections.abc import Mapping, Sequence +from collections.abc import Generator, Mapping, Sequence from datetime import datetime from enum import StrEnum from typing import TYPE_CHECKING, Any, Optional, Union, cast @@ -13,8 +13,11 @@ from core.file.constants import maybe_file_object from core.file.models import File from core.variables import utils as variable_utils from core.variables.variables import FloatVariable, IntegerVariable, StringVariable -from core.workflow.constants import CONVERSATION_VARIABLE_NODE_ID, SYSTEM_VARIABLE_NODE_ID -from core.workflow.enums import NodeType +from core.workflow.constants import ( + CONVERSATION_VARIABLE_NODE_ID, + SYSTEM_VARIABLE_NODE_ID, +) +from core.workflow.enums import NodeType, WorkflowExecutionStatus from extensions.ext_storage import Storage from factories.variable_factory import TypeMismatchError, build_segment_with_type from libs.datetime_utils import naive_utc_now @@ -35,7 +38,7 @@ from factories import variable_factory from libs import helper from .account import Account -from .base import Base +from .base import Base, DefaultFieldsMixin from .engine import db from .enums import CreatorUserRole, DraftVariableType, ExecutionOffLoadType from .types import EnumText, StringUUID @@ -137,8 +140,9 @@ class Workflow(Base): updated_at: Mapped[datetime] = mapped_column( DateTime, nullable=False, - default=naive_utc_now(), - server_onupdate=func.current_timestamp(), + default=func.current_timestamp(), + server_default=func.current_timestamp(), + onupdate=func.current_timestamp(), ) _environment_variables: Mapped[str] = mapped_column( "environment_variables", sa.Text, nullable=False, server_default="{}" @@ -147,7 +151,7 @@ class Workflow(Base): "conversation_variables", sa.Text, nullable=False, server_default="{}" ) _rag_pipeline_variables: Mapped[str] = mapped_column( - "rag_pipeline_variables", db.Text, nullable=False, server_default="{}" + "rag_pipeline_variables", sa.Text, nullable=False, server_default="{}" ) VERSION_DRAFT = "draft" @@ -247,7 +251,9 @@ class Workflow(Base): return node_type @staticmethod - def get_enclosing_node_type_and_id(node_config: Mapping[str, Any]) -> tuple[NodeType, str] | None: + def get_enclosing_node_type_and_id( + node_config: Mapping[str, Any], + ) -> tuple[NodeType, str] | None: in_loop = node_config.get("isInLoop", False) in_iteration = node_config.get("isInIteration", False) if in_loop: @@ -297,6 +303,54 @@ class Workflow(Base): def features_dict(self) -> dict[str, Any]: return json.loads(self.features) if self.features else {} + def walk_nodes( + self, specific_node_type: NodeType | None = None + ) -> Generator[tuple[str, Mapping[str, Any]], None, None]: + """ + Walk through the workflow nodes, yield each node configuration. + + Each node configuration is a tuple containing the node's id and the node's properties. + + Node properties example: + { + "type": "llm", + "title": "LLM", + "desc": "", + "variables": [], + "model": + { + "provider": "langgenius/openai/openai", + "name": "gpt-4", + "mode": "chat", + "completion_params": { "temperature": 0.7 }, + }, + "prompt_template": [{ "role": "system", "text": "" }], + "context": { "enabled": false, "variable_selector": [] }, + "vision": { "enabled": false }, + "memory": + { + "window": { "enabled": false, "size": 10 }, + "query_prompt_template": "{{#sys.query#}}\n\n{{#sys.files#}}", + "role_prefix": { "user": "", "assistant": "" }, + }, + "selected": false, + } + + For specific node type, refer to `core.workflow.nodes` + """ + graph_dict = self.graph_dict + if "nodes" not in graph_dict: + raise WorkflowDataError("nodes not found in workflow graph") + + if specific_node_type: + yield from ( + (node["id"], node["data"]) + for node in graph_dict["nodes"] + if node["data"]["type"] == specific_node_type.value + ) + else: + yield from ((node["id"], node["data"]) for node in graph_dict["nodes"]) + def user_input_form(self, to_old_structure: bool = False) -> list[Any]: # get start node from graph if not self.graph: @@ -306,7 +360,10 @@ class Workflow(Base): if "nodes" not in graph_dict: return [] - start_node = next((node for node in graph_dict["nodes"] if node["data"]["type"] == "start"), None) + start_node = next( + (node for node in graph_dict["nodes"] if node["data"]["type"] == "start"), + None, + ) if not start_node: return [] @@ -359,7 +416,9 @@ class Workflow(Base): return db.session.execute(stmt).scalar_one() @property - def environment_variables(self) -> Sequence[StringVariable | IntegerVariable | FloatVariable | SecretVariable]: + def environment_variables( + self, + ) -> Sequence[StringVariable | IntegerVariable | FloatVariable | SecretVariable]: # TODO: find some way to init `self._environment_variables` when instance created. if self._environment_variables is None: self._environment_variables = "{}" @@ -376,7 +435,9 @@ class Workflow(Base): ] # decrypt secret variables value - def decrypt_func(var: Variable) -> StringVariable | IntegerVariable | FloatVariable | SecretVariable: + def decrypt_func( + var: Variable, + ) -> StringVariable | IntegerVariable | FloatVariable | SecretVariable: if isinstance(var, SecretVariable): return var.model_copy(update={"value": encrypter.decrypt_token(tenant_id=tenant_id, token=var.value)}) elif isinstance(var, (StringVariable, IntegerVariable, FloatVariable)): @@ -537,7 +598,10 @@ class WorkflowRun(Base): version: Mapped[str] = mapped_column(String(255)) graph: Mapped[str | None] = mapped_column(sa.Text) inputs: Mapped[str | None] = mapped_column(sa.Text) - status: Mapped[str] = mapped_column(String(255)) # running, succeeded, failed, stopped, partial-succeeded + status: Mapped[str] = mapped_column( + EnumText(WorkflowExecutionStatus, length=255), + nullable=False, + ) outputs: Mapped[str | None] = mapped_column(sa.Text, default="{}") error: Mapped[str | None] = mapped_column(sa.Text) elapsed_time: Mapped[float] = mapped_column(sa.Float, nullable=False, server_default=sa.text("0")) @@ -549,6 +613,15 @@ class WorkflowRun(Base): finished_at: Mapped[datetime | None] = mapped_column(DateTime) exceptions_count: Mapped[int] = mapped_column(sa.Integer, server_default=sa.text("0"), nullable=True) + pause: Mapped[Optional["WorkflowPause"]] = orm.relationship( + "WorkflowPause", + primaryjoin="WorkflowRun.id == foreign(WorkflowPause.workflow_run_id)", + uselist=False, + # require explicit preloading. + lazy="raise", + back_populates="workflow_run", + ) + @property def created_by_account(self): created_by_role = CreatorUserRole(self.created_by_role) @@ -1034,7 +1107,16 @@ class WorkflowAppLog(Base): @property def workflow_run(self): - return db.session.get(WorkflowRun, self.workflow_run_id) + if self.workflow_run_id: + from sqlalchemy.orm import sessionmaker + + from repositories.factory import DifyAPIRepositoryFactory + + session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + return repo.get_workflow_run_by_id_without_tenant(run_id=self.workflow_run_id) + + return None @property def created_by_account(self): @@ -1073,7 +1155,10 @@ class ConversationVariable(Base): DateTime, nullable=False, server_default=func.current_timestamp(), index=True ) updated_at: Mapped[datetime] = mapped_column( - DateTime, nullable=False, server_default=func.current_timestamp(), onupdate=func.current_timestamp() + DateTime, + nullable=False, + server_default=func.current_timestamp(), + onupdate=func.current_timestamp(), ) def __init__(self, *, id: str, app_id: str, conversation_id: str, data: str): @@ -1101,10 +1186,6 @@ class ConversationVariable(Base): _EDITABLE_SYSTEM_VARIABLE = frozenset(["query", "files"]) -def _naive_utc_datetime(): - return naive_utc_now() - - class WorkflowDraftVariable(Base): """`WorkflowDraftVariable` record variables and outputs generated during debugging workflow or chatflow. @@ -1138,14 +1219,14 @@ class WorkflowDraftVariable(Base): created_at: Mapped[datetime] = mapped_column( DateTime, nullable=False, - default=_naive_utc_datetime, + default=naive_utc_now, server_default=func.current_timestamp(), ) updated_at: Mapped[datetime] = mapped_column( DateTime, nullable=False, - default=_naive_utc_datetime, + default=naive_utc_now, server_default=func.current_timestamp(), onupdate=func.current_timestamp(), ) @@ -1412,8 +1493,8 @@ class WorkflowDraftVariable(Base): file_id: str | None = None, ) -> "WorkflowDraftVariable": variable = WorkflowDraftVariable() - variable.created_at = _naive_utc_datetime() - variable.updated_at = _naive_utc_datetime() + variable.created_at = naive_utc_now() + variable.updated_at = naive_utc_now() variable.description = description variable.app_id = app_id variable.node_id = node_id @@ -1518,7 +1599,7 @@ class WorkflowDraftVariableFile(Base): created_at: Mapped[datetime] = mapped_column( DateTime, nullable=False, - default=_naive_utc_datetime, + default=naive_utc_now, server_default=func.current_timestamp(), ) @@ -1583,3 +1664,68 @@ class WorkflowDraftVariableFile(Base): def is_system_variable_editable(name: str) -> bool: return name in _EDITABLE_SYSTEM_VARIABLE + + +class WorkflowPause(DefaultFieldsMixin, Base): + """ + WorkflowPause records the paused state and related metadata for a specific workflow run. + + Each `WorkflowRun` can have zero or one associated `WorkflowPause`, depending on its execution status. + If a `WorkflowRun` is in the `PAUSED` state, there must be a corresponding `WorkflowPause` + that has not yet been resumed. + Otherwise, there should be no active (non-resumed) `WorkflowPause` linked to that run. + + This model captures the execution context required to resume workflow processing at a later time. + """ + + __tablename__ = "workflow_pauses" + __table_args__ = ( + # Design Note: + # Instead of adding a `pause_id` field to the `WorkflowRun` model—which would require a migration + # on a potentially large table—we reference `WorkflowRun` from `WorkflowPause` and enforce a unique + # constraint on `workflow_run_id` to guarantee a one-to-one relationship. + UniqueConstraint("workflow_run_id"), + ) + + # `workflow_id` represents the unique identifier of the workflow associated with this pause. + # It corresponds to the `id` field in the `Workflow` model. + # + # Since an application can have multiple versions of a workflow, each with its own unique ID, + # the `app_id` alone is insufficient to determine which workflow version should be loaded + # when resuming a suspended workflow. + workflow_id: Mapped[str] = mapped_column( + StringUUID, + nullable=False, + ) + + # `workflow_run_id` represents the identifier of the execution of workflow, + # correspond to the `id` field of `WorkflowRun`. + workflow_run_id: Mapped[str] = mapped_column( + StringUUID, + nullable=False, + ) + + # `resumed_at` records the timestamp when the suspended workflow was resumed. + # It is set to `NULL` if the workflow has not been resumed. + # + # NOTE: Resuming a suspended WorkflowPause does not delete the record immediately. + # It only set `resumed_at` to a non-null value. + resumed_at: Mapped[datetime | None] = mapped_column( + sa.DateTime, + nullable=True, + ) + + # state_object_key stores the object key referencing the serialized runtime state + # of the `GraphEngine`. This object captures the complete execution context of the + # workflow at the moment it was paused, enabling accurate resumption. + state_object_key: Mapped[str] = mapped_column(String(length=255), nullable=False) + + # Relationship to WorkflowRun + workflow_run: Mapped["WorkflowRun"] = orm.relationship( + foreign_keys=[workflow_run_id], + # require explicit preloading. + lazy="raise", + uselist=False, + primaryjoin="WorkflowPause.workflow_run_id == WorkflowRun.id", + back_populates="pause", + ) diff --git a/api/pyproject.toml b/api/pyproject.toml index 5a9becaaef..1cf7d719ea 100644 --- a/api/pyproject.toml +++ b/api/pyproject.toml @@ -1,6 +1,6 @@ [project] name = "dify-api" -version = "1.9.2" +version = "1.10.0" requires-python = ">=3.11,<3.13" dependencies = [ @@ -37,6 +37,7 @@ dependencies = [ "numpy~=1.26.4", "openpyxl~=3.1.5", "opik~=1.8.72", + "litellm==1.77.1", # Pinned to avoid madoka dependency issue "opentelemetry-api==1.27.0", "opentelemetry-distro==0.48b0", "opentelemetry-exporter-otlp==1.27.0", @@ -74,11 +75,10 @@ dependencies = [ "resend~=2.9.0", "sentry-sdk[flask]~=2.28.0", "sqlalchemy~=2.0.29", - "starlette==0.47.2", + "starlette==0.49.1", "tiktoken~=0.9.0", "transformers~=4.56.1", "unstructured[docx,epub,md,ppt,pptx]~=0.16.1", - "weave~=0.51.0", "yarl~=1.18.3", "webvtt-py~=0.5.1", "sseclient-py~=1.8.0", @@ -86,7 +86,10 @@ dependencies = [ "sendgrid~=6.12.3", "flask-restx~=1.3.0", "packaging~=23.2", + "croniter>=6.0.0", "weaviate-client==4.17.0", + "apscheduler>=3.11.0", + "weave>=0.52.16", ] # Before adding new dependency, consider place it in # alphabet order (a-z) and suitable group. @@ -117,7 +120,7 @@ dev = [ "pytest-cov~=4.1.0", "pytest-env~=1.1.3", "pytest-mock~=3.14.0", - "testcontainers~=4.10.0", + "testcontainers~=4.13.2", "types-aiofiles~=24.1.0", "types-beautifulsoup4~=4.12.0", "types-cachetools~=5.5.0", @@ -209,9 +212,9 @@ vdb = [ "pgvector==0.2.5", "pymilvus~=2.5.0", "pymochow==2.2.9", - "pyobvector~=0.2.15", + "pyobvector~=0.2.17", "qdrant-client==1.9.0", - "tablestore==6.2.0", + "tablestore==6.3.7", "tcvectordb~=1.6.4", "tidb-vector==0.0.9", "upstash-vector==0.6.0", diff --git a/api/repositories/api_workflow_run_repository.py b/api/repositories/api_workflow_run_repository.py index 72de9fed31..21fd57cd22 100644 --- a/api/repositories/api_workflow_run_repository.py +++ b/api/repositories/api_workflow_run_repository.py @@ -28,7 +28,7 @@ Example: runs = repo.get_paginated_workflow_runs( tenant_id="tenant-123", app_id="app-456", - triggered_from="debugging", + triggered_from=WorkflowRunTriggeredFrom.DEBUGGING, limit=20 ) ``` @@ -38,9 +38,17 @@ from collections.abc import Sequence from datetime import datetime from typing import Protocol +from core.workflow.entities.workflow_pause import WorkflowPauseEntity from core.workflow.repositories.workflow_execution_repository import WorkflowExecutionRepository from libs.infinite_scroll_pagination import InfiniteScrollPagination +from models.enums import WorkflowRunTriggeredFrom from models.workflow import WorkflowRun +from repositories.types import ( + AverageInteractionStats, + DailyRunsStats, + DailyTerminalsStats, + DailyTokenCostStats, +) class APIWorkflowRunRepository(WorkflowExecutionRepository, Protocol): @@ -56,7 +64,7 @@ class APIWorkflowRunRepository(WorkflowExecutionRepository, Protocol): self, tenant_id: str, app_id: str, - triggered_from: str, + triggered_from: WorkflowRunTriggeredFrom | Sequence[WorkflowRunTriggeredFrom], limit: int = 20, last_id: str | None = None, status: str | None = None, @@ -71,7 +79,7 @@ class APIWorkflowRunRepository(WorkflowExecutionRepository, Protocol): Args: tenant_id: Tenant identifier for multi-tenant isolation app_id: Application identifier - triggered_from: Filter by trigger source (e.g., "debugging", "app-run") + triggered_from: Filter by trigger source(s) (e.g., "debugging", "app-run", or list of values) limit: Maximum number of records to return (default: 20) last_id: Cursor for pagination - ID of the last record from previous page status: Optional filter by status (e.g., "running", "succeeded", "failed") @@ -109,6 +117,31 @@ class APIWorkflowRunRepository(WorkflowExecutionRepository, Protocol): """ ... + def get_workflow_run_by_id_without_tenant( + self, + run_id: str, + ) -> WorkflowRun | None: + """ + Get a specific workflow run by ID without tenant/app context. + + Retrieves a single workflow run using only the run ID, without + requiring tenant_id or app_id. This method is intended for internal + system operations like tracing and monitoring where the tenant context + is not available upfront. + + Args: + run_id: Workflow run identifier + + Returns: + WorkflowRun object if found, None otherwise + + Note: + This method bypasses tenant isolation checks and should only be used + in trusted system contexts like ops trace collection. For user-facing + operations, use get_workflow_run_by_id() with proper tenant isolation. + """ + ... + def get_workflow_runs_count( self, tenant_id: str, @@ -218,3 +251,229 @@ class APIWorkflowRunRepository(WorkflowExecutionRepository, Protocol): and ensure proper data retention policies are followed. """ ... + + def create_workflow_pause( + self, + workflow_run_id: str, + state_owner_user_id: str, + state: str, + ) -> WorkflowPauseEntity: + """ + Create a new workflow pause state. + + Creates a pause state for a workflow run, storing the current execution + state and marking the workflow as paused. This is used when a workflow + needs to be suspended and later resumed. + + Args: + workflow_run_id: Identifier of the workflow run to pause + state_owner_user_id: User ID who owns the pause state for file storage + state: Serialized workflow execution state (JSON string) + + Returns: + WorkflowPauseEntity representing the created pause state + + Raises: + ValueError: If workflow_run_id is invalid or workflow run doesn't exist + RuntimeError: If workflow is already paused or in invalid state + """ + # NOTE: we may get rid of the `state_owner_user_id` in parameter list. + # However, removing it would require an extra for `Workflow` model + # while creating pause. + ... + + def resume_workflow_pause( + self, + workflow_run_id: str, + pause_entity: WorkflowPauseEntity, + ) -> WorkflowPauseEntity: + """ + Resume a paused workflow. + + Marks a paused workflow as resumed, set the `resumed_at` field of WorkflowPauseEntity + and returning the workflow to running status. Returns the pause entity + that was resumed. + + The returned `WorkflowPauseEntity` model has `resumed_at` set. + + NOTE: this method does not delete the correspond `WorkflowPauseEntity` record and associated states. + It's the callers responsibility to clear the correspond state with `delete_workflow_pause`. + + Args: + workflow_run_id: Identifier of the workflow run to resume + pause_entity: The pause entity to resume + + Returns: + WorkflowPauseEntity representing the resumed pause state + + Raises: + ValueError: If workflow_run_id is invalid + RuntimeError: If workflow is not paused or already resumed + """ + ... + + def delete_workflow_pause( + self, + pause_entity: WorkflowPauseEntity, + ) -> None: + """ + Delete a workflow pause state. + + Permanently removes the pause state for a workflow run, including + the stored state file. Used for cleanup operations when a paused + workflow is no longer needed. + + Args: + pause_entity: The pause entity to delete + + Raises: + ValueError: If pause_entity is invalid + RuntimeError: If workflow is not paused + + Note: + This operation is irreversible. The stored workflow state will be + permanently deleted along with the pause record. + """ + ... + + def prune_pauses( + self, + expiration: datetime, + resumption_expiration: datetime, + limit: int | None = None, + ) -> Sequence[str]: + """ + Clean up expired and old pause states. + + Removes pause states that have expired (created before expiration time) + and pause states that were resumed more than resumption_duration ago. + This is used for maintenance and cleanup operations. + + Args: + expiration: Remove pause states created before this time + resumption_expiration: Remove pause states resumed before this time + limit: maximum number of records deleted in one call + + Returns: + a list of ids for pause records that were pruned + + Raises: + ValueError: If parameters are invalid + """ + ... + + def get_daily_runs_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[DailyRunsStats]: + """ + Get daily runs statistics. + + Retrieves daily workflow runs count grouped by date for a specific app + and trigger source. Used for workflow statistics dashboard. + + Args: + tenant_id: Tenant identifier for multi-tenant isolation + app_id: Application identifier + triggered_from: Filter by trigger source (e.g., "app-run") + start_date: Optional start date filter + end_date: Optional end date filter + timezone: Timezone for date grouping (default: "UTC") + + Returns: + List of dictionaries containing date and runs count: + [{"date": "2024-01-01", "runs": 10}, ...] + """ + ... + + def get_daily_terminals_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[DailyTerminalsStats]: + """ + Get daily terminals statistics. + + Retrieves daily unique terminal count grouped by date for a specific app + and trigger source. Used for workflow statistics dashboard. + + Args: + tenant_id: Tenant identifier for multi-tenant isolation + app_id: Application identifier + triggered_from: Filter by trigger source (e.g., "app-run") + start_date: Optional start date filter + end_date: Optional end date filter + timezone: Timezone for date grouping (default: "UTC") + + Returns: + List of dictionaries containing date and terminal count: + [{"date": "2024-01-01", "terminal_count": 5}, ...] + """ + ... + + def get_daily_token_cost_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[DailyTokenCostStats]: + """ + Get daily token cost statistics. + + Retrieves daily total token count grouped by date for a specific app + and trigger source. Used for workflow statistics dashboard. + + Args: + tenant_id: Tenant identifier for multi-tenant isolation + app_id: Application identifier + triggered_from: Filter by trigger source (e.g., "app-run") + start_date: Optional start date filter + end_date: Optional end date filter + timezone: Timezone for date grouping (default: "UTC") + + Returns: + List of dictionaries containing date and token count: + [{"date": "2024-01-01", "token_count": 1000}, ...] + """ + ... + + def get_average_app_interaction_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[AverageInteractionStats]: + """ + Get average app interaction statistics. + + Retrieves daily average interactions per user grouped by date for a specific app + and trigger source. Used for workflow statistics dashboard. + + Args: + tenant_id: Tenant identifier for multi-tenant isolation + app_id: Application identifier + triggered_from: Filter by trigger source (e.g., "app-run") + start_date: Optional start date filter + end_date: Optional end date filter + timezone: Timezone for date grouping (default: "UTC") + + Returns: + List of dictionaries containing date and average interactions: + [{"date": "2024-01-01", "interactions": 2.5}, ...] + """ + ... diff --git a/api/repositories/factory.py b/api/repositories/factory.py index 96f9f886a4..8e098a7059 100644 --- a/api/repositories/factory.py +++ b/api/repositories/factory.py @@ -5,7 +5,7 @@ This factory is specifically designed for DifyAPI repositories that handle service-layer operations with dependency injection patterns. """ -from sqlalchemy.orm import sessionmaker +from sqlalchemy.orm import Session, sessionmaker from configs import dify_config from core.repositories import DifyCoreRepositoryFactory, RepositoryImportError @@ -25,7 +25,7 @@ class DifyAPIRepositoryFactory(DifyCoreRepositoryFactory): @classmethod def create_api_workflow_node_execution_repository( - cls, session_maker: sessionmaker + cls, session_maker: sessionmaker[Session] ) -> DifyAPIWorkflowNodeExecutionRepository: """ Create a DifyAPIWorkflowNodeExecutionRepository instance based on configuration. @@ -55,7 +55,7 @@ class DifyAPIRepositoryFactory(DifyCoreRepositoryFactory): ) from e @classmethod - def create_api_workflow_run_repository(cls, session_maker: sessionmaker) -> APIWorkflowRunRepository: + def create_api_workflow_run_repository(cls, session_maker: sessionmaker[Session]) -> APIWorkflowRunRepository: """ Create an APIWorkflowRunRepository instance based on configuration. diff --git a/api/repositories/sqlalchemy_api_workflow_run_repository.py b/api/repositories/sqlalchemy_api_workflow_run_repository.py index 68affb59f3..0d52c56138 100644 --- a/api/repositories/sqlalchemy_api_workflow_run_repository.py +++ b/api/repositories/sqlalchemy_api_workflow_run_repository.py @@ -20,22 +20,42 @@ Implementation Notes: """ import logging +import uuid from collections.abc import Sequence from datetime import datetime -from typing import cast +from decimal import Decimal +from typing import Any, cast -from sqlalchemy import delete, func, select +import sqlalchemy as sa +from sqlalchemy import and_, delete, func, null, or_, select from sqlalchemy.engine import CursorResult -from sqlalchemy.orm import Session, sessionmaker +from sqlalchemy.orm import Session, selectinload, sessionmaker +from core.workflow.entities.workflow_pause import WorkflowPauseEntity +from core.workflow.enums import WorkflowExecutionStatus +from extensions.ext_storage import storage +from libs.datetime_utils import naive_utc_now from libs.infinite_scroll_pagination import InfiniteScrollPagination from libs.time_parser import get_time_threshold +from libs.uuid_utils import uuidv7 +from models.enums import WorkflowRunTriggeredFrom +from models.workflow import WorkflowPause as WorkflowPauseModel from models.workflow import WorkflowRun from repositories.api_workflow_run_repository import APIWorkflowRunRepository +from repositories.types import ( + AverageInteractionStats, + DailyRunsStats, + DailyTerminalsStats, + DailyTokenCostStats, +) logger = logging.getLogger(__name__) +class _WorkflowRunError(Exception): + pass + + class DifyAPISQLAlchemyWorkflowRunRepository(APIWorkflowRunRepository): """ SQLAlchemy implementation of APIWorkflowRunRepository. @@ -61,7 +81,7 @@ class DifyAPISQLAlchemyWorkflowRunRepository(APIWorkflowRunRepository): self, tenant_id: str, app_id: str, - triggered_from: str, + triggered_from: WorkflowRunTriggeredFrom | Sequence[WorkflowRunTriggeredFrom], limit: int = 20, last_id: str | None = None, status: str | None = None, @@ -78,9 +98,14 @@ class DifyAPISQLAlchemyWorkflowRunRepository(APIWorkflowRunRepository): base_stmt = select(WorkflowRun).where( WorkflowRun.tenant_id == tenant_id, WorkflowRun.app_id == app_id, - WorkflowRun.triggered_from == triggered_from, ) + # Handle triggered_from values + if isinstance(triggered_from, WorkflowRunTriggeredFrom): + triggered_from = [triggered_from] + if triggered_from: + base_stmt = base_stmt.where(WorkflowRun.triggered_from.in_(triggered_from)) + # Add optional status filter if status: base_stmt = base_stmt.where(WorkflowRun.status == status) @@ -126,6 +151,17 @@ class DifyAPISQLAlchemyWorkflowRunRepository(APIWorkflowRunRepository): ) return session.scalar(stmt) + def get_workflow_run_by_id_without_tenant( + self, + run_id: str, + ) -> WorkflowRun | None: + """ + Get a specific workflow run by ID without tenant/app context. + """ + with self._session_maker() as session: + stmt = select(WorkflowRun).where(WorkflowRun.id == run_id) + return session.scalar(stmt) + def get_workflow_runs_count( self, tenant_id: str, @@ -275,3 +311,554 @@ class DifyAPISQLAlchemyWorkflowRunRepository(APIWorkflowRunRepository): logger.info("Total deleted %s workflow runs for app %s", total_deleted, app_id) return total_deleted + + def create_workflow_pause( + self, + workflow_run_id: str, + state_owner_user_id: str, + state: str, + ) -> WorkflowPauseEntity: + """ + Create a new workflow pause state. + + Creates a pause state for a workflow run, storing the current execution + state and marking the workflow as paused. This is used when a workflow + needs to be suspended and later resumed. + + Args: + workflow_run_id: Identifier of the workflow run to pause + state_owner_user_id: User ID who owns the pause state for file storage + state: Serialized workflow execution state (JSON string) + + Returns: + RepositoryWorkflowPauseEntity representing the created pause state + + Raises: + ValueError: If workflow_run_id is invalid or workflow run doesn't exist + RuntimeError: If workflow is already paused or in invalid state + """ + previous_pause_model_query = select(WorkflowPauseModel).where( + WorkflowPauseModel.workflow_run_id == workflow_run_id + ) + with self._session_maker() as session, session.begin(): + # Get the workflow run + workflow_run = session.get(WorkflowRun, workflow_run_id) + if workflow_run is None: + raise ValueError(f"WorkflowRun not found: {workflow_run_id}") + + # Check if workflow is in RUNNING status + if workflow_run.status != WorkflowExecutionStatus.RUNNING: + raise _WorkflowRunError( + f"Only WorkflowRun with RUNNING status can be paused, " + f"workflow_run_id={workflow_run_id}, current_status={workflow_run.status}" + ) + # + previous_pause = session.scalars(previous_pause_model_query).first() + if previous_pause: + self._delete_pause_model(session, previous_pause) + # we need to flush here to ensure that the old one is actually deleted. + session.flush() + + state_obj_key = f"workflow-state-{uuid.uuid4()}.json" + storage.save(state_obj_key, state.encode()) + # Upload the state file + + # Create the pause record + pause_model = WorkflowPauseModel() + pause_model.id = str(uuidv7()) + pause_model.workflow_id = workflow_run.workflow_id + pause_model.workflow_run_id = workflow_run.id + pause_model.state_object_key = state_obj_key + pause_model.created_at = naive_utc_now() + + # Update workflow run status + workflow_run.status = WorkflowExecutionStatus.PAUSED + + # Save everything in a transaction + session.add(pause_model) + session.add(workflow_run) + + logger.info("Created workflow pause %s for workflow run %s", pause_model.id, workflow_run_id) + + return _PrivateWorkflowPauseEntity.from_models(pause_model) + + def get_workflow_pause( + self, + workflow_run_id: str, + ) -> WorkflowPauseEntity | None: + """ + Get an existing workflow pause state. + + Retrieves the pause state for a specific workflow run if it exists. + Used to check if a workflow is paused and to retrieve its saved state. + + Args: + workflow_run_id: Identifier of the workflow run to get pause state for + + Returns: + RepositoryWorkflowPauseEntity if pause state exists, None otherwise + + Raises: + ValueError: If workflow_run_id is invalid + """ + with self._session_maker() as session: + # Query workflow run with pause and state file + stmt = select(WorkflowRun).options(selectinload(WorkflowRun.pause)).where(WorkflowRun.id == workflow_run_id) + workflow_run = session.scalar(stmt) + + if workflow_run is None: + raise ValueError(f"WorkflowRun not found: {workflow_run_id}") + + pause_model = workflow_run.pause + if pause_model is None: + return None + + return _PrivateWorkflowPauseEntity.from_models(pause_model) + + def resume_workflow_pause( + self, + workflow_run_id: str, + pause_entity: WorkflowPauseEntity, + ) -> WorkflowPauseEntity: + """ + Resume a paused workflow. + + Marks a paused workflow as resumed, clearing the pause state and + returning the workflow to running status. Returns the pause entity + that was resumed. + + Args: + workflow_run_id: Identifier of the workflow run to resume + pause_entity: The pause entity to resume + + Returns: + RepositoryWorkflowPauseEntity representing the resumed pause state + + Raises: + ValueError: If workflow_run_id is invalid + RuntimeError: If workflow is not paused or already resumed + """ + with self._session_maker() as session, session.begin(): + # Get the workflow run with pause + stmt = select(WorkflowRun).options(selectinload(WorkflowRun.pause)).where(WorkflowRun.id == workflow_run_id) + workflow_run = session.scalar(stmt) + + if workflow_run is None: + raise ValueError(f"WorkflowRun not found: {workflow_run_id}") + + if workflow_run.status != WorkflowExecutionStatus.PAUSED: + raise _WorkflowRunError( + f"WorkflowRun is not in PAUSED status, workflow_run_id={workflow_run_id}, " + f"current_status={workflow_run.status}" + ) + pause_model = workflow_run.pause + if pause_model is None: + raise _WorkflowRunError(f"No pause state found for workflow run: {workflow_run_id}") + + if pause_model.id != pause_entity.id: + raise _WorkflowRunError( + "different id in WorkflowPause and WorkflowPauseEntity, " + f"WorkflowPause.id={pause_model.id}, " + f"WorkflowPauseEntity.id={pause_entity.id}" + ) + + if pause_model.resumed_at is not None: + raise _WorkflowRunError(f"Cannot resume an already resumed pause, pause_id={pause_model.id}") + + # Mark as resumed + pause_model.resumed_at = naive_utc_now() + workflow_run.pause_id = None # type: ignore + workflow_run.status = WorkflowExecutionStatus.RUNNING + + session.add(pause_model) + session.add(workflow_run) + + logger.info("Resumed workflow pause %s for workflow run %s", pause_model.id, workflow_run_id) + + return _PrivateWorkflowPauseEntity.from_models(pause_model) + + def delete_workflow_pause( + self, + pause_entity: WorkflowPauseEntity, + ) -> None: + """ + Delete a workflow pause state. + + Permanently removes the pause state for a workflow run, including + the stored state file. Used for cleanup operations when a paused + workflow is no longer needed. + + Args: + pause_entity: The pause entity to delete + + Raises: + ValueError: If pause_entity is invalid + _WorkflowRunError: If workflow is not paused + + Note: + This operation is irreversible. The stored workflow state will be + permanently deleted along with the pause record. + """ + with self._session_maker() as session, session.begin(): + # Get the pause model by ID + pause_model = session.get(WorkflowPauseModel, pause_entity.id) + if pause_model is None: + raise _WorkflowRunError(f"WorkflowPause not found: {pause_entity.id}") + self._delete_pause_model(session, pause_model) + + @staticmethod + def _delete_pause_model(session: Session, pause_model: WorkflowPauseModel): + storage.delete(pause_model.state_object_key) + + # Delete the pause record + session.delete(pause_model) + + logger.info("Deleted workflow pause %s for workflow run %s", pause_model.id, pause_model.workflow_run_id) + + def prune_pauses( + self, + expiration: datetime, + resumption_expiration: datetime, + limit: int | None = None, + ) -> Sequence[str]: + """ + Clean up expired and old pause states. + + Removes pause states that have expired (created before expiration time) + and pause states that were resumed more than resumption_duration ago. + This is used for maintenance and cleanup operations. + + Args: + expiration: Remove pause states created before this time + resumption_expiration: Remove pause states resumed before this time + limit: maximum number of records deleted in one call + + Returns: + a list of ids for pause records that were pruned + + Raises: + ValueError: If parameters are invalid + """ + _limit: int = limit or 1000 + pruned_record_ids: list[str] = [] + cond = or_( + WorkflowPauseModel.created_at < expiration, + and_( + WorkflowPauseModel.resumed_at.is_not(null()), + WorkflowPauseModel.resumed_at < resumption_expiration, + ), + ) + # First, collect pause records to delete with their state files + # Expired pauses (created before expiration time) + stmt = select(WorkflowPauseModel).where(cond).limit(_limit) + + with self._session_maker(expire_on_commit=False) as session: + # Old resumed pauses (resumed more than resumption_duration ago) + + # Get all records to delete + pauses_to_delete = session.scalars(stmt).all() + + # Delete state files from storage + for pause in pauses_to_delete: + with self._session_maker(expire_on_commit=False) as session, session.begin(): + # todo: this issues a separate query for each WorkflowPauseModel record. + # consider batching this lookup. + try: + storage.delete(pause.state_object_key) + logger.info( + "Deleted state object for pause, pause_id=%s, object_key=%s", + pause.id, + pause.state_object_key, + ) + except Exception: + logger.exception( + "Failed to delete state file for pause, pause_id=%s, object_key=%s", + pause.id, + pause.state_object_key, + ) + continue + session.delete(pause) + pruned_record_ids.append(pause.id) + logger.info( + "workflow pause records deleted, id=%s, resumed_at=%s", + pause.id, + pause.resumed_at, + ) + + return pruned_record_ids + + def get_daily_runs_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[DailyRunsStats]: + """ + Get daily runs statistics using raw SQL for optimal performance. + """ + sql_query = """SELECT + DATE(DATE_TRUNC('day', created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, + COUNT(id) AS runs +FROM + workflow_runs +WHERE + tenant_id = :tenant_id + AND app_id = :app_id + AND triggered_from = :triggered_from""" + + arg_dict: dict[str, Any] = { + "tz": timezone, + "tenant_id": tenant_id, + "app_id": app_id, + "triggered_from": triggered_from, + } + + if start_date: + sql_query += " AND created_at >= :start_date" + arg_dict["start_date"] = start_date + + if end_date: + sql_query += " AND created_at < :end_date" + arg_dict["end_date"] = end_date + + sql_query += " GROUP BY date ORDER BY date" + + response_data = [] + with self._session_maker() as session: + rs = session.execute(sa.text(sql_query), arg_dict) + for row in rs: + response_data.append({"date": str(row.date), "runs": row.runs}) + + return cast(list[DailyRunsStats], response_data) + + def get_daily_terminals_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[DailyTerminalsStats]: + """ + Get daily terminals statistics using raw SQL for optimal performance. + """ + sql_query = """SELECT + DATE(DATE_TRUNC('day', created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, + COUNT(DISTINCT created_by) AS terminal_count +FROM + workflow_runs +WHERE + tenant_id = :tenant_id + AND app_id = :app_id + AND triggered_from = :triggered_from""" + + arg_dict: dict[str, Any] = { + "tz": timezone, + "tenant_id": tenant_id, + "app_id": app_id, + "triggered_from": triggered_from, + } + + if start_date: + sql_query += " AND created_at >= :start_date" + arg_dict["start_date"] = start_date + + if end_date: + sql_query += " AND created_at < :end_date" + arg_dict["end_date"] = end_date + + sql_query += " GROUP BY date ORDER BY date" + + response_data = [] + with self._session_maker() as session: + rs = session.execute(sa.text(sql_query), arg_dict) + for row in rs: + response_data.append({"date": str(row.date), "terminal_count": row.terminal_count}) + + return cast(list[DailyTerminalsStats], response_data) + + def get_daily_token_cost_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[DailyTokenCostStats]: + """ + Get daily token cost statistics using raw SQL for optimal performance. + """ + sql_query = """SELECT + DATE(DATE_TRUNC('day', created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, + SUM(total_tokens) AS token_count +FROM + workflow_runs +WHERE + tenant_id = :tenant_id + AND app_id = :app_id + AND triggered_from = :triggered_from""" + + arg_dict: dict[str, Any] = { + "tz": timezone, + "tenant_id": tenant_id, + "app_id": app_id, + "triggered_from": triggered_from, + } + + if start_date: + sql_query += " AND created_at >= :start_date" + arg_dict["start_date"] = start_date + + if end_date: + sql_query += " AND created_at < :end_date" + arg_dict["end_date"] = end_date + + sql_query += " GROUP BY date ORDER BY date" + + response_data = [] + with self._session_maker() as session: + rs = session.execute(sa.text(sql_query), arg_dict) + for row in rs: + response_data.append( + { + "date": str(row.date), + "token_count": row.token_count, + } + ) + + return cast(list[DailyTokenCostStats], response_data) + + def get_average_app_interaction_statistics( + self, + tenant_id: str, + app_id: str, + triggered_from: str, + start_date: datetime | None = None, + end_date: datetime | None = None, + timezone: str = "UTC", + ) -> list[AverageInteractionStats]: + """ + Get average app interaction statistics using raw SQL for optimal performance. + """ + sql_query = """SELECT + AVG(sub.interactions) AS interactions, + sub.date +FROM + ( + SELECT + DATE(DATE_TRUNC('day', c.created_at AT TIME ZONE 'UTC' AT TIME ZONE :tz )) AS date, + c.created_by, + COUNT(c.id) AS interactions + FROM + workflow_runs c + WHERE + c.tenant_id = :tenant_id + AND c.app_id = :app_id + AND c.triggered_from = :triggered_from + {{start}} + {{end}} + GROUP BY + date, c.created_by + ) sub +GROUP BY + sub.date""" + + arg_dict: dict[str, Any] = { + "tz": timezone, + "tenant_id": tenant_id, + "app_id": app_id, + "triggered_from": triggered_from, + } + + if start_date: + sql_query = sql_query.replace("{{start}}", " AND c.created_at >= :start_date") + arg_dict["start_date"] = start_date + else: + sql_query = sql_query.replace("{{start}}", "") + + if end_date: + sql_query = sql_query.replace("{{end}}", " AND c.created_at < :end_date") + arg_dict["end_date"] = end_date + else: + sql_query = sql_query.replace("{{end}}", "") + + response_data = [] + with self._session_maker() as session: + rs = session.execute(sa.text(sql_query), arg_dict) + for row in rs: + response_data.append( + {"date": str(row.date), "interactions": float(row.interactions.quantize(Decimal("0.01")))} + ) + + return cast(list[AverageInteractionStats], response_data) + + +class _PrivateWorkflowPauseEntity(WorkflowPauseEntity): + """ + Private implementation of WorkflowPauseEntity for SQLAlchemy repository. + + This implementation is internal to the repository layer and provides + the concrete implementation of the WorkflowPauseEntity interface. + """ + + def __init__( + self, + *, + pause_model: WorkflowPauseModel, + ) -> None: + self._pause_model = pause_model + self._cached_state: bytes | None = None + + @classmethod + def from_models(cls, workflow_pause_model) -> "_PrivateWorkflowPauseEntity": + """ + Create a _PrivateWorkflowPauseEntity from database models. + + Args: + workflow_pause_model: The WorkflowPause database model + upload_file_model: The UploadFile database model + + Returns: + _PrivateWorkflowPauseEntity: The constructed entity + + Raises: + ValueError: If required model attributes are missing + """ + return cls(pause_model=workflow_pause_model) + + @property + def id(self) -> str: + return self._pause_model.id + + @property + def workflow_execution_id(self) -> str: + return self._pause_model.workflow_run_id + + def get_state(self) -> bytes: + """ + Retrieve the serialized workflow state from storage. + + Returns: + Mapping[str, Any]: The workflow state as a dictionary + + Raises: + FileNotFoundError: If the state file cannot be found + IOError: If there are issues reading the state file + _Workflow: If the state cannot be deserialized properly + """ + if self._cached_state is not None: + return self._cached_state + + # Load the state from storage + state_data = storage.load(self._pause_model.state_object_key) + self._cached_state = state_data + return state_data + + @property + def resumed_at(self) -> datetime | None: + return self._pause_model.resumed_at diff --git a/api/repositories/sqlalchemy_workflow_trigger_log_repository.py b/api/repositories/sqlalchemy_workflow_trigger_log_repository.py new file mode 100644 index 0000000000..0d67e286b0 --- /dev/null +++ b/api/repositories/sqlalchemy_workflow_trigger_log_repository.py @@ -0,0 +1,86 @@ +""" +SQLAlchemy implementation of WorkflowTriggerLogRepository. +""" + +from collections.abc import Sequence +from datetime import UTC, datetime, timedelta + +from sqlalchemy import and_, select +from sqlalchemy.orm import Session + +from models.enums import WorkflowTriggerStatus +from models.trigger import WorkflowTriggerLog +from repositories.workflow_trigger_log_repository import WorkflowTriggerLogRepository + + +class SQLAlchemyWorkflowTriggerLogRepository(WorkflowTriggerLogRepository): + """ + SQLAlchemy implementation of WorkflowTriggerLogRepository. + + Optimized for large table operations with proper indexing and batch processing. + """ + + def __init__(self, session: Session): + self.session = session + + def create(self, trigger_log: WorkflowTriggerLog) -> WorkflowTriggerLog: + """Create a new trigger log entry.""" + self.session.add(trigger_log) + self.session.flush() + return trigger_log + + def update(self, trigger_log: WorkflowTriggerLog) -> WorkflowTriggerLog: + """Update an existing trigger log entry.""" + self.session.merge(trigger_log) + self.session.flush() + return trigger_log + + def get_by_id(self, trigger_log_id: str, tenant_id: str | None = None) -> WorkflowTriggerLog | None: + """Get a trigger log by its ID.""" + query = select(WorkflowTriggerLog).where(WorkflowTriggerLog.id == trigger_log_id) + + if tenant_id: + query = query.where(WorkflowTriggerLog.tenant_id == tenant_id) + + return self.session.scalar(query) + + def get_failed_for_retry( + self, tenant_id: str, max_retry_count: int = 3, limit: int = 100 + ) -> Sequence[WorkflowTriggerLog]: + """Get failed trigger logs eligible for retry.""" + query = ( + select(WorkflowTriggerLog) + .where( + and_( + WorkflowTriggerLog.tenant_id == tenant_id, + WorkflowTriggerLog.status.in_([WorkflowTriggerStatus.FAILED, WorkflowTriggerStatus.RATE_LIMITED]), + WorkflowTriggerLog.retry_count < max_retry_count, + ) + ) + .order_by(WorkflowTriggerLog.created_at.asc()) + .limit(limit) + ) + + return list(self.session.scalars(query).all()) + + def get_recent_logs( + self, tenant_id: str, app_id: str, hours: int = 24, limit: int = 100, offset: int = 0 + ) -> Sequence[WorkflowTriggerLog]: + """Get recent trigger logs within specified hours.""" + since = datetime.now(UTC) - timedelta(hours=hours) + + query = ( + select(WorkflowTriggerLog) + .where( + and_( + WorkflowTriggerLog.tenant_id == tenant_id, + WorkflowTriggerLog.app_id == app_id, + WorkflowTriggerLog.created_at >= since, + ) + ) + .order_by(WorkflowTriggerLog.created_at.desc()) + .limit(limit) + .offset(offset) + ) + + return list(self.session.scalars(query).all()) diff --git a/api/repositories/types.py b/api/repositories/types.py new file mode 100644 index 0000000000..3b3ef7f635 --- /dev/null +++ b/api/repositories/types.py @@ -0,0 +1,21 @@ +from typing import TypedDict + + +class DailyRunsStats(TypedDict): + date: str + runs: int + + +class DailyTerminalsStats(TypedDict): + date: str + terminal_count: int + + +class DailyTokenCostStats(TypedDict): + date: str + token_count: int + + +class AverageInteractionStats(TypedDict): + date: str + interactions: float diff --git a/api/repositories/workflow_trigger_log_repository.py b/api/repositories/workflow_trigger_log_repository.py new file mode 100644 index 0000000000..138b8779ac --- /dev/null +++ b/api/repositories/workflow_trigger_log_repository.py @@ -0,0 +1,111 @@ +""" +Repository protocol for WorkflowTriggerLog operations. + +This module provides a protocol interface for operations on WorkflowTriggerLog, +designed to efficiently handle a potentially large volume of trigger logs with +proper indexing and batch operations. +""" + +from collections.abc import Sequence +from enum import StrEnum +from typing import Protocol + +from models.trigger import WorkflowTriggerLog + + +class TriggerLogOrderBy(StrEnum): + """Fields available for ordering trigger logs""" + + CREATED_AT = "created_at" + TRIGGERED_AT = "triggered_at" + FINISHED_AT = "finished_at" + STATUS = "status" + + +class WorkflowTriggerLogRepository(Protocol): + """ + Protocol for operations on WorkflowTriggerLog. + + This repository provides efficient access patterns for the trigger log table, + which is expected to grow large over time. It includes: + - Batch operations for cleanup + - Efficient queries with proper indexing + - Pagination support + - Status-based filtering + + Implementation notes: + - Leverage database indexes on (tenant_id, app_id), status, and created_at + - Use batch operations for deletions to avoid locking + - Support pagination for large result sets + """ + + def create(self, trigger_log: WorkflowTriggerLog) -> WorkflowTriggerLog: + """ + Create a new trigger log entry. + + Args: + trigger_log: The WorkflowTriggerLog instance to create + + Returns: + The created WorkflowTriggerLog with generated ID + """ + ... + + def update(self, trigger_log: WorkflowTriggerLog) -> WorkflowTriggerLog: + """ + Update an existing trigger log entry. + + Args: + trigger_log: The WorkflowTriggerLog instance to update + + Returns: + The updated WorkflowTriggerLog + """ + ... + + def get_by_id(self, trigger_log_id: str, tenant_id: str | None = None) -> WorkflowTriggerLog | None: + """ + Get a trigger log by its ID. + + Args: + trigger_log_id: The trigger log identifier + tenant_id: Optional tenant identifier for additional security + + Returns: + The WorkflowTriggerLog if found, None otherwise + """ + ... + + def get_failed_for_retry( + self, tenant_id: str, max_retry_count: int = 3, limit: int = 100 + ) -> Sequence[WorkflowTriggerLog]: + """ + Get failed trigger logs that are eligible for retry. + + Args: + tenant_id: The tenant identifier + max_retry_count: Maximum retry count to consider + limit: Maximum number of results + + Returns: + A sequence of WorkflowTriggerLog instances eligible for retry + """ + ... + + def get_recent_logs( + self, tenant_id: str, app_id: str, hours: int = 24, limit: int = 100, offset: int = 0 + ) -> Sequence[WorkflowTriggerLog]: + """ + Get recent trigger logs within specified hours. + + Args: + tenant_id: The tenant identifier + app_id: The application identifier + hours: Number of hours to look back + limit: Maximum number of results + offset: Number of results to skip + + Returns: + A sequence of recent WorkflowTriggerLog instances + """ + ... diff --git a/api/schedule/clean_messages.py b/api/schedule/clean_messages.py index 65038dce4d..352a84b592 100644 --- a/api/schedule/clean_messages.py +++ b/api/schedule/clean_messages.py @@ -7,6 +7,7 @@ from sqlalchemy.exc import SQLAlchemyError import app from configs import dify_config +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from extensions.ext_redis import redis_client from models.model import ( @@ -63,7 +64,7 @@ def clean_messages(): plan = features.billing.subscription.plan else: plan = plan_cache.decode() - if plan == "sandbox": + if plan == CloudPlan.SANDBOX: # clean related message db.session.query(MessageFeedback).where(MessageFeedback.message_id == message.id).delete( synchronize_session=False diff --git a/api/schedule/clean_unused_datasets_task.py b/api/schedule/clean_unused_datasets_task.py index 9efd46ba5d..d9fb6a24f1 100644 --- a/api/schedule/clean_unused_datasets_task.py +++ b/api/schedule/clean_unused_datasets_task.py @@ -9,6 +9,7 @@ from sqlalchemy.exc import SQLAlchemyError import app from configs import dify_config from core.rag.index_processor.index_processor_factory import IndexProcessorFactory +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from extensions.ext_redis import redis_client from models.dataset import Dataset, DatasetAutoDisableLog, DatasetQuery, Document @@ -35,7 +36,7 @@ def clean_unused_datasets_task(): }, { "clean_day": datetime.datetime.now() - datetime.timedelta(days=dify_config.PLAN_PRO_CLEAN_DAY_SETTING), - "plan_filter": "sandbox", + "plan_filter": CloudPlan.SANDBOX, "add_logs": False, }, ] diff --git a/api/schedule/mail_clean_document_notify_task.py b/api/schedule/mail_clean_document_notify_task.py index b70707b17e..d738bf46fa 100644 --- a/api/schedule/mail_clean_document_notify_task.py +++ b/api/schedule/mail_clean_document_notify_task.py @@ -7,6 +7,7 @@ from sqlalchemy import select import app from configs import dify_config +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from extensions.ext_mail import mail from libs.email_i18n import EmailType, get_email_i18n_service @@ -45,7 +46,7 @@ def mail_clean_document_notify_task(): for tenant_id, tenant_dataset_auto_disable_logs in dataset_auto_disable_logs_map.items(): features = FeatureService.get_features(tenant_id) plan = features.billing.subscription.plan - if plan != "sandbox": + if plan != CloudPlan.SANDBOX: knowledge_details = [] # check tenant tenant = db.session.query(Tenant).where(Tenant.id == tenant_id).first() diff --git a/api/schedule/trigger_provider_refresh_task.py b/api/schedule/trigger_provider_refresh_task.py new file mode 100644 index 0000000000..3b3e478793 --- /dev/null +++ b/api/schedule/trigger_provider_refresh_task.py @@ -0,0 +1,104 @@ +import logging +import math +import time +from collections.abc import Iterable, Sequence + +from sqlalchemy import ColumnElement, and_, func, or_, select +from sqlalchemy.engine.row import Row +from sqlalchemy.orm import Session + +import app +from configs import dify_config +from core.trigger.utils.locks import build_trigger_refresh_lock_keys +from extensions.ext_database import db +from extensions.ext_redis import redis_client +from models.trigger import TriggerSubscription +from tasks.trigger_subscription_refresh_tasks import trigger_subscription_refresh + +logger = logging.getLogger(__name__) + + +def _now_ts() -> int: + return int(time.time()) + + +def _build_due_filter(now_ts: int): + """Build SQLAlchemy filter for due credential or subscription refresh.""" + credential_due: ColumnElement[bool] = and_( + TriggerSubscription.credential_expires_at != -1, + TriggerSubscription.credential_expires_at + <= now_ts + int(dify_config.TRIGGER_PROVIDER_CREDENTIAL_THRESHOLD_SECONDS), + ) + subscription_due: ColumnElement[bool] = and_( + TriggerSubscription.expires_at != -1, + TriggerSubscription.expires_at <= now_ts + int(dify_config.TRIGGER_PROVIDER_SUBSCRIPTION_THRESHOLD_SECONDS), + ) + return or_(credential_due, subscription_due) + + +def _acquire_locks(keys: Iterable[str], ttl_seconds: int) -> list[bool]: + """Attempt to acquire locks in a single pipelined round-trip. + + Returns a list of booleans indicating which locks were acquired. + """ + pipe = redis_client.pipeline(transaction=False) + for key in keys: + pipe.set(key, b"1", ex=ttl_seconds, nx=True) + results = pipe.execute() + return [bool(r) for r in results] + + +@app.celery.task(queue="trigger_refresh_publisher") +def trigger_provider_refresh() -> None: + """ + Scan due trigger subscriptions and enqueue refresh tasks with in-flight locks. + """ + now: int = _now_ts() + + batch_size: int = int(dify_config.TRIGGER_PROVIDER_REFRESH_BATCH_SIZE) + lock_ttl: int = max(300, int(dify_config.TRIGGER_PROVIDER_SUBSCRIPTION_THRESHOLD_SECONDS)) + + with Session(db.engine, expire_on_commit=False) as session: + filter: ColumnElement[bool] = _build_due_filter(now_ts=now) + total_due: int = int(session.scalar(statement=select(func.count()).where(filter)) or 0) + logger.info("Trigger refresh scan start: due=%d", total_due) + if total_due == 0: + return + + pages: int = math.ceil(total_due / batch_size) + for page in range(pages): + offset: int = page * batch_size + subscription_rows: Sequence[Row[tuple[str, str]]] = session.execute( + select(TriggerSubscription.tenant_id, TriggerSubscription.id) + .where(filter) + .order_by(TriggerSubscription.updated_at.asc()) + .offset(offset) + .limit(batch_size) + ).all() + if not subscription_rows: + logger.debug("Trigger refresh page %d/%d empty", page + 1, pages) + continue + + subscriptions: list[tuple[str, str]] = [ + (str(tenant_id), str(subscription_id)) for tenant_id, subscription_id in subscription_rows + ] + lock_keys: list[str] = build_trigger_refresh_lock_keys(subscriptions) + acquired: list[bool] = _acquire_locks(keys=lock_keys, ttl_seconds=lock_ttl) + + enqueued: int = 0 + for (tenant_id, subscription_id), is_locked in zip(subscriptions, acquired): + if not is_locked: + continue + trigger_subscription_refresh.delay(tenant_id=tenant_id, subscription_id=subscription_id) + enqueued += 1 + + logger.info( + "Trigger refresh page %d/%d: scanned=%d locks_acquired=%d enqueued=%d", + page + 1, + pages, + len(subscriptions), + sum(1 for x in acquired if x), + enqueued, + ) + + logger.info("Trigger refresh scan done: due=%d", total_due) diff --git a/api/schedule/workflow_schedule_task.py b/api/schedule/workflow_schedule_task.py new file mode 100644 index 0000000000..41e2232353 --- /dev/null +++ b/api/schedule/workflow_schedule_task.py @@ -0,0 +1,127 @@ +import logging + +from celery import group, shared_task +from sqlalchemy import and_, select +from sqlalchemy.orm import Session, sessionmaker + +from configs import dify_config +from extensions.ext_database import db +from libs.datetime_utils import naive_utc_now +from libs.schedule_utils import calculate_next_run_at +from models.trigger import AppTrigger, AppTriggerStatus, AppTriggerType, WorkflowSchedulePlan +from services.workflow.queue_dispatcher import QueueDispatcherManager +from tasks.workflow_schedule_tasks import run_schedule_trigger + +logger = logging.getLogger(__name__) + + +@shared_task(queue="schedule_poller") +def poll_workflow_schedules() -> None: + """ + Poll and process due workflow schedules. + + Streaming flow: + 1. Fetch due schedules in batches + 2. Process each batch until all due schedules are handled + 3. Optional: Limit total dispatches per tick as a circuit breaker + """ + session_factory = sessionmaker(bind=db.engine, expire_on_commit=False) + + with session_factory() as session: + total_dispatched = 0 + total_rate_limited = 0 + + # Process in batches until we've handled all due schedules or hit the limit + while True: + due_schedules = _fetch_due_schedules(session) + + if not due_schedules: + break + + dispatched_count, rate_limited_count = _process_schedules(session, due_schedules) + total_dispatched += dispatched_count + total_rate_limited += rate_limited_count + + logger.debug("Batch processed: %d dispatched, %d rate limited", dispatched_count, rate_limited_count) + + # Circuit breaker: check if we've hit the per-tick limit (if enabled) + if ( + dify_config.WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK > 0 + and total_dispatched >= dify_config.WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK + ): + logger.warning( + "Circuit breaker activated: reached dispatch limit (%d), will continue next tick", + dify_config.WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK, + ) + break + + if total_dispatched > 0 or total_rate_limited > 0: + logger.info("Total processed: %d dispatched, %d rate limited", total_dispatched, total_rate_limited) + + +def _fetch_due_schedules(session: Session) -> list[WorkflowSchedulePlan]: + """ + Fetch a batch of due schedules, sorted by most overdue first. + + Returns up to WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE schedules per call. + Used in a loop to progressively process all due schedules. + """ + now = naive_utc_now() + + due_schedules = session.scalars( + ( + select(WorkflowSchedulePlan) + .join( + AppTrigger, + and_( + AppTrigger.app_id == WorkflowSchedulePlan.app_id, + AppTrigger.node_id == WorkflowSchedulePlan.node_id, + AppTrigger.trigger_type == AppTriggerType.TRIGGER_SCHEDULE, + ), + ) + .where( + WorkflowSchedulePlan.next_run_at <= now, + WorkflowSchedulePlan.next_run_at.isnot(None), + AppTrigger.status == AppTriggerStatus.ENABLED, + ) + ) + .order_by(WorkflowSchedulePlan.next_run_at.asc()) + .with_for_update(skip_locked=True) + .limit(dify_config.WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE) + ) + + return list(due_schedules) + + +def _process_schedules(session: Session, schedules: list[WorkflowSchedulePlan]) -> tuple[int, int]: + """Process schedules: check quota, update next run time and dispatch to Celery in parallel.""" + if not schedules: + return 0, 0 + + dispatcher_manager = QueueDispatcherManager() + tasks_to_dispatch: list[str] = [] + rate_limited_count = 0 + + for schedule in schedules: + next_run_at = calculate_next_run_at( + schedule.cron_expression, + schedule.timezone, + ) + schedule.next_run_at = next_run_at + + dispatcher = dispatcher_manager.get_dispatcher(schedule.tenant_id) + if not dispatcher.check_daily_quota(schedule.tenant_id): + logger.info("Tenant %s rate limited, skipping schedule_plan %s", schedule.tenant_id, schedule.id) + rate_limited_count += 1 + else: + tasks_to_dispatch.append(schedule.id) + + if tasks_to_dispatch: + job = group(run_schedule_trigger.s(schedule_id) for schedule_id in tasks_to_dispatch) + job.apply_async() + + logger.debug("Dispatched %d tasks in parallel", len(tasks_to_dispatch)) + + session.commit() + + return len(tasks_to_dispatch), rate_limited_count diff --git a/api/services/annotation_service.py b/api/services/annotation_service.py index c0d26cdd27..9258def907 100644 --- a/api/services/annotation_service.py +++ b/api/services/annotation_service.py @@ -32,41 +32,48 @@ class AppAnnotationService: if not app: raise NotFound("App not found") + + answer = args.get("answer") or args.get("content") + if answer is None: + raise ValueError("Either 'answer' or 'content' must be provided") + if args.get("message_id"): message_id = str(args["message_id"]) - # get message info message = db.session.query(Message).where(Message.id == message_id, Message.app_id == app.id).first() if not message: raise NotFound("Message Not Exists.") + question = args.get("question") or message.query or "" + annotation: MessageAnnotation | None = message.annotation - # save the message annotation if annotation: - annotation.content = args["answer"] - annotation.question = args["question"] + annotation.content = answer + annotation.question = question else: annotation = MessageAnnotation( app_id=app.id, conversation_id=message.conversation_id, message_id=message.id, - content=args["answer"], - question=args["question"], + content=answer, + question=question, account_id=current_user.id, ) else: - annotation = MessageAnnotation( - app_id=app.id, content=args["answer"], question=args["question"], account_id=current_user.id - ) + question = args.get("question") + if not question: + raise ValueError("'question' is required when 'message_id' is not provided") + + annotation = MessageAnnotation(app_id=app.id, content=answer, question=question, account_id=current_user.id) db.session.add(annotation) db.session.commit() - # if annotation reply is enabled , add annotation to index + annotation_setting = db.session.query(AppAnnotationSetting).where(AppAnnotationSetting.app_id == app_id).first() assert current_tenant_id is not None if annotation_setting: add_annotation_to_index_task.delay( annotation.id, - args["question"], + annotation.question, current_tenant_id, app_id, annotation_setting.collection_binding_id, diff --git a/api/services/app_dsl_service.py b/api/services/app_dsl_service.py index edb18a845a..15fefd6116 100644 --- a/api/services/app_dsl_service.py +++ b/api/services/app_dsl_service.py @@ -26,6 +26,7 @@ from core.workflow.nodes.llm.entities import LLMNodeData from core.workflow.nodes.parameter_extractor.entities import ParameterExtractorNodeData from core.workflow.nodes.question_classifier.entities import QuestionClassifierNodeData from core.workflow.nodes.tool.entities import ToolNodeData +from core.workflow.nodes.trigger_schedule.trigger_schedule_node import TriggerScheduleNode from events.app_event import app_model_config_was_updated, app_was_created from extensions.ext_redis import redis_client from factories import variable_factory @@ -43,7 +44,7 @@ IMPORT_INFO_REDIS_KEY_PREFIX = "app_import_info:" CHECK_DEPENDENCIES_REDIS_KEY_PREFIX = "app_check_dependencies:" IMPORT_INFO_REDIS_EXPIRY = 10 * 60 # 10 minutes DSL_MAX_SIZE = 10 * 1024 * 1024 # 10MB -CURRENT_DSL_VERSION = "0.4.0" +CURRENT_DSL_VERSION = "0.5.0" class ImportMode(StrEnum): @@ -599,6 +600,16 @@ class AppDslService: if not include_secret and data_type == NodeType.AGENT: for tool in node_data.get("agent_parameters", {}).get("tools", {}).get("value", []): tool.pop("credential_id", None) + if data_type == NodeType.TRIGGER_SCHEDULE.value: + # override the config with the default config + node_data["config"] = TriggerScheduleNode.get_default_config()["config"] + if data_type == NodeType.TRIGGER_WEBHOOK.value: + # clear the webhook_url + node_data["webhook_url"] = "" + node_data["webhook_debug_url"] = "" + if data_type == NodeType.TRIGGER_PLUGIN.value: + # clear the subscription_id + node_data["subscription_id"] = "" export_data["workflow"] = workflow_dict dependencies = cls._extract_dependencies_from_workflow(workflow) diff --git a/api/services/app_generate_service.py b/api/services/app_generate_service.py index b462ddf236..5b09bd9593 100644 --- a/api/services/app_generate_service.py +++ b/api/services/app_generate_service.py @@ -10,6 +10,7 @@ from core.app.apps.completion.app_generator import CompletionAppGenerator from core.app.apps.workflow.app_generator import WorkflowAppGenerator from core.app.entities.app_invoke_entities import InvokeFrom from core.app.features.rate_limiting import RateLimit +from enums.cloud_plan import CloudPlan from libs.helper import RateLimiter from models.model import Account, App, AppMode, EndUser from models.workflow import Workflow @@ -30,6 +31,7 @@ class AppGenerateService: args: Mapping[str, Any], invoke_from: InvokeFrom, streaming: bool = True, + root_node_id: str | None = None, ): """ App Content Generate @@ -44,7 +46,7 @@ class AppGenerateService: if dify_config.BILLING_ENABLED: # check if it's free plan limit_info = BillingService.get_info(app_model.tenant_id) - if limit_info["subscription"]["plan"] == "sandbox": + if limit_info["subscription"]["plan"] == CloudPlan.SANDBOX: if cls.system_rate_limiter.is_rate_limited(app_model.tenant_id): raise InvokeRateLimitError( "Rate limit exceeded, please upgrade your plan " @@ -113,6 +115,7 @@ class AppGenerateService: args=args, invoke_from=invoke_from, streaming=streaming, + root_node_id=root_node_id, call_depth=0, ), ), diff --git a/api/services/async_workflow_service.py b/api/services/async_workflow_service.py new file mode 100644 index 0000000000..034d7ffedb --- /dev/null +++ b/api/services/async_workflow_service.py @@ -0,0 +1,323 @@ +""" +Universal async workflow execution service. + +This service provides a centralized entry point for triggering workflows asynchronously +with support for different subscription tiers, rate limiting, and execution tracking. +""" + +import json +from datetime import UTC, datetime +from typing import Any, Union + +from celery.result import AsyncResult +from sqlalchemy import select +from sqlalchemy.orm import Session + +from extensions.ext_database import db +from extensions.ext_redis import redis_client +from models.account import Account +from models.enums import CreatorUserRole, WorkflowTriggerStatus +from models.model import App, EndUser +from models.trigger import WorkflowTriggerLog +from models.workflow import Workflow +from repositories.sqlalchemy_workflow_trigger_log_repository import SQLAlchemyWorkflowTriggerLogRepository +from services.errors.app import InvokeDailyRateLimitError, WorkflowNotFoundError +from services.workflow.entities import AsyncTriggerResponse, TriggerData, WorkflowTaskData +from services.workflow.queue_dispatcher import QueueDispatcherManager, QueuePriority +from services.workflow.rate_limiter import TenantDailyRateLimiter +from services.workflow_service import WorkflowService +from tasks.async_workflow_tasks import ( + execute_workflow_professional, + execute_workflow_sandbox, + execute_workflow_team, +) + + +class AsyncWorkflowService: + """ + Universal entry point for async workflow execution - ALL METHODS ARE NON-BLOCKING + + This service handles: + - Trigger data validation and processing + - Queue routing based on subscription tier + - Daily rate limiting with timezone support + - Execution tracking and logging + - Retry mechanisms for failed executions + + Important: All trigger methods return immediately after queuing tasks. + Actual workflow execution happens asynchronously in background Celery workers. + Use trigger log IDs to monitor execution status and results. + """ + + @classmethod + def trigger_workflow_async( + cls, session: Session, user: Union[Account, EndUser], trigger_data: TriggerData + ) -> AsyncTriggerResponse: + """ + Universal entry point for async workflow execution - THIS METHOD WILL NOT BLOCK + + Creates a trigger log and dispatches to appropriate queue based on subscription tier. + The workflow execution happens asynchronously in the background via Celery workers. + This method returns immediately after queuing the task, not after execution completion. + + Args: + session: Database session to use for operations + user: User (Account or EndUser) who initiated the workflow trigger + trigger_data: Validated Pydantic model containing trigger information + + Returns: + AsyncTriggerResponse with workflow_trigger_log_id, task_id, status="queued", and queue + Note: The actual workflow execution status must be checked separately via workflow_trigger_log_id + + Raises: + WorkflowNotFoundError: If app or workflow not found + InvokeDailyRateLimitError: If daily rate limit exceeded + + Behavior: + - Non-blocking: Returns immediately after queuing + - Asynchronous: Actual execution happens in background Celery workers + - Status tracking: Use workflow_trigger_log_id to monitor progress + - Queue-based: Routes to different queues based on subscription tier + """ + trigger_log_repo = SQLAlchemyWorkflowTriggerLogRepository(session) + dispatcher_manager = QueueDispatcherManager() + workflow_service = WorkflowService() + rate_limiter = TenantDailyRateLimiter(redis_client) + + # 1. Validate app exists + app_model = session.scalar(select(App).where(App.id == trigger_data.app_id)) + if not app_model: + raise WorkflowNotFoundError(f"App not found: {trigger_data.app_id}") + + # 2. Get workflow + workflow = cls._get_workflow(workflow_service, app_model, trigger_data.workflow_id) + + # 3. Get dispatcher based on tenant subscription + dispatcher = dispatcher_manager.get_dispatcher(trigger_data.tenant_id) + + # 4. Rate limiting check will be done without timezone first + + # 5. Determine user role and ID + if isinstance(user, Account): + created_by_role = CreatorUserRole.ACCOUNT + created_by = user.id + else: # EndUser + created_by_role = CreatorUserRole.END_USER + created_by = user.id + + # 6. Create trigger log entry first (for tracking) + trigger_log = WorkflowTriggerLog( + tenant_id=trigger_data.tenant_id, + app_id=trigger_data.app_id, + workflow_id=workflow.id, + root_node_id=trigger_data.root_node_id, + trigger_metadata=( + trigger_data.trigger_metadata.model_dump_json() if trigger_data.trigger_metadata else "{}" + ), + trigger_type=trigger_data.trigger_type, + trigger_data=trigger_data.model_dump_json(), + inputs=json.dumps(dict(trigger_data.inputs)), + status=WorkflowTriggerStatus.PENDING, + queue_name=dispatcher.get_queue_name(), + retry_count=0, + created_by_role=created_by_role, + created_by=created_by, + ) + + trigger_log = trigger_log_repo.create(trigger_log) + session.commit() + + # 7. Check and consume daily quota + if not dispatcher.consume_quota(trigger_data.tenant_id): + # Update trigger log status + trigger_log.status = WorkflowTriggerStatus.RATE_LIMITED + trigger_log.error = f"Daily limit reached for {dispatcher.get_queue_name()}" + trigger_log_repo.update(trigger_log) + session.commit() + + tenant_owner_tz = rate_limiter.get_tenant_owner_timezone(trigger_data.tenant_id) + + remaining = rate_limiter.get_remaining_quota(trigger_data.tenant_id, dispatcher.get_daily_limit()) + + reset_time = rate_limiter.get_quota_reset_time(trigger_data.tenant_id, tenant_owner_tz) + + raise InvokeDailyRateLimitError( + f"Daily workflow execution limit reached. " + f"Limit resets at {reset_time.strftime('%Y-%m-%d %H:%M:%S %Z')}. " + f"Remaining quota: {remaining}" + ) + + # 8. Create task data + queue_name = dispatcher.get_queue_name() + + task_data = WorkflowTaskData(workflow_trigger_log_id=trigger_log.id) + + # 9. Dispatch to appropriate queue + task_data_dict = task_data.model_dump(mode="json") + + task: AsyncResult[Any] | None = None + if queue_name == QueuePriority.PROFESSIONAL: + task = execute_workflow_professional.delay(task_data_dict) # type: ignore + elif queue_name == QueuePriority.TEAM: + task = execute_workflow_team.delay(task_data_dict) # type: ignore + else: # SANDBOX + task = execute_workflow_sandbox.delay(task_data_dict) # type: ignore + + # 10. Update trigger log with task info + trigger_log.status = WorkflowTriggerStatus.QUEUED + trigger_log.celery_task_id = task.id + trigger_log.triggered_at = datetime.now(UTC) + trigger_log_repo.update(trigger_log) + session.commit() + + return AsyncTriggerResponse( + workflow_trigger_log_id=trigger_log.id, + task_id=task.id, # type: ignore + status="queued", + queue=queue_name, + ) + + @classmethod + def reinvoke_trigger( + cls, session: Session, user: Union[Account, EndUser], workflow_trigger_log_id: str + ) -> AsyncTriggerResponse: + """ + Re-invoke a previously failed or rate-limited trigger - THIS METHOD WILL NOT BLOCK + + Updates the existing trigger log to retry status and creates a new async execution. + Returns immediately after queuing the retry, not after execution completion. + + Args: + session: Database session to use for operations + user: User (Account or EndUser) who initiated the retry + workflow_trigger_log_id: ID of the trigger log to re-invoke + + Returns: + AsyncTriggerResponse with new execution information (status="queued") + Note: This creates a new trigger log entry for the retry attempt + + Raises: + ValueError: If trigger log not found + + Behavior: + - Non-blocking: Returns immediately after queuing retry + - Creates new trigger log: Original log marked as retrying, new log for execution + - Preserves original trigger data: Uses same inputs and configuration + """ + trigger_log_repo = SQLAlchemyWorkflowTriggerLogRepository(session) + + trigger_log = trigger_log_repo.get_by_id(workflow_trigger_log_id) + + if not trigger_log: + raise ValueError(f"Trigger log not found: {workflow_trigger_log_id}") + + # Reconstruct trigger data from log + trigger_data = TriggerData.model_validate_json(trigger_log.trigger_data) + + # Reset log for retry + trigger_log.status = WorkflowTriggerStatus.RETRYING + trigger_log.retry_count += 1 + trigger_log.error = None + trigger_log.triggered_at = datetime.now(UTC) + trigger_log_repo.update(trigger_log) + session.commit() + + # Re-trigger workflow (this will create a new trigger log) + return cls.trigger_workflow_async(session, user, trigger_data) + + @classmethod + def get_trigger_log(cls, workflow_trigger_log_id: str, tenant_id: str | None = None) -> dict[str, Any] | None: + """ + Get trigger log by ID + + Args: + workflow_trigger_log_id: ID of the trigger log + tenant_id: Optional tenant ID for security check + + Returns: + Trigger log as dictionary or None if not found + """ + with Session(db.engine) as session: + trigger_log_repo = SQLAlchemyWorkflowTriggerLogRepository(session) + trigger_log = trigger_log_repo.get_by_id(workflow_trigger_log_id, tenant_id) + + if not trigger_log: + return None + + return trigger_log.to_dict() + + @classmethod + def get_recent_logs( + cls, tenant_id: str, app_id: str, hours: int = 24, limit: int = 100, offset: int = 0 + ) -> list[dict[str, Any]]: + """ + Get recent trigger logs + + Args: + tenant_id: Tenant ID + app_id: Application ID + hours: Number of hours to look back + limit: Maximum number of results + offset: Number of results to skip + + Returns: + List of trigger logs as dictionaries + """ + with Session(db.engine) as session: + trigger_log_repo = SQLAlchemyWorkflowTriggerLogRepository(session) + logs = trigger_log_repo.get_recent_logs( + tenant_id=tenant_id, app_id=app_id, hours=hours, limit=limit, offset=offset + ) + + return [log.to_dict() for log in logs] + + @classmethod + def get_failed_logs_for_retry( + cls, tenant_id: str, max_retry_count: int = 3, limit: int = 100 + ) -> list[dict[str, Any]]: + """ + Get failed logs eligible for retry + + Args: + tenant_id: Tenant ID + max_retry_count: Maximum retry count + limit: Maximum number of results + + Returns: + List of failed trigger logs as dictionaries + """ + with Session(db.engine) as session: + trigger_log_repo = SQLAlchemyWorkflowTriggerLogRepository(session) + logs = trigger_log_repo.get_failed_for_retry( + tenant_id=tenant_id, max_retry_count=max_retry_count, limit=limit + ) + + return [log.to_dict() for log in logs] + + @staticmethod + def _get_workflow(workflow_service: WorkflowService, app_model: App, workflow_id: str | None = None) -> Workflow: + """ + Get workflow for the app + + Args: + app_model: App model instance + workflow_id: Optional specific workflow ID + + Returns: + Workflow instance + + Raises: + WorkflowNotFoundError: If workflow not found + """ + if workflow_id: + # Get specific published workflow + workflow = workflow_service.get_published_workflow_by_id(app_model, workflow_id) + if not workflow: + raise WorkflowNotFoundError(f"Published workflow not found: {workflow_id}") + else: + # Get default published workflow + workflow = workflow_service.get_published_workflow(app_model) + if not workflow: + raise WorkflowNotFoundError(f"No published workflow found for app: {app_model.id}") + + return workflow diff --git a/api/services/audio_service.py b/api/services/audio_service.py index 1158fc5197..41ee9c88aa 100644 --- a/api/services/audio_service.py +++ b/api/services/audio_service.py @@ -82,54 +82,51 @@ class AudioService: message_id: str | None = None, is_draft: bool = False, ): - from app import app - def invoke_tts(text_content: str, app_model: App, voice: str | None = None, is_draft: bool = False): - with app.app_context(): - if voice is None: - if app_model.mode in {AppMode.ADVANCED_CHAT, AppMode.WORKFLOW}: - if is_draft: - workflow = WorkflowService().get_draft_workflow(app_model=app_model) - else: - workflow = app_model.workflow - if ( - workflow is None - or "text_to_speech" not in workflow.features_dict - or not workflow.features_dict["text_to_speech"].get("enabled") - ): + if voice is None: + if app_model.mode in {AppMode.ADVANCED_CHAT, AppMode.WORKFLOW}: + if is_draft: + workflow = WorkflowService().get_draft_workflow(app_model=app_model) + else: + workflow = app_model.workflow + if ( + workflow is None + or "text_to_speech" not in workflow.features_dict + or not workflow.features_dict["text_to_speech"].get("enabled") + ): + raise ValueError("TTS is not enabled") + + voice = workflow.features_dict["text_to_speech"].get("voice") + else: + if not is_draft: + if app_model.app_model_config is None: + raise ValueError("AppModelConfig not found") + text_to_speech_dict = app_model.app_model_config.text_to_speech_dict + + if not text_to_speech_dict.get("enabled"): raise ValueError("TTS is not enabled") - voice = workflow.features_dict["text_to_speech"].get("voice") - else: - if not is_draft: - if app_model.app_model_config is None: - raise ValueError("AppModelConfig not found") - text_to_speech_dict = app_model.app_model_config.text_to_speech_dict + voice = text_to_speech_dict.get("voice") - if not text_to_speech_dict.get("enabled"): - raise ValueError("TTS is not enabled") - - voice = text_to_speech_dict.get("voice") - - model_manager = ModelManager() - model_instance = model_manager.get_default_model_instance( - tenant_id=app_model.tenant_id, model_type=ModelType.TTS - ) - try: - if not voice: - voices = model_instance.get_tts_voices() - if voices: - voice = voices[0].get("value") - if not voice: - raise ValueError("Sorry, no voice available.") - else: + model_manager = ModelManager() + model_instance = model_manager.get_default_model_instance( + tenant_id=app_model.tenant_id, model_type=ModelType.TTS + ) + try: + if not voice: + voices = model_instance.get_tts_voices() + if voices: + voice = voices[0].get("value") + if not voice: raise ValueError("Sorry, no voice available.") + else: + raise ValueError("Sorry, no voice available.") - return model_instance.invoke_tts( - content_text=text_content.strip(), user=end_user, tenant_id=app_model.tenant_id, voice=voice - ) - except Exception as e: - raise e + return model_instance.invoke_tts( + content_text=text_content.strip(), user=end_user, tenant_id=app_model.tenant_id, voice=voice + ) + except Exception as e: + raise e if message_id: try: diff --git a/api/services/billing_service.py b/api/services/billing_service.py index a6851d2638..1650bad0f5 100644 --- a/api/services/billing_service.py +++ b/api/services/billing_service.py @@ -4,6 +4,7 @@ from typing import Literal import httpx from tenacity import retry, retry_if_exception_type, stop_before_delay, wait_fixed +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from extensions.ext_redis import redis_client from libs.helper import RateLimiter @@ -31,7 +32,7 @@ class BillingService: return { "limit": knowledge_rate_limit.get("limit", 10), - "subscription_plan": knowledge_rate_limit.get("subscription_plan", "sandbox"), + "subscription_plan": knowledge_rate_limit.get("subscription_plan", CloudPlan.SANDBOX), } @classmethod diff --git a/api/services/clear_free_plan_tenant_expired_logs.py b/api/services/clear_free_plan_tenant_expired_logs.py index f8f89d7428..aefc34fcae 100644 --- a/api/services/clear_free_plan_tenant_expired_logs.py +++ b/api/services/clear_free_plan_tenant_expired_logs.py @@ -11,6 +11,7 @@ from sqlalchemy.orm import Session, sessionmaker from configs import dify_config from core.model_runtime.utils.encoders import jsonable_encoder +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from extensions.ext_storage import storage from models.account import Tenant @@ -358,7 +359,7 @@ class ClearFreePlanTenantExpiredLogs: try: if ( not dify_config.BILLING_ENABLED - or BillingService.get_info(tenant_id)["subscription"]["plan"] == "sandbox" + or BillingService.get_info(tenant_id)["subscription"]["plan"] == CloudPlan.SANDBOX ): # only process sandbox tenant cls.process_tenant(flask_app, tenant_id, days, batch) diff --git a/api/services/dataset_service.py b/api/services/dataset_service.py index c97d419545..78de76df7e 100644 --- a/api/services/dataset_service.py +++ b/api/services/dataset_service.py @@ -22,6 +22,7 @@ from core.model_runtime.entities.model_entities import ModelType from core.rag.index_processor.constant.built_in_field import BuiltInField from core.rag.index_processor.constant.index_type import IndexType from core.rag.retrieval.retrieval_methods import RetrievalMethod +from enums.cloud_plan import CloudPlan from events.dataset_event import dataset_was_deleted from events.document_event import document_was_deleted from extensions.ext_database import db @@ -49,6 +50,7 @@ from models.model import UploadFile from models.provider_ids import ModelProviderID from models.source import DataSourceOauthBinding from models.workflow import Workflow +from services.document_indexing_task_proxy import DocumentIndexingTaskProxy from services.entities.knowledge_entities.knowledge_entities import ( ChildChunkUpdateArgs, KnowledgeConfig, @@ -78,7 +80,6 @@ from tasks.deal_dataset_vector_index_task import deal_dataset_vector_index_task from tasks.delete_segment_from_index_task import delete_segment_from_index_task from tasks.disable_segment_from_index_task import disable_segment_from_index_task from tasks.disable_segments_from_index_task import disable_segments_from_index_task -from tasks.document_indexing_task import document_indexing_task from tasks.document_indexing_update_task import document_indexing_update_task from tasks.duplicate_document_indexing_task import duplicate_document_indexing_task from tasks.enable_segments_to_index_task import enable_segments_to_index_task @@ -1042,7 +1043,7 @@ class DatasetService: assert isinstance(current_user, Account) assert current_user.current_tenant_id is not None features = FeatureService.get_features(current_user.current_tenant_id) - if not features.billing.enabled or features.billing.subscription.plan == "sandbox": + if not features.billing.enabled or features.billing.subscription.plan == CloudPlan.SANDBOX: return { "document_ids": [], "count": 0, @@ -1416,8 +1417,6 @@ class DocumentService: # check document limit assert isinstance(current_user, Account) assert current_user.current_tenant_id is not None - assert knowledge_config.data_source - assert knowledge_config.data_source.info_list.file_info_list features = FeatureService.get_features(current_user.current_tenant_id) @@ -1426,6 +1425,8 @@ class DocumentService: count = 0 if knowledge_config.data_source: if knowledge_config.data_source.info_list.data_source_type == "upload_file": + if not knowledge_config.data_source.info_list.file_info_list: + raise ValueError("File source info is required") upload_file_list = knowledge_config.data_source.info_list.file_info_list.file_ids count = len(upload_file_list) elif knowledge_config.data_source.info_list.data_source_type == "notion_import": @@ -1438,7 +1439,7 @@ class DocumentService: count = len(website_info.urls) batch_upload_limit = int(dify_config.BATCH_UPLOAD_LIMIT) - if features.billing.subscription.plan == "sandbox" and count > 1: + if features.billing.subscription.plan == CloudPlan.SANDBOX and count > 1: raise ValueError("Your current plan does not support batch upload, please upgrade your plan.") if count > batch_upload_limit: raise ValueError(f"You have reached the batch upload limit of {batch_upload_limit}.") @@ -1446,7 +1447,7 @@ class DocumentService: DocumentService.check_documents_upload_quota(count, features) # if dataset is empty, update dataset data_source_type - if not dataset.data_source_type: + if not dataset.data_source_type and knowledge_config.data_source: dataset.data_source_type = knowledge_config.data_source.info_list.data_source_type if not dataset.indexing_technique: @@ -1492,6 +1493,10 @@ class DocumentService: documents.append(document) batch = document.batch else: + # When creating new documents, data_source must be provided + if not knowledge_config.data_source: + raise ValueError("Data source is required when creating new documents") + batch = time.strftime("%Y%m%d%H%M%S") + str(100000 + secrets.randbelow(exclusive_upper_bound=900000)) # save process rule if not dataset_process_rule: @@ -1531,6 +1536,8 @@ class DocumentService: document_ids = [] duplicate_document_ids = [] if knowledge_config.data_source.info_list.data_source_type == "upload_file": + if not knowledge_config.data_source.info_list.file_info_list: + raise ValueError("File source info is required") upload_file_list = knowledge_config.data_source.info_list.file_info_list.file_ids for file_id in upload_file_list: file = ( @@ -1687,7 +1694,7 @@ class DocumentService: # trigger async task if document_ids: - document_indexing_task.delay(dataset.id, document_ids) + DocumentIndexingTaskProxy(dataset.tenant_id, dataset.id, document_ids).delay() if duplicate_document_ids: duplicate_document_indexing_task.delay(dataset.id, duplicate_document_ids) @@ -1721,7 +1728,7 @@ class DocumentService: # count = len(website_info.urls) # type: ignore # batch_upload_limit = int(dify_config.BATCH_UPLOAD_LIMIT) - # if features.billing.subscription.plan == "sandbox" and count > 1: + # if features.billing.subscription.plan == CloudPlan.SANDBOX and count > 1: # raise ValueError("Your current plan does not support batch upload, please upgrade your plan.") # if count > batch_upload_limit: # raise ValueError(f"You have reached the batch upload limit of {batch_upload_limit}.") @@ -2190,7 +2197,7 @@ class DocumentService: website_info = knowledge_config.data_source.info_list.website_info_list if website_info: count = len(website_info.urls) - if features.billing.subscription.plan == "sandbox" and count > 1: + if features.billing.subscription.plan == CloudPlan.SANDBOX and count > 1: raise ValueError("Your current plan does not support batch upload, please upgrade your plan.") batch_upload_limit = int(dify_config.BATCH_UPLOAD_LIMIT) if count > batch_upload_limit: diff --git a/api/services/datasource_provider_service.py b/api/services/datasource_provider_service.py index 1b690e2266..81e0c0ecd4 100644 --- a/api/services/datasource_provider_service.py +++ b/api/services/datasource_provider_service.py @@ -11,9 +11,9 @@ from core.helper import encrypter from core.helper.name_generator import generate_incremental_name from core.helper.provider_cache import NoOpProviderCredentialCache from core.model_runtime.entities.provider_entities import FormType +from core.plugin.entities.plugin_daemon import CredentialType from core.plugin.impl.datasource import PluginDatasourceManager from core.plugin.impl.oauth import OAuthHandler -from core.tools.entities.tool_entities import CredentialType from core.tools.utils.encryption import ProviderConfigCache, ProviderConfigEncrypter, create_provider_encrypter from extensions.ext_database import db from extensions.ext_redis import redis_client @@ -338,7 +338,7 @@ class DatasourceProviderService: key: value if value != HIDDEN_VALUE else original_params.get(key, UNKNOWN_VALUE) for key, value in client_params.items() } - tenant_oauth_client_params.client_params = encrypter.encrypt(new_params) + tenant_oauth_client_params.client_params = dict(encrypter.encrypt(new_params)) if enabled is not None: tenant_oauth_client_params.enabled = enabled @@ -374,7 +374,7 @@ class DatasourceProviderService: def get_tenant_oauth_client( self, tenant_id: str, datasource_provider_id: DatasourceProviderID, mask: bool = False - ) -> dict[str, Any] | None: + ) -> Mapping[str, Any] | None: """ get tenant oauth client """ @@ -390,7 +390,7 @@ class DatasourceProviderService: if tenant_oauth_client_params: encrypter, _ = self.get_oauth_encrypter(tenant_id, datasource_provider_id) if mask: - return encrypter.mask_tool_credentials(encrypter.decrypt(tenant_oauth_client_params.client_params)) + return encrypter.mask_plugin_credentials(encrypter.decrypt(tenant_oauth_client_params.client_params)) else: return encrypter.decrypt(tenant_oauth_client_params.client_params) return None @@ -434,7 +434,7 @@ class DatasourceProviderService: ) if tenant_oauth_client_params: encrypter, _ = self.get_oauth_encrypter(tenant_id, datasource_provider_id) - return encrypter.decrypt(tenant_oauth_client_params.client_params) + return dict(encrypter.decrypt(tenant_oauth_client_params.client_params)) provider_controller = self.provider_manager.fetch_datasource_provider( tenant_id=tenant_id, provider_id=str(datasource_provider_id) diff --git a/api/services/document_indexing_task_proxy.py b/api/services/document_indexing_task_proxy.py new file mode 100644 index 0000000000..861c84b586 --- /dev/null +++ b/api/services/document_indexing_task_proxy.py @@ -0,0 +1,83 @@ +import logging +from collections.abc import Callable, Sequence +from dataclasses import asdict +from functools import cached_property + +from core.entities.document_task import DocumentTask +from core.rag.pipeline.queue import TenantIsolatedTaskQueue +from enums.cloud_plan import CloudPlan +from services.feature_service import FeatureService +from tasks.document_indexing_task import normal_document_indexing_task, priority_document_indexing_task + +logger = logging.getLogger(__name__) + + +class DocumentIndexingTaskProxy: + def __init__(self, tenant_id: str, dataset_id: str, document_ids: Sequence[str]): + self._tenant_id = tenant_id + self._dataset_id = dataset_id + self._document_ids = document_ids + self._tenant_isolated_task_queue = TenantIsolatedTaskQueue(tenant_id, "document_indexing") + + @cached_property + def features(self): + return FeatureService.get_features(self._tenant_id) + + def _send_to_direct_queue(self, task_func: Callable[[str, str, Sequence[str]], None]): + logger.info("send dataset %s to direct queue", self._dataset_id) + task_func.delay( # type: ignore + tenant_id=self._tenant_id, dataset_id=self._dataset_id, document_ids=self._document_ids + ) + + def _send_to_tenant_queue(self, task_func: Callable[[str, str, Sequence[str]], None]): + logger.info("send dataset %s to tenant queue", self._dataset_id) + if self._tenant_isolated_task_queue.get_task_key(): + # Add to waiting queue using List operations (lpush) + self._tenant_isolated_task_queue.push_tasks( + [ + asdict( + DocumentTask( + tenant_id=self._tenant_id, dataset_id=self._dataset_id, document_ids=self._document_ids + ) + ) + ] + ) + logger.info("push tasks: %s - %s", self._dataset_id, self._document_ids) + else: + # Set flag and execute task + self._tenant_isolated_task_queue.set_task_waiting_time() + task_func.delay( # type: ignore + tenant_id=self._tenant_id, dataset_id=self._dataset_id, document_ids=self._document_ids + ) + logger.info("init tasks: %s - %s", self._dataset_id, self._document_ids) + + def _send_to_default_tenant_queue(self): + self._send_to_tenant_queue(normal_document_indexing_task) + + def _send_to_priority_tenant_queue(self): + self._send_to_tenant_queue(priority_document_indexing_task) + + def _send_to_priority_direct_queue(self): + self._send_to_direct_queue(priority_document_indexing_task) + + def _dispatch(self): + logger.info( + "dispatch args: %s - %s - %s", + self._tenant_id, + self.features.billing.enabled, + self.features.billing.subscription.plan, + ) + # dispatch to different indexing queue with tenant isolation when billing enabled + if self.features.billing.enabled: + if self.features.billing.subscription.plan == CloudPlan.SANDBOX: + # dispatch to normal pipeline queue with tenant self sub queue for sandbox plan + self._send_to_default_tenant_queue() + else: + # dispatch to priority pipeline queue with tenant self sub queue for other plans + self._send_to_priority_tenant_queue() + else: + # dispatch to priority queue without tenant isolation for others, e.g.: self-hosted or enterprise + self._send_to_priority_direct_queue() + + def delay(self): + self._dispatch() diff --git a/api/services/end_user_service.py b/api/services/end_user_service.py new file mode 100644 index 0000000000..aa4a2e46ec --- /dev/null +++ b/api/services/end_user_service.py @@ -0,0 +1,141 @@ +from collections.abc import Mapping + +from sqlalchemy.orm import Session + +from core.app.entities.app_invoke_entities import InvokeFrom +from extensions.ext_database import db +from models.model import App, DefaultEndUserSessionID, EndUser + + +class EndUserService: + """ + Service for managing end users. + """ + + @classmethod + def get_or_create_end_user(cls, app_model: App, user_id: str | None = None) -> EndUser: + """ + Get or create an end user for a given app. + """ + + return cls.get_or_create_end_user_by_type(InvokeFrom.SERVICE_API, app_model.tenant_id, app_model.id, user_id) + + @classmethod + def get_or_create_end_user_by_type( + cls, type: InvokeFrom, tenant_id: str, app_id: str, user_id: str | None = None + ) -> EndUser: + """ + Get or create an end user for a given app and type. + """ + + if not user_id: + user_id = DefaultEndUserSessionID.DEFAULT_SESSION_ID + + with Session(db.engine, expire_on_commit=False) as session: + end_user = ( + session.query(EndUser) + .where( + EndUser.tenant_id == tenant_id, + EndUser.app_id == app_id, + EndUser.session_id == user_id, + EndUser.type == type, + ) + .first() + ) + + if end_user is None: + end_user = EndUser( + tenant_id=tenant_id, + app_id=app_id, + type=type, + is_anonymous=user_id == DefaultEndUserSessionID.DEFAULT_SESSION_ID, + session_id=user_id, + external_user_id=user_id, + ) + session.add(end_user) + session.commit() + + return end_user + + @classmethod + def create_end_user_batch( + cls, type: InvokeFrom, tenant_id: str, app_ids: list[str], user_id: str + ) -> Mapping[str, EndUser]: + """Create end users in batch. + + Creates end users in batch for the specified tenant and application IDs in O(1) time. + + This batch creation is necessary because trigger subscriptions can span multiple applications, + and trigger events may be dispatched to multiple applications simultaneously. + + For each app_id in app_ids, check if an `EndUser` with the given + `user_id` (as session_id/external_user_id) already exists for the + tenant/app and type `type`. If it exists, return it; otherwise, + create it. Operates with minimal DB I/O by querying and inserting in + batches. + + Returns a mapping of `app_id -> EndUser`. + """ + + # Normalize user_id to default if empty + if not user_id: + user_id = DefaultEndUserSessionID.DEFAULT_SESSION_ID + + # Deduplicate app_ids while preserving input order + seen: set[str] = set() + unique_app_ids: list[str] = [] + for app_id in app_ids: + if app_id not in seen: + seen.add(app_id) + unique_app_ids.append(app_id) + + # Result is a simple app_id -> EndUser mapping + result: dict[str, EndUser] = {} + if not unique_app_ids: + return result + + with Session(db.engine, expire_on_commit=False) as session: + # Fetch existing end users for all target apps in a single query + existing_end_users: list[EndUser] = ( + session.query(EndUser) + .where( + EndUser.tenant_id == tenant_id, + EndUser.app_id.in_(unique_app_ids), + EndUser.session_id == user_id, + EndUser.type == type, + ) + .all() + ) + + found_app_ids: set[str] = set() + for eu in existing_end_users: + # If duplicates exist due to weak DB constraints, prefer the first + if eu.app_id not in result: + result[eu.app_id] = eu + found_app_ids.add(eu.app_id) + + # Determine which apps still need an EndUser created + missing_app_ids = [app_id for app_id in unique_app_ids if app_id not in found_app_ids] + + if missing_app_ids: + new_end_users: list[EndUser] = [] + is_anonymous = user_id == DefaultEndUserSessionID.DEFAULT_SESSION_ID + for app_id in missing_app_ids: + new_end_users.append( + EndUser( + tenant_id=tenant_id, + app_id=app_id, + type=type, + is_anonymous=is_anonymous, + session_id=user_id, + external_user_id=user_id, + ) + ) + + session.add_all(new_end_users) + session.commit() + + for eu in new_end_users: + result[eu.app_id] = eu + + return result diff --git a/api/services/enterprise/enterprise_service.py b/api/services/enterprise/enterprise_service.py index 974aa849db..83d0fcf296 100644 --- a/api/services/enterprise/enterprise_service.py +++ b/api/services/enterprise/enterprise_service.py @@ -92,16 +92,6 @@ class EnterpriseService: return ret - @classmethod - def get_app_access_mode_by_code(cls, app_code: str) -> WebAppSettings: - if not app_code: - raise ValueError("app_code must be provided.") - params = {"appCode": app_code} - data = EnterpriseRequest.send_request("GET", "/webapp/access-mode/code", params=params) - if not data: - raise ValueError("No data found.") - return WebAppSettings.model_validate(data) - @classmethod def update_app_access_mode(cls, app_id: str, access_mode: str): if not app_id: diff --git a/api/services/errors/app.py b/api/services/errors/app.py index 390716a47f..338636d9b6 100644 --- a/api/services/errors/app.py +++ b/api/services/errors/app.py @@ -16,3 +16,9 @@ class WorkflowNotFoundError(Exception): class WorkflowIdFormatError(Exception): pass + + +class InvokeDailyRateLimitError(Exception): + """Raised when daily rate limit is exceeded for workflow invocations.""" + + pass diff --git a/api/services/errors/file.py b/api/services/errors/file.py index 29f3f44eec..bf9d65a25b 100644 --- a/api/services/errors/file.py +++ b/api/services/errors/file.py @@ -11,3 +11,7 @@ class FileTooLargeError(BaseServiceError): class UnsupportedFileTypeError(BaseServiceError): pass + + +class BlockedFileExtensionError(BaseServiceError): + description = "File extension '{extension}' is not allowed for security reasons" diff --git a/api/services/feature_service.py b/api/services/feature_service.py index 148442f76e..44bea57769 100644 --- a/api/services/feature_service.py +++ b/api/services/feature_service.py @@ -3,12 +3,13 @@ from enum import StrEnum from pydantic import BaseModel, ConfigDict, Field from configs import dify_config +from enums.cloud_plan import CloudPlan from services.billing_service import BillingService from services.enterprise.enterprise_service import EnterpriseService class SubscriptionModel(BaseModel): - plan: str = "sandbox" + plan: str = CloudPlan.SANDBOX interval: str = "" @@ -186,7 +187,7 @@ class FeatureService: knowledge_rate_limit.enabled = True limit_info = BillingService.get_knowledge_rate_limit(tenant_id) knowledge_rate_limit.limit = limit_info.get("limit", 10) - knowledge_rate_limit.subscription_plan = limit_info.get("subscription_plan", "sandbox") + knowledge_rate_limit.subscription_plan = limit_info.get("subscription_plan", CloudPlan.SANDBOX) return knowledge_rate_limit @classmethod @@ -240,7 +241,7 @@ class FeatureService: features.billing.subscription.interval = billing_info["subscription"]["interval"] features.education.activated = billing_info["subscription"].get("education", False) - if features.billing.subscription.plan != "sandbox": + if features.billing.subscription.plan != CloudPlan.SANDBOX: features.webapp_copyright_enabled = True else: features.is_allow_transfer_workspace = False diff --git a/api/services/file_service.py b/api/services/file_service.py index dd6a829ea2..b0c5a32c9f 100644 --- a/api/services/file_service.py +++ b/api/services/file_service.py @@ -23,7 +23,7 @@ from models import Account from models.enums import CreatorUserRole from models.model import EndUser, UploadFile -from .errors.file import FileTooLargeError, UnsupportedFileTypeError +from .errors.file import BlockedFileExtensionError, FileTooLargeError, UnsupportedFileTypeError PREVIEW_WORDS_LIMIT = 3000 @@ -59,6 +59,10 @@ class FileService: if len(filename) > 200: filename = filename.split(".")[0][:200] + "." + extension + # check if extension is in blacklist + if extension and extension in dify_config.UPLOAD_FILE_EXTENSION_BLACKLIST: + raise BlockedFileExtensionError(f"File extension '.{extension}' is not allowed for security reasons") + if source == "datasets" and extension not in DOCUMENT_EXTENSIONS: raise UnsupportedFileTypeError() diff --git a/api/services/plugin/oauth_service.py b/api/services/plugin/oauth_service.py index 057b20428f..88dec062a0 100644 --- a/api/services/plugin/oauth_service.py +++ b/api/services/plugin/oauth_service.py @@ -16,6 +16,7 @@ class OAuthProxyService(BasePluginClient): tenant_id: str, plugin_id: str, provider: str, + extra_data: dict = {}, credential_id: str | None = None, ): """ @@ -32,6 +33,7 @@ class OAuthProxyService(BasePluginClient): """ context_id = str(uuid.uuid4()) data = { + **extra_data, "user_id": user_id, "plugin_id": plugin_id, "tenant_id": tenant_id, diff --git a/api/services/plugin/plugin_parameter_service.py b/api/services/plugin/plugin_parameter_service.py index 00b59dacb3..c517d9f966 100644 --- a/api/services/plugin/plugin_parameter_service.py +++ b/api/services/plugin/plugin_parameter_service.py @@ -4,11 +4,16 @@ from typing import Any, Literal from sqlalchemy.orm import Session from core.plugin.entities.parameters import PluginParameterOption +from core.plugin.entities.plugin_daemon import CredentialType from core.plugin.impl.dynamic_select import DynamicSelectClient from core.tools.tool_manager import ToolManager from core.tools.utils.encryption import create_tool_provider_encrypter +from core.trigger.entities.api_entities import TriggerProviderSubscriptionApiEntity +from core.trigger.entities.entities import SubscriptionBuilder from extensions.ext_database import db from models.tools import BuiltinToolProvider +from services.trigger.trigger_provider_service import TriggerProviderService +from services.trigger.trigger_subscription_builder_service import TriggerSubscriptionBuilderService class PluginParameterService: @@ -20,7 +25,8 @@ class PluginParameterService: provider: str, action: str, parameter: str, - provider_type: Literal["tool"], + credential_id: str | None, + provider_type: Literal["tool", "trigger"], ) -> Sequence[PluginParameterOption]: """ Get dynamic select options for a plugin parameter. @@ -33,7 +39,7 @@ class PluginParameterService: parameter: The parameter name. """ credentials: Mapping[str, Any] = {} - + credential_type: str = CredentialType.UNAUTHORIZED.value match provider_type: case "tool": provider_controller = ToolManager.get_builtin_provider(provider, tenant_id) @@ -49,24 +55,53 @@ class PluginParameterService: else: # fetch credentials from db with Session(db.engine) as session: - db_record = ( - session.query(BuiltinToolProvider) - .where( - BuiltinToolProvider.tenant_id == tenant_id, - BuiltinToolProvider.provider == provider, + if credential_id: + db_record = ( + session.query(BuiltinToolProvider) + .where( + BuiltinToolProvider.tenant_id == tenant_id, + BuiltinToolProvider.provider == provider, + BuiltinToolProvider.id == credential_id, + ) + .first() + ) + else: + db_record = ( + session.query(BuiltinToolProvider) + .where( + BuiltinToolProvider.tenant_id == tenant_id, + BuiltinToolProvider.provider == provider, + ) + .order_by(BuiltinToolProvider.is_default.desc(), BuiltinToolProvider.created_at.asc()) + .first() ) - .first() - ) if db_record is None: raise ValueError(f"Builtin provider {provider} not found when fetching credentials") credentials = encrypter.decrypt(db_record.credentials) - case _: - raise ValueError(f"Invalid provider type: {provider_type}") + credential_type = db_record.credential_type + case "trigger": + subscription: TriggerProviderSubscriptionApiEntity | SubscriptionBuilder | None + if credential_id: + subscription = TriggerSubscriptionBuilderService.get_subscription_builder(credential_id) + if not subscription: + trigger_subscription = TriggerProviderService.get_subscription_by_id(tenant_id, credential_id) + subscription = trigger_subscription.to_api_entity() if trigger_subscription else None + else: + trigger_subscription = TriggerProviderService.get_subscription_by_id(tenant_id) + subscription = trigger_subscription.to_api_entity() if trigger_subscription else None + + if subscription is None: + raise ValueError(f"Subscription {credential_id} not found") + + credentials = subscription.credentials + credential_type = subscription.credential_type or CredentialType.UNAUTHORIZED return ( DynamicSelectClient() - .fetch_dynamic_select_options(tenant_id, user_id, plugin_id, provider, action, credentials, parameter) + .fetch_dynamic_select_options( + tenant_id, user_id, plugin_id, provider, action, credentials, credential_type, parameter + ) .options ) diff --git a/api/services/plugin/plugin_service.py b/api/services/plugin/plugin_service.py index 525ccc9417..b8303eb724 100644 --- a/api/services/plugin/plugin_service.py +++ b/api/services/plugin/plugin_service.py @@ -3,6 +3,7 @@ from collections.abc import Mapping, Sequence from mimetypes import guess_type from pydantic import BaseModel +from yarl import URL from configs import dify_config from core.helper import marketplace @@ -175,6 +176,13 @@ class PluginService: manager = PluginInstaller() return manager.fetch_plugin_installation_by_ids(tenant_id, ids) + @classmethod + def get_plugin_icon_url(cls, tenant_id: str, filename: str) -> str: + url_prefix = ( + URL(dify_config.CONSOLE_API_URL or "/") / "console" / "api" / "workspaces" / "current" / "plugin" / "icon" + ) + return str(url_prefix % {"tenant_id": tenant_id, "filename": filename}) + @staticmethod def get_asset(tenant_id: str, asset_file: str) -> tuple[bytes, str]: """ @@ -185,6 +193,11 @@ class PluginService: mime_type, _ = guess_type(asset_file) return manager.fetch_asset(tenant_id, asset_file), mime_type or "application/octet-stream" + @staticmethod + def extract_asset(tenant_id: str, plugin_unique_identifier: str, file_name: str) -> bytes: + manager = PluginAssetManager() + return manager.extract_asset(tenant_id, plugin_unique_identifier, file_name) + @staticmethod def check_plugin_unique_identifier(tenant_id: str, plugin_unique_identifier: str) -> bool: """ @@ -502,3 +515,11 @@ class PluginService: """ manager = PluginInstaller() return manager.check_tools_existence(tenant_id, provider_ids) + + @staticmethod + def fetch_plugin_readme(tenant_id: str, plugin_unique_identifier: str, language: str) -> str: + """ + Fetch plugin readme + """ + manager = PluginInstaller() + return manager.fetch_plugin_readme(tenant_id, plugin_unique_identifier, language) diff --git a/api/services/rag_pipeline/rag_pipeline.py b/api/services/rag_pipeline/rag_pipeline.py index f6dddd75a3..fed7a25e21 100644 --- a/api/services/rag_pipeline/rag_pipeline.py +++ b/api/services/rag_pipeline/rag_pipeline.py @@ -9,7 +9,7 @@ from typing import Any, Union, cast from uuid import uuid4 from flask_login import current_user -from sqlalchemy import func, or_, select +from sqlalchemy import func, select from sqlalchemy.orm import Session, sessionmaker import contexts @@ -94,6 +94,7 @@ class RagPipelineService: self._node_execution_service_repo = DifyAPIRepositoryFactory.create_api_workflow_node_execution_repository( session_maker ) + self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) @classmethod def get_pipeline_templates(cls, type: str = "built-in", language: str = "en-US") -> dict: @@ -1015,48 +1016,21 @@ class RagPipelineService: :param args: request args """ limit = int(args.get("limit", 20)) + last_id = args.get("last_id") - base_query = db.session.query(WorkflowRun).where( - WorkflowRun.tenant_id == pipeline.tenant_id, - WorkflowRun.app_id == pipeline.id, - or_( - WorkflowRun.triggered_from == WorkflowRunTriggeredFrom.RAG_PIPELINE_RUN.value, - WorkflowRun.triggered_from == WorkflowRunTriggeredFrom.RAG_PIPELINE_DEBUGGING.value, - ), + triggered_from_values = [ + WorkflowRunTriggeredFrom.RAG_PIPELINE_RUN, + WorkflowRunTriggeredFrom.RAG_PIPELINE_DEBUGGING, + ] + + return self._workflow_run_repo.get_paginated_workflow_runs( + tenant_id=pipeline.tenant_id, + app_id=pipeline.id, + triggered_from=triggered_from_values, + limit=limit, + last_id=last_id, ) - if args.get("last_id"): - last_workflow_run = base_query.where( - WorkflowRun.id == args.get("last_id"), - ).first() - - if not last_workflow_run: - raise ValueError("Last workflow run not exists") - - workflow_runs = ( - base_query.where( - WorkflowRun.created_at < last_workflow_run.created_at, WorkflowRun.id != last_workflow_run.id - ) - .order_by(WorkflowRun.created_at.desc()) - .limit(limit) - .all() - ) - else: - workflow_runs = base_query.order_by(WorkflowRun.created_at.desc()).limit(limit).all() - - has_more = False - if len(workflow_runs) == limit: - current_page_first_workflow_run = workflow_runs[-1] - rest_count = base_query.where( - WorkflowRun.created_at < current_page_first_workflow_run.created_at, - WorkflowRun.id != current_page_first_workflow_run.id, - ).count() - - if rest_count > 0: - has_more = True - - return InfiniteScrollPagination(data=workflow_runs, limit=limit, has_more=has_more) - def get_rag_pipeline_workflow_run(self, pipeline: Pipeline, run_id: str) -> WorkflowRun | None: """ Get workflow run detail @@ -1064,18 +1038,12 @@ class RagPipelineService: :param app_model: app model :param run_id: workflow run id """ - workflow_run = ( - db.session.query(WorkflowRun) - .where( - WorkflowRun.tenant_id == pipeline.tenant_id, - WorkflowRun.app_id == pipeline.id, - WorkflowRun.id == run_id, - ) - .first() + return self._workflow_run_repo.get_workflow_run_by_id( + tenant_id=pipeline.tenant_id, + app_id=pipeline.id, + run_id=run_id, ) - return workflow_run - def get_rag_pipeline_workflow_run_node_executions( self, pipeline: Pipeline, @@ -1297,8 +1265,8 @@ class RagPipelineService: ) providers_map = {provider.plugin_id: provider.to_dict() for provider in providers} - plugin_manifests = marketplace.batch_fetch_plugin_manifests(plugin_ids) - plugin_manifests_map = {manifest.plugin_id: manifest for manifest in plugin_manifests} + plugin_manifests = marketplace.batch_fetch_plugin_by_ids(plugin_ids) + plugin_manifests_map = {manifest["plugin_id"]: manifest for manifest in plugin_manifests} installed_plugin_list = [] uninstalled_plugin_list = [] @@ -1308,14 +1276,7 @@ class RagPipelineService: else: plugin_manifest = plugin_manifests_map.get(plugin_id) if plugin_manifest: - uninstalled_plugin_list.append( - { - "plugin_id": plugin_id, - "name": plugin_manifest.name, - "icon": plugin_manifest.icon, - "plugin_unique_identifier": plugin_manifest.latest_package_identifier, - } - ) + uninstalled_plugin_list.append(plugin_manifest) # Build recommended plugins list return { diff --git a/api/services/rag_pipeline/rag_pipeline_task_proxy.py b/api/services/rag_pipeline/rag_pipeline_task_proxy.py new file mode 100644 index 0000000000..94dd7941da --- /dev/null +++ b/api/services/rag_pipeline/rag_pipeline_task_proxy.py @@ -0,0 +1,106 @@ +import json +import logging +from collections.abc import Callable, Sequence +from functools import cached_property + +from core.app.entities.rag_pipeline_invoke_entities import RagPipelineInvokeEntity +from core.rag.pipeline.queue import TenantIsolatedTaskQueue +from enums.cloud_plan import CloudPlan +from extensions.ext_database import db +from services.feature_service import FeatureService +from services.file_service import FileService +from tasks.rag_pipeline.priority_rag_pipeline_run_task import priority_rag_pipeline_run_task +from tasks.rag_pipeline.rag_pipeline_run_task import rag_pipeline_run_task + +logger = logging.getLogger(__name__) + + +class RagPipelineTaskProxy: + # Default uploaded file name for rag pipeline invoke entities + _RAG_PIPELINE_INVOKE_ENTITIES_FILE_NAME = "rag_pipeline_invoke_entities.json" + + def __init__( + self, dataset_tenant_id: str, user_id: str, rag_pipeline_invoke_entities: Sequence[RagPipelineInvokeEntity] + ): + self._dataset_tenant_id = dataset_tenant_id + self._user_id = user_id + self._rag_pipeline_invoke_entities = rag_pipeline_invoke_entities + self._tenant_isolated_task_queue = TenantIsolatedTaskQueue(dataset_tenant_id, "pipeline") + + @cached_property + def features(self): + return FeatureService.get_features(self._dataset_tenant_id) + + def _upload_invoke_entities(self) -> str: + text = [item.model_dump() for item in self._rag_pipeline_invoke_entities] + # Convert list to proper JSON string + json_text = json.dumps(text) + upload_file = FileService(db.engine).upload_text( + json_text, self._RAG_PIPELINE_INVOKE_ENTITIES_FILE_NAME, self._user_id, self._dataset_tenant_id + ) + return upload_file.id + + def _send_to_direct_queue(self, upload_file_id: str, task_func: Callable[[str, str], None]): + logger.info("send file %s to direct queue", upload_file_id) + task_func.delay( # type: ignore + rag_pipeline_invoke_entities_file_id=upload_file_id, + tenant_id=self._dataset_tenant_id, + ) + + def _send_to_tenant_queue(self, upload_file_id: str, task_func: Callable[[str, str], None]): + logger.info("send file %s to tenant queue", upload_file_id) + if self._tenant_isolated_task_queue.get_task_key(): + # Add to waiting queue using List operations (lpush) + self._tenant_isolated_task_queue.push_tasks([upload_file_id]) + logger.info("push tasks: %s", upload_file_id) + else: + # Set flag and execute task + self._tenant_isolated_task_queue.set_task_waiting_time() + task_func.delay( # type: ignore + rag_pipeline_invoke_entities_file_id=upload_file_id, + tenant_id=self._dataset_tenant_id, + ) + logger.info("init tasks: %s", upload_file_id) + + def _send_to_default_tenant_queue(self, upload_file_id: str): + self._send_to_tenant_queue(upload_file_id, rag_pipeline_run_task) + + def _send_to_priority_tenant_queue(self, upload_file_id: str): + self._send_to_tenant_queue(upload_file_id, priority_rag_pipeline_run_task) + + def _send_to_priority_direct_queue(self, upload_file_id: str): + self._send_to_direct_queue(upload_file_id, priority_rag_pipeline_run_task) + + def _dispatch(self): + upload_file_id = self._upload_invoke_entities() + if not upload_file_id: + raise ValueError("upload_file_id is empty") + + logger.info( + "dispatch args: %s - %s - %s", + self._dataset_tenant_id, + self.features.billing.enabled, + self.features.billing.subscription.plan, + ) + + # dispatch to different pipeline queue with tenant isolation when billing enabled + if self.features.billing.enabled: + if self.features.billing.subscription.plan == CloudPlan.SANDBOX: + # dispatch to normal pipeline queue with tenant isolation for sandbox plan + self._send_to_default_tenant_queue(upload_file_id) + else: + # dispatch to priority pipeline queue with tenant isolation for other plans + self._send_to_priority_tenant_queue(upload_file_id) + else: + # dispatch to priority pipeline queue without tenant isolation for others, e.g.: self-hosted or enterprise + self._send_to_priority_direct_queue(upload_file_id) + + def delay(self): + if not self._rag_pipeline_invoke_entities: + logger.warning( + "Received empty rag pipeline invoke entities, no tasks delivered: %s %s", + self._dataset_tenant_id, + self._user_id, + ) + return + self._dispatch() diff --git a/api/services/rag_pipeline/transform/website-crawl-general-economy.yml b/api/services/rag_pipeline/transform/website-crawl-general-economy.yml index 241d94c95d..a0f4b3bdd8 100644 --- a/api/services/rag_pipeline/transform/website-crawl-general-economy.yml +++ b/api/services/rag_pipeline/transform/website-crawl-general-economy.yml @@ -126,7 +126,7 @@ workflow: type: mixed value: '{{#rag.1752491761974.jina_use_sitemap#}}' plugin_id: langgenius/jina_datasource - provider_name: jina + provider_name: jinareader provider_type: website_crawl selected: false title: Jina Reader diff --git a/api/services/rag_pipeline/transform/website-crawl-general-high-quality.yml b/api/services/rag_pipeline/transform/website-crawl-general-high-quality.yml index 52b8f822c0..f58679fb6c 100644 --- a/api/services/rag_pipeline/transform/website-crawl-general-high-quality.yml +++ b/api/services/rag_pipeline/transform/website-crawl-general-high-quality.yml @@ -126,7 +126,7 @@ workflow: type: mixed value: '{{#rag.1752491761974.jina_use_sitemap#}}' plugin_id: langgenius/jina_datasource - provider_name: jina + provider_name: jinareader provider_type: website_crawl selected: false title: Jina Reader diff --git a/api/services/rag_pipeline/transform/website-crawl-parentchild.yml b/api/services/rag_pipeline/transform/website-crawl-parentchild.yml index 5d609bd12b..85b1cfd87d 100644 --- a/api/services/rag_pipeline/transform/website-crawl-parentchild.yml +++ b/api/services/rag_pipeline/transform/website-crawl-parentchild.yml @@ -419,7 +419,7 @@ workflow: type: mixed value: '{{#rag.1752491761974.jina_use_sitemap#}}' plugin_id: langgenius/jina_datasource - provider_name: jina + provider_name: jinareader provider_type: website_crawl selected: false title: Jina Reader diff --git a/api/services/tools/api_tools_manage_service.py b/api/services/tools/api_tools_manage_service.py index bb024cc846..250d29f335 100644 --- a/api/services/tools/api_tools_manage_service.py +++ b/api/services/tools/api_tools_manage_service.py @@ -300,13 +300,13 @@ class ApiToolManageService: ) original_credentials = encrypter.decrypt(provider.credentials) - masked_credentials = encrypter.mask_tool_credentials(original_credentials) + masked_credentials = encrypter.mask_plugin_credentials(original_credentials) # check if the credential has changed, save the original credential for name, value in credentials.items(): if name in masked_credentials and value == masked_credentials[name]: credentials[name] = original_credentials[name] - credentials = encrypter.encrypt(credentials) + credentials = dict(encrypter.encrypt(credentials)) provider.credentials_str = json.dumps(credentials) db.session.add(provider) @@ -417,7 +417,7 @@ class ApiToolManageService: ) decrypted_credentials = encrypter.decrypt(credentials) # check if the credential has changed, save the original credential - masked_credentials = encrypter.mask_tool_credentials(decrypted_credentials) + masked_credentials = encrypter.mask_plugin_credentials(decrypted_credentials) for name, value in credentials.items(): if name in masked_credentials and value == masked_credentials[name]: credentials[name] = decrypted_credentials[name] diff --git a/api/services/tools/builtin_tools_manage_service.py b/api/services/tools/builtin_tools_manage_service.py index 0628c8f22e..783f2f0d21 100644 --- a/api/services/tools/builtin_tools_manage_service.py +++ b/api/services/tools/builtin_tools_manage_service.py @@ -12,6 +12,7 @@ from constants import HIDDEN_VALUE, UNKNOWN_VALUE from core.helper.name_generator import generate_incremental_name from core.helper.position_helper import is_filtered from core.helper.provider_cache import NoOpProviderCredentialCache, ToolProviderCredentialsCache +from core.plugin.entities.plugin_daemon import CredentialType from core.tools.builtin_tool.provider import BuiltinToolProviderController from core.tools.builtin_tool.providers._positions import BuiltinToolProviderSort from core.tools.entities.api_entities import ( @@ -20,7 +21,6 @@ from core.tools.entities.api_entities import ( ToolProviderCredentialApiEntity, ToolProviderCredentialInfoApiEntity, ) -from core.tools.entities.tool_entities import CredentialType from core.tools.errors import ToolProviderNotFoundError from core.tools.plugin_tool.provider import PluginToolProviderController from core.tools.tool_label_manager import ToolLabelManager @@ -39,7 +39,6 @@ logger = logging.getLogger(__name__) class BuiltinToolManageService: __MAX_BUILTIN_TOOL_PROVIDER_COUNT__ = 100 - __DEFAULT_EXPIRES_AT__ = 2147483647 @staticmethod def delete_custom_oauth_client_params(tenant_id: str, provider: str): @@ -278,9 +277,7 @@ class BuiltinToolManageService: encrypted_credentials=json.dumps(encrypter.encrypt(credentials)), credential_type=api_type.value, name=name, - expires_at=expires_at - if expires_at is not None - else BuiltinToolManageService.__DEFAULT_EXPIRES_AT__, + expires_at=expires_at if expires_at is not None else -1, ) session.add(db_provider) @@ -353,10 +350,10 @@ class BuiltinToolManageService: encrypter, _ = BuiltinToolManageService.create_tool_encrypter( tenant_id, provider, provider.provider, provider_controller ) - decrypt_credential = encrypter.mask_tool_credentials(encrypter.decrypt(provider.credentials)) + decrypt_credential = encrypter.mask_plugin_credentials(encrypter.decrypt(provider.credentials)) credential_entity = ToolTransformService.convert_builtin_provider_to_credential_entity( provider=provider, - credentials=decrypt_credential, + credentials=dict(decrypt_credential), ) credentials.append(credential_entity) return credentials @@ -727,4 +724,4 @@ class BuiltinToolManageService: cache=NoOpProviderCredentialCache(), ) - return encrypter.mask_tool_credentials(encrypter.decrypt(custom_oauth_client_params.oauth_params)) + return encrypter.mask_plugin_credentials(encrypter.decrypt(custom_oauth_client_params.oauth_params)) diff --git a/api/services/tools/mcp_tools_manage_service.py b/api/services/tools/mcp_tools_manage_service.py index 92c33c1a49..d798e11ff1 100644 --- a/api/services/tools/mcp_tools_manage_service.py +++ b/api/services/tools/mcp_tools_manage_service.py @@ -1,86 +1,119 @@ import hashlib import json +import logging +from collections.abc import Mapping from datetime import datetime +from enum import StrEnum from typing import Any +from urllib.parse import urlparse -from sqlalchemy import or_ +from pydantic import BaseModel, Field +from sqlalchemy import or_, select from sqlalchemy.exc import IntegrityError +from sqlalchemy.orm import Session +from core.entities.mcp_provider import MCPAuthentication, MCPConfiguration, MCPProviderEntity from core.helper import encrypter from core.helper.provider_cache import NoOpProviderCredentialCache +from core.mcp.auth.auth_flow import auth +from core.mcp.auth_client import MCPClientWithAuthRetry from core.mcp.error import MCPAuthError, MCPError -from core.mcp.mcp_client import MCPClient from core.tools.entities.api_entities import ToolProviderApiEntity -from core.tools.entities.common_entities import I18nObject -from core.tools.entities.tool_entities import ToolProviderType -from core.tools.mcp_tool.provider import MCPToolProviderController from core.tools.utils.encryption import ProviderConfigEncrypter -from extensions.ext_database import db from models.tools import MCPToolProvider from services.tools.tools_transform_service import ToolTransformService +logger = logging.getLogger(__name__) + +# Constants UNCHANGED_SERVER_URL_PLACEHOLDER = "[__HIDDEN__]" +CLIENT_NAME = "Dify" +EMPTY_TOOLS_JSON = "[]" +EMPTY_CREDENTIALS_JSON = "{}" + + +class OAuthDataType(StrEnum): + """Types of OAuth data that can be saved.""" + + TOKENS = "tokens" + CLIENT_INFO = "client_info" + CODE_VERIFIER = "code_verifier" + MIXED = "mixed" + + +class ReconnectResult(BaseModel): + """Result of reconnecting to an MCP provider""" + + authed: bool = Field(description="Whether the provider is authenticated") + tools: str = Field(description="JSON string of tool list") + encrypted_credentials: str = Field(description="JSON string of encrypted credentials") + + +class ServerUrlValidationResult(BaseModel): + """Result of server URL validation check""" + + needs_validation: bool + validation_passed: bool = False + reconnect_result: ReconnectResult | None = None + encrypted_server_url: str | None = None + server_url_hash: str | None = None + + @property + def should_update_server_url(self) -> bool: + """Check if server URL should be updated based on validation result""" + return self.needs_validation and self.validation_passed and self.reconnect_result is not None class MCPToolManageService: - """ - Service class for managing mcp tools. - """ + """Service class for managing MCP tools and providers.""" - @staticmethod - def _encrypt_headers(headers: dict[str, str], tenant_id: str) -> dict[str, str]: + def __init__(self, session: Session): + self._session = session + + # ========== Provider CRUD Operations ========== + + def get_provider( + self, *, provider_id: str | None = None, server_identifier: str | None = None, tenant_id: str + ) -> MCPToolProvider: """ - Encrypt headers using ProviderConfigEncrypter with all headers as SECRET_INPUT. + Get MCP provider by ID or server identifier. Args: - headers: Dictionary of headers to encrypt - tenant_id: Tenant ID for encryption + provider_id: Provider ID (UUID) + server_identifier: Server identifier + tenant_id: Tenant ID Returns: - Dictionary with all headers encrypted + MCPToolProvider instance + + Raises: + ValueError: If provider not found """ - if not headers: - return {} + if server_identifier: + stmt = select(MCPToolProvider).where( + MCPToolProvider.tenant_id == tenant_id, MCPToolProvider.server_identifier == server_identifier + ) + else: + stmt = select(MCPToolProvider).where( + MCPToolProvider.tenant_id == tenant_id, MCPToolProvider.id == provider_id + ) - from core.entities.provider_entities import BasicProviderConfig - from core.helper.provider_cache import NoOpProviderCredentialCache - from core.tools.utils.encryption import create_provider_encrypter - - # Create dynamic config for all headers as SECRET_INPUT - config = [BasicProviderConfig(type=BasicProviderConfig.Type.SECRET_INPUT, name=key) for key in headers] - - encrypter_instance, _ = create_provider_encrypter( - tenant_id=tenant_id, - config=config, - cache=NoOpProviderCredentialCache(), - ) - - return encrypter_instance.encrypt(headers) - - @staticmethod - def get_mcp_provider_by_provider_id(provider_id: str, tenant_id: str) -> MCPToolProvider: - res = ( - db.session.query(MCPToolProvider) - .where(MCPToolProvider.tenant_id == tenant_id, MCPToolProvider.id == provider_id) - .first() - ) - if not res: + provider = self._session.scalar(stmt) + if not provider: raise ValueError("MCP tool not found") - return res + return provider - @staticmethod - def get_mcp_provider_by_server_identifier(server_identifier: str, tenant_id: str) -> MCPToolProvider: - res = ( - db.session.query(MCPToolProvider) - .where(MCPToolProvider.tenant_id == tenant_id, MCPToolProvider.server_identifier == server_identifier) - .first() - ) - if not res: - raise ValueError("MCP tool not found") - return res + def get_provider_entity(self, provider_id: str, tenant_id: str, by_server_id: bool = False) -> MCPProviderEntity: + """Get provider entity by ID or server identifier.""" + if by_server_id: + db_provider = self.get_provider(server_identifier=provider_id, tenant_id=tenant_id) + else: + db_provider = self.get_provider(provider_id=provider_id, tenant_id=tenant_id) + return db_provider.to_entity() - @staticmethod - def create_mcp_provider( + def create_provider( + self, + *, tenant_id: str, name: str, server_url: str, @@ -89,37 +122,30 @@ class MCPToolManageService: icon_type: str, icon_background: str, server_identifier: str, - timeout: float, - sse_read_timeout: float, + configuration: MCPConfiguration, + authentication: MCPAuthentication | None = None, headers: dict[str, str] | None = None, ) -> ToolProviderApiEntity: - server_url_hash = hashlib.sha256(server_url.encode()).hexdigest() - existing_provider = ( - db.session.query(MCPToolProvider) - .where( - MCPToolProvider.tenant_id == tenant_id, - or_( - MCPToolProvider.name == name, - MCPToolProvider.server_url_hash == server_url_hash, - MCPToolProvider.server_identifier == server_identifier, - ), - ) - .first() - ) - if existing_provider: - if existing_provider.name == name: - raise ValueError(f"MCP tool {name} already exists") - if existing_provider.server_url_hash == server_url_hash: - raise ValueError(f"MCP tool {server_url} already exists") - if existing_provider.server_identifier == server_identifier: - raise ValueError(f"MCP tool {server_identifier} already exists") - encrypted_server_url = encrypter.encrypt_token(tenant_id, server_url) - # Encrypt headers - encrypted_headers = None - if headers: - encrypted_headers_dict = MCPToolManageService._encrypt_headers(headers, tenant_id) - encrypted_headers = json.dumps(encrypted_headers_dict) + """Create a new MCP provider.""" + # Validate URL format + if not self._is_valid_url(server_url): + raise ValueError("Server URL is not valid.") + server_url_hash = hashlib.sha256(server_url.encode()).hexdigest() + + # Check for existing provider + self._check_provider_exists(tenant_id, name, server_url_hash, server_identifier) + + # Encrypt sensitive data + encrypted_server_url = encrypter.encrypt_token(tenant_id, server_url) + encrypted_headers = self._prepare_encrypted_dict(headers, tenant_id) if headers else None + encrypted_credentials = None + if authentication is not None and authentication.client_id: + encrypted_credentials = self._build_and_encrypt_credentials( + authentication.client_id, authentication.client_secret, tenant_id + ) + + # Create provider mcp_tool = MCPToolProvider( tenant_id=tenant_id, name=name, @@ -127,93 +153,23 @@ class MCPToolManageService: server_url_hash=server_url_hash, user_id=user_id, authed=False, - tools="[]", - icon=json.dumps({"content": icon, "background": icon_background}) if icon_type == "emoji" else icon, + tools=EMPTY_TOOLS_JSON, + icon=self._prepare_icon(icon, icon_type, icon_background), server_identifier=server_identifier, - timeout=timeout, - sse_read_timeout=sse_read_timeout, + timeout=configuration.timeout, + sse_read_timeout=configuration.sse_read_timeout, encrypted_headers=encrypted_headers, - ) - db.session.add(mcp_tool) - db.session.commit() - return ToolTransformService.mcp_provider_to_user_provider(mcp_tool, for_list=True) - - @staticmethod - def retrieve_mcp_tools(tenant_id: str, for_list: bool = False) -> list[ToolProviderApiEntity]: - mcp_providers = ( - db.session.query(MCPToolProvider) - .where(MCPToolProvider.tenant_id == tenant_id) - .order_by(MCPToolProvider.name) - .all() - ) - return [ - ToolTransformService.mcp_provider_to_user_provider(mcp_provider, for_list=for_list) - for mcp_provider in mcp_providers - ] - - @classmethod - def list_mcp_tool_from_remote_server(cls, tenant_id: str, provider_id: str) -> ToolProviderApiEntity: - mcp_provider = cls.get_mcp_provider_by_provider_id(provider_id, tenant_id) - server_url = mcp_provider.decrypted_server_url - authed = mcp_provider.authed - headers = mcp_provider.decrypted_headers - timeout = mcp_provider.timeout - sse_read_timeout = mcp_provider.sse_read_timeout - - try: - with MCPClient( - server_url, - provider_id, - tenant_id, - authed=authed, - for_list=True, - headers=headers, - timeout=timeout, - sse_read_timeout=sse_read_timeout, - ) as mcp_client: - tools = mcp_client.list_tools() - except MCPAuthError: - raise ValueError("Please auth the tool first") - except MCPError as e: - raise ValueError(f"Failed to connect to MCP server: {e}") - - try: - mcp_provider = cls.get_mcp_provider_by_provider_id(provider_id, tenant_id) - mcp_provider.tools = json.dumps([tool.model_dump() for tool in tools]) - mcp_provider.authed = True - mcp_provider.updated_at = datetime.now() - db.session.commit() - except Exception: - db.session.rollback() - raise - - user = mcp_provider.load_user() - if not mcp_provider.icon: - raise ValueError("MCP provider icon is required") - return ToolProviderApiEntity( - id=mcp_provider.id, - name=mcp_provider.name, - tools=ToolTransformService.mcp_tool_to_user_tool(mcp_provider, tools), - type=ToolProviderType.MCP, - icon=mcp_provider.icon, - author=user.name if user else "Anonymous", - server_url=mcp_provider.masked_server_url, - updated_at=int(mcp_provider.updated_at.timestamp()), - description=I18nObject(en_US="", zh_Hans=""), - label=I18nObject(en_US=mcp_provider.name, zh_Hans=mcp_provider.name), - plugin_unique_identifier=mcp_provider.server_identifier, + encrypted_credentials=encrypted_credentials, ) - @classmethod - def delete_mcp_tool(cls, tenant_id: str, provider_id: str): - mcp_tool = cls.get_mcp_provider_by_provider_id(provider_id, tenant_id) + self._session.add(mcp_tool) + self._session.flush() + mcp_providers = ToolTransformService.mcp_provider_to_user_provider(mcp_tool, for_list=True) + return mcp_providers - db.session.delete(mcp_tool) - db.session.commit() - - @classmethod - def update_mcp_provider( - cls, + def update_provider( + self, + *, tenant_id: str, provider_id: str, name: str, @@ -222,129 +178,546 @@ class MCPToolManageService: icon_type: str, icon_background: str, server_identifier: str, - timeout: float | None = None, - sse_read_timeout: float | None = None, headers: dict[str, str] | None = None, - ): - mcp_provider = cls.get_mcp_provider_by_provider_id(provider_id, tenant_id) + configuration: MCPConfiguration, + authentication: MCPAuthentication | None = None, + validation_result: ServerUrlValidationResult | None = None, + ) -> None: + """ + Update an MCP provider. - reconnect_result = None + Args: + validation_result: Pre-validation result from validate_server_url_change. + If provided and contains reconnect_result, it will be used + instead of performing network operations. + """ + mcp_provider = self.get_provider(provider_id=provider_id, tenant_id=tenant_id) + + # Check for duplicate name (excluding current provider) + if name != mcp_provider.name: + stmt = select(MCPToolProvider).where( + MCPToolProvider.tenant_id == tenant_id, + MCPToolProvider.name == name, + MCPToolProvider.id != provider_id, + ) + existing_provider = self._session.scalar(stmt) + if existing_provider: + raise ValueError(f"MCP tool {name} already exists") + + # Get URL update data from validation result encrypted_server_url = None server_url_hash = None + reconnect_result = None - if UNCHANGED_SERVER_URL_PLACEHOLDER not in server_url: - encrypted_server_url = encrypter.encrypt_token(tenant_id, server_url) - server_url_hash = hashlib.sha256(server_url.encode()).hexdigest() - - if server_url_hash != mcp_provider.server_url_hash: - reconnect_result = cls._re_connect_mcp_provider(server_url, provider_id, tenant_id) + if validation_result and validation_result.encrypted_server_url: + # Use all data from validation result + encrypted_server_url = validation_result.encrypted_server_url + server_url_hash = validation_result.server_url_hash + reconnect_result = validation_result.reconnect_result try: + # Update basic fields mcp_provider.updated_at = datetime.now() mcp_provider.name = name - mcp_provider.icon = ( - json.dumps({"content": icon, "background": icon_background}) if icon_type == "emoji" else icon - ) + mcp_provider.icon = self._prepare_icon(icon, icon_type, icon_background) mcp_provider.server_identifier = server_identifier - if encrypted_server_url is not None and server_url_hash is not None: + # Update server URL if changed + if encrypted_server_url and server_url_hash: mcp_provider.server_url = encrypted_server_url mcp_provider.server_url_hash = server_url_hash if reconnect_result: - mcp_provider.authed = reconnect_result["authed"] - mcp_provider.tools = reconnect_result["tools"] - mcp_provider.encrypted_credentials = reconnect_result["encrypted_credentials"] + mcp_provider.authed = reconnect_result.authed + mcp_provider.tools = reconnect_result.tools + mcp_provider.encrypted_credentials = reconnect_result.encrypted_credentials - if timeout is not None: - mcp_provider.timeout = timeout - if sse_read_timeout is not None: - mcp_provider.sse_read_timeout = sse_read_timeout + # Update optional configuration fields + self._update_optional_fields(mcp_provider, configuration) + + # Update headers if provided if headers is not None: - # Merge masked headers from frontend with existing real values - if headers: - # existing decrypted and masked headers - existing_decrypted = mcp_provider.decrypted_headers - existing_masked = mcp_provider.masked_headers + mcp_provider.encrypted_headers = self._process_headers(headers, mcp_provider, tenant_id) - # Build final headers: if value equals masked existing, keep original decrypted value - final_headers: dict[str, str] = {} - for key, incoming_value in headers.items(): - if ( - key in existing_masked - and key in existing_decrypted - and isinstance(incoming_value, str) - and incoming_value == existing_masked.get(key) - ): - # unchanged, use original decrypted value - final_headers[key] = str(existing_decrypted[key]) - else: - final_headers[key] = incoming_value + # Update credentials if provided + if authentication and authentication.client_id: + mcp_provider.encrypted_credentials = self._process_credentials(authentication, mcp_provider, tenant_id) - encrypted_headers_dict = MCPToolManageService._encrypt_headers(final_headers, tenant_id) - mcp_provider.encrypted_headers = json.dumps(encrypted_headers_dict) - else: - # Explicitly clear headers if empty dict passed - mcp_provider.encrypted_headers = None - db.session.commit() + # Flush changes to database + self._session.flush() except IntegrityError as e: - db.session.rollback() - error_msg = str(e.orig) - if "unique_mcp_provider_name" in error_msg: - raise ValueError(f"MCP tool {name} already exists") - if "unique_mcp_provider_server_url" in error_msg: - raise ValueError(f"MCP tool {server_url} already exists") - if "unique_mcp_provider_server_identifier" in error_msg: - raise ValueError(f"MCP tool {server_identifier} already exists") - raise - except Exception: - db.session.rollback() - raise + self._handle_integrity_error(e, name, server_url, server_identifier) - @classmethod - def update_mcp_provider_credentials( - cls, mcp_provider: MCPToolProvider, credentials: dict[str, Any], authed: bool = False - ): - provider_controller = MCPToolProviderController.from_db(mcp_provider) + def delete_provider(self, *, tenant_id: str, provider_id: str) -> None: + """Delete an MCP provider.""" + mcp_tool = self.get_provider(provider_id=provider_id, tenant_id=tenant_id) + self._session.delete(mcp_tool) + + def list_providers( + self, *, tenant_id: str, for_list: bool = False, include_sensitive: bool = True + ) -> list[ToolProviderApiEntity]: + """List all MCP providers for a tenant. + + Args: + tenant_id: Tenant ID + for_list: If True, return provider ID; if False, return server identifier + include_sensitive: If False, skip expensive decryption operations (default: True for backward compatibility) + """ + from models.account import Account + + stmt = select(MCPToolProvider).where(MCPToolProvider.tenant_id == tenant_id).order_by(MCPToolProvider.name) + mcp_providers = self._session.scalars(stmt).all() + + if not mcp_providers: + return [] + + # Batch query all users to avoid N+1 problem + user_ids = {provider.user_id for provider in mcp_providers} + users = self._session.query(Account).where(Account.id.in_(user_ids)).all() + user_name_map = {user.id: user.name for user in users} + + return [ + ToolTransformService.mcp_provider_to_user_provider( + provider, + for_list=for_list, + user_name=user_name_map.get(provider.user_id), + include_sensitive=include_sensitive, + ) + for provider in mcp_providers + ] + + # ========== Tool Operations ========== + + def list_provider_tools(self, *, tenant_id: str, provider_id: str) -> ToolProviderApiEntity: + """List tools from remote MCP server.""" + # Load provider and convert to entity + db_provider = self.get_provider(provider_id=provider_id, tenant_id=tenant_id) + provider_entity = db_provider.to_entity() + + # Verify authentication + if not provider_entity.authed: + raise ValueError("Please auth the tool first") + + # Prepare headers with auth token + headers = self._prepare_auth_headers(provider_entity) + + # Retrieve tools from remote server + server_url = provider_entity.decrypt_server_url() + try: + tools = self._retrieve_remote_mcp_tools(server_url, headers, provider_entity) + except MCPError as e: + raise ValueError(f"Failed to connect to MCP server: {e}") + + # Update database with retrieved tools + db_provider.tools = json.dumps([tool.model_dump() for tool in tools]) + db_provider.authed = True + db_provider.updated_at = datetime.now() + self._session.flush() + + # Build API response + return self._build_tool_provider_response(db_provider, provider_entity, tools) + + # ========== OAuth and Credentials Operations ========== + + def update_provider_credentials( + self, *, provider_id: str, tenant_id: str, credentials: dict[str, Any], authed: bool | None = None + ) -> None: + """ + Update provider credentials with encryption. + + Args: + provider_id: Provider ID + tenant_id: Tenant ID + credentials: Credentials to save + authed: Whether provider is authenticated (None means keep current state) + """ + from core.tools.mcp_tool.provider import MCPToolProviderController + + # Get provider from current session + provider = self.get_provider(provider_id=provider_id, tenant_id=tenant_id) + + # Encrypt new credentials + provider_controller = MCPToolProviderController.from_db(provider) tool_configuration = ProviderConfigEncrypter( - tenant_id=mcp_provider.tenant_id, + tenant_id=provider.tenant_id, config=list(provider_controller.get_credentials_schema()), provider_config_cache=NoOpProviderCredentialCache(), ) - credentials = tool_configuration.encrypt(credentials) - mcp_provider.updated_at = datetime.now() - mcp_provider.encrypted_credentials = json.dumps({**mcp_provider.credentials, **credentials}) - mcp_provider.authed = authed - if not authed: - mcp_provider.tools = "[]" - db.session.commit() + encrypted_credentials = tool_configuration.encrypt(credentials) - @classmethod - def _re_connect_mcp_provider(cls, server_url: str, provider_id: str, tenant_id: str): - # Get the existing provider to access headers and timeout settings - mcp_provider = cls.get_mcp_provider_by_provider_id(provider_id, tenant_id) - headers = mcp_provider.decrypted_headers - timeout = mcp_provider.timeout - sse_read_timeout = mcp_provider.sse_read_timeout + # Update provider + provider.updated_at = datetime.now() + provider.encrypted_credentials = json.dumps({**provider.credentials, **encrypted_credentials}) + + if authed is not None: + provider.authed = authed + if not authed: + provider.tools = EMPTY_TOOLS_JSON + + # Flush changes to database + self._session.flush() + + def save_oauth_data( + self, provider_id: str, tenant_id: str, data: dict[str, Any], data_type: OAuthDataType = OAuthDataType.MIXED + ) -> None: + """ + Save OAuth-related data (tokens, client info, code verifier). + + Args: + provider_id: Provider ID + tenant_id: Tenant ID + data: Data to save (tokens, client info, or code verifier) + data_type: Type of OAuth data to save + """ + # Determine if this makes the provider authenticated + authed = ( + data_type == OAuthDataType.TOKENS or (data_type == OAuthDataType.MIXED and "access_token" in data) or None + ) + + # update_provider_credentials will validate provider existence + self.update_provider_credentials(provider_id=provider_id, tenant_id=tenant_id, credentials=data, authed=authed) + + def clear_provider_credentials(self, *, provider_id: str, tenant_id: str) -> None: + """ + Clear all credentials for a provider. + + Args: + provider_id: Provider ID + tenant_id: Tenant ID + """ + # Get provider from current session + provider = self.get_provider(provider_id=provider_id, tenant_id=tenant_id) + + provider.tools = EMPTY_TOOLS_JSON + provider.encrypted_credentials = EMPTY_CREDENTIALS_JSON + provider.updated_at = datetime.now() + provider.authed = False + + # ========== Private Helper Methods ========== + + def _check_provider_exists(self, tenant_id: str, name: str, server_url_hash: str, server_identifier: str) -> None: + """Check if provider with same attributes already exists.""" + stmt = select(MCPToolProvider).where( + MCPToolProvider.tenant_id == tenant_id, + or_( + MCPToolProvider.name == name, + MCPToolProvider.server_url_hash == server_url_hash, + MCPToolProvider.server_identifier == server_identifier, + ), + ) + existing_provider = self._session.scalar(stmt) + + if existing_provider: + if existing_provider.name == name: + raise ValueError(f"MCP tool {name} already exists") + if existing_provider.server_url_hash == server_url_hash: + raise ValueError("MCP tool with this server URL already exists") + if existing_provider.server_identifier == server_identifier: + raise ValueError(f"MCP tool {server_identifier} already exists") + + def _prepare_icon(self, icon: str, icon_type: str, icon_background: str) -> str: + """Prepare icon data for storage.""" + if icon_type == "emoji": + return json.dumps({"content": icon, "background": icon_background}) + return icon + + def _encrypt_dict_fields(self, data: dict[str, Any], secret_fields: list[str], tenant_id: str) -> Mapping[str, str]: + """Encrypt specified fields in a dictionary. + + Args: + data: Dictionary containing data to encrypt + secret_fields: List of field names to encrypt + tenant_id: Tenant ID for encryption + + Returns: + JSON string of encrypted data + """ + from core.entities.provider_entities import BasicProviderConfig + from core.tools.utils.encryption import create_provider_encrypter + + # Create config for secret fields + config = [ + BasicProviderConfig(type=BasicProviderConfig.Type.SECRET_INPUT, name=field) for field in secret_fields + ] + + encrypter_instance, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=config, + cache=NoOpProviderCredentialCache(), + ) + + encrypted_data = encrypter_instance.encrypt(data) + return encrypted_data + + def _prepare_encrypted_dict(self, headers: dict[str, str], tenant_id: str) -> str: + """Encrypt headers and prepare for storage.""" + # All headers are treated as secret + return json.dumps(self._encrypt_dict_fields(headers, list(headers.keys()), tenant_id)) + + def _prepare_auth_headers(self, provider_entity: MCPProviderEntity) -> dict[str, str]: + """Prepare headers with OAuth token if available.""" + headers = provider_entity.decrypt_headers() + tokens = provider_entity.retrieve_tokens() + if tokens: + headers["Authorization"] = f"{tokens.token_type.capitalize()} {tokens.access_token}" + return headers + + def _retrieve_remote_mcp_tools( + self, + server_url: str, + headers: dict[str, str], + provider_entity: MCPProviderEntity, + ): + """Retrieve tools from remote MCP server.""" + with MCPClientWithAuthRetry( + server_url=server_url, + headers=headers, + timeout=provider_entity.timeout, + sse_read_timeout=provider_entity.sse_read_timeout, + provider_entity=provider_entity, + ) as mcp_client: + return mcp_client.list_tools() + + def execute_auth_actions(self, auth_result: Any) -> dict[str, str]: + """ + Execute the actions returned by the auth function. + + This method processes the AuthResult and performs the necessary database operations. + + Args: + auth_result: The result from the auth function + + Returns: + The response from the auth result + """ + from core.mcp.entities import AuthAction, AuthActionType + + action: AuthAction + for action in auth_result.actions: + if action.provider_id is None or action.tenant_id is None: + continue + + if action.action_type == AuthActionType.SAVE_CLIENT_INFO: + self.save_oauth_data(action.provider_id, action.tenant_id, action.data, OAuthDataType.CLIENT_INFO) + elif action.action_type == AuthActionType.SAVE_TOKENS: + self.save_oauth_data(action.provider_id, action.tenant_id, action.data, OAuthDataType.TOKENS) + elif action.action_type == AuthActionType.SAVE_CODE_VERIFIER: + self.save_oauth_data(action.provider_id, action.tenant_id, action.data, OAuthDataType.CODE_VERIFIER) + + return auth_result.response + + def auth_with_actions( + self, provider_entity: MCPProviderEntity, authorization_code: str | None = None + ) -> dict[str, str]: + """ + Perform authentication and execute all resulting actions. + + This method is used by MCPClientWithAuthRetry for automatic re-authentication. + + Args: + provider_entity: The MCP provider entity + authorization_code: Optional authorization code + + Returns: + Response dictionary from auth result + """ + auth_result = auth(provider_entity, authorization_code) + return self.execute_auth_actions(auth_result) + + def _reconnect_provider(self, *, server_url: str, provider: MCPToolProvider) -> ReconnectResult: + """Attempt to reconnect to MCP provider with new server URL.""" + provider_entity = provider.to_entity() + headers = provider_entity.headers try: - with MCPClient( - server_url, - provider_id, - tenant_id, - authed=False, - for_list=True, - headers=headers, - timeout=timeout, - sse_read_timeout=sse_read_timeout, - ) as mcp_client: - tools = mcp_client.list_tools() - return { - "authed": True, - "tools": json.dumps([tool.model_dump() for tool in tools]), - "encrypted_credentials": "{}", - } + tools = self._retrieve_remote_mcp_tools(server_url, headers, provider_entity) + return ReconnectResult( + authed=True, + tools=json.dumps([tool.model_dump() for tool in tools]), + encrypted_credentials=EMPTY_CREDENTIALS_JSON, + ) except MCPAuthError: - return {"authed": False, "tools": "[]", "encrypted_credentials": "{}"} + return ReconnectResult(authed=False, tools=EMPTY_TOOLS_JSON, encrypted_credentials=EMPTY_CREDENTIALS_JSON) except MCPError as e: raise ValueError(f"Failed to re-connect MCP server: {e}") from e + + def validate_server_url_change( + self, *, tenant_id: str, provider_id: str, new_server_url: str + ) -> ServerUrlValidationResult: + """ + Validate server URL change by attempting to connect to the new server. + This method should be called BEFORE update_provider to perform network operations + outside of the database transaction. + + Returns: + ServerUrlValidationResult: Validation result with connection status and tools if successful + """ + # Handle hidden/unchanged URL + if UNCHANGED_SERVER_URL_PLACEHOLDER in new_server_url: + return ServerUrlValidationResult(needs_validation=False) + + # Validate URL format + if not self._is_valid_url(new_server_url): + raise ValueError("Server URL is not valid.") + + # Always encrypt and hash the URL + encrypted_server_url = encrypter.encrypt_token(tenant_id, new_server_url) + new_server_url_hash = hashlib.sha256(new_server_url.encode()).hexdigest() + + # Get current provider + provider = self.get_provider(provider_id=provider_id, tenant_id=tenant_id) + + # Check if URL is actually different + if new_server_url_hash == provider.server_url_hash: + # URL hasn't changed, but still return the encrypted data + return ServerUrlValidationResult( + needs_validation=False, encrypted_server_url=encrypted_server_url, server_url_hash=new_server_url_hash + ) + + # Perform validation by attempting to connect + reconnect_result = self._reconnect_provider(server_url=new_server_url, provider=provider) + return ServerUrlValidationResult( + needs_validation=True, + validation_passed=True, + reconnect_result=reconnect_result, + encrypted_server_url=encrypted_server_url, + server_url_hash=new_server_url_hash, + ) + + def _build_tool_provider_response( + self, db_provider: MCPToolProvider, provider_entity: MCPProviderEntity, tools: list + ) -> ToolProviderApiEntity: + """Build API response for tool provider.""" + user = db_provider.load_user() + response = provider_entity.to_api_response( + user_name=user.name if user else None, + ) + response["tools"] = ToolTransformService.mcp_tool_to_user_tool(db_provider, tools) + response["plugin_unique_identifier"] = provider_entity.provider_id + return ToolProviderApiEntity(**response) + + def _handle_integrity_error( + self, error: IntegrityError, name: str, server_url: str, server_identifier: str + ) -> None: + """Handle database integrity errors with user-friendly messages.""" + error_msg = str(error.orig) + if "unique_mcp_provider_name" in error_msg: + raise ValueError(f"MCP tool {name} already exists") + if "unique_mcp_provider_server_url" in error_msg: + raise ValueError(f"MCP tool {server_url} already exists") + if "unique_mcp_provider_server_identifier" in error_msg: + raise ValueError(f"MCP tool {server_identifier} already exists") + raise + + def _is_valid_url(self, url: str) -> bool: + """Validate URL format.""" + if not url: + return False + try: + parsed = urlparse(url) + return all([parsed.scheme, parsed.netloc]) and parsed.scheme in ["http", "https"] + except (ValueError, TypeError): + return False + + def _update_optional_fields(self, mcp_provider: MCPToolProvider, configuration: MCPConfiguration) -> None: + """Update optional configuration fields using setattr for cleaner code.""" + field_mapping = {"timeout": configuration.timeout, "sse_read_timeout": configuration.sse_read_timeout} + + for field, value in field_mapping.items(): + if value is not None: + setattr(mcp_provider, field, value) + + def _process_headers(self, headers: dict[str, str], mcp_provider: MCPToolProvider, tenant_id: str) -> str | None: + """Process headers update, handling empty dict to clear headers.""" + if not headers: + return None + + # Merge with existing headers to preserve masked values + final_headers = self._merge_headers_with_masked(incoming_headers=headers, mcp_provider=mcp_provider) + return self._prepare_encrypted_dict(final_headers, tenant_id) + + def _process_credentials( + self, authentication: MCPAuthentication, mcp_provider: MCPToolProvider, tenant_id: str + ) -> str: + """Process credentials update, handling masked values.""" + # Merge with existing credentials + final_client_id, final_client_secret = self._merge_credentials_with_masked( + authentication.client_id, authentication.client_secret, mcp_provider + ) + + # Build and encrypt + return self._build_and_encrypt_credentials(final_client_id, final_client_secret, tenant_id) + + def _merge_headers_with_masked( + self, incoming_headers: dict[str, str], mcp_provider: MCPToolProvider + ) -> dict[str, str]: + """Merge incoming headers with existing ones, preserving unchanged masked values. + + Args: + incoming_headers: Headers from frontend (may contain masked values) + mcp_provider: The MCP provider instance + + Returns: + Final headers dict with proper values (original for unchanged masked, new for changed) + """ + mcp_provider_entity = mcp_provider.to_entity() + existing_decrypted = mcp_provider_entity.decrypt_headers() + existing_masked = mcp_provider_entity.masked_headers() + + return { + key: (str(existing_decrypted[key]) if key in existing_masked and value == existing_masked[key] else value) + for key, value in incoming_headers.items() + if key in existing_decrypted or value != existing_masked.get(key) + } + + def _merge_credentials_with_masked( + self, + client_id: str, + client_secret: str | None, + mcp_provider: MCPToolProvider, + ) -> tuple[ + str, + str | None, + ]: + """Merge incoming credentials with existing ones, preserving unchanged masked values. + + Args: + client_id: Client ID from frontend (may be masked) + client_secret: Client secret from frontend (may be masked) + mcp_provider: The MCP provider instance + + Returns: + Tuple of (final_client_id, final_client_secret) + """ + mcp_provider_entity = mcp_provider.to_entity() + existing_decrypted = mcp_provider_entity.decrypt_credentials() + existing_masked = mcp_provider_entity.masked_credentials() + + # Check if client_id is masked and unchanged + final_client_id = client_id + if existing_masked.get("client_id") and client_id == existing_masked["client_id"]: + # Use existing decrypted value + final_client_id = existing_decrypted.get("client_id", client_id) + + # Check if client_secret is masked and unchanged + final_client_secret = client_secret + if existing_masked.get("client_secret") and client_secret == existing_masked["client_secret"]: + # Use existing decrypted value + final_client_secret = existing_decrypted.get("client_secret", client_secret) + + return final_client_id, final_client_secret + + def _build_and_encrypt_credentials(self, client_id: str, client_secret: str | None, tenant_id: str) -> str: + """Build credentials and encrypt sensitive fields.""" + # Create a flat structure with all credential data + credentials_data = { + "client_id": client_id, + "client_name": CLIENT_NAME, + "is_dynamic_registration": False, + } + secret_fields = [] + if client_secret is not None: + credentials_data["encrypted_client_secret"] = client_secret + secret_fields = ["encrypted_client_secret"] + client_info = self._encrypt_dict_fields(credentials_data, secret_fields, tenant_id) + return json.dumps({"client_information": client_info}) diff --git a/api/services/tools/tools_transform_service.py b/api/services/tools/tools_transform_service.py index b7850ea150..3e976234ba 100644 --- a/api/services/tools/tools_transform_service.py +++ b/api/services/tools/tools_transform_service.py @@ -3,12 +3,13 @@ import logging from collections.abc import Mapping from typing import Any, Union +from pydantic import ValidationError from yarl import URL from configs import dify_config from core.helper.provider_cache import ToolProviderCredentialsCache from core.mcp.types import Tool as MCPTool -from core.plugin.entities.plugin_daemon import PluginDatasourceProviderEntity +from core.plugin.entities.plugin_daemon import CredentialType, PluginDatasourceProviderEntity from core.tools.__base.tool import Tool from core.tools.__base.tool_runtime import ToolRuntime from core.tools.builtin_tool.provider import BuiltinToolProviderController @@ -18,7 +19,6 @@ from core.tools.entities.common_entities import I18nObject from core.tools.entities.tool_bundle import ApiToolBundle from core.tools.entities.tool_entities import ( ApiProviderAuthType, - CredentialType, ToolParameter, ToolProviderType, ) @@ -27,18 +27,12 @@ from core.tools.utils.encryption import create_provider_encrypter, create_tool_p from core.tools.workflow_as_tool.provider import WorkflowToolProviderController from core.tools.workflow_as_tool.tool import WorkflowTool from models.tools import ApiToolProvider, BuiltinToolProvider, MCPToolProvider, WorkflowToolProvider +from services.plugin.plugin_service import PluginService logger = logging.getLogger(__name__) class ToolTransformService: - @classmethod - def get_plugin_icon_url(cls, tenant_id: str, filename: str) -> str: - url_prefix = ( - URL(dify_config.CONSOLE_API_URL or "/") / "console" / "api" / "workspaces" / "current" / "plugin" / "icon" - ) - return str(url_prefix % {"tenant_id": tenant_id, "filename": filename}) - @classmethod def get_tool_provider_icon_url( cls, provider_type: str, provider_name: str, icon: str | Mapping[str, str] @@ -78,11 +72,9 @@ class ToolTransformService: elif isinstance(provider, ToolProviderApiEntity): if provider.plugin_id: if isinstance(provider.icon, str): - provider.icon = ToolTransformService.get_plugin_icon_url( - tenant_id=tenant_id, filename=provider.icon - ) + provider.icon = PluginService.get_plugin_icon_url(tenant_id=tenant_id, filename=provider.icon) if isinstance(provider.icon_dark, str) and provider.icon_dark: - provider.icon_dark = ToolTransformService.get_plugin_icon_url( + provider.icon_dark = PluginService.get_plugin_icon_url( tenant_id=tenant_id, filename=provider.icon_dark ) else: @@ -96,7 +88,7 @@ class ToolTransformService: elif isinstance(provider, PluginDatasourceProviderEntity): if provider.plugin_id: if isinstance(provider.declaration.identity.icon, str): - provider.declaration.identity.icon = ToolTransformService.get_plugin_icon_url( + provider.declaration.identity.icon = PluginService.get_plugin_icon_url( tenant_id=tenant_id, filename=provider.declaration.identity.icon ) @@ -171,7 +163,7 @@ class ToolTransformService: ) # decrypt the credentials and mask the credentials decrypted_credentials = encrypter.decrypt(data=credentials) - masked_credentials = encrypter.mask_tool_credentials(data=decrypted_credentials) + masked_credentials = encrypter.mask_plugin_credentials(data=decrypted_credentials) result.masked_credentials = masked_credentials result.original_credentials = decrypted_credentials @@ -232,40 +224,59 @@ class ToolTransformService: ) @staticmethod - def mcp_provider_to_user_provider(db_provider: MCPToolProvider, for_list: bool = False) -> ToolProviderApiEntity: - user = db_provider.load_user() - return ToolProviderApiEntity( - id=db_provider.server_identifier if not for_list else db_provider.id, - author=user.name if user else "Anonymous", - name=db_provider.name, - icon=db_provider.provider_icon, - type=ToolProviderType.MCP, - is_team_authorization=db_provider.authed, - server_url=db_provider.masked_server_url, - tools=ToolTransformService.mcp_tool_to_user_tool( - db_provider, [MCPTool.model_validate(tool) for tool in json.loads(db_provider.tools)] - ), - updated_at=int(db_provider.updated_at.timestamp()), - label=I18nObject(en_US=db_provider.name, zh_Hans=db_provider.name), - description=I18nObject(en_US="", zh_Hans=""), - server_identifier=db_provider.server_identifier, - timeout=db_provider.timeout, - sse_read_timeout=db_provider.sse_read_timeout, - masked_headers=db_provider.masked_headers, - original_headers=db_provider.decrypted_headers, - ) + def mcp_provider_to_user_provider( + db_provider: MCPToolProvider, + for_list: bool = False, + user_name: str | None = None, + include_sensitive: bool = True, + ) -> ToolProviderApiEntity: + from core.entities.mcp_provider import MCPConfiguration + + # Use provided user_name to avoid N+1 query, fallback to load_user() if not provided + if user_name is None: + user = db_provider.load_user() + user_name = user.name if user else None + + # Convert to entity and use its API response method + provider_entity = db_provider.to_entity() + + response = provider_entity.to_api_response(user_name=user_name, include_sensitive=include_sensitive) + try: + mcp_tools = [MCPTool(**tool) for tool in json.loads(db_provider.tools)] + except (ValidationError, json.JSONDecodeError): + mcp_tools = [] + # Add additional fields specific to the transform + response["id"] = db_provider.server_identifier if not for_list else db_provider.id + response["tools"] = ToolTransformService.mcp_tool_to_user_tool(db_provider, mcp_tools, user_name=user_name) + response["server_identifier"] = db_provider.server_identifier + + # Convert configuration dict to MCPConfiguration object + if "configuration" in response and isinstance(response["configuration"], dict): + response["configuration"] = MCPConfiguration( + timeout=float(response["configuration"]["timeout"]), + sse_read_timeout=float(response["configuration"]["sse_read_timeout"]), + ) + + return ToolProviderApiEntity(**response) @staticmethod - def mcp_tool_to_user_tool(mcp_provider: MCPToolProvider, tools: list[MCPTool]) -> list[ToolApiEntity]: - user = mcp_provider.load_user() + def mcp_tool_to_user_tool( + mcp_provider: MCPToolProvider, tools: list[MCPTool], user_name: str | None = None + ) -> list[ToolApiEntity]: + # Use provided user_name to avoid N+1 query, fallback to load_user() if not provided + if user_name is None: + user = mcp_provider.load_user() + user_name = user.name if user else "Anonymous" + return [ ToolApiEntity( - author=user.name if user else "Anonymous", + author=user_name or "Anonymous", name=tool.name, label=I18nObject(en_US=tool.name, zh_Hans=tool.name), description=I18nObject(en_US=tool.description or "", zh_Hans=tool.description or ""), parameters=ToolTransformService.convert_mcp_schema_to_parameter(tool.inputSchema), labels=[], + output_schema=tool.outputSchema or {}, ) for tool in tools ] @@ -325,7 +336,7 @@ class ToolTransformService: # decrypt the credentials and mask the credentials decrypted_credentials = encrypter.decrypt(data=credentials) - masked_credentials = encrypter.mask_tool_credentials(data=decrypted_credentials) + masked_credentials = encrypter.mask_plugin_credentials(data=decrypted_credentials) result.masked_credentials = masked_credentials @@ -412,7 +423,7 @@ class ToolTransformService: ) @staticmethod - def convert_mcp_schema_to_parameter(schema: dict) -> list["ToolParameter"]: + def convert_mcp_schema_to_parameter(schema: dict[str, Any]) -> list["ToolParameter"]: """ Convert MCP JSON schema to tool parameters @@ -421,7 +432,7 @@ class ToolTransformService: """ def create_parameter( - name: str, description: str, param_type: str, required: bool, input_schema: dict | None = None + name: str, description: str, param_type: str, required: bool, input_schema: dict[str, Any] | None = None ) -> ToolParameter: """Create a ToolParameter instance with given attributes""" input_schema_dict: dict[str, Any] = {"input_schema": input_schema} if input_schema else {} @@ -436,7 +447,9 @@ class ToolTransformService: **input_schema_dict, ) - def process_properties(props: dict, required: list, prefix: str = "") -> list[ToolParameter]: + def process_properties( + props: dict[str, dict[str, Any]], required: list[str], prefix: str = "" + ) -> list[ToolParameter]: """Process properties recursively""" TYPE_MAPPING = {"integer": "number", "float": "number"} COMPLEX_TYPES = ["array", "object"] diff --git a/api/services/trigger/schedule_service.py b/api/services/trigger/schedule_service.py new file mode 100644 index 0000000000..b49d14f860 --- /dev/null +++ b/api/services/trigger/schedule_service.py @@ -0,0 +1,312 @@ +import json +import logging +from collections.abc import Mapping +from datetime import datetime +from typing import Any + +from sqlalchemy import select +from sqlalchemy.orm import Session + +from core.workflow.nodes import NodeType +from core.workflow.nodes.trigger_schedule.entities import ScheduleConfig, SchedulePlanUpdate, VisualConfig +from core.workflow.nodes.trigger_schedule.exc import ScheduleConfigError, ScheduleNotFoundError +from libs.schedule_utils import calculate_next_run_at, convert_12h_to_24h +from models.account import Account, TenantAccountJoin +from models.trigger import WorkflowSchedulePlan +from models.workflow import Workflow +from services.errors.account import AccountNotFoundError + +logger = logging.getLogger(__name__) + + +class ScheduleService: + @staticmethod + def create_schedule( + session: Session, + tenant_id: str, + app_id: str, + config: ScheduleConfig, + ) -> WorkflowSchedulePlan: + """ + Create a new schedule with validated configuration. + + Args: + session: Database session + tenant_id: Tenant ID + app_id: Application ID + config: Validated schedule configuration + + Returns: + Created WorkflowSchedulePlan instance + """ + next_run_at = calculate_next_run_at( + config.cron_expression, + config.timezone, + ) + + schedule = WorkflowSchedulePlan( + tenant_id=tenant_id, + app_id=app_id, + node_id=config.node_id, + cron_expression=config.cron_expression, + timezone=config.timezone, + next_run_at=next_run_at, + ) + + session.add(schedule) + session.flush() + + return schedule + + @staticmethod + def update_schedule( + session: Session, + schedule_id: str, + updates: SchedulePlanUpdate, + ) -> WorkflowSchedulePlan: + """ + Update an existing schedule with validated configuration. + + Args: + session: Database session + schedule_id: Schedule ID to update + updates: Validated update configuration + + Raises: + ScheduleNotFoundError: If schedule not found + + Returns: + Updated WorkflowSchedulePlan instance + """ + schedule = session.get(WorkflowSchedulePlan, schedule_id) + if not schedule: + raise ScheduleNotFoundError(f"Schedule not found: {schedule_id}") + + # If time-related fields are updated, synchronously update the next_run_at. + time_fields_updated = False + + if updates.node_id is not None: + schedule.node_id = updates.node_id + + if updates.cron_expression is not None: + schedule.cron_expression = updates.cron_expression + time_fields_updated = True + + if updates.timezone is not None: + schedule.timezone = updates.timezone + time_fields_updated = True + + if time_fields_updated: + schedule.next_run_at = calculate_next_run_at( + schedule.cron_expression, + schedule.timezone, + ) + + session.flush() + return schedule + + @staticmethod + def delete_schedule( + session: Session, + schedule_id: str, + ) -> None: + """ + Delete a schedule plan. + + Args: + session: Database session + schedule_id: Schedule ID to delete + """ + schedule = session.get(WorkflowSchedulePlan, schedule_id) + if not schedule: + raise ScheduleNotFoundError(f"Schedule not found: {schedule_id}") + + session.delete(schedule) + session.flush() + + @staticmethod + def get_tenant_owner(session: Session, tenant_id: str) -> Account: + """ + Returns an account to execute scheduled workflows on behalf of the tenant. + Prioritizes owner over admin to ensure proper authorization hierarchy. + """ + result = session.execute( + select(TenantAccountJoin) + .where(TenantAccountJoin.tenant_id == tenant_id, TenantAccountJoin.role == "owner") + .limit(1) + ).scalar_one_or_none() + + if not result: + # Owner may not exist in some tenant configurations, fallback to admin + result = session.execute( + select(TenantAccountJoin) + .where(TenantAccountJoin.tenant_id == tenant_id, TenantAccountJoin.role == "admin") + .limit(1) + ).scalar_one_or_none() + + if result: + account = session.get(Account, result.account_id) + if not account: + raise AccountNotFoundError(f"Account not found: {result.account_id}") + return account + else: + raise AccountNotFoundError(f"Account not found for tenant: {tenant_id}") + + @staticmethod + def update_next_run_at( + session: Session, + schedule_id: str, + ) -> datetime: + """ + Advances the schedule to its next execution time after a successful trigger. + Uses current time as base to prevent missing executions during delays. + """ + schedule = session.get(WorkflowSchedulePlan, schedule_id) + if not schedule: + raise ScheduleNotFoundError(f"Schedule not found: {schedule_id}") + + # Base on current time to handle execution delays gracefully + next_run_at = calculate_next_run_at( + schedule.cron_expression, + schedule.timezone, + ) + + schedule.next_run_at = next_run_at + session.flush() + return next_run_at + + @staticmethod + def to_schedule_config(node_config: Mapping[str, Any]) -> ScheduleConfig: + """ + Converts user-friendly visual schedule settings to cron expression. + Maintains consistency with frontend UI expectations while supporting croniter's extended syntax. + """ + node_data = node_config.get("data", {}) + mode = node_data.get("mode", "visual") + timezone = node_data.get("timezone", "UTC") + node_id = node_config.get("id", "start") + + cron_expression = None + if mode == "cron": + cron_expression = node_data.get("cron_expression") + if not cron_expression: + raise ScheduleConfigError("Cron expression is required for cron mode") + elif mode == "visual": + frequency = str(node_data.get("frequency")) + if not frequency: + raise ScheduleConfigError("Frequency is required for visual mode") + visual_config = VisualConfig(**node_data.get("visual_config", {})) + cron_expression = ScheduleService.visual_to_cron(frequency=frequency, visual_config=visual_config) + if not cron_expression: + raise ScheduleConfigError("Cron expression is required for visual mode") + else: + raise ScheduleConfigError(f"Invalid schedule mode: {mode}") + return ScheduleConfig(node_id=node_id, cron_expression=cron_expression, timezone=timezone) + + @staticmethod + def extract_schedule_config(workflow: Workflow) -> ScheduleConfig | None: + """ + Extracts schedule configuration from workflow graph. + + Searches for the first schedule trigger node in the workflow and converts + its configuration (either visual or cron mode) into a unified ScheduleConfig. + + Args: + workflow: The workflow containing the graph definition + + Returns: + ScheduleConfig if a valid schedule node is found, None if no schedule node exists + + Raises: + ScheduleConfigError: If graph parsing fails or schedule configuration is invalid + + Note: + Currently only returns the first schedule node found. + Multiple schedule nodes in the same workflow are not supported. + """ + try: + graph_data = workflow.graph_dict + except (json.JSONDecodeError, TypeError, AttributeError) as e: + raise ScheduleConfigError(f"Failed to parse workflow graph: {e}") + + if not graph_data: + raise ScheduleConfigError("Workflow graph is empty") + + nodes = graph_data.get("nodes", []) + for node in nodes: + node_data = node.get("data", {}) + + if node_data.get("type") != NodeType.TRIGGER_SCHEDULE.value: + continue + + mode = node_data.get("mode", "visual") + timezone = node_data.get("timezone", "UTC") + node_id = node.get("id", "start") + + cron_expression = None + if mode == "cron": + cron_expression = node_data.get("cron_expression") + if not cron_expression: + raise ScheduleConfigError("Cron expression is required for cron mode") + elif mode == "visual": + frequency = node_data.get("frequency") + visual_config_dict = node_data.get("visual_config", {}) + visual_config = VisualConfig(**visual_config_dict) + cron_expression = ScheduleService.visual_to_cron(frequency, visual_config) + else: + raise ScheduleConfigError(f"Invalid schedule mode: {mode}") + + return ScheduleConfig(node_id=node_id, cron_expression=cron_expression, timezone=timezone) + + return None + + @staticmethod + def visual_to_cron(frequency: str, visual_config: VisualConfig) -> str: + """ + Converts user-friendly visual schedule settings to cron expression. + Maintains consistency with frontend UI expectations while supporting croniter's extended syntax. + """ + if frequency == "hourly": + if visual_config.on_minute is None: + raise ScheduleConfigError("on_minute is required for hourly schedules") + return f"{visual_config.on_minute} * * * *" + + elif frequency == "daily": + if not visual_config.time: + raise ScheduleConfigError("time is required for daily schedules") + hour, minute = convert_12h_to_24h(visual_config.time) + return f"{minute} {hour} * * *" + + elif frequency == "weekly": + if not visual_config.time: + raise ScheduleConfigError("time is required for weekly schedules") + if not visual_config.weekdays: + raise ScheduleConfigError("Weekdays are required for weekly schedules") + hour, minute = convert_12h_to_24h(visual_config.time) + weekday_map = {"sun": "0", "mon": "1", "tue": "2", "wed": "3", "thu": "4", "fri": "5", "sat": "6"} + cron_weekdays = [weekday_map[day] for day in visual_config.weekdays] + return f"{minute} {hour} * * {','.join(sorted(cron_weekdays))}" + + elif frequency == "monthly": + if not visual_config.time: + raise ScheduleConfigError("time is required for monthly schedules") + if not visual_config.monthly_days: + raise ScheduleConfigError("Monthly days are required for monthly schedules") + hour, minute = convert_12h_to_24h(visual_config.time) + + numeric_days: list[int] = [] + has_last = False + for day in visual_config.monthly_days: + if day == "last": + has_last = True + else: + numeric_days.append(day) + + result_days = [str(d) for d in sorted(set(numeric_days))] + if has_last: + result_days.append("L") + + return f"{minute} {hour} {','.join(result_days)} * *" + + else: + raise ScheduleConfigError(f"Unsupported frequency: {frequency}") diff --git a/api/services/trigger/trigger_provider_service.py b/api/services/trigger/trigger_provider_service.py new file mode 100644 index 0000000000..076cc7e776 --- /dev/null +++ b/api/services/trigger/trigger_provider_service.py @@ -0,0 +1,687 @@ +import json +import logging +import time as _time +import uuid +from collections.abc import Mapping +from typing import Any + +from sqlalchemy import desc, func +from sqlalchemy.orm import Session + +from configs import dify_config +from constants import HIDDEN_VALUE, UNKNOWN_VALUE +from core.helper.provider_cache import NoOpProviderCredentialCache +from core.helper.provider_encryption import ProviderConfigEncrypter, create_provider_encrypter +from core.plugin.entities.plugin_daemon import CredentialType +from core.plugin.impl.oauth import OAuthHandler +from core.tools.utils.system_oauth_encryption import decrypt_system_oauth_params +from core.trigger.entities.api_entities import ( + TriggerProviderApiEntity, + TriggerProviderSubscriptionApiEntity, +) +from core.trigger.entities.entities import Subscription as TriggerSubscriptionEntity +from core.trigger.provider import PluginTriggerProviderController +from core.trigger.trigger_manager import TriggerManager +from core.trigger.utils.encryption import ( + create_trigger_provider_encrypter_for_properties, + create_trigger_provider_encrypter_for_subscription, + delete_cache_for_subscription, +) +from core.trigger.utils.endpoint import generate_plugin_trigger_endpoint_url +from extensions.ext_database import db +from extensions.ext_redis import redis_client +from models.provider_ids import TriggerProviderID +from models.trigger import ( + TriggerOAuthSystemClient, + TriggerOAuthTenantClient, + TriggerSubscription, + WorkflowPluginTrigger, +) +from services.plugin.plugin_service import PluginService + +logger = logging.getLogger(__name__) + + +class TriggerProviderService: + """Service for managing trigger providers and credentials""" + + ########################## + # Trigger provider + ########################## + __MAX_TRIGGER_PROVIDER_COUNT__ = 10 + + @classmethod + def get_trigger_provider(cls, tenant_id: str, provider: TriggerProviderID) -> TriggerProviderApiEntity: + """Get info for a trigger provider""" + return TriggerManager.get_trigger_provider(tenant_id, provider).to_api_entity() + + @classmethod + def list_trigger_providers(cls, tenant_id: str) -> list[TriggerProviderApiEntity]: + """List all trigger providers for the current tenant""" + return [provider.to_api_entity() for provider in TriggerManager.list_all_trigger_providers(tenant_id)] + + @classmethod + def list_trigger_provider_subscriptions( + cls, tenant_id: str, provider_id: TriggerProviderID + ) -> list[TriggerProviderSubscriptionApiEntity]: + """List all trigger subscriptions for the current tenant""" + subscriptions: list[TriggerProviderSubscriptionApiEntity] = [] + workflows_in_use_map: dict[str, int] = {} + with Session(db.engine, expire_on_commit=False) as session: + # Get all subscriptions + subscriptions_db = ( + session.query(TriggerSubscription) + .filter_by(tenant_id=tenant_id, provider_id=str(provider_id)) + .order_by(desc(TriggerSubscription.created_at)) + .all() + ) + subscriptions = [subscription.to_api_entity() for subscription in subscriptions_db] + if not subscriptions: + return [] + usage_counts = ( + session.query( + WorkflowPluginTrigger.subscription_id, + func.count(func.distinct(WorkflowPluginTrigger.app_id)).label("app_count"), + ) + .filter( + WorkflowPluginTrigger.tenant_id == tenant_id, + WorkflowPluginTrigger.subscription_id.in_([s.id for s in subscriptions]), + ) + .group_by(WorkflowPluginTrigger.subscription_id) + .all() + ) + workflows_in_use_map = {str(row.subscription_id): int(row.app_count) for row in usage_counts} + + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + for subscription in subscriptions: + encrypter, _ = create_trigger_provider_encrypter_for_subscription( + tenant_id=tenant_id, + controller=provider_controller, + subscription=subscription, + ) + subscription.credentials = dict( + encrypter.mask_credentials(dict(encrypter.decrypt(subscription.credentials))) + ) + subscription.properties = dict(encrypter.mask_credentials(dict(encrypter.decrypt(subscription.properties)))) + subscription.parameters = dict(encrypter.mask_credentials(dict(encrypter.decrypt(subscription.parameters)))) + count = workflows_in_use_map.get(subscription.id) + subscription.workflows_in_use = count if count is not None else 0 + + return subscriptions + + @classmethod + def add_trigger_subscription( + cls, + tenant_id: str, + user_id: str, + name: str, + provider_id: TriggerProviderID, + endpoint_id: str, + credential_type: CredentialType, + parameters: Mapping[str, Any], + properties: Mapping[str, Any], + credentials: Mapping[str, str], + subscription_id: str | None = None, + credential_expires_at: int = -1, + expires_at: int = -1, + ) -> Mapping[str, Any]: + """ + Add a new trigger provider with credentials. + Supports multiple credential instances per provider. + + :param tenant_id: Tenant ID + :param provider_id: Provider identifier (e.g., "plugin_id/provider_name") + :param credential_type: Type of credential (oauth or api_key) + :param credentials: Credential data to encrypt and store + :param name: Optional name for this credential instance + :param expires_at: OAuth token expiration timestamp + :return: Success response + """ + try: + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + with Session(db.engine, expire_on_commit=False) as session: + # Use distributed lock to prevent race conditions + lock_key = f"trigger_provider_create_lock:{tenant_id}_{provider_id}" + with redis_client.lock(lock_key, timeout=20): + # Check provider count limit + provider_count = ( + session.query(TriggerSubscription) + .filter_by(tenant_id=tenant_id, provider_id=str(provider_id)) + .count() + ) + + if provider_count >= cls.__MAX_TRIGGER_PROVIDER_COUNT__: + raise ValueError( + f"Maximum number of providers ({cls.__MAX_TRIGGER_PROVIDER_COUNT__}) " + f"reached for {provider_id}" + ) + + # Check if name already exists + existing = ( + session.query(TriggerSubscription) + .filter_by(tenant_id=tenant_id, provider_id=str(provider_id), name=name) + .first() + ) + if existing: + raise ValueError(f"Credential name '{name}' already exists for this provider") + + credential_encrypter: ProviderConfigEncrypter | None = None + if credential_type != CredentialType.UNAUTHORIZED: + credential_encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=provider_controller.get_credential_schema_config(credential_type), + cache=NoOpProviderCredentialCache(), + ) + + properties_encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=provider_controller.get_properties_schema(), + cache=NoOpProviderCredentialCache(), + ) + + # Create provider record + subscription = TriggerSubscription( + id=subscription_id or str(uuid.uuid4()), + tenant_id=tenant_id, + user_id=user_id, + name=name, + endpoint_id=endpoint_id, + provider_id=str(provider_id), + parameters=parameters, + properties=properties_encrypter.encrypt(dict(properties)), + credentials=credential_encrypter.encrypt(dict(credentials)) if credential_encrypter else {}, + credential_type=credential_type.value, + credential_expires_at=credential_expires_at, + expires_at=expires_at, + ) + + session.add(subscription) + session.commit() + + return { + "result": "success", + "id": str(subscription.id), + } + + except Exception as e: + logger.exception("Failed to add trigger provider") + raise ValueError(str(e)) + + @classmethod + def get_subscription_by_id(cls, tenant_id: str, subscription_id: str | None = None) -> TriggerSubscription | None: + """ + Get a trigger subscription by the ID. + """ + with Session(db.engine, expire_on_commit=False) as session: + subscription: TriggerSubscription | None = None + if subscription_id: + subscription = ( + session.query(TriggerSubscription).filter_by(tenant_id=tenant_id, id=subscription_id).first() + ) + else: + subscription = session.query(TriggerSubscription).filter_by(tenant_id=tenant_id).first() + if subscription: + provider_controller = TriggerManager.get_trigger_provider( + tenant_id, TriggerProviderID(subscription.provider_id) + ) + encrypter, _ = create_trigger_provider_encrypter_for_subscription( + tenant_id=tenant_id, + controller=provider_controller, + subscription=subscription, + ) + subscription.credentials = dict(encrypter.decrypt(subscription.credentials)) + properties_encrypter, _ = create_trigger_provider_encrypter_for_properties( + tenant_id=subscription.tenant_id, + controller=provider_controller, + subscription=subscription, + ) + subscription.properties = dict(properties_encrypter.decrypt(subscription.properties)) + return subscription + + @classmethod + def delete_trigger_provider(cls, session: Session, tenant_id: str, subscription_id: str): + """ + Delete a trigger provider subscription within an existing session. + + :param session: Database session + :param tenant_id: Tenant ID + :param subscription_id: Subscription instance ID + :return: Success response + """ + subscription: TriggerSubscription | None = ( + session.query(TriggerSubscription).filter_by(tenant_id=tenant_id, id=subscription_id).first() + ) + if not subscription: + raise ValueError(f"Trigger provider subscription {subscription_id} not found") + + credential_type: CredentialType = CredentialType.of(subscription.credential_type) + is_auto_created: bool = credential_type in [CredentialType.OAUTH2, CredentialType.API_KEY] + if is_auto_created: + provider_id = TriggerProviderID(subscription.provider_id) + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + encrypter, _ = create_trigger_provider_encrypter_for_subscription( + tenant_id=tenant_id, + controller=provider_controller, + subscription=subscription, + ) + try: + TriggerManager.unsubscribe_trigger( + tenant_id=tenant_id, + user_id=subscription.user_id, + provider_id=provider_id, + subscription=subscription.to_entity(), + credentials=encrypter.decrypt(subscription.credentials), + credential_type=credential_type, + ) + except Exception as e: + logger.exception("Error unsubscribing trigger", exc_info=e) + + # Clear cache + session.delete(subscription) + delete_cache_for_subscription( + tenant_id=tenant_id, + provider_id=subscription.provider_id, + subscription_id=subscription.id, + ) + + @classmethod + def refresh_oauth_token( + cls, + tenant_id: str, + subscription_id: str, + ) -> Mapping[str, Any]: + """ + Refresh OAuth token for a trigger provider. + + :param tenant_id: Tenant ID + :param subscription_id: Subscription instance ID + :return: New token info + """ + with Session(db.engine) as session: + subscription = session.query(TriggerSubscription).filter_by(tenant_id=tenant_id, id=subscription_id).first() + + if not subscription: + raise ValueError(f"Trigger provider subscription {subscription_id} not found") + + if subscription.credential_type != CredentialType.OAUTH2.value: + raise ValueError("Only OAuth credentials can be refreshed") + + provider_id = TriggerProviderID(subscription.provider_id) + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + # Create encrypter + encrypter, cache = create_provider_encrypter( + tenant_id=tenant_id, + config=[x.to_basic_provider_config() for x in provider_controller.get_oauth_client_schema()], + cache=NoOpProviderCredentialCache(), + ) + + # Decrypt current credentials + current_credentials = encrypter.decrypt(subscription.credentials) + + # Get OAuth client configuration + redirect_uri = ( + f"{dify_config.CONSOLE_API_URL}/console/api/oauth/plugin/{subscription.provider_id}/trigger/callback" + ) + system_credentials = cls.get_oauth_client(tenant_id, provider_id) + + # Refresh token + oauth_handler = OAuthHandler() + refreshed_credentials = oauth_handler.refresh_credentials( + tenant_id=tenant_id, + user_id=subscription.user_id, + plugin_id=provider_id.plugin_id, + provider=provider_id.provider_name, + redirect_uri=redirect_uri, + system_credentials=system_credentials or {}, + credentials=current_credentials, + ) + + # Update credentials + subscription.credentials = dict(encrypter.encrypt(dict(refreshed_credentials.credentials))) + subscription.credential_expires_at = refreshed_credentials.expires_at + session.commit() + + # Clear cache + cache.delete() + + return { + "result": "success", + "expires_at": refreshed_credentials.expires_at, + } + + @classmethod + def refresh_subscription( + cls, + tenant_id: str, + subscription_id: str, + now: int | None = None, + ) -> Mapping[str, Any]: + """ + Refresh trigger subscription if expired. + + Args: + tenant_id: Tenant ID + subscription_id: Subscription instance ID + now: Current timestamp, defaults to `int(time.time())` + + Returns: + Mapping with keys: `result` ("success"|"skipped") and `expires_at` (new or existing value) + """ + now_ts: int = int(now if now is not None else _time.time()) + + with Session(db.engine) as session: + subscription: TriggerSubscription | None = ( + session.query(TriggerSubscription).filter_by(tenant_id=tenant_id, id=subscription_id).first() + ) + if subscription is None: + raise ValueError(f"Trigger provider subscription {subscription_id} not found") + + if subscription.expires_at == -1 or int(subscription.expires_at) > now_ts: + logger.debug( + "Subscription not due for refresh: tenant=%s id=%s expires_at=%s now=%s", + tenant_id, + subscription_id, + subscription.expires_at, + now_ts, + ) + return {"result": "skipped", "expires_at": int(subscription.expires_at)} + + provider_id = TriggerProviderID(subscription.provider_id) + controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + + # Decrypt credentials and properties for runtime + credential_encrypter, _ = create_trigger_provider_encrypter_for_subscription( + tenant_id=tenant_id, + controller=controller, + subscription=subscription, + ) + properties_encrypter, properties_cache = create_trigger_provider_encrypter_for_properties( + tenant_id=tenant_id, + controller=controller, + subscription=subscription, + ) + + decrypted_credentials = credential_encrypter.decrypt(subscription.credentials) + decrypted_properties = properties_encrypter.decrypt(subscription.properties) + + sub_entity: TriggerSubscriptionEntity = TriggerSubscriptionEntity( + expires_at=int(subscription.expires_at), + endpoint=generate_plugin_trigger_endpoint_url(subscription.endpoint_id), + parameters=subscription.parameters, + properties=decrypted_properties, + ) + + refreshed: TriggerSubscriptionEntity = controller.refresh_trigger( + subscription=sub_entity, + credentials=decrypted_credentials, + credential_type=CredentialType.of(subscription.credential_type), + ) + + # Persist refreshed properties and expires_at + subscription.properties = dict(properties_encrypter.encrypt(dict(refreshed.properties))) + subscription.expires_at = int(refreshed.expires_at) + session.commit() + properties_cache.delete() + + logger.info( + "Subscription refreshed (service): tenant=%s id=%s new_expires_at=%s", + tenant_id, + subscription_id, + subscription.expires_at, + ) + + return {"result": "success", "expires_at": int(refreshed.expires_at)} + + @classmethod + def get_oauth_client(cls, tenant_id: str, provider_id: TriggerProviderID) -> Mapping[str, Any] | None: + """ + Get OAuth client configuration for a provider. + First tries tenant-level OAuth, then falls back to system OAuth. + + :param tenant_id: Tenant ID + :param provider_id: Provider identifier + :return: OAuth client configuration or None + """ + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + with Session(db.engine, expire_on_commit=False) as session: + tenant_client: TriggerOAuthTenantClient | None = ( + session.query(TriggerOAuthTenantClient) + .filter_by( + tenant_id=tenant_id, + provider=provider_id.provider_name, + plugin_id=provider_id.plugin_id, + enabled=True, + ) + .first() + ) + + oauth_params: Mapping[str, Any] | None = None + if tenant_client: + encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=[x.to_basic_provider_config() for x in provider_controller.get_oauth_client_schema()], + cache=NoOpProviderCredentialCache(), + ) + oauth_params = encrypter.decrypt(dict(tenant_client.oauth_params)) + return oauth_params + + is_verified = PluginService.is_plugin_verified(tenant_id, provider_id.plugin_id) + if not is_verified: + return None + + # Check for system-level OAuth client + system_client: TriggerOAuthSystemClient | None = ( + session.query(TriggerOAuthSystemClient) + .filter_by(plugin_id=provider_id.plugin_id, provider=provider_id.provider_name) + .first() + ) + + if system_client: + try: + oauth_params = decrypt_system_oauth_params(system_client.encrypted_oauth_params) + except Exception as e: + raise ValueError(f"Error decrypting system oauth params: {e}") + + return oauth_params + + @classmethod + def is_oauth_system_client_exists(cls, tenant_id: str, provider_id: TriggerProviderID) -> bool: + """ + Check if system OAuth client exists for a trigger provider. + """ + is_verified = PluginService.is_plugin_verified(tenant_id, provider_id.plugin_id) + if not is_verified: + return False + with Session(db.engine, expire_on_commit=False) as session: + system_client: TriggerOAuthSystemClient | None = ( + session.query(TriggerOAuthSystemClient) + .filter_by(plugin_id=provider_id.plugin_id, provider=provider_id.provider_name) + .first() + ) + return system_client is not None + + @classmethod + def save_custom_oauth_client_params( + cls, + tenant_id: str, + provider_id: TriggerProviderID, + client_params: Mapping[str, Any] | None = None, + enabled: bool | None = None, + ) -> Mapping[str, Any]: + """ + Save or update custom OAuth client parameters for a trigger provider. + + :param tenant_id: Tenant ID + :param provider_id: Provider identifier + :param client_params: OAuth client parameters (client_id, client_secret, etc.) + :param enabled: Enable/disable the custom OAuth client + :return: Success response + """ + if client_params is None and enabled is None: + return {"result": "success"} + + # Get provider controller to access schema + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + + with Session(db.engine) as session: + # Find existing custom client params + custom_client = ( + session.query(TriggerOAuthTenantClient) + .filter_by( + tenant_id=tenant_id, + plugin_id=provider_id.plugin_id, + provider=provider_id.provider_name, + ) + .first() + ) + + # Create new record if doesn't exist + if custom_client is None: + custom_client = TriggerOAuthTenantClient( + tenant_id=tenant_id, + plugin_id=provider_id.plugin_id, + provider=provider_id.provider_name, + ) + session.add(custom_client) + + # Update client params if provided + if client_params is None: + custom_client.encrypted_oauth_params = json.dumps({}) + else: + encrypter, cache = create_provider_encrypter( + tenant_id=tenant_id, + config=[x.to_basic_provider_config() for x in provider_controller.get_oauth_client_schema()], + cache=NoOpProviderCredentialCache(), + ) + + # Handle hidden values + original_params = encrypter.decrypt(dict(custom_client.oauth_params)) + new_params: dict[str, Any] = { + key: value if value != HIDDEN_VALUE else original_params.get(key, UNKNOWN_VALUE) + for key, value in client_params.items() + } + custom_client.encrypted_oauth_params = json.dumps(encrypter.encrypt(new_params)) + cache.delete() + + # Update enabled status if provided + if enabled is not None: + custom_client.enabled = enabled + + session.commit() + + return {"result": "success"} + + @classmethod + def get_custom_oauth_client_params(cls, tenant_id: str, provider_id: TriggerProviderID) -> Mapping[str, Any]: + """ + Get custom OAuth client parameters for a trigger provider. + + :param tenant_id: Tenant ID + :param provider_id: Provider identifier + :return: Masked OAuth client parameters + """ + with Session(db.engine) as session: + custom_client = ( + session.query(TriggerOAuthTenantClient) + .filter_by( + tenant_id=tenant_id, + plugin_id=provider_id.plugin_id, + provider=provider_id.provider_name, + ) + .first() + ) + + if custom_client is None: + return {} + + # Get provider controller to access schema + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=tenant_id, provider_id=provider_id + ) + + # Create encrypter to decrypt and mask values + encrypter, _ = create_provider_encrypter( + tenant_id=tenant_id, + config=[x.to_basic_provider_config() for x in provider_controller.get_oauth_client_schema()], + cache=NoOpProviderCredentialCache(), + ) + + return encrypter.mask_plugin_credentials(encrypter.decrypt(dict(custom_client.oauth_params))) + + @classmethod + def delete_custom_oauth_client_params(cls, tenant_id: str, provider_id: TriggerProviderID) -> Mapping[str, Any]: + """ + Delete custom OAuth client parameters for a trigger provider. + + :param tenant_id: Tenant ID + :param provider_id: Provider identifier + :return: Success response + """ + with Session(db.engine) as session: + session.query(TriggerOAuthTenantClient).filter_by( + tenant_id=tenant_id, + provider=provider_id.provider_name, + plugin_id=provider_id.plugin_id, + ).delete() + session.commit() + + return {"result": "success"} + + @classmethod + def is_oauth_custom_client_enabled(cls, tenant_id: str, provider_id: TriggerProviderID) -> bool: + """ + Check if custom OAuth client is enabled for a trigger provider. + + :param tenant_id: Tenant ID + :param provider_id: Provider identifier + :return: True if enabled, False otherwise + """ + with Session(db.engine, expire_on_commit=False) as session: + custom_client = ( + session.query(TriggerOAuthTenantClient) + .filter_by( + tenant_id=tenant_id, + plugin_id=provider_id.plugin_id, + provider=provider_id.provider_name, + enabled=True, + ) + .first() + ) + return custom_client is not None + + @classmethod + def get_subscription_by_endpoint(cls, endpoint_id: str) -> TriggerSubscription | None: + """ + Get a trigger subscription by the endpoint ID. + """ + with Session(db.engine, expire_on_commit=False) as session: + subscription = session.query(TriggerSubscription).filter_by(endpoint_id=endpoint_id).first() + if not subscription: + return None + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=subscription.tenant_id, provider_id=TriggerProviderID(subscription.provider_id) + ) + credential_encrypter, _ = create_trigger_provider_encrypter_for_subscription( + tenant_id=subscription.tenant_id, + controller=provider_controller, + subscription=subscription, + ) + subscription.credentials = dict(credential_encrypter.decrypt(subscription.credentials)) + + properties_encrypter, _ = create_trigger_provider_encrypter_for_properties( + tenant_id=subscription.tenant_id, + controller=provider_controller, + subscription=subscription, + ) + subscription.properties = dict(properties_encrypter.decrypt(subscription.properties)) + return subscription diff --git a/api/services/trigger/trigger_request_service.py b/api/services/trigger/trigger_request_service.py new file mode 100644 index 0000000000..91a838c265 --- /dev/null +++ b/api/services/trigger/trigger_request_service.py @@ -0,0 +1,65 @@ +from collections.abc import Mapping +from typing import Any + +from flask import Request +from pydantic import TypeAdapter + +from core.plugin.utils.http_parser import deserialize_request, serialize_request +from extensions.ext_storage import storage + + +class TriggerHttpRequestCachingService: + """ + Service for caching trigger requests. + """ + + _TRIGGER_STORAGE_PATH = "triggers" + + @classmethod + def get_request(cls, request_id: str) -> Request: + """ + Get the request object from the storage. + + Args: + request_id: The ID of the request. + + Returns: + The request object. + """ + return deserialize_request(storage.load_once(f"{cls._TRIGGER_STORAGE_PATH}/{request_id}.raw")) + + @classmethod + def get_payload(cls, request_id: str) -> Mapping[str, Any]: + """ + Get the payload from the storage. + + Args: + request_id: The ID of the request. + + Returns: + The payload. + """ + return TypeAdapter(Mapping[str, Any]).validate_json( + storage.load_once(f"{cls._TRIGGER_STORAGE_PATH}/{request_id}.payload") + ) + + @classmethod + def persist_request(cls, request_id: str, request: Request) -> None: + """ + Persist the request in the storage. + + Args: + request_id: The ID of the request. + request: The request object. + """ + storage.save(f"{cls._TRIGGER_STORAGE_PATH}/{request_id}.raw", serialize_request(request)) + + @classmethod + def persist_payload(cls, request_id: str, payload: Mapping[str, Any]) -> None: + """ + Persist the payload in the storage. + """ + storage.save( + f"{cls._TRIGGER_STORAGE_PATH}/{request_id}.payload", + TypeAdapter(Mapping[str, Any]).dump_json(payload), # type: ignore + ) diff --git a/api/services/trigger/trigger_service.py b/api/services/trigger/trigger_service.py new file mode 100644 index 0000000000..0255e42546 --- /dev/null +++ b/api/services/trigger/trigger_service.py @@ -0,0 +1,307 @@ +import logging +import secrets +import time +from collections.abc import Mapping +from typing import Any + +from flask import Request, Response +from pydantic import BaseModel +from sqlalchemy import select +from sqlalchemy.orm import Session + +from core.plugin.entities.plugin_daemon import CredentialType +from core.plugin.entities.request import TriggerDispatchResponse, TriggerInvokeEventResponse +from core.plugin.impl.exc import PluginNotFoundError +from core.trigger.debug.events import PluginTriggerDebugEvent +from core.trigger.provider import PluginTriggerProviderController +from core.trigger.trigger_manager import TriggerManager +from core.trigger.utils.encryption import create_trigger_provider_encrypter_for_subscription +from core.workflow.enums import NodeType +from core.workflow.nodes.trigger_plugin.entities import TriggerEventNodeData +from extensions.ext_database import db +from extensions.ext_redis import redis_client +from models.model import App +from models.provider_ids import TriggerProviderID +from models.trigger import TriggerSubscription, WorkflowPluginTrigger +from models.workflow import Workflow +from services.trigger.trigger_provider_service import TriggerProviderService +from services.trigger.trigger_request_service import TriggerHttpRequestCachingService +from services.workflow.entities import PluginTriggerDispatchData +from tasks.trigger_processing_tasks import dispatch_triggered_workflows_async + +logger = logging.getLogger(__name__) + + +class TriggerService: + __TEMPORARY_ENDPOINT_EXPIRE_MS__ = 5 * 60 * 1000 + __ENDPOINT_REQUEST_CACHE_COUNT__ = 10 + __ENDPOINT_REQUEST_CACHE_EXPIRE_MS__ = 5 * 60 * 1000 + __PLUGIN_TRIGGER_NODE_CACHE_KEY__ = "plugin_trigger_nodes" + MAX_PLUGIN_TRIGGER_NODES_PER_WORKFLOW = 5 # Maximum allowed plugin trigger nodes per workflow + + @classmethod + def invoke_trigger_event( + cls, tenant_id: str, user_id: str, node_config: Mapping[str, Any], event: PluginTriggerDebugEvent + ) -> TriggerInvokeEventResponse: + """Invoke a trigger event.""" + subscription: TriggerSubscription | None = TriggerProviderService.get_subscription_by_id( + tenant_id=tenant_id, + subscription_id=event.subscription_id, + ) + if not subscription: + raise ValueError("Subscription not found") + node_data: TriggerEventNodeData = TriggerEventNodeData.model_validate(node_config.get("data", {})) + request = TriggerHttpRequestCachingService.get_request(event.request_id) + payload = TriggerHttpRequestCachingService.get_payload(event.request_id) + # invoke triger + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id, TriggerProviderID(subscription.provider_id) + ) + return TriggerManager.invoke_trigger_event( + tenant_id=tenant_id, + user_id=user_id, + provider_id=TriggerProviderID(event.provider_id), + event_name=event.name, + parameters=node_data.resolve_parameters( + parameter_schemas=provider_controller.get_event_parameters(event_name=event.name) + ), + credentials=subscription.credentials, + credential_type=CredentialType.of(subscription.credential_type), + subscription=subscription.to_entity(), + request=request, + payload=payload, + ) + + @classmethod + def process_endpoint(cls, endpoint_id: str, request: Request) -> Response | None: + """ + Extract and process data from incoming endpoint request. + + Args: + endpoint_id: Endpoint ID + request: Request + """ + timestamp = int(time.time()) + subscription: TriggerSubscription | None = None + try: + subscription = TriggerProviderService.get_subscription_by_endpoint(endpoint_id) + except PluginNotFoundError: + return Response(status=404, response="Trigger provider not found") + except Exception: + return Response(status=500, response="Failed to get subscription by endpoint") + + if not subscription: + return None + + provider_id = TriggerProviderID(subscription.provider_id) + controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=subscription.tenant_id, provider_id=provider_id + ) + encrypter, _ = create_trigger_provider_encrypter_for_subscription( + tenant_id=subscription.tenant_id, + controller=controller, + subscription=subscription, + ) + dispatch_response: TriggerDispatchResponse = controller.dispatch( + request=request, + subscription=subscription.to_entity(), + credentials=encrypter.decrypt(subscription.credentials), + credential_type=CredentialType.of(subscription.credential_type), + ) + + if dispatch_response.events: + request_id = f"trigger_request_{timestamp}_{secrets.token_hex(6)}" + + # save the request and payload to storage as persistent data + TriggerHttpRequestCachingService.persist_request(request_id, request) + TriggerHttpRequestCachingService.persist_payload(request_id, dispatch_response.payload) + + # Validate event names + for event_name in dispatch_response.events: + if controller.get_event(event_name) is None: + logger.error( + "Event name %s not found in provider %s for endpoint %s", + event_name, + subscription.provider_id, + endpoint_id, + ) + raise ValueError(f"Event name {event_name} not found in provider {subscription.provider_id}") + + plugin_trigger_dispatch_data = PluginTriggerDispatchData( + user_id=dispatch_response.user_id, + tenant_id=subscription.tenant_id, + endpoint_id=endpoint_id, + provider_id=subscription.provider_id, + subscription_id=subscription.id, + timestamp=timestamp, + events=list(dispatch_response.events), + request_id=request_id, + ) + dispatch_data = plugin_trigger_dispatch_data.model_dump(mode="json") + dispatch_triggered_workflows_async.delay(dispatch_data) + + logger.info( + "Queued async dispatching for %d triggers on endpoint %s with request_id %s", + len(dispatch_response.events), + endpoint_id, + request_id, + ) + return dispatch_response.response + + @classmethod + def sync_plugin_trigger_relationships(cls, app: App, workflow: Workflow): + """ + Sync plugin trigger relationships in DB. + + 1. Check if the workflow has any plugin trigger nodes + 2. Fetch the nodes from DB, see if there were any plugin trigger records already + 3. Diff the nodes and the plugin trigger records, create/update/delete the records as needed + + Approach: + Frequent DB operations may cause performance issues, using Redis to cache it instead. + If any record exists, cache it. + + Limits: + - Maximum 5 plugin trigger nodes per workflow + """ + + class Cache(BaseModel): + """ + Cache model for plugin trigger nodes + """ + + record_id: str + node_id: str + provider_id: str + event_name: str + subscription_id: str + + # Walk nodes to find plugin triggers + nodes_in_graph: list[Mapping[str, Any]] = [] + for node_id, node_config in workflow.walk_nodes(NodeType.TRIGGER_PLUGIN): + # Extract plugin trigger configuration from node + plugin_id = node_config.get("plugin_id", "") + provider_id = node_config.get("provider_id", "") + event_name = node_config.get("event_name", "") + subscription_id = node_config.get("subscription_id", "") + + if not subscription_id: + continue + + nodes_in_graph.append( + { + "node_id": node_id, + "plugin_id": plugin_id, + "provider_id": provider_id, + "event_name": event_name, + "subscription_id": subscription_id, + } + ) + + # Check plugin trigger node limit + if len(nodes_in_graph) > cls.MAX_PLUGIN_TRIGGER_NODES_PER_WORKFLOW: + raise ValueError( + f"Workflow exceeds maximum plugin trigger node limit. " + f"Found {len(nodes_in_graph)} plugin trigger nodes, " + f"maximum allowed is {cls.MAX_PLUGIN_TRIGGER_NODES_PER_WORKFLOW}" + ) + + not_found_in_cache: list[Mapping[str, Any]] = [] + for node_info in nodes_in_graph: + node_id = node_info["node_id"] + # firstly check if the node exists in cache + if not redis_client.get(f"{cls.__PLUGIN_TRIGGER_NODE_CACHE_KEY__}:{node_id}"): + not_found_in_cache.append(node_info) + continue + + with Session(db.engine) as session: + try: + # lock the concurrent plugin trigger creation + redis_client.lock(f"{cls.__PLUGIN_TRIGGER_NODE_CACHE_KEY__}:apps:{app.id}:lock", timeout=10) + # fetch the non-cached nodes from DB + all_records = session.scalars( + select(WorkflowPluginTrigger).where( + WorkflowPluginTrigger.app_id == app.id, + WorkflowPluginTrigger.tenant_id == app.tenant_id, + ) + ).all() + + nodes_id_in_db = {node.node_id: node for node in all_records} + nodes_id_in_graph = {node["node_id"] for node in nodes_in_graph} + + # get the nodes not found both in cache and DB + nodes_not_found = [ + node_info for node_info in not_found_in_cache if node_info["node_id"] not in nodes_id_in_db + ] + + # create new plugin trigger records + for node_info in nodes_not_found: + plugin_trigger = WorkflowPluginTrigger( + app_id=app.id, + tenant_id=app.tenant_id, + node_id=node_info["node_id"], + provider_id=node_info["provider_id"], + event_name=node_info["event_name"], + subscription_id=node_info["subscription_id"], + ) + session.add(plugin_trigger) + session.flush() # Get the ID for caching + + cache = Cache( + record_id=plugin_trigger.id, + node_id=node_info["node_id"], + provider_id=node_info["provider_id"], + event_name=node_info["event_name"], + subscription_id=node_info["subscription_id"], + ) + redis_client.set( + f"{cls.__PLUGIN_TRIGGER_NODE_CACHE_KEY__}:{node_info['node_id']}", + cache.model_dump_json(), + ex=60 * 60, + ) + session.commit() + + # Update existing records if subscription_id changed + for node_info in nodes_in_graph: + node_id = node_info["node_id"] + if node_id in nodes_id_in_db: + existing_record = nodes_id_in_db[node_id] + if ( + existing_record.subscription_id != node_info["subscription_id"] + or existing_record.provider_id != node_info["provider_id"] + or existing_record.event_name != node_info["event_name"] + ): + existing_record.subscription_id = node_info["subscription_id"] + existing_record.provider_id = node_info["provider_id"] + existing_record.event_name = node_info["event_name"] + session.add(existing_record) + + # Update cache + cache = Cache( + record_id=existing_record.id, + node_id=node_id, + provider_id=node_info["provider_id"], + event_name=node_info["event_name"], + subscription_id=node_info["subscription_id"], + ) + redis_client.set( + f"{cls.__PLUGIN_TRIGGER_NODE_CACHE_KEY__}:{node_id}", + cache.model_dump_json(), + ex=60 * 60, + ) + session.commit() + + # delete the nodes not found in the graph + for node_id in nodes_id_in_db: + if node_id not in nodes_id_in_graph: + session.delete(nodes_id_in_db[node_id]) + redis_client.delete(f"{cls.__PLUGIN_TRIGGER_NODE_CACHE_KEY__}:{node_id}") + session.commit() + except Exception: + import logging + + logger = logging.getLogger(__name__) + logger.exception("Failed to sync plugin trigger relationships for app %s", app.id) + raise + finally: + redis_client.delete(f"{cls.__PLUGIN_TRIGGER_NODE_CACHE_KEY__}:apps:{app.id}:lock") diff --git a/api/services/trigger/trigger_subscription_builder_service.py b/api/services/trigger/trigger_subscription_builder_service.py new file mode 100644 index 0000000000..571393c782 --- /dev/null +++ b/api/services/trigger/trigger_subscription_builder_service.py @@ -0,0 +1,492 @@ +import json +import logging +import uuid +from collections.abc import Mapping +from contextlib import contextmanager +from datetime import datetime +from typing import Any + +from flask import Request, Response + +from core.plugin.entities.plugin_daemon import CredentialType +from core.plugin.entities.request import TriggerDispatchResponse +from core.tools.errors import ToolProviderCredentialValidationError +from core.trigger.entities.api_entities import SubscriptionBuilderApiEntity +from core.trigger.entities.entities import ( + RequestLog, + Subscription, + SubscriptionBuilder, + SubscriptionBuilderUpdater, + SubscriptionConstructor, +) +from core.trigger.provider import PluginTriggerProviderController +from core.trigger.trigger_manager import TriggerManager +from core.trigger.utils.encryption import masked_credentials +from core.trigger.utils.endpoint import generate_plugin_trigger_endpoint_url +from extensions.ext_redis import redis_client +from models.provider_ids import TriggerProviderID +from services.trigger.trigger_provider_service import TriggerProviderService + +logger = logging.getLogger(__name__) + + +class TriggerSubscriptionBuilderService: + """Service for managing trigger providers and credentials""" + + ########################## + # Trigger provider + ########################## + __MAX_TRIGGER_PROVIDER_COUNT__ = 10 + + ########################## + # Builder endpoint + ########################## + __BUILDER_CACHE_EXPIRE_SECONDS__ = 30 * 60 + + __VALIDATION_REQUEST_CACHE_COUNT__ = 10 + __VALIDATION_REQUEST_CACHE_EXPIRE_SECONDS__ = 30 * 60 + + ########################## + # Distributed lock + ########################## + __LOCK_EXPIRE_SECONDS__ = 30 + + @classmethod + def encode_cache_key(cls, subscription_id: str) -> str: + return f"trigger:subscription:builder:{subscription_id}" + + @classmethod + def encode_lock_key(cls, subscription_id: str) -> str: + return f"trigger:subscription:builder:lock:{subscription_id}" + + @classmethod + @contextmanager + def acquire_builder_lock(cls, subscription_id: str): + """ + Acquire a distributed lock for a subscription builder. + + :param subscription_id: The subscription builder ID + """ + lock_key = cls.encode_lock_key(subscription_id) + with redis_client.lock(lock_key, timeout=cls.__LOCK_EXPIRE_SECONDS__): + yield + + @classmethod + def verify_trigger_subscription_builder( + cls, + tenant_id: str, + user_id: str, + provider_id: TriggerProviderID, + subscription_builder_id: str, + ) -> Mapping[str, Any]: + """Verify a trigger subscription builder""" + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + if not provider_controller: + raise ValueError(f"Provider {provider_id} not found") + + subscription_builder = cls.get_subscription_builder(subscription_builder_id) + if not subscription_builder: + raise ValueError(f"Subscription builder {subscription_builder_id} not found") + + if subscription_builder.credential_type == CredentialType.OAUTH2: + return {"verified": bool(subscription_builder.credentials)} + + if subscription_builder.credential_type == CredentialType.API_KEY: + credentials_to_validate = subscription_builder.credentials + try: + provider_controller.validate_credentials(user_id, credentials_to_validate) + except ToolProviderCredentialValidationError as e: + raise ValueError(f"Invalid credentials: {e}") + return {"verified": True} + + return {"verified": True} + + @classmethod + def build_trigger_subscription_builder( + cls, tenant_id: str, user_id: str, provider_id: TriggerProviderID, subscription_builder_id: str + ) -> None: + """Build a trigger subscription builder""" + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + if not provider_controller: + raise ValueError(f"Provider {provider_id} not found") + + # Acquire lock to prevent concurrent build operations + with cls.acquire_builder_lock(subscription_builder_id): + subscription_builder = cls.get_subscription_builder(subscription_builder_id) + if not subscription_builder: + raise ValueError(f"Subscription builder {subscription_builder_id} not found") + + if not subscription_builder.name: + raise ValueError("Subscription builder name is required") + + credential_type = CredentialType.of( + subscription_builder.credential_type or CredentialType.UNAUTHORIZED.value + ) + if credential_type == CredentialType.UNAUTHORIZED: + # manually create + TriggerProviderService.add_trigger_subscription( + subscription_id=subscription_builder.id, + tenant_id=tenant_id, + user_id=user_id, + name=subscription_builder.name, + provider_id=provider_id, + endpoint_id=subscription_builder.endpoint_id, + parameters=subscription_builder.parameters, + properties=subscription_builder.properties, + credential_expires_at=subscription_builder.credential_expires_at or -1, + expires_at=subscription_builder.expires_at, + credentials=subscription_builder.credentials, + credential_type=credential_type, + ) + else: + # automatically create + subscription: Subscription = TriggerManager.subscribe_trigger( + tenant_id=tenant_id, + user_id=user_id, + provider_id=provider_id, + endpoint=generate_plugin_trigger_endpoint_url(subscription_builder.endpoint_id), + parameters=subscription_builder.parameters, + credentials=subscription_builder.credentials, + credential_type=credential_type, + ) + + TriggerProviderService.add_trigger_subscription( + subscription_id=subscription_builder.id, + tenant_id=tenant_id, + user_id=user_id, + name=subscription_builder.name, + provider_id=provider_id, + endpoint_id=subscription_builder.endpoint_id, + parameters=subscription_builder.parameters, + properties=subscription.properties, + credentials=subscription_builder.credentials, + credential_type=credential_type, + credential_expires_at=subscription_builder.credential_expires_at or -1, + expires_at=subscription_builder.expires_at, + ) + + # Delete the builder after successful subscription creation + cache_key = cls.encode_cache_key(subscription_builder_id) + redis_client.delete(cache_key) + + @classmethod + def create_trigger_subscription_builder( + cls, + tenant_id: str, + user_id: str, + provider_id: TriggerProviderID, + credential_type: CredentialType, + ) -> SubscriptionBuilderApiEntity: + """ + Add a new trigger subscription validation. + """ + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + if not provider_controller: + raise ValueError(f"Provider {provider_id} not found") + + subscription_constructor: SubscriptionConstructor | None = provider_controller.get_subscription_constructor() + subscription_id = str(uuid.uuid4()) + subscription_builder = SubscriptionBuilder( + id=subscription_id, + name=None, + endpoint_id=subscription_id, + tenant_id=tenant_id, + user_id=user_id, + provider_id=str(provider_id), + parameters=subscription_constructor.get_default_parameters() if subscription_constructor else {}, + properties=provider_controller.get_subscription_default_properties(), + credentials={}, + credential_type=credential_type, + credential_expires_at=-1, + expires_at=-1, + ) + cache_key = cls.encode_cache_key(subscription_id) + redis_client.setex(cache_key, cls.__BUILDER_CACHE_EXPIRE_SECONDS__, subscription_builder.model_dump_json()) + return cls.builder_to_api_entity(controller=provider_controller, entity=subscription_builder) + + @classmethod + def update_trigger_subscription_builder( + cls, + tenant_id: str, + provider_id: TriggerProviderID, + subscription_builder_id: str, + subscription_builder_updater: SubscriptionBuilderUpdater, + ) -> SubscriptionBuilderApiEntity: + """ + Update a trigger subscription validation. + """ + subscription_id = subscription_builder_id + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + if not provider_controller: + raise ValueError(f"Provider {provider_id} not found") + + # Acquire lock to prevent concurrent updates + with cls.acquire_builder_lock(subscription_id): + cache_key = cls.encode_cache_key(subscription_id) + subscription_builder_cache = cls.get_subscription_builder(subscription_builder_id) + if not subscription_builder_cache or subscription_builder_cache.tenant_id != tenant_id: + raise ValueError(f"Subscription {subscription_id} expired or not found") + + subscription_builder_updater.update(subscription_builder_cache) + + redis_client.setex( + cache_key, cls.__BUILDER_CACHE_EXPIRE_SECONDS__, subscription_builder_cache.model_dump_json() + ) + return cls.builder_to_api_entity(controller=provider_controller, entity=subscription_builder_cache) + + @classmethod + def update_and_verify_builder( + cls, + tenant_id: str, + user_id: str, + provider_id: TriggerProviderID, + subscription_builder_id: str, + subscription_builder_updater: SubscriptionBuilderUpdater, + ) -> Mapping[str, Any]: + """ + Atomically update and verify a subscription builder. + This ensures the verification is done on the exact data that was just updated. + """ + subscription_id = subscription_builder_id + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + if not provider_controller: + raise ValueError(f"Provider {provider_id} not found") + + # Acquire lock for the entire update + verify operation + with cls.acquire_builder_lock(subscription_id): + cache_key = cls.encode_cache_key(subscription_id) + subscription_builder_cache = cls.get_subscription_builder(subscription_builder_id) + if not subscription_builder_cache or subscription_builder_cache.tenant_id != tenant_id: + raise ValueError(f"Subscription {subscription_id} expired or not found") + + # Update + subscription_builder_updater.update(subscription_builder_cache) + redis_client.setex( + cache_key, cls.__BUILDER_CACHE_EXPIRE_SECONDS__, subscription_builder_cache.model_dump_json() + ) + + # Verify (using the just-updated data) + if subscription_builder_cache.credential_type == CredentialType.OAUTH2: + return {"verified": bool(subscription_builder_cache.credentials)} + + if subscription_builder_cache.credential_type == CredentialType.API_KEY: + credentials_to_validate = subscription_builder_cache.credentials + try: + provider_controller.validate_credentials(user_id, credentials_to_validate) + except ToolProviderCredentialValidationError as e: + raise ValueError(f"Invalid credentials: {e}") + return {"verified": True} + + return {"verified": True} + + @classmethod + def update_and_build_builder( + cls, + tenant_id: str, + user_id: str, + provider_id: TriggerProviderID, + subscription_builder_id: str, + subscription_builder_updater: SubscriptionBuilderUpdater, + ) -> None: + """ + Atomically update and build a subscription builder. + This ensures the build uses the exact data that was just updated. + """ + subscription_id = subscription_builder_id + provider_controller = TriggerManager.get_trigger_provider(tenant_id, provider_id) + if not provider_controller: + raise ValueError(f"Provider {provider_id} not found") + + # Acquire lock for the entire update + build operation + with cls.acquire_builder_lock(subscription_id): + cache_key = cls.encode_cache_key(subscription_id) + subscription_builder_cache = cls.get_subscription_builder(subscription_builder_id) + if not subscription_builder_cache or subscription_builder_cache.tenant_id != tenant_id: + raise ValueError(f"Subscription {subscription_id} expired or not found") + + # Update + subscription_builder_updater.update(subscription_builder_cache) + redis_client.setex( + cache_key, cls.__BUILDER_CACHE_EXPIRE_SECONDS__, subscription_builder_cache.model_dump_json() + ) + + # Re-fetch to ensure we have the latest data + subscription_builder = cls.get_subscription_builder(subscription_builder_id) + if not subscription_builder: + raise ValueError(f"Subscription builder {subscription_builder_id} not found") + + if not subscription_builder.name: + raise ValueError("Subscription builder name is required") + + # Build + credential_type = CredentialType.of( + subscription_builder.credential_type or CredentialType.UNAUTHORIZED.value + ) + if credential_type == CredentialType.UNAUTHORIZED: + # manually create + TriggerProviderService.add_trigger_subscription( + subscription_id=subscription_builder.id, + tenant_id=tenant_id, + user_id=user_id, + name=subscription_builder.name, + provider_id=provider_id, + endpoint_id=subscription_builder.endpoint_id, + parameters=subscription_builder.parameters, + properties=subscription_builder.properties, + credential_expires_at=subscription_builder.credential_expires_at or -1, + expires_at=subscription_builder.expires_at, + credentials=subscription_builder.credentials, + credential_type=credential_type, + ) + else: + # automatically create + subscription: Subscription = TriggerManager.subscribe_trigger( + tenant_id=tenant_id, + user_id=user_id, + provider_id=provider_id, + endpoint=generate_plugin_trigger_endpoint_url(subscription_builder.endpoint_id), + parameters=subscription_builder.parameters, + credentials=subscription_builder.credentials, + credential_type=credential_type, + ) + + TriggerProviderService.add_trigger_subscription( + subscription_id=subscription_builder.id, + tenant_id=tenant_id, + user_id=user_id, + name=subscription_builder.name, + provider_id=provider_id, + endpoint_id=subscription_builder.endpoint_id, + parameters=subscription_builder.parameters, + properties=subscription.properties, + credentials=subscription_builder.credentials, + credential_type=credential_type, + credential_expires_at=subscription_builder.credential_expires_at or -1, + expires_at=subscription_builder.expires_at, + ) + + # Delete the builder after successful subscription creation + cache_key = cls.encode_cache_key(subscription_builder_id) + redis_client.delete(cache_key) + + @classmethod + def builder_to_api_entity( + cls, controller: PluginTriggerProviderController, entity: SubscriptionBuilder + ) -> SubscriptionBuilderApiEntity: + credential_type = CredentialType.of(entity.credential_type or CredentialType.UNAUTHORIZED.value) + return SubscriptionBuilderApiEntity( + id=entity.id, + name=entity.name or "", + provider=entity.provider_id, + endpoint=generate_plugin_trigger_endpoint_url(entity.endpoint_id), + parameters=entity.parameters, + properties=entity.properties, + credential_type=credential_type, + credentials=masked_credentials( + schemas=controller.get_credentials_schema(credential_type), + credentials=entity.credentials, + ) + if controller.get_subscription_constructor() + else {}, + ) + + @classmethod + def get_subscription_builder(cls, endpoint_id: str) -> SubscriptionBuilder | None: + """ + Get a trigger subscription by the endpoint ID. + """ + cache_key = cls.encode_cache_key(endpoint_id) + subscription_cache = redis_client.get(cache_key) + if subscription_cache: + return SubscriptionBuilder.model_validate(json.loads(subscription_cache)) + + return None + + @classmethod + def append_log(cls, endpoint_id: str, request: Request, response: Response) -> None: + """Append validation request log to Redis.""" + log = RequestLog( + id=str(uuid.uuid4()), + endpoint=endpoint_id, + request={ + "method": request.method, + "url": request.url, + "headers": dict(request.headers), + "data": request.get_data(as_text=True), + }, + response={ + "status_code": response.status_code, + "headers": dict(response.headers), + "data": response.get_data(as_text=True), + }, + created_at=datetime.now(), + ) + + key = f"trigger:subscription:builder:logs:{endpoint_id}" + logs = json.loads(redis_client.get(key) or "[]") + logs.append(log.model_dump(mode="json")) + + # Keep last N logs + logs = logs[-cls.__VALIDATION_REQUEST_CACHE_COUNT__ :] + redis_client.setex(key, cls.__VALIDATION_REQUEST_CACHE_EXPIRE_SECONDS__, json.dumps(logs, default=str)) + + @classmethod + def list_logs(cls, endpoint_id: str) -> list[RequestLog]: + """List request logs for validation endpoint.""" + key = f"trigger:subscription:builder:logs:{endpoint_id}" + logs_json = redis_client.get(key) + if not logs_json: + return [] + return [RequestLog.model_validate(log) for log in json.loads(logs_json)] + + @classmethod + def process_builder_validation_endpoint(cls, endpoint_id: str, request: Request) -> Response | None: + """ + Process a temporary endpoint request. + + :param endpoint_id: The endpoint identifier + :param request: The Flask request object + :return: The Flask response object + """ + # check if validation endpoint exists + subscription_builder: SubscriptionBuilder | None = cls.get_subscription_builder(endpoint_id) + if not subscription_builder: + return None + + # response to validation endpoint + controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=subscription_builder.tenant_id, provider_id=TriggerProviderID(subscription_builder.provider_id) + ) + try: + dispatch_response: TriggerDispatchResponse = controller.dispatch( + request=request, + subscription=subscription_builder.to_subscription(), + credentials={}, + credential_type=CredentialType.UNAUTHORIZED, + ) + response: Response = dispatch_response.response + # append the request log + cls.append_log( + endpoint_id=endpoint_id, + request=request, + response=response, + ) + return response + except Exception: + logger.exception("Error during validation endpoint dispatch for endpoint_id=%s", endpoint_id) + error_response = Response(status=500, response="An internal error has occurred.") + cls.append_log(endpoint_id=endpoint_id, request=request, response=error_response) + return error_response + + @classmethod + def get_subscription_builder_by_id(cls, subscription_builder_id: str) -> SubscriptionBuilderApiEntity: + """Get a trigger subscription builder API entity.""" + subscription_builder = cls.get_subscription_builder(subscription_builder_id) + if not subscription_builder: + raise ValueError(f"Subscription builder {subscription_builder_id} not found") + return cls.builder_to_api_entity( + controller=TriggerManager.get_trigger_provider( + subscription_builder.tenant_id, TriggerProviderID(subscription_builder.provider_id) + ), + entity=subscription_builder, + ) diff --git a/api/services/trigger/trigger_subscription_operator_service.py b/api/services/trigger/trigger_subscription_operator_service.py new file mode 100644 index 0000000000..5d7785549e --- /dev/null +++ b/api/services/trigger/trigger_subscription_operator_service.py @@ -0,0 +1,70 @@ +from sqlalchemy import and_, select +from sqlalchemy.orm import Session + +from extensions.ext_database import db +from models.enums import AppTriggerStatus +from models.trigger import AppTrigger, WorkflowPluginTrigger + + +class TriggerSubscriptionOperatorService: + @classmethod + def get_subscriber_triggers( + cls, tenant_id: str, subscription_id: str, event_name: str + ) -> list[WorkflowPluginTrigger]: + """ + Get WorkflowPluginTriggers for a subscription and trigger. + + Args: + tenant_id: Tenant ID + subscription_id: Subscription ID + event_name: Event name + """ + with Session(db.engine, expire_on_commit=False) as session: + subscribers = session.scalars( + select(WorkflowPluginTrigger) + .join( + AppTrigger, + and_( + AppTrigger.tenant_id == WorkflowPluginTrigger.tenant_id, + AppTrigger.app_id == WorkflowPluginTrigger.app_id, + AppTrigger.node_id == WorkflowPluginTrigger.node_id, + ), + ) + .where( + WorkflowPluginTrigger.tenant_id == tenant_id, + WorkflowPluginTrigger.subscription_id == subscription_id, + WorkflowPluginTrigger.event_name == event_name, + AppTrigger.status == AppTriggerStatus.ENABLED, + ) + ).all() + return list(subscribers) + + @classmethod + def delete_plugin_trigger_by_subscription( + cls, + session: Session, + tenant_id: str, + subscription_id: str, + ) -> None: + """Delete a plugin trigger by tenant_id and subscription_id within an existing session + + Args: + session: Database session + tenant_id: The tenant ID + subscription_id: The subscription ID + + Raises: + NotFound: If plugin trigger not found + """ + # Find plugin trigger using indexed columns + plugin_trigger = session.scalar( + select(WorkflowPluginTrigger).where( + WorkflowPluginTrigger.tenant_id == tenant_id, + WorkflowPluginTrigger.subscription_id == subscription_id, + ) + ) + + if not plugin_trigger: + return + + session.delete(plugin_trigger) diff --git a/api/services/trigger/webhook_service.py b/api/services/trigger/webhook_service.py new file mode 100644 index 0000000000..946764c35c --- /dev/null +++ b/api/services/trigger/webhook_service.py @@ -0,0 +1,871 @@ +import json +import logging +import mimetypes +import secrets +from collections.abc import Mapping +from typing import Any + +from flask import request +from pydantic import BaseModel +from sqlalchemy import select +from sqlalchemy.orm import Session +from werkzeug.datastructures import FileStorage +from werkzeug.exceptions import RequestEntityTooLarge + +from configs import dify_config +from core.app.entities.app_invoke_entities import InvokeFrom +from core.file.models import FileTransferMethod +from core.tools.tool_file_manager import ToolFileManager +from core.variables.types import SegmentType +from core.workflow.enums import NodeType +from extensions.ext_database import db +from extensions.ext_redis import redis_client +from factories import file_factory +from models.enums import AppTriggerStatus, AppTriggerType +from models.model import App +from models.trigger import AppTrigger, WorkflowWebhookTrigger +from models.workflow import Workflow +from services.async_workflow_service import AsyncWorkflowService +from services.end_user_service import EndUserService +from services.workflow.entities import WebhookTriggerData + +logger = logging.getLogger(__name__) + + +class WebhookService: + """Service for handling webhook operations.""" + + __WEBHOOK_NODE_CACHE_KEY__ = "webhook_nodes" + MAX_WEBHOOK_NODES_PER_WORKFLOW = 5 # Maximum allowed webhook nodes per workflow + + @staticmethod + def _sanitize_key(key: str) -> str: + """Normalize external keys (headers/params) to workflow-safe variables.""" + if not isinstance(key, str): + return key + return key.replace("-", "_") + + @classmethod + def get_webhook_trigger_and_workflow( + cls, webhook_id: str, is_debug: bool = False + ) -> tuple[WorkflowWebhookTrigger, Workflow, Mapping[str, Any]]: + """Get webhook trigger, workflow, and node configuration. + + Args: + webhook_id: The webhook ID to look up + is_debug: If True, use the draft workflow graph and skip the trigger enabled status check + + Returns: + A tuple containing: + - WorkflowWebhookTrigger: The webhook trigger object + - Workflow: The associated workflow object + - Mapping[str, Any]: The node configuration data + + Raises: + ValueError: If webhook not found, app trigger not found, trigger disabled, or workflow not found + """ + with Session(db.engine) as session: + # Get webhook trigger + webhook_trigger = ( + session.query(WorkflowWebhookTrigger).where(WorkflowWebhookTrigger.webhook_id == webhook_id).first() + ) + if not webhook_trigger: + raise ValueError(f"Webhook not found: {webhook_id}") + + if is_debug: + workflow = ( + session.query(Workflow) + .filter( + Workflow.app_id == webhook_trigger.app_id, + Workflow.version == Workflow.VERSION_DRAFT, + ) + .order_by(Workflow.created_at.desc()) + .first() + ) + else: + # Check if the corresponding AppTrigger exists + app_trigger = ( + session.query(AppTrigger) + .filter( + AppTrigger.app_id == webhook_trigger.app_id, + AppTrigger.node_id == webhook_trigger.node_id, + AppTrigger.trigger_type == AppTriggerType.TRIGGER_WEBHOOK, + ) + .first() + ) + + if not app_trigger: + raise ValueError(f"App trigger not found for webhook {webhook_id}") + + # Only check enabled status if not in debug mode + if app_trigger.status != AppTriggerStatus.ENABLED: + raise ValueError(f"Webhook trigger is disabled for webhook {webhook_id}") + + # Get workflow + workflow = ( + session.query(Workflow) + .filter( + Workflow.app_id == webhook_trigger.app_id, + Workflow.version != Workflow.VERSION_DRAFT, + ) + .order_by(Workflow.created_at.desc()) + .first() + ) + if not workflow: + raise ValueError(f"Workflow not found for app {webhook_trigger.app_id}") + + node_config = workflow.get_node_config_by_id(webhook_trigger.node_id) + + return webhook_trigger, workflow, node_config + + @classmethod + def extract_and_validate_webhook_data( + cls, webhook_trigger: WorkflowWebhookTrigger, node_config: Mapping[str, Any] + ) -> dict[str, Any]: + """Extract and validate webhook data in a single unified process. + + Args: + webhook_trigger: The webhook trigger object containing metadata + node_config: The node configuration containing validation rules + + Returns: + dict[str, Any]: Processed and validated webhook data with correct types + + Raises: + ValueError: If validation fails (HTTP method mismatch, missing required fields, type errors) + """ + # Extract raw data first + raw_data = cls.extract_webhook_data(webhook_trigger) + + # Validate HTTP metadata (method, content-type) + node_data = node_config.get("data", {}) + validation_result = cls._validate_http_metadata(raw_data, node_data) + if not validation_result["valid"]: + raise ValueError(validation_result["error"]) + + # Process and validate data according to configuration + processed_data = cls._process_and_validate_data(raw_data, node_data) + + return processed_data + + @classmethod + def extract_webhook_data(cls, webhook_trigger: WorkflowWebhookTrigger) -> dict[str, Any]: + """Extract raw data from incoming webhook request without type conversion. + + Args: + webhook_trigger: The webhook trigger object for file processing context + + Returns: + dict[str, Any]: Raw webhook data containing: + - method: HTTP method + - headers: Request headers + - query_params: Query parameters as strings + - body: Request body (varies by content type) + - files: Uploaded files (if any) + """ + cls._validate_content_length() + + data = { + "method": request.method, + "headers": dict(request.headers), + "query_params": dict(request.args), + "body": {}, + "files": {}, + } + + # Extract and normalize content type + content_type = cls._extract_content_type(dict(request.headers)) + + # Route to appropriate extractor based on content type + extractors = { + "application/json": cls._extract_json_body, + "application/x-www-form-urlencoded": cls._extract_form_body, + "multipart/form-data": lambda: cls._extract_multipart_body(webhook_trigger), + "application/octet-stream": lambda: cls._extract_octet_stream_body(webhook_trigger), + "text/plain": cls._extract_text_body, + } + + extractor = extractors.get(content_type) + if not extractor: + # Default to text/plain for unknown content types + logger.warning("Unknown Content-Type: %s, treating as text/plain", content_type) + extractor = cls._extract_text_body + + # Extract body and files + body_data, files_data = extractor() + data["body"] = body_data + data["files"] = files_data + + return data + + @classmethod + def _process_and_validate_data(cls, raw_data: dict[str, Any], node_data: dict[str, Any]) -> dict[str, Any]: + """Process and validate webhook data according to node configuration. + + Args: + raw_data: Raw webhook data from extraction + node_data: Node configuration containing validation and type rules + + Returns: + dict[str, Any]: Processed data with validated types + + Raises: + ValueError: If validation fails or required fields are missing + """ + result = raw_data.copy() + + # Validate and process headers + cls._validate_required_headers(raw_data["headers"], node_data.get("headers", [])) + + # Process query parameters with type conversion and validation + result["query_params"] = cls._process_parameters( + raw_data["query_params"], node_data.get("params", []), is_form_data=True + ) + + # Process body parameters based on content type + configured_content_type = node_data.get("content_type", "application/json").lower() + result["body"] = cls._process_body_parameters( + raw_data["body"], node_data.get("body", []), configured_content_type + ) + + return result + + @classmethod + def _validate_content_length(cls) -> None: + """Validate request content length against maximum allowed size.""" + content_length = request.content_length + if content_length and content_length > dify_config.WEBHOOK_REQUEST_BODY_MAX_SIZE: + raise RequestEntityTooLarge( + f"Webhook request too large: {content_length} bytes exceeds maximum allowed size " + f"of {dify_config.WEBHOOK_REQUEST_BODY_MAX_SIZE} bytes" + ) + + @classmethod + def _extract_json_body(cls) -> tuple[dict[str, Any], dict[str, Any]]: + """Extract JSON body from request. + + Returns: + tuple: (body_data, files_data) where: + - body_data: Parsed JSON content or empty dict if parsing fails + - files_data: Empty dict (JSON requests don't contain files) + """ + try: + body = request.get_json() or {} + except Exception: + logger.warning("Failed to parse JSON body") + body = {} + return body, {} + + @classmethod + def _extract_form_body(cls) -> tuple[dict[str, Any], dict[str, Any]]: + """Extract form-urlencoded body from request. + + Returns: + tuple: (body_data, files_data) where: + - body_data: Form data as key-value pairs + - files_data: Empty dict (form-urlencoded requests don't contain files) + """ + return dict(request.form), {} + + @classmethod + def _extract_multipart_body(cls, webhook_trigger: WorkflowWebhookTrigger) -> tuple[dict[str, Any], dict[str, Any]]: + """Extract multipart/form-data body and files from request. + + Args: + webhook_trigger: Webhook trigger for file processing context + + Returns: + tuple: (body_data, files_data) where: + - body_data: Form data as key-value pairs + - files_data: Processed file objects indexed by field name + """ + body = dict(request.form) + files = cls._process_file_uploads(request.files, webhook_trigger) if request.files else {} + return body, files + + @classmethod + def _extract_octet_stream_body( + cls, webhook_trigger: WorkflowWebhookTrigger + ) -> tuple[dict[str, Any], dict[str, Any]]: + """Extract binary data as file from request. + + Args: + webhook_trigger: Webhook trigger for file processing context + + Returns: + tuple: (body_data, files_data) where: + - body_data: Dict with 'raw' key containing file object or None + - files_data: Empty dict + """ + try: + file_content = request.get_data() + if file_content: + file_obj = cls._create_file_from_binary(file_content, "application/octet-stream", webhook_trigger) + return {"raw": file_obj.to_dict()}, {} + else: + return {"raw": None}, {} + except Exception: + logger.exception("Failed to process octet-stream data") + return {"raw": None}, {} + + @classmethod + def _extract_text_body(cls) -> tuple[dict[str, Any], dict[str, Any]]: + """Extract text/plain body from request. + + Returns: + tuple: (body_data, files_data) where: + - body_data: Dict with 'raw' key containing text content + - files_data: Empty dict (text requests don't contain files) + """ + try: + body = {"raw": request.get_data(as_text=True)} + except Exception: + logger.warning("Failed to extract text body") + body = {"raw": ""} + return body, {} + + @classmethod + def _process_file_uploads( + cls, files: Mapping[str, FileStorage], webhook_trigger: WorkflowWebhookTrigger + ) -> dict[str, Any]: + """Process file uploads using ToolFileManager. + + Args: + files: Flask request files object containing uploaded files + webhook_trigger: Webhook trigger for tenant and user context + + Returns: + dict[str, Any]: Processed file objects indexed by field name + """ + processed_files = {} + + for name, file in files.items(): + if file and file.filename: + try: + file_content = file.read() + mimetype = file.content_type or mimetypes.guess_type(file.filename)[0] or "application/octet-stream" + file_obj = cls._create_file_from_binary(file_content, mimetype, webhook_trigger) + processed_files[name] = file_obj.to_dict() + except Exception: + logger.exception("Failed to process file upload '%s'", name) + # Continue processing other files + + return processed_files + + @classmethod + def _create_file_from_binary( + cls, file_content: bytes, mimetype: str, webhook_trigger: WorkflowWebhookTrigger + ) -> Any: + """Create a file object from binary content using ToolFileManager. + + Args: + file_content: The binary content of the file + mimetype: The MIME type of the file + webhook_trigger: Webhook trigger for tenant and user context + + Returns: + Any: A file object built from the binary content + """ + tool_file_manager = ToolFileManager() + + # Create file using ToolFileManager + tool_file = tool_file_manager.create_file_by_raw( + user_id=webhook_trigger.created_by, + tenant_id=webhook_trigger.tenant_id, + conversation_id=None, + file_binary=file_content, + mimetype=mimetype, + ) + + # Build File object + mapping = { + "tool_file_id": tool_file.id, + "transfer_method": FileTransferMethod.TOOL_FILE.value, + } + return file_factory.build_from_mapping( + mapping=mapping, + tenant_id=webhook_trigger.tenant_id, + ) + + @classmethod + def _process_parameters( + cls, raw_params: dict[str, str], param_configs: list, is_form_data: bool = False + ) -> dict[str, Any]: + """Process parameters with unified validation and type conversion. + + Args: + raw_params: Raw parameter values as strings + param_configs: List of parameter configuration dictionaries + is_form_data: Whether the parameters are from form data (requiring string conversion) + + Returns: + dict[str, Any]: Processed parameters with validated types + + Raises: + ValueError: If required parameters are missing or validation fails + """ + processed = {} + configured_params = {config.get("name", ""): config for config in param_configs} + + # Process configured parameters + for param_config in param_configs: + name = param_config.get("name", "") + param_type = param_config.get("type", SegmentType.STRING) + required = param_config.get("required", False) + + # Check required parameters + if required and name not in raw_params: + raise ValueError(f"Required parameter missing: {name}") + + if name in raw_params: + raw_value = raw_params[name] + processed[name] = cls._validate_and_convert_value(name, raw_value, param_type, is_form_data) + + # Include unconfigured parameters as strings + for name, value in raw_params.items(): + if name not in configured_params: + processed[name] = value + + return processed + + @classmethod + def _process_body_parameters( + cls, raw_body: dict[str, Any], body_configs: list, content_type: str + ) -> dict[str, Any]: + """Process body parameters based on content type and configuration. + + Args: + raw_body: Raw body data from request + body_configs: List of body parameter configuration dictionaries + content_type: The request content type + + Returns: + dict[str, Any]: Processed body parameters with validated types + + Raises: + ValueError: If required body parameters are missing or validation fails + """ + if content_type in ["text/plain", "application/octet-stream"]: + # For text/plain and octet-stream, validate required content exists + if body_configs and any(config.get("required", False) for config in body_configs): + raw_content = raw_body.get("raw") + if not raw_content: + raise ValueError(f"Required body content missing for {content_type} request") + return raw_body + + # For structured data (JSON, form-data, etc.) + processed = {} + configured_params = {config.get("name", ""): config for config in body_configs} + + for body_config in body_configs: + name = body_config.get("name", "") + param_type = body_config.get("type", SegmentType.STRING) + required = body_config.get("required", False) + + # Handle file parameters for multipart data + if param_type == SegmentType.FILE and content_type == "multipart/form-data": + # File validation is handled separately in extract phase + continue + + # Check required parameters + if required and name not in raw_body: + raise ValueError(f"Required body parameter missing: {name}") + + if name in raw_body: + raw_value = raw_body[name] + is_form_data = content_type in ["application/x-www-form-urlencoded", "multipart/form-data"] + processed[name] = cls._validate_and_convert_value(name, raw_value, param_type, is_form_data) + + # Include unconfigured parameters + for name, value in raw_body.items(): + if name not in configured_params: + processed[name] = value + + return processed + + @classmethod + def _validate_and_convert_value(cls, param_name: str, value: Any, param_type: str, is_form_data: bool) -> Any: + """Unified validation and type conversion for parameter values. + + Args: + param_name: Name of the parameter for error reporting + value: The value to validate and convert + param_type: The expected parameter type (SegmentType) + is_form_data: Whether the value is from form data (requiring string conversion) + + Returns: + Any: The validated and converted value + + Raises: + ValueError: If validation or conversion fails + """ + try: + if is_form_data: + # Form data comes as strings and needs conversion + return cls._convert_form_value(param_name, value, param_type) + else: + # JSON data should already be in correct types, just validate + return cls._validate_json_value(param_name, value, param_type) + except Exception as e: + raise ValueError(f"Parameter '{param_name}' validation failed: {str(e)}") + + @classmethod + def _convert_form_value(cls, param_name: str, value: str, param_type: str) -> Any: + """Convert form data string values to specified types. + + Args: + param_name: Name of the parameter for error reporting + value: The string value to convert + param_type: The target type to convert to (SegmentType) + + Returns: + Any: The converted value in the appropriate type + + Raises: + ValueError: If the value cannot be converted to the specified type + """ + if param_type == SegmentType.STRING: + return value + elif param_type == SegmentType.NUMBER: + if not cls._can_convert_to_number(value): + raise ValueError(f"Cannot convert '{value}' to number") + numeric_value = float(value) + return int(numeric_value) if numeric_value.is_integer() else numeric_value + elif param_type == SegmentType.BOOLEAN: + lower_value = value.lower() + bool_map = {"true": True, "false": False, "1": True, "0": False, "yes": True, "no": False} + if lower_value not in bool_map: + raise ValueError(f"Cannot convert '{value}' to boolean") + return bool_map[lower_value] + else: + raise ValueError(f"Unsupported type '{param_type}' for form data parameter '{param_name}'") + + @classmethod + def _validate_json_value(cls, param_name: str, value: Any, param_type: str) -> Any: + """Validate JSON values against expected types. + + Args: + param_name: Name of the parameter for error reporting + value: The value to validate + param_type: The expected parameter type (SegmentType) + + Returns: + Any: The validated value (unchanged if valid) + + Raises: + ValueError: If the value type doesn't match the expected type + """ + type_validators = { + SegmentType.STRING: (lambda v: isinstance(v, str), "string"), + SegmentType.NUMBER: (lambda v: isinstance(v, (int, float)), "number"), + SegmentType.BOOLEAN: (lambda v: isinstance(v, bool), "boolean"), + SegmentType.OBJECT: (lambda v: isinstance(v, dict), "object"), + SegmentType.ARRAY_STRING: ( + lambda v: isinstance(v, list) and all(isinstance(item, str) for item in v), + "array of strings", + ), + SegmentType.ARRAY_NUMBER: ( + lambda v: isinstance(v, list) and all(isinstance(item, (int, float)) for item in v), + "array of numbers", + ), + SegmentType.ARRAY_BOOLEAN: ( + lambda v: isinstance(v, list) and all(isinstance(item, bool) for item in v), + "array of booleans", + ), + SegmentType.ARRAY_OBJECT: ( + lambda v: isinstance(v, list) and all(isinstance(item, dict) for item in v), + "array of objects", + ), + } + + validator_info = type_validators.get(SegmentType(param_type)) + if not validator_info: + logger.warning("Unknown parameter type: %s for parameter %s", param_type, param_name) + return value + + validator, expected_type = validator_info + if not validator(value): + actual_type = type(value).__name__ + raise ValueError(f"Expected {expected_type}, got {actual_type}") + + return value + + @classmethod + def _validate_required_headers(cls, headers: dict[str, Any], header_configs: list) -> None: + """Validate required headers are present. + + Args: + headers: Request headers dictionary + header_configs: List of header configuration dictionaries + + Raises: + ValueError: If required headers are missing + """ + headers_lower = {k.lower(): v for k, v in headers.items()} + headers_sanitized = {cls._sanitize_key(k).lower(): v for k, v in headers.items()} + for header_config in header_configs: + if header_config.get("required", False): + header_name = header_config.get("name", "") + sanitized_name = cls._sanitize_key(header_name).lower() + if header_name.lower() not in headers_lower and sanitized_name not in headers_sanitized: + raise ValueError(f"Required header missing: {header_name}") + + @classmethod + def _validate_http_metadata(cls, webhook_data: dict[str, Any], node_data: dict[str, Any]) -> dict[str, Any]: + """Validate HTTP method and content-type. + + Args: + webhook_data: Extracted webhook data containing method and headers + node_data: Node configuration containing expected method and content-type + + Returns: + dict[str, Any]: Validation result with 'valid' key and optional 'error' key + """ + # Validate HTTP method + configured_method = node_data.get("method", "get").upper() + request_method = webhook_data["method"].upper() + if configured_method != request_method: + return cls._validation_error(f"HTTP method mismatch. Expected {configured_method}, got {request_method}") + + # Validate Content-type + configured_content_type = node_data.get("content_type", "application/json").lower() + request_content_type = cls._extract_content_type(webhook_data["headers"]) + + if configured_content_type != request_content_type: + return cls._validation_error( + f"Content-type mismatch. Expected {configured_content_type}, got {request_content_type}" + ) + + return {"valid": True} + + @classmethod + def _extract_content_type(cls, headers: dict[str, Any]) -> str: + """Extract and normalize content-type from headers. + + Args: + headers: Request headers dictionary + + Returns: + str: Normalized content-type (main type without parameters) + """ + content_type = headers.get("Content-Type", "").lower() + if not content_type: + content_type = headers.get("content-type", "application/json").lower() + # Extract the main content type (ignore parameters like boundary) + return content_type.split(";")[0].strip() + + @classmethod + def _validation_error(cls, error_message: str) -> dict[str, Any]: + """Create a standard validation error response. + + Args: + error_message: The error message to include + + Returns: + dict[str, Any]: Validation error response with 'valid' and 'error' keys + """ + return {"valid": False, "error": error_message} + + @classmethod + def _can_convert_to_number(cls, value: str) -> bool: + """Check if a string can be converted to a number.""" + try: + float(value) + return True + except ValueError: + return False + + @classmethod + def build_workflow_inputs(cls, webhook_data: dict[str, Any]) -> dict[str, Any]: + """Construct workflow inputs payload from webhook data. + + Args: + webhook_data: Processed webhook data containing headers, query params, and body + + Returns: + dict[str, Any]: Workflow inputs formatted for execution + """ + return { + "webhook_data": webhook_data, + "webhook_headers": webhook_data.get("headers", {}), + "webhook_query_params": webhook_data.get("query_params", {}), + "webhook_body": webhook_data.get("body", {}), + } + + @classmethod + def trigger_workflow_execution( + cls, webhook_trigger: WorkflowWebhookTrigger, webhook_data: dict[str, Any], workflow: Workflow + ) -> None: + """Trigger workflow execution via AsyncWorkflowService. + + Args: + webhook_trigger: The webhook trigger object + webhook_data: Processed webhook data for workflow inputs + workflow: The workflow to execute + + Raises: + ValueError: If tenant owner is not found + Exception: If workflow execution fails + """ + try: + with Session(db.engine) as session: + # Prepare inputs for the webhook node + # The webhook node expects webhook_data in the inputs + workflow_inputs = cls.build_workflow_inputs(webhook_data) + + # Create trigger data + trigger_data = WebhookTriggerData( + app_id=webhook_trigger.app_id, + workflow_id=workflow.id, + root_node_id=webhook_trigger.node_id, # Start from the webhook node + inputs=workflow_inputs, + tenant_id=webhook_trigger.tenant_id, + ) + + end_user = EndUserService.get_or_create_end_user_by_type( + type=InvokeFrom.TRIGGER, + tenant_id=webhook_trigger.tenant_id, + app_id=webhook_trigger.app_id, + user_id=None, + ) + + # Trigger workflow execution asynchronously + AsyncWorkflowService.trigger_workflow_async( + session, + end_user, + trigger_data, + ) + + except Exception: + logger.exception("Failed to trigger workflow for webhook %s", webhook_trigger.webhook_id) + raise + + @classmethod + def generate_webhook_response(cls, node_config: Mapping[str, Any]) -> tuple[dict[str, Any], int]: + """Generate HTTP response based on node configuration. + + Args: + node_config: Node configuration containing response settings + + Returns: + tuple[dict[str, Any], int]: Response data and HTTP status code + """ + node_data = node_config.get("data", {}) + + # Get configured status code and response body + status_code = node_data.get("status_code", 200) + response_body = node_data.get("response_body", "") + + # Parse response body as JSON if it's valid JSON, otherwise return as text + try: + if response_body: + try: + response_data = ( + json.loads(response_body) + if response_body.strip().startswith(("{", "[")) + else {"message": response_body} + ) + except json.JSONDecodeError: + response_data = {"message": response_body} + else: + response_data = {"status": "success", "message": "Webhook processed successfully"} + except: + response_data = {"message": response_body or "Webhook processed successfully"} + + return response_data, status_code + + @classmethod + def sync_webhook_relationships(cls, app: App, workflow: Workflow): + """ + Sync webhook relationships in DB. + + 1. Check if the workflow has any webhook trigger nodes + 2. Fetch the nodes from DB, see if there were any webhook records already + 3. Diff the nodes and the webhook records, create/update/delete the webhook records as needed + + Approach: + Frequent DB operations may cause performance issues, using Redis to cache it instead. + If any record exists, cache it. + + Limits: + - Maximum 5 webhook nodes per workflow + """ + + class Cache(BaseModel): + """ + Cache model for webhook nodes + """ + + record_id: str + node_id: str + webhook_id: str + + nodes_id_in_graph = [node_id for node_id, _ in workflow.walk_nodes(NodeType.TRIGGER_WEBHOOK)] + + # Check webhook node limit + if len(nodes_id_in_graph) > cls.MAX_WEBHOOK_NODES_PER_WORKFLOW: + raise ValueError( + f"Workflow exceeds maximum webhook node limit. " + f"Found {len(nodes_id_in_graph)} webhook nodes, maximum allowed is {cls.MAX_WEBHOOK_NODES_PER_WORKFLOW}" + ) + + not_found_in_cache: list[str] = [] + for node_id in nodes_id_in_graph: + # firstly check if the node exists in cache + if not redis_client.get(f"{cls.__WEBHOOK_NODE_CACHE_KEY__}:{node_id}"): + not_found_in_cache.append(node_id) + continue + + with Session(db.engine) as session: + try: + # lock the concurrent webhook trigger creation + redis_client.lock(f"{cls.__WEBHOOK_NODE_CACHE_KEY__}:apps:{app.id}:lock", timeout=10) + # fetch the non-cached nodes from DB + all_records = session.scalars( + select(WorkflowWebhookTrigger).where( + WorkflowWebhookTrigger.app_id == app.id, + WorkflowWebhookTrigger.tenant_id == app.tenant_id, + ) + ).all() + + nodes_id_in_db = {node.node_id: node for node in all_records} + + # get the nodes not found both in cache and DB + nodes_not_found = [node_id for node_id in not_found_in_cache if node_id not in nodes_id_in_db] + + # create new webhook records + for node_id in nodes_not_found: + webhook_record = WorkflowWebhookTrigger( + app_id=app.id, + tenant_id=app.tenant_id, + node_id=node_id, + webhook_id=cls.generate_webhook_id(), + created_by=app.created_by, + ) + session.add(webhook_record) + session.flush() + cache = Cache(record_id=webhook_record.id, node_id=node_id, webhook_id=webhook_record.webhook_id) + redis_client.set(f"{cls.__WEBHOOK_NODE_CACHE_KEY__}:{node_id}", cache.model_dump_json(), ex=60 * 60) + session.commit() + + # delete the nodes not found in the graph + for node_id in nodes_id_in_db: + if node_id not in nodes_id_in_graph: + session.delete(nodes_id_in_db[node_id]) + redis_client.delete(f"{cls.__WEBHOOK_NODE_CACHE_KEY__}:{node_id}") + session.commit() + except Exception: + logger.exception("Failed to sync webhook relationships for app %s", app.id) + raise + finally: + redis_client.delete(f"{cls.__WEBHOOK_NODE_CACHE_KEY__}:apps:{app.id}:lock") + + @classmethod + def generate_webhook_id(cls) -> str: + """ + Generate unique 24-character webhook ID + + Deduplication is not needed, DB already has unique constraint on webhook_id. + """ + # Generate 24-character random string + return secrets.token_urlsafe(18)[:24] # token_urlsafe gives base64url, take first 24 chars diff --git a/api/services/variable_truncator.py b/api/services/variable_truncator.py index 4e13d2d964..6eb8d0031d 100644 --- a/api/services/variable_truncator.py +++ b/api/services/variable_truncator.py @@ -1,4 +1,5 @@ import dataclasses +from abc import ABC, abstractmethod from collections.abc import Mapping from typing import Any, Generic, TypeAlias, TypeVar, overload @@ -66,7 +67,17 @@ class TruncationResult: truncated: bool -class VariableTruncator: +class BaseTruncator(ABC): + @abstractmethod + def truncate(self, segment: Segment) -> TruncationResult: + pass + + @abstractmethod + def truncate_variable_mapping(self, v: Mapping[str, Any]) -> tuple[Mapping[str, Any], bool]: + pass + + +class VariableTruncator(BaseTruncator): """ Handles variable truncation with structure-preserving strategies. @@ -283,7 +294,7 @@ class VariableTruncator: break remaining_budget = target_size - used_size - if item is None or isinstance(item, (str, list, dict, bool, int, float)): + if item is None or isinstance(item, (str, list, dict, bool, int, float, UpdatedVariable)): part_result = self._truncate_json_primitives(item, remaining_budget) else: raise UnknownTypeError(f"got unknown type {type(item)} in array truncation") @@ -373,6 +384,11 @@ class VariableTruncator: return _PartResult(truncated_obj, used_size, truncated) + @overload + def _truncate_json_primitives( + self, val: UpdatedVariable, target_size: int + ) -> _PartResult[Mapping[str, object]]: ... + @overload def _truncate_json_primitives(self, val: str, target_size: int) -> _PartResult[str]: ... @@ -413,3 +429,38 @@ class VariableTruncator: return _PartResult(val, self.calculate_json_size(val), False) else: raise AssertionError("this statement should be unreachable.") + + +class DummyVariableTruncator(BaseTruncator): + """ + A no-op variable truncator that doesn't truncate any data. + + This is used for Service API calls where truncation should be disabled + to maintain backward compatibility and provide complete data. + """ + + def truncate_variable_mapping(self, v: Mapping[str, Any]) -> tuple[Mapping[str, Any], bool]: + """ + Return original mapping without truncation. + + Args: + v: The variable mapping to process + + Returns: + Tuple of (original_mapping, False) where False indicates no truncation occurred + """ + return v, False + + def truncate(self, segment: Segment) -> TruncationResult: + """ + Return original segment without truncation. + + Args: + segment: The segment to process + + Returns: + The original segment unchanged + """ + # For Service API, we want to preserve the original segment + # without any truncation, so just return it as-is + return TruncationResult(result=segment, truncated=False) diff --git a/api/services/workflow/entities.py b/api/services/workflow/entities.py new file mode 100644 index 0000000000..70ec8d6e2a --- /dev/null +++ b/api/services/workflow/entities.py @@ -0,0 +1,165 @@ +""" +Pydantic models for async workflow trigger system. +""" + +from collections.abc import Mapping, Sequence +from enum import StrEnum +from typing import Any + +from pydantic import BaseModel, ConfigDict, Field + +from models.enums import AppTriggerType, WorkflowRunTriggeredFrom + + +class AsyncTriggerStatus(StrEnum): + """Async trigger execution status""" + + COMPLETED = "completed" + FAILED = "failed" + TIMEOUT = "timeout" + + +class TriggerMetadata(BaseModel): + """Trigger metadata""" + + type: AppTriggerType = Field(default=AppTriggerType.UNKNOWN) + + +class TriggerData(BaseModel): + """Base trigger data model for async workflow execution""" + + app_id: str + tenant_id: str + workflow_id: str | None = None + root_node_id: str + inputs: Mapping[str, Any] + files: Sequence[Mapping[str, Any]] = Field(default_factory=list) + trigger_type: AppTriggerType + trigger_from: WorkflowRunTriggeredFrom + trigger_metadata: TriggerMetadata | None = None + + model_config = ConfigDict(use_enum_values=True) + + +class WebhookTriggerData(TriggerData): + """Webhook-specific trigger data""" + + trigger_type: AppTriggerType = AppTriggerType.TRIGGER_WEBHOOK + trigger_from: WorkflowRunTriggeredFrom = WorkflowRunTriggeredFrom.WEBHOOK + + +class ScheduleTriggerData(TriggerData): + """Schedule-specific trigger data""" + + trigger_type: AppTriggerType = AppTriggerType.TRIGGER_SCHEDULE + trigger_from: WorkflowRunTriggeredFrom = WorkflowRunTriggeredFrom.SCHEDULE + + +class PluginTriggerMetadata(TriggerMetadata): + """Plugin trigger metadata""" + + type: AppTriggerType = AppTriggerType.TRIGGER_PLUGIN + + endpoint_id: str + plugin_unique_identifier: str + provider_id: str + event_name: str + icon_filename: str + icon_dark_filename: str + + +class PluginTriggerData(TriggerData): + """Plugin webhook trigger data""" + + trigger_type: AppTriggerType = AppTriggerType.TRIGGER_PLUGIN + trigger_from: WorkflowRunTriggeredFrom = WorkflowRunTriggeredFrom.PLUGIN + plugin_id: str + endpoint_id: str + + +class PluginTriggerDispatchData(BaseModel): + """Plugin trigger dispatch data for Celery tasks""" + + user_id: str + tenant_id: str + endpoint_id: str + provider_id: str + subscription_id: str + timestamp: int + events: list[str] + request_id: str + + +class WorkflowTaskData(BaseModel): + """Lightweight data structure for Celery workflow tasks""" + + workflow_trigger_log_id: str # Primary tracking ID - all other data can be fetched from DB + + model_config = ConfigDict(arbitrary_types_allowed=True) + + +class AsyncTriggerExecutionResult(BaseModel): + """Result from async trigger-based workflow execution""" + + execution_id: str + status: AsyncTriggerStatus + result: Mapping[str, Any] | None = None + error: str | None = None + elapsed_time: float | None = None + total_tokens: int | None = None + + model_config = ConfigDict(use_enum_values=True) + + +class AsyncTriggerResponse(BaseModel): + """Response from triggering an async workflow""" + + workflow_trigger_log_id: str + task_id: str + status: str + queue: str + + model_config = ConfigDict(use_enum_values=True) + + +class TriggerLogResponse(BaseModel): + """Response model for trigger log data""" + + id: str + tenant_id: str + app_id: str + workflow_id: str + trigger_type: WorkflowRunTriggeredFrom + status: str + queue_name: str + retry_count: int + celery_task_id: str | None = None + workflow_run_id: str | None = None + error: str | None = None + outputs: str | None = None + elapsed_time: float | None = None + total_tokens: int | None = None + created_at: str | None = None + triggered_at: str | None = None + finished_at: str | None = None + + model_config = ConfigDict(use_enum_values=True) + + +class WorkflowScheduleCFSPlanEntity(BaseModel): + """ + CFS plan entity. + Ensure each workflow run inside Dify is associated with a CFS(Completely Fair Scheduler) plan. + + """ + + class Strategy(StrEnum): + """ + CFS plan strategy. + """ + + TimeSlice = "time-slice" # time-slice based plan + Nop = "nop" # no plan, just run the workflow + + schedule_strategy: Strategy + granularity: int = Field(default=-1) # -1 means infinite diff --git a/api/services/workflow/queue_dispatcher.py b/api/services/workflow/queue_dispatcher.py new file mode 100644 index 0000000000..c55de7a085 --- /dev/null +++ b/api/services/workflow/queue_dispatcher.py @@ -0,0 +1,151 @@ +""" +Queue dispatcher system for async workflow execution. + +Implements an ABC-based pattern for handling different subscription tiers +with appropriate queue routing and rate limiting. +""" + +from abc import ABC, abstractmethod +from enum import StrEnum + +from configs import dify_config +from extensions.ext_redis import redis_client +from services.billing_service import BillingService +from services.workflow.rate_limiter import TenantDailyRateLimiter + + +class QueuePriority(StrEnum): + """Queue priorities for different subscription tiers""" + + PROFESSIONAL = "workflow_professional" # Highest priority + TEAM = "workflow_team" + SANDBOX = "workflow_sandbox" # Free tier + + +class BaseQueueDispatcher(ABC): + """Abstract base class for queue dispatchers""" + + def __init__(self): + self.rate_limiter = TenantDailyRateLimiter(redis_client) + + @abstractmethod + def get_queue_name(self) -> str: + """Get the queue name for this dispatcher""" + pass + + @abstractmethod + def get_daily_limit(self) -> int: + """Get daily execution limit""" + pass + + @abstractmethod + def get_priority(self) -> int: + """Get task priority level""" + pass + + def check_daily_quota(self, tenant_id: str) -> bool: + """ + Check if tenant has remaining daily quota + + Args: + tenant_id: The tenant identifier + + Returns: + True if quota available, False otherwise + """ + # Check without consuming + remaining = self.rate_limiter.get_remaining_quota(tenant_id=tenant_id, max_daily_limit=self.get_daily_limit()) + return remaining > 0 + + def consume_quota(self, tenant_id: str) -> bool: + """ + Consume one execution from daily quota + + Args: + tenant_id: The tenant identifier + + Returns: + True if quota consumed successfully, False if limit reached + """ + return self.rate_limiter.check_and_consume(tenant_id=tenant_id, max_daily_limit=self.get_daily_limit()) + + +class ProfessionalQueueDispatcher(BaseQueueDispatcher): + """Dispatcher for professional tier""" + + def get_queue_name(self) -> str: + return QueuePriority.PROFESSIONAL + + def get_daily_limit(self) -> int: + return int(1e9) + + def get_priority(self) -> int: + return 100 + + +class TeamQueueDispatcher(BaseQueueDispatcher): + """Dispatcher for team tier""" + + def get_queue_name(self) -> str: + return QueuePriority.TEAM + + def get_daily_limit(self) -> int: + return int(1e9) + + def get_priority(self) -> int: + return 50 + + +class SandboxQueueDispatcher(BaseQueueDispatcher): + """Dispatcher for free/sandbox tier""" + + def get_queue_name(self) -> str: + return QueuePriority.SANDBOX + + def get_daily_limit(self) -> int: + return dify_config.APP_DAILY_RATE_LIMIT + + def get_priority(self) -> int: + return 10 + + +class QueueDispatcherManager: + """Factory for creating appropriate dispatcher based on tenant subscription""" + + # Mapping of billing plans to dispatchers + PLAN_DISPATCHER_MAP = { + "professional": ProfessionalQueueDispatcher, + "team": TeamQueueDispatcher, + "sandbox": SandboxQueueDispatcher, + # Add new tiers here as they're created + # For any unknown plan, default to sandbox + } + + @classmethod + def get_dispatcher(cls, tenant_id: str) -> BaseQueueDispatcher: + """ + Get dispatcher based on tenant's subscription plan + + Args: + tenant_id: The tenant identifier + + Returns: + Appropriate queue dispatcher instance + """ + if dify_config.BILLING_ENABLED: + try: + billing_info = BillingService.get_info(tenant_id) + plan = billing_info.get("subscription", {}).get("plan", "sandbox") + except Exception: + # If billing service fails, default to sandbox + plan = "sandbox" + else: + # If billing is disabled, use team tier as default + plan = "team" + + dispatcher_class = cls.PLAN_DISPATCHER_MAP.get( + plan, + SandboxQueueDispatcher, # Default to sandbox for unknown plans + ) + + return dispatcher_class() # type: ignore diff --git a/api/services/workflow/rate_limiter.py b/api/services/workflow/rate_limiter.py new file mode 100644 index 0000000000..1ccb4e1961 --- /dev/null +++ b/api/services/workflow/rate_limiter.py @@ -0,0 +1,183 @@ +""" +Day-based rate limiter for workflow executions. + +Implements UTC-based daily quotas that reset at midnight UTC for consistent rate limiting. +""" + +from datetime import UTC, datetime, time, timedelta +from typing import Union + +import pytz +from redis import Redis +from sqlalchemy import select + +from extensions.ext_database import db +from extensions.ext_redis import RedisClientWrapper +from models.account import Account, TenantAccountJoin, TenantAccountRole + + +class TenantDailyRateLimiter: + """ + Day-based rate limiter that resets at midnight UTC + + This class provides Redis-based rate limiting with the following features: + - Daily quotas that reset at midnight UTC for consistency + - Atomic check-and-consume operations + - Automatic cleanup of stale counters + - Timezone-aware error messages for better UX + """ + + def __init__(self, redis_client: Union[Redis, RedisClientWrapper]): + self.redis = redis_client + + def get_tenant_owner_timezone(self, tenant_id: str) -> str: + """ + Get timezone of tenant owner + + Args: + tenant_id: The tenant identifier + + Returns: + Timezone string (e.g., 'America/New_York', 'UTC') + """ + # Query to get tenant owner's timezone using scalar and select + owner = db.session.scalar( + select(Account) + .join(TenantAccountJoin, TenantAccountJoin.account_id == Account.id) + .where(TenantAccountJoin.tenant_id == tenant_id, TenantAccountJoin.role == TenantAccountRole.OWNER) + ) + + if not owner: + return "UTC" + + return owner.timezone or "UTC" + + def _get_day_key(self, tenant_id: str) -> str: + """ + Get Redis key for current UTC day + + Args: + tenant_id: The tenant identifier + + Returns: + Redis key for the current UTC day + """ + utc_now = datetime.now(UTC) + date_str = utc_now.strftime("%Y-%m-%d") + return f"workflow:daily_limit:{tenant_id}:{date_str}" + + def _get_ttl_seconds(self) -> int: + """ + Calculate seconds until UTC midnight + + Returns: + Number of seconds until UTC midnight + """ + utc_now = datetime.now(UTC) + + # Get next midnight in UTC + next_midnight = datetime.combine(utc_now.date() + timedelta(days=1), time.min) + next_midnight = next_midnight.replace(tzinfo=UTC) + + return int((next_midnight - utc_now).total_seconds()) + + def check_and_consume(self, tenant_id: str, max_daily_limit: int) -> bool: + """ + Check if quota available and consume one execution + + Args: + tenant_id: The tenant identifier + max_daily_limit: Maximum daily limit + + Returns: + True if quota consumed successfully, False if limit reached + """ + key = self._get_day_key(tenant_id) + ttl = self._get_ttl_seconds() + + # Check current usage + current = self.redis.get(key) + + if current is None: + # First execution of the day - set to 1 + self.redis.setex(key, ttl, 1) + return True + + current_count = int(current) + if current_count < max_daily_limit: + # Within limit, increment + new_count = self.redis.incr(key) + # Update TTL + self.redis.expire(key, ttl) + + # Double-check in case of race condition + if new_count <= max_daily_limit: + return True + else: + # Race condition occurred, decrement back + self.redis.decr(key) + return False + else: + # Limit exceeded + return False + + def get_remaining_quota(self, tenant_id: str, max_daily_limit: int) -> int: + """ + Get remaining quota for the day + + Args: + tenant_id: The tenant identifier + max_daily_limit: Maximum daily limit + + Returns: + Number of remaining executions for the day + """ + key = self._get_day_key(tenant_id) + used = int(self.redis.get(key) or 0) + return max(0, max_daily_limit - used) + + def get_current_usage(self, tenant_id: str) -> int: + """ + Get current usage for the day + + Args: + tenant_id: The tenant identifier + + Returns: + Number of executions used today + """ + key = self._get_day_key(tenant_id) + return int(self.redis.get(key) or 0) + + def reset_quota(self, tenant_id: str) -> bool: + """ + Reset quota for testing purposes + + Args: + tenant_id: The tenant identifier + + Returns: + True if key was deleted, False if key didn't exist + """ + key = self._get_day_key(tenant_id) + return bool(self.redis.delete(key)) + + def get_quota_reset_time(self, tenant_id: str, timezone_str: str) -> datetime: + """ + Get the time when quota will reset (next UTC midnight in tenant's timezone) + + Args: + tenant_id: The tenant identifier + timezone_str: Tenant's timezone for display purposes + + Returns: + Datetime when quota resets (next UTC midnight in tenant's timezone) + """ + tz = pytz.timezone(timezone_str) + utc_now = datetime.now(UTC) + + # Get next midnight in UTC, then convert to tenant's timezone + next_utc_midnight = datetime.combine(utc_now.date() + timedelta(days=1), time.min) + next_utc_midnight = pytz.UTC.localize(next_utc_midnight) + + return next_utc_midnight.astimezone(tz) diff --git a/api/services/workflow/scheduler.py b/api/services/workflow/scheduler.py new file mode 100644 index 0000000000..7728c7f470 --- /dev/null +++ b/api/services/workflow/scheduler.py @@ -0,0 +1,34 @@ +from abc import ABC, abstractmethod +from enum import StrEnum + +from services.workflow.entities import WorkflowScheduleCFSPlanEntity + + +class SchedulerCommand(StrEnum): + """ + Scheduler command. + """ + + RESOURCE_LIMIT_REACHED = "resource_limit_reached" + NONE = "none" + + +class CFSPlanScheduler(ABC): + """ + CFS plan scheduler. + """ + + def __init__(self, plan: WorkflowScheduleCFSPlanEntity): + """ + Initialize the CFS plan scheduler. + + Args: + plan: The CFS plan. + """ + self.plan = plan + + @abstractmethod + def can_schedule(self) -> SchedulerCommand: + """ + Whether a workflow run can be scheduled. + """ diff --git a/api/services/workflow/workflow_converter.py b/api/services/workflow/workflow_converter.py index e70b2b5c95..067feb994f 100644 --- a/api/services/workflow/workflow_converter.py +++ b/api/services/workflow/workflow_converter.py @@ -1,5 +1,5 @@ import json -from typing import Any +from typing import Any, TypedDict from core.app.app_config.entities import ( DatasetEntity, @@ -28,6 +28,12 @@ from models.model import App, AppMode, AppModelConfig from models.workflow import Workflow, WorkflowType +class _NodeType(TypedDict): + id: str + position: None + data: dict[str, Any] + + class WorkflowConverter: """ App Convert to Workflow Mode @@ -217,7 +223,7 @@ class WorkflowConverter: return app_config - def _convert_to_start_node(self, variables: list[VariableEntity]): + def _convert_to_start_node(self, variables: list[VariableEntity]) -> _NodeType: """ Convert to Start Node :param variables: list of variables @@ -235,7 +241,7 @@ class WorkflowConverter: def _convert_to_http_request_node( self, app_model: App, variables: list[VariableEntity], external_data_variables: list[ExternalDataVariableEntity] - ) -> tuple[list[dict], dict[str, str]]: + ) -> tuple[list[_NodeType], dict[str, str]]: """ Convert API Based Extension to HTTP Request Node :param app_model: App instance @@ -285,7 +291,7 @@ class WorkflowConverter: request_body_json = json.dumps(request_body) request_body_json = request_body_json.replace(r"\{\{", "{{").replace(r"\}\}", "}}") - http_request_node = { + http_request_node: _NodeType = { "id": f"http_request_{index}", "position": None, "data": { @@ -303,7 +309,7 @@ class WorkflowConverter: nodes.append(http_request_node) # append code node for response body parsing - code_node: dict[str, Any] = { + code_node: _NodeType = { "id": f"code_{index}", "position": None, "data": { @@ -326,7 +332,7 @@ class WorkflowConverter: def _convert_to_knowledge_retrieval_node( self, new_app_mode: AppMode, dataset_config: DatasetEntity, model_config: ModelConfigEntity - ) -> dict | None: + ) -> _NodeType | None: """ Convert datasets to Knowledge Retrieval Node :param new_app_mode: new app mode @@ -384,7 +390,7 @@ class WorkflowConverter: prompt_template: PromptTemplateEntity, file_upload: FileUploadConfig | None = None, external_data_variable_node_mapping: dict[str, str] | None = None, - ): + ) -> _NodeType: """ Convert to LLM Node :param original_app_mode: original app mode @@ -561,7 +567,7 @@ class WorkflowConverter: return template - def _convert_to_end_node(self): + def _convert_to_end_node(self) -> _NodeType: """ Convert to End Node :return: @@ -577,7 +583,7 @@ class WorkflowConverter: }, } - def _convert_to_answer_node(self): + def _convert_to_answer_node(self) -> _NodeType: """ Convert to Answer Node :return: @@ -598,7 +604,7 @@ class WorkflowConverter: """ return {"id": f"{source}-{target}", "source": source, "target": target} - def _append_node(self, graph: dict, node: dict): + def _append_node(self, graph: dict[str, Any], node: _NodeType): """ Append Node to Graph diff --git a/api/services/workflow_app_service.py b/api/services/workflow_app_service.py index 23dd436675..01f0c7a55a 100644 --- a/api/services/workflow_app_service.py +++ b/api/services/workflow_app_service.py @@ -1,12 +1,37 @@ +import json import uuid from datetime import datetime +from typing import Any from sqlalchemy import and_, func, or_, select from sqlalchemy.orm import Session from core.workflow.enums import WorkflowExecutionStatus from models import Account, App, EndUser, WorkflowAppLog, WorkflowRun -from models.enums import CreatorUserRole +from models.enums import AppTriggerType, CreatorUserRole +from models.trigger import WorkflowTriggerLog +from services.plugin.plugin_service import PluginService +from services.workflow.entities import TriggerMetadata + + +# Since the workflow_app_log table has exceeded 100 million records, we use an additional details field to extend it +class LogView: + """Lightweight wrapper for WorkflowAppLog with computed details. + + - Exposes `details_` for marshalling to `details` in API response + - Proxies all other attributes to the underlying `WorkflowAppLog` + """ + + def __init__(self, log: WorkflowAppLog, details: dict | None): + self.log = log + self.details_ = details + + @property + def details(self) -> dict | None: + return self.details_ + + def __getattr__(self, name): + return getattr(self.log, name) class WorkflowAppService: @@ -21,6 +46,7 @@ class WorkflowAppService: created_at_after: datetime | None = None, page: int = 1, limit: int = 20, + detail: bool = False, created_by_end_user_session_id: str | None = None, created_by_account: str | None = None, ): @@ -34,6 +60,7 @@ class WorkflowAppService: :param created_at_after: filter logs created after this timestamp :param page: page number :param limit: items per page + :param detail: whether to return detailed logs :param created_by_end_user_session_id: filter by end user session id :param created_by_account: filter by account email :return: Pagination object @@ -43,8 +70,20 @@ class WorkflowAppService: WorkflowAppLog.tenant_id == app_model.tenant_id, WorkflowAppLog.app_id == app_model.id ) + if detail: + # Simple left join by workflow_run_id to fetch trigger_metadata + stmt = stmt.outerjoin( + WorkflowTriggerLog, + and_( + WorkflowTriggerLog.tenant_id == app_model.tenant_id, + WorkflowTriggerLog.app_id == app_model.id, + WorkflowTriggerLog.workflow_run_id == WorkflowAppLog.workflow_run_id, + ), + ).add_columns(WorkflowTriggerLog.trigger_metadata) + if keyword or status: stmt = stmt.join(WorkflowRun, WorkflowRun.id == WorkflowAppLog.workflow_run_id) + # Join to workflow run for filtering when needed. if keyword: keyword_like_val = f"%{keyword[:30].encode('unicode_escape').decode('utf-8')}%".replace(r"\u", r"\\u") @@ -108,9 +147,17 @@ class WorkflowAppService: # Apply pagination limits offset_stmt = stmt.offset((page - 1) * limit).limit(limit) - # Execute query and get items - items = list(session.scalars(offset_stmt).all()) + # wrapper moved to module scope as `LogView` + # Execute query and get items + if detail: + rows = session.execute(offset_stmt).all() + items = [ + LogView(log, {"trigger_metadata": self.handle_trigger_metadata(app_model.tenant_id, meta_val)}) + for log, meta_val in rows + ] + else: + items = [LogView(log, None) for log in session.scalars(offset_stmt).all()] return { "page": page, "limit": limit, @@ -119,6 +166,31 @@ class WorkflowAppService: "data": items, } + def handle_trigger_metadata(self, tenant_id: str, meta_val: str) -> dict[str, Any]: + metadata: dict[str, Any] | None = self._safe_json_loads(meta_val) + if not metadata: + return {} + trigger_metadata = TriggerMetadata.model_validate(metadata) + if trigger_metadata.type == AppTriggerType.TRIGGER_PLUGIN: + icon = metadata.get("icon_filename") + icon_dark = metadata.get("icon_dark_filename") + metadata["icon"] = PluginService.get_plugin_icon_url(tenant_id=tenant_id, filename=icon) if icon else None + metadata["icon_dark"] = ( + PluginService.get_plugin_icon_url(tenant_id=tenant_id, filename=icon_dark) if icon_dark else None + ) + return metadata + + @staticmethod + def _safe_json_loads(val): + if not val: + return None + if isinstance(val, str): + try: + return json.loads(val) + except Exception: + return None + return val + @staticmethod def _safe_parse_uuid(value: str): # fast check diff --git a/api/services/workflow_draft_variable_service.py b/api/services/workflow_draft_variable_service.py index 5e63a83bb1..c5d1f6ab13 100644 --- a/api/services/workflow_draft_variable_service.py +++ b/api/services/workflow_draft_variable_service.py @@ -808,7 +808,11 @@ class DraftVariableSaver: # We only save conversation variable here. if selector[0] != CONVERSATION_VARIABLE_NODE_ID: continue - segment = WorkflowDraftVariable.build_segment_with_type(segment_type=item.value_type, value=item.new_value) + # Conversation variables are exposed as NUMBER in the UI even if their + # persisted type is INTEGER. Allow float updates by loosening the type + # to NUMBER here so downstream storage infers the precise subtype. + segment_type = SegmentType.NUMBER if item.value_type == SegmentType.INTEGER else item.value_type + segment = WorkflowDraftVariable.build_segment_with_type(segment_type=segment_type, value=item.new_value) draft_vars.append( WorkflowDraftVariable.new_conversation_variable( app_id=self._app_id, @@ -1026,7 +1030,7 @@ class DraftVariableSaver: return if self._node_type == NodeType.VARIABLE_ASSIGNER: draft_vars = self._build_from_variable_assigner_mapping(process_data=process_data) - elif self._node_type == NodeType.START: + elif self._node_type == NodeType.START or self._node_type.is_trigger_node: draft_vars = self._build_variables_from_start_mapping(outputs) else: draft_vars = self._build_variables_from_mapping(outputs) diff --git a/api/services/workflow_run_service.py b/api/services/workflow_run_service.py index 5c8719b499..b903d8df5f 100644 --- a/api/services/workflow_run_service.py +++ b/api/services/workflow_run_service.py @@ -1,6 +1,7 @@ import threading from collections.abc import Sequence +from sqlalchemy import Engine from sqlalchemy.orm import sessionmaker import contexts @@ -14,17 +15,26 @@ from models import ( WorkflowRun, WorkflowRunTriggeredFrom, ) +from repositories.api_workflow_run_repository import APIWorkflowRunRepository from repositories.factory import DifyAPIRepositoryFactory class WorkflowRunService: - def __init__(self): + _session_factory: sessionmaker + _workflow_run_repo: APIWorkflowRunRepository + + def __init__(self, session_factory: Engine | sessionmaker | None = None): """Initialize WorkflowRunService with repository dependencies.""" - session_maker = sessionmaker(bind=db.engine, expire_on_commit=False) + if session_factory is None: + session_factory = sessionmaker(bind=db.engine, expire_on_commit=False) + elif isinstance(session_factory, Engine): + session_factory = sessionmaker(bind=session_factory, expire_on_commit=False) + + self._session_factory = session_factory self._node_execution_service_repo = DifyAPIRepositoryFactory.create_api_workflow_node_execution_repository( - session_maker + self._session_factory ) - self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(session_maker) + self._workflow_run_repo = DifyAPIRepositoryFactory.create_api_workflow_run_repository(self._session_factory) def get_paginate_advanced_chat_workflow_runs( self, app_model: App, args: dict, triggered_from: WorkflowRunTriggeredFrom = WorkflowRunTriggeredFrom.DEBUGGING diff --git a/api/services/workflow_service.py b/api/services/workflow_service.py index 2f69e46074..e8088e17c1 100644 --- a/api/services/workflow_service.py +++ b/api/services/workflow_service.py @@ -14,7 +14,7 @@ from core.file import File from core.repositories import DifyCoreRepositoryFactory from core.variables import Variable from core.variables.variables import VariableUnion -from core.workflow.entities import WorkflowNodeExecution +from core.workflow.entities import VariablePool, WorkflowNodeExecution from core.workflow.enums import ErrorStrategy, WorkflowNodeExecutionMetadataKey, WorkflowNodeExecutionStatus from core.workflow.errors import WorkflowNodeRunFailedError from core.workflow.graph_events import GraphNodeEventBase, NodeRunFailedEvent, NodeRunSucceededEvent @@ -23,7 +23,6 @@ from core.workflow.nodes import NodeType from core.workflow.nodes.base.node import Node from core.workflow.nodes.node_mapping import LATEST_VERSION, NODE_TYPE_CLASSES_MAPPING from core.workflow.nodes.start.entities import StartNodeData -from core.workflow.runtime import VariablePool from core.workflow.system_variable import SystemVariable from core.workflow.workflow_entry import WorkflowEntry from events.app_event import app_draft_workflow_was_synced, app_published_workflow_was_updated @@ -211,6 +210,9 @@ class WorkflowService: # validate features structure self.validate_features_structure(app_model=app_model, features=features) + # validate graph structure + self.validate_graph_structure(graph=graph) + # create draft workflow if not found if not workflow: workflow = Workflow( @@ -267,6 +269,9 @@ class WorkflowService: if FeatureService.get_system_features().plugin_manager.enabled: self._validate_workflow_credentials(draft_workflow) + # validate graph structure + self.validate_graph_structure(graph=draft_workflow.graph_dict) + # create new workflow workflow = Workflow.new( tenant_id=app_model.tenant_id, @@ -623,7 +628,7 @@ class WorkflowService: node_config = draft_workflow.get_node_config_by_id(node_id) node_type = Workflow.get_node_type_from_node_config(node_config) node_data = node_config.get("data", {}) - if node_type == NodeType.START: + if node_type.is_start_node: with Session(bind=db.engine) as session, session.begin(): draft_var_srv = WorkflowDraftVariableService(session) conversation_id = draft_var_srv.get_or_create_conversation( @@ -631,10 +636,11 @@ class WorkflowService: app=app_model, workflow=draft_workflow, ) - start_data = StartNodeData.model_validate(node_data) - user_inputs = _rebuild_file_for_user_inputs_in_start_node( - tenant_id=draft_workflow.tenant_id, start_node_data=start_data, user_inputs=user_inputs - ) + if node_type is NodeType.START: + start_data = StartNodeData.model_validate(node_data) + user_inputs = _rebuild_file_for_user_inputs_in_start_node( + tenant_id=draft_workflow.tenant_id, start_node_data=start_data, user_inputs=user_inputs + ) # init variable pool variable_pool = _setup_variable_pool( query=query, @@ -895,6 +901,31 @@ class WorkflowService: return new_app + def validate_graph_structure(self, graph: Mapping[str, Any]): + """ + Validate workflow graph structure. + + This performs a lightweight validation on the graph, checking for structural + inconsistencies such as the coexistence of start and trigger nodes. + """ + node_configs = graph.get("nodes", []) + node_configs = cast(list[dict[str, Any]], node_configs) + + # is empty graph + if not node_configs: + return + + node_types: set[NodeType] = set() + for node in node_configs: + node_type = node.get("data", {}).get("type") + if node_type: + node_types.add(NodeType(node_type)) + + # start node and trigger node cannot coexist + if NodeType.START in node_types: + if any(nt.is_trigger_node for nt in node_types): + raise ValueError("Start node and trigger nodes cannot coexist in the same workflow") + def validate_features_structure(self, app_model: App, features: dict): if app_model.mode == AppMode.ADVANCED_CHAT: return AdvancedChatAppConfigManager.config_validate( @@ -997,10 +1028,11 @@ def _setup_variable_pool( conversation_variables: list[Variable], ): # Only inject system variables for START node type. - if node_type == NodeType.START: + if node_type == NodeType.START or node_type.is_trigger_node: system_variable = SystemVariable( user_id=user_id, app_id=workflow.app_id, + timestamp=int(naive_utc_now().timestamp()), workflow_id=workflow.id, files=files or [], workflow_execution_id=str(uuid.uuid4()), diff --git a/api/tasks/add_document_to_index_task.py b/api/tasks/add_document_to_index_task.py index 5df9888acc..933ad6b9e2 100644 --- a/api/tasks/add_document_to_index_task.py +++ b/api/tasks/add_document_to_index_task.py @@ -48,7 +48,6 @@ def add_document_to_index_task(dataset_document_id: str): db.session.query(DocumentSegment) .where( DocumentSegment.document_id == dataset_document.id, - DocumentSegment.enabled == False, DocumentSegment.status == "completed", ) .order_by(DocumentSegment.position.asc()) diff --git a/api/tasks/async_workflow_tasks.py b/api/tasks/async_workflow_tasks.py new file mode 100644 index 0000000000..a9907ac981 --- /dev/null +++ b/api/tasks/async_workflow_tasks.py @@ -0,0 +1,186 @@ +""" +Celery tasks for async workflow execution. + +These tasks handle workflow execution for different subscription tiers +with appropriate retry policies and error handling. +""" + +from datetime import UTC, datetime +from typing import Any + +from celery import shared_task +from sqlalchemy import select +from sqlalchemy.orm import Session, sessionmaker + +from configs import dify_config +from core.app.apps.workflow.app_generator import WorkflowAppGenerator +from core.app.entities.app_invoke_entities import InvokeFrom +from core.app.layers.timeslice_layer import TimeSliceLayer +from core.app.layers.trigger_post_layer import TriggerPostLayer +from extensions.ext_database import db +from models.account import Account +from models.enums import CreatorUserRole, WorkflowTriggerStatus +from models.model import App, EndUser, Tenant +from models.trigger import WorkflowTriggerLog +from models.workflow import Workflow +from repositories.sqlalchemy_workflow_trigger_log_repository import SQLAlchemyWorkflowTriggerLogRepository +from services.errors.app import WorkflowNotFoundError +from services.workflow.entities import ( + TriggerData, + WorkflowTaskData, +) +from tasks.workflow_cfs_scheduler.cfs_scheduler import AsyncWorkflowCFSPlanEntity, AsyncWorkflowCFSPlanScheduler +from tasks.workflow_cfs_scheduler.entities import AsyncWorkflowQueue, AsyncWorkflowSystemStrategy + + +@shared_task(queue=AsyncWorkflowQueue.PROFESSIONAL_QUEUE) +def execute_workflow_professional(task_data_dict: dict[str, Any]): + """Execute workflow for professional tier with highest priority""" + task_data = WorkflowTaskData.model_validate(task_data_dict) + cfs_plan_scheduler_entity = AsyncWorkflowCFSPlanEntity( + queue=AsyncWorkflowQueue.PROFESSIONAL_QUEUE, + schedule_strategy=AsyncWorkflowSystemStrategy, + granularity=dify_config.ASYNC_WORKFLOW_SCHEDULER_GRANULARITY, + ) + _execute_workflow_common( + task_data, + AsyncWorkflowCFSPlanScheduler(plan=cfs_plan_scheduler_entity), + cfs_plan_scheduler_entity, + ) + + +@shared_task(queue=AsyncWorkflowQueue.TEAM_QUEUE) +def execute_workflow_team(task_data_dict: dict[str, Any]): + """Execute workflow for team tier""" + task_data = WorkflowTaskData.model_validate(task_data_dict) + cfs_plan_scheduler_entity = AsyncWorkflowCFSPlanEntity( + queue=AsyncWorkflowQueue.TEAM_QUEUE, + schedule_strategy=AsyncWorkflowSystemStrategy, + granularity=dify_config.ASYNC_WORKFLOW_SCHEDULER_GRANULARITY, + ) + _execute_workflow_common( + task_data, + AsyncWorkflowCFSPlanScheduler(plan=cfs_plan_scheduler_entity), + cfs_plan_scheduler_entity, + ) + + +@shared_task(queue=AsyncWorkflowQueue.SANDBOX_QUEUE) +def execute_workflow_sandbox(task_data_dict: dict[str, Any]): + """Execute workflow for free tier with lower retry limit""" + task_data = WorkflowTaskData.model_validate(task_data_dict) + cfs_plan_scheduler_entity = AsyncWorkflowCFSPlanEntity( + queue=AsyncWorkflowQueue.SANDBOX_QUEUE, + schedule_strategy=AsyncWorkflowSystemStrategy, + granularity=dify_config.ASYNC_WORKFLOW_SCHEDULER_GRANULARITY, + ) + _execute_workflow_common( + task_data, + AsyncWorkflowCFSPlanScheduler(plan=cfs_plan_scheduler_entity), + cfs_plan_scheduler_entity, + ) + + +def _execute_workflow_common( + task_data: WorkflowTaskData, + cfs_plan_scheduler: AsyncWorkflowCFSPlanScheduler, + cfs_plan_scheduler_entity: AsyncWorkflowCFSPlanEntity, +): + """Execute workflow with common logic and trigger log updates.""" + + # Create a new session for this task + session_factory = sessionmaker(bind=db.engine, expire_on_commit=False) + + with session_factory() as session: + trigger_log_repo = SQLAlchemyWorkflowTriggerLogRepository(session) + + # Get trigger log + trigger_log = trigger_log_repo.get_by_id(task_data.workflow_trigger_log_id) + + if not trigger_log: + # This should not happen, but handle gracefully + return + + # Reconstruct execution data from trigger log + trigger_data = TriggerData.model_validate_json(trigger_log.trigger_data) + + # Update status to running + trigger_log.status = WorkflowTriggerStatus.RUNNING + trigger_log_repo.update(trigger_log) + session.commit() + + start_time = datetime.now(UTC) + + try: + # Get app and workflow models + app_model = session.scalar(select(App).where(App.id == trigger_log.app_id)) + + if not app_model: + raise WorkflowNotFoundError(f"App not found: {trigger_log.app_id}") + + workflow = session.scalar(select(Workflow).where(Workflow.id == trigger_log.workflow_id)) + if not workflow: + raise WorkflowNotFoundError(f"Workflow not found: {trigger_log.workflow_id}") + + user = _get_user(session, trigger_log) + + # Execute workflow using WorkflowAppGenerator + generator = WorkflowAppGenerator() + + # Prepare args matching AppGenerateService.generate format + args: dict[str, Any] = {"inputs": dict(trigger_data.inputs), "files": list(trigger_data.files)} + + # If workflow_id was specified, add it to args + if trigger_data.workflow_id: + args["workflow_id"] = str(trigger_data.workflow_id) + + # Execute the workflow with the trigger type + generator.generate( + app_model=app_model, + workflow=workflow, + user=user, + args=args, + invoke_from=InvokeFrom.SERVICE_API, + streaming=False, + call_depth=0, + triggered_from=trigger_data.trigger_from, + root_node_id=trigger_data.root_node_id, + graph_engine_layers=[ + TimeSliceLayer(cfs_plan_scheduler), + TriggerPostLayer(cfs_plan_scheduler_entity, start_time, trigger_log.id, session_factory), + ], + ) + + except Exception as e: + # Calculate elapsed time for failed execution + elapsed_time = (datetime.now(UTC) - start_time).total_seconds() + + # Update trigger log with failure + trigger_log.status = WorkflowTriggerStatus.FAILED + trigger_log.error = str(e) + trigger_log.finished_at = datetime.now(UTC) + trigger_log.elapsed_time = elapsed_time + trigger_log_repo.update(trigger_log) + + # Final failure - no retry logic (simplified like RAG tasks) + session.commit() + + +def _get_user(session: Session, trigger_log: WorkflowTriggerLog) -> Account | EndUser: + """Compose user from trigger log""" + tenant = session.scalar(select(Tenant).where(Tenant.id == trigger_log.tenant_id)) + if not tenant: + raise ValueError(f"Tenant not found: {trigger_log.tenant_id}") + + # Get user from trigger log + if trigger_log.created_by_role == CreatorUserRole.ACCOUNT: + user = session.scalar(select(Account).where(Account.id == trigger_log.created_by)) + if user: + user.current_tenant = tenant + else: # CreatorUserRole.END_USER + user = session.scalar(select(EndUser).where(EndUser.id == trigger_log.created_by)) + + if not user: + raise ValueError(f"User not found: {trigger_log.created_by} (role: {trigger_log.created_by_role})") + + return user diff --git a/api/tasks/document_indexing_task.py b/api/tasks/document_indexing_task.py index 012ae8f706..fee4430612 100644 --- a/api/tasks/document_indexing_task.py +++ b/api/tasks/document_indexing_task.py @@ -1,11 +1,15 @@ import logging import time +from collections.abc import Callable, Sequence import click from celery import shared_task from configs import dify_config +from core.entities.document_task import DocumentTask from core.indexing_runner import DocumentIsPausedError, IndexingRunner +from core.rag.pipeline.queue import TenantIsolatedTaskQueue +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from libs.datetime_utils import naive_utc_now from models.dataset import Dataset, Document @@ -21,8 +25,24 @@ def document_indexing_task(dataset_id: str, document_ids: list): :param dataset_id: :param document_ids: + .. warning:: TO BE DEPRECATED + This function will be deprecated and removed in a future version. + Use normal_document_indexing_task or priority_document_indexing_task instead. + Usage: document_indexing_task.delay(dataset_id, document_ids) """ + logger.warning("document indexing legacy mode received: %s - %s", dataset_id, document_ids) + _document_indexing(dataset_id, document_ids) + + +def _document_indexing(dataset_id: str, document_ids: Sequence[str]): + """ + Process document for tasks + :param dataset_id: + :param document_ids: + + Usage: _document_indexing(dataset_id, document_ids) + """ documents = [] start_at = time.perf_counter() @@ -38,7 +58,7 @@ def document_indexing_task(dataset_id: str, document_ids: list): vector_space = features.vector_space count = len(document_ids) batch_upload_limit = int(dify_config.BATCH_UPLOAD_LIMIT) - if features.billing.subscription.plan == "sandbox" and count > 1: + if features.billing.subscription.plan == CloudPlan.SANDBOX and count > 1: raise ValueError("Your current plan does not support batch upload, please upgrade your plan.") if count > batch_upload_limit: raise ValueError(f"You have reached the batch upload limit of {batch_upload_limit}.") @@ -86,3 +106,63 @@ def document_indexing_task(dataset_id: str, document_ids: list): logger.exception("Document indexing task failed, dataset_id: %s", dataset_id) finally: db.session.close() + + +def _document_indexing_with_tenant_queue( + tenant_id: str, dataset_id: str, document_ids: Sequence[str], task_func: Callable[[str, str, Sequence[str]], None] +): + try: + _document_indexing(dataset_id, document_ids) + except Exception: + logger.exception("Error processing document indexing %s for tenant %s: %s", dataset_id, tenant_id) + finally: + tenant_isolated_task_queue = TenantIsolatedTaskQueue(tenant_id, "document_indexing") + + # Check if there are waiting tasks in the queue + # Use rpop to get the next task from the queue (FIFO order) + next_tasks = tenant_isolated_task_queue.pull_tasks(count=dify_config.TENANT_ISOLATED_TASK_CONCURRENCY) + + logger.info("document indexing tenant isolation queue next tasks: %s", next_tasks) + + if next_tasks: + for next_task in next_tasks: + document_task = DocumentTask(**next_task) + # Process the next waiting task + # Keep the flag set to indicate a task is running + tenant_isolated_task_queue.set_task_waiting_time() + task_func.delay( # type: ignore + tenant_id=document_task.tenant_id, + dataset_id=document_task.dataset_id, + document_ids=document_task.document_ids, + ) + else: + # No more waiting tasks, clear the flag + tenant_isolated_task_queue.delete_task_key() + + +@shared_task(queue="dataset") +def normal_document_indexing_task(tenant_id: str, dataset_id: str, document_ids: Sequence[str]): + """ + Async process document + :param tenant_id: + :param dataset_id: + :param document_ids: + + Usage: normal_document_indexing_task.delay(tenant_id, dataset_id, document_ids) + """ + logger.info("normal document indexing task received: %s - %s - %s", tenant_id, dataset_id, document_ids) + _document_indexing_with_tenant_queue(tenant_id, dataset_id, document_ids, normal_document_indexing_task) + + +@shared_task(queue="priority_dataset") +def priority_document_indexing_task(tenant_id: str, dataset_id: str, document_ids: Sequence[str]): + """ + Priority async process document + :param tenant_id: + :param dataset_id: + :param document_ids: + + Usage: priority_document_indexing_task.delay(tenant_id, dataset_id, document_ids) + """ + logger.info("priority document indexing task received: %s - %s - %s", tenant_id, dataset_id, document_ids) + _document_indexing_with_tenant_queue(tenant_id, dataset_id, document_ids, priority_document_indexing_task) diff --git a/api/tasks/duplicate_document_indexing_task.py b/api/tasks/duplicate_document_indexing_task.py index 2020179cd9..6492e356a3 100644 --- a/api/tasks/duplicate_document_indexing_task.py +++ b/api/tasks/duplicate_document_indexing_task.py @@ -8,6 +8,7 @@ from sqlalchemy import select from configs import dify_config from core.indexing_runner import DocumentIsPausedError, IndexingRunner from core.rag.index_processor.index_processor_factory import IndexProcessorFactory +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from libs.datetime_utils import naive_utc_now from models.dataset import Dataset, Document, DocumentSegment @@ -41,7 +42,7 @@ def duplicate_document_indexing_task(dataset_id: str, document_ids: list): if features.billing.enabled: vector_space = features.vector_space count = len(document_ids) - if features.billing.subscription.plan == "sandbox" and count > 1: + if features.billing.subscription.plan == CloudPlan.SANDBOX and count > 1: raise ValueError("Your current plan does not support batch upload, please upgrade your plan.") batch_upload_limit = int(dify_config.BATCH_UPLOAD_LIMIT) if count > batch_upload_limit: diff --git a/api/tasks/rag_pipeline/priority_rag_pipeline_run_task.py b/api/tasks/rag_pipeline/priority_rag_pipeline_run_task.py index 6de95a3b85..a7f61d9811 100644 --- a/api/tasks/rag_pipeline/priority_rag_pipeline_run_task.py +++ b/api/tasks/rag_pipeline/priority_rag_pipeline_run_task.py @@ -12,8 +12,10 @@ from celery import shared_task # type: ignore from flask import current_app, g from sqlalchemy.orm import Session, sessionmaker +from configs import dify_config from core.app.entities.app_invoke_entities import InvokeFrom, RagPipelineGenerateEntity from core.app.entities.rag_pipeline_invoke_entities import RagPipelineInvokeEntity +from core.rag.pipeline.queue import TenantIsolatedTaskQueue from core.repositories.factory import DifyCoreRepositoryFactory from extensions.ext_database import db from models import Account, Tenant @@ -22,6 +24,8 @@ from models.enums import WorkflowRunTriggeredFrom from models.workflow import Workflow, WorkflowNodeExecutionTriggeredFrom from services.file_service import FileService +logger = logging.getLogger(__name__) + @shared_task(queue="priority_pipeline") def priority_rag_pipeline_run_task( @@ -69,6 +73,27 @@ def priority_rag_pipeline_run_task( logging.exception(click.style(f"Error running rag pipeline, tenant_id: {tenant_id}", fg="red")) raise finally: + tenant_isolated_task_queue = TenantIsolatedTaskQueue(tenant_id, "pipeline") + + # Check if there are waiting tasks in the queue + # Use rpop to get the next task from the queue (FIFO order) + next_file_ids = tenant_isolated_task_queue.pull_tasks(count=dify_config.TENANT_ISOLATED_TASK_CONCURRENCY) + logger.info("priority rag pipeline tenant isolation queue next files: %s", next_file_ids) + + if next_file_ids: + for next_file_id in next_file_ids: + # Process the next waiting task + # Keep the flag set to indicate a task is running + tenant_isolated_task_queue.set_task_waiting_time() + priority_rag_pipeline_run_task.delay( # type: ignore + rag_pipeline_invoke_entities_file_id=next_file_id.decode("utf-8") + if isinstance(next_file_id, bytes) + else next_file_id, + tenant_id=tenant_id, + ) + else: + # No more waiting tasks, clear the flag + tenant_isolated_task_queue.delete_task_key() file_service = FileService(db.engine) file_service.delete_file(rag_pipeline_invoke_entities_file_id) db.session.close() diff --git a/api/tasks/rag_pipeline/rag_pipeline_run_task.py b/api/tasks/rag_pipeline/rag_pipeline_run_task.py index f4a092d97e..92f1dfb73d 100644 --- a/api/tasks/rag_pipeline/rag_pipeline_run_task.py +++ b/api/tasks/rag_pipeline/rag_pipeline_run_task.py @@ -12,17 +12,20 @@ from celery import shared_task # type: ignore from flask import current_app, g from sqlalchemy.orm import Session, sessionmaker +from configs import dify_config from core.app.entities.app_invoke_entities import InvokeFrom, RagPipelineGenerateEntity from core.app.entities.rag_pipeline_invoke_entities import RagPipelineInvokeEntity +from core.rag.pipeline.queue import TenantIsolatedTaskQueue from core.repositories.factory import DifyCoreRepositoryFactory from extensions.ext_database import db -from extensions.ext_redis import redis_client from models import Account, Tenant from models.dataset import Pipeline from models.enums import WorkflowRunTriggeredFrom from models.workflow import Workflow, WorkflowNodeExecutionTriggeredFrom from services.file_service import FileService +logger = logging.getLogger(__name__) + @shared_task(queue="pipeline") def rag_pipeline_run_task( @@ -70,26 +73,27 @@ def rag_pipeline_run_task( logging.exception(click.style(f"Error running rag pipeline, tenant_id: {tenant_id}", fg="red")) raise finally: - tenant_self_pipeline_task_queue = f"tenant_self_pipeline_task_queue:{tenant_id}" - tenant_pipeline_task_key = f"tenant_pipeline_task:{tenant_id}" + tenant_isolated_task_queue = TenantIsolatedTaskQueue(tenant_id, "pipeline") # Check if there are waiting tasks in the queue # Use rpop to get the next task from the queue (FIFO order) - next_file_id = redis_client.rpop(tenant_self_pipeline_task_queue) + next_file_ids = tenant_isolated_task_queue.pull_tasks(count=dify_config.TENANT_ISOLATED_TASK_CONCURRENCY) + logger.info("rag pipeline tenant isolation queue next files: %s", next_file_ids) - if next_file_id: - # Process the next waiting task - # Keep the flag set to indicate a task is running - redis_client.setex(tenant_pipeline_task_key, 60 * 60, 1) - rag_pipeline_run_task.delay( # type: ignore - rag_pipeline_invoke_entities_file_id=next_file_id.decode("utf-8") - if isinstance(next_file_id, bytes) - else next_file_id, - tenant_id=tenant_id, - ) + if next_file_ids: + for next_file_id in next_file_ids: + # Process the next waiting task + # Keep the flag set to indicate a task is running + tenant_isolated_task_queue.set_task_waiting_time() + rag_pipeline_run_task.delay( # type: ignore + rag_pipeline_invoke_entities_file_id=next_file_id.decode("utf-8") + if isinstance(next_file_id, bytes) + else next_file_id, + tenant_id=tenant_id, + ) else: # No more waiting tasks, clear the flag - redis_client.delete(tenant_pipeline_task_key) + tenant_isolated_task_queue.delete_task_key() file_service = FileService(db.engine) file_service.delete_file(rag_pipeline_invoke_entities_file_id) db.session.close() diff --git a/api/tasks/remove_app_and_related_data_task.py b/api/tasks/remove_app_and_related_data_task.py index f8f39583ac..3227f6da96 100644 --- a/api/tasks/remove_app_and_related_data_task.py +++ b/api/tasks/remove_app_and_related_data_task.py @@ -17,6 +17,7 @@ from models import ( AppDatasetJoin, AppMCPServer, AppModelConfig, + AppTrigger, Conversation, EndUser, InstalledApp, @@ -30,8 +31,10 @@ from models import ( Site, TagBinding, TraceAppConfig, + WorkflowSchedulePlan, ) from models.tools import WorkflowToolProvider +from models.trigger import WorkflowPluginTrigger, WorkflowTriggerLog, WorkflowWebhookTrigger from models.web import PinnedConversation, SavedMessage from models.workflow import ( ConversationVariable, @@ -69,6 +72,11 @@ def remove_app_and_related_data_task(self, tenant_id: str, app_id: str): _delete_trace_app_configs(tenant_id, app_id) _delete_conversation_variables(app_id=app_id) _delete_draft_variables(app_id) + _delete_app_triggers(tenant_id, app_id) + _delete_workflow_plugin_triggers(tenant_id, app_id) + _delete_workflow_webhook_triggers(tenant_id, app_id) + _delete_workflow_schedule_plans(tenant_id, app_id) + _delete_workflow_trigger_logs(tenant_id, app_id) end_at = time.perf_counter() logger.info(click.style(f"App and related data deleted: {app_id} latency: {end_at - start_at}", fg="green")) @@ -484,6 +492,72 @@ def _delete_draft_variable_offload_data(conn, file_ids: list[str]) -> int: return files_deleted +def _delete_app_triggers(tenant_id: str, app_id: str): + def del_app_trigger(trigger_id: str): + db.session.query(AppTrigger).where(AppTrigger.id == trigger_id).delete(synchronize_session=False) + + _delete_records( + """select id from app_triggers where tenant_id=:tenant_id and app_id=:app_id limit 1000""", + {"tenant_id": tenant_id, "app_id": app_id}, + del_app_trigger, + "app trigger", + ) + + +def _delete_workflow_plugin_triggers(tenant_id: str, app_id: str): + def del_plugin_trigger(trigger_id: str): + db.session.query(WorkflowPluginTrigger).where(WorkflowPluginTrigger.id == trigger_id).delete( + synchronize_session=False + ) + + _delete_records( + """select id from workflow_plugin_triggers where tenant_id=:tenant_id and app_id=:app_id limit 1000""", + {"tenant_id": tenant_id, "app_id": app_id}, + del_plugin_trigger, + "workflow plugin trigger", + ) + + +def _delete_workflow_webhook_triggers(tenant_id: str, app_id: str): + def del_webhook_trigger(trigger_id: str): + db.session.query(WorkflowWebhookTrigger).where(WorkflowWebhookTrigger.id == trigger_id).delete( + synchronize_session=False + ) + + _delete_records( + """select id from workflow_webhook_triggers where tenant_id=:tenant_id and app_id=:app_id limit 1000""", + {"tenant_id": tenant_id, "app_id": app_id}, + del_webhook_trigger, + "workflow webhook trigger", + ) + + +def _delete_workflow_schedule_plans(tenant_id: str, app_id: str): + def del_schedule_plan(plan_id: str): + db.session.query(WorkflowSchedulePlan).where(WorkflowSchedulePlan.id == plan_id).delete( + synchronize_session=False + ) + + _delete_records( + """select id from workflow_schedule_plans where tenant_id=:tenant_id and app_id=:app_id limit 1000""", + {"tenant_id": tenant_id, "app_id": app_id}, + del_schedule_plan, + "workflow schedule plan", + ) + + +def _delete_workflow_trigger_logs(tenant_id: str, app_id: str): + def del_trigger_log(log_id: str): + db.session.query(WorkflowTriggerLog).where(WorkflowTriggerLog.id == log_id).delete(synchronize_session=False) + + _delete_records( + """select id from workflow_trigger_logs where tenant_id=:tenant_id and app_id=:app_id limit 1000""", + {"tenant_id": tenant_id, "app_id": app_id}, + del_trigger_log, + "workflow trigger log", + ) + + def _delete_records(query_sql: str, params: dict, delete_func: Callable, name: str) -> None: while True: with db.engine.begin() as conn: diff --git a/api/tasks/trigger_processing_tasks.py b/api/tasks/trigger_processing_tasks.py new file mode 100644 index 0000000000..985125e66b --- /dev/null +++ b/api/tasks/trigger_processing_tasks.py @@ -0,0 +1,492 @@ +""" +Celery tasks for async trigger processing. + +These tasks handle trigger workflow execution asynchronously +to avoid blocking the main request thread. +""" + +import json +import logging +from collections.abc import Mapping, Sequence +from datetime import UTC, datetime +from typing import Any + +from celery import shared_task +from sqlalchemy import func, select +from sqlalchemy.orm import Session + +from core.app.entities.app_invoke_entities import InvokeFrom +from core.plugin.entities.plugin_daemon import CredentialType +from core.plugin.entities.request import TriggerInvokeEventResponse +from core.plugin.impl.exc import PluginInvokeError +from core.trigger.debug.event_bus import TriggerDebugEventBus +from core.trigger.debug.events import PluginTriggerDebugEvent, build_plugin_pool_key +from core.trigger.entities.entities import TriggerProviderEntity +from core.trigger.provider import PluginTriggerProviderController +from core.trigger.trigger_manager import TriggerManager +from core.workflow.enums import NodeType, WorkflowExecutionStatus +from core.workflow.nodes.trigger_plugin.entities import TriggerEventNodeData +from extensions.ext_database import db +from models.enums import AppTriggerType, CreatorUserRole, WorkflowRunTriggeredFrom, WorkflowTriggerStatus +from models.model import EndUser +from models.provider_ids import TriggerProviderID +from models.trigger import TriggerSubscription, WorkflowPluginTrigger, WorkflowTriggerLog +from models.workflow import Workflow, WorkflowAppLog, WorkflowAppLogCreatedFrom, WorkflowRun +from services.async_workflow_service import AsyncWorkflowService +from services.end_user_service import EndUserService +from services.trigger.trigger_provider_service import TriggerProviderService +from services.trigger.trigger_request_service import TriggerHttpRequestCachingService +from services.trigger.trigger_subscription_operator_service import TriggerSubscriptionOperatorService +from services.workflow.entities import PluginTriggerData, PluginTriggerDispatchData, PluginTriggerMetadata +from services.workflow.queue_dispatcher import QueueDispatcherManager + +logger = logging.getLogger(__name__) + +# Use workflow queue for trigger processing +TRIGGER_QUEUE = "triggered_workflow_dispatcher" + + +def dispatch_trigger_debug_event( + events: list[str], + user_id: str, + timestamp: int, + request_id: str, + subscription: TriggerSubscription, +) -> int: + debug_dispatched = 0 + try: + for event_name in events: + pool_key: str = build_plugin_pool_key( + name=event_name, + tenant_id=subscription.tenant_id, + subscription_id=subscription.id, + provider_id=subscription.provider_id, + ) + trigger_debug_event: PluginTriggerDebugEvent = PluginTriggerDebugEvent( + timestamp=timestamp, + user_id=user_id, + name=event_name, + request_id=request_id, + subscription_id=subscription.id, + provider_id=subscription.provider_id, + ) + debug_dispatched += TriggerDebugEventBus.dispatch( + tenant_id=subscription.tenant_id, + event=trigger_debug_event, + pool_key=pool_key, + ) + logger.debug( + "Trigger debug dispatched %d sessions to pool %s for event %s for subscription %s provider %s", + debug_dispatched, + pool_key, + event_name, + subscription.id, + subscription.provider_id, + ) + return debug_dispatched + except Exception: + logger.exception("Failed to dispatch to debug sessions") + return 0 + + +def _get_latest_workflows_by_app_ids( + session: Session, subscribers: Sequence[WorkflowPluginTrigger] +) -> Mapping[str, Workflow]: + """Get the latest workflows by app_ids""" + workflow_query = ( + select(Workflow.app_id, func.max(Workflow.created_at).label("max_created_at")) + .where( + Workflow.app_id.in_({t.app_id for t in subscribers}), + Workflow.version != Workflow.VERSION_DRAFT, + ) + .group_by(Workflow.app_id) + .subquery() + ) + workflows = session.scalars( + select(Workflow).join( + workflow_query, + (Workflow.app_id == workflow_query.c.app_id) & (Workflow.created_at == workflow_query.c.max_created_at), + ) + ).all() + return {w.app_id: w for w in workflows} + + +def _record_trigger_failure_log( + *, + session: Session, + workflow: Workflow, + plugin_trigger: WorkflowPluginTrigger, + subscription: TriggerSubscription, + trigger_metadata: PluginTriggerMetadata, + end_user: EndUser | None, + error_message: str, + event_name: str, + request_id: str, +) -> None: + """ + Persist a workflow run, workflow app log, and trigger log entry for failed trigger invocations. + """ + now = datetime.now(UTC) + if end_user: + created_by_role = CreatorUserRole.END_USER + created_by = end_user.id + else: + created_by_role = CreatorUserRole.ACCOUNT + created_by = subscription.user_id + + failure_inputs = { + "event_name": event_name, + "subscription_id": subscription.id, + "request_id": request_id, + "plugin_trigger_id": plugin_trigger.id, + } + + workflow_run = WorkflowRun( + tenant_id=workflow.tenant_id, + app_id=workflow.app_id, + workflow_id=workflow.id, + type=workflow.type, + triggered_from=WorkflowRunTriggeredFrom.PLUGIN.value, + version=workflow.version, + graph=workflow.graph, + inputs=json.dumps(failure_inputs), + status=WorkflowExecutionStatus.FAILED.value, + outputs="{}", + error=error_message, + elapsed_time=0.0, + total_tokens=0, + total_steps=0, + created_by_role=created_by_role.value, + created_by=created_by, + created_at=now, + finished_at=now, + exceptions_count=0, + ) + session.add(workflow_run) + session.flush() + + workflow_app_log = WorkflowAppLog( + tenant_id=workflow.tenant_id, + app_id=workflow.app_id, + workflow_id=workflow.id, + workflow_run_id=workflow_run.id, + created_from=WorkflowAppLogCreatedFrom.SERVICE_API.value, + created_by_role=created_by_role.value, + created_by=created_by, + ) + session.add(workflow_app_log) + + dispatcher = QueueDispatcherManager.get_dispatcher(subscription.tenant_id) + queue_name = dispatcher.get_queue_name() + + trigger_data = PluginTriggerData( + app_id=plugin_trigger.app_id, + tenant_id=subscription.tenant_id, + workflow_id=workflow.id, + root_node_id=plugin_trigger.node_id, + inputs={}, + trigger_metadata=trigger_metadata, + plugin_id=subscription.provider_id, + endpoint_id=subscription.endpoint_id, + ) + + trigger_log = WorkflowTriggerLog( + tenant_id=workflow.tenant_id, + app_id=workflow.app_id, + workflow_id=workflow.id, + workflow_run_id=workflow_run.id, + root_node_id=plugin_trigger.node_id, + trigger_metadata=trigger_metadata.model_dump_json(), + trigger_type=AppTriggerType.TRIGGER_PLUGIN, + trigger_data=trigger_data.model_dump_json(), + inputs=json.dumps({}), + status=WorkflowTriggerStatus.FAILED, + error=error_message, + queue_name=queue_name, + retry_count=0, + created_by_role=created_by_role.value, + created_by=created_by, + triggered_at=now, + finished_at=now, + elapsed_time=0.0, + total_tokens=0, + ) + session.add(trigger_log) + session.commit() + + +def dispatch_triggered_workflow( + user_id: str, + subscription: TriggerSubscription, + event_name: str, + request_id: str, +) -> int: + """Process triggered workflows. + + Args: + subscription: The trigger subscription + event: The trigger entity that was activated + request_id: The ID of the stored request in storage system + """ + request = TriggerHttpRequestCachingService.get_request(request_id) + payload = TriggerHttpRequestCachingService.get_payload(request_id) + + subscribers: list[WorkflowPluginTrigger] = TriggerSubscriptionOperatorService.get_subscriber_triggers( + tenant_id=subscription.tenant_id, subscription_id=subscription.id, event_name=event_name + ) + if not subscribers: + logger.warning( + "No workflows found for trigger event '%s' in subscription '%s'", + event_name, + subscription.id, + ) + return 0 + + dispatched_count = 0 + provider_controller: PluginTriggerProviderController = TriggerManager.get_trigger_provider( + tenant_id=subscription.tenant_id, provider_id=TriggerProviderID(subscription.provider_id) + ) + trigger_entity: TriggerProviderEntity = provider_controller.entity + with Session(db.engine) as session: + workflows: Mapping[str, Workflow] = _get_latest_workflows_by_app_ids(session, subscribers) + + end_users: Mapping[str, EndUser] = EndUserService.create_end_user_batch( + type=InvokeFrom.TRIGGER, + tenant_id=subscription.tenant_id, + app_ids=[plugin_trigger.app_id for plugin_trigger in subscribers], + user_id=user_id, + ) + for plugin_trigger in subscribers: + # Get workflow from mapping + workflow: Workflow | None = workflows.get(plugin_trigger.app_id) + if not workflow: + logger.error( + "Workflow not found for app %s", + plugin_trigger.app_id, + ) + continue + + # Find the trigger node in the workflow + event_node = None + for node_id, node_config in workflow.walk_nodes(NodeType.TRIGGER_PLUGIN): + if node_id == plugin_trigger.node_id: + event_node = node_config + break + + if not event_node: + logger.error("Trigger event node not found for app %s", plugin_trigger.app_id) + continue + + # invoke trigger + trigger_metadata = PluginTriggerMetadata( + plugin_unique_identifier=provider_controller.plugin_unique_identifier or "", + endpoint_id=subscription.endpoint_id, + provider_id=subscription.provider_id, + event_name=event_name, + icon_filename=trigger_entity.identity.icon or "", + icon_dark_filename=trigger_entity.identity.icon_dark or "", + ) + + node_data: TriggerEventNodeData = TriggerEventNodeData.model_validate(event_node) + invoke_response: TriggerInvokeEventResponse | None = None + try: + invoke_response = TriggerManager.invoke_trigger_event( + tenant_id=subscription.tenant_id, + user_id=user_id, + provider_id=TriggerProviderID(subscription.provider_id), + event_name=event_name, + parameters=node_data.resolve_parameters( + parameter_schemas=provider_controller.get_event_parameters(event_name=event_name) + ), + credentials=subscription.credentials, + credential_type=CredentialType.of(subscription.credential_type), + subscription=subscription.to_entity(), + request=request, + payload=payload, + ) + except PluginInvokeError as e: + error_message = e.to_user_friendly_error(plugin_name=trigger_entity.identity.name) + try: + end_user = end_users.get(plugin_trigger.app_id) + _record_trigger_failure_log( + session=session, + workflow=workflow, + plugin_trigger=plugin_trigger, + subscription=subscription, + trigger_metadata=trigger_metadata, + end_user=end_user, + error_message=error_message, + event_name=event_name, + request_id=request_id, + ) + except Exception: + logger.exception( + "Failed to record trigger failure log for app %s", + plugin_trigger.app_id, + ) + continue + except Exception: + logger.exception( + "Failed to invoke trigger event for app %s", + plugin_trigger.app_id, + ) + continue + + if invoke_response is not None and invoke_response.cancelled: + logger.info( + "Trigger ignored for app %s with trigger event %s", + plugin_trigger.app_id, + event_name, + ) + continue + + # Create trigger data for async execution + trigger_data = PluginTriggerData( + app_id=plugin_trigger.app_id, + tenant_id=subscription.tenant_id, + workflow_id=workflow.id, + root_node_id=plugin_trigger.node_id, + plugin_id=subscription.provider_id, + endpoint_id=subscription.endpoint_id, + inputs=invoke_response.variables, + trigger_metadata=trigger_metadata, + ) + + # Trigger async workflow + try: + end_user = end_users.get(plugin_trigger.app_id) + if not end_user: + raise ValueError(f"End user not found for app {plugin_trigger.app_id}") + + AsyncWorkflowService.trigger_workflow_async(session=session, user=end_user, trigger_data=trigger_data) + dispatched_count += 1 + logger.info( + "Triggered workflow for app %s with trigger event %s", + plugin_trigger.app_id, + event_name, + ) + except Exception: + logger.exception( + "Failed to trigger workflow for app %s", + plugin_trigger.app_id, + ) + + return dispatched_count + + +def dispatch_triggered_workflows( + user_id: str, + events: list[str], + subscription: TriggerSubscription, + request_id: str, +) -> int: + dispatched_count = 0 + for event_name in events: + try: + dispatched_count += dispatch_triggered_workflow( + user_id=user_id, + subscription=subscription, + event_name=event_name, + request_id=request_id, + ) + except Exception: + logger.exception( + "Failed to dispatch trigger '%s' for subscription %s and provider %s. Continuing...", + event_name, + subscription.id, + subscription.provider_id, + ) + # Continue processing other triggers even if one fails + continue + + logger.info( + "Completed async trigger dispatching: processed %d/%d triggers for subscription %s and provider %s", + dispatched_count, + len(events), + subscription.id, + subscription.provider_id, + ) + return dispatched_count + + +@shared_task(queue=TRIGGER_QUEUE) +def dispatch_triggered_workflows_async( + dispatch_data: Mapping[str, Any], +) -> Mapping[str, Any]: + """ + Dispatch triggers asynchronously. + + Args: + endpoint_id: Endpoint ID + provider_id: Provider ID + subscription_id: Subscription ID + timestamp: Timestamp of the event + triggers: List of triggers to dispatch + request_id: Unique ID of the stored request + + Returns: + dict: Execution result with status and dispatched trigger count + """ + dispatch_params: PluginTriggerDispatchData = PluginTriggerDispatchData.model_validate(dispatch_data) + user_id = dispatch_params.user_id + tenant_id = dispatch_params.tenant_id + endpoint_id = dispatch_params.endpoint_id + provider_id = dispatch_params.provider_id + subscription_id = dispatch_params.subscription_id + timestamp = dispatch_params.timestamp + events = dispatch_params.events + request_id = dispatch_params.request_id + + try: + logger.info( + "Starting trigger dispatching uid=%s, endpoint=%s, events=%s, req_id=%s, sub_id=%s, provider_id=%s", + user_id, + endpoint_id, + events, + request_id, + subscription_id, + provider_id, + ) + + subscription: TriggerSubscription | None = TriggerProviderService.get_subscription_by_id( + tenant_id=tenant_id, + subscription_id=subscription_id, + ) + if not subscription: + logger.error("Subscription not found: %s", subscription_id) + return {"status": "failed", "error": "Subscription not found"} + + workflow_dispatched = dispatch_triggered_workflows( + user_id=user_id, + events=events, + subscription=subscription, + request_id=request_id, + ) + + debug_dispatched = dispatch_trigger_debug_event( + events=events, + user_id=user_id, + timestamp=timestamp, + request_id=request_id, + subscription=subscription, + ) + + return { + "status": "completed", + "total_count": len(events), + "workflows": workflow_dispatched, + "debug_events": debug_dispatched, + } + + except Exception as e: + logger.exception( + "Error in async trigger dispatching for endpoint %s data %s for subscription %s and provider %s", + endpoint_id, + dispatch_data, + subscription_id, + provider_id, + ) + return { + "status": "failed", + "error": str(e), + } diff --git a/api/tasks/trigger_subscription_refresh_tasks.py b/api/tasks/trigger_subscription_refresh_tasks.py new file mode 100644 index 0000000000..11324df881 --- /dev/null +++ b/api/tasks/trigger_subscription_refresh_tasks.py @@ -0,0 +1,115 @@ +import logging +import time +from collections.abc import Mapping +from typing import Any + +from celery import shared_task +from sqlalchemy.orm import Session + +from core.plugin.entities.plugin_daemon import CredentialType +from core.trigger.utils.locks import build_trigger_refresh_lock_key +from extensions.ext_database import db +from extensions.ext_redis import redis_client +from models.trigger import TriggerSubscription +from services.trigger.trigger_provider_service import TriggerProviderService + +logger = logging.getLogger(__name__) + + +def _now_ts() -> int: + return int(time.time()) + + +def _load_subscription(session: Session, tenant_id: str, subscription_id: str) -> TriggerSubscription | None: + return session.query(TriggerSubscription).filter_by(tenant_id=tenant_id, id=subscription_id).first() + + +def _refresh_oauth_if_expired(tenant_id: str, subscription: TriggerSubscription, now: int) -> None: + if ( + subscription.credential_expires_at != -1 + and int(subscription.credential_expires_at) <= now + and CredentialType.of(subscription.credential_type) == CredentialType.OAUTH2 + ): + logger.info( + "Refreshing OAuth token: tenant=%s subscription_id=%s expires_at=%s now=%s", + tenant_id, + subscription.id, + subscription.credential_expires_at, + now, + ) + try: + result: Mapping[str, Any] = TriggerProviderService.refresh_oauth_token( + tenant_id=tenant_id, subscription_id=subscription.id + ) + logger.info( + "OAuth token refreshed: tenant=%s subscription_id=%s result=%s", tenant_id, subscription.id, result + ) + except Exception: + logger.exception("OAuth refresh failed: tenant=%s subscription_id=%s", tenant_id, subscription.id) + + +def _refresh_subscription_if_expired( + tenant_id: str, + subscription: TriggerSubscription, + now: int, +) -> None: + if subscription.expires_at == -1 or int(subscription.expires_at) > now: + logger.debug( + "Subscription not due: tenant=%s subscription_id=%s expires_at=%s now=%s", + tenant_id, + subscription.id, + subscription.expires_at, + now, + ) + return + + try: + result: Mapping[str, Any] = TriggerProviderService.refresh_subscription( + tenant_id=tenant_id, subscription_id=subscription.id, now=now + ) + logger.info( + "Subscription refreshed: tenant=%s subscription_id=%s result=%s", + tenant_id, + subscription.id, + result.get("result"), + ) + except Exception: + logger.exception("Subscription refresh failed: tenant=%s id=%s", tenant_id, subscription.id) + + +@shared_task(queue="trigger_refresh_executor") +def trigger_subscription_refresh(tenant_id: str, subscription_id: str) -> None: + """Refresh a trigger subscription if needed, guarded by a Redis in-flight lock.""" + lock_key: str = build_trigger_refresh_lock_key(tenant_id, subscription_id) + if not redis_client.get(lock_key): + logger.debug("Refresh lock missing, skip: %s", lock_key) + return + + logger.info("Begin subscription refresh: tenant=%s id=%s", tenant_id, subscription_id) + try: + now: int = _now_ts() + with Session(db.engine) as session: + subscription: TriggerSubscription | None = _load_subscription(session, tenant_id, subscription_id) + + if not subscription: + logger.warning("Subscription not found: tenant=%s id=%s", tenant_id, subscription_id) + return + + logger.debug( + "Loaded subscription: tenant=%s id=%s cred_exp=%s sub_exp=%s now=%s", + tenant_id, + subscription.id, + subscription.credential_expires_at, + subscription.expires_at, + now, + ) + + _refresh_oauth_if_expired(tenant_id=tenant_id, subscription=subscription, now=now) + _refresh_subscription_if_expired(tenant_id=tenant_id, subscription=subscription, now=now) + finally: + try: + redis_client.delete(lock_key) + logger.debug("Lock released: %s", lock_key) + except Exception: + # Best-effort lock cleanup + logger.warning("Failed to release lock: %s", lock_key, exc_info=True) diff --git a/api/tasks/workflow_cfs_scheduler/cfs_scheduler.py b/api/tasks/workflow_cfs_scheduler/cfs_scheduler.py new file mode 100644 index 0000000000..218e61f6d9 --- /dev/null +++ b/api/tasks/workflow_cfs_scheduler/cfs_scheduler.py @@ -0,0 +1,32 @@ +from services.workflow.entities import WorkflowScheduleCFSPlanEntity +from services.workflow.scheduler import CFSPlanScheduler, SchedulerCommand +from tasks.workflow_cfs_scheduler.entities import AsyncWorkflowQueue + + +class AsyncWorkflowCFSPlanEntity(WorkflowScheduleCFSPlanEntity): + """ + Trigger workflow CFS plan entity. + """ + + queue: AsyncWorkflowQueue + + +class AsyncWorkflowCFSPlanScheduler(CFSPlanScheduler): + """ + Trigger workflow CFS plan scheduler. + """ + + plan: AsyncWorkflowCFSPlanEntity + + def can_schedule(self) -> SchedulerCommand: + """ + Check if the workflow can be scheduled. + """ + if self.plan.queue in [AsyncWorkflowQueue.PROFESSIONAL_QUEUE, AsyncWorkflowQueue.TEAM_QUEUE]: + """ + permitted all paid users to schedule the workflow any time + """ + return SchedulerCommand.NONE + + # FIXME: avoid the sandbox user's workflow at a running state for ever + return SchedulerCommand.RESOURCE_LIMIT_REACHED diff --git a/api/tasks/workflow_cfs_scheduler/entities.py b/api/tasks/workflow_cfs_scheduler/entities.py new file mode 100644 index 0000000000..6990f6968a --- /dev/null +++ b/api/tasks/workflow_cfs_scheduler/entities.py @@ -0,0 +1,25 @@ +from enum import StrEnum + +from configs import dify_config +from services.workflow.entities import WorkflowScheduleCFSPlanEntity + +# Determine queue names based on edition +if dify_config.EDITION == "CLOUD": + # Cloud edition: separate queues for different tiers + _professional_queue = "workflow_professional" + _team_queue = "workflow_team" + _sandbox_queue = "workflow_sandbox" + AsyncWorkflowSystemStrategy = WorkflowScheduleCFSPlanEntity.Strategy.TimeSlice +else: + # Community edition: single workflow queue (not dataset) + _professional_queue = "workflow" + _team_queue = "workflow" + _sandbox_queue = "workflow" + AsyncWorkflowSystemStrategy = WorkflowScheduleCFSPlanEntity.Strategy.Nop + + +class AsyncWorkflowQueue(StrEnum): + # Define constants + PROFESSIONAL_QUEUE = _professional_queue + TEAM_QUEUE = _team_queue + SANDBOX_QUEUE = _sandbox_queue diff --git a/api/tasks/workflow_schedule_tasks.py b/api/tasks/workflow_schedule_tasks.py new file mode 100644 index 0000000000..f0596a8f4a --- /dev/null +++ b/api/tasks/workflow_schedule_tasks.py @@ -0,0 +1,60 @@ +import logging + +from celery import shared_task +from sqlalchemy.orm import sessionmaker + +from core.workflow.nodes.trigger_schedule.exc import ( + ScheduleExecutionError, + ScheduleNotFoundError, + TenantOwnerNotFoundError, +) +from extensions.ext_database import db +from models.trigger import WorkflowSchedulePlan +from services.async_workflow_service import AsyncWorkflowService +from services.trigger.schedule_service import ScheduleService +from services.workflow.entities import ScheduleTriggerData + +logger = logging.getLogger(__name__) + + +@shared_task(queue="schedule_executor") +def run_schedule_trigger(schedule_id: str) -> None: + """ + Execute a scheduled workflow trigger. + + Note: No retry logic needed as schedules will run again at next interval. + The execution result is tracked via WorkflowTriggerLog. + + Raises: + ScheduleNotFoundError: If schedule doesn't exist + TenantOwnerNotFoundError: If no owner/admin for tenant + ScheduleExecutionError: If workflow trigger fails + """ + session_factory = sessionmaker(bind=db.engine, expire_on_commit=False) + + with session_factory() as session: + schedule = session.get(WorkflowSchedulePlan, schedule_id) + if not schedule: + raise ScheduleNotFoundError(f"Schedule {schedule_id} not found") + + tenant_owner = ScheduleService.get_tenant_owner(session, schedule.tenant_id) + if not tenant_owner: + raise TenantOwnerNotFoundError(f"No owner or admin found for tenant {schedule.tenant_id}") + + try: + # Production dispatch: Trigger the workflow normally + response = AsyncWorkflowService.trigger_workflow_async( + session=session, + user=tenant_owner, + trigger_data=ScheduleTriggerData( + app_id=schedule.app_id, + root_node_id=schedule.node_id, + inputs={}, + tenant_id=schedule.tenant_id, + ), + ) + logger.info("Schedule %s triggered workflow: %s", schedule_id, response.workflow_trigger_log_id) + except Exception as e: + raise ScheduleExecutionError( + f"Failed to trigger workflow for schedule {schedule_id}, app {schedule.app_id}" + ) from e diff --git a/api/tests/fixtures/workflow/iteration_flatten_output_disabled_workflow.yml b/api/tests/fixtures/workflow/iteration_flatten_output_disabled_workflow.yml new file mode 100644 index 0000000000..9cae6385c8 --- /dev/null +++ b/api/tests/fixtures/workflow/iteration_flatten_output_disabled_workflow.yml @@ -0,0 +1,258 @@ +app: + description: 'This workflow tests the iteration node with flatten_output=False. + + + It processes [1, 2, 3], outputs [item, item*2] for each iteration. + + + With flatten_output=False, it should output nested arrays: + + + ``` + + {"output": [[1, 2], [2, 4], [3, 6]]} + + ```' + icon: 🤖 + icon_background: '#FFEAD5' + mode: workflow + name: test_iteration_flatten_disabled + use_icon_as_answer_icon: false +dependencies: [] +kind: app +version: 0.3.1 +workflow: + conversation_variables: [] + environment_variables: [] + features: + file_upload: + enabled: false + opening_statement: '' + retriever_resource: + enabled: true + sensitive_word_avoidance: + enabled: false + speech_to_text: + enabled: false + suggested_questions: [] + suggested_questions_after_answer: + enabled: false + text_to_speech: + enabled: false + graph: + edges: + - data: + isInIteration: false + isInLoop: false + sourceType: start + targetType: code + id: start-source-code-target + source: start_node + sourceHandle: source + target: code_node + targetHandle: target + type: custom + zIndex: 0 + - data: + isInIteration: false + isInLoop: false + sourceType: code + targetType: iteration + id: code-source-iteration-target + source: code_node + sourceHandle: source + target: iteration_node + targetHandle: target + type: custom + zIndex: 0 + - data: + isInIteration: true + isInLoop: false + iteration_id: iteration_node + sourceType: iteration-start + targetType: code + id: iteration-start-source-code-inner-target + source: iteration_nodestart + sourceHandle: source + target: code_inner_node + targetHandle: target + type: custom + zIndex: 1002 + - data: + isInIteration: false + isInLoop: false + sourceType: iteration + targetType: end + id: iteration-source-end-target + source: iteration_node + sourceHandle: source + target: end_node + targetHandle: target + type: custom + zIndex: 0 + nodes: + - data: + desc: '' + selected: false + title: Start + type: start + variables: [] + height: 54 + id: start_node + position: + x: 80 + y: 282 + positionAbsolute: + x: 80 + y: 282 + sourcePosition: right + targetPosition: left + type: custom + width: 244 + - data: + code: "\ndef main() -> dict:\n return {\n \"result\": [1, 2, 3],\n\ + \ }\n" + code_language: python3 + desc: '' + outputs: + result: + children: null + type: array[number] + selected: false + title: Generate Array + type: code + variables: [] + height: 54 + id: code_node + position: + x: 384 + y: 282 + positionAbsolute: + x: 384 + y: 282 + selected: false + sourcePosition: right + targetPosition: left + type: custom + width: 244 + - data: + desc: '' + error_handle_mode: terminated + flatten_output: false + height: 178 + is_parallel: false + iterator_input_type: array[number] + iterator_selector: + - code_node + - result + output_selector: + - code_inner_node + - result + output_type: array[array[number]] + parallel_nums: 10 + selected: false + start_node_id: iteration_nodestart + title: Iteration with Flatten Disabled + type: iteration + width: 388 + height: 178 + id: iteration_node + position: + x: 684 + y: 282 + positionAbsolute: + x: 684 + y: 282 + selected: false + sourcePosition: right + targetPosition: left + type: custom + width: 388 + zIndex: 1 + - data: + desc: '' + isInIteration: true + selected: false + title: '' + type: iteration-start + draggable: false + height: 48 + id: iteration_nodestart + parentId: iteration_node + position: + x: 24 + y: 68 + positionAbsolute: + x: 708 + y: 350 + selectable: false + sourcePosition: right + targetPosition: left + type: custom-iteration-start + width: 44 + zIndex: 1002 + - data: + code: "\ndef main(arg1: int) -> dict:\n return {\n \"result\": [arg1,\ + \ arg1 * 2],\n }\n" + code_language: python3 + desc: '' + isInIteration: true + isInLoop: false + iteration_id: iteration_node + outputs: + result: + children: null + type: array[number] + selected: false + title: Generate Pair + type: code + variables: + - value_selector: + - iteration_node + - item + value_type: number + variable: arg1 + height: 54 + id: code_inner_node + parentId: iteration_node + position: + x: 128 + y: 68 + positionAbsolute: + x: 812 + y: 350 + selected: false + sourcePosition: right + targetPosition: left + type: custom + width: 244 + zIndex: 1002 + - data: + desc: '' + outputs: + - value_selector: + - iteration_node + - output + value_type: array[array[number]] + variable: output + selected: false + title: End + type: end + height: 90 + id: end_node + position: + x: 1132 + y: 282 + positionAbsolute: + x: 1132 + y: 282 + selected: true + sourcePosition: right + targetPosition: left + type: custom + width: 244 + viewport: + x: -476 + y: 3 + zoom: 1 + diff --git a/api/tests/fixtures/workflow/iteration_flatten_output_enabled_workflow.yml b/api/tests/fixtures/workflow/iteration_flatten_output_enabled_workflow.yml new file mode 100644 index 0000000000..0fc76df768 --- /dev/null +++ b/api/tests/fixtures/workflow/iteration_flatten_output_enabled_workflow.yml @@ -0,0 +1,258 @@ +app: + description: 'This workflow tests the iteration node with flatten_output=True. + + + It processes [1, 2, 3], outputs [item, item*2] for each iteration. + + + With flatten_output=True (default), it should output: + + + ``` + + {"output": [1, 2, 2, 4, 3, 6]} + + ```' + icon: 🤖 + icon_background: '#FFEAD5' + mode: workflow + name: test_iteration_flatten_enabled + use_icon_as_answer_icon: false +dependencies: [] +kind: app +version: 0.3.1 +workflow: + conversation_variables: [] + environment_variables: [] + features: + file_upload: + enabled: false + opening_statement: '' + retriever_resource: + enabled: true + sensitive_word_avoidance: + enabled: false + speech_to_text: + enabled: false + suggested_questions: [] + suggested_questions_after_answer: + enabled: false + text_to_speech: + enabled: false + graph: + edges: + - data: + isInIteration: false + isInLoop: false + sourceType: start + targetType: code + id: start-source-code-target + source: start_node + sourceHandle: source + target: code_node + targetHandle: target + type: custom + zIndex: 0 + - data: + isInIteration: false + isInLoop: false + sourceType: code + targetType: iteration + id: code-source-iteration-target + source: code_node + sourceHandle: source + target: iteration_node + targetHandle: target + type: custom + zIndex: 0 + - data: + isInIteration: true + isInLoop: false + iteration_id: iteration_node + sourceType: iteration-start + targetType: code + id: iteration-start-source-code-inner-target + source: iteration_nodestart + sourceHandle: source + target: code_inner_node + targetHandle: target + type: custom + zIndex: 1002 + - data: + isInIteration: false + isInLoop: false + sourceType: iteration + targetType: end + id: iteration-source-end-target + source: iteration_node + sourceHandle: source + target: end_node + targetHandle: target + type: custom + zIndex: 0 + nodes: + - data: + desc: '' + selected: false + title: Start + type: start + variables: [] + height: 54 + id: start_node + position: + x: 80 + y: 282 + positionAbsolute: + x: 80 + y: 282 + sourcePosition: right + targetPosition: left + type: custom + width: 244 + - data: + code: "\ndef main() -> dict:\n return {\n \"result\": [1, 2, 3],\n\ + \ }\n" + code_language: python3 + desc: '' + outputs: + result: + children: null + type: array[number] + selected: false + title: Generate Array + type: code + variables: [] + height: 54 + id: code_node + position: + x: 384 + y: 282 + positionAbsolute: + x: 384 + y: 282 + selected: false + sourcePosition: right + targetPosition: left + type: custom + width: 244 + - data: + desc: '' + error_handle_mode: terminated + flatten_output: true + height: 178 + is_parallel: false + iterator_input_type: array[number] + iterator_selector: + - code_node + - result + output_selector: + - code_inner_node + - result + output_type: array[array[number]] + parallel_nums: 10 + selected: false + start_node_id: iteration_nodestart + title: Iteration with Flatten Enabled + type: iteration + width: 388 + height: 178 + id: iteration_node + position: + x: 684 + y: 282 + positionAbsolute: + x: 684 + y: 282 + selected: false + sourcePosition: right + targetPosition: left + type: custom + width: 388 + zIndex: 1 + - data: + desc: '' + isInIteration: true + selected: false + title: '' + type: iteration-start + draggable: false + height: 48 + id: iteration_nodestart + parentId: iteration_node + position: + x: 24 + y: 68 + positionAbsolute: + x: 708 + y: 350 + selectable: false + sourcePosition: right + targetPosition: left + type: custom-iteration-start + width: 44 + zIndex: 1002 + - data: + code: "\ndef main(arg1: int) -> dict:\n return {\n \"result\": [arg1,\ + \ arg1 * 2],\n }\n" + code_language: python3 + desc: '' + isInIteration: true + isInLoop: false + iteration_id: iteration_node + outputs: + result: + children: null + type: array[number] + selected: false + title: Generate Pair + type: code + variables: + - value_selector: + - iteration_node + - item + value_type: number + variable: arg1 + height: 54 + id: code_inner_node + parentId: iteration_node + position: + x: 128 + y: 68 + positionAbsolute: + x: 812 + y: 350 + selected: false + sourcePosition: right + targetPosition: left + type: custom + width: 244 + zIndex: 1002 + - data: + desc: '' + outputs: + - value_selector: + - iteration_node + - output + value_type: array[number] + variable: output + selected: false + title: End + type: end + height: 90 + id: end_node + position: + x: 1132 + y: 282 + positionAbsolute: + x: 1132 + y: 282 + selected: true + sourcePosition: right + targetPosition: left + type: custom + width: 244 + viewport: + x: -476 + y: 3 + zoom: 1 + diff --git a/api/tests/integration_tests/.env.example b/api/tests/integration_tests/.env.example index 23a0ecf714..e4c534f046 100644 --- a/api/tests/integration_tests/.env.example +++ b/api/tests/integration_tests/.env.example @@ -144,6 +144,9 @@ HTTP_REQUEST_MAX_WRITE_TIMEOUT=600 HTTP_REQUEST_NODE_MAX_BINARY_SIZE=10485760 HTTP_REQUEST_NODE_MAX_TEXT_SIZE=1048576 +# Webhook configuration +WEBHOOK_REQUEST_BODY_MAX_SIZE=10485760 + # Respect X-* headers to redirect clients RESPECT_XFORWARD_HEADERS_ENABLED=false diff --git a/api/tests/integration_tests/vdb/opensearch/test_opensearch.py b/api/tests/integration_tests/vdb/opensearch/test_opensearch.py index 192c995ce5..210dee4c36 100644 --- a/api/tests/integration_tests/vdb/opensearch/test_opensearch.py +++ b/api/tests/integration_tests/vdb/opensearch/test_opensearch.py @@ -182,6 +182,28 @@ class TestOpenSearchVector: assert len(ids) == 1 assert ids[0] == "mock_id" + def test_delete_nonexistent_index(self): + """Test deleting a non-existent index.""" + # Create a vector instance with a non-existent collection name + self.vector._client.indices.exists.return_value = False + + # Should not raise an exception + self.vector.delete() + + # Verify that exists was called but delete was not + self.vector._client.indices.exists.assert_called_once_with(index=self.collection_name.lower()) + self.vector._client.indices.delete.assert_not_called() + + def test_delete_existing_index(self): + """Test deleting an existing index.""" + self.vector._client.indices.exists.return_value = True + + self.vector.delete() + + # Verify both exists and delete were called + self.vector._client.indices.exists.assert_called_once_with(index=self.collection_name.lower()) + self.vector._client.indices.delete.assert_called_once_with(index=self.collection_name.lower()) + @pytest.mark.usefixtures("setup_mock_redis") class TestOpenSearchVectorWithRedis: diff --git a/api/tests/test_containers_integration_tests/core/__init__.py b/api/tests/test_containers_integration_tests/core/__init__.py new file mode 100644 index 0000000000..5860ad0399 --- /dev/null +++ b/api/tests/test_containers_integration_tests/core/__init__.py @@ -0,0 +1 @@ +# Core integration tests package diff --git a/api/tests/test_containers_integration_tests/core/app/__init__.py b/api/tests/test_containers_integration_tests/core/app/__init__.py new file mode 100644 index 0000000000..0822a865b7 --- /dev/null +++ b/api/tests/test_containers_integration_tests/core/app/__init__.py @@ -0,0 +1 @@ +# App integration tests package diff --git a/api/tests/test_containers_integration_tests/core/app/layers/__init__.py b/api/tests/test_containers_integration_tests/core/app/layers/__init__.py new file mode 100644 index 0000000000..90e5229b1a --- /dev/null +++ b/api/tests/test_containers_integration_tests/core/app/layers/__init__.py @@ -0,0 +1 @@ +# Layers integration tests package diff --git a/api/tests/test_containers_integration_tests/core/app/layers/test_pause_state_persist_layer.py b/api/tests/test_containers_integration_tests/core/app/layers/test_pause_state_persist_layer.py new file mode 100644 index 0000000000..bec3517d66 --- /dev/null +++ b/api/tests/test_containers_integration_tests/core/app/layers/test_pause_state_persist_layer.py @@ -0,0 +1,577 @@ +"""Comprehensive TestContainers-based integration tests for PauseStatePersistenceLayer class. + +This test suite covers complete integration scenarios including: +- Real database interactions using containerized PostgreSQL +- Real storage operations using test storage backend +- Complete workflow: event -> state serialization -> database save -> storage save +- Testing with actual WorkflowRunService (not mocked) +- Real Workflow and WorkflowRun instances in database +- Database transactions and rollback behavior +- Actual file upload and retrieval through storage +- Workflow status transitions in database +- Error handling with real database constraints +- Multiple pause events in sequence +- Integration with real ReadOnlyGraphRuntimeState implementations + +These tests use TestContainers to spin up real services for integration testing, +providing more reliable and realistic test scenarios than mocks. +""" + +import json +import uuid +from time import time + +import pytest +from sqlalchemy import Engine, delete, select +from sqlalchemy.orm import Session + +from core.app.app_config.entities import WorkflowUIBasedAppConfig +from core.app.entities.app_invoke_entities import InvokeFrom, WorkflowAppGenerateEntity +from core.app.layers.pause_state_persist_layer import ( + PauseStatePersistenceLayer, + WorkflowResumptionContext, +) +from core.model_runtime.entities.llm_entities import LLMUsage +from core.workflow.entities.pause_reason import SchedulingPause +from core.workflow.enums import WorkflowExecutionStatus +from core.workflow.graph_engine.entities.commands import GraphEngineCommand +from core.workflow.graph_events.graph import GraphRunPausedEvent +from core.workflow.runtime.graph_runtime_state import GraphRuntimeState +from core.workflow.runtime.graph_runtime_state_protocol import ReadOnlyGraphRuntimeState +from core.workflow.runtime.read_only_wrappers import ReadOnlyGraphRuntimeStateWrapper +from core.workflow.runtime.variable_pool import SystemVariable, VariablePool +from extensions.ext_storage import storage +from libs.datetime_utils import naive_utc_now +from models import Account +from models import WorkflowPause as WorkflowPauseModel +from models.model import AppMode, UploadFile +from models.workflow import Workflow, WorkflowRun +from services.file_service import FileService +from services.workflow_run_service import WorkflowRunService + + +class _TestCommandChannelImpl: + """Real implementation of CommandChannel for testing.""" + + def __init__(self): + self._commands: list[GraphEngineCommand] = [] + + def fetch_commands(self) -> list[GraphEngineCommand]: + """Fetch pending commands for this GraphEngine instance.""" + return self._commands.copy() + + def send_command(self, command: GraphEngineCommand) -> None: + """Send a command to be processed by this GraphEngine instance.""" + self._commands.append(command) + + +class TestPauseStatePersistenceLayerTestContainers: + """Comprehensive TestContainers-based integration tests for PauseStatePersistenceLayer class.""" + + @pytest.fixture + def engine(self, db_session_with_containers: Session): + """Get database engine from TestContainers session.""" + bind = db_session_with_containers.get_bind() + assert isinstance(bind, Engine) + return bind + + @pytest.fixture + def file_service(self, engine: Engine): + """Create FileService instance with TestContainers engine.""" + return FileService(engine) + + @pytest.fixture + def workflow_run_service(self, engine: Engine, file_service: FileService): + """Create WorkflowRunService instance with TestContainers engine and FileService.""" + return WorkflowRunService(engine) + + @pytest.fixture(autouse=True) + def setup_test_data(self, db_session_with_containers, file_service, workflow_run_service): + """Set up test data for each test method using TestContainers.""" + # Create test tenant and account + from models.account import Tenant, TenantAccountJoin, TenantAccountRole + + tenant = Tenant( + name="Test Tenant", + status="normal", + ) + db_session_with_containers.add(tenant) + db_session_with_containers.commit() + + account = Account( + email="test@example.com", + name="Test User", + interface_language="en-US", + status="active", + ) + db_session_with_containers.add(account) + db_session_with_containers.commit() + + # Create tenant-account join + tenant_join = TenantAccountJoin( + tenant_id=tenant.id, + account_id=account.id, + role=TenantAccountRole.OWNER, + current=True, + ) + db_session_with_containers.add(tenant_join) + db_session_with_containers.commit() + + # Set test data + self.test_tenant_id = tenant.id + self.test_user_id = account.id + self.test_app_id = str(uuid.uuid4()) + self.test_workflow_id = str(uuid.uuid4()) + self.test_workflow_run_id = str(uuid.uuid4()) + + # Create test workflow + self.test_workflow = Workflow( + id=self.test_workflow_id, + tenant_id=self.test_tenant_id, + app_id=self.test_app_id, + type="workflow", + version="draft", + graph='{"nodes": [], "edges": []}', + features='{"file_upload": {"enabled": false}}', + created_by=self.test_user_id, + created_at=naive_utc_now(), + ) + + # Create test workflow run + self.test_workflow_run = WorkflowRun( + id=self.test_workflow_run_id, + tenant_id=self.test_tenant_id, + app_id=self.test_app_id, + workflow_id=self.test_workflow_id, + type="workflow", + triggered_from="debugging", + version="draft", + status=WorkflowExecutionStatus.RUNNING, + created_by=self.test_user_id, + created_by_role="account", + created_at=naive_utc_now(), + ) + + # Store session and service instances + self.session = db_session_with_containers + self.file_service = file_service + self.workflow_run_service = workflow_run_service + + # Save test data to database + self.session.add(self.test_workflow) + self.session.add(self.test_workflow_run) + self.session.commit() + + yield + + # Cleanup + self._cleanup_test_data() + + def _cleanup_test_data(self): + """Clean up test data after each test method.""" + try: + # Clean up workflow pauses + self.session.execute(delete(WorkflowPauseModel)) + # Clean up upload files + self.session.execute( + delete(UploadFile).where( + UploadFile.tenant_id == self.test_tenant_id, + ) + ) + # Clean up workflow runs + self.session.execute( + delete(WorkflowRun).where( + WorkflowRun.tenant_id == self.test_tenant_id, + WorkflowRun.app_id == self.test_app_id, + ) + ) + # Clean up workflows + self.session.execute( + delete(Workflow).where( + Workflow.tenant_id == self.test_tenant_id, + Workflow.app_id == self.test_app_id, + ) + ) + self.session.commit() + except Exception as e: + self.session.rollback() + raise e + + def _create_graph_runtime_state( + self, + outputs: dict[str, object] | None = None, + total_tokens: int = 0, + node_run_steps: int = 0, + variables: dict[tuple[str, str], object] | None = None, + workflow_run_id: str | None = None, + ) -> ReadOnlyGraphRuntimeState: + """Create a real GraphRuntimeState for testing.""" + start_at = time() + + execution_id = workflow_run_id or getattr(self, "test_workflow_run_id", None) or str(uuid.uuid4()) + + # Create variable pool + variable_pool = VariablePool(system_variables=SystemVariable(workflow_execution_id=execution_id)) + if variables: + for (node_id, var_key), value in variables.items(): + variable_pool.add([node_id, var_key], value) + + # Create LLM usage + llm_usage = LLMUsage.empty_usage() + + # Create graph runtime state + graph_runtime_state = GraphRuntimeState( + variable_pool=variable_pool, + start_at=start_at, + total_tokens=total_tokens, + llm_usage=llm_usage, + outputs=outputs or {}, + node_run_steps=node_run_steps, + ) + + return ReadOnlyGraphRuntimeStateWrapper(graph_runtime_state) + + def _create_generate_entity( + self, + workflow_execution_id: str | None = None, + user_id: str | None = None, + workflow_id: str | None = None, + ) -> WorkflowAppGenerateEntity: + execution_id = workflow_execution_id or getattr(self, "test_workflow_run_id", str(uuid.uuid4())) + wf_id = workflow_id or getattr(self, "test_workflow_id", str(uuid.uuid4())) + tenant_id = getattr(self, "test_tenant_id", "tenant-123") + app_id = getattr(self, "test_app_id", "app-123") + app_config = WorkflowUIBasedAppConfig( + tenant_id=str(tenant_id), + app_id=str(app_id), + app_mode=AppMode.WORKFLOW, + workflow_id=str(wf_id), + ) + return WorkflowAppGenerateEntity( + task_id=str(uuid.uuid4()), + app_config=app_config, + inputs={}, + files=[], + user_id=user_id or getattr(self, "test_user_id", str(uuid.uuid4())), + stream=False, + invoke_from=InvokeFrom.DEBUGGER, + workflow_execution_id=execution_id, + ) + + def _create_pause_state_persistence_layer( + self, + workflow_run: WorkflowRun | None = None, + workflow: Workflow | None = None, + state_owner_user_id: str | None = None, + generate_entity: WorkflowAppGenerateEntity | None = None, + ) -> PauseStatePersistenceLayer: + """Create PauseStatePersistenceLayer with real dependencies.""" + owner_id = state_owner_user_id + if owner_id is None: + if workflow is not None and workflow.created_by: + owner_id = workflow.created_by + elif workflow_run is not None and workflow_run.created_by: + owner_id = workflow_run.created_by + else: + owner_id = getattr(self, "test_user_id", None) + + assert owner_id is not None + owner_id = str(owner_id) + workflow_execution_id = ( + workflow_run.id if workflow_run is not None else getattr(self, "test_workflow_run_id", None) + ) + assert workflow_execution_id is not None + workflow_id = workflow.id if workflow is not None else getattr(self, "test_workflow_id", None) + assert workflow_id is not None + entity_user_id = getattr(self, "test_user_id", owner_id) + entity = generate_entity or self._create_generate_entity( + workflow_execution_id=str(workflow_execution_id), + user_id=entity_user_id, + workflow_id=str(workflow_id), + ) + + return PauseStatePersistenceLayer( + session_factory=self.session.get_bind(), + state_owner_user_id=owner_id, + generate_entity=entity, + ) + + def test_complete_pause_flow_with_real_dependencies(self, db_session_with_containers): + """Test complete pause flow: event -> state serialization -> database save -> storage save.""" + # Arrange + layer = self._create_pause_state_persistence_layer() + + # Create real graph runtime state with test data + test_outputs = {"result": "test_output", "step": "intermediate"} + test_variables = { + ("node1", "var1"): "string_value", + ("node2", "var2"): {"complex": "object"}, + } + graph_runtime_state = self._create_graph_runtime_state( + outputs=test_outputs, + total_tokens=100, + node_run_steps=5, + variables=test_variables, + ) + + command_channel = _TestCommandChannelImpl() + layer.initialize(graph_runtime_state, command_channel) + + # Create pause event + event = GraphRunPausedEvent( + reason=SchedulingPause(message="test pause"), + outputs={"intermediate": "result"}, + ) + + # Act + layer.on_event(event) + + # Assert - Verify pause state was saved to database + self.session.refresh(self.test_workflow_run) + workflow_run = self.session.get(WorkflowRun, self.test_workflow_run_id) + assert workflow_run is not None + assert workflow_run.status == WorkflowExecutionStatus.PAUSED + + # Verify pause state exists in database + pause_model = self.session.scalars( + select(WorkflowPauseModel).where(WorkflowPauseModel.workflow_run_id == workflow_run.id) + ).first() + assert pause_model is not None + assert pause_model.workflow_id == self.test_workflow_id + assert pause_model.workflow_run_id == self.test_workflow_run_id + assert pause_model.state_object_key != "" + assert pause_model.resumed_at is None + + storage_content = storage.load(pause_model.state_object_key).decode() + resumption_context = WorkflowResumptionContext.loads(storage_content) + assert resumption_context.version == "1" + assert resumption_context.serialized_graph_runtime_state == graph_runtime_state.dumps() + expected_state = json.loads(graph_runtime_state.dumps()) + actual_state = json.loads(resumption_context.serialized_graph_runtime_state) + assert actual_state == expected_state + persisted_entity = resumption_context.get_generate_entity() + assert isinstance(persisted_entity, WorkflowAppGenerateEntity) + assert persisted_entity.workflow_execution_id == self.test_workflow_run_id + + def test_state_persistence_and_retrieval(self, db_session_with_containers): + """Test that pause state can be persisted and retrieved correctly.""" + # Arrange + layer = self._create_pause_state_persistence_layer() + + # Create complex test data + complex_outputs = { + "nested": {"key": "value", "number": 42}, + "list": [1, 2, 3, {"nested": "item"}], + "boolean": True, + "null_value": None, + } + complex_variables = { + ("node1", "var1"): "string_value", + ("node2", "var2"): {"complex": "object"}, + ("node3", "var3"): [1, 2, 3], + } + + graph_runtime_state = self._create_graph_runtime_state( + outputs=complex_outputs, + total_tokens=250, + node_run_steps=10, + variables=complex_variables, + ) + + command_channel = _TestCommandChannelImpl() + layer.initialize(graph_runtime_state, command_channel) + + event = GraphRunPausedEvent(reason=SchedulingPause(message="test pause")) + + # Act - Save pause state + layer.on_event(event) + + # Assert - Retrieve and verify + pause_entity = self.workflow_run_service._workflow_run_repo.get_workflow_pause(self.test_workflow_run_id) + assert pause_entity is not None + assert pause_entity.workflow_execution_id == self.test_workflow_run_id + + state_bytes = pause_entity.get_state() + resumption_context = WorkflowResumptionContext.loads(state_bytes.decode()) + retrieved_state = json.loads(resumption_context.serialized_graph_runtime_state) + expected_state = json.loads(graph_runtime_state.dumps()) + + assert retrieved_state == expected_state + assert retrieved_state["outputs"] == complex_outputs + assert retrieved_state["total_tokens"] == 250 + assert retrieved_state["node_run_steps"] == 10 + assert resumption_context.get_generate_entity().workflow_execution_id == self.test_workflow_run_id + + def test_database_transaction_handling(self, db_session_with_containers): + """Test that database transactions are handled correctly.""" + # Arrange + layer = self._create_pause_state_persistence_layer() + graph_runtime_state = self._create_graph_runtime_state( + outputs={"test": "transaction"}, + total_tokens=50, + ) + + command_channel = _TestCommandChannelImpl() + layer.initialize(graph_runtime_state, command_channel) + + event = GraphRunPausedEvent(reason=SchedulingPause(message="test pause")) + + # Act + layer.on_event(event) + + # Assert - Verify data is committed and accessible in new session + with Session(bind=self.session.get_bind(), expire_on_commit=False) as new_session: + workflow_run = new_session.get(WorkflowRun, self.test_workflow_run_id) + assert workflow_run is not None + assert workflow_run.status == WorkflowExecutionStatus.PAUSED + + pause_model = new_session.scalars( + select(WorkflowPauseModel).where(WorkflowPauseModel.workflow_run_id == workflow_run.id) + ).first() + assert pause_model is not None + assert pause_model.workflow_run_id == self.test_workflow_run_id + assert pause_model.resumed_at is None + assert pause_model.state_object_key != "" + + def test_file_storage_integration(self, db_session_with_containers): + """Test integration with file storage system.""" + # Arrange + layer = self._create_pause_state_persistence_layer() + + # Create large state data to test storage + large_outputs = {"data": "x" * 10000} # 10KB of data + graph_runtime_state = self._create_graph_runtime_state( + outputs=large_outputs, + total_tokens=1000, + ) + + command_channel = _TestCommandChannelImpl() + layer.initialize(graph_runtime_state, command_channel) + + event = GraphRunPausedEvent(reason=SchedulingPause(message="test pause")) + + # Act + layer.on_event(event) + + # Assert - Verify file was uploaded to storage + self.session.refresh(self.test_workflow_run) + pause_model = self.session.scalars( + select(WorkflowPauseModel).where(WorkflowPauseModel.workflow_run_id == self.test_workflow_run.id) + ).first() + assert pause_model is not None + assert pause_model.state_object_key != "" + + # Verify content in storage + storage_content = storage.load(pause_model.state_object_key).decode() + resumption_context = WorkflowResumptionContext.loads(storage_content) + assert resumption_context.serialized_graph_runtime_state == graph_runtime_state.dumps() + assert resumption_context.get_generate_entity().workflow_execution_id == self.test_workflow_run_id + + def test_workflow_with_different_creators(self, db_session_with_containers): + """Test pause state with workflows created by different users.""" + # Arrange - Create workflow with different creator + different_user_id = str(uuid.uuid4()) + different_workflow = Workflow( + id=str(uuid.uuid4()), + tenant_id=self.test_tenant_id, + app_id=self.test_app_id, + type="workflow", + version="draft", + graph='{"nodes": [], "edges": []}', + features='{"file_upload": {"enabled": false}}', + created_by=different_user_id, + created_at=naive_utc_now(), + ) + + different_workflow_run = WorkflowRun( + id=str(uuid.uuid4()), + tenant_id=self.test_tenant_id, + app_id=self.test_app_id, + workflow_id=different_workflow.id, + type="workflow", + triggered_from="debugging", + version="draft", + status=WorkflowExecutionStatus.RUNNING, + created_by=self.test_user_id, # Run created by different user + created_by_role="account", + created_at=naive_utc_now(), + ) + + self.session.add(different_workflow) + self.session.add(different_workflow_run) + self.session.commit() + + layer = self._create_pause_state_persistence_layer( + workflow_run=different_workflow_run, + workflow=different_workflow, + ) + + graph_runtime_state = self._create_graph_runtime_state( + outputs={"creator_test": "different_creator"}, + workflow_run_id=different_workflow_run.id, + ) + + command_channel = _TestCommandChannelImpl() + layer.initialize(graph_runtime_state, command_channel) + + event = GraphRunPausedEvent(reason=SchedulingPause(message="test pause")) + + # Act + layer.on_event(event) + + # Assert - Should use workflow creator (not run creator) + self.session.refresh(different_workflow_run) + pause_model = self.session.scalars( + select(WorkflowPauseModel).where(WorkflowPauseModel.workflow_run_id == different_workflow_run.id) + ).first() + assert pause_model is not None + + # Verify the state owner is the workflow creator + pause_entity = self.workflow_run_service._workflow_run_repo.get_workflow_pause(different_workflow_run.id) + assert pause_entity is not None + resumption_context = WorkflowResumptionContext.loads(pause_entity.get_state().decode()) + assert resumption_context.get_generate_entity().workflow_execution_id == different_workflow_run.id + + def test_layer_ignores_non_pause_events(self, db_session_with_containers): + """Test that layer ignores non-pause events.""" + # Arrange + layer = self._create_pause_state_persistence_layer() + graph_runtime_state = self._create_graph_runtime_state() + + command_channel = _TestCommandChannelImpl() + layer.initialize(graph_runtime_state, command_channel) + + # Import other event types + from core.workflow.graph_events.graph import ( + GraphRunFailedEvent, + GraphRunStartedEvent, + GraphRunSucceededEvent, + ) + + # Act - Send non-pause events + layer.on_event(GraphRunStartedEvent()) + layer.on_event(GraphRunSucceededEvent(outputs={"result": "success"})) + layer.on_event(GraphRunFailedEvent(error="test error", exceptions_count=1)) + + # Assert - No pause state should be created + self.session.refresh(self.test_workflow_run) + assert self.test_workflow_run.status == WorkflowExecutionStatus.RUNNING + + pause_states = ( + self.session.query(WorkflowPauseModel) + .filter(WorkflowPauseModel.workflow_run_id == self.test_workflow_run_id) + .all() + ) + assert len(pause_states) == 0 + + def test_layer_requires_initialization(self, db_session_with_containers): + """Test that layer requires proper initialization before handling events.""" + # Arrange + layer = self._create_pause_state_persistence_layer() + # Don't initialize - graph_runtime_state should not be set + + event = GraphRunPausedEvent(reason=SchedulingPause(message="test pause")) + + # Act & Assert - Should raise AttributeError + with pytest.raises(AttributeError): + layer.on_event(event) diff --git a/api/tests/test_containers_integration_tests/core/rag/__init__.py b/api/tests/test_containers_integration_tests/core/rag/__init__.py new file mode 100644 index 0000000000..8b13789179 --- /dev/null +++ b/api/tests/test_containers_integration_tests/core/rag/__init__.py @@ -0,0 +1 @@ + diff --git a/api/tests/test_containers_integration_tests/core/rag/pipeline/__init__.py b/api/tests/test_containers_integration_tests/core/rag/pipeline/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/test_containers_integration_tests/core/rag/pipeline/test_queue_integration.py b/api/tests/test_containers_integration_tests/core/rag/pipeline/test_queue_integration.py new file mode 100644 index 0000000000..cdf390b327 --- /dev/null +++ b/api/tests/test_containers_integration_tests/core/rag/pipeline/test_queue_integration.py @@ -0,0 +1,595 @@ +""" +Integration tests for TenantIsolatedTaskQueue using testcontainers. + +These tests verify the Redis-based task queue functionality with real Redis instances, +testing tenant isolation, task serialization, and queue operations in a realistic environment. +Includes compatibility tests for migrating from legacy string-only queues. + +All tests use generic naming to avoid coupling to specific business implementations. +""" + +import time +from dataclasses import dataclass +from typing import Any +from uuid import uuid4 + +import pytest +from faker import Faker + +from core.rag.pipeline.queue import TaskWrapper, TenantIsolatedTaskQueue +from extensions.ext_redis import redis_client +from models import Account, Tenant, TenantAccountJoin, TenantAccountRole + + +@dataclass +class TestTask: + """Test task data structure for testing complex object serialization.""" + + task_id: str + tenant_id: str + data: dict[str, Any] + metadata: dict[str, Any] + + +class TestTenantIsolatedTaskQueueIntegration: + """Integration tests for TenantIsolatedTaskQueue using testcontainers.""" + + @pytest.fixture + def fake(self): + """Faker instance for generating test data.""" + return Faker() + + @pytest.fixture + def test_tenant_and_account(self, db_session_with_containers, fake): + """Create test tenant and account for testing.""" + # Create account + account = Account( + email=fake.email(), + name=fake.name(), + interface_language="en-US", + status="active", + ) + db_session_with_containers.add(account) + db_session_with_containers.commit() + + # Create tenant + tenant = Tenant( + name=fake.company(), + status="normal", + ) + db_session_with_containers.add(tenant) + db_session_with_containers.commit() + + # Create tenant-account join + join = TenantAccountJoin( + tenant_id=tenant.id, + account_id=account.id, + role=TenantAccountRole.OWNER, + current=True, + ) + db_session_with_containers.add(join) + db_session_with_containers.commit() + + return tenant, account + + @pytest.fixture + def test_queue(self, test_tenant_and_account): + """Create a generic test queue for testing.""" + tenant, _ = test_tenant_and_account + return TenantIsolatedTaskQueue(tenant.id, "test_queue") + + @pytest.fixture + def secondary_queue(self, test_tenant_and_account): + """Create a secondary test queue for testing isolation.""" + tenant, _ = test_tenant_and_account + return TenantIsolatedTaskQueue(tenant.id, "secondary_queue") + + def test_queue_initialization(self, test_tenant_and_account): + """Test queue initialization with correct key generation.""" + tenant, _ = test_tenant_and_account + queue = TenantIsolatedTaskQueue(tenant.id, "test-key") + + assert queue._tenant_id == tenant.id + assert queue._unique_key == "test-key" + assert queue._queue == f"tenant_self_test-key_task_queue:{tenant.id}" + assert queue._task_key == f"tenant_test-key_task:{tenant.id}" + + def test_tenant_isolation(self, test_tenant_and_account, db_session_with_containers, fake): + """Test that different tenants have isolated queues.""" + tenant1, _ = test_tenant_and_account + + # Create second tenant + tenant2 = Tenant( + name=fake.company(), + status="normal", + ) + db_session_with_containers.add(tenant2) + db_session_with_containers.commit() + + queue1 = TenantIsolatedTaskQueue(tenant1.id, "same-key") + queue2 = TenantIsolatedTaskQueue(tenant2.id, "same-key") + + assert queue1._queue != queue2._queue + assert queue1._task_key != queue2._task_key + assert queue1._queue == f"tenant_self_same-key_task_queue:{tenant1.id}" + assert queue2._queue == f"tenant_self_same-key_task_queue:{tenant2.id}" + + def test_key_isolation(self, test_tenant_and_account): + """Test that different keys have isolated queues.""" + tenant, _ = test_tenant_and_account + queue1 = TenantIsolatedTaskQueue(tenant.id, "key1") + queue2 = TenantIsolatedTaskQueue(tenant.id, "key2") + + assert queue1._queue != queue2._queue + assert queue1._task_key != queue2._task_key + assert queue1._queue == f"tenant_self_key1_task_queue:{tenant.id}" + assert queue2._queue == f"tenant_self_key2_task_queue:{tenant.id}" + + def test_task_key_operations(self, test_queue): + """Test task key operations (get, set, delete).""" + # Initially no task key should exist + assert test_queue.get_task_key() is None + + # Set task waiting time with default TTL + test_queue.set_task_waiting_time() + task_key = test_queue.get_task_key() + # Redis returns bytes, convert to string for comparison + assert task_key in (b"1", "1") + + # Set task waiting time with custom TTL + custom_ttl = 30 + test_queue.set_task_waiting_time(custom_ttl) + task_key = test_queue.get_task_key() + assert task_key in (b"1", "1") + + # Delete task key + test_queue.delete_task_key() + assert test_queue.get_task_key() is None + + def test_push_and_pull_string_tasks(self, test_queue): + """Test pushing and pulling string tasks.""" + tasks = ["task1", "task2", "task3"] + + # Push tasks + test_queue.push_tasks(tasks) + + # Pull tasks (FIFO order) + pulled_tasks = test_queue.pull_tasks(3) + + # Should get tasks in FIFO order (lpush + rpop = FIFO) + assert pulled_tasks == ["task1", "task2", "task3"] + + def test_push_and_pull_multiple_tasks(self, test_queue): + """Test pushing and pulling multiple tasks at once.""" + tasks = ["task1", "task2", "task3", "task4", "task5"] + + # Push tasks + test_queue.push_tasks(tasks) + + # Pull multiple tasks + pulled_tasks = test_queue.pull_tasks(3) + assert len(pulled_tasks) == 3 + assert pulled_tasks == ["task1", "task2", "task3"] + + # Pull remaining tasks + remaining_tasks = test_queue.pull_tasks(5) + assert len(remaining_tasks) == 2 + assert remaining_tasks == ["task4", "task5"] + + def test_push_and_pull_complex_objects(self, test_queue, fake): + """Test pushing and pulling complex object tasks.""" + # Create complex task objects as dictionaries (not dataclass instances) + tasks = [ + { + "task_id": str(uuid4()), + "tenant_id": test_queue._tenant_id, + "data": { + "file_id": str(uuid4()), + "content": fake.text(), + "metadata": {"size": fake.random_int(1000, 10000)}, + }, + "metadata": {"created_at": fake.iso8601(), "tags": fake.words(3)}, + }, + { + "task_id": str(uuid4()), + "tenant_id": test_queue._tenant_id, + "data": { + "file_id": str(uuid4()), + "content": "测试中文内容", + "metadata": {"size": fake.random_int(1000, 10000)}, + }, + "metadata": {"created_at": fake.iso8601(), "tags": ["中文", "测试", "emoji🚀"]}, + }, + ] + + # Push complex tasks + test_queue.push_tasks(tasks) + + # Pull tasks + pulled_tasks = test_queue.pull_tasks(2) + assert len(pulled_tasks) == 2 + + # Verify deserialized tasks match original (FIFO order) + for i, pulled_task in enumerate(pulled_tasks): + original_task = tasks[i] # FIFO order + assert isinstance(pulled_task, dict) + assert pulled_task["task_id"] == original_task["task_id"] + assert pulled_task["tenant_id"] == original_task["tenant_id"] + assert pulled_task["data"] == original_task["data"] + assert pulled_task["metadata"] == original_task["metadata"] + + def test_mixed_task_types(self, test_queue, fake): + """Test pushing and pulling mixed string and object tasks.""" + string_task = "simple_string_task" + object_task = { + "task_id": str(uuid4()), + "dataset_id": str(uuid4()), + "document_ids": [str(uuid4()) for _ in range(3)], + } + + tasks = [string_task, object_task, "another_string"] + + # Push mixed tasks + test_queue.push_tasks(tasks) + + # Pull all tasks + pulled_tasks = test_queue.pull_tasks(3) + assert len(pulled_tasks) == 3 + + # Verify types and content + assert pulled_tasks[0] == string_task + assert isinstance(pulled_tasks[1], dict) + assert pulled_tasks[1] == object_task + assert pulled_tasks[2] == "another_string" + + def test_empty_queue_operations(self, test_queue): + """Test operations on empty queue.""" + # Pull from empty queue + tasks = test_queue.pull_tasks(5) + assert tasks == [] + + # Pull zero or negative count + assert test_queue.pull_tasks(0) == [] + assert test_queue.pull_tasks(-1) == [] + + def test_task_ttl_expiration(self, test_queue): + """Test task key TTL expiration.""" + # Set task with short TTL + short_ttl = 2 + test_queue.set_task_waiting_time(short_ttl) + + # Verify task key exists + assert test_queue.get_task_key() == b"1" or test_queue.get_task_key() == "1" + + # Wait for TTL to expire + time.sleep(short_ttl + 1) + + # Verify task key has expired + assert test_queue.get_task_key() is None + + def test_large_task_batch(self, test_queue, fake): + """Test handling large batches of tasks.""" + # Create large batch of tasks + large_batch = [] + for i in range(100): + task = { + "task_id": str(uuid4()), + "index": i, + "data": fake.text(max_nb_chars=100), + "metadata": {"batch_id": str(uuid4())}, + } + large_batch.append(task) + + # Push large batch + test_queue.push_tasks(large_batch) + + # Pull all tasks + pulled_tasks = test_queue.pull_tasks(100) + assert len(pulled_tasks) == 100 + + # Verify all tasks were retrieved correctly (FIFO order) + for i, task in enumerate(pulled_tasks): + assert isinstance(task, dict) + assert task["index"] == i # FIFO order + + def test_queue_operations_isolation(self, test_tenant_and_account, fake): + """Test concurrent operations on different queues.""" + tenant, _ = test_tenant_and_account + + # Create multiple queues for the same tenant + queue1 = TenantIsolatedTaskQueue(tenant.id, "queue1") + queue2 = TenantIsolatedTaskQueue(tenant.id, "queue2") + + # Push tasks to different queues + queue1.push_tasks(["task1_queue1", "task2_queue1"]) + queue2.push_tasks(["task1_queue2", "task2_queue2"]) + + # Verify queues are isolated + tasks1 = queue1.pull_tasks(2) + tasks2 = queue2.pull_tasks(2) + + assert tasks1 == ["task1_queue1", "task2_queue1"] + assert tasks2 == ["task1_queue2", "task2_queue2"] + assert tasks1 != tasks2 + + def test_task_wrapper_serialization_roundtrip(self, test_queue, fake): + """Test TaskWrapper serialization and deserialization roundtrip.""" + # Create complex nested data + complex_data = { + "id": str(uuid4()), + "nested": {"deep": {"value": "test", "numbers": [1, 2, 3, 4, 5], "unicode": "测试中文", "emoji": "🚀"}}, + "metadata": {"created_at": fake.iso8601(), "tags": ["tag1", "tag2", "tag3"]}, + } + + # Create wrapper and serialize + wrapper = TaskWrapper(data=complex_data) + serialized = wrapper.serialize() + + # Verify serialization + assert isinstance(serialized, str) + assert "测试中文" in serialized + assert "🚀" in serialized + + # Deserialize and verify + deserialized_wrapper = TaskWrapper.deserialize(serialized) + assert deserialized_wrapper.data == complex_data + + def test_error_handling_invalid_json(self, test_queue): + """Test error handling for invalid JSON in wrapped tasks.""" + # Manually create invalid JSON task (not a valid TaskWrapper JSON) + invalid_json_task = "invalid json data" + + # Push invalid task directly to Redis + redis_client.lpush(test_queue._queue, invalid_json_task) + + # Pull task - should fall back to string since it's not valid JSON + task = test_queue.pull_tasks(1) + assert task[0] == invalid_json_task + + def test_real_world_batch_processing_scenario(self, test_queue, fake): + """Test realistic batch processing scenario.""" + # Simulate batch processing tasks + batch_tasks = [] + for i in range(3): + task = { + "file_id": str(uuid4()), + "tenant_id": test_queue._tenant_id, + "user_id": str(uuid4()), + "processing_config": { + "model": fake.random_element(["model_a", "model_b", "model_c"]), + "temperature": fake.random.uniform(0.1, 1.0), + "max_tokens": fake.random_int(1000, 4000), + }, + "metadata": { + "source": fake.random_element(["upload", "api", "webhook"]), + "priority": fake.random_element(["low", "normal", "high"]), + }, + } + batch_tasks.append(task) + + # Push tasks + test_queue.push_tasks(batch_tasks) + + # Process tasks in batches + batch_size = 2 + processed_tasks = [] + + while True: + batch = test_queue.pull_tasks(batch_size) + if not batch: + break + + processed_tasks.extend(batch) + + # Verify all tasks were processed + assert len(processed_tasks) == 3 + + # Verify task structure + for task in processed_tasks: + assert isinstance(task, dict) + assert "file_id" in task + assert "tenant_id" in task + assert "processing_config" in task + assert "metadata" in task + assert task["tenant_id"] == test_queue._tenant_id + + +class TestTenantIsolatedTaskQueueCompatibility: + """Compatibility tests for migrating from legacy string-only queues.""" + + @pytest.fixture + def fake(self): + """Faker instance for generating test data.""" + return Faker() + + @pytest.fixture + def test_tenant_and_account(self, db_session_with_containers, fake): + """Create test tenant and account for testing.""" + # Create account + account = Account( + email=fake.email(), + name=fake.name(), + interface_language="en-US", + status="active", + ) + db_session_with_containers.add(account) + db_session_with_containers.commit() + + # Create tenant + tenant = Tenant( + name=fake.company(), + status="normal", + ) + db_session_with_containers.add(tenant) + db_session_with_containers.commit() + + # Create tenant-account join + join = TenantAccountJoin( + tenant_id=tenant.id, + account_id=account.id, + role=TenantAccountRole.OWNER, + current=True, + ) + db_session_with_containers.add(join) + db_session_with_containers.commit() + + return tenant, account + + def test_legacy_string_queue_compatibility(self, test_tenant_and_account, fake): + """ + Test compatibility with legacy queues containing only string data. + + This simulates the scenario where Redis queues already contain string data + from the old architecture, and we need to ensure the new code can read them. + """ + tenant, _ = test_tenant_and_account + queue = TenantIsolatedTaskQueue(tenant.id, "legacy_queue") + + # Simulate legacy string data in Redis queue (using old format) + legacy_strings = ["legacy_task_1", "legacy_task_2", "legacy_task_3", "legacy_task_4", "legacy_task_5"] + + # Manually push legacy strings directly to Redis (simulating old system) + for legacy_string in legacy_strings: + redis_client.lpush(queue._queue, legacy_string) + + # Verify new code can read legacy string data + pulled_tasks = queue.pull_tasks(5) + assert len(pulled_tasks) == 5 + + # Verify all tasks are strings (not wrapped) + for task in pulled_tasks: + assert isinstance(task, str) + assert task.startswith("legacy_task_") + + # Verify order (FIFO from Redis list) + expected_order = ["legacy_task_1", "legacy_task_2", "legacy_task_3", "legacy_task_4", "legacy_task_5"] + assert pulled_tasks == expected_order + + def test_legacy_queue_migration_scenario(self, test_tenant_and_account, fake): + """ + Test complete migration scenario from legacy to new system. + + This simulates the real-world scenario where: + 1. Legacy system has string data in Redis + 2. New system starts processing the same queue + 3. Both legacy and new tasks coexist during migration + 4. New system can handle both formats seamlessly + """ + tenant, _ = test_tenant_and_account + queue = TenantIsolatedTaskQueue(tenant.id, "migration_queue") + + # Phase 1: Legacy system has data + legacy_tasks = [f"legacy_resource_{i}" for i in range(1, 6)] + redis_client.lpush(queue._queue, *legacy_tasks) + + # Phase 2: New system starts processing legacy data + processed_legacy = [] + while True: + tasks = queue.pull_tasks(1) + if not tasks: + break + processed_legacy.extend(tasks) + + # Verify legacy data was processed correctly + assert len(processed_legacy) == 5 + for task in processed_legacy: + assert isinstance(task, str) + assert task.startswith("legacy_resource_") + + # Phase 3: New system adds new tasks (mixed types) + new_string_tasks = ["new_resource_1", "new_resource_2"] + new_object_tasks = [ + { + "resource_id": str(uuid4()), + "tenant_id": tenant.id, + "processing_type": "new_system", + "metadata": {"version": "2.0", "features": ["ai", "ml"]}, + }, + { + "resource_id": str(uuid4()), + "tenant_id": tenant.id, + "processing_type": "new_system", + "metadata": {"version": "2.0", "features": ["ai", "ml"]}, + }, + ] + + # Push new tasks using new system + queue.push_tasks(new_string_tasks) + queue.push_tasks(new_object_tasks) + + # Phase 4: Process all new tasks + processed_new = [] + while True: + tasks = queue.pull_tasks(1) + if not tasks: + break + processed_new.extend(tasks) + + # Verify new tasks were processed correctly + assert len(processed_new) == 4 + + string_tasks = [task for task in processed_new if isinstance(task, str)] + object_tasks = [task for task in processed_new if isinstance(task, dict)] + + assert len(string_tasks) == 2 + assert len(object_tasks) == 2 + + # Verify string tasks + for task in string_tasks: + assert task.startswith("new_resource_") + + # Verify object tasks + for task in object_tasks: + assert isinstance(task, dict) + assert "resource_id" in task + assert "tenant_id" in task + assert task["tenant_id"] == tenant.id + assert task["processing_type"] == "new_system" + + def test_legacy_queue_error_recovery(self, test_tenant_and_account, fake): + """ + Test error recovery when legacy queue contains malformed data. + + This ensures the new system can gracefully handle corrupted or + malformed legacy data without crashing. + """ + tenant, _ = test_tenant_and_account + queue = TenantIsolatedTaskQueue(tenant.id, "error_recovery_queue") + + # Create mix of valid and malformed legacy data + mixed_legacy_data = [ + "valid_legacy_task_1", + "valid_legacy_task_2", + "malformed_data_string", # This should be treated as string + "valid_legacy_task_3", + "invalid_json_not_taskwrapper_format", # This should fall back to string (not valid TaskWrapper JSON) + "valid_legacy_task_4", + ] + + # Manually push mixed data directly to Redis + redis_client.lpush(queue._queue, *mixed_legacy_data) + + # Process all tasks + processed_tasks = [] + while True: + tasks = queue.pull_tasks(1) + if not tasks: + break + processed_tasks.extend(tasks) + + # Verify all tasks were processed (no crashes) + assert len(processed_tasks) == 6 + + # Verify all tasks are strings (malformed data falls back to string) + for task in processed_tasks: + assert isinstance(task, str) + + # Verify valid tasks are preserved + valid_tasks = [task for task in processed_tasks if task.startswith("valid_legacy_task_")] + assert len(valid_tasks) == 4 + + # Verify malformed data is handled gracefully + malformed_tasks = [task for task in processed_tasks if not task.startswith("valid_legacy_task_")] + assert len(malformed_tasks) == 2 + assert "malformed_data_string" in malformed_tasks + assert "invalid_json_not_taskwrapper_format" in malformed_tasks diff --git a/api/tests/test_containers_integration_tests/libs/broadcast_channel/redis/test_channel.py b/api/tests/test_containers_integration_tests/libs/broadcast_channel/redis/test_channel.py new file mode 100644 index 0000000000..c2e17328d6 --- /dev/null +++ b/api/tests/test_containers_integration_tests/libs/broadcast_channel/redis/test_channel.py @@ -0,0 +1,311 @@ +""" +Integration tests for Redis broadcast channel implementation using TestContainers. + +This test suite covers real Redis interactions including: +- Multiple producer/consumer scenarios +- Network failure scenarios +- Performance under load +- Real-world usage patterns +""" + +import threading +import time +import uuid +from collections.abc import Iterator +from concurrent.futures import ThreadPoolExecutor, as_completed + +import pytest +import redis +from testcontainers.redis import RedisContainer + +from libs.broadcast_channel.channel import BroadcastChannel, Subscription, Topic +from libs.broadcast_channel.exc import SubscriptionClosedError +from libs.broadcast_channel.redis.channel import BroadcastChannel as RedisBroadcastChannel + + +class TestRedisBroadcastChannelIntegration: + """Integration tests for Redis broadcast channel with real Redis instance.""" + + @pytest.fixture(scope="class") + def redis_container(self) -> Iterator[RedisContainer]: + """Create a Redis container for integration testing.""" + with RedisContainer(image="redis:6-alpine") as container: + yield container + + @pytest.fixture(scope="class") + def redis_client(self, redis_container: RedisContainer) -> redis.Redis: + """Create a Redis client connected to the test container.""" + host = redis_container.get_container_host_ip() + port = redis_container.get_exposed_port(6379) + return redis.Redis(host=host, port=port, decode_responses=False) + + @pytest.fixture + def broadcast_channel(self, redis_client: redis.Redis) -> BroadcastChannel: + """Create a BroadcastChannel instance with real Redis client.""" + return RedisBroadcastChannel(redis_client) + + @classmethod + def _get_test_topic_name(cls): + return f"test_topic_{uuid.uuid4()}" + + # ==================== Basic Functionality Tests ====================' + + def test_close_an_active_subscription_should_stop_iteration(self, broadcast_channel): + topic_name = self._get_test_topic_name() + topic = broadcast_channel.topic(topic_name) + subscription = topic.subscribe() + consuming_event = threading.Event() + + def consume(): + msgs = [] + consuming_event.set() + for msg in subscription: + msgs.append(msg) + return msgs + + with ThreadPoolExecutor(max_workers=1) as executor: + producer_future = executor.submit(consume) + consuming_event.wait() + subscription.close() + msgs = producer_future.result(timeout=1) + assert msgs == [] + + def test_end_to_end_messaging(self, broadcast_channel: BroadcastChannel): + """Test complete end-to-end messaging flow.""" + topic_name = "test-topic" + message = b"hello world" + + # Create producer and subscriber + topic = broadcast_channel.topic(topic_name) + producer = topic.as_producer() + subscription = topic.subscribe() + + # Publish and receive message + + def producer_thread(): + time.sleep(0.1) # Small delay to ensure subscriber is ready + producer.publish(message) + time.sleep(0.1) + subscription.close() + + def consumer_thread() -> list[bytes]: + received_messages = [] + for msg in subscription: + received_messages.append(msg) + return received_messages + + # Run producer and consumer + with ThreadPoolExecutor(max_workers=2) as executor: + producer_future = executor.submit(producer_thread) + consumer_future = executor.submit(consumer_thread) + + # Wait for completion + producer_future.result(timeout=5.0) + received_messages = consumer_future.result(timeout=5.0) + + assert len(received_messages) == 1 + assert received_messages[0] == message + + def test_multiple_subscribers_same_topic(self, broadcast_channel: BroadcastChannel): + """Test message broadcasting to multiple subscribers.""" + topic_name = "broadcast-topic" + message = b"broadcast message" + subscriber_count = 5 + + # Create producer and multiple subscribers + topic = broadcast_channel.topic(topic_name) + producer = topic.as_producer() + subscriptions = [topic.subscribe() for _ in range(subscriber_count)] + + def producer_thread(): + time.sleep(0.2) # Allow all subscribers to connect + producer.publish(message) + time.sleep(0.2) + for sub in subscriptions: + sub.close() + + def consumer_thread(subscription: Subscription) -> list[bytes]: + received_msgs = [] + while True: + try: + msg = subscription.receive(0.1) + except SubscriptionClosedError: + break + if msg is None: + continue + received_msgs.append(msg) + if len(received_msgs) >= 1: + break + return received_msgs + + # Run producer and consumers + with ThreadPoolExecutor(max_workers=subscriber_count + 1) as executor: + producer_future = executor.submit(producer_thread) + consumer_futures = [executor.submit(consumer_thread, subscription) for subscription in subscriptions] + + # Wait for completion + producer_future.result(timeout=10.0) + msgs_by_consumers = [] + for future in as_completed(consumer_futures, timeout=10.0): + msgs_by_consumers.append(future.result()) + + # Close all subscriptions + for subscription in subscriptions: + subscription.close() + + # Verify all subscribers received the message + for msgs in msgs_by_consumers: + assert len(msgs) == 1 + assert msgs[0] == message + + def test_topic_isolation(self, broadcast_channel: BroadcastChannel): + """Test that different topics are isolated from each other.""" + topic1_name = "topic1" + topic2_name = "topic2" + message1 = b"message for topic1" + message2 = b"message for topic2" + + # Create producers and subscribers for different topics + topic1 = broadcast_channel.topic(topic1_name) + topic2 = broadcast_channel.topic(topic2_name) + + def producer_thread(): + time.sleep(0.1) + topic1.publish(message1) + topic2.publish(message2) + + def consumer_by_thread(topic: Topic) -> list[bytes]: + subscription = topic.subscribe() + received = [] + with subscription: + for msg in subscription: + received.append(msg) + if len(received) >= 1: + break + return received + + # Run all threads + with ThreadPoolExecutor(max_workers=3) as executor: + producer_future = executor.submit(producer_thread) + consumer1_future = executor.submit(consumer_by_thread, topic1) + consumer2_future = executor.submit(consumer_by_thread, topic2) + + # Wait for completion + producer_future.result(timeout=5.0) + received_by_topic1 = consumer1_future.result(timeout=5.0) + received_by_topic2 = consumer2_future.result(timeout=5.0) + + # Verify topic isolation + assert len(received_by_topic1) == 1 + assert len(received_by_topic2) == 1 + assert received_by_topic1[0] == message1 + assert received_by_topic2[0] == message2 + + # ==================== Performance Tests ==================== + + def test_concurrent_producers(self, broadcast_channel: BroadcastChannel): + """Test multiple producers publishing to the same topic.""" + topic_name = "concurrent-producers-topic" + producer_count = 5 + messages_per_producer = 5 + + topic = broadcast_channel.topic(topic_name) + subscription = topic.subscribe() + + expected_total = producer_count * messages_per_producer + consumer_ready = threading.Event() + + def producer_thread(producer_idx: int) -> set[bytes]: + producer = topic.as_producer() + produced = set() + for i in range(messages_per_producer): + message = f"producer_{producer_idx}_msg_{i}".encode() + produced.add(message) + producer.publish(message) + time.sleep(0.001) # Small delay to avoid overwhelming + return produced + + def consumer_thread() -> set[bytes]: + received_msgs: set[bytes] = set() + with subscription: + consumer_ready.set() + while True: + try: + msg = subscription.receive(timeout=0.1) + except SubscriptionClosedError: + break + if msg is None: + if len(received_msgs) >= expected_total: + break + else: + continue + + received_msgs.add(msg) + return received_msgs + + # Run producers and consumer + with ThreadPoolExecutor(max_workers=producer_count + 1) as executor: + consumer_future = executor.submit(consumer_thread) + consumer_ready.wait() + producer_futures = [executor.submit(producer_thread, i) for i in range(producer_count)] + + sent_msgs: set[bytes] = set() + # Wait for completion + for future in as_completed(producer_futures, timeout=30.0): + sent_msgs.update(future.result()) + + subscription.close() + consumer_received_msgs = consumer_future.result(timeout=30.0) + + # Verify message content + assert sent_msgs == consumer_received_msgs + + # ==================== Resource Management Tests ==================== + + def test_subscription_cleanup(self, broadcast_channel: BroadcastChannel, redis_client: redis.Redis): + """Test proper cleanup of subscription resources.""" + topic_name = "cleanup-test-topic" + + # Create multiple subscriptions + topic = broadcast_channel.topic(topic_name) + + def _consume(sub: Subscription): + for i in sub: + pass + + subscriptions = [] + for i in range(5): + subscription = topic.subscribe() + subscriptions.append(subscription) + + # Start all subscriptions + thread = threading.Thread(target=_consume, args=(subscription,)) + thread.start() + time.sleep(0.01) + + # Verify subscriptions are active + pubsub_info = redis_client.pubsub_numsub(topic_name) + # pubsub_numsub returns list of tuples, find our topic + topic_subscribers = 0 + for channel, count in pubsub_info: + # the channel name returned by redis is bytes. + if channel == topic_name.encode(): + topic_subscribers = count + break + assert topic_subscribers >= 5 + + # Close all subscriptions + for subscription in subscriptions: + subscription.close() + + # Wait a bit for cleanup + time.sleep(1) + + # Verify subscriptions are cleaned up + pubsub_info_after = redis_client.pubsub_numsub(topic_name) + topic_subscribers_after = 0 + for channel, count in pubsub_info_after: + if channel == topic_name.encode(): + topic_subscribers_after = count + break + assert topic_subscribers_after == 0 diff --git a/api/tests/test_containers_integration_tests/services/test_app_generate_service.py b/api/tests/test_containers_integration_tests/services/test_app_generate_service.py index 9386687a04..8b8739d557 100644 --- a/api/tests/test_containers_integration_tests/services/test_app_generate_service.py +++ b/api/tests/test_containers_integration_tests/services/test_app_generate_service.py @@ -5,6 +5,7 @@ import pytest from faker import Faker from core.app.entities.app_invoke_entities import InvokeFrom +from enums.cloud_plan import CloudPlan from models.model import EndUser from models.workflow import Workflow from services.app_generate_service import AppGenerateService @@ -32,7 +33,7 @@ class TestAppGenerateService: patch("services.app_generate_service.dify_config") as mock_dify_config, ): # Setup default mock returns for billing service - mock_billing_service.get_info.return_value = {"subscription": {"plan": "sandbox"}} + mock_billing_service.get_info.return_value = {"subscription": {"plan": CloudPlan.SANDBOX}} # Setup default mock returns for workflow service mock_workflow_service_instance = mock_workflow_service.return_value @@ -430,7 +431,7 @@ class TestAppGenerateService: # Setup billing service mock for sandbox plan mock_external_service_dependencies["billing_service"].get_info.return_value = { - "subscription": {"plan": "sandbox"} + "subscription": {"plan": CloudPlan.SANDBOX} } # Set BILLING_ENABLED to True for this test @@ -461,7 +462,7 @@ class TestAppGenerateService: # Setup billing service mock for sandbox plan mock_external_service_dependencies["billing_service"].get_info.return_value = { - "subscription": {"plan": "sandbox"} + "subscription": {"plan": CloudPlan.SANDBOX} } # Set BILLING_ENABLED to True for this test diff --git a/api/tests/test_containers_integration_tests/services/test_feature_service.py b/api/tests/test_containers_integration_tests/services/test_feature_service.py index 8bd5440411..40380b09d2 100644 --- a/api/tests/test_containers_integration_tests/services/test_feature_service.py +++ b/api/tests/test_containers_integration_tests/services/test_feature_service.py @@ -3,6 +3,7 @@ from unittest.mock import patch import pytest from faker import Faker +from enums.cloud_plan import CloudPlan from services.feature_service import FeatureModel, FeatureService, KnowledgeRateLimitModel, SystemFeatureModel @@ -173,7 +174,7 @@ class TestFeatureService: # Set mock return value inside the patch context mock_external_service_dependencies["billing_service"].get_info.return_value = { "enabled": True, - "subscription": {"plan": "sandbox", "interval": "monthly", "education": False}, + "subscription": {"plan": CloudPlan.SANDBOX, "interval": "monthly", "education": False}, "members": {"size": 1, "limit": 3}, "apps": {"size": 1, "limit": 5}, "vector_space": {"size": 1, "limit": 2}, @@ -189,7 +190,7 @@ class TestFeatureService: result = FeatureService.get_features(tenant_id) # Assert: Verify sandbox-specific limitations - assert result.billing.subscription.plan == "sandbox" + assert result.billing.subscription.plan == CloudPlan.SANDBOX assert result.education.activated is False # Verify sandbox limitations diff --git a/api/tests/test_containers_integration_tests/services/test_file_service.py b/api/tests/test_containers_integration_tests/services/test_file_service.py index 4c94e42f3e..93516a0030 100644 --- a/api/tests/test_containers_integration_tests/services/test_file_service.py +++ b/api/tests/test_containers_integration_tests/services/test_file_service.py @@ -11,7 +11,7 @@ from configs import dify_config from models import Account, Tenant from models.enums import CreatorUserRole from models.model import EndUser, UploadFile -from services.errors.file import FileTooLargeError, UnsupportedFileTypeError +from services.errors.file import BlockedFileExtensionError, FileTooLargeError, UnsupportedFileTypeError from services.file_service import FileService @@ -943,3 +943,150 @@ class TestFileService: # Should have the signed URL when source_url is empty assert upload_file2.source_url == "https://example.com/signed-url" + + # Test file extension blacklist + def test_upload_file_blocked_extension( + self, db_session_with_containers, engine, mock_external_service_dependencies + ): + """ + Test file upload with blocked extension. + """ + fake = Faker() + account = self._create_test_account(db_session_with_containers, mock_external_service_dependencies) + + # Mock blacklist configuration by patching the inner field + with patch.object(dify_config, "inner_UPLOAD_FILE_EXTENSION_BLACKLIST", "exe,bat,sh"): + filename = "malware.exe" + content = b"test content" + mimetype = "application/x-msdownload" + + with pytest.raises(BlockedFileExtensionError): + FileService(engine).upload_file( + filename=filename, + content=content, + mimetype=mimetype, + user=account, + ) + + def test_upload_file_blocked_extension_case_insensitive( + self, db_session_with_containers, engine, mock_external_service_dependencies + ): + """ + Test file upload with blocked extension (case insensitive). + """ + fake = Faker() + account = self._create_test_account(db_session_with_containers, mock_external_service_dependencies) + + # Mock blacklist configuration by patching the inner field + with patch.object(dify_config, "inner_UPLOAD_FILE_EXTENSION_BLACKLIST", "exe,bat"): + # Test with uppercase extension + filename = "malware.EXE" + content = b"test content" + mimetype = "application/x-msdownload" + + with pytest.raises(BlockedFileExtensionError): + FileService(engine).upload_file( + filename=filename, + content=content, + mimetype=mimetype, + user=account, + ) + + def test_upload_file_not_in_blacklist(self, db_session_with_containers, engine, mock_external_service_dependencies): + """ + Test file upload with extension not in blacklist. + """ + fake = Faker() + account = self._create_test_account(db_session_with_containers, mock_external_service_dependencies) + + # Mock blacklist configuration by patching the inner field + with patch.object(dify_config, "inner_UPLOAD_FILE_EXTENSION_BLACKLIST", "exe,bat,sh"): + filename = "document.pdf" + content = b"test content" + mimetype = "application/pdf" + + upload_file = FileService(engine).upload_file( + filename=filename, + content=content, + mimetype=mimetype, + user=account, + ) + + assert upload_file is not None + assert upload_file.name == filename + assert upload_file.extension == "pdf" + + def test_upload_file_empty_blacklist(self, db_session_with_containers, engine, mock_external_service_dependencies): + """ + Test file upload with empty blacklist (default behavior). + """ + fake = Faker() + account = self._create_test_account(db_session_with_containers, mock_external_service_dependencies) + + # Mock empty blacklist configuration by patching the inner field + with patch.object(dify_config, "inner_UPLOAD_FILE_EXTENSION_BLACKLIST", ""): + # Should allow all file types when blacklist is empty + filename = "script.sh" + content = b"#!/bin/bash\necho test" + mimetype = "application/x-sh" + + upload_file = FileService(engine).upload_file( + filename=filename, + content=content, + mimetype=mimetype, + user=account, + ) + + assert upload_file is not None + assert upload_file.extension == "sh" + + def test_upload_file_multiple_blocked_extensions( + self, db_session_with_containers, engine, mock_external_service_dependencies + ): + """ + Test file upload with multiple blocked extensions. + """ + fake = Faker() + account = self._create_test_account(db_session_with_containers, mock_external_service_dependencies) + + # Mock blacklist with multiple extensions by patching the inner field + blacklist_str = "exe,bat,cmd,com,scr,vbs,ps1,msi,dll" + with patch.object(dify_config, "inner_UPLOAD_FILE_EXTENSION_BLACKLIST", blacklist_str): + for ext in blacklist_str.split(","): + filename = f"malware.{ext}" + content = b"test content" + mimetype = "application/octet-stream" + + with pytest.raises(BlockedFileExtensionError): + FileService(engine).upload_file( + filename=filename, + content=content, + mimetype=mimetype, + user=account, + ) + + def test_upload_file_no_extension_with_blacklist( + self, db_session_with_containers, engine, mock_external_service_dependencies + ): + """ + Test file upload with no extension when blacklist is configured. + """ + fake = Faker() + account = self._create_test_account(db_session_with_containers, mock_external_service_dependencies) + + # Mock blacklist configuration by patching the inner field + with patch.object(dify_config, "inner_UPLOAD_FILE_EXTENSION_BLACKLIST", "exe,bat"): + # Files with no extension should not be blocked + filename = "README" + content = b"test content" + mimetype = "text/plain" + + upload_file = FileService(engine).upload_file( + filename=filename, + content=content, + mimetype=mimetype, + user=account, + ) + + assert upload_file is not None + assert upload_file.extension == "" diff --git a/api/tests/test_containers_integration_tests/services/test_webapp_auth_service.py b/api/tests/test_containers_integration_tests/services/test_webapp_auth_service.py index 73e622b061..72b119b4ff 100644 --- a/api/tests/test_containers_integration_tests/services/test_webapp_auth_service.py +++ b/api/tests/test_containers_integration_tests/services/test_webapp_auth_service.py @@ -35,9 +35,7 @@ class TestWebAppAuthService: mock_enterprise_service.WebAppAuth.get_app_access_mode_by_id.return_value = type( "MockWebAppAuth", (), {"access_mode": "private"} )() - mock_enterprise_service.WebAppAuth.get_app_access_mode_by_code.return_value = type( - "MockWebAppAuth", (), {"access_mode": "private"} - )() + # Note: get_app_access_mode_by_code method was removed in refactoring yield { "passport_service": mock_passport_service, diff --git a/api/tests/test_containers_integration_tests/services/test_webhook_service.py b/api/tests/test_containers_integration_tests/services/test_webhook_service.py new file mode 100644 index 0000000000..09a2deb8cc --- /dev/null +++ b/api/tests/test_containers_integration_tests/services/test_webhook_service.py @@ -0,0 +1,569 @@ +import json +from io import BytesIO +from unittest.mock import MagicMock, patch + +import pytest +from faker import Faker +from flask import Flask +from werkzeug.datastructures import FileStorage + +from models.enums import AppTriggerStatus, AppTriggerType +from models.model import App +from models.trigger import AppTrigger, WorkflowWebhookTrigger +from models.workflow import Workflow +from services.account_service import AccountService, TenantService +from services.trigger.webhook_service import WebhookService + + +class TestWebhookService: + """Integration tests for WebhookService using testcontainers.""" + + @pytest.fixture + def mock_external_dependencies(self): + """Mock external service dependencies.""" + with ( + patch("services.trigger.webhook_service.AsyncWorkflowService") as mock_async_service, + patch("services.trigger.webhook_service.ToolFileManager") as mock_tool_file_manager, + patch("services.trigger.webhook_service.file_factory") as mock_file_factory, + patch("services.account_service.FeatureService") as mock_feature_service, + ): + # Mock ToolFileManager + mock_tool_file_instance = MagicMock() + mock_tool_file_manager.return_value = mock_tool_file_instance + + # Mock file creation + mock_tool_file = MagicMock() + mock_tool_file.id = "test_file_id" + mock_tool_file_instance.create_file_by_raw.return_value = mock_tool_file + + # Mock file factory + mock_file_obj = MagicMock() + mock_file_factory.build_from_mapping.return_value = mock_file_obj + + # Mock feature service + mock_feature_service.get_system_features.return_value.is_allow_register = True + mock_feature_service.get_system_features.return_value.is_allow_create_workspace = True + + yield { + "async_service": mock_async_service, + "tool_file_manager": mock_tool_file_manager, + "file_factory": mock_file_factory, + "tool_file": mock_tool_file, + "file_obj": mock_file_obj, + "feature_service": mock_feature_service, + } + + @pytest.fixture + def test_data(self, db_session_with_containers, mock_external_dependencies): + """Create test data for webhook service tests.""" + fake = Faker() + + # Create account and tenant + account = AccountService.create_account( + email=fake.email(), + name=fake.name(), + interface_language="en-US", + password=fake.password(length=12), + ) + TenantService.create_owner_tenant_if_not_exist(account, name=fake.company()) + tenant = account.current_tenant + + # Create app + app = App( + tenant_id=tenant.id, + name=fake.company(), + description=fake.text(), + mode="workflow", + icon="", + icon_background="", + enable_site=True, + enable_api=True, + ) + db_session_with_containers.add(app) + db_session_with_containers.flush() + + # Create workflow + workflow_data = { + "nodes": [ + { + "id": "webhook_node", + "type": "webhook", + "data": { + "title": "Test Webhook", + "method": "post", + "content_type": "application/json", + "headers": [ + {"name": "Authorization", "required": True}, + {"name": "Content-Type", "required": False}, + ], + "params": [{"name": "version", "required": True}, {"name": "format", "required": False}], + "body": [ + {"name": "message", "type": "string", "required": True}, + {"name": "count", "type": "number", "required": False}, + {"name": "upload", "type": "file", "required": False}, + ], + "status_code": 200, + "response_body": '{"status": "success"}', + "timeout": 30, + }, + } + ], + "edges": [], + } + + workflow = Workflow( + tenant_id=tenant.id, + app_id=app.id, + type="workflow", + graph=json.dumps(workflow_data), + features=json.dumps({}), + created_by=account.id, + environment_variables=[], + conversation_variables=[], + version="1.0", + ) + db_session_with_containers.add(workflow) + db_session_with_containers.flush() + + # Create webhook trigger + webhook_id = fake.uuid4()[:16] + webhook_trigger = WorkflowWebhookTrigger( + app_id=app.id, + node_id="webhook_node", + tenant_id=tenant.id, + webhook_id=webhook_id, + created_by=account.id, + ) + db_session_with_containers.add(webhook_trigger) + db_session_with_containers.flush() + + # Create app trigger (required for non-debug mode) + app_trigger = AppTrigger( + tenant_id=tenant.id, + app_id=app.id, + node_id="webhook_node", + trigger_type=AppTriggerType.TRIGGER_WEBHOOK, + title="Test Webhook", + status=AppTriggerStatus.ENABLED, + ) + db_session_with_containers.add(app_trigger) + db_session_with_containers.commit() + + return { + "tenant": tenant, + "account": account, + "app": app, + "workflow": workflow, + "webhook_trigger": webhook_trigger, + "webhook_id": webhook_id, + "app_trigger": app_trigger, + } + + def test_get_webhook_trigger_and_workflow_success(self, test_data, flask_app_with_containers): + """Test successful retrieval of webhook trigger and workflow.""" + webhook_id = test_data["webhook_id"] + + with flask_app_with_containers.app_context(): + webhook_trigger, workflow, node_config = WebhookService.get_webhook_trigger_and_workflow(webhook_id) + + assert webhook_trigger is not None + assert webhook_trigger.webhook_id == webhook_id + assert workflow is not None + assert workflow.app_id == test_data["app"].id + assert node_config is not None + assert node_config["id"] == "webhook_node" + assert node_config["data"]["title"] == "Test Webhook" + + def test_get_webhook_trigger_and_workflow_not_found(self, flask_app_with_containers): + """Test webhook trigger not found scenario.""" + with flask_app_with_containers.app_context(): + with pytest.raises(ValueError, match="Webhook not found"): + WebhookService.get_webhook_trigger_and_workflow("nonexistent_webhook") + + def test_extract_webhook_data_json(self): + """Test webhook data extraction from JSON request.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json", "Authorization": "Bearer token"}, + query_string="version=1&format=json", + json={"message": "hello", "count": 42}, + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["headers"]["Authorization"] == "Bearer token" + assert webhook_data["query_params"]["version"] == "1" + assert webhook_data["query_params"]["format"] == "json" + assert webhook_data["body"]["message"] == "hello" + assert webhook_data["body"]["count"] == 42 + assert webhook_data["files"] == {} + + def test_extract_webhook_data_form_urlencoded(self): + """Test webhook data extraction from form URL encoded request.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/x-www-form-urlencoded"}, + data={"username": "test", "password": "secret"}, + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["body"]["username"] == "test" + assert webhook_data["body"]["password"] == "secret" + + def test_extract_webhook_data_multipart_with_files(self, mock_external_dependencies): + """Test webhook data extraction from multipart form with files.""" + app = Flask(__name__) + + # Create a mock file + file_content = b"test file content" + file_storage = FileStorage(stream=BytesIO(file_content), filename="test.txt", content_type="text/plain") + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "multipart/form-data"}, + data={"message": "test", "upload": file_storage}, + ): + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["body"]["message"] == "test" + assert "upload" in webhook_data["files"] + + # Verify file processing was called + mock_external_dependencies["tool_file_manager"].assert_called_once() + mock_external_dependencies["file_factory"].build_from_mapping.assert_called_once() + + def test_extract_webhook_data_raw_text(self): + """Test webhook data extraction from raw text request.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", method="POST", headers={"Content-Type": "text/plain"}, data="raw text content" + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["body"]["raw"] == "raw text content" + + def test_extract_and_validate_webhook_request_success(self): + """Test successful webhook request validation and type conversion.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json", "Authorization": "Bearer token"}, + query_string="version=1", + json={"message": "hello"}, + ): + webhook_trigger = MagicMock() + node_config = { + "data": { + "method": "post", + "content_type": "application/json", + "headers": [ + {"name": "Authorization", "required": True}, + {"name": "Content-Type", "required": False}, + ], + "params": [{"name": "version", "required": True}], + "body": [{"name": "message", "type": "string", "required": True}], + } + } + + result = WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + assert result["headers"]["Authorization"] == "Bearer token" + assert result["query_params"]["version"] == "1" + assert result["body"]["message"] == "hello" + + def test_extract_and_validate_webhook_request_method_mismatch(self): + """Test webhook validation with HTTP method mismatch.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="GET", + headers={"Content-Type": "application/json"}, + ): + webhook_trigger = MagicMock() + node_config = {"data": {"method": "post", "content_type": "application/json"}} + + with pytest.raises(ValueError, match="HTTP method mismatch"): + WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + def test_extract_and_validate_webhook_request_missing_required_header(self): + """Test webhook validation with missing required header.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json"}, + ): + webhook_trigger = MagicMock() + node_config = { + "data": { + "method": "post", + "content_type": "application/json", + "headers": [{"name": "Authorization", "required": True}], + } + } + + with pytest.raises(ValueError, match="Required header missing: Authorization"): + WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + def test_extract_and_validate_webhook_request_case_insensitive_headers(self): + """Test webhook validation with case-insensitive header matching.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json", "authorization": "Bearer token"}, + json={"message": "hello"}, + ): + webhook_trigger = MagicMock() + node_config = { + "data": { + "method": "post", + "content_type": "application/json", + "headers": [{"name": "Authorization", "required": True}], + "body": [{"name": "message", "type": "string", "required": True}], + } + } + + result = WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + assert result["headers"].get("Authorization") == "Bearer token" + + def test_extract_and_validate_webhook_request_missing_required_param(self): + """Test webhook validation with missing required query parameter.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json"}, + json={"message": "hello"}, + ): + webhook_trigger = MagicMock() + node_config = { + "data": { + "method": "post", + "content_type": "application/json", + "params": [{"name": "version", "required": True}], + "body": [{"name": "message", "type": "string", "required": True}], + } + } + + with pytest.raises(ValueError, match="Required parameter missing: version"): + WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + def test_extract_and_validate_webhook_request_missing_required_body_param(self): + """Test webhook validation with missing required body parameter.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json"}, + json={}, + ): + webhook_trigger = MagicMock() + node_config = { + "data": { + "method": "post", + "content_type": "application/json", + "body": [{"name": "message", "type": "string", "required": True}], + } + } + + with pytest.raises(ValueError, match="Required body parameter missing: message"): + WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + def test_extract_and_validate_webhook_request_missing_required_file(self): + """Test webhook validation when required file is missing from multipart request.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + data={"note": "test"}, + content_type="multipart/form-data", + ): + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "tenant" + webhook_trigger.created_by = "user" + node_config = { + "data": { + "method": "post", + "content_type": "multipart/form-data", + "body": [{"name": "upload", "type": "file", "required": True}], + } + } + + result = WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + assert result["files"] == {} + + def test_trigger_workflow_execution_success(self, test_data, mock_external_dependencies, flask_app_with_containers): + """Test successful workflow execution trigger.""" + webhook_data = { + "method": "POST", + "headers": {"Authorization": "Bearer token"}, + "query_params": {"version": "1"}, + "body": {"message": "hello"}, + "files": {}, + } + + with flask_app_with_containers.app_context(): + # Mock tenant owner lookup to return the test account + with patch("services.trigger.webhook_service.select") as mock_select: + mock_query = MagicMock() + mock_select.return_value.join.return_value.where.return_value = mock_query + + # Mock the session to return our test account + with patch("services.trigger.webhook_service.Session") as mock_session: + mock_session_instance = MagicMock() + mock_session.return_value.__enter__.return_value = mock_session_instance + mock_session_instance.scalar.return_value = test_data["account"] + + # Should not raise any exceptions + WebhookService.trigger_workflow_execution( + test_data["webhook_trigger"], webhook_data, test_data["workflow"] + ) + + # Verify AsyncWorkflowService was called + mock_external_dependencies["async_service"].trigger_workflow_async.assert_called_once() + + def test_trigger_workflow_execution_end_user_service_failure( + self, test_data, mock_external_dependencies, flask_app_with_containers + ): + """Test workflow execution trigger when EndUserService fails.""" + webhook_data = {"method": "POST", "headers": {}, "query_params": {}, "body": {}, "files": {}} + + with flask_app_with_containers.app_context(): + # Mock EndUserService to raise an exception + with patch( + "services.trigger.webhook_service.EndUserService.get_or_create_end_user_by_type" + ) as mock_end_user: + mock_end_user.side_effect = ValueError("Failed to create end user") + + with pytest.raises(ValueError, match="Failed to create end user"): + WebhookService.trigger_workflow_execution( + test_data["webhook_trigger"], webhook_data, test_data["workflow"] + ) + + def test_generate_webhook_response_default(self): + """Test webhook response generation with default values.""" + node_config = {"data": {}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 200 + assert response_data["status"] == "success" + assert "Webhook processed successfully" in response_data["message"] + + def test_generate_webhook_response_custom_json(self): + """Test webhook response generation with custom JSON response.""" + node_config = {"data": {"status_code": 201, "response_body": '{"result": "created", "id": 123}'}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 201 + assert response_data["result"] == "created" + assert response_data["id"] == 123 + + def test_generate_webhook_response_custom_text(self): + """Test webhook response generation with custom text response.""" + node_config = {"data": {"status_code": 202, "response_body": "Request accepted for processing"}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 202 + assert response_data["message"] == "Request accepted for processing" + + def test_generate_webhook_response_invalid_json(self): + """Test webhook response generation with invalid JSON response.""" + node_config = {"data": {"status_code": 400, "response_body": '{"invalid": json}'}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 400 + assert response_data["message"] == '{"invalid": json}' + + def test_process_file_uploads_success(self, mock_external_dependencies): + """Test successful file upload processing.""" + # Create mock files + files = { + "file1": MagicMock(filename="test1.txt", content_type="text/plain"), + "file2": MagicMock(filename="test2.jpg", content_type="image/jpeg"), + } + + # Mock file reads + files["file1"].read.return_value = b"content1" + files["file2"].read.return_value = b"content2" + + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + result = WebhookService._process_file_uploads(files, webhook_trigger) + + assert len(result) == 2 + assert "file1" in result + assert "file2" in result + + # Verify file processing was called for each file + assert mock_external_dependencies["tool_file_manager"].call_count == 2 + assert mock_external_dependencies["file_factory"].build_from_mapping.call_count == 2 + + def test_process_file_uploads_with_errors(self, mock_external_dependencies): + """Test file upload processing with errors.""" + # Create mock files, one will fail + files = { + "good_file": MagicMock(filename="test.txt", content_type="text/plain"), + "bad_file": MagicMock(filename="test.bad", content_type="text/plain"), + } + + files["good_file"].read.return_value = b"content" + files["bad_file"].read.side_effect = Exception("Read error") + + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + result = WebhookService._process_file_uploads(files, webhook_trigger) + + # Should process the good file and skip the bad one + assert len(result) == 1 + assert "good_file" in result + assert "bad_file" not in result + + def test_process_file_uploads_empty_filename(self, mock_external_dependencies): + """Test file upload processing with empty filename.""" + files = { + "no_filename": MagicMock(filename="", content_type="text/plain"), + "none_filename": MagicMock(filename=None, content_type="text/plain"), + } + + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + result = WebhookService._process_file_uploads(files, webhook_trigger) + + # Should skip files without filenames + assert len(result) == 0 + mock_external_dependencies["tool_file_manager"].assert_not_called() diff --git a/api/tests/test_containers_integration_tests/services/test_workflow_service.py b/api/tests/test_containers_integration_tests/services/test_workflow_service.py index 4741eba1f5..88c6313f64 100644 --- a/api/tests/test_containers_integration_tests/services/test_workflow_service.py +++ b/api/tests/test_containers_integration_tests/services/test_workflow_service.py @@ -584,7 +584,16 @@ class TestWorkflowService: account = self._create_test_account(db_session_with_containers, fake) app = self._create_test_app(db_session_with_containers, fake) - graph = {"nodes": [{"id": "start", "type": "start"}], "edges": []} + graph = { + "nodes": [ + { + "id": "start", + "type": "start", + "data": {"type": "start", "title": "Start"}, + } + ], + "edges": [], + } features = {"features": ["feature1", "feature2"]} # Don't pre-calculate hash, let the service generate it unique_hash = None @@ -632,7 +641,25 @@ class TestWorkflowService: # Get the actual hash that was generated original_hash = existing_workflow.unique_hash - new_graph = {"nodes": [{"id": "start", "type": "start"}, {"id": "end", "type": "end"}], "edges": []} + new_graph = { + "nodes": [ + { + "id": "start", + "type": "start", + "data": {"type": "start", "title": "Start"}, + }, + { + "id": "end", + "type": "end", + "data": { + "type": "end", + "title": "End", + "outputs": [{"variable": "output", "value_selector": ["start", "text"]}], + }, + }, + ], + "edges": [], + } new_features = {"features": ["feature1", "feature2", "feature3"]} environment_variables = [] @@ -679,7 +706,16 @@ class TestWorkflowService: # Get the actual hash that was generated original_hash = existing_workflow.unique_hash - new_graph = {"nodes": [{"id": "start", "type": "start"}], "edges": []} + new_graph = { + "nodes": [ + { + "id": "start", + "type": "start", + "data": {"type": "start", "title": "Start"}, + } + ], + "edges": [], + } new_features = {"features": ["feature1"]} # Use a different hash to trigger the error mismatched_hash = "different_hash_12345" diff --git a/api/tests/test_containers_integration_tests/services/tools/test_mcp_tools_manage_service.py b/api/tests/test_containers_integration_tests/services/tools/test_mcp_tools_manage_service.py index 71d55c3ade..8c190762cf 100644 --- a/api/tests/test_containers_integration_tests/services/tools/test_mcp_tools_manage_service.py +++ b/api/tests/test_containers_integration_tests/services/tools/test_mcp_tools_manage_service.py @@ -20,12 +20,21 @@ class TestMCPToolManageService: patch("services.tools.mcp_tools_manage_service.ToolTransformService") as mock_tool_transform_service, ): # Setup default mock returns + from core.tools.entities.api_entities import ToolProviderApiEntity + from core.tools.entities.common_entities import I18nObject + mock_encrypter.encrypt_token.return_value = "encrypted_server_url" - mock_tool_transform_service.mcp_provider_to_user_provider.return_value = { - "id": "test_id", - "name": "test_name", - "type": ToolProviderType.MCP, - } + mock_tool_transform_service.mcp_provider_to_user_provider.return_value = ToolProviderApiEntity( + id="test_id", + author="test_author", + name="test_name", + type=ToolProviderType.MCP, + description=I18nObject(en_US="Test Description", zh_Hans="测试描述"), + icon={"type": "emoji", "content": "🤖"}, + label=I18nObject(en_US="Test Label", zh_Hans="测试标签"), + labels=[], + tools=[], + ) yield { "encrypter": mock_encrypter, @@ -104,9 +113,9 @@ class TestMCPToolManageService: mcp_provider = MCPToolProvider( tenant_id=tenant_id, name=fake.company(), - server_identifier=fake.uuid4(), + server_identifier=str(fake.uuid4()), server_url="encrypted_server_url", - server_url_hash=fake.sha256(), + server_url_hash=str(fake.sha256()), user_id=user_id, authed=False, tools="[]", @@ -144,7 +153,10 @@ class TestMCPToolManageService: ) # Act: Execute the method under test - result = MCPToolManageService.get_mcp_provider_by_provider_id(mcp_provider.id, tenant.id) + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service.get_provider(provider_id=mcp_provider.id, tenant_id=tenant.id) # Assert: Verify the expected outcomes assert result is not None @@ -154,8 +166,6 @@ class TestMCPToolManageService: assert result.user_id == account.id # Verify database state - from extensions.ext_database import db - db.session.refresh(result) assert result.id is not None assert result.server_identifier == mcp_provider.server_identifier @@ -177,11 +187,14 @@ class TestMCPToolManageService: db_session_with_containers, mock_external_service_dependencies ) - non_existent_id = fake.uuid4() + non_existent_id = str(fake.uuid4()) # Act & Assert: Verify proper error handling + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="MCP tool not found"): - MCPToolManageService.get_mcp_provider_by_provider_id(non_existent_id, tenant.id) + service.get_provider(provider_id=non_existent_id, tenant_id=tenant.id) def test_get_mcp_provider_by_provider_id_tenant_isolation( self, db_session_with_containers, mock_external_service_dependencies @@ -210,8 +223,11 @@ class TestMCPToolManageService: ) # Act & Assert: Verify tenant isolation + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="MCP tool not found"): - MCPToolManageService.get_mcp_provider_by_provider_id(mcp_provider1.id, tenant2.id) + service.get_provider(provider_id=mcp_provider1.id, tenant_id=tenant2.id) def test_get_mcp_provider_by_server_identifier_success( self, db_session_with_containers, mock_external_service_dependencies @@ -235,7 +251,10 @@ class TestMCPToolManageService: ) # Act: Execute the method under test - result = MCPToolManageService.get_mcp_provider_by_server_identifier(mcp_provider.server_identifier, tenant.id) + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service.get_provider(server_identifier=mcp_provider.server_identifier, tenant_id=tenant.id) # Assert: Verify the expected outcomes assert result is not None @@ -245,8 +264,6 @@ class TestMCPToolManageService: assert result.user_id == account.id # Verify database state - from extensions.ext_database import db - db.session.refresh(result) assert result.id is not None assert result.name == mcp_provider.name @@ -268,11 +285,14 @@ class TestMCPToolManageService: db_session_with_containers, mock_external_service_dependencies ) - non_existent_identifier = fake.uuid4() + non_existent_identifier = str(fake.uuid4()) # Act & Assert: Verify proper error handling + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="MCP tool not found"): - MCPToolManageService.get_mcp_provider_by_server_identifier(non_existent_identifier, tenant.id) + service.get_provider(server_identifier=non_existent_identifier, tenant_id=tenant.id) def test_get_mcp_provider_by_server_identifier_tenant_isolation( self, db_session_with_containers, mock_external_service_dependencies @@ -301,8 +321,11 @@ class TestMCPToolManageService: ) # Act & Assert: Verify tenant isolation + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="MCP tool not found"): - MCPToolManageService.get_mcp_provider_by_server_identifier(mcp_provider1.server_identifier, tenant2.id) + service.get_provider(server_identifier=mcp_provider1.server_identifier, tenant_id=tenant2.id) def test_create_mcp_provider_success(self, db_session_with_containers, mock_external_service_dependencies): """ @@ -322,15 +345,30 @@ class TestMCPToolManageService: ) # Setup mocks for provider creation + from core.tools.entities.api_entities import ToolProviderApiEntity + from core.tools.entities.common_entities import I18nObject + mock_external_service_dependencies["encrypter"].encrypt_token.return_value = "encrypted_server_url" - mock_external_service_dependencies["tool_transform_service"].mcp_provider_to_user_provider.return_value = { - "id": "new_provider_id", - "name": "Test MCP Provider", - "type": ToolProviderType.MCP, - } + mock_external_service_dependencies[ + "tool_transform_service" + ].mcp_provider_to_user_provider.return_value = ToolProviderApiEntity( + id="new_provider_id", + author=account.name, + name="Test MCP Provider", + type=ToolProviderType.MCP, + description=I18nObject(en_US="Test MCP Provider Description", zh_Hans="测试MCP提供者描述"), + icon={"type": "emoji", "content": "🤖"}, + label=I18nObject(en_US="Test MCP Provider", zh_Hans="测试MCP提供者"), + labels=[], + tools=[], + ) # Act: Execute the method under test - result = MCPToolManageService.create_mcp_provider( + from core.entities.mcp_provider import MCPConfiguration + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service.create_provider( tenant_id=tenant.id, name="Test MCP Provider", server_url="https://example.com/mcp", @@ -339,14 +377,16 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#FF6B6B", server_identifier="test_identifier_123", - timeout=30.0, - sse_read_timeout=300.0, + configuration=MCPConfiguration( + timeout=30.0, + sse_read_timeout=300.0, + ), ) # Assert: Verify the expected outcomes assert result is not None - assert result["name"] == "Test MCP Provider" - assert result["type"] == ToolProviderType.MCP + assert result.name == "Test MCP Provider" + assert result.type == ToolProviderType.MCP # Verify database state from extensions.ext_database import db @@ -386,7 +426,11 @@ class TestMCPToolManageService: ) # Create first provider - MCPToolManageService.create_mcp_provider( + from core.entities.mcp_provider import MCPConfiguration + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + service.create_provider( tenant_id=tenant.id, name="Test MCP Provider", server_url="https://example1.com/mcp", @@ -395,13 +439,15 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#FF6B6B", server_identifier="test_identifier_1", - timeout=30.0, - sse_read_timeout=300.0, + configuration=MCPConfiguration( + timeout=30.0, + sse_read_timeout=300.0, + ), ) # Act & Assert: Verify proper error handling for duplicate name with pytest.raises(ValueError, match="MCP tool Test MCP Provider already exists"): - MCPToolManageService.create_mcp_provider( + service.create_provider( tenant_id=tenant.id, name="Test MCP Provider", # Duplicate name server_url="https://example2.com/mcp", @@ -410,8 +456,10 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#4ECDC4", server_identifier="test_identifier_2", - timeout=45.0, - sse_read_timeout=400.0, + configuration=MCPConfiguration( + timeout=45.0, + sse_read_timeout=400.0, + ), ) def test_create_mcp_provider_duplicate_server_url( @@ -432,7 +480,11 @@ class TestMCPToolManageService: ) # Create first provider - MCPToolManageService.create_mcp_provider( + from core.entities.mcp_provider import MCPConfiguration + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + service.create_provider( tenant_id=tenant.id, name="Test MCP Provider 1", server_url="https://example.com/mcp", @@ -441,13 +493,15 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#FF6B6B", server_identifier="test_identifier_1", - timeout=30.0, - sse_read_timeout=300.0, + configuration=MCPConfiguration( + timeout=30.0, + sse_read_timeout=300.0, + ), ) # Act & Assert: Verify proper error handling for duplicate server URL - with pytest.raises(ValueError, match="MCP tool https://example.com/mcp already exists"): - MCPToolManageService.create_mcp_provider( + with pytest.raises(ValueError, match="MCP tool with this server URL already exists"): + service.create_provider( tenant_id=tenant.id, name="Test MCP Provider 2", server_url="https://example.com/mcp", # Duplicate URL @@ -456,8 +510,10 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#4ECDC4", server_identifier="test_identifier_2", - timeout=45.0, - sse_read_timeout=400.0, + configuration=MCPConfiguration( + timeout=45.0, + sse_read_timeout=400.0, + ), ) def test_create_mcp_provider_duplicate_server_identifier( @@ -478,7 +534,11 @@ class TestMCPToolManageService: ) # Create first provider - MCPToolManageService.create_mcp_provider( + from core.entities.mcp_provider import MCPConfiguration + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + service.create_provider( tenant_id=tenant.id, name="Test MCP Provider 1", server_url="https://example1.com/mcp", @@ -487,13 +547,15 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#FF6B6B", server_identifier="test_identifier_123", - timeout=30.0, - sse_read_timeout=300.0, + configuration=MCPConfiguration( + timeout=30.0, + sse_read_timeout=300.0, + ), ) # Act & Assert: Verify proper error handling for duplicate server identifier with pytest.raises(ValueError, match="MCP tool test_identifier_123 already exists"): - MCPToolManageService.create_mcp_provider( + service.create_provider( tenant_id=tenant.id, name="Test MCP Provider 2", server_url="https://example2.com/mcp", @@ -502,8 +564,10 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#4ECDC4", server_identifier="test_identifier_123", # Duplicate identifier - timeout=45.0, - sse_read_timeout=400.0, + configuration=MCPConfiguration( + timeout=45.0, + sse_read_timeout=400.0, + ), ) def test_retrieve_mcp_tools_success(self, db_session_with_containers, mock_external_service_dependencies): @@ -543,23 +607,59 @@ class TestMCPToolManageService: db.session.commit() # Setup mock for transformation service + from core.tools.entities.api_entities import ToolProviderApiEntity + from core.tools.entities.common_entities import I18nObject + mock_external_service_dependencies["tool_transform_service"].mcp_provider_to_user_provider.side_effect = [ - {"id": provider1.id, "name": provider1.name, "type": ToolProviderType.MCP}, - {"id": provider2.id, "name": provider2.name, "type": ToolProviderType.MCP}, - {"id": provider3.id, "name": provider3.name, "type": ToolProviderType.MCP}, + ToolProviderApiEntity( + id=provider1.id, + author=account.name, + name=provider1.name, + type=ToolProviderType.MCP, + description=I18nObject(en_US="Alpha Provider Description", zh_Hans="Alpha提供者描述"), + icon={"type": "emoji", "content": "🅰️"}, + label=I18nObject(en_US=provider1.name, zh_Hans=provider1.name), + labels=[], + tools=[], + ), + ToolProviderApiEntity( + id=provider2.id, + author=account.name, + name=provider2.name, + type=ToolProviderType.MCP, + description=I18nObject(en_US="Beta Provider Description", zh_Hans="Beta提供者描述"), + icon={"type": "emoji", "content": "🅱️"}, + label=I18nObject(en_US=provider2.name, zh_Hans=provider2.name), + labels=[], + tools=[], + ), + ToolProviderApiEntity( + id=provider3.id, + author=account.name, + name=provider3.name, + type=ToolProviderType.MCP, + description=I18nObject(en_US="Gamma Provider Description", zh_Hans="Gamma提供者描述"), + icon={"type": "emoji", "content": "Γ"}, + label=I18nObject(en_US=provider3.name, zh_Hans=provider3.name), + labels=[], + tools=[], + ), ] # Act: Execute the method under test - result = MCPToolManageService.retrieve_mcp_tools(tenant.id, for_list=True) + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service.list_providers(tenant_id=tenant.id, for_list=True) # Assert: Verify the expected outcomes assert result is not None assert len(result) == 3 # Verify correct ordering by name - assert result[0]["name"] == "Alpha Provider" - assert result[1]["name"] == "Beta Provider" - assert result[2]["name"] == "Gamma Provider" + assert result[0].name == "Alpha Provider" + assert result[1].name == "Beta Provider" + assert result[2].name == "Gamma Provider" # Verify mock interactions assert ( @@ -584,7 +684,10 @@ class TestMCPToolManageService: # No MCP providers created for this tenant # Act: Execute the method under test - result = MCPToolManageService.retrieve_mcp_tools(tenant.id, for_list=False) + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service.list_providers(tenant_id=tenant.id, for_list=False) # Assert: Verify the expected outcomes assert result is not None @@ -624,20 +727,46 @@ class TestMCPToolManageService: ) # Setup mock for transformation service + from core.tools.entities.api_entities import ToolProviderApiEntity + from core.tools.entities.common_entities import I18nObject + mock_external_service_dependencies["tool_transform_service"].mcp_provider_to_user_provider.side_effect = [ - {"id": provider1.id, "name": provider1.name, "type": ToolProviderType.MCP}, - {"id": provider2.id, "name": provider2.name, "type": ToolProviderType.MCP}, + ToolProviderApiEntity( + id=provider1.id, + author=account1.name, + name=provider1.name, + type=ToolProviderType.MCP, + description=I18nObject(en_US="Provider 1 Description", zh_Hans="提供者1描述"), + icon={"type": "emoji", "content": "1️⃣"}, + label=I18nObject(en_US=provider1.name, zh_Hans=provider1.name), + labels=[], + tools=[], + ), + ToolProviderApiEntity( + id=provider2.id, + author=account2.name, + name=provider2.name, + type=ToolProviderType.MCP, + description=I18nObject(en_US="Provider 2 Description", zh_Hans="提供者2描述"), + icon={"type": "emoji", "content": "2️⃣"}, + label=I18nObject(en_US=provider2.name, zh_Hans=provider2.name), + labels=[], + tools=[], + ), ] # Act: Execute the method under test for both tenants - result1 = MCPToolManageService.retrieve_mcp_tools(tenant1.id, for_list=True) - result2 = MCPToolManageService.retrieve_mcp_tools(tenant2.id, for_list=True) + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result1 = service.list_providers(tenant_id=tenant1.id, for_list=True) + result2 = service.list_providers(tenant_id=tenant2.id, for_list=True) # Assert: Verify tenant isolation assert len(result1) == 1 assert len(result2) == 1 - assert result1[0]["id"] == provider1.id - assert result2[0]["id"] == provider2.id + assert result1[0].id == provider1.id + assert result2[0].id == provider2.id def test_list_mcp_tool_from_remote_server_success( self, db_session_with_containers, mock_external_service_dependencies @@ -661,17 +790,20 @@ class TestMCPToolManageService: mcp_provider = self._create_test_mcp_provider( db_session_with_containers, mock_external_service_dependencies, tenant.id, account.id ) - mcp_provider.server_url = "encrypted_server_url" - mcp_provider.authed = False + # Use a valid base64 encoded string to avoid decryption errors + import base64 + + mcp_provider.server_url = base64.b64encode(b"encrypted_server_url").decode() + mcp_provider.authed = True # Provider must be authenticated to list tools mcp_provider.tools = "[]" from extensions.ext_database import db db.session.commit() - # Mock the decrypted_server_url property to avoid encryption issues - with patch("models.tools.encrypter") as mock_encrypter: - mock_encrypter.decrypt_token.return_value = "https://example.com/mcp" + # Mock the decryption process at the rsa level to avoid key file issues + with patch("libs.rsa.decrypt") as mock_decrypt: + mock_decrypt.return_value = "https://example.com/mcp" # Mock MCPClient and its context manager mock_tools = [ @@ -683,13 +815,16 @@ class TestMCPToolManageService: )(), ] - with patch("services.tools.mcp_tools_manage_service.MCPClient") as mock_mcp_client: + with patch("services.tools.mcp_tools_manage_service.MCPClientWithAuthRetry") as mock_mcp_client: # Setup mock client mock_client_instance = mock_mcp_client.return_value.__enter__.return_value mock_client_instance.list_tools.return_value = mock_tools # Act: Execute the method under test - result = MCPToolManageService.list_mcp_tool_from_remote_server(tenant.id, mcp_provider.id) + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service.list_provider_tools(tenant_id=tenant.id, provider_id=mcp_provider.id) # Assert: Verify the expected outcomes assert result is not None @@ -705,16 +840,8 @@ class TestMCPToolManageService: assert mcp_provider.updated_at is not None # Verify mock interactions - mock_mcp_client.assert_called_once_with( - "https://example.com/mcp", - mcp_provider.id, - tenant.id, - authed=False, - for_list=True, - headers={}, - timeout=30.0, - sse_read_timeout=300.0, - ) + # MCPClientWithAuthRetry is called with different parameters + mock_mcp_client.assert_called_once() def test_list_mcp_tool_from_remote_server_auth_error( self, db_session_with_containers, mock_external_service_dependencies @@ -737,7 +864,10 @@ class TestMCPToolManageService: mcp_provider = self._create_test_mcp_provider( db_session_with_containers, mock_external_service_dependencies, tenant.id, account.id ) - mcp_provider.server_url = "encrypted_server_url" + # Use a valid base64 encoded string to avoid decryption errors + import base64 + + mcp_provider.server_url = base64.b64encode(b"encrypted_server_url").decode() mcp_provider.authed = False mcp_provider.tools = "[]" @@ -745,20 +875,23 @@ class TestMCPToolManageService: db.session.commit() - # Mock the decrypted_server_url property to avoid encryption issues - with patch("models.tools.encrypter") as mock_encrypter: - mock_encrypter.decrypt_token.return_value = "https://example.com/mcp" + # Mock the decryption process at the rsa level to avoid key file issues + with patch("libs.rsa.decrypt") as mock_decrypt: + mock_decrypt.return_value = "https://example.com/mcp" # Mock MCPClient to raise authentication error - with patch("services.tools.mcp_tools_manage_service.MCPClient") as mock_mcp_client: + with patch("services.tools.mcp_tools_manage_service.MCPClientWithAuthRetry") as mock_mcp_client: from core.mcp.error import MCPAuthError mock_client_instance = mock_mcp_client.return_value.__enter__.return_value mock_client_instance.list_tools.side_effect = MCPAuthError("Authentication required") # Act & Assert: Verify proper error handling + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="Please auth the tool first"): - MCPToolManageService.list_mcp_tool_from_remote_server(tenant.id, mcp_provider.id) + service.list_provider_tools(tenant_id=tenant.id, provider_id=mcp_provider.id) # Verify database state was not changed db.session.refresh(mcp_provider) @@ -786,32 +919,38 @@ class TestMCPToolManageService: mcp_provider = self._create_test_mcp_provider( db_session_with_containers, mock_external_service_dependencies, tenant.id, account.id ) - mcp_provider.server_url = "encrypted_server_url" - mcp_provider.authed = False + # Use a valid base64 encoded string to avoid decryption errors + import base64 + + mcp_provider.server_url = base64.b64encode(b"encrypted_server_url").decode() + mcp_provider.authed = True # Provider must be authenticated to test connection errors mcp_provider.tools = "[]" from extensions.ext_database import db db.session.commit() - # Mock the decrypted_server_url property to avoid encryption issues - with patch("models.tools.encrypter") as mock_encrypter: - mock_encrypter.decrypt_token.return_value = "https://example.com/mcp" + # Mock the decryption process at the rsa level to avoid key file issues + with patch("libs.rsa.decrypt") as mock_decrypt: + mock_decrypt.return_value = "https://example.com/mcp" # Mock MCPClient to raise connection error - with patch("services.tools.mcp_tools_manage_service.MCPClient") as mock_mcp_client: + with patch("services.tools.mcp_tools_manage_service.MCPClientWithAuthRetry") as mock_mcp_client: from core.mcp.error import MCPError mock_client_instance = mock_mcp_client.return_value.__enter__.return_value mock_client_instance.list_tools.side_effect = MCPError("Connection failed") # Act & Assert: Verify proper error handling + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="Failed to connect to MCP server: Connection failed"): - MCPToolManageService.list_mcp_tool_from_remote_server(tenant.id, mcp_provider.id) + service.list_provider_tools(tenant_id=tenant.id, provider_id=mcp_provider.id) # Verify database state was not changed db.session.refresh(mcp_provider) - assert mcp_provider.authed is False + assert mcp_provider.authed is True # Provider remains authenticated assert mcp_provider.tools == "[]" def test_delete_mcp_tool_success(self, db_session_with_containers, mock_external_service_dependencies): @@ -840,7 +979,8 @@ class TestMCPToolManageService: assert db.session.query(MCPToolProvider).filter_by(id=mcp_provider.id).first() is not None # Act: Execute the method under test - MCPToolManageService.delete_mcp_tool(tenant.id, mcp_provider.id) + service = MCPToolManageService(db.session()) + service.delete_provider(tenant_id=tenant.id, provider_id=mcp_provider.id) # Assert: Verify the expected outcomes # Provider should be deleted from database @@ -862,11 +1002,14 @@ class TestMCPToolManageService: db_session_with_containers, mock_external_service_dependencies ) - non_existent_id = fake.uuid4() + non_existent_id = str(fake.uuid4()) # Act & Assert: Verify proper error handling + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="MCP tool not found"): - MCPToolManageService.delete_mcp_tool(tenant.id, non_existent_id) + service.delete_provider(tenant_id=tenant.id, provider_id=non_existent_id) def test_delete_mcp_tool_tenant_isolation(self, db_session_with_containers, mock_external_service_dependencies): """ @@ -893,8 +1036,11 @@ class TestMCPToolManageService: ) # Act & Assert: Verify tenant isolation + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="MCP tool not found"): - MCPToolManageService.delete_mcp_tool(tenant2.id, mcp_provider1.id) + service.delete_provider(tenant_id=tenant2.id, provider_id=mcp_provider1.id) # Verify provider still exists in tenant1 from extensions.ext_database import db @@ -929,7 +1075,10 @@ class TestMCPToolManageService: db.session.commit() # Act: Execute the method under test - MCPToolManageService.update_mcp_provider( + from core.entities.mcp_provider import MCPConfiguration + + service = MCPToolManageService(db.session()) + service.update_provider( tenant_id=tenant.id, provider_id=mcp_provider.id, name="Updated MCP Provider", @@ -938,8 +1087,10 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#4ECDC4", server_identifier="updated_identifier_123", - timeout=45.0, - sse_read_timeout=400.0, + configuration=MCPConfiguration( + timeout=45.0, + sse_read_timeout=400.0, + ), ) # Assert: Verify the expected outcomes @@ -953,70 +1104,10 @@ class TestMCPToolManageService: # Verify icon was updated import json - icon_data = json.loads(mcp_provider.icon) + icon_data = json.loads(mcp_provider.icon or "{}") assert icon_data["content"] == "🚀" assert icon_data["background"] == "#4ECDC4" - def test_update_mcp_provider_with_server_url_change( - self, db_session_with_containers, mock_external_service_dependencies - ): - """ - Test successful update of MCP provider with server URL change. - - This test verifies: - - Proper handling of server URL changes - - Correct reconnection logic - - Database state updates - - External service integration - """ - # Arrange: Create test data - fake = Faker() - account, tenant = self._create_test_account_and_tenant( - db_session_with_containers, mock_external_service_dependencies - ) - - # Create MCP provider - mcp_provider = self._create_test_mcp_provider( - db_session_with_containers, mock_external_service_dependencies, tenant.id, account.id - ) - - from extensions.ext_database import db - - db.session.commit() - - # Mock the reconnection method - with patch.object(MCPToolManageService, "_re_connect_mcp_provider") as mock_reconnect: - mock_reconnect.return_value = { - "authed": True, - "tools": '[{"name": "test_tool"}]', - "encrypted_credentials": "{}", - } - - # Act: Execute the method under test - MCPToolManageService.update_mcp_provider( - tenant_id=tenant.id, - provider_id=mcp_provider.id, - name="Updated MCP Provider", - server_url="https://new-example.com/mcp", - icon="🚀", - icon_type="emoji", - icon_background="#4ECDC4", - server_identifier="updated_identifier_123", - timeout=45.0, - sse_read_timeout=400.0, - ) - - # Assert: Verify the expected outcomes - db.session.refresh(mcp_provider) - assert mcp_provider.name == "Updated MCP Provider" - assert mcp_provider.server_identifier == "updated_identifier_123" - assert mcp_provider.timeout == 45.0 - assert mcp_provider.sse_read_timeout == 400.0 - assert mcp_provider.updated_at is not None - - # Verify reconnection was called - mock_reconnect.assert_called_once_with("https://new-example.com/mcp", mcp_provider.id, tenant.id) - def test_update_mcp_provider_duplicate_name(self, db_session_with_containers, mock_external_service_dependencies): """ Test error handling when updating MCP provider with duplicate name. @@ -1048,8 +1139,12 @@ class TestMCPToolManageService: db.session.commit() # Act & Assert: Verify proper error handling for duplicate name + from core.entities.mcp_provider import MCPConfiguration + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="MCP tool First Provider already exists"): - MCPToolManageService.update_mcp_provider( + service.update_provider( tenant_id=tenant.id, provider_id=provider2.id, name="First Provider", # Duplicate name @@ -1058,8 +1153,10 @@ class TestMCPToolManageService: icon_type="emoji", icon_background="#4ECDC4", server_identifier="unique_identifier", - timeout=45.0, - sse_read_timeout=400.0, + configuration=MCPConfiguration( + timeout=45.0, + sse_read_timeout=400.0, + ), ) def test_update_mcp_provider_credentials_success( @@ -1094,19 +1191,25 @@ class TestMCPToolManageService: # Mock the provider controller and encryption with ( - patch("services.tools.mcp_tools_manage_service.MCPToolProviderController") as mock_controller, - patch("services.tools.mcp_tools_manage_service.ProviderConfigEncrypter") as mock_encrypter, + patch("core.tools.mcp_tool.provider.MCPToolProviderController") as mock_controller, + patch("core.tools.utils.encryption.ProviderConfigEncrypter") as mock_encrypter, ): # Setup mocks - mock_controller_instance = mock_controller._from_db.return_value + mock_controller_instance = mock_controller.from_db.return_value mock_controller_instance.get_credentials_schema.return_value = [] mock_encrypter_instance = mock_encrypter.return_value mock_encrypter_instance.encrypt.return_value = {"new_key": "encrypted_value"} # Act: Execute the method under test - MCPToolManageService.update_mcp_provider_credentials( - mcp_provider=mcp_provider, credentials={"new_key": "new_value"}, authed=True + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + service.update_provider_credentials( + provider_id=mcp_provider.id, + tenant_id=tenant.id, + credentials={"new_key": "new_value"}, + authed=True, ) # Assert: Verify the expected outcomes @@ -1117,7 +1220,7 @@ class TestMCPToolManageService: # Verify credentials were encrypted and merged import json - credentials = json.loads(mcp_provider.encrypted_credentials) + credentials = json.loads(mcp_provider.encrypted_credentials or "{}") assert "existing_key" in credentials assert "new_key" in credentials @@ -1152,19 +1255,25 @@ class TestMCPToolManageService: # Mock the provider controller and encryption with ( - patch("services.tools.mcp_tools_manage_service.MCPToolProviderController") as mock_controller, - patch("services.tools.mcp_tools_manage_service.ProviderConfigEncrypter") as mock_encrypter, + patch("core.tools.mcp_tool.provider.MCPToolProviderController") as mock_controller, + patch("core.tools.utils.encryption.ProviderConfigEncrypter") as mock_encrypter, ): # Setup mocks - mock_controller_instance = mock_controller._from_db.return_value + mock_controller_instance = mock_controller.from_db.return_value mock_controller_instance.get_credentials_schema.return_value = [] mock_encrypter_instance = mock_encrypter.return_value mock_encrypter_instance.encrypt.return_value = {"new_key": "encrypted_value"} # Act: Execute the method under test - MCPToolManageService.update_mcp_provider_credentials( - mcp_provider=mcp_provider, credentials={"new_key": "new_value"}, authed=False + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + service.update_provider_credentials( + provider_id=mcp_provider.id, + tenant_id=tenant.id, + credentials={"new_key": "new_value"}, + authed=False, ) # Assert: Verify the expected outcomes @@ -1199,41 +1308,37 @@ class TestMCPToolManageService: type("MockTool", (), {"model_dump": lambda self: {"name": "test_tool_2", "description": "Test tool 2"}})(), ] - with patch("services.tools.mcp_tools_manage_service.MCPClient") as mock_mcp_client: + with patch("services.tools.mcp_tools_manage_service.MCPClientWithAuthRetry") as mock_mcp_client: # Setup mock client mock_client_instance = mock_mcp_client.return_value.__enter__.return_value mock_client_instance.list_tools.return_value = mock_tools # Act: Execute the method under test - result = MCPToolManageService._re_connect_mcp_provider( - "https://example.com/mcp", mcp_provider.id, tenant.id + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service._reconnect_provider( + server_url="https://example.com/mcp", + provider=mcp_provider, ) # Assert: Verify the expected outcomes assert result is not None - assert result["authed"] is True - assert result["tools"] is not None - assert result["encrypted_credentials"] == "{}" + assert result.authed is True + assert result.tools is not None + assert result.encrypted_credentials == "{}" # Verify tools were properly serialized import json - tools_data = json.loads(result["tools"]) + tools_data = json.loads(result.tools) assert len(tools_data) == 2 assert tools_data[0]["name"] == "test_tool_1" assert tools_data[1]["name"] == "test_tool_2" # Verify mock interactions - mock_mcp_client.assert_called_once_with( - "https://example.com/mcp", - mcp_provider.id, - tenant.id, - authed=False, - for_list=True, - headers={}, - timeout=30.0, - sse_read_timeout=300.0, - ) + provider_entity = mcp_provider.to_entity() + mock_mcp_client.assert_called_once() def test_re_connect_mcp_provider_auth_error(self, db_session_with_containers, mock_external_service_dependencies): """ @@ -1256,22 +1361,26 @@ class TestMCPToolManageService: ) # Mock MCPClient to raise authentication error - with patch("services.tools.mcp_tools_manage_service.MCPClient") as mock_mcp_client: + with patch("services.tools.mcp_tools_manage_service.MCPClientWithAuthRetry") as mock_mcp_client: from core.mcp.error import MCPAuthError mock_client_instance = mock_mcp_client.return_value.__enter__.return_value mock_client_instance.list_tools.side_effect = MCPAuthError("Authentication required") # Act: Execute the method under test - result = MCPToolManageService._re_connect_mcp_provider( - "https://example.com/mcp", mcp_provider.id, tenant.id + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) + result = service._reconnect_provider( + server_url="https://example.com/mcp", + provider=mcp_provider, ) # Assert: Verify the expected outcomes assert result is not None - assert result["authed"] is False - assert result["tools"] == "[]" - assert result["encrypted_credentials"] == "{}" + assert result.authed is False + assert result.tools == "[]" + assert result.encrypted_credentials == "{}" def test_re_connect_mcp_provider_connection_error( self, db_session_with_containers, mock_external_service_dependencies @@ -1295,12 +1404,18 @@ class TestMCPToolManageService: ) # Mock MCPClient to raise connection error - with patch("services.tools.mcp_tools_manage_service.MCPClient") as mock_mcp_client: + with patch("services.tools.mcp_tools_manage_service.MCPClientWithAuthRetry") as mock_mcp_client: from core.mcp.error import MCPError mock_client_instance = mock_mcp_client.return_value.__enter__.return_value mock_client_instance.list_tools.side_effect = MCPError("Connection failed") # Act & Assert: Verify proper error handling + from extensions.ext_database import db + + service = MCPToolManageService(db.session()) with pytest.raises(ValueError, match="Failed to re-connect MCP server: Connection failed"): - MCPToolManageService._re_connect_mcp_provider("https://example.com/mcp", mcp_provider.id, tenant.id) + service._reconnect_provider( + server_url="https://example.com/mcp", + provider=mcp_provider, + ) diff --git a/api/tests/test_containers_integration_tests/services/tools/test_tools_transform_service.py b/api/tests/test_containers_integration_tests/services/tools/test_tools_transform_service.py index e2c616420f..9b86671954 100644 --- a/api/tests/test_containers_integration_tests/services/tools/test_tools_transform_service.py +++ b/api/tests/test_containers_integration_tests/services/tools/test_tools_transform_service.py @@ -8,6 +8,7 @@ from core.tools.entities.common_entities import I18nObject from core.tools.entities.tool_entities import ToolProviderType from libs.uuid_utils import uuidv7 from models.tools import ApiToolProvider, BuiltinToolProvider, MCPToolProvider, WorkflowToolProvider +from services.plugin.plugin_service import PluginService from services.tools.tools_transform_service import ToolTransformService @@ -17,15 +18,14 @@ class TestToolTransformService: @pytest.fixture def mock_external_service_dependencies(self): """Mock setup for external service dependencies.""" - with ( - patch("services.tools.tools_transform_service.dify_config") as mock_dify_config, - ): - # Setup default mock returns - mock_dify_config.CONSOLE_API_URL = "https://console.example.com" + with patch("services.tools.tools_transform_service.dify_config") as mock_dify_config: + with patch("services.plugin.plugin_service.dify_config", new=mock_dify_config): + # Setup default mock returns + mock_dify_config.CONSOLE_API_URL = "https://console.example.com" - yield { - "dify_config": mock_dify_config, - } + yield { + "dify_config": mock_dify_config, + } def _create_test_tool_provider( self, db_session_with_containers, mock_external_service_dependencies, provider_type="api" @@ -113,13 +113,13 @@ class TestToolTransformService: filename = "test_icon.png" # Act: Execute the method under test - result = ToolTransformService.get_plugin_icon_url(tenant_id, filename) + result = PluginService.get_plugin_icon_url(str(tenant_id), filename) # Assert: Verify the expected outcomes assert result is not None assert isinstance(result, str) assert "console/api/workspaces/current/plugin/icon" in result - assert tenant_id in result + assert str(tenant_id) in result assert filename in result assert result.startswith("https://console.example.com") @@ -144,13 +144,13 @@ class TestToolTransformService: filename = "test_icon.png" # Act: Execute the method under test - result = ToolTransformService.get_plugin_icon_url(tenant_id, filename) + result = PluginService.get_plugin_icon_url(str(tenant_id), filename) # Assert: Verify the expected outcomes assert result is not None assert isinstance(result, str) assert result.startswith("/console/api/workspaces/current/plugin/icon") - assert tenant_id in result + assert str(tenant_id) in result assert filename in result # Verify URL structure @@ -334,7 +334,7 @@ class TestToolTransformService: provider = {"type": ToolProviderType.BUILT_IN, "name": fake.company(), "icon": "🔧"} # Act: Execute the method under test - ToolTransformService.repack_provider(tenant_id, provider) + ToolTransformService.repack_provider(str(tenant_id), provider) # Assert: Verify the expected outcomes assert "icon" in provider @@ -358,7 +358,7 @@ class TestToolTransformService: # Create provider entity with plugin_id provider = ToolProviderApiEntity( - id=fake.uuid4(), + id=str(fake.uuid4()), author=fake.name(), name=fake.company(), description=I18nObject(en_US=fake.text(max_nb_chars=100)), @@ -380,14 +380,14 @@ class TestToolTransformService: assert provider.icon is not None assert isinstance(provider.icon, str) assert "console/api/workspaces/current/plugin/icon" in provider.icon - assert tenant_id in provider.icon + assert str(tenant_id) in provider.icon assert "test_icon.png" in provider.icon # Verify dark icon handling assert provider.icon_dark is not None assert isinstance(provider.icon_dark, str) assert "console/api/workspaces/current/plugin/icon" in provider.icon_dark - assert tenant_id in provider.icon_dark + assert str(tenant_id) in provider.icon_dark assert "test_icon_dark.png" in provider.icon_dark def test_repack_provider_entity_no_plugin_success( @@ -423,7 +423,7 @@ class TestToolTransformService: ) # Act: Execute the method under test - ToolTransformService.repack_provider(tenant_id, provider) + ToolTransformService.repack_provider(str(tenant_id), provider) # Assert: Verify the expected outcomes assert provider.icon is not None @@ -521,7 +521,7 @@ class TestToolTransformService: with patch("services.tools.tools_transform_service.create_provider_encrypter") as mock_encrypter: mock_encrypter_instance = Mock() mock_encrypter_instance.decrypt.return_value = {"api_key": "decrypted_key"} - mock_encrypter_instance.mask_tool_credentials.return_value = {"api_key": ""} + mock_encrypter_instance.mask_plugin_credentials.return_value = {"api_key": ""} mock_encrypter.return_value = (mock_encrypter_instance, None) # Act: Execute the method under test diff --git a/api/tests/test_containers_integration_tests/tasks/test_add_document_to_index_task.py b/api/tests/test_containers_integration_tests/tasks/test_add_document_to_index_task.py index 68e485107c..f1530bcac6 100644 --- a/api/tests/test_containers_integration_tests/tasks/test_add_document_to_index_task.py +++ b/api/tests/test_containers_integration_tests/tasks/test_add_document_to_index_task.py @@ -256,7 +256,7 @@ class TestAddDocumentToIndexTask: """ # Arrange: Use non-existent document ID fake = Faker() - non_existent_id = fake.uuid4() + non_existent_id = str(fake.uuid4()) # Act: Execute the task with non-existent document add_document_to_index_task(non_existent_id) @@ -282,7 +282,7 @@ class TestAddDocumentToIndexTask: - Redis cache key not affected """ # Arrange: Create test data with invalid indexing status - dataset, document = self._create_test_dataset_and_document( + _, document = self._create_test_dataset_and_document( db_session_with_containers, mock_external_service_dependencies ) @@ -417,15 +417,15 @@ class TestAddDocumentToIndexTask: # Verify redis cache was cleared assert redis_client.exists(indexing_cache_key) == 0 - def test_add_document_to_index_with_no_segments_to_process( + def test_add_document_to_index_with_already_enabled_segments( self, db_session_with_containers, mock_external_service_dependencies ): """ - Test document indexing when no segments need processing. + Test document indexing when segments are already enabled. This test verifies: - - Proper handling when all segments are already enabled - - Index processing still occurs but with empty documents list + - Segments with status="completed" are processed regardless of enabled status + - Index processing occurs with all completed segments - Auto disable log deletion still occurs - Redis cache is cleared """ @@ -465,15 +465,16 @@ class TestAddDocumentToIndexTask: # Act: Execute the task add_document_to_index_task(document.id) - # Assert: Verify index processing occurred but with empty documents list + # Assert: Verify index processing occurred with all completed segments mock_external_service_dependencies["index_processor_factory"].assert_called_once_with(IndexType.PARAGRAPH_INDEX) mock_external_service_dependencies["index_processor"].load.assert_called_once() - # Verify the load method was called with empty documents list + # Verify the load method was called with all completed segments + # (implementation doesn't filter by enabled status, only by status="completed") call_args = mock_external_service_dependencies["index_processor"].load.call_args assert call_args is not None documents = call_args[0][1] # Second argument should be documents list - assert len(documents) == 0 # No segments to process + assert len(documents) == 3 # All completed segments are processed # Verify redis cache was cleared assert redis_client.exists(indexing_cache_key) == 0 @@ -499,7 +500,7 @@ class TestAddDocumentToIndexTask: # Create some auto disable log entries fake = Faker() auto_disable_logs = [] - for i in range(2): + for _ in range(2): log_entry = DatasetAutoDisableLog( id=fake.uuid4(), tenant_id=document.tenant_id, @@ -595,9 +596,11 @@ class TestAddDocumentToIndexTask: Test segment filtering with various edge cases. This test verifies: - - Only segments with enabled=False and status="completed" are processed + - Only segments with status="completed" are processed (regardless of enabled status) + - Segments with status!="completed" are NOT processed - Segments are ordered by position correctly - Mixed segment states are handled properly + - All segments are updated to enabled=True after processing - Redis cache key deletion """ # Arrange: Create test data @@ -628,7 +631,8 @@ class TestAddDocumentToIndexTask: db.session.add(segment1) segments.append(segment1) - # Segment 2: Should NOT be processed (enabled=True, status="completed") + # Segment 2: Should be processed (enabled=True, status="completed") + # Note: Implementation doesn't filter by enabled status, only by status="completed" segment2 = DocumentSegment( id=fake.uuid4(), tenant_id=document.tenant_id, @@ -640,7 +644,7 @@ class TestAddDocumentToIndexTask: tokens=len(fake.text(max_nb_chars=200).split()) * 2, index_node_id="node_1", index_node_hash="hash_1", - enabled=True, # Already enabled + enabled=True, # Already enabled, but will still be processed status="completed", created_by=document.created_by, ) @@ -702,11 +706,14 @@ class TestAddDocumentToIndexTask: call_args = mock_external_service_dependencies["index_processor"].load.call_args assert call_args is not None documents = call_args[0][1] # Second argument should be documents list - assert len(documents) == 2 # Only 2 segments should be processed + assert len(documents) == 3 # 3 segments with status="completed" should be processed # Verify correct segments were processed (by position order) - assert documents[0].metadata["doc_id"] == "node_0" # position 0 - assert documents[1].metadata["doc_id"] == "node_3" # position 3 + # Segments 1, 2, 4 should be processed (positions 0, 1, 3) + # Segment 3 is skipped (position 2, status="processing") + assert documents[0].metadata["doc_id"] == "node_0" # segment1, position 0 + assert documents[1].metadata["doc_id"] == "node_1" # segment2, position 1 + assert documents[2].metadata["doc_id"] == "node_3" # segment4, position 3 # Verify database state changes db.session.refresh(document) @@ -717,7 +724,7 @@ class TestAddDocumentToIndexTask: # All segments should be enabled because the task updates ALL segments for the document assert segment1.enabled is True - assert segment2.enabled is True # Was already enabled, now updated to True + assert segment2.enabled is True # Was already enabled, stays True assert segment3.enabled is True # Was not processed but still updated to True assert segment4.enabled is True diff --git a/api/tests/test_containers_integration_tests/tasks/test_document_indexing_task.py b/api/tests/test_containers_integration_tests/tasks/test_document_indexing_task.py index 448f6da5ec..c015d7ec9c 100644 --- a/api/tests/test_containers_integration_tests/tasks/test_document_indexing_task.py +++ b/api/tests/test_containers_integration_tests/tasks/test_document_indexing_task.py @@ -1,16 +1,33 @@ +from dataclasses import asdict from unittest.mock import MagicMock, patch import pytest from faker import Faker +from core.entities.document_task import DocumentTask +from enums.cloud_plan import CloudPlan from extensions.ext_database import db from models import Account, Tenant, TenantAccountJoin, TenantAccountRole from models.dataset import Dataset, Document -from tasks.document_indexing_task import document_indexing_task +from tasks.document_indexing_task import ( + _document_indexing, # Core function + _document_indexing_with_tenant_queue, # Tenant queue wrapper function + document_indexing_task, # Deprecated old interface + normal_document_indexing_task, # New normal task + priority_document_indexing_task, # New priority task +) -class TestDocumentIndexingTask: - """Integration tests for document_indexing_task using testcontainers.""" +class TestDocumentIndexingTasks: + """Integration tests for document indexing tasks using testcontainers. + + This test class covers: + - Core _document_indexing function + - Deprecated document_indexing_task function + - New normal_document_indexing_task function + - New priority_document_indexing_task function + - Tenant queue wrapper _document_indexing_with_tenant_queue function + """ @pytest.fixture def mock_external_service_dependencies(self): @@ -197,7 +214,7 @@ class TestDocumentIndexingTask: # Configure billing features mock_external_service_dependencies["features"].billing.enabled = billing_enabled if billing_enabled: - mock_external_service_dependencies["features"].billing.subscription.plan = "sandbox" + mock_external_service_dependencies["features"].billing.subscription.plan = CloudPlan.SANDBOX mock_external_service_dependencies["features"].vector_space.limit = 100 mock_external_service_dependencies["features"].vector_space.size = 50 @@ -223,7 +240,7 @@ class TestDocumentIndexingTask: document_ids = [doc.id for doc in documents] # Act: Execute the task - document_indexing_task(dataset.id, document_ids) + _document_indexing(dataset.id, document_ids) # Assert: Verify the expected outcomes # Verify indexing runner was called correctly @@ -231,10 +248,11 @@ class TestDocumentIndexingTask: mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() # Verify documents were updated to parsing status - for document in documents: - db.session.refresh(document) - assert document.indexing_status == "parsing" - assert document.processing_started_at is not None + # Re-query documents from database since _document_indexing uses a different session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None # Verify the run method was called with correct documents call_args = mock_external_service_dependencies["indexing_runner_instance"].run.call_args @@ -260,7 +278,7 @@ class TestDocumentIndexingTask: document_ids = [fake.uuid4() for _ in range(3)] # Act: Execute the task with non-existent dataset - document_indexing_task(non_existent_dataset_id, document_ids) + _document_indexing(non_existent_dataset_id, document_ids) # Assert: Verify no processing occurred mock_external_service_dependencies["indexing_runner"].assert_not_called() @@ -290,17 +308,18 @@ class TestDocumentIndexingTask: all_document_ids = existing_document_ids + non_existent_document_ids # Act: Execute the task with mixed document IDs - document_indexing_task(dataset.id, all_document_ids) + _document_indexing(dataset.id, all_document_ids) # Assert: Verify only existing documents were processed mock_external_service_dependencies["indexing_runner"].assert_called_once() mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() # Verify only existing documents were updated - for document in documents: - db.session.refresh(document) - assert document.indexing_status == "parsing" - assert document.processing_started_at is not None + # Re-query documents from database since _document_indexing uses a different session + for doc_id in existing_document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None # Verify the run method was called with only existing documents call_args = mock_external_service_dependencies["indexing_runner_instance"].run.call_args @@ -332,7 +351,7 @@ class TestDocumentIndexingTask: ) # Act: Execute the task - document_indexing_task(dataset.id, document_ids) + _document_indexing(dataset.id, document_ids) # Assert: Verify exception was handled gracefully # The task should complete without raising exceptions @@ -340,10 +359,11 @@ class TestDocumentIndexingTask: mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() # Verify documents were still updated to parsing status before the exception - for document in documents: - db.session.refresh(document) - assert document.indexing_status == "parsing" - assert document.processing_started_at is not None + # Re-query documents from database since _document_indexing close the session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None def test_document_indexing_task_mixed_document_states( self, db_session_with_containers, mock_external_service_dependencies @@ -406,17 +426,18 @@ class TestDocumentIndexingTask: document_ids = [doc.id for doc in all_documents] # Act: Execute the task with mixed document states - document_indexing_task(dataset.id, document_ids) + _document_indexing(dataset.id, document_ids) # Assert: Verify processing mock_external_service_dependencies["indexing_runner"].assert_called_once() mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() # Verify all documents were updated to parsing status - for document in all_documents: - db.session.refresh(document) - assert document.indexing_status == "parsing" - assert document.processing_started_at is not None + # Re-query documents from database since _document_indexing uses a different session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None # Verify the run method was called with all documents call_args = mock_external_service_dependencies["indexing_runner_instance"].run.call_args @@ -442,7 +463,7 @@ class TestDocumentIndexingTask: ) # Configure sandbox plan with batch limit - mock_external_service_dependencies["features"].billing.subscription.plan = "sandbox" + mock_external_service_dependencies["features"].billing.subscription.plan = CloudPlan.SANDBOX # Create more documents than sandbox plan allows (limit is 1) fake = Faker() @@ -469,15 +490,16 @@ class TestDocumentIndexingTask: document_ids = [doc.id for doc in all_documents] # Act: Execute the task with too many documents for sandbox plan - document_indexing_task(dataset.id, document_ids) + _document_indexing(dataset.id, document_ids) # Assert: Verify error handling - for document in all_documents: - db.session.refresh(document) - assert document.indexing_status == "error" - assert document.error is not None - assert "batch upload" in document.error - assert document.stopped_at is not None + # Re-query documents from database since _document_indexing uses a different session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "error" + assert updated_document.error is not None + assert "batch upload" in updated_document.error + assert updated_document.stopped_at is not None # Verify no indexing runner was called mock_external_service_dependencies["indexing_runner"].assert_not_called() @@ -502,17 +524,18 @@ class TestDocumentIndexingTask: document_ids = [doc.id for doc in documents] # Act: Execute the task with billing disabled - document_indexing_task(dataset.id, document_ids) + _document_indexing(dataset.id, document_ids) # Assert: Verify successful processing mock_external_service_dependencies["indexing_runner"].assert_called_once() mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() # Verify documents were updated to parsing status - for document in documents: - db.session.refresh(document) - assert document.indexing_status == "parsing" - assert document.processing_started_at is not None + # Re-query documents from database since _document_indexing uses a different session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None def test_document_indexing_task_document_is_paused_error( self, db_session_with_containers, mock_external_service_dependencies @@ -540,7 +563,7 @@ class TestDocumentIndexingTask: ) # Act: Execute the task - document_indexing_task(dataset.id, document_ids) + _document_indexing(dataset.id, document_ids) # Assert: Verify exception was handled gracefully # The task should complete without raising exceptions @@ -548,7 +571,317 @@ class TestDocumentIndexingTask: mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() # Verify documents were still updated to parsing status before the exception - for document in documents: - db.session.refresh(document) - assert document.indexing_status == "parsing" - assert document.processing_started_at is not None + # Re-query documents from database since _document_indexing uses a different session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None + + # ==================== NEW TESTS FOR REFACTORED FUNCTIONS ==================== + def test_old_document_indexing_task_success(self, db_session_with_containers, mock_external_service_dependencies): + """ + Test document_indexing_task basic functionality. + + This test verifies: + - Task function calls the wrapper correctly + - Basic parameter passing works + """ + # Arrange: Create test data + dataset, documents = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=1 + ) + document_ids = [doc.id for doc in documents] + + # Act: Execute the deprecated task (it only takes 2 parameters) + document_indexing_task(dataset.id, document_ids) + + # Assert: Verify processing occurred (core logic is tested in _document_indexing tests) + mock_external_service_dependencies["indexing_runner"].assert_called_once() + mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() + + def test_normal_document_indexing_task_success( + self, db_session_with_containers, mock_external_service_dependencies + ): + """ + Test normal_document_indexing_task basic functionality. + + This test verifies: + - Task function calls the wrapper correctly + - Basic parameter passing works + """ + # Arrange: Create test data + dataset, documents = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=1 + ) + document_ids = [doc.id for doc in documents] + tenant_id = dataset.tenant_id + + # Act: Execute the new normal task + normal_document_indexing_task(tenant_id, dataset.id, document_ids) + + # Assert: Verify processing occurred (core logic is tested in _document_indexing tests) + mock_external_service_dependencies["indexing_runner"].assert_called_once() + mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() + + def test_priority_document_indexing_task_success( + self, db_session_with_containers, mock_external_service_dependencies + ): + """ + Test priority_document_indexing_task basic functionality. + + This test verifies: + - Task function calls the wrapper correctly + - Basic parameter passing works + """ + # Arrange: Create test data + dataset, documents = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=1 + ) + document_ids = [doc.id for doc in documents] + tenant_id = dataset.tenant_id + + # Act: Execute the new priority task + priority_document_indexing_task(tenant_id, dataset.id, document_ids) + + # Assert: Verify processing occurred (core logic is tested in _document_indexing tests) + mock_external_service_dependencies["indexing_runner"].assert_called_once() + mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() + + def test_document_indexing_with_tenant_queue_success( + self, db_session_with_containers, mock_external_service_dependencies + ): + """ + Test _document_indexing_with_tenant_queue function with no waiting tasks. + + This test verifies: + - Core indexing logic execution (same as _document_indexing) + - Tenant queue cleanup when no waiting tasks + - Task function parameter passing + - Queue management after processing + """ + # Arrange: Create test data + dataset, documents = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=2 + ) + document_ids = [doc.id for doc in documents] + tenant_id = dataset.tenant_id + + # Mock the task function + from unittest.mock import MagicMock + + mock_task_func = MagicMock() + + # Act: Execute the wrapper function + _document_indexing_with_tenant_queue(tenant_id, dataset.id, document_ids, mock_task_func) + + # Assert: Verify core processing occurred (same as _document_indexing) + mock_external_service_dependencies["indexing_runner"].assert_called_once() + mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() + + # Verify documents were updated (same as _document_indexing) + # Re-query documents from database since _document_indexing uses a different session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None + + # Verify the run method was called with correct documents + call_args = mock_external_service_dependencies["indexing_runner_instance"].run.call_args + assert call_args is not None + processed_documents = call_args[0][0] + assert len(processed_documents) == 2 + + # Verify task function was not called (no waiting tasks) + mock_task_func.delay.assert_not_called() + + def test_document_indexing_with_tenant_queue_with_waiting_tasks( + self, db_session_with_containers, mock_external_service_dependencies + ): + """ + Test _document_indexing_with_tenant_queue function with waiting tasks in queue using real Redis. + + This test verifies: + - Core indexing logic execution + - Real Redis-based tenant queue processing of waiting tasks + - Task function calls for waiting tasks + - Queue management with multiple tasks using actual Redis operations + """ + # Arrange: Create test data + dataset, documents = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=1 + ) + document_ids = [doc.id for doc in documents] + tenant_id = dataset.tenant_id + dataset_id = dataset.id + + # Mock the task function + from unittest.mock import MagicMock + + mock_task_func = MagicMock() + + # Use real Redis for TenantIsolatedTaskQueue + from core.rag.pipeline.queue import TenantIsolatedTaskQueue + + # Create real queue instance + queue = TenantIsolatedTaskQueue(tenant_id, "document_indexing") + + # Add waiting tasks to the real Redis queue + waiting_tasks = [ + DocumentTask(tenant_id=tenant_id, dataset_id=dataset.id, document_ids=["waiting-doc-1"]), + DocumentTask(tenant_id=tenant_id, dataset_id=dataset.id, document_ids=["waiting-doc-2"]), + ] + # Convert DocumentTask objects to dictionaries for serialization + waiting_task_dicts = [asdict(task) for task in waiting_tasks] + queue.push_tasks(waiting_task_dicts) + + # Act: Execute the wrapper function + _document_indexing_with_tenant_queue(tenant_id, dataset.id, document_ids, mock_task_func) + + # Assert: Verify core processing occurred + mock_external_service_dependencies["indexing_runner"].assert_called_once() + mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() + + # Verify task function was called for each waiting task + assert mock_task_func.delay.call_count == 1 + + # Verify correct parameters for each call + calls = mock_task_func.delay.call_args_list + assert calls[0][1] == {"tenant_id": tenant_id, "dataset_id": dataset_id, "document_ids": ["waiting-doc-1"]} + + # Verify queue is empty after processing (tasks were pulled) + remaining_tasks = queue.pull_tasks(count=10) # Pull more than we added + assert len(remaining_tasks) == 1 + + def test_document_indexing_with_tenant_queue_error_handling( + self, db_session_with_containers, mock_external_service_dependencies + ): + """ + Test error handling in _document_indexing_with_tenant_queue using real Redis. + + This test verifies: + - Exception handling during core processing + - Tenant queue cleanup even on errors using real Redis + - Proper error logging + - Function completes without raising exceptions + - Queue management continues despite core processing errors + """ + # Arrange: Create test data + dataset, documents = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=1 + ) + document_ids = [doc.id for doc in documents] + tenant_id = dataset.tenant_id + dataset_id = dataset.id + + # Mock IndexingRunner to raise an exception + mock_external_service_dependencies["indexing_runner_instance"].run.side_effect = Exception("Test error") + + # Mock the task function + from unittest.mock import MagicMock + + mock_task_func = MagicMock() + + # Use real Redis for TenantIsolatedTaskQueue + from core.rag.pipeline.queue import TenantIsolatedTaskQueue + + # Create real queue instance + queue = TenantIsolatedTaskQueue(tenant_id, "document_indexing") + + # Add waiting task to the real Redis queue + waiting_task = DocumentTask(tenant_id=tenant_id, dataset_id=dataset.id, document_ids=["waiting-doc-1"]) + queue.push_tasks([asdict(waiting_task)]) + + # Act: Execute the wrapper function + _document_indexing_with_tenant_queue(tenant_id, dataset.id, document_ids, mock_task_func) + + # Assert: Verify error was handled gracefully + # The function should not raise exceptions + mock_external_service_dependencies["indexing_runner"].assert_called_once() + mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() + + # Verify documents were still updated to parsing status before the exception + # Re-query documents from database since _document_indexing uses a different session + for doc_id in document_ids: + updated_document = db.session.query(Document).where(Document.id == doc_id).first() + assert updated_document.indexing_status == "parsing" + assert updated_document.processing_started_at is not None + + # Verify waiting task was still processed despite core processing error + mock_task_func.delay.assert_called_once() + + # Verify correct parameters for the call + call = mock_task_func.delay.call_args + assert call[1] == {"tenant_id": tenant_id, "dataset_id": dataset_id, "document_ids": ["waiting-doc-1"]} + + # Verify queue is empty after processing (task was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 0 + + def test_document_indexing_with_tenant_queue_tenant_isolation( + self, db_session_with_containers, mock_external_service_dependencies + ): + """ + Test tenant isolation in _document_indexing_with_tenant_queue using real Redis. + + This test verifies: + - Different tenants have isolated queues + - Tasks from one tenant don't affect another tenant's queue + - Queue operations are properly scoped to tenant + """ + # Arrange: Create test data for two different tenants + dataset1, documents1 = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=1 + ) + dataset2, documents2 = self._create_test_dataset_and_documents( + db_session_with_containers, mock_external_service_dependencies, document_count=1 + ) + + tenant1_id = dataset1.tenant_id + tenant2_id = dataset2.tenant_id + dataset1_id = dataset1.id + dataset2_id = dataset2.id + document_ids1 = [doc.id for doc in documents1] + document_ids2 = [doc.id for doc in documents2] + + # Mock the task function + from unittest.mock import MagicMock + + mock_task_func = MagicMock() + + # Use real Redis for TenantIsolatedTaskQueue + from core.rag.pipeline.queue import TenantIsolatedTaskQueue + + # Create queue instances for both tenants + queue1 = TenantIsolatedTaskQueue(tenant1_id, "document_indexing") + queue2 = TenantIsolatedTaskQueue(tenant2_id, "document_indexing") + + # Add waiting tasks to both queues + waiting_task1 = DocumentTask(tenant_id=tenant1_id, dataset_id=dataset1.id, document_ids=["tenant1-doc-1"]) + waiting_task2 = DocumentTask(tenant_id=tenant2_id, dataset_id=dataset2.id, document_ids=["tenant2-doc-1"]) + + queue1.push_tasks([asdict(waiting_task1)]) + queue2.push_tasks([asdict(waiting_task2)]) + + # Act: Execute the wrapper function for tenant1 only + _document_indexing_with_tenant_queue(tenant1_id, dataset1.id, document_ids1, mock_task_func) + + # Assert: Verify core processing occurred for tenant1 + mock_external_service_dependencies["indexing_runner"].assert_called_once() + mock_external_service_dependencies["indexing_runner_instance"].run.assert_called_once() + + # Verify only tenant1's waiting task was processed + mock_task_func.delay.assert_called_once() + call = mock_task_func.delay.call_args + assert call[1] == {"tenant_id": tenant1_id, "dataset_id": dataset1_id, "document_ids": ["tenant1-doc-1"]} + + # Verify tenant1's queue is empty + remaining_tasks1 = queue1.pull_tasks(count=10) + assert len(remaining_tasks1) == 0 + + # Verify tenant2's queue still has its task (isolation) + remaining_tasks2 = queue2.pull_tasks(count=10) + assert len(remaining_tasks2) == 1 + + # Verify queue keys are different + assert queue1._queue != queue2._queue + assert queue1._task_key != queue2._task_key diff --git a/api/tests/test_containers_integration_tests/tasks/test_rag_pipeline_run_tasks.py b/api/tests/test_containers_integration_tests/tasks/test_rag_pipeline_run_tasks.py new file mode 100644 index 0000000000..c82162238c --- /dev/null +++ b/api/tests/test_containers_integration_tests/tasks/test_rag_pipeline_run_tasks.py @@ -0,0 +1,936 @@ +import json +import uuid +from unittest.mock import patch + +import pytest +from faker import Faker + +from core.app.entities.app_invoke_entities import InvokeFrom, RagPipelineGenerateEntity +from core.app.entities.rag_pipeline_invoke_entities import RagPipelineInvokeEntity +from core.rag.pipeline.queue import TenantIsolatedTaskQueue +from extensions.ext_database import db +from models import Account, Tenant, TenantAccountJoin, TenantAccountRole +from models.dataset import Pipeline +from models.workflow import Workflow +from tasks.rag_pipeline.priority_rag_pipeline_run_task import ( + priority_rag_pipeline_run_task, + run_single_rag_pipeline_task, +) +from tasks.rag_pipeline.rag_pipeline_run_task import rag_pipeline_run_task + + +class TestRagPipelineRunTasks: + """Integration tests for RAG pipeline run tasks using testcontainers. + + This test class covers: + - priority_rag_pipeline_run_task function + - rag_pipeline_run_task function + - run_single_rag_pipeline_task function + - Real Redis-based TenantIsolatedTaskQueue operations + - PipelineGenerator._generate method mocking and parameter validation + - File operations and cleanup + - Error handling and queue management + """ + + @pytest.fixture + def mock_pipeline_generator(self): + """Mock PipelineGenerator._generate method.""" + with patch("core.app.apps.pipeline.pipeline_generator.PipelineGenerator._generate") as mock_generate: + # Mock the _generate method to return a simple response + mock_generate.return_value = {"answer": "Test response", "metadata": {"test": "data"}} + yield mock_generate + + @pytest.fixture + def mock_file_service(self): + """Mock FileService for file operations.""" + with ( + patch("services.file_service.FileService.get_file_content") as mock_get_content, + patch("services.file_service.FileService.delete_file") as mock_delete_file, + ): + yield { + "get_content": mock_get_content, + "delete_file": mock_delete_file, + } + + def _create_test_pipeline_and_workflow(self, db_session_with_containers): + """ + Helper method to create test pipeline and workflow for testing. + + Args: + db_session_with_containers: Database session from testcontainers infrastructure + + Returns: + tuple: (account, tenant, pipeline, workflow) - Created entities + """ + fake = Faker() + + # Create account and tenant + account = Account( + email=fake.email(), + name=fake.name(), + interface_language="en-US", + status="active", + ) + db.session.add(account) + db.session.commit() + + tenant = Tenant( + name=fake.company(), + status="normal", + ) + db.session.add(tenant) + db.session.commit() + + # Create tenant-account join + join = TenantAccountJoin( + tenant_id=tenant.id, + account_id=account.id, + role=TenantAccountRole.OWNER, + current=True, + ) + db.session.add(join) + db.session.commit() + + # Create workflow + workflow = Workflow( + id=str(uuid.uuid4()), + tenant_id=tenant.id, + app_id=str(uuid.uuid4()), + type="workflow", + version="draft", + graph="{}", + features="{}", + marked_name=fake.company(), + marked_comment=fake.text(max_nb_chars=100), + created_by=account.id, + environment_variables=[], + conversation_variables=[], + rag_pipeline_variables=[], + ) + db.session.add(workflow) + db.session.commit() + + # Create pipeline + pipeline = Pipeline( + id=str(uuid.uuid4()), + tenant_id=tenant.id, + workflow_id=workflow.id, + name=fake.company(), + description=fake.text(max_nb_chars=100), + created_by=account.id, + ) + db.session.add(pipeline) + db.session.commit() + + # Refresh entities to ensure they're properly loaded + db.session.refresh(account) + db.session.refresh(tenant) + db.session.refresh(workflow) + db.session.refresh(pipeline) + + return account, tenant, pipeline, workflow + + def _create_rag_pipeline_invoke_entities(self, account, tenant, pipeline, workflow, count=2): + """ + Helper method to create RAG pipeline invoke entities for testing. + + Args: + account: Account instance + tenant: Tenant instance + pipeline: Pipeline instance + workflow: Workflow instance + count: Number of entities to create + + Returns: + list: List of RagPipelineInvokeEntity instances + """ + fake = Faker() + entities = [] + + for i in range(count): + # Create application generate entity + app_config = { + "app_id": str(uuid.uuid4()), + "app_name": fake.company(), + "mode": "workflow", + "workflow_id": workflow.id, + "tenant_id": tenant.id, + "app_mode": "workflow", + } + + application_generate_entity = { + "task_id": str(uuid.uuid4()), + "app_config": app_config, + "inputs": {"query": f"Test query {i}"}, + "files": [], + "user_id": account.id, + "stream": False, + "invoke_from": "published", + "workflow_execution_id": str(uuid.uuid4()), + "pipeline_config": { + "app_id": str(uuid.uuid4()), + "app_name": fake.company(), + "mode": "workflow", + "workflow_id": workflow.id, + "tenant_id": tenant.id, + "app_mode": "workflow", + }, + "datasource_type": "upload_file", + "datasource_info": {}, + "dataset_id": str(uuid.uuid4()), + "batch": "test_batch", + } + + entity = RagPipelineInvokeEntity( + pipeline_id=pipeline.id, + application_generate_entity=application_generate_entity, + user_id=account.id, + tenant_id=tenant.id, + workflow_id=workflow.id, + streaming=False, + workflow_execution_id=str(uuid.uuid4()), + workflow_thread_pool_id=str(uuid.uuid4()), + ) + entities.append(entity) + + return entities + + def _create_file_content_for_entities(self, entities): + """ + Helper method to create file content for RAG pipeline invoke entities. + + Args: + entities: List of RagPipelineInvokeEntity instances + + Returns: + str: JSON string containing serialized entities + """ + entities_data = [entity.model_dump() for entity in entities] + return json.dumps(entities_data) + + def test_priority_rag_pipeline_run_task_success( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test successful priority RAG pipeline run task execution. + + This test verifies: + - Task execution with multiple RAG pipeline invoke entities + - File content retrieval and parsing + - PipelineGenerator._generate method calls with correct parameters + - Thread pool execution + - File cleanup after execution + - Queue management with no waiting tasks + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=2) + file_content = self._create_file_content_for_entities(entities) + + # Mock file service + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].return_value = file_content + + # Act: Execute the priority task + priority_rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify expected outcomes + # Verify file operations + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_file_service["delete_file"].assert_called_once_with(file_id) + + # Verify PipelineGenerator._generate was called for each entity + assert mock_pipeline_generator.call_count == 2 + + # Verify call parameters for each entity + calls = mock_pipeline_generator.call_args_list + for call in calls: + call_kwargs = call[1] # Get keyword arguments + assert call_kwargs["pipeline"].id == pipeline.id + assert call_kwargs["workflow_id"] == workflow.id + assert call_kwargs["user"].id == account.id + assert call_kwargs["invoke_from"] == InvokeFrom.PUBLISHED + assert call_kwargs["streaming"] == False + assert isinstance(call_kwargs["application_generate_entity"], RagPipelineGenerateEntity) + + def test_rag_pipeline_run_task_success( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test successful regular RAG pipeline run task execution. + + This test verifies: + - Task execution with multiple RAG pipeline invoke entities + - File content retrieval and parsing + - PipelineGenerator._generate method calls with correct parameters + - Thread pool execution + - File cleanup after execution + - Queue management with no waiting tasks + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=3) + file_content = self._create_file_content_for_entities(entities) + + # Mock file service + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].return_value = file_content + + # Act: Execute the regular task + rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify expected outcomes + # Verify file operations + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_file_service["delete_file"].assert_called_once_with(file_id) + + # Verify PipelineGenerator._generate was called for each entity + assert mock_pipeline_generator.call_count == 3 + + # Verify call parameters for each entity + calls = mock_pipeline_generator.call_args_list + for call in calls: + call_kwargs = call[1] # Get keyword arguments + assert call_kwargs["pipeline"].id == pipeline.id + assert call_kwargs["workflow_id"] == workflow.id + assert call_kwargs["user"].id == account.id + assert call_kwargs["invoke_from"] == InvokeFrom.PUBLISHED + assert call_kwargs["streaming"] == False + assert isinstance(call_kwargs["application_generate_entity"], RagPipelineGenerateEntity) + + def test_priority_rag_pipeline_run_task_with_waiting_tasks( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test priority RAG pipeline run task with waiting tasks in queue using real Redis. + + This test verifies: + - Core task execution + - Real Redis-based tenant queue processing of waiting tasks + - Task function calls for waiting tasks + - Queue management with multiple tasks using actual Redis operations + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=1) + file_content = self._create_file_content_for_entities(entities) + + # Mock file service + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].return_value = file_content + + # Use real Redis for TenantIsolatedTaskQueue + queue = TenantIsolatedTaskQueue(tenant.id, "pipeline") + + # Add waiting tasks to the real Redis queue + waiting_file_ids = [str(uuid.uuid4()) for _ in range(2)] + queue.push_tasks(waiting_file_ids) + + # Mock the task function calls + with patch( + "tasks.rag_pipeline.priority_rag_pipeline_run_task.priority_rag_pipeline_run_task.delay" + ) as mock_delay: + # Act: Execute the priority task + priority_rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify core processing occurred + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_file_service["delete_file"].assert_called_once_with(file_id) + assert mock_pipeline_generator.call_count == 1 + + # Verify waiting tasks were processed, pull 1 task a time by default + assert mock_delay.call_count == 1 + + # Verify correct parameters for the call + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_ids[0] + assert call_kwargs.get("tenant_id") == tenant.id + + # Verify queue still has remaining tasks (only 1 was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 1 # 2 original - 1 pulled = 1 remaining + + def test_rag_pipeline_run_task_legacy_compatibility( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test regular RAG pipeline run task with legacy Redis queue format for backward compatibility. + + This test simulates the scenario where: + - Old code writes file IDs directly to Redis list using lpush + - New worker processes these legacy queue entries + - Ensures backward compatibility during deployment transition + + Legacy format: redis_client.lpush(tenant_self_pipeline_task_queue, upload_file.id) + New format: TenantIsolatedTaskQueue.push_tasks([file_id]) + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=1) + file_content = self._create_file_content_for_entities(entities) + + # Mock file service + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].return_value = file_content + + # Simulate legacy Redis queue format - direct file IDs in Redis list + from extensions.ext_redis import redis_client + + # Legacy queue key format (old code) + legacy_queue_key = f"tenant_self_pipeline_task_queue:{tenant.id}" + legacy_task_key = f"tenant_pipeline_task:{tenant.id}" + + # Add legacy format data to Redis (simulating old code behavior) + legacy_file_ids = [str(uuid.uuid4()) for _ in range(3)] + for file_id_legacy in legacy_file_ids: + redis_client.lpush(legacy_queue_key, file_id_legacy) + + # Set the task key to indicate there are waiting tasks (legacy behavior) + redis_client.set(legacy_task_key, 1, ex=60 * 60) + + # Mock the task function calls + with patch("tasks.rag_pipeline.rag_pipeline_run_task.rag_pipeline_run_task.delay") as mock_delay: + # Act: Execute the priority task with new code but legacy queue data + rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify core processing occurred + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_file_service["delete_file"].assert_called_once_with(file_id) + assert mock_pipeline_generator.call_count == 1 + + # Verify waiting tasks were processed, pull 1 task a time by default + assert mock_delay.call_count == 1 + + # Verify correct parameters for the call + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == legacy_file_ids[0] + assert call_kwargs.get("tenant_id") == tenant.id + + # Verify that new code can process legacy queue entries + # The new TenantIsolatedTaskQueue should be able to read from the legacy format + queue = TenantIsolatedTaskQueue(tenant.id, "pipeline") + + # Verify queue still has remaining tasks (only 1 was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 2 # 3 original - 1 pulled = 2 remaining + + # Cleanup: Remove legacy test data + redis_client.delete(legacy_queue_key) + redis_client.delete(legacy_task_key) + + def test_rag_pipeline_run_task_with_waiting_tasks( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test regular RAG pipeline run task with waiting tasks in queue using real Redis. + + This test verifies: + - Core task execution + - Real Redis-based tenant queue processing of waiting tasks + - Task function calls for waiting tasks + - Queue management with multiple tasks using actual Redis operations + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=1) + file_content = self._create_file_content_for_entities(entities) + + # Mock file service + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].return_value = file_content + + # Use real Redis for TenantIsolatedTaskQueue + queue = TenantIsolatedTaskQueue(tenant.id, "pipeline") + + # Add waiting tasks to the real Redis queue + waiting_file_ids = [str(uuid.uuid4()) for _ in range(3)] + queue.push_tasks(waiting_file_ids) + + # Mock the task function calls + with patch("tasks.rag_pipeline.rag_pipeline_run_task.rag_pipeline_run_task.delay") as mock_delay: + # Act: Execute the regular task + rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify core processing occurred + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_file_service["delete_file"].assert_called_once_with(file_id) + assert mock_pipeline_generator.call_count == 1 + + # Verify waiting tasks were processed, pull 1 task a time by default + assert mock_delay.call_count == 1 + + # Verify correct parameters for the call + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_ids[0] + assert call_kwargs.get("tenant_id") == tenant.id + + # Verify queue still has remaining tasks (only 1 was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 2 # 3 original - 1 pulled = 2 remaining + + def test_priority_rag_pipeline_run_task_error_handling( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test error handling in priority RAG pipeline run task using real Redis. + + This test verifies: + - Exception handling during core processing + - Tenant queue cleanup even on errors using real Redis + - Proper error logging + - Function completes without raising exceptions + - Queue management continues despite core processing errors + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=1) + file_content = self._create_file_content_for_entities(entities) + + # Mock file service + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].return_value = file_content + + # Mock PipelineGenerator to raise an exception + mock_pipeline_generator.side_effect = Exception("Pipeline generation failed") + + # Use real Redis for TenantIsolatedTaskQueue + queue = TenantIsolatedTaskQueue(tenant.id, "pipeline") + + # Add waiting task to the real Redis queue + waiting_file_id = str(uuid.uuid4()) + queue.push_tasks([waiting_file_id]) + + # Mock the task function calls + with patch( + "tasks.rag_pipeline.priority_rag_pipeline_run_task.priority_rag_pipeline_run_task.delay" + ) as mock_delay: + # Act: Execute the priority task (should not raise exception) + priority_rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify error was handled gracefully + # The function should not raise exceptions + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_file_service["delete_file"].assert_called_once_with(file_id) + assert mock_pipeline_generator.call_count == 1 + + # Verify waiting task was still processed despite core processing error + mock_delay.assert_called_once() + + # Verify correct parameters for the call + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_id + assert call_kwargs.get("tenant_id") == tenant.id + + # Verify queue is empty after processing (task was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 0 + + def test_rag_pipeline_run_task_error_handling( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test error handling in regular RAG pipeline run task using real Redis. + + This test verifies: + - Exception handling during core processing + - Tenant queue cleanup even on errors using real Redis + - Proper error logging + - Function completes without raising exceptions + - Queue management continues despite core processing errors + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=1) + file_content = self._create_file_content_for_entities(entities) + + # Mock file service + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].return_value = file_content + + # Mock PipelineGenerator to raise an exception + mock_pipeline_generator.side_effect = Exception("Pipeline generation failed") + + # Use real Redis for TenantIsolatedTaskQueue + queue = TenantIsolatedTaskQueue(tenant.id, "pipeline") + + # Add waiting task to the real Redis queue + waiting_file_id = str(uuid.uuid4()) + queue.push_tasks([waiting_file_id]) + + # Mock the task function calls + with patch("tasks.rag_pipeline.rag_pipeline_run_task.rag_pipeline_run_task.delay") as mock_delay: + # Act: Execute the regular task (should not raise exception) + rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify error was handled gracefully + # The function should not raise exceptions + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_file_service["delete_file"].assert_called_once_with(file_id) + assert mock_pipeline_generator.call_count == 1 + + # Verify waiting task was still processed despite core processing error + mock_delay.assert_called_once() + + # Verify correct parameters for the call + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_id + assert call_kwargs.get("tenant_id") == tenant.id + + # Verify queue is empty after processing (task was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 0 + + def test_priority_rag_pipeline_run_task_tenant_isolation( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test tenant isolation in priority RAG pipeline run task using real Redis. + + This test verifies: + - Different tenants have isolated queues + - Tasks from one tenant don't affect another tenant's queue + - Queue operations are properly scoped to tenant + """ + # Arrange: Create test data for two different tenants + account1, tenant1, pipeline1, workflow1 = self._create_test_pipeline_and_workflow(db_session_with_containers) + account2, tenant2, pipeline2, workflow2 = self._create_test_pipeline_and_workflow(db_session_with_containers) + + entities1 = self._create_rag_pipeline_invoke_entities(account1, tenant1, pipeline1, workflow1, count=1) + entities2 = self._create_rag_pipeline_invoke_entities(account2, tenant2, pipeline2, workflow2, count=1) + + file_content1 = self._create_file_content_for_entities(entities1) + file_content2 = self._create_file_content_for_entities(entities2) + + # Mock file service + file_id1 = str(uuid.uuid4()) + file_id2 = str(uuid.uuid4()) + mock_file_service["get_content"].side_effect = [file_content1, file_content2] + + # Use real Redis for TenantIsolatedTaskQueue + queue1 = TenantIsolatedTaskQueue(tenant1.id, "pipeline") + queue2 = TenantIsolatedTaskQueue(tenant2.id, "pipeline") + + # Add waiting tasks to both queues + waiting_file_id1 = str(uuid.uuid4()) + waiting_file_id2 = str(uuid.uuid4()) + + queue1.push_tasks([waiting_file_id1]) + queue2.push_tasks([waiting_file_id2]) + + # Mock the task function calls + with patch( + "tasks.rag_pipeline.priority_rag_pipeline_run_task.priority_rag_pipeline_run_task.delay" + ) as mock_delay: + # Act: Execute the priority task for tenant1 only + priority_rag_pipeline_run_task(file_id1, tenant1.id) + + # Assert: Verify core processing occurred for tenant1 + assert mock_file_service["get_content"].call_count == 1 + assert mock_file_service["delete_file"].call_count == 1 + assert mock_pipeline_generator.call_count == 1 + + # Verify only tenant1's waiting task was processed + mock_delay.assert_called_once() + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_id1 + assert call_kwargs.get("tenant_id") == tenant1.id + + # Verify tenant1's queue is empty + remaining_tasks1 = queue1.pull_tasks(count=10) + assert len(remaining_tasks1) == 0 + + # Verify tenant2's queue still has its task (isolation) + remaining_tasks2 = queue2.pull_tasks(count=10) + assert len(remaining_tasks2) == 1 + + # Verify queue keys are different + assert queue1._queue != queue2._queue + assert queue1._task_key != queue2._task_key + + def test_rag_pipeline_run_task_tenant_isolation( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test tenant isolation in regular RAG pipeline run task using real Redis. + + This test verifies: + - Different tenants have isolated queues + - Tasks from one tenant don't affect another tenant's queue + - Queue operations are properly scoped to tenant + """ + # Arrange: Create test data for two different tenants + account1, tenant1, pipeline1, workflow1 = self._create_test_pipeline_and_workflow(db_session_with_containers) + account2, tenant2, pipeline2, workflow2 = self._create_test_pipeline_and_workflow(db_session_with_containers) + + entities1 = self._create_rag_pipeline_invoke_entities(account1, tenant1, pipeline1, workflow1, count=1) + entities2 = self._create_rag_pipeline_invoke_entities(account2, tenant2, pipeline2, workflow2, count=1) + + file_content1 = self._create_file_content_for_entities(entities1) + file_content2 = self._create_file_content_for_entities(entities2) + + # Mock file service + file_id1 = str(uuid.uuid4()) + file_id2 = str(uuid.uuid4()) + mock_file_service["get_content"].side_effect = [file_content1, file_content2] + + # Use real Redis for TenantIsolatedTaskQueue + queue1 = TenantIsolatedTaskQueue(tenant1.id, "pipeline") + queue2 = TenantIsolatedTaskQueue(tenant2.id, "pipeline") + + # Add waiting tasks to both queues + waiting_file_id1 = str(uuid.uuid4()) + waiting_file_id2 = str(uuid.uuid4()) + + queue1.push_tasks([waiting_file_id1]) + queue2.push_tasks([waiting_file_id2]) + + # Mock the task function calls + with patch("tasks.rag_pipeline.rag_pipeline_run_task.rag_pipeline_run_task.delay") as mock_delay: + # Act: Execute the regular task for tenant1 only + rag_pipeline_run_task(file_id1, tenant1.id) + + # Assert: Verify core processing occurred for tenant1 + assert mock_file_service["get_content"].call_count == 1 + assert mock_file_service["delete_file"].call_count == 1 + assert mock_pipeline_generator.call_count == 1 + + # Verify only tenant1's waiting task was processed + mock_delay.assert_called_once() + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_id1 + assert call_kwargs.get("tenant_id") == tenant1.id + + # Verify tenant1's queue is empty + remaining_tasks1 = queue1.pull_tasks(count=10) + assert len(remaining_tasks1) == 0 + + # Verify tenant2's queue still has its task (isolation) + remaining_tasks2 = queue2.pull_tasks(count=10) + assert len(remaining_tasks2) == 1 + + # Verify queue keys are different + assert queue1._queue != queue2._queue + assert queue1._task_key != queue2._task_key + + def test_run_single_rag_pipeline_task_success( + self, db_session_with_containers, mock_pipeline_generator, flask_app_with_containers + ): + """ + Test successful run_single_rag_pipeline_task execution. + + This test verifies: + - Single RAG pipeline task execution within Flask app context + - Entity validation and database queries + - PipelineGenerator._generate method call with correct parameters + - Proper Flask context handling + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + entities = self._create_rag_pipeline_invoke_entities(account, tenant, pipeline, workflow, count=1) + entity_data = entities[0].model_dump() + + # Act: Execute the single task + with flask_app_with_containers.app_context(): + run_single_rag_pipeline_task(entity_data, flask_app_with_containers) + + # Assert: Verify expected outcomes + # Verify PipelineGenerator._generate was called + assert mock_pipeline_generator.call_count == 1 + + # Verify call parameters + call = mock_pipeline_generator.call_args + call_kwargs = call[1] # Get keyword arguments + assert call_kwargs["pipeline"].id == pipeline.id + assert call_kwargs["workflow_id"] == workflow.id + assert call_kwargs["user"].id == account.id + assert call_kwargs["invoke_from"] == InvokeFrom.PUBLISHED + assert call_kwargs["streaming"] == False + assert isinstance(call_kwargs["application_generate_entity"], RagPipelineGenerateEntity) + + def test_run_single_rag_pipeline_task_entity_validation_error( + self, db_session_with_containers, mock_pipeline_generator, flask_app_with_containers + ): + """ + Test run_single_rag_pipeline_task with invalid entity data. + + This test verifies: + - Proper error handling for invalid entity data + - Exception logging + - Function raises ValueError for missing entities + """ + # Arrange: Create entity data with valid UUIDs but non-existent entities + fake = Faker() + invalid_entity_data = { + "pipeline_id": str(uuid.uuid4()), + "application_generate_entity": { + "app_config": { + "app_id": str(uuid.uuid4()), + "app_name": "Test App", + "mode": "workflow", + "workflow_id": str(uuid.uuid4()), + }, + "inputs": {"query": "Test query"}, + "query": "Test query", + "response_mode": "blocking", + "user": str(uuid.uuid4()), + "files": [], + "conversation_id": str(uuid.uuid4()), + }, + "user_id": str(uuid.uuid4()), + "tenant_id": str(uuid.uuid4()), + "workflow_id": str(uuid.uuid4()), + "streaming": False, + "workflow_execution_id": str(uuid.uuid4()), + "workflow_thread_pool_id": str(uuid.uuid4()), + } + + # Act & Assert: Execute the single task with non-existent entities (should raise ValueError) + with flask_app_with_containers.app_context(): + with pytest.raises(ValueError, match="Account .* not found"): + run_single_rag_pipeline_task(invalid_entity_data, flask_app_with_containers) + + # Assert: Pipeline generator should not be called + mock_pipeline_generator.assert_not_called() + + def test_run_single_rag_pipeline_task_database_entity_not_found( + self, db_session_with_containers, mock_pipeline_generator, flask_app_with_containers + ): + """ + Test run_single_rag_pipeline_task with non-existent database entities. + + This test verifies: + - Proper error handling for missing database entities + - Exception logging + - Function raises ValueError for missing entities + """ + # Arrange: Create test data with non-existent IDs + fake = Faker() + entity_data = { + "pipeline_id": str(uuid.uuid4()), + "application_generate_entity": { + "app_config": { + "app_id": str(uuid.uuid4()), + "app_name": "Test App", + "mode": "workflow", + "workflow_id": str(uuid.uuid4()), + }, + "inputs": {"query": "Test query"}, + "query": "Test query", + "response_mode": "blocking", + "user": str(uuid.uuid4()), + "files": [], + "conversation_id": str(uuid.uuid4()), + }, + "user_id": str(uuid.uuid4()), + "tenant_id": str(uuid.uuid4()), + "workflow_id": str(uuid.uuid4()), + "streaming": False, + "workflow_execution_id": str(uuid.uuid4()), + "workflow_thread_pool_id": str(uuid.uuid4()), + } + + # Act & Assert: Execute the single task with non-existent entities (should raise ValueError) + with flask_app_with_containers.app_context(): + with pytest.raises(ValueError, match="Account .* not found"): + run_single_rag_pipeline_task(entity_data, flask_app_with_containers) + + # Assert: Pipeline generator should not be called + mock_pipeline_generator.assert_not_called() + + def test_priority_rag_pipeline_run_task_file_not_found( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test priority RAG pipeline run task with non-existent file. + + This test verifies: + - Proper error handling for missing files + - Exception logging + - Function raises Exception for file errors + - Queue management continues despite file errors + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + + # Mock file service to raise exception + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].side_effect = Exception("File not found") + + # Use real Redis for TenantIsolatedTaskQueue + queue = TenantIsolatedTaskQueue(tenant.id, "pipeline") + + # Add waiting task to the real Redis queue + waiting_file_id = str(uuid.uuid4()) + queue.push_tasks([waiting_file_id]) + + # Mock the task function calls + with patch( + "tasks.rag_pipeline.priority_rag_pipeline_run_task.priority_rag_pipeline_run_task.delay" + ) as mock_delay: + # Act & Assert: Execute the priority task (should raise Exception) + with pytest.raises(Exception, match="File not found"): + priority_rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify error was handled gracefully + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_pipeline_generator.assert_not_called() + + # Verify waiting task was still processed despite file error + mock_delay.assert_called_once() + + # Verify correct parameters for the call + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_id + assert call_kwargs.get("tenant_id") == tenant.id + + # Verify queue is empty after processing (task was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 0 + + def test_rag_pipeline_run_task_file_not_found( + self, db_session_with_containers, mock_pipeline_generator, mock_file_service + ): + """ + Test regular RAG pipeline run task with non-existent file. + + This test verifies: + - Proper error handling for missing files + - Exception logging + - Function raises Exception for file errors + - Queue management continues despite file errors + """ + # Arrange: Create test data + account, tenant, pipeline, workflow = self._create_test_pipeline_and_workflow(db_session_with_containers) + + # Mock file service to raise exception + file_id = str(uuid.uuid4()) + mock_file_service["get_content"].side_effect = Exception("File not found") + + # Use real Redis for TenantIsolatedTaskQueue + queue = TenantIsolatedTaskQueue(tenant.id, "pipeline") + + # Add waiting task to the real Redis queue + waiting_file_id = str(uuid.uuid4()) + queue.push_tasks([waiting_file_id]) + + # Mock the task function calls + with patch("tasks.rag_pipeline.rag_pipeline_run_task.rag_pipeline_run_task.delay") as mock_delay: + # Act & Assert: Execute the regular task (should raise Exception) + with pytest.raises(Exception, match="File not found"): + rag_pipeline_run_task(file_id, tenant.id) + + # Assert: Verify error was handled gracefully + mock_file_service["get_content"].assert_called_once_with(file_id) + mock_pipeline_generator.assert_not_called() + + # Verify waiting task was still processed despite file error + mock_delay.assert_called_once() + + # Verify correct parameters for the call + call_kwargs = mock_delay.call_args[1] if mock_delay.call_args else {} + assert call_kwargs.get("rag_pipeline_invoke_entities_file_id") == waiting_file_id + assert call_kwargs.get("tenant_id") == tenant.id + + # Verify queue is empty after processing (task was pulled) + remaining_tasks = queue.pull_tasks(count=10) + assert len(remaining_tasks) == 0 diff --git a/api/tests/test_containers_integration_tests/test_workflow_pause_integration.py b/api/tests/test_containers_integration_tests/test_workflow_pause_integration.py new file mode 100644 index 0000000000..79da5d4d0e --- /dev/null +++ b/api/tests/test_containers_integration_tests/test_workflow_pause_integration.py @@ -0,0 +1,948 @@ +"""Comprehensive integration tests for workflow pause functionality. + +This test suite covers complete workflow pause functionality including: +- Real database interactions using containerized PostgreSQL +- Real storage operations using the test storage backend +- Complete workflow: create -> pause -> resume -> delete +- Testing with actual FileService (not mocked) +- Database transactions and rollback behavior +- Actual file upload and retrieval through storage +- Workflow status transitions in the database +- Error handling with real database constraints +- Concurrent access scenarios +- Multi-tenant isolation +- Prune functionality +- File storage integration + +These tests use TestContainers to spin up real services for integration testing, +providing more reliable and realistic test scenarios than mocks. +""" + +import json +import uuid +from dataclasses import dataclass +from datetime import timedelta + +import pytest +from sqlalchemy import delete, select +from sqlalchemy.orm import Session, selectinload, sessionmaker + +from core.workflow.entities import WorkflowExecution +from core.workflow.enums import WorkflowExecutionStatus +from extensions.ext_storage import storage +from libs.datetime_utils import naive_utc_now +from models import Account +from models import WorkflowPause as WorkflowPauseModel +from models.account import Tenant, TenantAccountJoin, TenantAccountRole +from models.model import UploadFile +from models.workflow import Workflow, WorkflowRun +from repositories.sqlalchemy_api_workflow_run_repository import ( + DifyAPISQLAlchemyWorkflowRunRepository, + _WorkflowRunError, +) + + +@dataclass +class PauseWorkflowSuccessCase: + """Test case for successful pause workflow operations.""" + + name: str + initial_status: WorkflowExecutionStatus + description: str = "" + + +@dataclass +class PauseWorkflowFailureCase: + """Test case for pause workflow failure scenarios.""" + + name: str + initial_status: WorkflowExecutionStatus + description: str = "" + + +@dataclass +class ResumeWorkflowSuccessCase: + """Test case for successful resume workflow operations.""" + + name: str + initial_status: WorkflowExecutionStatus + description: str = "" + + +@dataclass +class ResumeWorkflowFailureCase: + """Test case for resume workflow failure scenarios.""" + + name: str + initial_status: WorkflowExecutionStatus + pause_resumed: bool + set_running_status: bool = False + description: str = "" + + +@dataclass +class PrunePausesTestCase: + """Test case for prune pauses operations.""" + + name: str + pause_age: timedelta + resume_age: timedelta | None + expected_pruned_count: int + description: str = "" + + +def pause_workflow_failure_cases() -> list[PauseWorkflowFailureCase]: + """Create test cases for pause workflow failure scenarios.""" + return [ + PauseWorkflowFailureCase( + name="pause_already_paused_workflow", + initial_status=WorkflowExecutionStatus.PAUSED, + description="Should fail to pause an already paused workflow", + ), + PauseWorkflowFailureCase( + name="pause_completed_workflow", + initial_status=WorkflowExecutionStatus.SUCCEEDED, + description="Should fail to pause a completed workflow", + ), + PauseWorkflowFailureCase( + name="pause_failed_workflow", + initial_status=WorkflowExecutionStatus.FAILED, + description="Should fail to pause a failed workflow", + ), + ] + + +def resume_workflow_success_cases() -> list[ResumeWorkflowSuccessCase]: + """Create test cases for successful resume workflow operations.""" + return [ + ResumeWorkflowSuccessCase( + name="resume_paused_workflow", + initial_status=WorkflowExecutionStatus.PAUSED, + description="Should successfully resume a paused workflow", + ), + ] + + +def resume_workflow_failure_cases() -> list[ResumeWorkflowFailureCase]: + """Create test cases for resume workflow failure scenarios.""" + return [ + ResumeWorkflowFailureCase( + name="resume_already_resumed_workflow", + initial_status=WorkflowExecutionStatus.PAUSED, + pause_resumed=True, + description="Should fail to resume an already resumed workflow", + ), + ResumeWorkflowFailureCase( + name="resume_running_workflow", + initial_status=WorkflowExecutionStatus.RUNNING, + pause_resumed=False, + set_running_status=True, + description="Should fail to resume a running workflow", + ), + ] + + +def prune_pauses_test_cases() -> list[PrunePausesTestCase]: + """Create test cases for prune pauses operations.""" + return [ + PrunePausesTestCase( + name="prune_old_active_pauses", + pause_age=timedelta(days=7), + resume_age=None, + expected_pruned_count=1, + description="Should prune old active pauses", + ), + PrunePausesTestCase( + name="prune_old_resumed_pauses", + pause_age=timedelta(hours=12), # Created 12 hours ago (recent) + resume_age=timedelta(days=7), + expected_pruned_count=1, + description="Should prune old resumed pauses", + ), + PrunePausesTestCase( + name="keep_recent_active_pauses", + pause_age=timedelta(hours=1), + resume_age=None, + expected_pruned_count=0, + description="Should keep recent active pauses", + ), + PrunePausesTestCase( + name="keep_recent_resumed_pauses", + pause_age=timedelta(days=1), + resume_age=timedelta(hours=1), + expected_pruned_count=0, + description="Should keep recent resumed pauses", + ), + ] + + +class TestWorkflowPauseIntegration: + """Comprehensive integration tests for workflow pause functionality.""" + + @pytest.fixture(autouse=True) + def setup_test_data(self, db_session_with_containers): + """Set up test data for each test method using TestContainers.""" + # Create test tenant and account + + tenant = Tenant( + name="Test Tenant", + status="normal", + ) + db_session_with_containers.add(tenant) + db_session_with_containers.commit() + + account = Account( + email="test@example.com", + name="Test User", + interface_language="en-US", + status="active", + ) + db_session_with_containers.add(account) + db_session_with_containers.commit() + + # Create tenant-account join + tenant_join = TenantAccountJoin( + tenant_id=tenant.id, + account_id=account.id, + role=TenantAccountRole.OWNER, + current=True, + ) + db_session_with_containers.add(tenant_join) + db_session_with_containers.commit() + + # Set test data + self.test_tenant_id = tenant.id + self.test_user_id = account.id + self.test_app_id = str(uuid.uuid4()) + self.test_workflow_id = str(uuid.uuid4()) + + # Create test workflow + self.test_workflow = Workflow( + id=self.test_workflow_id, + tenant_id=self.test_tenant_id, + app_id=self.test_app_id, + type="workflow", + version="draft", + graph='{"nodes": [], "edges": []}', + features='{"file_upload": {"enabled": false}}', + created_by=self.test_user_id, + created_at=naive_utc_now(), + ) + + # Store session instance + self.session = db_session_with_containers + + # Save test data to database + self.session.add(self.test_workflow) + self.session.commit() + + yield + + # Cleanup + self._cleanup_test_data() + + def _cleanup_test_data(self): + """Clean up test data after each test method.""" + # Clean up workflow pauses + self.session.execute(delete(WorkflowPauseModel)) + # Clean up upload files + self.session.execute( + delete(UploadFile).where( + UploadFile.tenant_id == self.test_tenant_id, + ) + ) + # Clean up workflow runs + self.session.execute( + delete(WorkflowRun).where( + WorkflowRun.tenant_id == self.test_tenant_id, + WorkflowRun.app_id == self.test_app_id, + ) + ) + # Clean up workflows + self.session.execute( + delete(Workflow).where( + Workflow.tenant_id == self.test_tenant_id, + Workflow.app_id == self.test_app_id, + ) + ) + self.session.commit() + + def _create_test_workflow_run( + self, status: WorkflowExecutionStatus = WorkflowExecutionStatus.RUNNING + ) -> WorkflowRun: + """Create a test workflow run with specified status.""" + workflow_run = WorkflowRun( + id=str(uuid.uuid4()), + tenant_id=self.test_tenant_id, + app_id=self.test_app_id, + workflow_id=self.test_workflow_id, + type="workflow", + triggered_from="debugging", + version="draft", + status=status, + created_by=self.test_user_id, + created_by_role="account", + created_at=naive_utc_now(), + ) + self.session.add(workflow_run) + self.session.commit() + return workflow_run + + def _create_test_state(self) -> str: + """Create a test state string.""" + return json.dumps( + { + "node_id": "test-node", + "node_type": "llm", + "status": "paused", + "data": {"key": "value"}, + "timestamp": naive_utc_now().isoformat(), + } + ) + + def _get_workflow_run_repository(self): + """Get workflow run repository instance for testing.""" + # Create session factory from the test session + engine = self.session.get_bind() + session_factory = sessionmaker(bind=engine, expire_on_commit=False) + + # Create a test-specific repository that implements the missing save method + class TestWorkflowRunRepository(DifyAPISQLAlchemyWorkflowRunRepository): + """Test-specific repository that implements the missing save method.""" + + def save(self, execution: WorkflowExecution): + """Implement the missing save method for testing.""" + # For testing purposes, we don't need to implement this method + # as it's not used in the pause functionality tests + pass + + # Create and return repository instance + repository = TestWorkflowRunRepository(session_maker=session_factory) + return repository + + # ==================== Complete Pause Workflow Tests ==================== + + def test_complete_pause_resume_workflow(self): + """Test complete workflow: create -> pause -> resume -> delete.""" + # Arrange + workflow_run = self._create_test_workflow_run() + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + # Act - Create pause state + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + # Assert - Pause state created + assert pause_entity is not None + assert pause_entity.id is not None + assert pause_entity.workflow_execution_id == workflow_run.id + # Convert both to strings for comparison + retrieved_state = pause_entity.get_state() + if isinstance(retrieved_state, bytes): + retrieved_state = retrieved_state.decode() + assert retrieved_state == test_state + + # Verify database state + query = select(WorkflowPauseModel).where(WorkflowPauseModel.workflow_run_id == workflow_run.id) + pause_model = self.session.scalars(query).first() + assert pause_model is not None + assert pause_model.resumed_at is None + assert pause_model.id == pause_entity.id + + self.session.refresh(workflow_run) + assert workflow_run.status == WorkflowExecutionStatus.PAUSED + + # Act - Get pause state + retrieved_entity = repository.get_workflow_pause(workflow_run.id) + + # Assert - Pause state retrieved + assert retrieved_entity is not None + assert retrieved_entity.id == pause_entity.id + retrieved_state = retrieved_entity.get_state() + if isinstance(retrieved_state, bytes): + retrieved_state = retrieved_state.decode() + assert retrieved_state == test_state + + # Act - Resume workflow + resumed_entity = repository.resume_workflow_pause( + workflow_run_id=workflow_run.id, + pause_entity=pause_entity, + ) + + # Assert - Workflow resumed + assert resumed_entity is not None + assert resumed_entity.id == pause_entity.id + assert resumed_entity.resumed_at is not None + + # Verify database state + self.session.refresh(workflow_run) + assert workflow_run.status == WorkflowExecutionStatus.RUNNING + self.session.refresh(pause_model) + assert pause_model.resumed_at is not None + + # Act - Delete pause state + repository.delete_workflow_pause(pause_entity) + + # Assert - Pause state deleted + with Session(bind=self.session.get_bind()) as session: + deleted_pause = session.get(WorkflowPauseModel, pause_entity.id) + assert deleted_pause is None + + def test_pause_workflow_success(self): + """Test successful pause workflow scenarios.""" + workflow_run = self._create_test_workflow_run(status=WorkflowExecutionStatus.RUNNING) + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + assert pause_entity is not None + assert pause_entity.workflow_execution_id == workflow_run.id + + retrieved_state = pause_entity.get_state() + if isinstance(retrieved_state, bytes): + retrieved_state = retrieved_state.decode() + assert retrieved_state == test_state + + self.session.refresh(workflow_run) + assert workflow_run.status == WorkflowExecutionStatus.PAUSED + pause_query = select(WorkflowPauseModel).where(WorkflowPauseModel.workflow_run_id == workflow_run.id) + pause_model = self.session.scalars(pause_query).first() + assert pause_model is not None + assert pause_model.id == pause_entity.id + assert pause_model.resumed_at is None + + @pytest.mark.parametrize("test_case", pause_workflow_failure_cases(), ids=lambda tc: tc.name) + def test_pause_workflow_failure(self, test_case: PauseWorkflowFailureCase): + """Test pause workflow failure scenarios.""" + workflow_run = self._create_test_workflow_run(status=test_case.initial_status) + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + with pytest.raises(_WorkflowRunError): + repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + @pytest.mark.parametrize("test_case", resume_workflow_success_cases(), ids=lambda tc: tc.name) + def test_resume_workflow_success(self, test_case: ResumeWorkflowSuccessCase): + """Test successful resume workflow scenarios.""" + workflow_run = self._create_test_workflow_run(status=test_case.initial_status) + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + if workflow_run.status != WorkflowExecutionStatus.RUNNING: + workflow_run.status = WorkflowExecutionStatus.RUNNING + self.session.commit() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + self.session.refresh(workflow_run) + assert workflow_run.status == WorkflowExecutionStatus.PAUSED + + resumed_entity = repository.resume_workflow_pause( + workflow_run_id=workflow_run.id, + pause_entity=pause_entity, + ) + assert resumed_entity is not None + assert resumed_entity.id == pause_entity.id + assert resumed_entity.resumed_at is not None + + self.session.refresh(workflow_run) + assert workflow_run.status == WorkflowExecutionStatus.RUNNING + pause_query = select(WorkflowPauseModel).where(WorkflowPauseModel.workflow_run_id == workflow_run.id) + pause_model = self.session.scalars(pause_query).first() + assert pause_model is not None + assert pause_model.id == pause_entity.id + assert pause_model.resumed_at is not None + + def test_resume_running_workflow(self): + """Test resume workflow failure scenarios.""" + workflow_run = self._create_test_workflow_run(status=WorkflowExecutionStatus.RUNNING) + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + self.session.refresh(workflow_run) + workflow_run.status = WorkflowExecutionStatus.RUNNING + self.session.add(workflow_run) + self.session.commit() + + with pytest.raises(_WorkflowRunError): + repository.resume_workflow_pause( + workflow_run_id=workflow_run.id, + pause_entity=pause_entity, + ) + + def test_resume_resumed_pause(self): + """Test resume workflow failure scenarios.""" + workflow_run = self._create_test_workflow_run(status=WorkflowExecutionStatus.RUNNING) + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + pause_model = self.session.get(WorkflowPauseModel, pause_entity.id) + pause_model.resumed_at = naive_utc_now() + self.session.add(pause_model) + self.session.commit() + + with pytest.raises(_WorkflowRunError): + repository.resume_workflow_pause( + workflow_run_id=workflow_run.id, + pause_entity=pause_entity, + ) + + # ==================== Error Scenario Tests ==================== + + def test_pause_nonexistent_workflow_run(self): + """Test pausing a non-existent workflow run.""" + # Arrange + nonexistent_id = str(uuid.uuid4()) + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + # Act & Assert + with pytest.raises(ValueError, match="WorkflowRun not found"): + repository.create_workflow_pause( + workflow_run_id=nonexistent_id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + def test_resume_nonexistent_workflow_run(self): + """Test resuming a non-existent workflow run.""" + # Arrange + workflow_run = self._create_test_workflow_run() + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + nonexistent_id = str(uuid.uuid4()) + + # Act & Assert + with pytest.raises(ValueError, match="WorkflowRun not found"): + repository.resume_workflow_pause( + workflow_run_id=nonexistent_id, + pause_entity=pause_entity, + ) + + # ==================== Prune Functionality Tests ==================== + + @pytest.mark.parametrize("test_case", prune_pauses_test_cases(), ids=lambda tc: tc.name) + def test_prune_pauses_scenarios(self, test_case: PrunePausesTestCase): + """Test various prune pauses scenarios.""" + now = naive_utc_now() + + # Create pause state + workflow_run = self._create_test_workflow_run() + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + # Manually adjust timestamps for testing + pause_model = self.session.get(WorkflowPauseModel, pause_entity.id) + pause_model.created_at = now - test_case.pause_age + + if test_case.resume_age is not None: + # Resume pause and adjust resume time + repository.resume_workflow_pause( + workflow_run_id=workflow_run.id, + pause_entity=pause_entity, + ) + # Need to refresh to get the updated model + self.session.refresh(pause_model) + # Manually set the resumed_at to an older time for testing + pause_model.resumed_at = now - test_case.resume_age + self.session.commit() # Commit the resumed_at change + # Refresh again to ensure the change is persisted + self.session.refresh(pause_model) + + self.session.commit() + + # Act - Prune pauses + expiration_time = now - timedelta(days=1, seconds=1) # Expire pauses older than 1 day (plus 1 second) + resumption_time = now - timedelta( + days=7, seconds=1 + ) # Clean up pauses resumed more than 7 days ago (plus 1 second) + + # Debug: Check pause state before pruning + self.session.refresh(pause_model) + print(f"Pause created_at: {pause_model.created_at}") + print(f"Pause resumed_at: {pause_model.resumed_at}") + print(f"Expiration time: {expiration_time}") + print(f"Resumption time: {resumption_time}") + + # Force commit to ensure timestamps are saved + self.session.commit() + + # Determine if the pause should be pruned based on timestamps + should_be_pruned = False + if test_case.resume_age is not None: + # If resumed, check if resumed_at is older than resumption_time + should_be_pruned = pause_model.resumed_at < resumption_time + else: + # If not resumed, check if created_at is older than expiration_time + should_be_pruned = pause_model.created_at < expiration_time + + # Act - Prune pauses + pruned_ids = repository.prune_pauses( + expiration=expiration_time, + resumption_expiration=resumption_time, + ) + + # Assert - Check pruning results + if should_be_pruned: + assert len(pruned_ids) == test_case.expected_pruned_count + # Verify pause was actually deleted + # The pause should be in the pruned_ids list if it was pruned + assert pause_entity.id in pruned_ids + else: + assert len(pruned_ids) == 0 + + def test_prune_pauses_with_limit(self): + """Test prune pauses with limit parameter.""" + now = naive_utc_now() + + # Create multiple pause states + pause_entities = [] + repository = self._get_workflow_run_repository() + + for i in range(5): + workflow_run = self._create_test_workflow_run() + test_state = self._create_test_state() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + pause_entities.append(pause_entity) + + # Make all pauses old enough to be pruned + pause_model = self.session.get(WorkflowPauseModel, pause_entity.id) + pause_model.created_at = now - timedelta(days=7) + + self.session.commit() + + # Act - Prune with limit + expiration_time = now - timedelta(days=1) + resumption_time = now - timedelta(days=7) + + pruned_ids = repository.prune_pauses( + expiration=expiration_time, + resumption_expiration=resumption_time, + limit=3, + ) + + # Assert + assert len(pruned_ids) == 3 + + # Verify only 3 were deleted + remaining_count = ( + self.session.query(WorkflowPauseModel) + .filter(WorkflowPauseModel.id.in_([pe.id for pe in pause_entities])) + .count() + ) + assert remaining_count == 2 + + # ==================== Multi-tenant Isolation Tests ==================== + + def test_multi_tenant_pause_isolation(self): + """Test that pause states are properly isolated by tenant.""" + # Arrange - Create second tenant + + tenant2 = Tenant( + name="Test Tenant 2", + status="normal", + ) + self.session.add(tenant2) + self.session.commit() + + account2 = Account( + email="test2@example.com", + name="Test User 2", + interface_language="en-US", + status="active", + ) + self.session.add(account2) + self.session.commit() + + tenant2_join = TenantAccountJoin( + tenant_id=tenant2.id, + account_id=account2.id, + role=TenantAccountRole.OWNER, + current=True, + ) + self.session.add(tenant2_join) + self.session.commit() + + # Create workflow for tenant 2 + workflow2 = Workflow( + id=str(uuid.uuid4()), + tenant_id=tenant2.id, + app_id=str(uuid.uuid4()), + type="workflow", + version="draft", + graph='{"nodes": [], "edges": []}', + features='{"file_upload": {"enabled": false}}', + created_by=account2.id, + created_at=naive_utc_now(), + ) + self.session.add(workflow2) + self.session.commit() + + # Create workflow runs for both tenants + workflow_run1 = self._create_test_workflow_run() + workflow_run2 = WorkflowRun( + id=str(uuid.uuid4()), + tenant_id=tenant2.id, + app_id=workflow2.app_id, + workflow_id=workflow2.id, + type="workflow", + triggered_from="debugging", + version="draft", + status=WorkflowExecutionStatus.RUNNING, + created_by=account2.id, + created_by_role="account", + created_at=naive_utc_now(), + ) + self.session.add(workflow_run2) + self.session.commit() + + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + # Act - Create pause for tenant 1 + pause_entity1 = repository.create_workflow_pause( + workflow_run_id=workflow_run1.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + # Try to access pause from tenant 2 using tenant 1's repository + # This should work because we're using the same repository + pause_entity2 = repository.get_workflow_pause(workflow_run2.id) + assert pause_entity2 is None # No pause for tenant 2 yet + + # Create pause for tenant 2 + pause_entity2 = repository.create_workflow_pause( + workflow_run_id=workflow_run2.id, + state_owner_user_id=account2.id, + state=test_state, + ) + + # Assert - Both pauses should exist and be separate + assert pause_entity1 is not None + assert pause_entity2 is not None + assert pause_entity1.id != pause_entity2.id + assert pause_entity1.workflow_execution_id != pause_entity2.workflow_execution_id + + def test_cross_tenant_access_restriction(self): + """Test that cross-tenant access is properly restricted.""" + # This test would require tenant-specific repositories + # For now, we test that pause entities are properly scoped by tenant_id + workflow_run = self._create_test_workflow_run() + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + # Verify pause is properly scoped + pause_model = self.session.get(WorkflowPauseModel, pause_entity.id) + assert pause_model.workflow_id == self.test_workflow_id + + # ==================== File Storage Integration Tests ==================== + + def test_file_storage_integration(self): + """Test that state files are properly stored and retrieved.""" + # Arrange + workflow_run = self._create_test_workflow_run() + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + # Act - Create pause state + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + # Assert - Verify file was uploaded to storage + pause_model = self.session.get(WorkflowPauseModel, pause_entity.id) + assert pause_model.state_object_key != "" + + # Verify file content in storage + + file_key = pause_model.state_object_key + storage_content = storage.load(file_key).decode() + assert storage_content == test_state + + # Verify retrieval through entity + retrieved_state = pause_entity.get_state() + if isinstance(retrieved_state, bytes): + retrieved_state = retrieved_state.decode() + assert retrieved_state == test_state + + def test_file_cleanup_on_pause_deletion(self): + """Test that files are properly handled on pause deletion.""" + # Arrange + workflow_run = self._create_test_workflow_run() + test_state = self._create_test_state() + repository = self._get_workflow_run_repository() + + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=test_state, + ) + + # Get file info before deletion + pause_model = self.session.get(WorkflowPauseModel, pause_entity.id) + file_key = pause_model.state_object_key + + # Act - Delete pause state + repository.delete_workflow_pause(pause_entity) + + # Assert - Pause record should be deleted + self.session.expire_all() # Clear session to ensure fresh query + deleted_pause = self.session.get(WorkflowPauseModel, pause_entity.id) + assert deleted_pause is None + + try: + content = storage.load(file_key).decode() + pytest.fail("File should be deleted from storage after pause deletion") + except FileNotFoundError: + # This is expected - file should be deleted from storage + pass + except Exception as e: + pytest.fail(f"Unexpected error when checking file deletion: {e}") + + def test_large_state_file_handling(self): + """Test handling of large state files.""" + # Arrange - Create a large state (1MB) + large_state = "x" * (1024 * 1024) # 1MB of data + large_state_json = json.dumps({"large_data": large_state}) + + workflow_run = self._create_test_workflow_run() + repository = self._get_workflow_run_repository() + + # Act + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=large_state_json, + ) + + # Assert + assert pause_entity is not None + retrieved_state = pause_entity.get_state() + if isinstance(retrieved_state, bytes): + retrieved_state = retrieved_state.decode() + assert retrieved_state == large_state_json + + # Verify file size in database + pause_model = self.session.get(WorkflowPauseModel, pause_entity.id) + assert pause_model.state_object_key != "" + loaded_state = storage.load(pause_model.state_object_key) + assert loaded_state.decode() == large_state_json + + def test_multiple_pause_resume_cycles(self): + """Test multiple pause/resume cycles on the same workflow run.""" + # Arrange + workflow_run = self._create_test_workflow_run() + repository = self._get_workflow_run_repository() + + # Act & Assert - Multiple cycles + for i in range(3): + state = json.dumps({"cycle": i, "data": f"state_{i}"}) + + # Reset workflow run status to RUNNING before each pause (after first cycle) + if i > 0: + self.session.refresh(workflow_run) # Refresh to get latest state from session + workflow_run.status = WorkflowExecutionStatus.RUNNING + self.session.commit() + self.session.refresh(workflow_run) # Refresh again after commit + + # Pause + pause_entity = repository.create_workflow_pause( + workflow_run_id=workflow_run.id, + state_owner_user_id=self.test_user_id, + state=state, + ) + assert pause_entity is not None + + # Verify pause + self.session.expire_all() # Clear session to ensure fresh query + self.session.refresh(workflow_run) + + # Use the test session directly to verify the pause + stmt = select(WorkflowRun).options(selectinload(WorkflowRun.pause)).where(WorkflowRun.id == workflow_run.id) + workflow_run_with_pause = self.session.scalar(stmt) + pause_model = workflow_run_with_pause.pause + + # Verify pause using test session directly + assert pause_model is not None + assert pause_model.id == pause_entity.id + assert pause_model.state_object_key != "" + + # Load file content using storage directly + file_content = storage.load(pause_model.state_object_key) + if isinstance(file_content, bytes): + file_content = file_content.decode() + assert file_content == state + + # Resume + resumed_entity = repository.resume_workflow_pause( + workflow_run_id=workflow_run.id, + pause_entity=pause_entity, + ) + assert resumed_entity is not None + assert resumed_entity.resumed_at is not None + + # Verify resume - check that pause is marked as resumed + self.session.expire_all() # Clear session to ensure fresh query + stmt = select(WorkflowPauseModel).where(WorkflowPauseModel.id == pause_entity.id) + resumed_pause_model = self.session.scalar(stmt) + assert resumed_pause_model is not None + assert resumed_pause_model.resumed_at is not None + + # Verify workflow run status + self.session.refresh(workflow_run) + assert workflow_run.status == WorkflowExecutionStatus.RUNNING diff --git a/api/tests/unit_tests/core/app/apps/common/test_workflow_response_converter_process_data.py b/api/tests/unit_tests/core/app/apps/common/test_workflow_response_converter_process_data.py deleted file mode 100644 index abe09fb8a4..0000000000 --- a/api/tests/unit_tests/core/app/apps/common/test_workflow_response_converter_process_data.py +++ /dev/null @@ -1,324 +0,0 @@ -""" -Unit tests for WorkflowResponseConverter focusing on process_data truncation functionality. -""" - -import uuid -from collections.abc import Mapping -from typing import Any -from unittest.mock import Mock - -import pytest - -from core.app.apps.common.workflow_response_converter import WorkflowResponseConverter -from core.app.entities.app_invoke_entities import WorkflowAppGenerateEntity -from core.app.entities.queue_entities import ( - QueueNodeRetryEvent, - QueueNodeStartedEvent, - QueueNodeSucceededEvent, -) -from core.workflow.enums import NodeType -from core.workflow.system_variable import SystemVariable -from libs.datetime_utils import naive_utc_now -from models import Account - - -class TestWorkflowResponseConverterCenarios: - """Test process_data truncation in WorkflowResponseConverter.""" - - def create_mock_generate_entity(self) -> WorkflowAppGenerateEntity: - """Create a mock WorkflowAppGenerateEntity.""" - mock_entity = Mock(spec=WorkflowAppGenerateEntity) - mock_app_config = Mock() - mock_app_config.tenant_id = "test-tenant-id" - mock_entity.app_config = mock_app_config - mock_entity.inputs = {} - return mock_entity - - def create_workflow_response_converter(self) -> WorkflowResponseConverter: - """Create a WorkflowResponseConverter for testing.""" - - mock_entity = self.create_mock_generate_entity() - mock_user = Mock(spec=Account) - mock_user.id = "test-user-id" - mock_user.name = "Test User" - mock_user.email = "test@example.com" - - system_variables = SystemVariable(workflow_id="wf-id", workflow_execution_id="initial-run-id") - return WorkflowResponseConverter( - application_generate_entity=mock_entity, - user=mock_user, - system_variables=system_variables, - ) - - def create_node_started_event(self, *, node_execution_id: str | None = None) -> QueueNodeStartedEvent: - """Create a QueueNodeStartedEvent for testing.""" - return QueueNodeStartedEvent( - node_execution_id=node_execution_id or str(uuid.uuid4()), - node_id="test-node-id", - node_title="Test Node", - node_type=NodeType.CODE, - start_at=naive_utc_now(), - predecessor_node_id=None, - in_iteration_id=None, - in_loop_id=None, - provider_type="built-in", - provider_id="code", - ) - - def create_node_succeeded_event( - self, - *, - node_execution_id: str, - process_data: Mapping[str, Any] | None = None, - ) -> QueueNodeSucceededEvent: - """Create a QueueNodeSucceededEvent for testing.""" - return QueueNodeSucceededEvent( - node_id="test-node-id", - node_type=NodeType.CODE, - node_execution_id=node_execution_id, - start_at=naive_utc_now(), - in_iteration_id=None, - in_loop_id=None, - inputs={}, - process_data=process_data or {}, - outputs={}, - execution_metadata={}, - ) - - def create_node_retry_event( - self, - *, - node_execution_id: str, - process_data: Mapping[str, Any] | None = None, - ) -> QueueNodeRetryEvent: - """Create a QueueNodeRetryEvent for testing.""" - return QueueNodeRetryEvent( - inputs={"data": "inputs"}, - outputs={"data": "outputs"}, - process_data=process_data or {}, - error="oops", - retry_index=1, - node_id="test-node-id", - node_type=NodeType.CODE, - node_title="test code", - provider_type="built-in", - provider_id="code", - node_execution_id=node_execution_id, - start_at=naive_utc_now(), - in_iteration_id=None, - in_loop_id=None, - ) - - def test_workflow_node_finish_response_uses_truncated_process_data(self): - """Test that node finish response uses get_response_process_data().""" - converter = self.create_workflow_response_converter() - - original_data = {"large_field": "x" * 10000, "metadata": "info"} - truncated_data = {"large_field": "[TRUNCATED]", "metadata": "info"} - - converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") - start_event = self.create_node_started_event() - converter.workflow_node_start_to_stream_response( - event=start_event, - task_id="test-task-id", - ) - - event = self.create_node_succeeded_event( - node_execution_id=start_event.node_execution_id, - process_data=original_data, - ) - - def fake_truncate(mapping): - if mapping == dict(original_data): - return truncated_data, True - return mapping, False - - converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] - - response = converter.workflow_node_finish_to_stream_response( - event=event, - task_id="test-task-id", - ) - - # Response should use truncated data, not original - assert response is not None - assert response.data.process_data == truncated_data - assert response.data.process_data != original_data - assert response.data.process_data_truncated is True - - def test_workflow_node_finish_response_without_truncation(self): - """Test node finish response when no truncation is applied.""" - converter = self.create_workflow_response_converter() - - original_data = {"small": "data"} - - converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") - start_event = self.create_node_started_event() - converter.workflow_node_start_to_stream_response( - event=start_event, - task_id="test-task-id", - ) - - event = self.create_node_succeeded_event( - node_execution_id=start_event.node_execution_id, - process_data=original_data, - ) - - def fake_truncate(mapping): - return mapping, False - - converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] - - response = converter.workflow_node_finish_to_stream_response( - event=event, - task_id="test-task-id", - ) - - # Response should use original data - assert response is not None - assert response.data.process_data == original_data - assert response.data.process_data_truncated is False - - def test_workflow_node_finish_response_with_none_process_data(self): - """Test node finish response when process_data is None.""" - converter = self.create_workflow_response_converter() - - converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") - start_event = self.create_node_started_event() - converter.workflow_node_start_to_stream_response( - event=start_event, - task_id="test-task-id", - ) - - event = self.create_node_succeeded_event( - node_execution_id=start_event.node_execution_id, - process_data=None, - ) - - def fake_truncate(mapping): - return mapping, False - - converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] - - response = converter.workflow_node_finish_to_stream_response( - event=event, - task_id="test-task-id", - ) - - # Response should normalize missing process_data to an empty mapping - assert response is not None - assert response.data.process_data == {} - assert response.data.process_data_truncated is False - - def test_workflow_node_retry_response_uses_truncated_process_data(self): - """Test that node retry response uses get_response_process_data().""" - converter = self.create_workflow_response_converter() - - original_data = {"large_field": "x" * 10000, "metadata": "info"} - truncated_data = {"large_field": "[TRUNCATED]", "metadata": "info"} - - converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") - start_event = self.create_node_started_event() - converter.workflow_node_start_to_stream_response( - event=start_event, - task_id="test-task-id", - ) - - event = self.create_node_retry_event( - node_execution_id=start_event.node_execution_id, - process_data=original_data, - ) - - def fake_truncate(mapping): - if mapping == dict(original_data): - return truncated_data, True - return mapping, False - - converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] - - response = converter.workflow_node_retry_to_stream_response( - event=event, - task_id="test-task-id", - ) - - # Response should use truncated data, not original - assert response is not None - assert response.data.process_data == truncated_data - assert response.data.process_data != original_data - assert response.data.process_data_truncated is True - - def test_workflow_node_retry_response_without_truncation(self): - """Test node retry response when no truncation is applied.""" - converter = self.create_workflow_response_converter() - - original_data = {"small": "data"} - - converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") - start_event = self.create_node_started_event() - converter.workflow_node_start_to_stream_response( - event=start_event, - task_id="test-task-id", - ) - - event = self.create_node_retry_event( - node_execution_id=start_event.node_execution_id, - process_data=original_data, - ) - - def fake_truncate(mapping): - return mapping, False - - converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] - - response = converter.workflow_node_retry_to_stream_response( - event=event, - task_id="test-task-id", - ) - - assert response is not None - assert response.data.process_data == original_data - assert response.data.process_data_truncated is False - - def test_iteration_and_loop_nodes_return_none(self): - """Test that iteration and loop nodes return None (no streaming events).""" - converter = self.create_workflow_response_converter() - - iteration_event = QueueNodeSucceededEvent( - node_id="iteration-node", - node_type=NodeType.ITERATION, - node_execution_id=str(uuid.uuid4()), - start_at=naive_utc_now(), - in_iteration_id=None, - in_loop_id=None, - inputs={}, - process_data={}, - outputs={}, - execution_metadata={}, - ) - - response = converter.workflow_node_finish_to_stream_response( - event=iteration_event, - task_id="test-task-id", - ) - assert response is None - - loop_event = iteration_event.model_copy(update={"node_type": NodeType.LOOP}) - response = converter.workflow_node_finish_to_stream_response( - event=loop_event, - task_id="test-task-id", - ) - assert response is None - - def test_finish_without_start_raises(self): - """Ensure finish responses require a prior workflow start.""" - converter = self.create_workflow_response_converter() - event = self.create_node_succeeded_event( - node_execution_id=str(uuid.uuid4()), - process_data={}, - ) - - with pytest.raises(ValueError): - converter.workflow_node_finish_to_stream_response( - event=event, - task_id="test-task-id", - ) diff --git a/api/tests/unit_tests/core/app/apps/common/test_workflow_response_converter_truncation.py b/api/tests/unit_tests/core/app/apps/common/test_workflow_response_converter_truncation.py new file mode 100644 index 0000000000..1c9f577a50 --- /dev/null +++ b/api/tests/unit_tests/core/app/apps/common/test_workflow_response_converter_truncation.py @@ -0,0 +1,810 @@ +""" +Unit tests for WorkflowResponseConverter focusing on process_data truncation functionality. +""" + +import uuid +from collections.abc import Mapping +from dataclasses import dataclass +from typing import Any +from unittest.mock import Mock + +import pytest + +from core.app.app_config.entities import WorkflowUIBasedAppConfig +from core.app.apps.common.workflow_response_converter import WorkflowResponseConverter +from core.app.entities.app_invoke_entities import InvokeFrom, WorkflowAppGenerateEntity +from core.app.entities.queue_entities import ( + QueueEvent, + QueueIterationStartEvent, + QueueLoopStartEvent, + QueueNodeExceptionEvent, + QueueNodeFailedEvent, + QueueNodeRetryEvent, + QueueNodeStartedEvent, + QueueNodeSucceededEvent, +) +from core.workflow.enums import NodeType +from core.workflow.system_variable import SystemVariable +from libs.datetime_utils import naive_utc_now +from models import Account +from models.model import AppMode + + +class TestWorkflowResponseConverter: + """Test truncation in WorkflowResponseConverter.""" + + def create_mock_generate_entity(self) -> WorkflowAppGenerateEntity: + """Create a mock WorkflowAppGenerateEntity.""" + mock_entity = Mock(spec=WorkflowAppGenerateEntity) + mock_app_config = Mock() + mock_app_config.tenant_id = "test-tenant-id" + mock_entity.invoke_from = InvokeFrom.WEB_APP + mock_entity.app_config = mock_app_config + mock_entity.inputs = {} + return mock_entity + + def create_workflow_response_converter(self) -> WorkflowResponseConverter: + """Create a WorkflowResponseConverter for testing.""" + + mock_entity = self.create_mock_generate_entity() + mock_user = Mock(spec=Account) + mock_user.id = "test-user-id" + mock_user.name = "Test User" + mock_user.email = "test@example.com" + + system_variables = SystemVariable(workflow_id="wf-id", workflow_execution_id="initial-run-id") + return WorkflowResponseConverter( + application_generate_entity=mock_entity, + user=mock_user, + system_variables=system_variables, + ) + + def create_node_started_event(self, *, node_execution_id: str | None = None) -> QueueNodeStartedEvent: + """Create a QueueNodeStartedEvent for testing.""" + return QueueNodeStartedEvent( + node_execution_id=node_execution_id or str(uuid.uuid4()), + node_id="test-node-id", + node_title="Test Node", + node_type=NodeType.CODE, + start_at=naive_utc_now(), + in_iteration_id=None, + in_loop_id=None, + provider_type="built-in", + provider_id="code", + ) + + def create_node_succeeded_event( + self, + *, + node_execution_id: str, + process_data: Mapping[str, Any] | None = None, + ) -> QueueNodeSucceededEvent: + """Create a QueueNodeSucceededEvent for testing.""" + return QueueNodeSucceededEvent( + node_id="test-node-id", + node_type=NodeType.CODE, + node_execution_id=node_execution_id, + start_at=naive_utc_now(), + in_iteration_id=None, + in_loop_id=None, + inputs={}, + process_data=process_data or {}, + outputs={}, + execution_metadata={}, + ) + + def create_node_retry_event( + self, + *, + node_execution_id: str, + process_data: Mapping[str, Any] | None = None, + ) -> QueueNodeRetryEvent: + """Create a QueueNodeRetryEvent for testing.""" + return QueueNodeRetryEvent( + inputs={"data": "inputs"}, + outputs={"data": "outputs"}, + process_data=process_data or {}, + error="oops", + retry_index=1, + node_id="test-node-id", + node_type=NodeType.CODE, + node_title="test code", + provider_type="built-in", + provider_id="code", + node_execution_id=node_execution_id, + start_at=naive_utc_now(), + in_iteration_id=None, + in_loop_id=None, + ) + + def test_workflow_node_finish_response_uses_truncated_process_data(self): + """Test that node finish response uses get_response_process_data().""" + converter = self.create_workflow_response_converter() + + original_data = {"large_field": "x" * 10000, "metadata": "info"} + truncated_data = {"large_field": "[TRUNCATED]", "metadata": "info"} + + converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") + start_event = self.create_node_started_event() + converter.workflow_node_start_to_stream_response( + event=start_event, + task_id="test-task-id", + ) + + event = self.create_node_succeeded_event( + node_execution_id=start_event.node_execution_id, + process_data=original_data, + ) + + def fake_truncate(mapping): + if mapping == dict(original_data): + return truncated_data, True + return mapping, False + + converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test-task-id", + ) + + # Response should use truncated data, not original + assert response is not None + assert response.data.process_data == truncated_data + assert response.data.process_data != original_data + assert response.data.process_data_truncated is True + + def test_workflow_node_finish_response_without_truncation(self): + """Test node finish response when no truncation is applied.""" + converter = self.create_workflow_response_converter() + + original_data = {"small": "data"} + + converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") + start_event = self.create_node_started_event() + converter.workflow_node_start_to_stream_response( + event=start_event, + task_id="test-task-id", + ) + + event = self.create_node_succeeded_event( + node_execution_id=start_event.node_execution_id, + process_data=original_data, + ) + + def fake_truncate(mapping): + return mapping, False + + converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test-task-id", + ) + + # Response should use original data + assert response is not None + assert response.data.process_data == original_data + assert response.data.process_data_truncated is False + + def test_workflow_node_finish_response_with_none_process_data(self): + """Test node finish response when process_data is None.""" + converter = self.create_workflow_response_converter() + + converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") + start_event = self.create_node_started_event() + converter.workflow_node_start_to_stream_response( + event=start_event, + task_id="test-task-id", + ) + + event = self.create_node_succeeded_event( + node_execution_id=start_event.node_execution_id, + process_data=None, + ) + + def fake_truncate(mapping): + return mapping, False + + converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test-task-id", + ) + + # Response should normalize missing process_data to an empty mapping + assert response is not None + assert response.data.process_data == {} + assert response.data.process_data_truncated is False + + def test_workflow_node_retry_response_uses_truncated_process_data(self): + """Test that node retry response uses get_response_process_data().""" + converter = self.create_workflow_response_converter() + + original_data = {"large_field": "x" * 10000, "metadata": "info"} + truncated_data = {"large_field": "[TRUNCATED]", "metadata": "info"} + + converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") + start_event = self.create_node_started_event() + converter.workflow_node_start_to_stream_response( + event=start_event, + task_id="test-task-id", + ) + + event = self.create_node_retry_event( + node_execution_id=start_event.node_execution_id, + process_data=original_data, + ) + + def fake_truncate(mapping): + if mapping == dict(original_data): + return truncated_data, True + return mapping, False + + converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] + + response = converter.workflow_node_retry_to_stream_response( + event=event, + task_id="test-task-id", + ) + + # Response should use truncated data, not original + assert response is not None + assert response.data.process_data == truncated_data + assert response.data.process_data != original_data + assert response.data.process_data_truncated is True + + def test_workflow_node_retry_response_without_truncation(self): + """Test node retry response when no truncation is applied.""" + converter = self.create_workflow_response_converter() + + original_data = {"small": "data"} + + converter.workflow_start_to_stream_response(task_id="bootstrap", workflow_run_id="run-id", workflow_id="wf-id") + start_event = self.create_node_started_event() + converter.workflow_node_start_to_stream_response( + event=start_event, + task_id="test-task-id", + ) + + event = self.create_node_retry_event( + node_execution_id=start_event.node_execution_id, + process_data=original_data, + ) + + def fake_truncate(mapping): + return mapping, False + + converter._truncator.truncate_variable_mapping = fake_truncate # type: ignore[assignment] + + response = converter.workflow_node_retry_to_stream_response( + event=event, + task_id="test-task-id", + ) + + assert response is not None + assert response.data.process_data == original_data + assert response.data.process_data_truncated is False + + def test_iteration_and_loop_nodes_return_none(self): + """Test that iteration and loop nodes return None (no streaming events).""" + converter = self.create_workflow_response_converter() + + iteration_event = QueueNodeSucceededEvent( + node_id="iteration-node", + node_type=NodeType.ITERATION, + node_execution_id=str(uuid.uuid4()), + start_at=naive_utc_now(), + in_iteration_id=None, + in_loop_id=None, + inputs={}, + process_data={}, + outputs={}, + execution_metadata={}, + ) + + response = converter.workflow_node_finish_to_stream_response( + event=iteration_event, + task_id="test-task-id", + ) + assert response is None + + loop_event = iteration_event.model_copy(update={"node_type": NodeType.LOOP}) + response = converter.workflow_node_finish_to_stream_response( + event=loop_event, + task_id="test-task-id", + ) + assert response is None + + def test_finish_without_start_raises(self): + """Ensure finish responses require a prior workflow start.""" + converter = self.create_workflow_response_converter() + event = self.create_node_succeeded_event( + node_execution_id=str(uuid.uuid4()), + process_data={}, + ) + + with pytest.raises(ValueError): + converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test-task-id", + ) + + +@dataclass +class TestCase: + """Test case data for table-driven tests.""" + + name: str + invoke_from: InvokeFrom + expected_truncation_enabled: bool + description: str + + +class TestWorkflowResponseConverterServiceApiTruncation: + """Test class for Service API truncation functionality in WorkflowResponseConverter.""" + + def create_test_app_generate_entity(self, invoke_from: InvokeFrom) -> WorkflowAppGenerateEntity: + """Create a test WorkflowAppGenerateEntity with specified invoke_from.""" + # Create a minimal WorkflowUIBasedAppConfig for testing + app_config = WorkflowUIBasedAppConfig( + tenant_id="test_tenant", + app_id="test_app", + app_mode=AppMode.WORKFLOW, + workflow_id="test_workflow_id", + ) + + entity = WorkflowAppGenerateEntity( + task_id="test_task_id", + app_id="test_app_id", + app_config=app_config, + tenant_id="test_tenant", + app_mode=AppMode.WORKFLOW, + invoke_from=invoke_from, + inputs={"test_input": "test_value"}, + user_id="test_user_id", + stream=True, + files=[], + workflow_execution_id="test_workflow_exec_id", + ) + return entity + + def create_test_user(self) -> Account: + """Create a test user account.""" + account = Account( + name="Test User", + email="test@example.com", + ) + # Manually set the ID for testing purposes + account.id = "test_user_id" + return account + + def create_test_system_variables(self) -> SystemVariable: + """Create test system variables.""" + return SystemVariable() + + def create_test_converter(self, invoke_from: InvokeFrom) -> WorkflowResponseConverter: + """Create WorkflowResponseConverter with specified invoke_from.""" + entity = self.create_test_app_generate_entity(invoke_from) + user = self.create_test_user() + system_variables = self.create_test_system_variables() + + converter = WorkflowResponseConverter( + application_generate_entity=entity, + user=user, + system_variables=system_variables, + ) + # ensure `workflow_run_id` is set. + converter.workflow_start_to_stream_response( + task_id="test-task-id", + workflow_run_id="test-workflow-run-id", + workflow_id="test-workflow-id", + ) + return converter + + @pytest.mark.parametrize( + "test_case", + [ + TestCase( + name="service_api_truncation_disabled", + invoke_from=InvokeFrom.SERVICE_API, + expected_truncation_enabled=False, + description="Service API calls should have truncation disabled", + ), + TestCase( + name="web_app_truncation_enabled", + invoke_from=InvokeFrom.WEB_APP, + expected_truncation_enabled=True, + description="Web app calls should have truncation enabled", + ), + TestCase( + name="debugger_truncation_enabled", + invoke_from=InvokeFrom.DEBUGGER, + expected_truncation_enabled=True, + description="Debugger calls should have truncation enabled", + ), + TestCase( + name="explore_truncation_enabled", + invoke_from=InvokeFrom.EXPLORE, + expected_truncation_enabled=True, + description="Explore calls should have truncation enabled", + ), + TestCase( + name="published_truncation_enabled", + invoke_from=InvokeFrom.PUBLISHED, + expected_truncation_enabled=True, + description="Published app calls should have truncation enabled", + ), + ], + ids=lambda x: x.name, + ) + def test_truncator_selection_based_on_invoke_from(self, test_case: TestCase): + """Test that the correct truncator is selected based on invoke_from.""" + converter = self.create_test_converter(test_case.invoke_from) + + # Test truncation behavior instead of checking private attribute + + # Create a test event with large data + large_value = {"key": ["x"] * 2000} # Large data that would be truncated + + event = QueueNodeSucceededEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + start_at=naive_utc_now(), + inputs=large_value, + process_data=large_value, + outputs=large_value, + error=None, + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + ) + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test_task", + ) + + # Verify response is not None + assert response is not None + + # Verify truncation behavior matches expectations + if test_case.expected_truncation_enabled: + # Truncation should be enabled for non-service-api calls + assert response.data.inputs_truncated + assert response.data.process_data_truncated + assert response.data.outputs_truncated + else: + # SERVICE_API should not truncate + assert not response.data.inputs_truncated + assert not response.data.process_data_truncated + assert not response.data.outputs_truncated + + def test_service_api_truncator_no_op_mapping(self): + """Test that Service API truncator doesn't truncate variable mappings.""" + converter = self.create_test_converter(InvokeFrom.SERVICE_API) + + # Create a test event with large data + large_value: dict[str, Any] = { + "large_string": "x" * 10000, # Large string + "large_list": list(range(2000)), # Large array + "nested_data": {"deep_nested": {"very_deep": {"value": "x" * 5000}}}, + } + + event = QueueNodeSucceededEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + start_at=naive_utc_now(), + inputs=large_value, + process_data=large_value, + outputs=large_value, + error=None, + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + ) + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test_task", + ) + + # Verify response is not None + data = response.data + assert data.inputs == large_value + assert data.process_data == large_value + assert data.outputs == large_value + # Service API should not truncate + assert data.inputs_truncated is False + assert data.process_data_truncated is False + assert data.outputs_truncated is False + + def test_web_app_truncator_works_normally(self): + """Test that web app truncator still works normally.""" + converter = self.create_test_converter(InvokeFrom.WEB_APP) + + # Create a test event with large data + large_value = { + "large_string": "x" * 10000, # Large string + "large_list": list(range(2000)), # Large array + } + + event = QueueNodeSucceededEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + start_at=naive_utc_now(), + inputs=large_value, + process_data=large_value, + outputs=large_value, + error=None, + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + ) + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test_task", + ) + + # Verify response is not None + assert response is not None + + # Web app should truncate + data = response.data + assert data.inputs != large_value + assert data.process_data != large_value + assert data.outputs != large_value + # The exact behavior depends on VariableTruncator implementation + # Just verify that truncation flags are present + assert data.inputs_truncated is True + assert data.process_data_truncated is True + assert data.outputs_truncated is True + + @staticmethod + def _create_event_by_type( + type_: QueueEvent, inputs: Mapping[str, Any], process_data: Mapping[str, Any], outputs: Mapping[str, Any] + ) -> QueueNodeSucceededEvent | QueueNodeFailedEvent | QueueNodeExceptionEvent: + if type_ == QueueEvent.NODE_SUCCEEDED: + return QueueNodeSucceededEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + start_at=naive_utc_now(), + inputs=inputs, + process_data=process_data, + outputs=outputs, + error=None, + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + ) + elif type_ == QueueEvent.NODE_FAILED: + return QueueNodeFailedEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + start_at=naive_utc_now(), + inputs=inputs, + process_data=process_data, + outputs=outputs, + error="oops", + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + ) + elif type_ == QueueEvent.NODE_EXCEPTION: + return QueueNodeExceptionEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + start_at=naive_utc_now(), + inputs=inputs, + process_data=process_data, + outputs=outputs, + error="oops", + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + ) + else: + raise Exception("unknown type.") + + @pytest.mark.parametrize( + "event_type", + [ + QueueEvent.NODE_SUCCEEDED, + QueueEvent.NODE_FAILED, + QueueEvent.NODE_EXCEPTION, + ], + ) + def test_service_api_node_finish_event_no_truncation(self, event_type: QueueEvent): + """Test that Service API doesn't truncate node finish events.""" + converter = self.create_test_converter(InvokeFrom.SERVICE_API) + # Create test event with large data + large_inputs = {"input1": "x" * 5000, "input2": list(range(2000))} + large_process_data = {"process1": "y" * 5000, "process2": {"nested": ["z"] * 2000}} + large_outputs = {"output1": "result" * 1000, "output2": list(range(2000))} + + event = TestWorkflowResponseConverterServiceApiTruncation._create_event_by_type( + event_type, large_inputs, large_process_data, large_outputs + ) + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test_task", + ) + + # Verify response is not None + assert response is not None + + # Verify response contains full data (not truncated) + assert response.data.inputs == large_inputs + assert response.data.process_data == large_process_data + assert response.data.outputs == large_outputs + assert not response.data.inputs_truncated + assert not response.data.process_data_truncated + assert not response.data.outputs_truncated + + def test_service_api_node_retry_event_no_truncation(self): + """Test that Service API doesn't truncate node retry events.""" + converter = self.create_test_converter(InvokeFrom.SERVICE_API) + + # Create test event with large data + large_inputs = {"retry_input": "x" * 5000} + large_process_data = {"retry_process": "y" * 5000} + large_outputs = {"retry_output": "z" * 5000} + + # First, we need to store a snapshot by simulating a start event + start_event = QueueNodeStartedEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + node_title="Test Node", + node_run_index=1, + start_at=naive_utc_now(), + in_iteration_id=None, + in_loop_id=None, + agent_strategy=None, + provider_type="plugin", + provider_id="test/test_plugin", + ) + converter.workflow_node_start_to_stream_response(event=start_event, task_id="test_task") + + # Now create retry event + event = QueueNodeRetryEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + node_title="Test Node", + node_run_index=1, + start_at=naive_utc_now(), + inputs=large_inputs, + process_data=large_process_data, + outputs=large_outputs, + error="Retry error", + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + retry_index=1, + provider_type="plugin", + provider_id="test/test_plugin", + ) + + response = converter.workflow_node_retry_to_stream_response( + event=event, + task_id="test_task", + ) + + # Verify response is not None + assert response is not None + + # Verify response contains full data (not truncated) + assert response.data.inputs == large_inputs + assert response.data.process_data == large_process_data + assert response.data.outputs == large_outputs + assert not response.data.inputs_truncated + assert not response.data.process_data_truncated + assert not response.data.outputs_truncated + + def test_service_api_iteration_events_no_truncation(self): + """Test that Service API doesn't truncate iteration events.""" + converter = self.create_test_converter(InvokeFrom.SERVICE_API) + + # Test iteration start event + large_value = {"iteration_input": ["x"] * 2000} + + start_event = QueueIterationStartEvent( + node_execution_id="test_iter_exec_id", + node_id="test_iteration", + node_type=NodeType.ITERATION, + node_title="Test Iteration", + node_run_index=0, + start_at=naive_utc_now(), + inputs=large_value, + metadata={}, + ) + + response = converter.workflow_iteration_start_to_stream_response( + task_id="test_task", + workflow_execution_id="test_workflow_exec_id", + event=start_event, + ) + + assert response is not None + assert response.data.inputs == large_value + assert not response.data.inputs_truncated + + def test_service_api_loop_events_no_truncation(self): + """Test that Service API doesn't truncate loop events.""" + converter = self.create_test_converter(InvokeFrom.SERVICE_API) + + # Test loop start event + large_inputs = {"loop_input": ["x"] * 2000} + + start_event = QueueLoopStartEvent( + node_execution_id="test_loop_exec_id", + node_id="test_loop", + node_type=NodeType.LOOP, + node_title="Test Loop", + start_at=naive_utc_now(), + inputs=large_inputs, + metadata={}, + node_run_index=0, + ) + + response = converter.workflow_loop_start_to_stream_response( + task_id="test_task", + workflow_execution_id="test_workflow_exec_id", + event=start_event, + ) + + assert response is not None + assert response.data.inputs == large_inputs + assert not response.data.inputs_truncated + + def test_web_app_node_finish_event_truncation_works(self): + """Test that web app still truncates node finish events.""" + converter = self.create_test_converter(InvokeFrom.WEB_APP) + + # Create test event with large data that should be truncated + large_inputs = {"input1": ["x"] * 2000} + large_process_data = {"process1": ["y"] * 2000} + large_outputs = {"output1": ["z"] * 2000} + + event = QueueNodeSucceededEvent( + node_execution_id="test_node_exec_id", + node_id="test_node", + node_type=NodeType.LLM, + start_at=naive_utc_now(), + inputs=large_inputs, + process_data=large_process_data, + outputs=large_outputs, + error=None, + execution_metadata=None, + in_iteration_id=None, + in_loop_id=None, + ) + + response = converter.workflow_node_finish_to_stream_response( + event=event, + task_id="test_task", + ) + + # Verify response is not None + assert response is not None + + # Verify response contains truncated data + # The exact behavior depends on VariableTruncator implementation + # Just verify truncation flags are set correctly (may or may not be truncated depending on size) + # At minimum, the truncation mechanism should work + assert isinstance(response.data.inputs, dict) + assert response.data.inputs_truncated + assert isinstance(response.data.process_data, dict) + assert response.data.process_data_truncated + assert isinstance(response.data.outputs, dict) + assert response.data.outputs_truncated diff --git a/api/tests/unit_tests/core/app/apps/test_workflow_app_generator.py b/api/tests/unit_tests/core/app/apps/test_workflow_app_generator.py new file mode 100644 index 0000000000..83ac3a5591 --- /dev/null +++ b/api/tests/unit_tests/core/app/apps/test_workflow_app_generator.py @@ -0,0 +1,19 @@ +from core.app.apps.workflow.app_generator import SKIP_PREPARE_USER_INPUTS_KEY, WorkflowAppGenerator + + +def test_should_prepare_user_inputs_defaults_to_true(): + args = {"inputs": {}} + + assert WorkflowAppGenerator()._should_prepare_user_inputs(args) + + +def test_should_prepare_user_inputs_skips_when_flag_truthy(): + args = {"inputs": {}, SKIP_PREPARE_USER_INPUTS_KEY: True} + + assert not WorkflowAppGenerator()._should_prepare_user_inputs(args) + + +def test_should_prepare_user_inputs_keeps_validation_when_flag_false(): + args = {"inputs": {}, SKIP_PREPARE_USER_INPUTS_KEY: False} + + assert WorkflowAppGenerator()._should_prepare_user_inputs(args) diff --git a/api/tests/unit_tests/core/app/layers/test_pause_state_persist_layer.py b/api/tests/unit_tests/core/app/layers/test_pause_state_persist_layer.py new file mode 100644 index 0000000000..807f5e0fa5 --- /dev/null +++ b/api/tests/unit_tests/core/app/layers/test_pause_state_persist_layer.py @@ -0,0 +1,410 @@ +import json +from time import time +from unittest.mock import Mock + +import pytest + +from core.app.app_config.entities import WorkflowUIBasedAppConfig +from core.app.entities.app_invoke_entities import AdvancedChatAppGenerateEntity, InvokeFrom, WorkflowAppGenerateEntity +from core.app.layers.pause_state_persist_layer import ( + PauseStatePersistenceLayer, + WorkflowResumptionContext, + _AdvancedChatAppGenerateEntityWrapper, + _WorkflowGenerateEntityWrapper, +) +from core.variables.segments import Segment +from core.workflow.entities.pause_reason import SchedulingPause +from core.workflow.graph_engine.entities.commands import GraphEngineCommand +from core.workflow.graph_events.graph import ( + GraphRunFailedEvent, + GraphRunPausedEvent, + GraphRunStartedEvent, + GraphRunSucceededEvent, +) +from core.workflow.runtime.graph_runtime_state_protocol import ReadOnlyVariablePool +from models.model import AppMode +from repositories.factory import DifyAPIRepositoryFactory + + +class TestDataFactory: + """Factory helpers for constructing graph events used in tests.""" + + @staticmethod + def create_graph_run_paused_event(outputs: dict[str, object] | None = None) -> GraphRunPausedEvent: + return GraphRunPausedEvent(reason=SchedulingPause(message="test pause"), outputs=outputs or {}) + + @staticmethod + def create_graph_run_started_event() -> GraphRunStartedEvent: + return GraphRunStartedEvent() + + @staticmethod + def create_graph_run_succeeded_event(outputs: dict[str, object] | None = None) -> GraphRunSucceededEvent: + return GraphRunSucceededEvent(outputs=outputs or {}) + + @staticmethod + def create_graph_run_failed_event( + error: str = "Test error", + exceptions_count: int = 1, + ) -> GraphRunFailedEvent: + return GraphRunFailedEvent(error=error, exceptions_count=exceptions_count) + + +class MockSystemVariableReadOnlyView: + """Minimal read-only system variable view for testing.""" + + def __init__(self, workflow_execution_id: str | None = None) -> None: + self._workflow_execution_id = workflow_execution_id + + @property + def workflow_execution_id(self) -> str | None: + return self._workflow_execution_id + + +class MockReadOnlyVariablePool: + """Mock implementation of ReadOnlyVariablePool for testing.""" + + def __init__(self, variables: dict[tuple[str, str], object] | None = None): + self._variables = variables or {} + + def get(self, node_id: str, variable_key: str) -> Segment | None: + value = self._variables.get((node_id, variable_key)) + if value is None: + return None + mock_segment = Mock(spec=Segment) + mock_segment.value = value + return mock_segment + + def get_all_by_node(self, node_id: str) -> dict[str, object]: + return {key: value for (nid, key), value in self._variables.items() if nid == node_id} + + def get_by_prefix(self, prefix: str) -> dict[str, object]: + return {f"{nid}.{key}": value for (nid, key), value in self._variables.items() if nid.startswith(prefix)} + + +class MockReadOnlyGraphRuntimeState: + """Mock implementation of ReadOnlyGraphRuntimeState for testing.""" + + def __init__( + self, + start_at: float | None = None, + total_tokens: int = 0, + node_run_steps: int = 0, + ready_queue_size: int = 0, + exceptions_count: int = 0, + outputs: dict[str, object] | None = None, + variables: dict[tuple[str, str], object] | None = None, + workflow_execution_id: str | None = None, + ): + self._start_at = start_at or time() + self._total_tokens = total_tokens + self._node_run_steps = node_run_steps + self._ready_queue_size = ready_queue_size + self._exceptions_count = exceptions_count + self._outputs = outputs or {} + self._variable_pool = MockReadOnlyVariablePool(variables) + self._system_variable = MockSystemVariableReadOnlyView(workflow_execution_id) + + @property + def system_variable(self) -> MockSystemVariableReadOnlyView: + return self._system_variable + + @property + def variable_pool(self) -> ReadOnlyVariablePool: + return self._variable_pool + + @property + def start_at(self) -> float: + return self._start_at + + @property + def total_tokens(self) -> int: + return self._total_tokens + + @property + def node_run_steps(self) -> int: + return self._node_run_steps + + @property + def ready_queue_size(self) -> int: + return self._ready_queue_size + + @property + def exceptions_count(self) -> int: + return self._exceptions_count + + @property + def outputs(self) -> dict[str, object]: + return self._outputs.copy() + + @property + def llm_usage(self): + mock_usage = Mock() + mock_usage.prompt_tokens = 10 + mock_usage.completion_tokens = 20 + mock_usage.total_tokens = 30 + return mock_usage + + def get_output(self, key: str, default: object = None) -> object: + return self._outputs.get(key, default) + + def dumps(self) -> str: + return json.dumps( + { + "start_at": self._start_at, + "total_tokens": self._total_tokens, + "node_run_steps": self._node_run_steps, + "ready_queue_size": self._ready_queue_size, + "exceptions_count": self._exceptions_count, + "outputs": self._outputs, + "variables": {f"{k[0]}.{k[1]}": v for k, v in self._variable_pool._variables.items()}, + "workflow_execution_id": self._system_variable.workflow_execution_id, + } + ) + + +class MockCommandChannel: + """Mock implementation of CommandChannel for testing.""" + + def __init__(self): + self._commands: list[GraphEngineCommand] = [] + + def fetch_commands(self) -> list[GraphEngineCommand]: + return self._commands.copy() + + def send_command(self, command: GraphEngineCommand) -> None: + self._commands.append(command) + + +class TestPauseStatePersistenceLayer: + """Unit tests for PauseStatePersistenceLayer.""" + + @staticmethod + def _create_generate_entity(workflow_execution_id: str = "run-123") -> WorkflowAppGenerateEntity: + app_config = WorkflowUIBasedAppConfig( + tenant_id="tenant-123", + app_id="app-123", + app_mode=AppMode.WORKFLOW, + workflow_id="workflow-123", + ) + return WorkflowAppGenerateEntity( + task_id="task-123", + app_config=app_config, + inputs={}, + files=[], + user_id="user-123", + stream=False, + invoke_from=InvokeFrom.DEBUGGER, + workflow_execution_id=workflow_execution_id, + ) + + def test_init_with_dependency_injection(self): + session_factory = Mock(name="session_factory") + state_owner_user_id = "user-123" + + layer = PauseStatePersistenceLayer( + session_factory=session_factory, + state_owner_user_id=state_owner_user_id, + generate_entity=self._create_generate_entity(), + ) + + assert layer._session_maker is session_factory + assert layer._state_owner_user_id == state_owner_user_id + assert not hasattr(layer, "graph_runtime_state") + assert not hasattr(layer, "command_channel") + + def test_initialize_sets_dependencies(self): + session_factory = Mock(name="session_factory") + layer = PauseStatePersistenceLayer( + session_factory=session_factory, + state_owner_user_id="owner", + generate_entity=self._create_generate_entity(), + ) + + graph_runtime_state = MockReadOnlyGraphRuntimeState() + command_channel = MockCommandChannel() + + layer.initialize(graph_runtime_state, command_channel) + + assert layer.graph_runtime_state is graph_runtime_state + assert layer.command_channel is command_channel + + def test_on_event_with_graph_run_paused_event(self, monkeypatch: pytest.MonkeyPatch): + session_factory = Mock(name="session_factory") + generate_entity = self._create_generate_entity(workflow_execution_id="run-123") + layer = PauseStatePersistenceLayer( + session_factory=session_factory, + state_owner_user_id="owner-123", + generate_entity=generate_entity, + ) + + mock_repo = Mock() + mock_factory = Mock(return_value=mock_repo) + monkeypatch.setattr(DifyAPIRepositoryFactory, "create_api_workflow_run_repository", mock_factory) + + graph_runtime_state = MockReadOnlyGraphRuntimeState( + outputs={"result": "test_output"}, + total_tokens=100, + workflow_execution_id="run-123", + ) + command_channel = MockCommandChannel() + layer.initialize(graph_runtime_state, command_channel) + + event = TestDataFactory.create_graph_run_paused_event(outputs={"intermediate": "result"}) + expected_state = graph_runtime_state.dumps() + + layer.on_event(event) + + mock_factory.assert_called_once_with(session_factory) + mock_repo.create_workflow_pause.assert_called_once_with( + workflow_run_id="run-123", + state_owner_user_id="owner-123", + state=mock_repo.create_workflow_pause.call_args.kwargs["state"], + ) + serialized_state = mock_repo.create_workflow_pause.call_args.kwargs["state"] + resumption_context = WorkflowResumptionContext.loads(serialized_state) + assert resumption_context.serialized_graph_runtime_state == expected_state + assert resumption_context.get_generate_entity().model_dump() == generate_entity.model_dump() + + def test_on_event_ignores_non_paused_events(self, monkeypatch: pytest.MonkeyPatch): + session_factory = Mock(name="session_factory") + layer = PauseStatePersistenceLayer( + session_factory=session_factory, + state_owner_user_id="owner-123", + generate_entity=self._create_generate_entity(), + ) + + mock_repo = Mock() + mock_factory = Mock(return_value=mock_repo) + monkeypatch.setattr(DifyAPIRepositoryFactory, "create_api_workflow_run_repository", mock_factory) + + graph_runtime_state = MockReadOnlyGraphRuntimeState() + command_channel = MockCommandChannel() + layer.initialize(graph_runtime_state, command_channel) + + events = [ + TestDataFactory.create_graph_run_started_event(), + TestDataFactory.create_graph_run_succeeded_event(), + TestDataFactory.create_graph_run_failed_event(), + ] + + for event in events: + layer.on_event(event) + + mock_factory.assert_not_called() + mock_repo.create_workflow_pause.assert_not_called() + + def test_on_event_raises_attribute_error_when_graph_runtime_state_is_none(self): + session_factory = Mock(name="session_factory") + layer = PauseStatePersistenceLayer( + session_factory=session_factory, + state_owner_user_id="owner-123", + generate_entity=self._create_generate_entity(), + ) + + event = TestDataFactory.create_graph_run_paused_event() + + with pytest.raises(AttributeError): + layer.on_event(event) + + def test_on_event_asserts_when_workflow_execution_id_missing(self, monkeypatch: pytest.MonkeyPatch): + session_factory = Mock(name="session_factory") + layer = PauseStatePersistenceLayer( + session_factory=session_factory, + state_owner_user_id="owner-123", + generate_entity=self._create_generate_entity(), + ) + + mock_repo = Mock() + mock_factory = Mock(return_value=mock_repo) + monkeypatch.setattr(DifyAPIRepositoryFactory, "create_api_workflow_run_repository", mock_factory) + + graph_runtime_state = MockReadOnlyGraphRuntimeState(workflow_execution_id=None) + command_channel = MockCommandChannel() + layer.initialize(graph_runtime_state, command_channel) + + event = TestDataFactory.create_graph_run_paused_event() + + with pytest.raises(AssertionError): + layer.on_event(event) + + mock_factory.assert_not_called() + mock_repo.create_workflow_pause.assert_not_called() + + +def _build_workflow_generate_entity_for_roundtrip() -> WorkflowResumptionContext: + """Create a WorkflowAppGenerateEntity with realistic data for WorkflowResumptionContext tests.""" + app_config = WorkflowUIBasedAppConfig( + tenant_id="tenant-roundtrip", + app_id="app-roundtrip", + app_mode=AppMode.WORKFLOW, + workflow_id="workflow-roundtrip", + ) + serialized_state = json.dumps({"state": "workflow"}) + + return WorkflowResumptionContext( + serialized_graph_runtime_state=serialized_state, + generate_entity=_WorkflowGenerateEntityWrapper( + entity=WorkflowAppGenerateEntity( + task_id="workflow-task", + app_config=app_config, + inputs={"input_key": "input_value"}, + files=[], + user_id="user-roundtrip", + stream=False, + invoke_from=InvokeFrom.DEBUGGER, + workflow_execution_id="workflow-exec-roundtrip", + ) + ), + ) + + +def _build_advanced_chat_generate_entity_for_roundtrip() -> WorkflowResumptionContext: + """Create an AdvancedChatAppGenerateEntity with realistic data for WorkflowResumptionContext tests.""" + app_config = WorkflowUIBasedAppConfig( + tenant_id="tenant-advanced", + app_id="app-advanced", + app_mode=AppMode.ADVANCED_CHAT, + workflow_id="workflow-advanced", + ) + serialized_state = json.dumps({"state": "workflow"}) + + return WorkflowResumptionContext( + serialized_graph_runtime_state=serialized_state, + generate_entity=_AdvancedChatAppGenerateEntityWrapper( + entity=AdvancedChatAppGenerateEntity( + task_id="advanced-task", + app_config=app_config, + inputs={"topic": "roundtrip"}, + files=[], + user_id="advanced-user", + stream=False, + invoke_from=InvokeFrom.DEBUGGER, + workflow_run_id="advanced-run-id", + query="Explain serialization behavior", + ) + ), + ) + + +@pytest.mark.parametrize( + "state", + [ + pytest.param( + _build_advanced_chat_generate_entity_for_roundtrip(), + id="advanced_chat", + ), + pytest.param( + _build_workflow_generate_entity_for_roundtrip(), + id="workflow", + ), + ], +) +def test_workflow_resumption_context_dumps_loads_roundtrip(state: WorkflowResumptionContext): + """WorkflowResumptionContext roundtrip preserves workflow generate entity metadata.""" + dumped = state.dumps() + loaded = WorkflowResumptionContext.loads(dumped) + + assert loaded == state + assert loaded.serialized_graph_runtime_state == state.serialized_graph_runtime_state + restored_entity = loaded.get_generate_entity() + assert isinstance(restored_entity, type(state.generate_entity.entity)) diff --git a/api/tests/unit_tests/core/file/test_models.py b/api/tests/unit_tests/core/file/test_models.py index 3ada2087c6..f55063ee1a 100644 --- a/api/tests/unit_tests/core/file/test_models.py +++ b/api/tests/unit_tests/core/file/test_models.py @@ -23,3 +23,32 @@ def test_file(): assert file.extension == ".png" assert file.mime_type == "image/png" assert file.size == 67 + + +def test_file_model_validate_with_legacy_fields(): + """Test `File` model can handle data containing compatibility fields.""" + data = { + "id": "test-file", + "tenant_id": "test-tenant-id", + "type": "image", + "transfer_method": "tool_file", + "related_id": "test-related-id", + "filename": "image.png", + "extension": ".png", + "mime_type": "image/png", + "size": 67, + "storage_key": "test-storage-key", + "url": "https://example.com/image.png", + # Extra legacy fields + "tool_file_id": "tool-file-123", + "upload_file_id": "upload-file-456", + "datasource_file_id": "datasource-file-789", + } + + # Should be able to create `File` object without raising an exception + file = File.model_validate(data) + + # The File object does not have tool_file_id, upload_file_id, or datasource_file_id as attributes. + # Instead, check it does not expose unrecognized legacy fields (should raise on getattr). + for legacy_field in ("tool_file_id", "upload_file_id", "datasource_file_id"): + assert not hasattr(file, legacy_field) diff --git a/api/tests/unit_tests/core/helper/code_executor/__init__.py b/api/tests/unit_tests/core/helper/code_executor/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/unit_tests/core/helper/code_executor/javascript/__init__.py b/api/tests/unit_tests/core/helper/code_executor/javascript/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/unit_tests/core/helper/code_executor/javascript/test_javascript_transformer.py b/api/tests/unit_tests/core/helper/code_executor/javascript/test_javascript_transformer.py new file mode 100644 index 0000000000..03f37756d7 --- /dev/null +++ b/api/tests/unit_tests/core/helper/code_executor/javascript/test_javascript_transformer.py @@ -0,0 +1,12 @@ +from core.helper.code_executor.javascript.javascript_code_provider import JavascriptCodeProvider +from core.helper.code_executor.javascript.javascript_transformer import NodeJsTemplateTransformer + + +def test_get_runner_script(): + code = JavascriptCodeProvider.get_default_code() + inputs = {"arg1": "hello, ", "arg2": "world!"} + script = NodeJsTemplateTransformer.assemble_runner_script(code, inputs) + script_lines = script.splitlines() + code_lines = code.splitlines() + # Check that the first lines of script are exactly the same as code + assert script_lines[: len(code_lines)] == code_lines diff --git a/api/tests/unit_tests/core/helper/code_executor/python3/__init__.py b/api/tests/unit_tests/core/helper/code_executor/python3/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/unit_tests/core/helper/code_executor/python3/test_python3_transformer.py b/api/tests/unit_tests/core/helper/code_executor/python3/test_python3_transformer.py new file mode 100644 index 0000000000..1166cb8892 --- /dev/null +++ b/api/tests/unit_tests/core/helper/code_executor/python3/test_python3_transformer.py @@ -0,0 +1,12 @@ +from core.helper.code_executor.python3.python3_code_provider import Python3CodeProvider +from core.helper.code_executor.python3.python3_transformer import Python3TemplateTransformer + + +def test_get_runner_script(): + code = Python3CodeProvider.get_default_code() + inputs = {"arg1": "hello, ", "arg2": "world!"} + script = Python3TemplateTransformer.assemble_runner_script(code, inputs) + script_lines = script.splitlines() + code_lines = code.splitlines() + # Check that the first lines of script are exactly the same as code + assert script_lines[: len(code_lines)] == code_lines diff --git a/api/tests/unit_tests/core/mcp/__init__.py b/api/tests/unit_tests/core/mcp/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/unit_tests/core/mcp/auth/__init__.py b/api/tests/unit_tests/core/mcp/auth/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/unit_tests/core/mcp/auth/test_auth_flow.py b/api/tests/unit_tests/core/mcp/auth/test_auth_flow.py new file mode 100644 index 0000000000..12a9f11205 --- /dev/null +++ b/api/tests/unit_tests/core/mcp/auth/test_auth_flow.py @@ -0,0 +1,740 @@ +"""Unit tests for MCP OAuth authentication flow.""" + +from unittest.mock import Mock, patch + +import pytest + +from core.entities.mcp_provider import MCPProviderEntity +from core.mcp.auth.auth_flow import ( + OAUTH_STATE_EXPIRY_SECONDS, + OAUTH_STATE_REDIS_KEY_PREFIX, + OAuthCallbackState, + _create_secure_redis_state, + _retrieve_redis_state, + auth, + check_support_resource_discovery, + discover_oauth_metadata, + exchange_authorization, + generate_pkce_challenge, + handle_callback, + refresh_authorization, + register_client, + start_authorization, +) +from core.mcp.entities import AuthActionType, AuthResult +from core.mcp.types import ( + OAuthClientInformation, + OAuthClientInformationFull, + OAuthClientMetadata, + OAuthMetadata, + OAuthTokens, +) + + +class TestPKCEGeneration: + """Test PKCE challenge generation.""" + + def test_generate_pkce_challenge(self): + """Test PKCE challenge and verifier generation.""" + code_verifier, code_challenge = generate_pkce_challenge() + + # Verify format - should be URL-safe base64 without padding + assert "=" not in code_verifier + assert "+" not in code_verifier + assert "/" not in code_verifier + assert "=" not in code_challenge + assert "+" not in code_challenge + assert "/" not in code_challenge + + # Verify length + assert len(code_verifier) > 40 # Should be around 54 characters + assert len(code_challenge) > 40 # Should be around 43 characters + + def test_generate_pkce_challenge_uniqueness(self): + """Test that PKCE generation produces unique values.""" + results = set() + for _ in range(10): + code_verifier, code_challenge = generate_pkce_challenge() + results.add((code_verifier, code_challenge)) + + # All should be unique + assert len(results) == 10 + + +class TestRedisStateManagement: + """Test Redis state management functions.""" + + @patch("core.mcp.auth.auth_flow.redis_client") + def test_create_secure_redis_state(self, mock_redis): + """Test creating secure Redis state.""" + state_data = OAuthCallbackState( + provider_id="test-provider", + tenant_id="test-tenant", + server_url="https://example.com", + metadata=None, + client_information=OAuthClientInformation(client_id="test-client"), + code_verifier="test-verifier", + redirect_uri="https://redirect.example.com", + ) + + state_key = _create_secure_redis_state(state_data) + + # Verify state key format + assert len(state_key) > 20 # Should be a secure random token + + # Verify Redis call + mock_redis.setex.assert_called_once() + call_args = mock_redis.setex.call_args + assert call_args[0][0].startswith(OAUTH_STATE_REDIS_KEY_PREFIX) + assert call_args[0][1] == OAUTH_STATE_EXPIRY_SECONDS + assert state_data.model_dump_json() in call_args[0][2] + + @patch("core.mcp.auth.auth_flow.redis_client") + def test_retrieve_redis_state_success(self, mock_redis): + """Test retrieving state from Redis.""" + state_data = OAuthCallbackState( + provider_id="test-provider", + tenant_id="test-tenant", + server_url="https://example.com", + metadata=None, + client_information=OAuthClientInformation(client_id="test-client"), + code_verifier="test-verifier", + redirect_uri="https://redirect.example.com", + ) + mock_redis.get.return_value = state_data.model_dump_json() + + result = _retrieve_redis_state("test-state-key") + + # Verify result + assert result.provider_id == "test-provider" + assert result.tenant_id == "test-tenant" + assert result.server_url == "https://example.com" + + # Verify Redis calls + mock_redis.get.assert_called_once_with(f"{OAUTH_STATE_REDIS_KEY_PREFIX}test-state-key") + mock_redis.delete.assert_called_once_with(f"{OAUTH_STATE_REDIS_KEY_PREFIX}test-state-key") + + @patch("core.mcp.auth.auth_flow.redis_client") + def test_retrieve_redis_state_not_found(self, mock_redis): + """Test retrieving non-existent state from Redis.""" + mock_redis.get.return_value = None + + with pytest.raises(ValueError) as exc_info: + _retrieve_redis_state("nonexistent-key") + + assert "State parameter has expired or does not exist" in str(exc_info.value) + + @patch("core.mcp.auth.auth_flow.redis_client") + def test_retrieve_redis_state_invalid_json(self, mock_redis): + """Test retrieving invalid JSON state from Redis.""" + mock_redis.get.return_value = '{"invalid": json}' + + with pytest.raises(ValueError) as exc_info: + _retrieve_redis_state("test-key") + + assert "Invalid state parameter" in str(exc_info.value) + # State should still be deleted + mock_redis.delete.assert_called_once() + + +class TestOAuthDiscovery: + """Test OAuth discovery functions.""" + + @patch("core.helper.ssrf_proxy.get") + def test_check_support_resource_discovery_success(self, mock_get): + """Test successful resource discovery check.""" + mock_response = Mock() + mock_response.status_code = 200 + mock_response.json.return_value = {"authorization_server_url": ["https://auth.example.com"]} + mock_get.return_value = mock_response + + supported, auth_url = check_support_resource_discovery("https://api.example.com/endpoint") + + assert supported is True + assert auth_url == "https://auth.example.com" + mock_get.assert_called_once_with( + "https://api.example.com/.well-known/oauth-protected-resource", + headers={"MCP-Protocol-Version": "2025-03-26", "User-Agent": "Dify"}, + ) + + @patch("core.helper.ssrf_proxy.get") + def test_check_support_resource_discovery_not_supported(self, mock_get): + """Test resource discovery not supported.""" + mock_response = Mock() + mock_response.status_code = 404 + mock_get.return_value = mock_response + + supported, auth_url = check_support_resource_discovery("https://api.example.com") + + assert supported is False + assert auth_url == "" + + @patch("core.helper.ssrf_proxy.get") + def test_check_support_resource_discovery_with_query_fragment(self, mock_get): + """Test resource discovery with query and fragment.""" + mock_response = Mock() + mock_response.status_code = 200 + mock_response.json.return_value = {"authorization_server_url": ["https://auth.example.com"]} + mock_get.return_value = mock_response + + supported, auth_url = check_support_resource_discovery("https://api.example.com/path?query=1#fragment") + + assert supported is True + assert auth_url == "https://auth.example.com" + mock_get.assert_called_once_with( + "https://api.example.com/.well-known/oauth-protected-resource?query=1#fragment", + headers={"MCP-Protocol-Version": "2025-03-26", "User-Agent": "Dify"}, + ) + + @patch("core.helper.ssrf_proxy.get") + def test_discover_oauth_metadata_with_resource_discovery(self, mock_get): + """Test OAuth metadata discovery with resource discovery support.""" + with patch("core.mcp.auth.auth_flow.check_support_resource_discovery") as mock_check: + mock_check.return_value = (True, "https://auth.example.com") + + mock_response = Mock() + mock_response.status_code = 200 + mock_response.is_success = True + mock_response.json.return_value = { + "authorization_endpoint": "https://auth.example.com/authorize", + "token_endpoint": "https://auth.example.com/token", + "response_types_supported": ["code"], + } + mock_get.return_value = mock_response + + metadata = discover_oauth_metadata("https://api.example.com") + + assert metadata is not None + assert metadata.authorization_endpoint == "https://auth.example.com/authorize" + assert metadata.token_endpoint == "https://auth.example.com/token" + mock_get.assert_called_once_with( + "https://auth.example.com/.well-known/oauth-authorization-server", + headers={"MCP-Protocol-Version": "2025-03-26"}, + ) + + @patch("core.helper.ssrf_proxy.get") + def test_discover_oauth_metadata_without_resource_discovery(self, mock_get): + """Test OAuth metadata discovery without resource discovery.""" + with patch("core.mcp.auth.auth_flow.check_support_resource_discovery") as mock_check: + mock_check.return_value = (False, "") + + mock_response = Mock() + mock_response.status_code = 200 + mock_response.is_success = True + mock_response.json.return_value = { + "authorization_endpoint": "https://api.example.com/oauth/authorize", + "token_endpoint": "https://api.example.com/oauth/token", + "response_types_supported": ["code"], + } + mock_get.return_value = mock_response + + metadata = discover_oauth_metadata("https://api.example.com") + + assert metadata is not None + assert metadata.authorization_endpoint == "https://api.example.com/oauth/authorize" + mock_get.assert_called_once_with( + "https://api.example.com/.well-known/oauth-authorization-server", + headers={"MCP-Protocol-Version": "2025-03-26"}, + ) + + @patch("core.helper.ssrf_proxy.get") + def test_discover_oauth_metadata_not_found(self, mock_get): + """Test OAuth metadata discovery when not found.""" + with patch("core.mcp.auth.auth_flow.check_support_resource_discovery") as mock_check: + mock_check.return_value = (False, "") + + mock_response = Mock() + mock_response.status_code = 404 + mock_get.return_value = mock_response + + metadata = discover_oauth_metadata("https://api.example.com") + + assert metadata is None + + +class TestAuthorizationFlow: + """Test authorization flow functions.""" + + @patch("core.mcp.auth.auth_flow._create_secure_redis_state") + def test_start_authorization_with_metadata(self, mock_create_state): + """Test starting authorization with metadata.""" + mock_create_state.return_value = "secure-state-key" + + metadata = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + code_challenge_methods_supported=["S256"], + ) + client_info = OAuthClientInformation(client_id="test-client-id") + + auth_url, code_verifier = start_authorization( + "https://api.example.com", + metadata, + client_info, + "https://redirect.example.com", + "provider-id", + "tenant-id", + ) + + # Verify URL format + assert auth_url.startswith("https://auth.example.com/authorize?") + assert "response_type=code" in auth_url + assert "client_id=test-client-id" in auth_url + assert "code_challenge=" in auth_url + assert "code_challenge_method=S256" in auth_url + assert "redirect_uri=https%3A%2F%2Fredirect.example.com" in auth_url + assert "state=secure-state-key" in auth_url + + # Verify code verifier + assert len(code_verifier) > 40 + + # Verify state was stored + mock_create_state.assert_called_once() + state_data = mock_create_state.call_args[0][0] + assert state_data.provider_id == "provider-id" + assert state_data.tenant_id == "tenant-id" + assert state_data.code_verifier == code_verifier + + def test_start_authorization_without_metadata(self): + """Test starting authorization without metadata.""" + with patch("core.mcp.auth.auth_flow._create_secure_redis_state") as mock_create_state: + mock_create_state.return_value = "secure-state-key" + + client_info = OAuthClientInformation(client_id="test-client-id") + + auth_url, code_verifier = start_authorization( + "https://api.example.com", + None, + client_info, + "https://redirect.example.com", + "provider-id", + "tenant-id", + ) + + # Should use default authorization endpoint + assert auth_url.startswith("https://api.example.com/authorize?") + + def test_start_authorization_invalid_metadata(self): + """Test starting authorization with invalid metadata.""" + metadata = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["token"], # No "code" support + code_challenge_methods_supported=["plain"], # No "S256" support + ) + client_info = OAuthClientInformation(client_id="test-client-id") + + with pytest.raises(ValueError) as exc_info: + start_authorization( + "https://api.example.com", + metadata, + client_info, + "https://redirect.example.com", + "provider-id", + "tenant-id", + ) + + assert "does not support response type code" in str(exc_info.value) + + @patch("core.helper.ssrf_proxy.post") + def test_exchange_authorization_success(self, mock_post): + """Test successful authorization code exchange.""" + mock_response = Mock() + mock_response.is_success = True + mock_response.json.return_value = { + "access_token": "new-access-token", + "token_type": "Bearer", + "expires_in": 3600, + "refresh_token": "new-refresh-token", + } + mock_post.return_value = mock_response + + metadata = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + grant_types_supported=["authorization_code"], + ) + client_info = OAuthClientInformation(client_id="test-client-id", client_secret="test-secret") + + tokens = exchange_authorization( + "https://api.example.com", + metadata, + client_info, + "auth-code-123", + "code-verifier-xyz", + "https://redirect.example.com", + ) + + assert tokens.access_token == "new-access-token" + assert tokens.token_type == "Bearer" + assert tokens.expires_in == 3600 + assert tokens.refresh_token == "new-refresh-token" + + # Verify request + mock_post.assert_called_once_with( + "https://auth.example.com/token", + data={ + "grant_type": "authorization_code", + "client_id": "test-client-id", + "client_secret": "test-secret", + "code": "auth-code-123", + "code_verifier": "code-verifier-xyz", + "redirect_uri": "https://redirect.example.com", + }, + ) + + @patch("core.helper.ssrf_proxy.post") + def test_exchange_authorization_failure(self, mock_post): + """Test failed authorization code exchange.""" + mock_response = Mock() + mock_response.is_success = False + mock_response.status_code = 400 + mock_post.return_value = mock_response + + client_info = OAuthClientInformation(client_id="test-client-id") + + with pytest.raises(ValueError) as exc_info: + exchange_authorization( + "https://api.example.com", + None, + client_info, + "invalid-code", + "code-verifier", + "https://redirect.example.com", + ) + + assert "Token exchange failed: HTTP 400" in str(exc_info.value) + + @patch("core.helper.ssrf_proxy.post") + def test_refresh_authorization_success(self, mock_post): + """Test successful token refresh.""" + mock_response = Mock() + mock_response.is_success = True + mock_response.json.return_value = { + "access_token": "refreshed-access-token", + "token_type": "Bearer", + "expires_in": 3600, + "refresh_token": "new-refresh-token", + } + mock_post.return_value = mock_response + + metadata = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + grant_types_supported=["refresh_token"], + ) + client_info = OAuthClientInformation(client_id="test-client-id") + + tokens = refresh_authorization("https://api.example.com", metadata, client_info, "old-refresh-token") + + assert tokens.access_token == "refreshed-access-token" + assert tokens.refresh_token == "new-refresh-token" + + # Verify request + mock_post.assert_called_once_with( + "https://auth.example.com/token", + data={ + "grant_type": "refresh_token", + "client_id": "test-client-id", + "refresh_token": "old-refresh-token", + }, + ) + + @patch("core.helper.ssrf_proxy.post") + def test_register_client_success(self, mock_post): + """Test successful client registration.""" + mock_response = Mock() + mock_response.is_success = True + mock_response.json.return_value = { + "client_id": "new-client-id", + "client_secret": "new-client-secret", + "client_name": "Dify", + "redirect_uris": ["https://redirect.example.com"], + } + mock_post.return_value = mock_response + + metadata = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + registration_endpoint="https://auth.example.com/register", + response_types_supported=["code"], + ) + client_metadata = OAuthClientMetadata( + client_name="Dify", + redirect_uris=["https://redirect.example.com"], + grant_types=["authorization_code"], + response_types=["code"], + ) + + client_info = register_client("https://api.example.com", metadata, client_metadata) + + assert isinstance(client_info, OAuthClientInformationFull) + assert client_info.client_id == "new-client-id" + assert client_info.client_secret == "new-client-secret" + + # Verify request + mock_post.assert_called_once_with( + "https://auth.example.com/register", + json=client_metadata.model_dump(), + headers={"Content-Type": "application/json"}, + ) + + def test_register_client_no_endpoint(self): + """Test client registration when no endpoint available.""" + metadata = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + registration_endpoint=None, + response_types_supported=["code"], + ) + client_metadata = OAuthClientMetadata(client_name="Dify", redirect_uris=["https://redirect.example.com"]) + + with pytest.raises(ValueError) as exc_info: + register_client("https://api.example.com", metadata, client_metadata) + + assert "does not support dynamic client registration" in str(exc_info.value) + + +class TestCallbackHandling: + """Test OAuth callback handling.""" + + @patch("core.mcp.auth.auth_flow._retrieve_redis_state") + @patch("core.mcp.auth.auth_flow.exchange_authorization") + def test_handle_callback_success(self, mock_exchange, mock_retrieve_state): + """Test successful callback handling.""" + # Setup state + state_data = OAuthCallbackState( + provider_id="test-provider", + tenant_id="test-tenant", + server_url="https://api.example.com", + metadata=None, + client_information=OAuthClientInformation(client_id="test-client"), + code_verifier="test-verifier", + redirect_uri="https://redirect.example.com", + ) + mock_retrieve_state.return_value = state_data + + # Setup token exchange + tokens = OAuthTokens( + access_token="new-token", + token_type="Bearer", + expires_in=3600, + ) + mock_exchange.return_value = tokens + + # Setup service + mock_service = Mock() + + state_result, tokens_result = handle_callback("state-key", "auth-code") + + assert state_result == state_data + assert tokens_result == tokens + + # Verify calls + mock_retrieve_state.assert_called_once_with("state-key") + mock_exchange.assert_called_once_with( + "https://api.example.com", + None, + state_data.client_information, + "auth-code", + "test-verifier", + "https://redirect.example.com", + ) + # Note: handle_callback no longer saves tokens directly, it just returns them + # The caller (e.g., controller) is responsible for saving via execute_auth_actions + + +class TestAuthOrchestration: + """Test the main auth orchestration function.""" + + @pytest.fixture + def mock_provider(self): + """Create a mock provider entity.""" + provider = Mock(spec=MCPProviderEntity) + provider.id = "provider-id" + provider.tenant_id = "tenant-id" + provider.decrypt_server_url.return_value = "https://api.example.com" + provider.client_metadata = OAuthClientMetadata( + client_name="Dify", + redirect_uris=["https://redirect.example.com"], + ) + provider.redirect_url = "https://redirect.example.com" + provider.retrieve_client_information.return_value = None + provider.retrieve_tokens.return_value = None + return provider + + @pytest.fixture + def mock_service(self): + """Create a mock MCP service.""" + return Mock() + + @patch("core.mcp.auth.auth_flow.discover_oauth_metadata") + @patch("core.mcp.auth.auth_flow.register_client") + @patch("core.mcp.auth.auth_flow.start_authorization") + def test_auth_new_registration(self, mock_start_auth, mock_register, mock_discover, mock_provider, mock_service): + """Test auth flow for new client registration.""" + # Setup + mock_discover.return_value = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + grant_types_supported=["authorization_code"], + ) + mock_register.return_value = OAuthClientInformationFull( + client_id="new-client-id", + client_name="Dify", + redirect_uris=["https://redirect.example.com"], + ) + mock_start_auth.return_value = ("https://auth.example.com/authorize?...", "code-verifier") + + result = auth(mock_provider) + + # auth() now returns AuthResult + assert isinstance(result, AuthResult) + assert result.response == {"authorization_url": "https://auth.example.com/authorize?..."} + + # Verify that the result contains the correct actions + assert len(result.actions) == 2 + # Check for SAVE_CLIENT_INFO action + client_info_action = next(a for a in result.actions if a.action_type == AuthActionType.SAVE_CLIENT_INFO) + assert client_info_action.data == {"client_information": mock_register.return_value.model_dump()} + assert client_info_action.provider_id == "provider-id" + assert client_info_action.tenant_id == "tenant-id" + + # Check for SAVE_CODE_VERIFIER action + verifier_action = next(a for a in result.actions if a.action_type == AuthActionType.SAVE_CODE_VERIFIER) + assert verifier_action.data == {"code_verifier": "code-verifier"} + assert verifier_action.provider_id == "provider-id" + assert verifier_action.tenant_id == "tenant-id" + + # Verify calls + mock_register.assert_called_once() + + @patch("core.mcp.auth.auth_flow.discover_oauth_metadata") + @patch("core.mcp.auth.auth_flow._retrieve_redis_state") + @patch("core.mcp.auth.auth_flow.exchange_authorization") + def test_auth_exchange_code(self, mock_exchange, mock_retrieve_state, mock_discover, mock_provider, mock_service): + """Test auth flow for exchanging authorization code.""" + # Setup metadata discovery + mock_discover.return_value = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + grant_types_supported=["authorization_code"], + ) + + # Setup existing client + mock_provider.retrieve_client_information.return_value = OAuthClientInformation(client_id="existing-client") + + # Setup state retrieval + state_data = OAuthCallbackState( + provider_id="provider-id", + tenant_id="tenant-id", + server_url="https://api.example.com", + metadata=None, + client_information=OAuthClientInformation(client_id="existing-client"), + code_verifier="test-verifier", + redirect_uri="https://redirect.example.com", + ) + mock_retrieve_state.return_value = state_data + + # Setup token exchange + tokens = OAuthTokens(access_token="new-token", token_type="Bearer", expires_in=3600) + mock_exchange.return_value = tokens + + result = auth(mock_provider, authorization_code="auth-code", state_param="state-key") + + # auth() now returns AuthResult, not a dict + assert isinstance(result, AuthResult) + assert result.response == {"result": "success"} + + # Verify that the result contains the correct action + assert len(result.actions) == 1 + assert result.actions[0].action_type == AuthActionType.SAVE_TOKENS + assert result.actions[0].data == tokens.model_dump() + assert result.actions[0].provider_id == "provider-id" + assert result.actions[0].tenant_id == "tenant-id" + + @patch("core.mcp.auth.auth_flow.discover_oauth_metadata") + def test_auth_exchange_code_without_state(self, mock_discover, mock_provider, mock_service): + """Test auth flow fails when exchanging code without state.""" + # Setup metadata discovery + mock_discover.return_value = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + grant_types_supported=["authorization_code"], + ) + + mock_provider.retrieve_client_information.return_value = OAuthClientInformation(client_id="existing-client") + + with pytest.raises(ValueError) as exc_info: + auth(mock_provider, authorization_code="auth-code") + + assert "State parameter is required" in str(exc_info.value) + + @patch("core.mcp.auth.auth_flow.refresh_authorization") + def test_auth_refresh_token(self, mock_refresh, mock_provider, mock_service): + """Test auth flow for refreshing tokens.""" + # Setup existing client and tokens + mock_provider.retrieve_client_information.return_value = OAuthClientInformation(client_id="existing-client") + mock_provider.retrieve_tokens.return_value = OAuthTokens( + access_token="old-token", + token_type="Bearer", + expires_in=0, + refresh_token="refresh-token", + ) + + # Setup refresh + new_tokens = OAuthTokens( + access_token="refreshed-token", + token_type="Bearer", + expires_in=3600, + refresh_token="new-refresh-token", + ) + mock_refresh.return_value = new_tokens + + with patch("core.mcp.auth.auth_flow.discover_oauth_metadata") as mock_discover: + mock_discover.return_value = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + grant_types_supported=["authorization_code"], + ) + + result = auth(mock_provider) + + # auth() now returns AuthResult + assert isinstance(result, AuthResult) + assert result.response == {"result": "success"} + + # Verify that the result contains the correct action + assert len(result.actions) == 1 + assert result.actions[0].action_type == AuthActionType.SAVE_TOKENS + assert result.actions[0].data == new_tokens.model_dump() + assert result.actions[0].provider_id == "provider-id" + assert result.actions[0].tenant_id == "tenant-id" + + # Verify refresh was called + mock_refresh.assert_called_once() + + @patch("core.mcp.auth.auth_flow.discover_oauth_metadata") + def test_auth_registration_fails_with_code(self, mock_discover, mock_provider, mock_service): + """Test auth fails when no client info exists but code is provided.""" + # Setup metadata discovery + mock_discover.return_value = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + response_types_supported=["code"], + grant_types_supported=["authorization_code"], + ) + + mock_provider.retrieve_client_information.return_value = None + + with pytest.raises(ValueError) as exc_info: + auth(mock_provider, authorization_code="auth-code") + + assert "Existing OAuth client information is required" in str(exc_info.value) diff --git a/api/tests/unit_tests/core/mcp/client/test_session.py b/api/tests/unit_tests/core/mcp/client/test_session.py index 08d5b7d21c..8b24c8ce75 100644 --- a/api/tests/unit_tests/core/mcp/client/test_session.py +++ b/api/tests/unit_tests/core/mcp/client/test_session.py @@ -395,9 +395,6 @@ def test_client_capabilities_default(): # Assert default capabilities assert received_capabilities is not None - assert received_capabilities.sampling is not None - assert received_capabilities.roots is not None - assert received_capabilities.roots.listChanged is True def test_client_capabilities_with_custom_callbacks(): diff --git a/api/tests/unit_tests/core/mcp/test_auth_client_inheritance.py b/api/tests/unit_tests/core/mcp/test_auth_client_inheritance.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/unit_tests/core/mcp/test_entities.py b/api/tests/unit_tests/core/mcp/test_entities.py new file mode 100644 index 0000000000..3fede55916 --- /dev/null +++ b/api/tests/unit_tests/core/mcp/test_entities.py @@ -0,0 +1,239 @@ +"""Unit tests for MCP entities module.""" + +from unittest.mock import Mock + +from core.mcp.entities import ( + SUPPORTED_PROTOCOL_VERSIONS, + LifespanContextT, + RequestContext, + SessionT, +) +from core.mcp.session.base_session import BaseSession +from core.mcp.types import LATEST_PROTOCOL_VERSION, RequestParams + + +class TestProtocolVersions: + """Test protocol version constants.""" + + def test_supported_protocol_versions(self): + """Test supported protocol versions list.""" + assert isinstance(SUPPORTED_PROTOCOL_VERSIONS, list) + assert len(SUPPORTED_PROTOCOL_VERSIONS) >= 3 + assert "2024-11-05" in SUPPORTED_PROTOCOL_VERSIONS + assert "2025-03-26" in SUPPORTED_PROTOCOL_VERSIONS + assert LATEST_PROTOCOL_VERSION in SUPPORTED_PROTOCOL_VERSIONS + + def test_latest_protocol_version_is_supported(self): + """Test that latest protocol version is in supported versions.""" + assert LATEST_PROTOCOL_VERSION in SUPPORTED_PROTOCOL_VERSIONS + + +class TestRequestContext: + """Test RequestContext dataclass.""" + + def test_request_context_creation(self): + """Test creating a RequestContext instance.""" + mock_session = Mock(spec=BaseSession) + mock_lifespan = {"key": "value"} + mock_meta = RequestParams.Meta(progressToken="test-token") + + context = RequestContext( + request_id="test-request-123", + meta=mock_meta, + session=mock_session, + lifespan_context=mock_lifespan, + ) + + assert context.request_id == "test-request-123" + assert context.meta == mock_meta + assert context.session == mock_session + assert context.lifespan_context == mock_lifespan + + def test_request_context_with_none_meta(self): + """Test creating RequestContext with None meta.""" + mock_session = Mock(spec=BaseSession) + + context = RequestContext( + request_id=42, # Can be int or string + meta=None, + session=mock_session, + lifespan_context=None, + ) + + assert context.request_id == 42 + assert context.meta is None + assert context.session == mock_session + assert context.lifespan_context is None + + def test_request_context_attributes(self): + """Test RequestContext attributes are accessible.""" + mock_session = Mock(spec=BaseSession) + + context = RequestContext( + request_id="test-123", + meta=None, + session=mock_session, + lifespan_context=None, + ) + + # Verify attributes are accessible + assert hasattr(context, "request_id") + assert hasattr(context, "meta") + assert hasattr(context, "session") + assert hasattr(context, "lifespan_context") + + # Verify values + assert context.request_id == "test-123" + assert context.meta is None + assert context.session == mock_session + assert context.lifespan_context is None + + def test_request_context_generic_typing(self): + """Test RequestContext with different generic types.""" + # Create a mock session with specific type + mock_session = Mock(spec=BaseSession) + + # Create context with string lifespan context + context_str = RequestContext[BaseSession, str]( + request_id="test-1", + meta=None, + session=mock_session, + lifespan_context="string-context", + ) + assert isinstance(context_str.lifespan_context, str) + + # Create context with dict lifespan context + context_dict = RequestContext[BaseSession, dict]( + request_id="test-2", + meta=None, + session=mock_session, + lifespan_context={"key": "value"}, + ) + assert isinstance(context_dict.lifespan_context, dict) + + # Create context with custom object lifespan context + class CustomLifespan: + def __init__(self, data): + self.data = data + + custom_lifespan = CustomLifespan("test-data") + context_custom = RequestContext[BaseSession, CustomLifespan]( + request_id="test-3", + meta=None, + session=mock_session, + lifespan_context=custom_lifespan, + ) + assert isinstance(context_custom.lifespan_context, CustomLifespan) + assert context_custom.lifespan_context.data == "test-data" + + def test_request_context_with_progress_meta(self): + """Test RequestContext with progress metadata.""" + mock_session = Mock(spec=BaseSession) + progress_meta = RequestParams.Meta(progressToken="progress-123") + + context = RequestContext( + request_id="req-456", + meta=progress_meta, + session=mock_session, + lifespan_context=None, + ) + + assert context.meta is not None + assert context.meta.progressToken == "progress-123" + + def test_request_context_equality(self): + """Test RequestContext equality comparison.""" + mock_session1 = Mock(spec=BaseSession) + mock_session2 = Mock(spec=BaseSession) + + context1 = RequestContext( + request_id="test-123", + meta=None, + session=mock_session1, + lifespan_context="context", + ) + + context2 = RequestContext( + request_id="test-123", + meta=None, + session=mock_session1, + lifespan_context="context", + ) + + context3 = RequestContext( + request_id="test-456", + meta=None, + session=mock_session1, + lifespan_context="context", + ) + + # Same values should be equal + assert context1 == context2 + + # Different request_id should not be equal + assert context1 != context3 + + # Different session should not be equal + context4 = RequestContext( + request_id="test-123", + meta=None, + session=mock_session2, + lifespan_context="context", + ) + assert context1 != context4 + + def test_request_context_repr(self): + """Test RequestContext string representation.""" + mock_session = Mock(spec=BaseSession) + mock_session.__repr__ = Mock(return_value="") + + context = RequestContext( + request_id="test-123", + meta=None, + session=mock_session, + lifespan_context={"data": "test"}, + ) + + repr_str = repr(context) + assert "RequestContext" in repr_str + assert "test-123" in repr_str + assert "MockSession" in repr_str + + +class TestTypeVariables: + """Test type variables defined in the module.""" + + def test_session_type_var(self): + """Test SessionT type variable.""" + + # Create a custom session class + class CustomSession(BaseSession): + pass + + # Use in generic context + def process_session(session: SessionT) -> SessionT: + return session + + mock_session = Mock(spec=CustomSession) + result = process_session(mock_session) + assert result == mock_session + + def test_lifespan_context_type_var(self): + """Test LifespanContextT type variable.""" + + # Use in generic context + def process_lifespan(context: LifespanContextT) -> LifespanContextT: + return context + + # Test with different types + str_context = "string-context" + assert process_lifespan(str_context) == str_context + + dict_context = {"key": "value"} + assert process_lifespan(dict_context) == dict_context + + class CustomContext: + pass + + custom_context = CustomContext() + assert process_lifespan(custom_context) == custom_context diff --git a/api/tests/unit_tests/core/mcp/test_error.py b/api/tests/unit_tests/core/mcp/test_error.py new file mode 100644 index 0000000000..3a95fae673 --- /dev/null +++ b/api/tests/unit_tests/core/mcp/test_error.py @@ -0,0 +1,205 @@ +"""Unit tests for MCP error classes.""" + +import pytest + +from core.mcp.error import MCPAuthError, MCPConnectionError, MCPError + + +class TestMCPError: + """Test MCPError base exception class.""" + + def test_mcp_error_creation(self): + """Test creating MCPError instance.""" + error = MCPError("Test error message") + assert str(error) == "Test error message" + assert isinstance(error, Exception) + + def test_mcp_error_inheritance(self): + """Test MCPError inherits from Exception.""" + error = MCPError() + assert isinstance(error, Exception) + assert type(error).__name__ == "MCPError" + + def test_mcp_error_with_empty_message(self): + """Test MCPError with empty message.""" + error = MCPError() + assert str(error) == "" + + def test_mcp_error_raise(self): + """Test raising MCPError.""" + with pytest.raises(MCPError) as exc_info: + raise MCPError("Something went wrong") + + assert str(exc_info.value) == "Something went wrong" + + +class TestMCPConnectionError: + """Test MCPConnectionError exception class.""" + + def test_mcp_connection_error_creation(self): + """Test creating MCPConnectionError instance.""" + error = MCPConnectionError("Connection failed") + assert str(error) == "Connection failed" + assert isinstance(error, MCPError) + assert isinstance(error, Exception) + + def test_mcp_connection_error_inheritance(self): + """Test MCPConnectionError inheritance chain.""" + error = MCPConnectionError() + assert isinstance(error, MCPConnectionError) + assert isinstance(error, MCPError) + assert isinstance(error, Exception) + + def test_mcp_connection_error_raise(self): + """Test raising MCPConnectionError.""" + with pytest.raises(MCPConnectionError) as exc_info: + raise MCPConnectionError("Unable to connect to server") + + assert str(exc_info.value) == "Unable to connect to server" + + def test_mcp_connection_error_catch_as_mcp_error(self): + """Test catching MCPConnectionError as MCPError.""" + with pytest.raises(MCPError) as exc_info: + raise MCPConnectionError("Connection issue") + + assert isinstance(exc_info.value, MCPConnectionError) + assert str(exc_info.value) == "Connection issue" + + +class TestMCPAuthError: + """Test MCPAuthError exception class.""" + + def test_mcp_auth_error_creation(self): + """Test creating MCPAuthError instance.""" + error = MCPAuthError("Authentication failed") + assert str(error) == "Authentication failed" + assert isinstance(error, MCPConnectionError) + assert isinstance(error, MCPError) + assert isinstance(error, Exception) + + def test_mcp_auth_error_inheritance(self): + """Test MCPAuthError inheritance chain.""" + error = MCPAuthError() + assert isinstance(error, MCPAuthError) + assert isinstance(error, MCPConnectionError) + assert isinstance(error, MCPError) + assert isinstance(error, Exception) + + def test_mcp_auth_error_raise(self): + """Test raising MCPAuthError.""" + with pytest.raises(MCPAuthError) as exc_info: + raise MCPAuthError("Invalid credentials") + + assert str(exc_info.value) == "Invalid credentials" + + def test_mcp_auth_error_catch_hierarchy(self): + """Test catching MCPAuthError at different levels.""" + # Catch as MCPAuthError + with pytest.raises(MCPAuthError) as exc_info: + raise MCPAuthError("Auth specific error") + assert str(exc_info.value) == "Auth specific error" + + # Catch as MCPConnectionError + with pytest.raises(MCPConnectionError) as exc_info: + raise MCPAuthError("Auth connection error") + assert isinstance(exc_info.value, MCPAuthError) + assert str(exc_info.value) == "Auth connection error" + + # Catch as MCPError + with pytest.raises(MCPError) as exc_info: + raise MCPAuthError("Auth base error") + assert isinstance(exc_info.value, MCPAuthError) + assert str(exc_info.value) == "Auth base error" + + +class TestErrorHierarchy: + """Test the complete error hierarchy.""" + + def test_exception_hierarchy(self): + """Test the complete exception hierarchy.""" + # Create instances + base_error = MCPError("base") + connection_error = MCPConnectionError("connection") + auth_error = MCPAuthError("auth") + + # Test type relationships + assert not isinstance(base_error, MCPConnectionError) + assert not isinstance(base_error, MCPAuthError) + + assert isinstance(connection_error, MCPError) + assert not isinstance(connection_error, MCPAuthError) + + assert isinstance(auth_error, MCPError) + assert isinstance(auth_error, MCPConnectionError) + + def test_error_handling_patterns(self): + """Test common error handling patterns.""" + + def raise_auth_error(): + raise MCPAuthError("401 Unauthorized") + + def raise_connection_error(): + raise MCPConnectionError("Connection timeout") + + def raise_base_error(): + raise MCPError("Generic error") + + # Pattern 1: Catch specific errors first + errors_caught = [] + + for error_func in [raise_auth_error, raise_connection_error, raise_base_error]: + try: + error_func() + except MCPAuthError: + errors_caught.append("auth") + except MCPConnectionError: + errors_caught.append("connection") + except MCPError: + errors_caught.append("base") + + assert errors_caught == ["auth", "connection", "base"] + + # Pattern 2: Catch all as base error + for error_func in [raise_auth_error, raise_connection_error, raise_base_error]: + with pytest.raises(MCPError) as exc_info: + error_func() + assert isinstance(exc_info.value, MCPError) + + def test_error_with_cause(self): + """Test errors with cause (chained exceptions).""" + original_error = ValueError("Original error") + + def raise_chained_error(): + try: + raise original_error + except ValueError as e: + raise MCPConnectionError("Connection failed") from e + + with pytest.raises(MCPConnectionError) as exc_info: + raise_chained_error() + + assert str(exc_info.value) == "Connection failed" + assert exc_info.value.__cause__ == original_error + + def test_error_comparison(self): + """Test error instance comparison.""" + error1 = MCPError("Test message") + error2 = MCPError("Test message") + error3 = MCPError("Different message") + + # Errors are not equal even with same message (different instances) + assert error1 != error2 + assert error1 != error3 + + # But they have the same type + assert type(error1) == type(error2) == type(error3) + + def test_error_representation(self): + """Test error string representation.""" + base_error = MCPError("Base error message") + connection_error = MCPConnectionError("Connection error message") + auth_error = MCPAuthError("Auth error message") + + assert repr(base_error) == "MCPError('Base error message')" + assert repr(connection_error) == "MCPConnectionError('Connection error message')" + assert repr(auth_error) == "MCPAuthError('Auth error message')" diff --git a/api/tests/unit_tests/core/mcp/test_mcp_client.py b/api/tests/unit_tests/core/mcp/test_mcp_client.py new file mode 100644 index 0000000000..c0420d3371 --- /dev/null +++ b/api/tests/unit_tests/core/mcp/test_mcp_client.py @@ -0,0 +1,382 @@ +"""Unit tests for MCP client.""" + +from contextlib import ExitStack +from types import TracebackType +from unittest.mock import Mock, patch + +import pytest + +from core.mcp.error import MCPConnectionError +from core.mcp.mcp_client import MCPClient +from core.mcp.types import CallToolResult, ListToolsResult, TextContent, Tool, ToolAnnotations + + +class TestMCPClient: + """Test suite for MCPClient.""" + + def test_init(self): + """Test client initialization.""" + client = MCPClient( + server_url="http://test.example.com/mcp", + headers={"Authorization": "Bearer test"}, + timeout=30.0, + sse_read_timeout=60.0, + ) + + assert client.server_url == "http://test.example.com/mcp" + assert client.headers == {"Authorization": "Bearer test"} + assert client.timeout == 30.0 + assert client.sse_read_timeout == 60.0 + assert client._session is None + assert isinstance(client._exit_stack, ExitStack) + assert client._initialized is False + + def test_init_defaults(self): + """Test client initialization with defaults.""" + client = MCPClient(server_url="http://test.example.com") + + assert client.server_url == "http://test.example.com" + assert client.headers == {} + assert client.timeout is None + assert client.sse_read_timeout is None + + @patch("core.mcp.mcp_client.streamablehttp_client") + @patch("core.mcp.mcp_client.ClientSession") + def test_initialize_with_mcp_url(self, mock_client_session, mock_streamable_client): + """Test initialization with MCP URL.""" + # Setup mocks + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_client_context = Mock() + mock_streamable_client.return_value.__enter__.return_value = ( + mock_read_stream, + mock_write_stream, + mock_client_context, + ) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + client = MCPClient(server_url="http://test.example.com/mcp") + client._initialize() + + # Verify streamable client was called + mock_streamable_client.assert_called_once_with( + url="http://test.example.com/mcp", + headers={}, + timeout=None, + sse_read_timeout=None, + ) + + # Verify session was created + mock_client_session.assert_called_once_with(mock_read_stream, mock_write_stream) + mock_session.initialize.assert_called_once() + assert client._session == mock_session + + @patch("core.mcp.mcp_client.sse_client") + @patch("core.mcp.mcp_client.ClientSession") + def test_initialize_with_sse_url(self, mock_client_session, mock_sse_client): + """Test initialization with SSE URL.""" + # Setup mocks + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_sse_client.return_value.__enter__.return_value = (mock_read_stream, mock_write_stream) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + client = MCPClient(server_url="http://test.example.com/sse") + client._initialize() + + # Verify SSE client was called + mock_sse_client.assert_called_once_with( + url="http://test.example.com/sse", + headers={}, + timeout=None, + sse_read_timeout=None, + ) + + # Verify session was created + mock_client_session.assert_called_once_with(mock_read_stream, mock_write_stream) + mock_session.initialize.assert_called_once() + assert client._session == mock_session + + @patch("core.mcp.mcp_client.sse_client") + @patch("core.mcp.mcp_client.streamablehttp_client") + @patch("core.mcp.mcp_client.ClientSession") + def test_initialize_with_unknown_method_fallback_to_sse( + self, mock_client_session, mock_streamable_client, mock_sse_client + ): + """Test initialization with unknown method falls back to SSE.""" + # Setup mocks + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_sse_client.return_value.__enter__.return_value = (mock_read_stream, mock_write_stream) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + client = MCPClient(server_url="http://test.example.com/unknown") + client._initialize() + + # Verify SSE client was tried + mock_sse_client.assert_called_once() + mock_streamable_client.assert_not_called() + + # Verify session was created + assert client._session == mock_session + + @patch("core.mcp.mcp_client.sse_client") + @patch("core.mcp.mcp_client.streamablehttp_client") + @patch("core.mcp.mcp_client.ClientSession") + def test_initialize_fallback_from_sse_to_mcp(self, mock_client_session, mock_streamable_client, mock_sse_client): + """Test initialization falls back from SSE to MCP on connection error.""" + # Setup SSE to fail + mock_sse_client.side_effect = MCPConnectionError("SSE connection failed") + + # Setup MCP to succeed + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_client_context = Mock() + mock_streamable_client.return_value.__enter__.return_value = ( + mock_read_stream, + mock_write_stream, + mock_client_context, + ) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + client = MCPClient(server_url="http://test.example.com/unknown") + client._initialize() + + # Verify both were tried + mock_sse_client.assert_called_once() + mock_streamable_client.assert_called_once() + + # Verify session was created with MCP + assert client._session == mock_session + + @patch("core.mcp.mcp_client.streamablehttp_client") + @patch("core.mcp.mcp_client.ClientSession") + def test_connect_server_mcp(self, mock_client_session, mock_streamable_client): + """Test connect_server with MCP method.""" + # Setup mocks + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_client_context = Mock() + mock_streamable_client.return_value.__enter__.return_value = ( + mock_read_stream, + mock_write_stream, + mock_client_context, + ) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + client = MCPClient(server_url="http://test.example.com") + client.connect_server(mock_streamable_client, "mcp") + + # Verify correct streams were passed + mock_client_session.assert_called_once_with(mock_read_stream, mock_write_stream) + mock_session.initialize.assert_called_once() + + @patch("core.mcp.mcp_client.sse_client") + @patch("core.mcp.mcp_client.ClientSession") + def test_connect_server_sse(self, mock_client_session, mock_sse_client): + """Test connect_server with SSE method.""" + # Setup mocks + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_sse_client.return_value.__enter__.return_value = (mock_read_stream, mock_write_stream) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + client = MCPClient(server_url="http://test.example.com") + client.connect_server(mock_sse_client, "sse") + + # Verify correct streams were passed + mock_client_session.assert_called_once_with(mock_read_stream, mock_write_stream) + mock_session.initialize.assert_called_once() + + def test_context_manager_enter(self): + """Test context manager enter.""" + client = MCPClient(server_url="http://test.example.com") + + with patch.object(client, "_initialize") as mock_initialize: + result = client.__enter__() + + assert result == client + assert client._initialized is True + mock_initialize.assert_called_once() + + def test_context_manager_exit(self): + """Test context manager exit.""" + client = MCPClient(server_url="http://test.example.com") + + with patch.object(client, "cleanup") as mock_cleanup: + exc_type: type[BaseException] | None = None + exc_val: BaseException | None = None + exc_tb: TracebackType | None = None + client.__exit__(exc_type, exc_val, exc_tb) + + mock_cleanup.assert_called_once() + + def test_list_tools_not_initialized(self): + """Test list_tools when session not initialized.""" + client = MCPClient(server_url="http://test.example.com") + + with pytest.raises(ValueError) as exc_info: + client.list_tools() + + assert "Session not initialized" in str(exc_info.value) + + def test_list_tools_success(self): + """Test successful list_tools call.""" + client = MCPClient(server_url="http://test.example.com") + + # Setup mock session + mock_session = Mock() + expected_tools = [ + Tool( + name="test-tool", + description="A test tool", + inputSchema={"type": "object", "properties": {}}, + annotations=ToolAnnotations(title="Test Tool"), + ) + ] + mock_session.list_tools.return_value = ListToolsResult(tools=expected_tools) + client._session = mock_session + + result = client.list_tools() + + assert result == expected_tools + mock_session.list_tools.assert_called_once() + + def test_invoke_tool_not_initialized(self): + """Test invoke_tool when session not initialized.""" + client = MCPClient(server_url="http://test.example.com") + + with pytest.raises(ValueError) as exc_info: + client.invoke_tool("test-tool", {"arg": "value"}) + + assert "Session not initialized" in str(exc_info.value) + + def test_invoke_tool_success(self): + """Test successful invoke_tool call.""" + client = MCPClient(server_url="http://test.example.com") + + # Setup mock session + mock_session = Mock() + expected_result = CallToolResult( + content=[TextContent(type="text", text="Tool executed successfully")], + isError=False, + ) + mock_session.call_tool.return_value = expected_result + client._session = mock_session + + result = client.invoke_tool("test-tool", {"arg": "value"}) + + assert result == expected_result + mock_session.call_tool.assert_called_once_with("test-tool", {"arg": "value"}) + + def test_cleanup(self): + """Test cleanup method.""" + client = MCPClient(server_url="http://test.example.com") + mock_exit_stack = Mock(spec=ExitStack) + client._exit_stack = mock_exit_stack + client._session = Mock() + client._initialized = True + + client.cleanup() + + mock_exit_stack.close.assert_called_once() + assert client._session is None + assert client._initialized is False + + def test_cleanup_with_error(self): + """Test cleanup method with error.""" + client = MCPClient(server_url="http://test.example.com") + mock_exit_stack = Mock(spec=ExitStack) + mock_exit_stack.close.side_effect = Exception("Cleanup error") + client._exit_stack = mock_exit_stack + client._session = Mock() + client._initialized = True + + with pytest.raises(ValueError) as exc_info: + client.cleanup() + + assert "Error during cleanup: Cleanup error" in str(exc_info.value) + assert client._session is None + assert client._initialized is False + + @patch("core.mcp.mcp_client.streamablehttp_client") + @patch("core.mcp.mcp_client.ClientSession") + def test_full_context_manager_flow(self, mock_client_session, mock_streamable_client): + """Test full context manager flow.""" + # Setup mocks + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_client_context = Mock() + mock_streamable_client.return_value.__enter__.return_value = ( + mock_read_stream, + mock_write_stream, + mock_client_context, + ) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + expected_tools = [Tool(name="test-tool", description="Test", inputSchema={})] + mock_session.list_tools.return_value = ListToolsResult(tools=expected_tools) + + with MCPClient(server_url="http://test.example.com/mcp") as client: + assert client._initialized is True + assert client._session == mock_session + + # Test tool operations + tools = client.list_tools() + assert tools == expected_tools + + # After exit, should be cleaned up + assert client._initialized is False + assert client._session is None + + def test_headers_passed_to_clients(self): + """Test that headers are properly passed to underlying clients.""" + custom_headers = { + "Authorization": "Bearer test-token", + "X-Custom-Header": "test-value", + } + + with patch("core.mcp.mcp_client.streamablehttp_client") as mock_streamable_client: + with patch("core.mcp.mcp_client.ClientSession") as mock_client_session: + # Setup mocks + mock_read_stream = Mock() + mock_write_stream = Mock() + mock_client_context = Mock() + mock_streamable_client.return_value.__enter__.return_value = ( + mock_read_stream, + mock_write_stream, + mock_client_context, + ) + + mock_session = Mock() + mock_client_session.return_value.__enter__.return_value = mock_session + + client = MCPClient( + server_url="http://test.example.com/mcp", + headers=custom_headers, + timeout=30.0, + sse_read_timeout=60.0, + ) + client._initialize() + + # Verify headers were passed + mock_streamable_client.assert_called_once_with( + url="http://test.example.com/mcp", + headers=custom_headers, + timeout=30.0, + sse_read_timeout=60.0, + ) diff --git a/api/tests/unit_tests/core/mcp/test_types.py b/api/tests/unit_tests/core/mcp/test_types.py new file mode 100644 index 0000000000..6d8130bd13 --- /dev/null +++ b/api/tests/unit_tests/core/mcp/test_types.py @@ -0,0 +1,492 @@ +"""Unit tests for MCP types module.""" + +import pytest +from pydantic import ValidationError + +from core.mcp.types import ( + INTERNAL_ERROR, + INVALID_PARAMS, + INVALID_REQUEST, + LATEST_PROTOCOL_VERSION, + METHOD_NOT_FOUND, + PARSE_ERROR, + SERVER_LATEST_PROTOCOL_VERSION, + Annotations, + CallToolRequest, + CallToolRequestParams, + CallToolResult, + ClientCapabilities, + CompleteRequest, + CompleteRequestParams, + CompleteResult, + Completion, + CompletionArgument, + CompletionContext, + ErrorData, + ImageContent, + Implementation, + InitializeRequest, + InitializeRequestParams, + InitializeResult, + JSONRPCError, + JSONRPCMessage, + JSONRPCNotification, + JSONRPCRequest, + JSONRPCResponse, + ListToolsRequest, + ListToolsResult, + OAuthClientInformation, + OAuthClientMetadata, + OAuthMetadata, + OAuthTokens, + PingRequest, + ProgressNotification, + ProgressNotificationParams, + PromptReference, + RequestParams, + ResourceTemplateReference, + Result, + ServerCapabilities, + TextContent, + Tool, + ToolAnnotations, +) + + +class TestConstants: + """Test module constants.""" + + def test_protocol_versions(self): + """Test protocol version constants.""" + assert LATEST_PROTOCOL_VERSION == "2025-03-26" + assert SERVER_LATEST_PROTOCOL_VERSION == "2024-11-05" + + def test_error_codes(self): + """Test JSON-RPC error code constants.""" + assert PARSE_ERROR == -32700 + assert INVALID_REQUEST == -32600 + assert METHOD_NOT_FOUND == -32601 + assert INVALID_PARAMS == -32602 + assert INTERNAL_ERROR == -32603 + + +class TestRequestParams: + """Test RequestParams and related classes.""" + + def test_request_params_basic(self): + """Test basic RequestParams creation.""" + params = RequestParams() + assert params.meta is None + + def test_request_params_with_meta(self): + """Test RequestParams with meta.""" + meta = RequestParams.Meta(progressToken="test-token") + params = RequestParams(_meta=meta) + assert params.meta is not None + assert params.meta.progressToken == "test-token" + + def test_request_params_meta_extra_fields(self): + """Test RequestParams.Meta allows extra fields.""" + meta = RequestParams.Meta(progressToken="token", customField="value") + assert meta.progressToken == "token" + assert meta.customField == "value" # type: ignore + + def test_request_params_serialization(self): + """Test RequestParams serialization with _meta alias.""" + meta = RequestParams.Meta(progressToken="test") + params = RequestParams(_meta=meta) + + # Model dump should use the alias + dumped = params.model_dump(by_alias=True) + assert "_meta" in dumped + assert dumped["_meta"] is not None + assert dumped["_meta"]["progressToken"] == "test" + + +class TestJSONRPCMessages: + """Test JSON-RPC message types.""" + + def test_jsonrpc_request(self): + """Test JSONRPCRequest creation and validation.""" + request = JSONRPCRequest(jsonrpc="2.0", id="test-123", method="test_method", params={"key": "value"}) + + assert request.jsonrpc == "2.0" + assert request.id == "test-123" + assert request.method == "test_method" + assert request.params == {"key": "value"} + + def test_jsonrpc_request_numeric_id(self): + """Test JSONRPCRequest with numeric ID.""" + request = JSONRPCRequest(jsonrpc="2.0", id=123, method="test", params=None) + assert request.id == 123 + + def test_jsonrpc_notification(self): + """Test JSONRPCNotification creation.""" + notification = JSONRPCNotification(jsonrpc="2.0", method="notification_method", params={"data": "test"}) + + assert notification.jsonrpc == "2.0" + assert notification.method == "notification_method" + assert not hasattr(notification, "id") # Notifications don't have ID + + def test_jsonrpc_response(self): + """Test JSONRPCResponse creation.""" + response = JSONRPCResponse(jsonrpc="2.0", id="req-123", result={"success": True}) + + assert response.jsonrpc == "2.0" + assert response.id == "req-123" + assert response.result == {"success": True} + + def test_jsonrpc_error(self): + """Test JSONRPCError creation.""" + error_data = ErrorData(code=INVALID_PARAMS, message="Invalid parameters", data={"field": "missing"}) + + error = JSONRPCError(jsonrpc="2.0", id="req-123", error=error_data) + + assert error.jsonrpc == "2.0" + assert error.id == "req-123" + assert error.error.code == INVALID_PARAMS + assert error.error.message == "Invalid parameters" + assert error.error.data == {"field": "missing"} + + def test_jsonrpc_message_parsing(self): + """Test JSONRPCMessage parsing different message types.""" + # Parse request + request_json = '{"jsonrpc": "2.0", "id": 1, "method": "test", "params": null}' + msg = JSONRPCMessage.model_validate_json(request_json) + assert isinstance(msg.root, JSONRPCRequest) + + # Parse response + response_json = '{"jsonrpc": "2.0", "id": 1, "result": {"data": "test"}}' + msg = JSONRPCMessage.model_validate_json(response_json) + assert isinstance(msg.root, JSONRPCResponse) + + # Parse error + error_json = '{"jsonrpc": "2.0", "id": 1, "error": {"code": -32600, "message": "Invalid Request"}}' + msg = JSONRPCMessage.model_validate_json(error_json) + assert isinstance(msg.root, JSONRPCError) + + +class TestCapabilities: + """Test capability classes.""" + + def test_client_capabilities(self): + """Test ClientCapabilities creation.""" + caps = ClientCapabilities( + experimental={"feature": {"enabled": True}}, + sampling={"model_config": {"extra": "allow"}}, + roots={"listChanged": True}, + ) + + assert caps.experimental == {"feature": {"enabled": True}} + assert caps.sampling is not None + assert caps.roots.listChanged is True # type: ignore + + def test_server_capabilities(self): + """Test ServerCapabilities creation.""" + caps = ServerCapabilities( + tools={"listChanged": True}, + resources={"subscribe": True, "listChanged": False}, + prompts={"listChanged": True}, + logging={}, + completions={}, + ) + + assert caps.tools.listChanged is True # type: ignore + assert caps.resources.subscribe is True # type: ignore + assert caps.resources.listChanged is False # type: ignore + + +class TestInitialization: + """Test initialization request/response types.""" + + def test_initialize_request(self): + """Test InitializeRequest creation.""" + client_info = Implementation(name="test-client", version="1.0.0") + capabilities = ClientCapabilities() + + params = InitializeRequestParams( + protocolVersion=LATEST_PROTOCOL_VERSION, capabilities=capabilities, clientInfo=client_info + ) + + request = InitializeRequest(params=params) + + assert request.method == "initialize" + assert request.params.protocolVersion == LATEST_PROTOCOL_VERSION + assert request.params.clientInfo.name == "test-client" + + def test_initialize_result(self): + """Test InitializeResult creation.""" + server_info = Implementation(name="test-server", version="1.0.0") + capabilities = ServerCapabilities() + + result = InitializeResult( + protocolVersion=LATEST_PROTOCOL_VERSION, + capabilities=capabilities, + serverInfo=server_info, + instructions="Welcome to test server", + ) + + assert result.protocolVersion == LATEST_PROTOCOL_VERSION + assert result.serverInfo.name == "test-server" + assert result.instructions == "Welcome to test server" + + +class TestTools: + """Test tool-related types.""" + + def test_tool_creation(self): + """Test Tool creation with all fields.""" + tool = Tool( + name="test_tool", + title="Test Tool", + description="A tool for testing", + inputSchema={"type": "object", "properties": {"input": {"type": "string"}}, "required": ["input"]}, + outputSchema={"type": "object", "properties": {"result": {"type": "string"}}}, + annotations=ToolAnnotations( + title="Test Tool", readOnlyHint=False, destructiveHint=False, idempotentHint=True + ), + ) + + assert tool.name == "test_tool" + assert tool.title == "Test Tool" + assert tool.description == "A tool for testing" + assert tool.inputSchema["properties"]["input"]["type"] == "string" + assert tool.annotations.idempotentHint is True + + def test_call_tool_request(self): + """Test CallToolRequest creation.""" + params = CallToolRequestParams(name="test_tool", arguments={"input": "test value"}) + + request = CallToolRequest(params=params) + + assert request.method == "tools/call" + assert request.params.name == "test_tool" + assert request.params.arguments == {"input": "test value"} + + def test_call_tool_result(self): + """Test CallToolResult creation.""" + result = CallToolResult( + content=[TextContent(type="text", text="Tool executed successfully")], + structuredContent={"status": "success", "data": "test"}, + isError=False, + ) + + assert len(result.content) == 1 + assert result.content[0].text == "Tool executed successfully" # type: ignore + assert result.structuredContent == {"status": "success", "data": "test"} + assert result.isError is False + + def test_list_tools_request(self): + """Test ListToolsRequest creation.""" + request = ListToolsRequest() + assert request.method == "tools/list" + + def test_list_tools_result(self): + """Test ListToolsResult creation.""" + tool1 = Tool(name="tool1", inputSchema={}) + tool2 = Tool(name="tool2", inputSchema={}) + + result = ListToolsResult(tools=[tool1, tool2]) + + assert len(result.tools) == 2 + assert result.tools[0].name == "tool1" + assert result.tools[1].name == "tool2" + + +class TestContent: + """Test content types.""" + + def test_text_content(self): + """Test TextContent creation.""" + annotations = Annotations(audience=["user"], priority=0.8) + content = TextContent(type="text", text="Hello, world!", annotations=annotations) + + assert content.type == "text" + assert content.text == "Hello, world!" + assert content.annotations is not None + assert content.annotations.priority == 0.8 + + def test_image_content(self): + """Test ImageContent creation.""" + content = ImageContent(type="image", data="base64encodeddata", mimeType="image/png") + + assert content.type == "image" + assert content.data == "base64encodeddata" + assert content.mimeType == "image/png" + + +class TestOAuth: + """Test OAuth-related types.""" + + def test_oauth_client_metadata(self): + """Test OAuthClientMetadata creation.""" + metadata = OAuthClientMetadata( + client_name="Test Client", + redirect_uris=["https://example.com/callback"], + grant_types=["authorization_code", "refresh_token"], + response_types=["code"], + token_endpoint_auth_method="none", + client_uri="https://example.com", + scope="read write", + ) + + assert metadata.client_name == "Test Client" + assert len(metadata.redirect_uris) == 1 + assert "authorization_code" in metadata.grant_types + + def test_oauth_client_information(self): + """Test OAuthClientInformation creation.""" + info = OAuthClientInformation(client_id="test-client-id", client_secret="test-secret") + + assert info.client_id == "test-client-id" + assert info.client_secret == "test-secret" + + def test_oauth_client_information_without_secret(self): + """Test OAuthClientInformation without secret.""" + info = OAuthClientInformation(client_id="public-client") + + assert info.client_id == "public-client" + assert info.client_secret is None + + def test_oauth_tokens(self): + """Test OAuthTokens creation.""" + tokens = OAuthTokens( + access_token="access-token-123", + token_type="Bearer", + expires_in=3600, + refresh_token="refresh-token-456", + scope="read write", + ) + + assert tokens.access_token == "access-token-123" + assert tokens.token_type == "Bearer" + assert tokens.expires_in == 3600 + assert tokens.refresh_token == "refresh-token-456" + assert tokens.scope == "read write" + + def test_oauth_metadata(self): + """Test OAuthMetadata creation.""" + metadata = OAuthMetadata( + authorization_endpoint="https://auth.example.com/authorize", + token_endpoint="https://auth.example.com/token", + registration_endpoint="https://auth.example.com/register", + response_types_supported=["code", "token"], + grant_types_supported=["authorization_code", "refresh_token"], + code_challenge_methods_supported=["plain", "S256"], + ) + + assert metadata.authorization_endpoint == "https://auth.example.com/authorize" + assert "code" in metadata.response_types_supported + assert "S256" in metadata.code_challenge_methods_supported + + +class TestNotifications: + """Test notification types.""" + + def test_progress_notification(self): + """Test ProgressNotification creation.""" + params = ProgressNotificationParams( + progressToken="progress-123", progress=50.0, total=100.0, message="Processing... 50%" + ) + + notification = ProgressNotification(params=params) + + assert notification.method == "notifications/progress" + assert notification.params.progressToken == "progress-123" + assert notification.params.progress == 50.0 + assert notification.params.total == 100.0 + assert notification.params.message == "Processing... 50%" + + def test_ping_request(self): + """Test PingRequest creation.""" + request = PingRequest() + assert request.method == "ping" + assert request.params is None + + +class TestCompletion: + """Test completion-related types.""" + + def test_completion_context(self): + """Test CompletionContext creation.""" + context = CompletionContext(arguments={"template_var": "value"}) + assert context.arguments == {"template_var": "value"} + + def test_resource_template_reference(self): + """Test ResourceTemplateReference creation.""" + ref = ResourceTemplateReference(type="ref/resource", uri="file:///path/to/{filename}") + assert ref.type == "ref/resource" + assert ref.uri == "file:///path/to/{filename}" + + def test_prompt_reference(self): + """Test PromptReference creation.""" + ref = PromptReference(type="ref/prompt", name="test_prompt") + assert ref.type == "ref/prompt" + assert ref.name == "test_prompt" + + def test_complete_request(self): + """Test CompleteRequest creation.""" + ref = PromptReference(type="ref/prompt", name="test_prompt") + arg = CompletionArgument(name="arg1", value="val") + + params = CompleteRequestParams(ref=ref, argument=arg, context=CompletionContext(arguments={"key": "value"})) + + request = CompleteRequest(params=params) + + assert request.method == "completion/complete" + assert request.params.ref.name == "test_prompt" # type: ignore + assert request.params.argument.name == "arg1" + + def test_complete_result(self): + """Test CompleteResult creation.""" + completion = Completion(values=["option1", "option2", "option3"], total=10, hasMore=True) + + result = CompleteResult(completion=completion) + + assert len(result.completion.values) == 3 + assert result.completion.total == 10 + assert result.completion.hasMore is True + + +class TestValidation: + """Test validation of various types.""" + + def test_invalid_jsonrpc_version(self): + """Test invalid JSON-RPC version validation.""" + with pytest.raises(ValidationError): + JSONRPCRequest( + jsonrpc="1.0", # Invalid version + id=1, + method="test", + ) + + def test_tool_annotations_validation(self): + """Test ToolAnnotations with invalid values.""" + # Valid annotations + annotations = ToolAnnotations( + title="Test", readOnlyHint=True, destructiveHint=False, idempotentHint=True, openWorldHint=False + ) + assert annotations.title == "Test" + + def test_extra_fields_allowed(self): + """Test that extra fields are allowed in models.""" + # Most models should allow extra fields + tool = Tool( + name="test", + inputSchema={}, + customField="allowed", # type: ignore + ) + assert tool.customField == "allowed" # type: ignore + + def test_result_meta_alias(self): + """Test Result model with _meta alias.""" + # Create with the field name (not alias) + result = Result(_meta={"key": "value"}) + + # Verify the field is set correctly + assert result.meta == {"key": "value"} + + # Dump with alias + dumped = result.model_dump(by_alias=True) + assert "_meta" in dumped + assert dumped["_meta"] == {"key": "value"} diff --git a/api/tests/unit_tests/core/mcp/test_utils.py b/api/tests/unit_tests/core/mcp/test_utils.py new file mode 100644 index 0000000000..ca41d5f4c1 --- /dev/null +++ b/api/tests/unit_tests/core/mcp/test_utils.py @@ -0,0 +1,355 @@ +"""Unit tests for MCP utils module.""" + +import json +from collections.abc import Generator +from unittest.mock import MagicMock, Mock, patch + +import httpx +import httpx_sse +import pytest + +from core.mcp.utils import ( + STATUS_FORCELIST, + create_mcp_error_response, + create_ssrf_proxy_mcp_http_client, + ssrf_proxy_sse_connect, +) + + +class TestConstants: + """Test module constants.""" + + def test_status_forcelist(self): + """Test STATUS_FORCELIST contains expected HTTP status codes.""" + assert STATUS_FORCELIST == [429, 500, 502, 503, 504] + assert 429 in STATUS_FORCELIST # Too Many Requests + assert 500 in STATUS_FORCELIST # Internal Server Error + assert 502 in STATUS_FORCELIST # Bad Gateway + assert 503 in STATUS_FORCELIST # Service Unavailable + assert 504 in STATUS_FORCELIST # Gateway Timeout + + +class TestCreateSSRFProxyMCPHTTPClient: + """Test create_ssrf_proxy_mcp_http_client function.""" + + @patch("core.mcp.utils.dify_config") + def test_create_client_with_all_url_proxy(self, mock_config): + """Test client creation with SSRF_PROXY_ALL_URL configured.""" + mock_config.SSRF_PROXY_ALL_URL = "http://proxy.example.com:8080" + mock_config.HTTP_REQUEST_NODE_SSL_VERIFY = True + + client = create_ssrf_proxy_mcp_http_client( + headers={"Authorization": "Bearer token"}, timeout=httpx.Timeout(30.0) + ) + + assert isinstance(client, httpx.Client) + assert client.headers["Authorization"] == "Bearer token" + assert client.timeout.connect == 30.0 + assert client.follow_redirects is True + + # Clean up + client.close() + + @patch("core.mcp.utils.dify_config") + def test_create_client_with_http_https_proxies(self, mock_config): + """Test client creation with separate HTTP/HTTPS proxies.""" + mock_config.SSRF_PROXY_ALL_URL = None + mock_config.SSRF_PROXY_HTTP_URL = "http://http-proxy.example.com:8080" + mock_config.SSRF_PROXY_HTTPS_URL = "http://https-proxy.example.com:8443" + mock_config.HTTP_REQUEST_NODE_SSL_VERIFY = False + + client = create_ssrf_proxy_mcp_http_client() + + assert isinstance(client, httpx.Client) + assert client.follow_redirects is True + + # Clean up + client.close() + + @patch("core.mcp.utils.dify_config") + def test_create_client_without_proxy(self, mock_config): + """Test client creation without proxy configuration.""" + mock_config.SSRF_PROXY_ALL_URL = None + mock_config.SSRF_PROXY_HTTP_URL = None + mock_config.SSRF_PROXY_HTTPS_URL = None + mock_config.HTTP_REQUEST_NODE_SSL_VERIFY = True + + headers = {"X-Custom-Header": "value"} + timeout = httpx.Timeout(timeout=30.0, connect=5.0, read=10.0, write=30.0) + + client = create_ssrf_proxy_mcp_http_client(headers=headers, timeout=timeout) + + assert isinstance(client, httpx.Client) + assert client.headers["X-Custom-Header"] == "value" + assert client.timeout.connect == 5.0 + assert client.timeout.read == 10.0 + assert client.follow_redirects is True + + # Clean up + client.close() + + @patch("core.mcp.utils.dify_config") + def test_create_client_default_params(self, mock_config): + """Test client creation with default parameters.""" + mock_config.SSRF_PROXY_ALL_URL = None + mock_config.SSRF_PROXY_HTTP_URL = None + mock_config.SSRF_PROXY_HTTPS_URL = None + mock_config.HTTP_REQUEST_NODE_SSL_VERIFY = True + + client = create_ssrf_proxy_mcp_http_client() + + assert isinstance(client, httpx.Client) + # httpx.Client adds default headers, so we just check it's a Headers object + assert isinstance(client.headers, httpx.Headers) + # When no timeout is provided, httpx uses its default timeout + assert client.timeout is not None + + # Clean up + client.close() + + +class TestSSRFProxySSEConnect: + """Test ssrf_proxy_sse_connect function.""" + + @patch("core.mcp.utils.connect_sse") + @patch("core.mcp.utils.create_ssrf_proxy_mcp_http_client") + def test_sse_connect_with_provided_client(self, mock_create_client, mock_connect_sse): + """Test SSE connection with pre-configured client.""" + # Setup mocks + mock_client = Mock(spec=httpx.Client) + mock_event_source = Mock(spec=httpx_sse.EventSource) + mock_context = MagicMock() + mock_context.__enter__.return_value = mock_event_source + mock_connect_sse.return_value = mock_context + + # Call with provided client + result = ssrf_proxy_sse_connect( + "http://example.com/sse", client=mock_client, method="POST", headers={"Authorization": "Bearer token"} + ) + + # Verify client creation was not called + mock_create_client.assert_not_called() + + # Verify connect_sse was called correctly + mock_connect_sse.assert_called_once_with( + mock_client, "POST", "http://example.com/sse", headers={"Authorization": "Bearer token"} + ) + + # Verify result + assert result == mock_context + + @patch("core.mcp.utils.connect_sse") + @patch("core.mcp.utils.create_ssrf_proxy_mcp_http_client") + @patch("core.mcp.utils.dify_config") + def test_sse_connect_without_client(self, mock_config, mock_create_client, mock_connect_sse): + """Test SSE connection without pre-configured client.""" + # Setup config + mock_config.SSRF_DEFAULT_TIME_OUT = 30.0 + mock_config.SSRF_DEFAULT_CONNECT_TIME_OUT = 10.0 + mock_config.SSRF_DEFAULT_READ_TIME_OUT = 60.0 + mock_config.SSRF_DEFAULT_WRITE_TIME_OUT = 30.0 + + # Setup mocks + mock_client = Mock(spec=httpx.Client) + mock_create_client.return_value = mock_client + + mock_event_source = Mock(spec=httpx_sse.EventSource) + mock_context = MagicMock() + mock_context.__enter__.return_value = mock_event_source + mock_connect_sse.return_value = mock_context + + # Call without client + result = ssrf_proxy_sse_connect("http://example.com/sse", headers={"X-Custom": "value"}) + + # Verify client was created + mock_create_client.assert_called_once() + call_args = mock_create_client.call_args + assert call_args[1]["headers"] == {"X-Custom": "value"} + + timeout = call_args[1]["timeout"] + # httpx.Timeout object has these attributes + assert isinstance(timeout, httpx.Timeout) + assert timeout.connect == 10.0 + assert timeout.read == 60.0 + assert timeout.write == 30.0 + + # Verify connect_sse was called + mock_connect_sse.assert_called_once_with( + mock_client, + "GET", # Default method + "http://example.com/sse", + ) + + # Verify result + assert result == mock_context + + @patch("core.mcp.utils.connect_sse") + @patch("core.mcp.utils.create_ssrf_proxy_mcp_http_client") + def test_sse_connect_with_custom_timeout(self, mock_create_client, mock_connect_sse): + """Test SSE connection with custom timeout.""" + # Setup mocks + mock_client = Mock(spec=httpx.Client) + mock_create_client.return_value = mock_client + + mock_event_source = Mock(spec=httpx_sse.EventSource) + mock_context = MagicMock() + mock_context.__enter__.return_value = mock_event_source + mock_connect_sse.return_value = mock_context + + custom_timeout = httpx.Timeout(timeout=60.0, read=120.0) + + # Call with custom timeout + result = ssrf_proxy_sse_connect("http://example.com/sse", timeout=custom_timeout) + + # Verify client was created with custom timeout + mock_create_client.assert_called_once() + call_args = mock_create_client.call_args + assert call_args[1]["timeout"] == custom_timeout + + # Verify result + assert result == mock_context + + @patch("core.mcp.utils.connect_sse") + @patch("core.mcp.utils.create_ssrf_proxy_mcp_http_client") + def test_sse_connect_error_cleanup(self, mock_create_client, mock_connect_sse): + """Test SSE connection cleans up client on error.""" + # Setup mocks + mock_client = Mock(spec=httpx.Client) + mock_create_client.return_value = mock_client + + # Make connect_sse raise an exception + mock_connect_sse.side_effect = httpx.ConnectError("Connection failed") + + # Call should raise the exception + with pytest.raises(httpx.ConnectError): + ssrf_proxy_sse_connect("http://example.com/sse") + + # Verify client was cleaned up + mock_client.close.assert_called_once() + + @patch("core.mcp.utils.connect_sse") + def test_sse_connect_error_no_cleanup_with_provided_client(self, mock_connect_sse): + """Test SSE connection doesn't clean up provided client on error.""" + # Setup mocks + mock_client = Mock(spec=httpx.Client) + + # Make connect_sse raise an exception + mock_connect_sse.side_effect = httpx.ConnectError("Connection failed") + + # Call should raise the exception + with pytest.raises(httpx.ConnectError): + ssrf_proxy_sse_connect("http://example.com/sse", client=mock_client) + + # Verify client was NOT cleaned up (because it was provided) + mock_client.close.assert_not_called() + + +class TestCreateMCPErrorResponse: + """Test create_mcp_error_response function.""" + + def test_create_error_response_basic(self): + """Test creating basic error response.""" + generator = create_mcp_error_response(request_id="req-123", code=-32600, message="Invalid Request") + + # Generator should yield bytes + assert isinstance(generator, Generator) + + # Get the response + response_bytes = next(generator) + assert isinstance(response_bytes, bytes) + + # Parse the response + response_str = response_bytes.decode("utf-8") + response_json = json.loads(response_str) + + assert response_json["jsonrpc"] == "2.0" + assert response_json["id"] == "req-123" + assert response_json["error"]["code"] == -32600 + assert response_json["error"]["message"] == "Invalid Request" + assert response_json["error"]["data"] is None + + # Generator should be exhausted + with pytest.raises(StopIteration): + next(generator) + + def test_create_error_response_with_data(self): + """Test creating error response with additional data.""" + error_data = {"field": "username", "reason": "required"} + + generator = create_mcp_error_response( + request_id=456, # Numeric ID + code=-32602, + message="Invalid params", + data=error_data, + ) + + response_bytes = next(generator) + response_json = json.loads(response_bytes.decode("utf-8")) + + assert response_json["id"] == 456 + assert response_json["error"]["code"] == -32602 + assert response_json["error"]["message"] == "Invalid params" + assert response_json["error"]["data"] == error_data + + def test_create_error_response_without_request_id(self): + """Test creating error response without request ID.""" + generator = create_mcp_error_response(request_id=None, code=-32700, message="Parse error") + + response_bytes = next(generator) + response_json = json.loads(response_bytes.decode("utf-8")) + + # Should default to ID 1 + assert response_json["id"] == 1 + assert response_json["error"]["code"] == -32700 + assert response_json["error"]["message"] == "Parse error" + + def test_create_error_response_with_complex_data(self): + """Test creating error response with complex error data.""" + complex_data = { + "errors": [{"field": "name", "message": "Too short"}, {"field": "email", "message": "Invalid format"}], + "timestamp": "2024-01-01T00:00:00Z", + } + + generator = create_mcp_error_response( + request_id="complex-req", code=-32602, message="Validation failed", data=complex_data + ) + + response_bytes = next(generator) + response_json = json.loads(response_bytes.decode("utf-8")) + + assert response_json["error"]["data"] == complex_data + assert len(response_json["error"]["data"]["errors"]) == 2 + + def test_create_error_response_encoding(self): + """Test error response with non-ASCII characters.""" + generator = create_mcp_error_response( + request_id="unicode-req", + code=-32603, + message="内部错误", # Chinese characters + data={"details": "エラー詳細"}, # Japanese characters + ) + + response_bytes = next(generator) + + # Should be valid UTF-8 + response_str = response_bytes.decode("utf-8") + response_json = json.loads(response_str) + + assert response_json["error"]["message"] == "内部错误" + assert response_json["error"]["data"]["details"] == "エラー詳細" + + def test_create_error_response_yields_once(self): + """Test that error response generator yields exactly once.""" + generator = create_mcp_error_response(request_id="test", code=-32600, message="Test") + + # First yield should work + first_yield = next(generator) + assert isinstance(first_yield, bytes) + + # Second yield should raise StopIteration + with pytest.raises(StopIteration): + next(generator) + + # Subsequent calls should also raise + with pytest.raises(StopIteration): + next(generator) diff --git a/api/tests/unit_tests/core/plugin/utils/test_http_parser.py b/api/tests/unit_tests/core/plugin/utils/test_http_parser.py new file mode 100644 index 0000000000..1c2e0c96f8 --- /dev/null +++ b/api/tests/unit_tests/core/plugin/utils/test_http_parser.py @@ -0,0 +1,655 @@ +import pytest +from flask import Request, Response + +from core.plugin.utils.http_parser import ( + deserialize_request, + deserialize_response, + serialize_request, + serialize_response, +) + + +class TestSerializeRequest: + def test_serialize_simple_get_request(self): + # Create a simple GET request + environ = { + "REQUEST_METHOD": "GET", + "PATH_INFO": "/api/test", + "QUERY_STRING": "", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8000", + "wsgi.input": None, + "wsgi.url_scheme": "http", + } + request = Request(environ) + + raw_data = serialize_request(request) + + assert raw_data.startswith(b"GET /api/test HTTP/1.1\r\n") + assert b"\r\n\r\n" in raw_data # Empty line between headers and body + + def test_serialize_request_with_query_params(self): + # Create a GET request with query parameters + environ = { + "REQUEST_METHOD": "GET", + "PATH_INFO": "/api/search", + "QUERY_STRING": "q=test&limit=10", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8000", + "wsgi.input": None, + "wsgi.url_scheme": "http", + } + request = Request(environ) + + raw_data = serialize_request(request) + + assert raw_data.startswith(b"GET /api/search?q=test&limit=10 HTTP/1.1\r\n") + + def test_serialize_post_request_with_body(self): + # Create a POST request with body + from io import BytesIO + + body = b'{"name": "test", "value": 123}' + environ = { + "REQUEST_METHOD": "POST", + "PATH_INFO": "/api/data", + "QUERY_STRING": "", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8000", + "wsgi.input": BytesIO(body), + "wsgi.url_scheme": "http", + "CONTENT_LENGTH": str(len(body)), + "CONTENT_TYPE": "application/json", + "HTTP_CONTENT_TYPE": "application/json", + } + request = Request(environ) + + raw_data = serialize_request(request) + + assert b"POST /api/data HTTP/1.1\r\n" in raw_data + assert b"Content-Type: application/json" in raw_data + assert raw_data.endswith(body) + + def test_serialize_request_with_custom_headers(self): + # Create a request with custom headers + environ = { + "REQUEST_METHOD": "GET", + "PATH_INFO": "/api/test", + "QUERY_STRING": "", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8000", + "wsgi.input": None, + "wsgi.url_scheme": "http", + "HTTP_AUTHORIZATION": "Bearer token123", + "HTTP_X_CUSTOM_HEADER": "custom-value", + } + request = Request(environ) + + raw_data = serialize_request(request) + + assert b"Authorization: Bearer token123" in raw_data + assert b"X-Custom-Header: custom-value" in raw_data + + +class TestDeserializeRequest: + def test_deserialize_simple_get_request(self): + raw_data = b"GET /api/test HTTP/1.1\r\nHost: localhost:8000\r\n\r\n" + + request = deserialize_request(raw_data) + + assert request.method == "GET" + assert request.path == "/api/test" + assert request.headers.get("Host") == "localhost:8000" + + def test_deserialize_request_with_query_params(self): + raw_data = b"GET /api/search?q=test&limit=10 HTTP/1.1\r\nHost: example.com\r\n\r\n" + + request = deserialize_request(raw_data) + + assert request.method == "GET" + assert request.path == "/api/search" + assert request.query_string == b"q=test&limit=10" + assert request.args.get("q") == "test" + assert request.args.get("limit") == "10" + + def test_deserialize_post_request_with_body(self): + body = b'{"name": "test", "value": 123}' + raw_data = ( + b"POST /api/data HTTP/1.1\r\n" + b"Host: localhost\r\n" + b"Content-Type: application/json\r\n" + b"Content-Length: " + str(len(body)).encode() + b"\r\n" + b"\r\n" + body + ) + + request = deserialize_request(raw_data) + + assert request.method == "POST" + assert request.path == "/api/data" + assert request.content_type == "application/json" + assert request.get_data() == body + + def test_deserialize_request_with_custom_headers(self): + raw_data = ( + b"GET /api/protected HTTP/1.1\r\n" + b"Host: api.example.com\r\n" + b"Authorization: Bearer token123\r\n" + b"X-Custom-Header: custom-value\r\n" + b"User-Agent: TestClient/1.0\r\n" + b"\r\n" + ) + + request = deserialize_request(raw_data) + + assert request.method == "GET" + assert request.headers.get("Authorization") == "Bearer token123" + assert request.headers.get("X-Custom-Header") == "custom-value" + assert request.headers.get("User-Agent") == "TestClient/1.0" + + def test_deserialize_request_with_multiline_body(self): + body = b"line1\r\nline2\r\nline3" + raw_data = b"PUT /api/text HTTP/1.1\r\nHost: localhost\r\nContent-Type: text/plain\r\n\r\n" + body + + request = deserialize_request(raw_data) + + assert request.method == "PUT" + assert request.get_data() == body + + def test_deserialize_invalid_request_line(self): + raw_data = b"INVALID\r\n\r\n" # Only one part, should fail + + with pytest.raises(ValueError, match="Invalid request line"): + deserialize_request(raw_data) + + def test_roundtrip_request(self): + # Test that serialize -> deserialize produces equivalent request + from io import BytesIO + + body = b"test body content" + environ = { + "REQUEST_METHOD": "POST", + "PATH_INFO": "/api/echo", + "QUERY_STRING": "format=json", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8080", + "wsgi.input": BytesIO(body), + "wsgi.url_scheme": "http", + "CONTENT_LENGTH": str(len(body)), + "CONTENT_TYPE": "text/plain", + "HTTP_CONTENT_TYPE": "text/plain", + "HTTP_X_REQUEST_ID": "req-123", + } + original_request = Request(environ) + + # Serialize and deserialize + raw_data = serialize_request(original_request) + restored_request = deserialize_request(raw_data) + + # Verify key properties are preserved + assert restored_request.method == original_request.method + assert restored_request.path == original_request.path + assert restored_request.query_string == original_request.query_string + assert restored_request.get_data() == body + assert restored_request.headers.get("X-Request-Id") == "req-123" + + +class TestSerializeResponse: + def test_serialize_simple_response(self): + response = Response("Hello, World!", status=200) + + raw_data = serialize_response(response) + + assert raw_data.startswith(b"HTTP/1.1 200 OK\r\n") + assert b"\r\n\r\n" in raw_data + assert raw_data.endswith(b"Hello, World!") + + def test_serialize_response_with_headers(self): + response = Response( + '{"status": "success"}', + status=201, + headers={ + "Content-Type": "application/json", + "X-Request-Id": "req-456", + }, + ) + + raw_data = serialize_response(response) + + assert b"HTTP/1.1 201 CREATED\r\n" in raw_data + assert b"Content-Type: application/json" in raw_data + assert b"X-Request-Id: req-456" in raw_data + assert raw_data.endswith(b'{"status": "success"}') + + def test_serialize_error_response(self): + response = Response( + "Not Found", + status=404, + headers={"Content-Type": "text/plain"}, + ) + + raw_data = serialize_response(response) + + assert b"HTTP/1.1 404 NOT FOUND\r\n" in raw_data + assert b"Content-Type: text/plain" in raw_data + assert raw_data.endswith(b"Not Found") + + def test_serialize_response_without_body(self): + response = Response(status=204) # No Content + + raw_data = serialize_response(response) + + assert b"HTTP/1.1 204 NO CONTENT\r\n" in raw_data + assert raw_data.endswith(b"\r\n\r\n") # Should end with empty line + + def test_serialize_response_with_binary_body(self): + binary_data = b"\x00\x01\x02\x03\x04\x05" + response = Response( + binary_data, + status=200, + headers={"Content-Type": "application/octet-stream"}, + ) + + raw_data = serialize_response(response) + + assert b"HTTP/1.1 200 OK\r\n" in raw_data + assert b"Content-Type: application/octet-stream" in raw_data + assert raw_data.endswith(binary_data) + + +class TestDeserializeResponse: + def test_deserialize_simple_response(self): + raw_data = b"HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\n\r\nHello, World!" + + response = deserialize_response(raw_data) + + assert response.status_code == 200 + assert response.get_data() == b"Hello, World!" + assert response.headers.get("Content-Type") == "text/plain" + + def test_deserialize_response_with_json(self): + body = b'{"result": "success", "data": [1, 2, 3]}' + raw_data = ( + b"HTTP/1.1 201 Created\r\n" + b"Content-Type: application/json\r\n" + b"Content-Length: " + str(len(body)).encode() + b"\r\n" + b"X-Custom-Header: test-value\r\n" + b"\r\n" + body + ) + + response = deserialize_response(raw_data) + + assert response.status_code == 201 + assert response.get_data() == body + assert response.headers.get("Content-Type") == "application/json" + assert response.headers.get("X-Custom-Header") == "test-value" + + def test_deserialize_error_response(self): + raw_data = b"HTTP/1.1 404 Not Found\r\nContent-Type: text/html\r\n\r\nPage not found" + + response = deserialize_response(raw_data) + + assert response.status_code == 404 + assert response.get_data() == b"Page not found" + + def test_deserialize_response_without_body(self): + raw_data = b"HTTP/1.1 204 No Content\r\n\r\n" + + response = deserialize_response(raw_data) + + assert response.status_code == 204 + assert response.get_data() == b"" + + def test_deserialize_response_with_multiline_body(self): + body = b"Line 1\r\nLine 2\r\nLine 3" + raw_data = b"HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\n\r\n" + body + + response = deserialize_response(raw_data) + + assert response.status_code == 200 + assert response.get_data() == body + + def test_deserialize_response_minimal_status_line(self): + # Test with minimal status line (no status text) + raw_data = b"HTTP/1.1 200\r\n\r\nOK" + + response = deserialize_response(raw_data) + + assert response.status_code == 200 + assert response.get_data() == b"OK" + + def test_deserialize_invalid_status_line(self): + raw_data = b"INVALID\r\n\r\n" + + with pytest.raises(ValueError, match="Invalid status line"): + deserialize_response(raw_data) + + def test_roundtrip_response(self): + # Test that serialize -> deserialize produces equivalent response + original_response = Response( + '{"message": "test"}', + status=200, + headers={ + "Content-Type": "application/json", + "X-Request-Id": "abc-123", + "Cache-Control": "no-cache", + }, + ) + + # Serialize and deserialize + raw_data = serialize_response(original_response) + restored_response = deserialize_response(raw_data) + + # Verify key properties are preserved + assert restored_response.status_code == original_response.status_code + assert restored_response.get_data() == original_response.get_data() + assert restored_response.headers.get("Content-Type") == "application/json" + assert restored_response.headers.get("X-Request-Id") == "abc-123" + assert restored_response.headers.get("Cache-Control") == "no-cache" + + +class TestEdgeCases: + def test_request_with_empty_headers(self): + raw_data = b"GET / HTTP/1.1\r\n\r\n" + + request = deserialize_request(raw_data) + + assert request.method == "GET" + assert request.path == "/" + + def test_response_with_empty_headers(self): + raw_data = b"HTTP/1.1 200 OK\r\n\r\nSuccess" + + response = deserialize_response(raw_data) + + assert response.status_code == 200 + assert response.get_data() == b"Success" + + def test_request_with_special_characters_in_path(self): + raw_data = b"GET /api/test%20path?key=%26value HTTP/1.1\r\n\r\n" + + request = deserialize_request(raw_data) + + assert request.method == "GET" + assert "/api/test%20path" in request.full_path + + def test_response_with_binary_content(self): + binary_body = bytes(range(256)) # All possible byte values + raw_data = b"HTTP/1.1 200 OK\r\nContent-Type: application/octet-stream\r\n\r\n" + binary_body + + response = deserialize_response(raw_data) + + assert response.status_code == 200 + assert response.get_data() == binary_body + + +class TestFileUploads: + def test_serialize_request_with_text_file_upload(self): + # Test multipart/form-data request with text file + from io import BytesIO + + boundary = "----WebKitFormBoundary7MA4YWxkTrZu0gW" + text_content = "Hello, this is a test file content!\nWith multiple lines." + body = ( + f"------{boundary}\r\n" + f'Content-Disposition: form-data; name="file"; filename="test.txt"\r\n' + f"Content-Type: text/plain\r\n" + f"\r\n" + f"{text_content}\r\n" + f"------{boundary}\r\n" + f'Content-Disposition: form-data; name="description"\r\n' + f"\r\n" + f"Test file upload\r\n" + f"------{boundary}--\r\n" + ).encode() + + environ = { + "REQUEST_METHOD": "POST", + "PATH_INFO": "/api/upload", + "QUERY_STRING": "", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8000", + "wsgi.input": BytesIO(body), + "wsgi.url_scheme": "http", + "CONTENT_LENGTH": str(len(body)), + "CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + "HTTP_CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + } + request = Request(environ) + + raw_data = serialize_request(request) + + assert b"POST /api/upload HTTP/1.1\r\n" in raw_data + assert f"Content-Type: multipart/form-data; boundary={boundary}".encode() in raw_data + assert b'Content-Disposition: form-data; name="file"; filename="test.txt"' in raw_data + assert text_content.encode() in raw_data + + def test_deserialize_request_with_text_file_upload(self): + # Test deserializing multipart/form-data request with text file + boundary = "----WebKitFormBoundary7MA4YWxkTrZu0gW" + text_content = "Sample text file content\nLine 2\nLine 3" + body = ( + f"------{boundary}\r\n" + f'Content-Disposition: form-data; name="document"; filename="document.txt"\r\n' + f"Content-Type: text/plain\r\n" + f"\r\n" + f"{text_content}\r\n" + f"------{boundary}\r\n" + f'Content-Disposition: form-data; name="title"\r\n' + f"\r\n" + f"My Document\r\n" + f"------{boundary}--\r\n" + ).encode() + + raw_data = ( + b"POST /api/documents HTTP/1.1\r\n" + b"Host: example.com\r\n" + b"Content-Type: multipart/form-data; boundary=" + boundary.encode() + b"\r\n" + b"Content-Length: " + str(len(body)).encode() + b"\r\n" + b"\r\n" + body + ) + + request = deserialize_request(raw_data) + + assert request.method == "POST" + assert request.path == "/api/documents" + assert "multipart/form-data" in request.content_type + # The body should contain the multipart data + request_body = request.get_data() + assert b"document.txt" in request_body + assert text_content.encode() in request_body + + def test_serialize_request_with_binary_file_upload(self): + # Test multipart/form-data request with binary file (e.g., image) + from io import BytesIO + + boundary = "----BoundaryString123" + # Simulate a small PNG file header + binary_content = b"\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10" + + # Build multipart body + body_parts = [] + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="image"; filename="test.png"') + body_parts.append(b"Content-Type: image/png") + body_parts.append(b"") + body_parts.append(binary_content) + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="caption"') + body_parts.append(b"") + body_parts.append(b"Test image") + body_parts.append(f"------{boundary}--".encode()) + + body = b"\r\n".join(body_parts) + + environ = { + "REQUEST_METHOD": "POST", + "PATH_INFO": "/api/images", + "QUERY_STRING": "", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8000", + "wsgi.input": BytesIO(body), + "wsgi.url_scheme": "http", + "CONTENT_LENGTH": str(len(body)), + "CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + "HTTP_CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + } + request = Request(environ) + + raw_data = serialize_request(request) + + assert b"POST /api/images HTTP/1.1\r\n" in raw_data + assert f"Content-Type: multipart/form-data; boundary={boundary}".encode() in raw_data + assert b'filename="test.png"' in raw_data + assert b"Content-Type: image/png" in raw_data + assert binary_content in raw_data + + def test_deserialize_request_with_binary_file_upload(self): + # Test deserializing multipart/form-data request with binary file + boundary = "----BoundaryABC123" + # Simulate a small JPEG file header + binary_content = b"\xff\xd8\xff\xe0\x00\x10JFIF\x00\x01\x01\x00\x00\x01\x00\x01\x00\x00" + + body_parts = [] + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="photo"; filename="photo.jpg"') + body_parts.append(b"Content-Type: image/jpeg") + body_parts.append(b"") + body_parts.append(binary_content) + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="album"') + body_parts.append(b"") + body_parts.append(b"Vacation 2024") + body_parts.append(f"------{boundary}--".encode()) + + body = b"\r\n".join(body_parts) + + raw_data = ( + b"POST /api/photos HTTP/1.1\r\n" + b"Host: api.example.com\r\n" + b"Content-Type: multipart/form-data; boundary=" + boundary.encode() + b"\r\n" + b"Content-Length: " + str(len(body)).encode() + b"\r\n" + b"Accept: application/json\r\n" + b"\r\n" + body + ) + + request = deserialize_request(raw_data) + + assert request.method == "POST" + assert request.path == "/api/photos" + assert "multipart/form-data" in request.content_type + assert request.headers.get("Accept") == "application/json" + + # Verify the binary content is preserved + request_body = request.get_data() + assert b"photo.jpg" in request_body + assert b"image/jpeg" in request_body + assert binary_content in request_body + assert b"Vacation 2024" in request_body + + def test_serialize_request_with_multiple_files(self): + # Test request with multiple file uploads + from io import BytesIO + + boundary = "----MultiFilesBoundary" + text_file = b"Text file contents" + binary_file = b"\x00\x01\x02\x03\x04\x05" + + body_parts = [] + # First file (text) + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="files"; filename="doc.txt"') + body_parts.append(b"Content-Type: text/plain") + body_parts.append(b"") + body_parts.append(text_file) + # Second file (binary) + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="files"; filename="data.bin"') + body_parts.append(b"Content-Type: application/octet-stream") + body_parts.append(b"") + body_parts.append(binary_file) + # Additional form field + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="folder"') + body_parts.append(b"") + body_parts.append(b"uploads/2024") + body_parts.append(f"------{boundary}--".encode()) + + body = b"\r\n".join(body_parts) + + environ = { + "REQUEST_METHOD": "POST", + "PATH_INFO": "/api/batch-upload", + "QUERY_STRING": "", + "SERVER_NAME": "localhost", + "SERVER_PORT": "8000", + "wsgi.input": BytesIO(body), + "wsgi.url_scheme": "https", + "CONTENT_LENGTH": str(len(body)), + "CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + "HTTP_CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + "HTTP_X_FORWARDED_PROTO": "https", + } + request = Request(environ) + + raw_data = serialize_request(request) + + assert b"POST /api/batch-upload HTTP/1.1\r\n" in raw_data + assert b"doc.txt" in raw_data + assert b"data.bin" in raw_data + assert text_file in raw_data + assert binary_file in raw_data + assert b"uploads/2024" in raw_data + + def test_roundtrip_file_upload_request(self): + # Test that file upload request survives serialize -> deserialize + from io import BytesIO + + boundary = "----RoundTripBoundary" + file_content = b"This is my file content with special chars: \xf0\x9f\x98\x80" + + body_parts = [] + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="upload"; filename="emoji.txt"') + body_parts.append(b"Content-Type: text/plain; charset=utf-8") + body_parts.append(b"") + body_parts.append(file_content) + body_parts.append(f"------{boundary}".encode()) + body_parts.append(b'Content-Disposition: form-data; name="metadata"') + body_parts.append(b"") + body_parts.append(b'{"encoding": "utf-8", "size": 42}') + body_parts.append(f"------{boundary}--".encode()) + + body = b"\r\n".join(body_parts) + + environ = { + "REQUEST_METHOD": "PUT", + "PATH_INFO": "/api/files/123", + "QUERY_STRING": "version=2", + "SERVER_NAME": "storage.example.com", + "SERVER_PORT": "443", + "wsgi.input": BytesIO(body), + "wsgi.url_scheme": "https", + "CONTENT_LENGTH": str(len(body)), + "CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + "HTTP_CONTENT_TYPE": f"multipart/form-data; boundary={boundary}", + "HTTP_AUTHORIZATION": "Bearer token123", + "HTTP_X_FORWARDED_PROTO": "https", + } + original_request = Request(environ) + + # Serialize and deserialize + raw_data = serialize_request(original_request) + restored_request = deserialize_request(raw_data) + + # Verify the request is preserved + assert restored_request.method == "PUT" + assert restored_request.path == "/api/files/123" + assert restored_request.query_string == b"version=2" + assert "multipart/form-data" in restored_request.content_type + assert boundary in restored_request.content_type + + # Verify file content is preserved + restored_body = restored_request.get_data() + assert b"emoji.txt" in restored_body + assert file_content in restored_body + assert b'{"encoding": "utf-8", "size": 42}' in restored_body diff --git a/api/tests/unit_tests/core/rag/extractor/test_word_extractor.py b/api/tests/unit_tests/core/rag/extractor/test_word_extractor.py new file mode 100644 index 0000000000..3635e4dbf9 --- /dev/null +++ b/api/tests/unit_tests/core/rag/extractor/test_word_extractor.py @@ -0,0 +1,49 @@ +"""Primarily used for testing merged cell scenarios""" + +from docx import Document + +from core.rag.extractor.word_extractor import WordExtractor + + +def _generate_table_with_merged_cells(): + doc = Document() + + """ + The table looks like this: + +-----+-----+-----+ + | 1-1 & 1-2 | 1-3 | + +-----+-----+-----+ + | 2-1 | 2-2 | 2-3 | + | & |-----+-----+ + | 3-1 | 3-2 | 3-3 | + +-----+-----+-----+ + """ + table = doc.add_table(rows=3, cols=3) + table.style = "Table Grid" + + for i in range(3): + for j in range(3): + cell = table.cell(i, j) + cell.text = f"{i + 1}-{j + 1}" + + # Merge cells + cell_0_0 = table.cell(0, 0) + cell_0_1 = table.cell(0, 1) + merged_cell_1 = cell_0_0.merge(cell_0_1) + merged_cell_1.text = "1-1 & 1-2" + + cell_1_0 = table.cell(1, 0) + cell_2_0 = table.cell(2, 0) + merged_cell_2 = cell_1_0.merge(cell_2_0) + merged_cell_2.text = "2-1 & 3-1" + + ground_truth = [["1-1 & 1-2", "", "1-3"], ["2-1 & 3-1", "2-2", "2-3"], ["2-1 & 3-1", "3-2", "3-3"]] + + return doc.tables[0], ground_truth + + +def test_parse_row(): + table, gt = _generate_table_with_merged_cells() + extractor = object.__new__(WordExtractor) + for idx, row in enumerate(table.rows): + assert extractor._parse_row(row, {}, 3) == gt[idx] diff --git a/api/tests/unit_tests/core/rag/pipeline/test_queue.py b/api/tests/unit_tests/core/rag/pipeline/test_queue.py new file mode 100644 index 0000000000..17c5f3c6b7 --- /dev/null +++ b/api/tests/unit_tests/core/rag/pipeline/test_queue.py @@ -0,0 +1,301 @@ +""" +Unit tests for TenantIsolatedTaskQueue. + +These tests verify the Redis-based task queue functionality for tenant-specific +task management with proper serialization and deserialization. +""" + +import json +from unittest.mock import MagicMock, patch +from uuid import uuid4 + +import pytest +from pydantic import ValidationError + +from core.rag.pipeline.queue import TaskWrapper, TenantIsolatedTaskQueue + + +class TestTaskWrapper: + """Test cases for TaskWrapper serialization/deserialization.""" + + def test_serialize_simple_data(self): + """Test serialization of simple data types.""" + data = {"key": "value", "number": 42, "list": [1, 2, 3]} + wrapper = TaskWrapper(data=data) + + serialized = wrapper.serialize() + assert isinstance(serialized, str) + + # Verify it's valid JSON + parsed = json.loads(serialized) + assert parsed["data"] == data + + def test_serialize_complex_data(self): + """Test serialization of complex nested data.""" + data = { + "nested": {"deep": {"value": "test", "numbers": [1, 2, 3, 4, 5]}}, + "unicode": "测试中文", + "special_chars": "!@#$%^&*()", + } + wrapper = TaskWrapper(data=data) + + serialized = wrapper.serialize() + parsed = json.loads(serialized) + assert parsed["data"] == data + + def test_deserialize_valid_data(self): + """Test deserialization of valid JSON data.""" + original_data = {"key": "value", "number": 42} + # Serialize using TaskWrapper to get the correct format + wrapper = TaskWrapper(data=original_data) + serialized = wrapper.serialize() + + wrapper = TaskWrapper.deserialize(serialized) + assert wrapper.data == original_data + + def test_deserialize_invalid_json(self): + """Test deserialization handles invalid JSON gracefully.""" + invalid_json = "{invalid json}" + + # Pydantic will raise ValidationError for invalid JSON + with pytest.raises(ValidationError): + TaskWrapper.deserialize(invalid_json) + + def test_serialize_ensure_ascii_false(self): + """Test that serialization preserves Unicode characters.""" + data = {"chinese": "中文测试", "emoji": "🚀"} + wrapper = TaskWrapper(data=data) + + serialized = wrapper.serialize() + assert "中文测试" in serialized + assert "🚀" in serialized + + +class TestTenantIsolatedTaskQueue: + """Test cases for TenantIsolatedTaskQueue functionality.""" + + @pytest.fixture + def mock_redis_client(self): + """Mock Redis client for testing.""" + mock_redis = MagicMock() + return mock_redis + + @pytest.fixture + def sample_queue(self, mock_redis_client): + """Create a sample TenantIsolatedTaskQueue instance.""" + return TenantIsolatedTaskQueue("tenant-123", "test-key") + + def test_initialization(self, sample_queue): + """Test queue initialization with correct key generation.""" + assert sample_queue._tenant_id == "tenant-123" + assert sample_queue._unique_key == "test-key" + assert sample_queue._queue == "tenant_self_test-key_task_queue:tenant-123" + assert sample_queue._task_key == "tenant_test-key_task:tenant-123" + + @patch("core.rag.pipeline.queue.redis_client") + def test_get_task_key_exists(self, mock_redis, sample_queue): + """Test getting task key when it exists.""" + mock_redis.get.return_value = "1" + + result = sample_queue.get_task_key() + + assert result == "1" + mock_redis.get.assert_called_once_with("tenant_test-key_task:tenant-123") + + @patch("core.rag.pipeline.queue.redis_client") + def test_get_task_key_not_exists(self, mock_redis, sample_queue): + """Test getting task key when it doesn't exist.""" + mock_redis.get.return_value = None + + result = sample_queue.get_task_key() + + assert result is None + mock_redis.get.assert_called_once_with("tenant_test-key_task:tenant-123") + + @patch("core.rag.pipeline.queue.redis_client") + def test_set_task_waiting_time_default_ttl(self, mock_redis, sample_queue): + """Test setting task waiting flag with default TTL.""" + sample_queue.set_task_waiting_time() + + mock_redis.setex.assert_called_once_with( + "tenant_test-key_task:tenant-123", + 3600, # DEFAULT_TASK_TTL + 1, + ) + + @patch("core.rag.pipeline.queue.redis_client") + def test_set_task_waiting_time_custom_ttl(self, mock_redis, sample_queue): + """Test setting task waiting flag with custom TTL.""" + custom_ttl = 1800 + sample_queue.set_task_waiting_time(custom_ttl) + + mock_redis.setex.assert_called_once_with("tenant_test-key_task:tenant-123", custom_ttl, 1) + + @patch("core.rag.pipeline.queue.redis_client") + def test_delete_task_key(self, mock_redis, sample_queue): + """Test deleting task key.""" + sample_queue.delete_task_key() + + mock_redis.delete.assert_called_once_with("tenant_test-key_task:tenant-123") + + @patch("core.rag.pipeline.queue.redis_client") + def test_push_tasks_string_list(self, mock_redis, sample_queue): + """Test pushing string tasks directly.""" + tasks = ["task1", "task2", "task3"] + + sample_queue.push_tasks(tasks) + + mock_redis.lpush.assert_called_once_with( + "tenant_self_test-key_task_queue:tenant-123", "task1", "task2", "task3" + ) + + @patch("core.rag.pipeline.queue.redis_client") + def test_push_tasks_mixed_types(self, mock_redis, sample_queue): + """Test pushing mixed string and object tasks.""" + tasks = ["string_task", {"object_task": "data", "id": 123}, "another_string"] + + sample_queue.push_tasks(tasks) + + # Verify lpush was called + mock_redis.lpush.assert_called_once() + call_args = mock_redis.lpush.call_args + + # Check queue name + assert call_args[0][0] == "tenant_self_test-key_task_queue:tenant-123" + + # Check serialized tasks + serialized_tasks = call_args[0][1:] + assert len(serialized_tasks) == 3 + assert serialized_tasks[0] == "string_task" + assert serialized_tasks[2] == "another_string" + + # Check object task is serialized as TaskWrapper JSON (without prefix) + # It should be a valid JSON string that can be deserialized by TaskWrapper + wrapper = TaskWrapper.deserialize(serialized_tasks[1]) + assert wrapper.data == {"object_task": "data", "id": 123} + + @patch("core.rag.pipeline.queue.redis_client") + def test_push_tasks_empty_list(self, mock_redis, sample_queue): + """Test pushing empty task list.""" + sample_queue.push_tasks([]) + + mock_redis.lpush.assert_not_called() + + @patch("core.rag.pipeline.queue.redis_client") + def test_pull_tasks_default_count(self, mock_redis, sample_queue): + """Test pulling tasks with default count (1).""" + mock_redis.rpop.side_effect = ["task1", None] + + result = sample_queue.pull_tasks() + + assert result == ["task1"] + assert mock_redis.rpop.call_count == 1 + + @patch("core.rag.pipeline.queue.redis_client") + def test_pull_tasks_custom_count(self, mock_redis, sample_queue): + """Test pulling tasks with custom count.""" + # First test: pull 3 tasks + mock_redis.rpop.side_effect = ["task1", "task2", "task3", None] + + result = sample_queue.pull_tasks(3) + + assert result == ["task1", "task2", "task3"] + assert mock_redis.rpop.call_count == 3 + + # Reset mock for second test + mock_redis.reset_mock() + mock_redis.rpop.side_effect = ["task1", "task2", None] + + result = sample_queue.pull_tasks(3) + + assert result == ["task1", "task2"] + assert mock_redis.rpop.call_count == 3 + + @patch("core.rag.pipeline.queue.redis_client") + def test_pull_tasks_zero_count(self, mock_redis, sample_queue): + """Test pulling tasks with zero count returns empty list.""" + result = sample_queue.pull_tasks(0) + + assert result == [] + mock_redis.rpop.assert_not_called() + + @patch("core.rag.pipeline.queue.redis_client") + def test_pull_tasks_negative_count(self, mock_redis, sample_queue): + """Test pulling tasks with negative count returns empty list.""" + result = sample_queue.pull_tasks(-1) + + assert result == [] + mock_redis.rpop.assert_not_called() + + @patch("core.rag.pipeline.queue.redis_client") + def test_pull_tasks_with_wrapped_objects(self, mock_redis, sample_queue): + """Test pulling tasks that include wrapped objects.""" + # Create a wrapped task + task_data = {"task_id": 123, "data": "test"} + wrapper = TaskWrapper(data=task_data) + wrapped_task = wrapper.serialize() + + mock_redis.rpop.side_effect = [ + "string_task", + wrapped_task.encode("utf-8"), # Simulate bytes from Redis + None, + ] + + result = sample_queue.pull_tasks(2) + + assert len(result) == 2 + assert result[0] == "string_task" + assert result[1] == {"task_id": 123, "data": "test"} + + @patch("core.rag.pipeline.queue.redis_client") + def test_pull_tasks_with_invalid_wrapped_data(self, mock_redis, sample_queue): + """Test pulling tasks with invalid JSON falls back to string.""" + # Invalid JSON string that cannot be deserialized + invalid_json = "invalid json data" + mock_redis.rpop.side_effect = [invalid_json, None] + + result = sample_queue.pull_tasks(1) + + assert result == [invalid_json] + + @patch("core.rag.pipeline.queue.redis_client") + def test_pull_tasks_bytes_decoding(self, mock_redis, sample_queue): + """Test pulling tasks handles bytes from Redis correctly.""" + mock_redis.rpop.side_effect = [ + b"task1", # bytes + "task2", # string + None, + ] + + result = sample_queue.pull_tasks(2) + + assert result == ["task1", "task2"] + + @patch("core.rag.pipeline.queue.redis_client") + def test_complex_object_serialization_roundtrip(self, mock_redis, sample_queue): + """Test complex object serialization and deserialization roundtrip.""" + complex_task = { + "id": uuid4().hex, + "data": {"nested": {"deep": [1, 2, 3], "unicode": "测试中文", "special": "!@#$%^&*()"}}, + "metadata": {"created_at": "2024-01-01T00:00:00Z", "tags": ["tag1", "tag2", "tag3"]}, + } + + # Push the complex task + sample_queue.push_tasks([complex_task]) + + # Verify it was serialized as TaskWrapper JSON + call_args = mock_redis.lpush.call_args + wrapped_task = call_args[0][1] + # Verify it's a valid TaskWrapper JSON (starts with {"data":) + assert wrapped_task.startswith('{"data":') + + # Verify it can be deserialized + wrapper = TaskWrapper.deserialize(wrapped_task) + assert wrapper.data == complex_task + + # Simulate pulling it back + mock_redis.rpop.return_value = wrapped_task + result = sample_queue.pull_tasks(1) + + assert len(result) == 1 + assert result[0] == complex_task diff --git a/api/tests/unit_tests/core/test_trigger_debug_event_selectors.py b/api/tests/unit_tests/core/test_trigger_debug_event_selectors.py new file mode 100644 index 0000000000..2b508ca654 --- /dev/null +++ b/api/tests/unit_tests/core/test_trigger_debug_event_selectors.py @@ -0,0 +1,102 @@ +import hashlib +import json +from datetime import UTC, datetime + +import pytest +import pytz + +from core.trigger.debug import event_selectors +from core.workflow.nodes.trigger_schedule.entities import ScheduleConfig + + +class _DummyRedis: + def __init__(self): + self.store: dict[str, str] = {} + + def get(self, key: str): + return self.store.get(key) + + def setex(self, name: str, time: int, value: str): + self.store[name] = value + + def expire(self, name: str, ttl: int): + # Expiration not required for these tests. + pass + + def delete(self, name: str): + self.store.pop(name, None) + + +@pytest.fixture +def dummy_schedule_config() -> ScheduleConfig: + return ScheduleConfig( + node_id="node-1", + cron_expression="* * * * *", + timezone="Asia/Shanghai", + ) + + +@pytest.fixture(autouse=True) +def patch_schedule_service(monkeypatch: pytest.MonkeyPatch, dummy_schedule_config: ScheduleConfig): + # Ensure poller always receives the deterministic config. + monkeypatch.setattr( + "services.trigger.schedule_service.ScheduleService.to_schedule_config", + staticmethod(lambda *_args, **_kwargs: dummy_schedule_config), + ) + + +def _make_poller( + monkeypatch: pytest.MonkeyPatch, redis_client: _DummyRedis +) -> event_selectors.ScheduleTriggerDebugEventPoller: + monkeypatch.setattr(event_selectors, "redis_client", redis_client) + return event_selectors.ScheduleTriggerDebugEventPoller( + tenant_id="tenant-1", + user_id="user-1", + app_id="app-1", + node_config={"id": "node-1", "data": {"mode": "cron"}}, + node_id="node-1", + ) + + +def test_schedule_poller_handles_aware_next_run(monkeypatch: pytest.MonkeyPatch): + redis_client = _DummyRedis() + poller = _make_poller(monkeypatch, redis_client) + + base_now = datetime(2025, 1, 1, 12, 0, 10) + aware_next_run = datetime(2025, 1, 1, 12, 0, 5, tzinfo=UTC) + + monkeypatch.setattr(event_selectors, "naive_utc_now", lambda: base_now) + monkeypatch.setattr(event_selectors, "calculate_next_run_at", lambda *_: aware_next_run) + + event = poller.poll() + + assert event is not None + assert event.node_id == "node-1" + assert event.workflow_args["inputs"] == {} + + +def test_schedule_runtime_cache_normalizes_timezone( + monkeypatch: pytest.MonkeyPatch, dummy_schedule_config: ScheduleConfig +): + redis_client = _DummyRedis() + poller = _make_poller(monkeypatch, redis_client) + + localized_time = pytz.timezone("Asia/Shanghai").localize(datetime(2025, 1, 1, 20, 0, 0)) + + cron_hash = hashlib.sha256(dummy_schedule_config.cron_expression.encode()).hexdigest() + cache_key = poller.schedule_debug_runtime_key(cron_hash) + + redis_client.store[cache_key] = json.dumps( + { + "cache_key": cache_key, + "timezone": dummy_schedule_config.timezone, + "cron_expression": dummy_schedule_config.cron_expression, + "next_run_at": localized_time.isoformat(), + } + ) + + runtime = poller.get_or_create_schedule_debug_runtime() + + expected = localized_time.astimezone(UTC).replace(tzinfo=None) + assert runtime.next_run_at == expected + assert runtime.next_run_at.tzinfo is None diff --git a/api/tests/unit_tests/core/tools/utils/test_encryption.py b/api/tests/unit_tests/core/tools/utils/test_encryption.py index 6425ab0b8d..94be0bb573 100644 --- a/api/tests/unit_tests/core/tools/utils/test_encryption.py +++ b/api/tests/unit_tests/core/tools/utils/test_encryption.py @@ -4,7 +4,7 @@ from unittest.mock import patch import pytest from core.entities.provider_entities import BasicProviderConfig -from core.tools.utils.encryption import ProviderConfigEncrypter +from core.helper.provider_encryption import ProviderConfigEncrypter # --------------------------- @@ -70,7 +70,7 @@ def test_encrypt_only_secret_is_encrypted_and_non_secret_unchanged(encrypter_obj data_in = {"username": "alice", "password": "plain_pwd"} data_copy = copy.deepcopy(data_in) - with patch("core.tools.utils.encryption.encrypter.encrypt_token", return_value="CIPHERTEXT") as mock_encrypt: + with patch("core.helper.provider_encryption.encrypter.encrypt_token", return_value="CIPHERTEXT") as mock_encrypt: out = encrypter_obj.encrypt(data_in) assert out["username"] == "alice" @@ -81,14 +81,14 @@ def test_encrypt_only_secret_is_encrypted_and_non_secret_unchanged(encrypter_obj def test_encrypt_missing_secret_key_is_ok(encrypter_obj): """If secret field missing in input, no error and no encryption called.""" - with patch("core.tools.utils.encryption.encrypter.encrypt_token") as mock_encrypt: + with patch("core.helper.provider_encryption.encrypter.encrypt_token") as mock_encrypt: out = encrypter_obj.encrypt({"username": "alice"}) assert out["username"] == "alice" mock_encrypt.assert_not_called() # ============================================================ -# ProviderConfigEncrypter.mask_tool_credentials() +# ProviderConfigEncrypter.mask_plugin_credentials() # ============================================================ @@ -107,7 +107,7 @@ def test_mask_tool_credentials_long_secret(encrypter_obj, raw, prefix, suffix): data_in = {"username": "alice", "password": raw} data_copy = copy.deepcopy(data_in) - out = encrypter_obj.mask_tool_credentials(data_in) + out = encrypter_obj.mask_plugin_credentials(data_in) masked = out["password"] assert masked.startswith(prefix) @@ -122,7 +122,7 @@ def test_mask_tool_credentials_short_secret(encrypter_obj, raw): """ For length <= 6: fully mask with '*' of same length. """ - out = encrypter_obj.mask_tool_credentials({"password": raw}) + out = encrypter_obj.mask_plugin_credentials({"password": raw}) assert out["password"] == ("*" * len(raw)) @@ -131,7 +131,7 @@ def test_mask_tool_credentials_missing_key_noop(encrypter_obj): data_in = {"username": "alice"} data_copy = copy.deepcopy(data_in) - out = encrypter_obj.mask_tool_credentials(data_in) + out = encrypter_obj.mask_plugin_credentials(data_in) assert out["username"] == "alice" assert data_in == data_copy @@ -151,7 +151,7 @@ def test_decrypt_normal_flow(encrypter_obj): data_in = {"username": "alice", "password": "ENC"} data_copy = copy.deepcopy(data_in) - with patch("core.tools.utils.encryption.encrypter.decrypt_token", return_value="PLAIN") as mock_decrypt: + with patch("core.helper.provider_encryption.encrypter.decrypt_token", return_value="PLAIN") as mock_decrypt: out = encrypter_obj.decrypt(data_in) assert out["username"] == "alice" @@ -163,7 +163,7 @@ def test_decrypt_normal_flow(encrypter_obj): @pytest.mark.parametrize("empty_val", ["", None]) def test_decrypt_skip_empty_values(encrypter_obj, empty_val): """Skip decrypt if value is empty or None, keep original.""" - with patch("core.tools.utils.encryption.encrypter.decrypt_token") as mock_decrypt: + with patch("core.helper.provider_encryption.encrypter.decrypt_token") as mock_decrypt: out = encrypter_obj.decrypt({"password": empty_val}) mock_decrypt.assert_not_called() @@ -175,7 +175,7 @@ def test_decrypt_swallow_exception_and_keep_original(encrypter_obj): If decrypt_token raises, exception should be swallowed, and original value preserved. """ - with patch("core.tools.utils.encryption.encrypter.decrypt_token", side_effect=Exception("boom")): + with patch("core.helper.provider_encryption.encrypter.decrypt_token", side_effect=Exception("boom")): out = encrypter_obj.decrypt({"password": "ENC_ERR"}) assert out["password"] == "ENC_ERR" diff --git a/api/tests/unit_tests/core/variables/test_segment_type.py b/api/tests/unit_tests/core/variables/test_segment_type.py index a197b617f3..3bfc5a957f 100644 --- a/api/tests/unit_tests/core/variables/test_segment_type.py +++ b/api/tests/unit_tests/core/variables/test_segment_type.py @@ -1,3 +1,5 @@ +import pytest + from core.variables.types import ArrayValidation, SegmentType @@ -83,3 +85,81 @@ class TestSegmentTypeIsValidArrayValidation: value = [1, 2, 3] # validation is None, skip assert SegmentType.ARRAY_STRING.is_valid(value, array_validation=ArrayValidation.NONE) + + +class TestSegmentTypeGetZeroValue: + """ + Test class for SegmentType.get_zero_value static method. + + Provides comprehensive coverage of all supported SegmentType values to ensure + correct zero value generation for each type. + """ + + def test_array_types_return_empty_list(self): + """Test that all array types return empty list segments.""" + array_types = [ + SegmentType.ARRAY_ANY, + SegmentType.ARRAY_STRING, + SegmentType.ARRAY_NUMBER, + SegmentType.ARRAY_OBJECT, + SegmentType.ARRAY_BOOLEAN, + ] + + for seg_type in array_types: + result = SegmentType.get_zero_value(seg_type) + assert result.value == [] + assert result.value_type == seg_type + + def test_object_returns_empty_dict(self): + """Test that OBJECT type returns empty dictionary segment.""" + result = SegmentType.get_zero_value(SegmentType.OBJECT) + assert result.value == {} + assert result.value_type == SegmentType.OBJECT + + def test_string_returns_empty_string(self): + """Test that STRING type returns empty string segment.""" + result = SegmentType.get_zero_value(SegmentType.STRING) + assert result.value == "" + assert result.value_type == SegmentType.STRING + + def test_integer_returns_zero(self): + """Test that INTEGER type returns zero segment.""" + result = SegmentType.get_zero_value(SegmentType.INTEGER) + assert result.value == 0 + assert result.value_type == SegmentType.INTEGER + + def test_float_returns_zero_point_zero(self): + """Test that FLOAT type returns 0.0 segment.""" + result = SegmentType.get_zero_value(SegmentType.FLOAT) + assert result.value == 0.0 + assert result.value_type == SegmentType.FLOAT + + def test_number_returns_zero(self): + """Test that NUMBER type returns zero segment.""" + result = SegmentType.get_zero_value(SegmentType.NUMBER) + assert result.value == 0 + # NUMBER type with integer value returns INTEGER segment type + # (NUMBER is a union type that can be INTEGER or FLOAT) + assert result.value_type == SegmentType.INTEGER + # Verify that exposed_type returns NUMBER for frontend compatibility + assert result.value_type.exposed_type() == SegmentType.NUMBER + + def test_boolean_returns_false(self): + """Test that BOOLEAN type returns False segment.""" + result = SegmentType.get_zero_value(SegmentType.BOOLEAN) + assert result.value is False + assert result.value_type == SegmentType.BOOLEAN + + def test_unsupported_types_raise_value_error(self): + """Test that unsupported types raise ValueError.""" + unsupported_types = [ + SegmentType.SECRET, + SegmentType.FILE, + SegmentType.NONE, + SegmentType.GROUP, + SegmentType.ARRAY_FILE, + ] + + for seg_type in unsupported_types: + with pytest.raises(ValueError, match="unsupported variable type"): + SegmentType.get_zero_value(seg_type) diff --git a/api/tests/unit_tests/core/workflow/entities/test_graph_runtime_state.py b/api/tests/unit_tests/core/workflow/entities/test_graph_runtime_state.py index 5ecaeb60ac..deff06fc5d 100644 --- a/api/tests/unit_tests/core/workflow/entities/test_graph_runtime_state.py +++ b/api/tests/unit_tests/core/workflow/entities/test_graph_runtime_state.py @@ -8,6 +8,18 @@ from core.model_runtime.entities.llm_entities import LLMUsage from core.workflow.runtime import GraphRuntimeState, ReadOnlyGraphRuntimeStateWrapper, VariablePool +class StubCoordinator: + def __init__(self) -> None: + self.state = "initial" + + def dumps(self) -> str: + return json.dumps({"state": self.state}) + + def loads(self, data: str) -> None: + payload = json.loads(data) + self.state = payload["state"] + + class TestGraphRuntimeState: def test_property_getters_and_setters(self): # FIXME(-LAN-): Mock VariablePool if needed @@ -191,17 +203,6 @@ class TestGraphRuntimeState: graph_execution.exceptions_count = 4 graph_execution.started = True - class StubCoordinator: - def __init__(self) -> None: - self.state = "initial" - - def dumps(self) -> str: - return json.dumps({"state": self.state}) - - def loads(self, data: str) -> None: - payload = json.loads(data) - self.state = payload["state"] - mock_graph = MagicMock() stub = StubCoordinator() with patch.object(GraphRuntimeState, "_build_response_coordinator", return_value=stub): @@ -211,8 +212,7 @@ class TestGraphRuntimeState: snapshot = state.dumps() - restored = GraphRuntimeState(variable_pool=VariablePool(), start_at=0.0) - restored.loads(snapshot) + restored = GraphRuntimeState.from_snapshot(snapshot) assert restored.total_tokens == 10 assert restored.node_run_steps == 3 @@ -235,3 +235,47 @@ class TestGraphRuntimeState: restored.attach_graph(mock_graph) assert new_stub.state == "configured" + + def test_loads_rehydrates_existing_instance(self): + variable_pool = VariablePool() + variable_pool.add(("node", "key"), "value") + + state = GraphRuntimeState(variable_pool=variable_pool, start_at=time()) + state.total_tokens = 7 + state.node_run_steps = 2 + state.set_output("foo", "bar") + state.ready_queue.put("node-1") + + execution = state.graph_execution + execution.workflow_id = "wf-456" + execution.started = True + + mock_graph = MagicMock() + original_stub = StubCoordinator() + with patch.object(GraphRuntimeState, "_build_response_coordinator", return_value=original_stub): + state.attach_graph(mock_graph) + + original_stub.state = "configured" + snapshot = state.dumps() + + new_stub = StubCoordinator() + with patch.object(GraphRuntimeState, "_build_response_coordinator", return_value=new_stub): + restored = GraphRuntimeState(variable_pool=VariablePool(), start_at=0.0) + restored.attach_graph(mock_graph) + restored.loads(snapshot) + + assert restored.total_tokens == 7 + assert restored.node_run_steps == 2 + assert restored.get_output("foo") == "bar" + assert restored.ready_queue.qsize() == 1 + assert restored.ready_queue.get(timeout=0.01) == "node-1" + + restored_segment = restored.variable_pool.get(("node", "key")) + assert restored_segment is not None + assert restored_segment.value == "value" + + restored_execution = restored.graph_execution + assert restored_execution.workflow_id == "wf-456" + assert restored_execution.started is True + + assert new_stub.state == "configured" diff --git a/api/tests/unit_tests/core/workflow/entities/test_private_workflow_pause.py b/api/tests/unit_tests/core/workflow/entities/test_private_workflow_pause.py new file mode 100644 index 0000000000..ccb2dff85a --- /dev/null +++ b/api/tests/unit_tests/core/workflow/entities/test_private_workflow_pause.py @@ -0,0 +1,171 @@ +"""Tests for _PrivateWorkflowPauseEntity implementation.""" + +from datetime import datetime +from unittest.mock import MagicMock, patch + +from models.workflow import WorkflowPause as WorkflowPauseModel +from repositories.sqlalchemy_api_workflow_run_repository import _PrivateWorkflowPauseEntity + + +class TestPrivateWorkflowPauseEntity: + """Test _PrivateWorkflowPauseEntity implementation.""" + + def test_entity_initialization(self): + """Test entity initialization with required parameters.""" + # Create mock models + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.id = "pause-123" + mock_pause_model.workflow_run_id = "execution-456" + mock_pause_model.resumed_at = None + + # Create entity + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + # Verify initialization + assert entity._pause_model is mock_pause_model + assert entity._cached_state is None + + def test_from_models_classmethod(self): + """Test from_models class method.""" + # Create mock models + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.id = "pause-123" + mock_pause_model.workflow_run_id = "execution-456" + + # Create entity using from_models + entity = _PrivateWorkflowPauseEntity.from_models( + workflow_pause_model=mock_pause_model, + ) + + # Verify entity creation + assert isinstance(entity, _PrivateWorkflowPauseEntity) + assert entity._pause_model is mock_pause_model + + def test_id_property(self): + """Test id property returns pause model ID.""" + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.id = "pause-123" + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + assert entity.id == "pause-123" + + def test_workflow_execution_id_property(self): + """Test workflow_execution_id property returns workflow run ID.""" + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.workflow_run_id = "execution-456" + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + assert entity.workflow_execution_id == "execution-456" + + def test_resumed_at_property(self): + """Test resumed_at property returns pause model resumed_at.""" + resumed_at = datetime(2023, 12, 25, 15, 30, 45) + + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.resumed_at = resumed_at + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + assert entity.resumed_at == resumed_at + + def test_resumed_at_property_none(self): + """Test resumed_at property returns None when not set.""" + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.resumed_at = None + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + assert entity.resumed_at is None + + @patch("repositories.sqlalchemy_api_workflow_run_repository.storage") + def test_get_state_first_call(self, mock_storage): + """Test get_state loads from storage on first call.""" + state_data = b'{"test": "data", "step": 5}' + mock_storage.load.return_value = state_data + + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.state_object_key = "test-state-key" + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + # First call should load from storage + result = entity.get_state() + + assert result == state_data + mock_storage.load.assert_called_once_with("test-state-key") + assert entity._cached_state == state_data + + @patch("repositories.sqlalchemy_api_workflow_run_repository.storage") + def test_get_state_cached_call(self, mock_storage): + """Test get_state returns cached data on subsequent calls.""" + state_data = b'{"test": "data", "step": 5}' + mock_storage.load.return_value = state_data + + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + mock_pause_model.state_object_key = "test-state-key" + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + # First call + result1 = entity.get_state() + # Second call should use cache + result2 = entity.get_state() + + assert result1 == state_data + assert result2 == state_data + # Storage should only be called once + mock_storage.load.assert_called_once_with("test-state-key") + + @patch("repositories.sqlalchemy_api_workflow_run_repository.storage") + def test_get_state_with_pre_cached_data(self, mock_storage): + """Test get_state returns pre-cached data.""" + state_data = b'{"test": "data", "step": 5}' + + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + # Pre-cache data + entity._cached_state = state_data + + # Should return cached data without calling storage + result = entity.get_state() + + assert result == state_data + mock_storage.load.assert_not_called() + + def test_entity_with_binary_state_data(self): + """Test entity with binary state data.""" + # Test with binary data that's not valid JSON + binary_data = b"\x00\x01\x02\x03\x04\x05\xff\xfe" + + with patch("repositories.sqlalchemy_api_workflow_run_repository.storage") as mock_storage: + mock_storage.load.return_value = binary_data + + mock_pause_model = MagicMock(spec=WorkflowPauseModel) + + entity = _PrivateWorkflowPauseEntity( + pause_model=mock_pause_model, + ) + + result = entity.get_state() + + assert result == binary_data diff --git a/api/tests/unit_tests/core/workflow/entities/test_variable_pool.py b/api/tests/unit_tests/core/workflow/entities/test_variable_pool.py index f9de456b19..18f6753b05 100644 --- a/api/tests/unit_tests/core/workflow/entities/test_variable_pool.py +++ b/api/tests/unit_tests/core/workflow/entities/test_variable_pool.py @@ -111,3 +111,26 @@ class TestVariablePoolGetAndNestedAttribute: assert segment_false is not None assert isinstance(segment_false, BooleanSegment) assert segment_false.value is False + + +class TestVariablePoolGetNotModifyVariableDictionary: + _NODE_ID = "start" + _VAR_NAME = "name" + + def test_convert_to_template_should_not_introduce_extra_keys(self): + pool = VariablePool.empty() + pool.add([self._NODE_ID, self._VAR_NAME], 0) + pool.convert_template("The start.name is {{#start.name#}}") + assert "The start" not in pool.variable_dictionary + + def test_get_should_not_modify_variable_dictionary(self): + pool = VariablePool.empty() + pool.get([self._NODE_ID, self._VAR_NAME]) + assert len(pool.variable_dictionary) == 1 # only contains `sys` node id + assert "start" not in pool.variable_dictionary + + pool = VariablePool.empty() + pool.add([self._NODE_ID, self._VAR_NAME], "Joe") + pool.get([self._NODE_ID, "count"]) + start_subdict = pool.variable_dictionary[self._NODE_ID] + assert "count" not in start_subdict diff --git a/api/tests/unit_tests/core/workflow/graph/test_graph_validation.py b/api/tests/unit_tests/core/workflow/graph/test_graph_validation.py index b55d4998c4..c55c40c5b4 100644 --- a/api/tests/unit_tests/core/workflow/graph/test_graph_validation.py +++ b/api/tests/unit_tests/core/workflow/graph/test_graph_validation.py @@ -64,6 +64,15 @@ class _TestNode(Node): ) self.data = dict(data) + node_type_value = data.get("type") + if isinstance(node_type_value, NodeType): + self.node_type = node_type_value + elif isinstance(node_type_value, str): + try: + self.node_type = NodeType(node_type_value) + except ValueError: + pass + def _run(self): raise NotImplementedError @@ -179,3 +188,22 @@ def test_graph_promotes_fail_branch_nodes_to_branch_execution_type( graph = Graph.init(graph_config=graph_config, node_factory=node_factory) assert graph.nodes["branch"].execution_type == NodeExecutionType.BRANCH + + +def test_graph_validation_blocks_start_and_trigger_coexistence( + graph_init_dependencies: tuple[_SimpleNodeFactory, dict[str, object]], +) -> None: + node_factory, graph_config = graph_init_dependencies + graph_config["nodes"] = [ + {"id": "start", "data": {"type": NodeType.START, "title": "Start", "execution_type": NodeExecutionType.ROOT}}, + { + "id": "trigger", + "data": {"type": NodeType.TRIGGER_WEBHOOK, "title": "Webhook", "execution_type": NodeExecutionType.ROOT}, + }, + ] + graph_config["edges"] = [] + + with pytest.raises(GraphValidationError) as exc_info: + Graph.init(graph_config=graph_config, node_factory=node_factory) + + assert any(issue.code == "TRIGGER_START_NODE_CONFLICT" for issue in exc_info.value.issues) diff --git a/api/tests/unit_tests/core/workflow/graph_engine/test_command_system.py b/api/tests/unit_tests/core/workflow/graph_engine/test_command_system.py index d451e7e608..b29baf5a9f 100644 --- a/api/tests/unit_tests/core/workflow/graph_engine/test_command_system.py +++ b/api/tests/unit_tests/core/workflow/graph_engine/test_command_system.py @@ -3,6 +3,7 @@ import time from unittest.mock import MagicMock +from core.workflow.entities.pause_reason import SchedulingPause from core.workflow.graph import Graph from core.workflow.graph_engine import GraphEngine from core.workflow.graph_engine.command_channels import InMemoryChannel @@ -149,8 +150,8 @@ def test_pause_command(): assert any(isinstance(e, GraphRunStartedEvent) for e in events) pause_events = [e for e in events if isinstance(e, GraphRunPausedEvent)] assert len(pause_events) == 1 - assert pause_events[0].reason == "User requested pause" + assert pause_events[0].reason == SchedulingPause(message="User requested pause") graph_execution = engine.graph_runtime_state.graph_execution assert graph_execution.is_paused - assert graph_execution.pause_reason == "User requested pause" + assert graph_execution.pause_reason == SchedulingPause(message="User requested pause") diff --git a/api/tests/unit_tests/core/workflow/graph_engine/test_database_utils.py b/api/tests/unit_tests/core/workflow/graph_engine/test_database_utils.py new file mode 100644 index 0000000000..ae7dd48bb1 --- /dev/null +++ b/api/tests/unit_tests/core/workflow/graph_engine/test_database_utils.py @@ -0,0 +1,46 @@ +""" +Utilities for detecting if database service is available for workflow tests. +""" + +import psycopg2 +import pytest + +from configs import dify_config + + +def is_database_available() -> bool: + """ + Check if the database service is available by attempting to connect to it. + + Returns: + True if database is available, False otherwise. + """ + try: + # Try to establish a database connection using a context manager + with psycopg2.connect( + host=dify_config.DB_HOST, + port=dify_config.DB_PORT, + database=dify_config.DB_DATABASE, + user=dify_config.DB_USERNAME, + password=dify_config.DB_PASSWORD, + connect_timeout=2, # 2 second timeout + ) as conn: + pass # Connection established and will be closed automatically + return True + except (psycopg2.OperationalError, psycopg2.Error): + return False + + +def skip_if_database_unavailable(): + """ + Pytest skip decorator that skips tests when database service is unavailable. + + Usage: + @skip_if_database_unavailable() + def test_my_workflow(): + ... + """ + return pytest.mark.skipif( + not is_database_available(), + reason="Database service is not available (connection refused or authentication failed)", + ) diff --git a/api/tests/unit_tests/core/workflow/graph_engine/test_iteration_flatten_output.py b/api/tests/unit_tests/core/workflow/graph_engine/test_iteration_flatten_output.py new file mode 100644 index 0000000000..98f344babf --- /dev/null +++ b/api/tests/unit_tests/core/workflow/graph_engine/test_iteration_flatten_output.py @@ -0,0 +1,100 @@ +""" +Test cases for the Iteration node's flatten_output functionality. + +This module tests the iteration node's ability to: +1. Flatten array outputs when flatten_output=True (default) +2. Preserve nested array structure when flatten_output=False +""" + +from .test_database_utils import skip_if_database_unavailable +from .test_table_runner import TableTestRunner, WorkflowTestCase + + +@skip_if_database_unavailable() +def test_iteration_with_flatten_output_enabled(): + """ + Test iteration node with flatten_output=True (default behavior). + + The fixture implements an iteration that: + 1. Iterates over [1, 2, 3] + 2. For each item, outputs [item, item*2] + 3. With flatten_output=True, should output [1, 2, 2, 4, 3, 6] + """ + runner = TableTestRunner() + + test_case = WorkflowTestCase( + fixture_path="iteration_flatten_output_enabled_workflow", + inputs={}, + expected_outputs={"output": [1, 2, 2, 4, 3, 6]}, + description="Iteration with flatten_output=True flattens nested arrays", + use_auto_mock=False, # Run code nodes directly + ) + + result = runner.run_test_case(test_case) + + assert result.success, f"Test failed: {result.error}" + assert result.actual_outputs is not None, "Should have outputs" + assert result.actual_outputs == {"output": [1, 2, 2, 4, 3, 6]}, ( + f"Expected flattened output [1, 2, 2, 4, 3, 6], got {result.actual_outputs}" + ) + + +@skip_if_database_unavailable() +def test_iteration_with_flatten_output_disabled(): + """ + Test iteration node with flatten_output=False. + + The fixture implements an iteration that: + 1. Iterates over [1, 2, 3] + 2. For each item, outputs [item, item*2] + 3. With flatten_output=False, should output [[1, 2], [2, 4], [3, 6]] + """ + runner = TableTestRunner() + + test_case = WorkflowTestCase( + fixture_path="iteration_flatten_output_disabled_workflow", + inputs={}, + expected_outputs={"output": [[1, 2], [2, 4], [3, 6]]}, + description="Iteration with flatten_output=False preserves nested structure", + use_auto_mock=False, # Run code nodes directly + ) + + result = runner.run_test_case(test_case) + + assert result.success, f"Test failed: {result.error}" + assert result.actual_outputs is not None, "Should have outputs" + assert result.actual_outputs == {"output": [[1, 2], [2, 4], [3, 6]]}, ( + f"Expected nested output [[1, 2], [2, 4], [3, 6]], got {result.actual_outputs}" + ) + + +@skip_if_database_unavailable() +def test_iteration_flatten_output_comparison(): + """ + Run both flatten_output configurations in parallel to verify the difference. + """ + runner = TableTestRunner() + + test_cases = [ + WorkflowTestCase( + fixture_path="iteration_flatten_output_enabled_workflow", + inputs={}, + expected_outputs={"output": [1, 2, 2, 4, 3, 6]}, + description="flatten_output=True: Flattened output", + use_auto_mock=False, # Run code nodes directly + ), + WorkflowTestCase( + fixture_path="iteration_flatten_output_disabled_workflow", + inputs={}, + expected_outputs={"output": [[1, 2], [2, 4], [3, 6]]}, + description="flatten_output=False: Nested output", + use_auto_mock=False, # Run code nodes directly + ), + ] + + suite_result = runner.run_table_tests(test_cases, parallel=True) + + # Assert all tests passed + assert suite_result.passed_tests == 2, f"Expected 2 passed tests, got {suite_result.passed_tests}" + assert suite_result.failed_tests == 0, f"Expected 0 failed tests, got {suite_result.failed_tests}" + assert suite_result.success_rate == 100.0, f"Expected 100% success rate, got {suite_result.success_rate}" diff --git a/api/tests/unit_tests/core/workflow/nodes/webhook/__init__.py b/api/tests/unit_tests/core/workflow/nodes/webhook/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/api/tests/unit_tests/core/workflow/nodes/webhook/test_entities.py b/api/tests/unit_tests/core/workflow/nodes/webhook/test_entities.py new file mode 100644 index 0000000000..4fa9a01b61 --- /dev/null +++ b/api/tests/unit_tests/core/workflow/nodes/webhook/test_entities.py @@ -0,0 +1,308 @@ +import pytest +from pydantic import ValidationError + +from core.workflow.nodes.trigger_webhook.entities import ( + ContentType, + Method, + WebhookBodyParameter, + WebhookData, + WebhookParameter, +) + + +def test_method_enum(): + """Test Method enum values.""" + assert Method.GET == "get" + assert Method.POST == "post" + assert Method.HEAD == "head" + assert Method.PATCH == "patch" + assert Method.PUT == "put" + assert Method.DELETE == "delete" + + # Test all enum values are strings + for method in Method: + assert isinstance(method.value, str) + + +def test_content_type_enum(): + """Test ContentType enum values.""" + assert ContentType.JSON == "application/json" + assert ContentType.FORM_DATA == "multipart/form-data" + assert ContentType.FORM_URLENCODED == "application/x-www-form-urlencoded" + assert ContentType.TEXT == "text/plain" + assert ContentType.BINARY == "application/octet-stream" + + # Test all enum values are strings + for content_type in ContentType: + assert isinstance(content_type.value, str) + + +def test_webhook_parameter_creation(): + """Test WebhookParameter model creation and validation.""" + # Test with all fields + param = WebhookParameter(name="api_key", required=True) + assert param.name == "api_key" + assert param.required is True + + # Test with defaults + param_default = WebhookParameter(name="optional_param") + assert param_default.name == "optional_param" + assert param_default.required is False + + # Test validation - name is required + with pytest.raises(ValidationError): + WebhookParameter() + + +def test_webhook_body_parameter_creation(): + """Test WebhookBodyParameter model creation and validation.""" + # Test with all fields + body_param = WebhookBodyParameter( + name="user_data", + type="object", + required=True, + ) + assert body_param.name == "user_data" + assert body_param.type == "object" + assert body_param.required is True + + # Test with defaults + body_param_default = WebhookBodyParameter(name="message") + assert body_param_default.name == "message" + assert body_param_default.type == "string" # Default type + assert body_param_default.required is False + + # Test validation - name is required + with pytest.raises(ValidationError): + WebhookBodyParameter() + + +def test_webhook_body_parameter_types(): + """Test WebhookBodyParameter type validation.""" + valid_types = [ + "string", + "number", + "boolean", + "object", + "array[string]", + "array[number]", + "array[boolean]", + "array[object]", + "file", + ] + + for param_type in valid_types: + param = WebhookBodyParameter(name="test", type=param_type) + assert param.type == param_type + + # Test invalid type + with pytest.raises(ValidationError): + WebhookBodyParameter(name="test", type="invalid_type") + + +def test_webhook_data_creation_minimal(): + """Test WebhookData creation with minimal required fields.""" + data = WebhookData(title="Test Webhook") + + assert data.title == "Test Webhook" + assert data.method == Method.GET # Default + assert data.content_type == ContentType.JSON # Default + assert data.headers == [] # Default + assert data.params == [] # Default + assert data.body == [] # Default + assert data.status_code == 200 # Default + assert data.response_body == "" # Default + assert data.webhook_id is None # Default + assert data.timeout == 30 # Default + + +def test_webhook_data_creation_full(): + """Test WebhookData creation with all fields.""" + headers = [ + WebhookParameter(name="Authorization", required=True), + WebhookParameter(name="Content-Type", required=False), + ] + params = [ + WebhookParameter(name="version", required=True), + WebhookParameter(name="format", required=False), + ] + body = [ + WebhookBodyParameter(name="message", type="string", required=True), + WebhookBodyParameter(name="count", type="number", required=False), + WebhookBodyParameter(name="upload", type="file", required=True), + ] + + # Use the alias for content_type to test it properly + data = WebhookData( + title="Full Webhook Test", + desc="A comprehensive webhook test", + method=Method.POST, + content_type=ContentType.FORM_DATA, + headers=headers, + params=params, + body=body, + status_code=201, + response_body='{"success": true}', + webhook_id="webhook_123", + timeout=60, + ) + + assert data.title == "Full Webhook Test" + assert data.desc == "A comprehensive webhook test" + assert data.method == Method.POST + assert data.content_type == ContentType.FORM_DATA + assert len(data.headers) == 2 + assert len(data.params) == 2 + assert len(data.body) == 3 + assert data.status_code == 201 + assert data.response_body == '{"success": true}' + assert data.webhook_id == "webhook_123" + assert data.timeout == 60 + + +def test_webhook_data_content_type_alias(): + """Test WebhookData content_type accepts both strings and enum values.""" + data1 = WebhookData(title="Test", content_type="application/json") + assert data1.content_type == ContentType.JSON + + data2 = WebhookData(title="Test", content_type=ContentType.FORM_DATA) + assert data2.content_type == ContentType.FORM_DATA + + +def test_webhook_data_model_dump(): + """Test WebhookData model serialization.""" + data = WebhookData( + title="Test Webhook", + method=Method.POST, + content_type=ContentType.JSON, + headers=[WebhookParameter(name="Authorization", required=True)], + params=[WebhookParameter(name="version", required=False)], + body=[WebhookBodyParameter(name="message", type="string", required=True)], + status_code=200, + response_body="OK", + timeout=30, + ) + + dumped = data.model_dump() + + assert dumped["title"] == "Test Webhook" + assert dumped["method"] == "post" + assert dumped["content_type"] == "application/json" + assert len(dumped["headers"]) == 1 + assert dumped["headers"][0]["name"] == "Authorization" + assert dumped["headers"][0]["required"] is True + assert len(dumped["params"]) == 1 + assert len(dumped["body"]) == 1 + assert dumped["body"][0]["type"] == "string" + + +def test_webhook_data_model_dump_with_alias(): + """Test WebhookData model serialization includes alias.""" + data = WebhookData( + title="Test Webhook", + content_type=ContentType.FORM_DATA, + ) + + dumped = data.model_dump(by_alias=True) + assert "content_type" in dumped + assert dumped["content_type"] == "multipart/form-data" + + +def test_webhook_data_validation_errors(): + """Test WebhookData validation errors.""" + # Title is required (inherited from BaseNodeData) + with pytest.raises(ValidationError): + WebhookData() + + # Invalid method + with pytest.raises(ValidationError): + WebhookData(title="Test", method="invalid_method") + + # Invalid content_type + with pytest.raises(ValidationError): + WebhookData(title="Test", content_type="invalid/type") + + # Invalid status_code (should be int) - use non-numeric string + with pytest.raises(ValidationError): + WebhookData(title="Test", status_code="invalid") + + # Invalid timeout (should be int) - use non-numeric string + with pytest.raises(ValidationError): + WebhookData(title="Test", timeout="invalid") + + # Valid cases that should NOT raise errors + # These should work fine (pydantic converts string numbers to int) + valid_data = WebhookData(title="Test", status_code="200", timeout="30") + assert valid_data.status_code == 200 + assert valid_data.timeout == 30 + + +def test_webhook_data_sequence_fields(): + """Test WebhookData sequence field behavior.""" + # Test empty sequences + data = WebhookData(title="Test") + assert data.headers == [] + assert data.params == [] + assert data.body == [] + + # Test immutable sequences + headers = [WebhookParameter(name="test")] + data = WebhookData(title="Test", headers=headers) + + # Original list shouldn't affect the model + headers.append(WebhookParameter(name="test2")) + assert len(data.headers) == 1 # Should still be 1 + + +def test_webhook_data_sync_mode(): + """Test WebhookData SyncMode nested enum.""" + # Test that SyncMode enum exists and has expected value + assert hasattr(WebhookData, "SyncMode") + assert WebhookData.SyncMode.SYNC == "async" # Note: confusingly named but correct + + +def test_webhook_parameter_edge_cases(): + """Test WebhookParameter edge cases.""" + # Test with special characters in name + param = WebhookParameter(name="X-Custom-Header-123", required=True) + assert param.name == "X-Custom-Header-123" + + # Test with empty string name (should be valid if pydantic allows it) + param_empty = WebhookParameter(name="", required=False) + assert param_empty.name == "" + + +def test_webhook_body_parameter_edge_cases(): + """Test WebhookBodyParameter edge cases.""" + # Test file type parameter + file_param = WebhookBodyParameter(name="upload", type="file", required=True) + assert file_param.type == "file" + assert file_param.required is True + + # Test all valid types + for param_type in [ + "string", + "number", + "boolean", + "object", + "array[string]", + "array[number]", + "array[boolean]", + "array[object]", + "file", + ]: + param = WebhookBodyParameter(name=f"test_{param_type}", type=param_type) + assert param.type == param_type + + +def test_webhook_data_inheritance(): + """Test WebhookData inherits from BaseNodeData correctly.""" + from core.workflow.nodes.base import BaseNodeData + + # Test that WebhookData is a subclass of BaseNodeData + assert issubclass(WebhookData, BaseNodeData) + + # Test that instances have BaseNodeData properties + data = WebhookData(title="Test") + assert hasattr(data, "title") + assert hasattr(data, "desc") # Inherited from BaseNodeData diff --git a/api/tests/unit_tests/core/workflow/nodes/webhook/test_exceptions.py b/api/tests/unit_tests/core/workflow/nodes/webhook/test_exceptions.py new file mode 100644 index 0000000000..374d5183c8 --- /dev/null +++ b/api/tests/unit_tests/core/workflow/nodes/webhook/test_exceptions.py @@ -0,0 +1,195 @@ +import pytest + +from core.workflow.nodes.base.exc import BaseNodeError +from core.workflow.nodes.trigger_webhook.exc import ( + WebhookConfigError, + WebhookNodeError, + WebhookNotFoundError, + WebhookTimeoutError, +) + + +def test_webhook_node_error_inheritance(): + """Test WebhookNodeError inherits from BaseNodeError.""" + assert issubclass(WebhookNodeError, BaseNodeError) + + # Test instantiation + error = WebhookNodeError("Test error message") + assert str(error) == "Test error message" + assert isinstance(error, BaseNodeError) + + +def test_webhook_timeout_error(): + """Test WebhookTimeoutError functionality.""" + # Test inheritance + assert issubclass(WebhookTimeoutError, WebhookNodeError) + assert issubclass(WebhookTimeoutError, BaseNodeError) + + # Test instantiation with message + error = WebhookTimeoutError("Webhook request timed out") + assert str(error) == "Webhook request timed out" + + # Test instantiation without message + error_no_msg = WebhookTimeoutError() + assert isinstance(error_no_msg, WebhookTimeoutError) + + +def test_webhook_not_found_error(): + """Test WebhookNotFoundError functionality.""" + # Test inheritance + assert issubclass(WebhookNotFoundError, WebhookNodeError) + assert issubclass(WebhookNotFoundError, BaseNodeError) + + # Test instantiation with message + error = WebhookNotFoundError("Webhook trigger not found") + assert str(error) == "Webhook trigger not found" + + # Test instantiation without message + error_no_msg = WebhookNotFoundError() + assert isinstance(error_no_msg, WebhookNotFoundError) + + +def test_webhook_config_error(): + """Test WebhookConfigError functionality.""" + # Test inheritance + assert issubclass(WebhookConfigError, WebhookNodeError) + assert issubclass(WebhookConfigError, BaseNodeError) + + # Test instantiation with message + error = WebhookConfigError("Invalid webhook configuration") + assert str(error) == "Invalid webhook configuration" + + # Test instantiation without message + error_no_msg = WebhookConfigError() + assert isinstance(error_no_msg, WebhookConfigError) + + +def test_webhook_error_hierarchy(): + """Test the complete webhook error hierarchy.""" + # All webhook errors should inherit from WebhookNodeError + webhook_errors = [ + WebhookTimeoutError, + WebhookNotFoundError, + WebhookConfigError, + ] + + for error_class in webhook_errors: + assert issubclass(error_class, WebhookNodeError) + assert issubclass(error_class, BaseNodeError) + + +def test_webhook_error_instantiation_with_args(): + """Test webhook error instantiation with various arguments.""" + # Test with single string argument + error1 = WebhookNodeError("Simple error message") + assert str(error1) == "Simple error message" + + # Test with multiple arguments + error2 = WebhookTimeoutError("Timeout after", 30, "seconds") + # Note: The exact string representation depends on Exception.__str__ implementation + assert "Timeout after" in str(error2) + + # Test with keyword arguments (if supported by base Exception) + error3 = WebhookConfigError("Config error in field: timeout") + assert "Config error in field: timeout" in str(error3) + + +def test_webhook_error_as_exceptions(): + """Test that webhook errors can be raised and caught properly.""" + # Test raising and catching WebhookNodeError + with pytest.raises(WebhookNodeError) as exc_info: + raise WebhookNodeError("Base webhook error") + assert str(exc_info.value) == "Base webhook error" + + # Test raising and catching specific errors + with pytest.raises(WebhookTimeoutError) as exc_info: + raise WebhookTimeoutError("Request timeout") + assert str(exc_info.value) == "Request timeout" + + with pytest.raises(WebhookNotFoundError) as exc_info: + raise WebhookNotFoundError("Webhook not found") + assert str(exc_info.value) == "Webhook not found" + + with pytest.raises(WebhookConfigError) as exc_info: + raise WebhookConfigError("Invalid config") + assert str(exc_info.value) == "Invalid config" + + +def test_webhook_error_catching_hierarchy(): + """Test that webhook errors can be caught by their parent classes.""" + # WebhookTimeoutError should be catchable as WebhookNodeError + with pytest.raises(WebhookNodeError): + raise WebhookTimeoutError("Timeout error") + + # WebhookNotFoundError should be catchable as WebhookNodeError + with pytest.raises(WebhookNodeError): + raise WebhookNotFoundError("Not found error") + + # WebhookConfigError should be catchable as WebhookNodeError + with pytest.raises(WebhookNodeError): + raise WebhookConfigError("Config error") + + # All webhook errors should be catchable as BaseNodeError + with pytest.raises(BaseNodeError): + raise WebhookTimeoutError("Timeout as base error") + + with pytest.raises(BaseNodeError): + raise WebhookNotFoundError("Not found as base error") + + with pytest.raises(BaseNodeError): + raise WebhookConfigError("Config as base error") + + +def test_webhook_error_attributes(): + """Test webhook error class attributes.""" + # Test that all error classes have proper __name__ + assert WebhookNodeError.__name__ == "WebhookNodeError" + assert WebhookTimeoutError.__name__ == "WebhookTimeoutError" + assert WebhookNotFoundError.__name__ == "WebhookNotFoundError" + assert WebhookConfigError.__name__ == "WebhookConfigError" + + # Test that all error classes have proper __module__ + expected_module = "core.workflow.nodes.trigger_webhook.exc" + assert WebhookNodeError.__module__ == expected_module + assert WebhookTimeoutError.__module__ == expected_module + assert WebhookNotFoundError.__module__ == expected_module + assert WebhookConfigError.__module__ == expected_module + + +def test_webhook_error_docstrings(): + """Test webhook error class docstrings.""" + assert WebhookNodeError.__doc__ == "Base webhook node error." + assert WebhookTimeoutError.__doc__ == "Webhook timeout error." + assert WebhookNotFoundError.__doc__ == "Webhook not found error." + assert WebhookConfigError.__doc__ == "Webhook configuration error." + + +def test_webhook_error_repr_and_str(): + """Test webhook error string representations.""" + error = WebhookNodeError("Test message") + + # Test __str__ method + assert str(error) == "Test message" + + # Test __repr__ method (should include class name) + repr_str = repr(error) + assert "WebhookNodeError" in repr_str + assert "Test message" in repr_str + + +def test_webhook_error_with_no_message(): + """Test webhook errors with no message.""" + # Test that errors can be instantiated without messages + errors = [ + WebhookNodeError(), + WebhookTimeoutError(), + WebhookNotFoundError(), + WebhookConfigError(), + ] + + for error in errors: + # Should be instances of their respective classes + assert isinstance(error, type(error)) + # Should be able to be raised + with pytest.raises(type(error)): + raise error diff --git a/api/tests/unit_tests/core/workflow/nodes/webhook/test_webhook_node.py b/api/tests/unit_tests/core/workflow/nodes/webhook/test_webhook_node.py new file mode 100644 index 0000000000..d7094ae5f2 --- /dev/null +++ b/api/tests/unit_tests/core/workflow/nodes/webhook/test_webhook_node.py @@ -0,0 +1,468 @@ +import pytest + +from core.app.entities.app_invoke_entities import InvokeFrom +from core.file import File, FileTransferMethod, FileType +from core.variables import StringVariable +from core.workflow.entities.graph_init_params import GraphInitParams +from core.workflow.entities.workflow_node_execution import WorkflowNodeExecutionStatus +from core.workflow.nodes.trigger_webhook.entities import ( + ContentType, + Method, + WebhookBodyParameter, + WebhookData, + WebhookParameter, +) +from core.workflow.nodes.trigger_webhook.node import TriggerWebhookNode +from core.workflow.runtime.graph_runtime_state import GraphRuntimeState +from core.workflow.runtime.variable_pool import VariablePool +from core.workflow.system_variable import SystemVariable +from models.enums import UserFrom +from models.workflow import WorkflowType + + +def create_webhook_node(webhook_data: WebhookData, variable_pool: VariablePool) -> TriggerWebhookNode: + """Helper function to create a webhook node with proper initialization.""" + node_config = { + "id": "1", + "data": webhook_data.model_dump(), + } + + node = TriggerWebhookNode( + id="1", + config=node_config, + graph_init_params=GraphInitParams( + tenant_id="1", + app_id="1", + workflow_type=WorkflowType.WORKFLOW, + workflow_id="1", + graph_config={}, + user_id="1", + user_from=UserFrom.ACCOUNT, + invoke_from=InvokeFrom.SERVICE_API, + call_depth=0, + ), + graph_runtime_state=GraphRuntimeState( + variable_pool=variable_pool, + start_at=0, + ), + ) + + node.init_node_data(node_config["data"]) + return node + + +def test_webhook_node_basic_initialization(): + """Test basic webhook node initialization and configuration.""" + data = WebhookData( + title="Test Webhook", + method=Method.POST, + content_type=ContentType.JSON, + headers=[WebhookParameter(name="X-API-Key", required=True)], + params=[WebhookParameter(name="version", required=False)], + body=[WebhookBodyParameter(name="message", type="string", required=True)], + status_code=200, + response_body="OK", + timeout=30, + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={}, + ) + + node = create_webhook_node(data, variable_pool) + + assert node.node_type.value == "trigger-webhook" + assert node.version() == "1" + assert node._get_title() == "Test Webhook" + assert node._node_data.method == Method.POST + assert node._node_data.content_type == ContentType.JSON + assert len(node._node_data.headers) == 1 + assert len(node._node_data.params) == 1 + assert len(node._node_data.body) == 1 + + +def test_webhook_node_default_config(): + """Test webhook node default configuration.""" + config = TriggerWebhookNode.get_default_config() + + assert config["type"] == "webhook" + assert config["config"]["method"] == "get" + assert config["config"]["content_type"] == "application/json" + assert config["config"]["headers"] == [] + assert config["config"]["params"] == [] + assert config["config"]["body"] == [] + assert config["config"]["async_mode"] is True + assert config["config"]["status_code"] == 200 + assert config["config"]["response_body"] == "" + assert config["config"]["timeout"] == 30 + + +def test_webhook_node_run_with_headers(): + """Test webhook node execution with header extraction.""" + data = WebhookData( + title="Test Webhook", + headers=[ + WebhookParameter(name="Authorization", required=True), + WebhookParameter(name="Content-Type", required=False), + ], + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": { + "headers": { + "Authorization": "Bearer token123", + "content-type": "application/json", # Different case + "X-Custom": "custom-value", + }, + "query_params": {}, + "body": {}, + "files": {}, + } + }, + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert result.outputs["Authorization"] == "Bearer token123" + assert result.outputs["Content_Type"] == "application/json" # Case-insensitive match + assert "_webhook_raw" in result.outputs + + +def test_webhook_node_run_with_query_params(): + """Test webhook node execution with query parameter extraction.""" + data = WebhookData( + title="Test Webhook", + params=[ + WebhookParameter(name="page", required=True), + WebhookParameter(name="limit", required=False), + WebhookParameter(name="missing", required=False), + ], + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": { + "headers": {}, + "query_params": { + "page": "1", + "limit": "10", + }, + "body": {}, + "files": {}, + } + }, + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert result.outputs["page"] == "1" + assert result.outputs["limit"] == "10" + assert result.outputs["missing"] is None # Missing parameter should be None + + +def test_webhook_node_run_with_body_params(): + """Test webhook node execution with body parameter extraction.""" + data = WebhookData( + title="Test Webhook", + body=[ + WebhookBodyParameter(name="message", type="string", required=True), + WebhookBodyParameter(name="count", type="number", required=False), + WebhookBodyParameter(name="active", type="boolean", required=False), + WebhookBodyParameter(name="metadata", type="object", required=False), + ], + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": { + "headers": {}, + "query_params": {}, + "body": { + "message": "Hello World", + "count": 42, + "active": True, + "metadata": {"key": "value"}, + }, + "files": {}, + } + }, + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert result.outputs["message"] == "Hello World" + assert result.outputs["count"] == 42 + assert result.outputs["active"] is True + assert result.outputs["metadata"] == {"key": "value"} + + +def test_webhook_node_run_with_file_params(): + """Test webhook node execution with file parameter extraction.""" + # Create mock file objects + file1 = File( + tenant_id="1", + type=FileType.IMAGE, + transfer_method=FileTransferMethod.LOCAL_FILE, + related_id="file1", + filename="image.jpg", + mime_type="image/jpeg", + storage_key="", + ) + + file2 = File( + tenant_id="1", + type=FileType.DOCUMENT, + transfer_method=FileTransferMethod.LOCAL_FILE, + related_id="file2", + filename="document.pdf", + mime_type="application/pdf", + storage_key="", + ) + + data = WebhookData( + title="Test Webhook", + body=[ + WebhookBodyParameter(name="upload", type="file", required=True), + WebhookBodyParameter(name="document", type="file", required=False), + WebhookBodyParameter(name="missing_file", type="file", required=False), + ], + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": { + "headers": {}, + "query_params": {}, + "body": {}, + "files": { + "upload": file1, + "document": file2, + }, + } + }, + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert result.outputs["upload"] == file1 + assert result.outputs["document"] == file2 + assert result.outputs["missing_file"] is None + + +def test_webhook_node_run_mixed_parameters(): + """Test webhook node execution with mixed parameter types.""" + file_obj = File( + tenant_id="1", + type=FileType.IMAGE, + transfer_method=FileTransferMethod.LOCAL_FILE, + related_id="file1", + filename="test.jpg", + mime_type="image/jpeg", + storage_key="", + ) + + data = WebhookData( + title="Test Webhook", + headers=[WebhookParameter(name="Authorization", required=True)], + params=[WebhookParameter(name="version", required=False)], + body=[ + WebhookBodyParameter(name="message", type="string", required=True), + WebhookBodyParameter(name="upload", type="file", required=False), + ], + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": { + "headers": {"Authorization": "Bearer token"}, + "query_params": {"version": "v1"}, + "body": {"message": "Test message"}, + "files": {"upload": file_obj}, + } + }, + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert result.outputs["Authorization"] == "Bearer token" + assert result.outputs["version"] == "v1" + assert result.outputs["message"] == "Test message" + assert result.outputs["upload"] == file_obj + assert "_webhook_raw" in result.outputs + + +def test_webhook_node_run_empty_webhook_data(): + """Test webhook node execution with empty webhook data.""" + data = WebhookData( + title="Test Webhook", + headers=[WebhookParameter(name="Authorization", required=False)], + params=[WebhookParameter(name="page", required=False)], + body=[WebhookBodyParameter(name="message", type="string", required=False)], + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={}, # No webhook_data + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert result.outputs["Authorization"] is None + assert result.outputs["page"] is None + assert result.outputs["message"] is None + assert result.outputs["_webhook_raw"] == {} + + +def test_webhook_node_run_case_insensitive_headers(): + """Test webhook node header extraction is case-insensitive.""" + data = WebhookData( + title="Test Webhook", + headers=[ + WebhookParameter(name="Content-Type", required=True), + WebhookParameter(name="X-API-KEY", required=True), + WebhookParameter(name="authorization", required=True), + ], + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": { + "headers": { + "content-type": "application/json", # lowercase + "x-api-key": "key123", # lowercase + "Authorization": "Bearer token", # different case + }, + "query_params": {}, + "body": {}, + "files": {}, + } + }, + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert result.outputs["Content_Type"] == "application/json" + assert result.outputs["X_API_KEY"] == "key123" + assert result.outputs["authorization"] == "Bearer token" + + +def test_webhook_node_variable_pool_user_inputs(): + """Test that webhook node uses user_inputs from variable pool correctly.""" + data = WebhookData(title="Test Webhook") + + # Add some additional variables to the pool + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": {"headers": {}, "query_params": {}, "body": {}, "files": {}}, + "other_var": "should_be_included", + }, + ) + variable_pool.add(["node1", "extra"], StringVariable(name="extra", value="extra_value")) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + # Check that all user_inputs are included in the inputs (they get converted to dict) + inputs_dict = dict(result.inputs) + assert "webhook_data" in inputs_dict + assert "other_var" in inputs_dict + assert inputs_dict["other_var"] == "should_be_included" + + +@pytest.mark.parametrize( + "method", + [Method.GET, Method.POST, Method.PUT, Method.DELETE, Method.PATCH, Method.HEAD], +) +def test_webhook_node_different_methods(method): + """Test webhook node with different HTTP methods.""" + data = WebhookData( + title="Test Webhook", + method=method, + ) + + variable_pool = VariablePool( + system_variables=SystemVariable.empty(), + user_inputs={ + "webhook_data": { + "headers": {}, + "query_params": {}, + "body": {}, + "files": {}, + } + }, + ) + + node = create_webhook_node(data, variable_pool) + result = node._run() + + assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED + assert node._node_data.method == method + + +def test_webhook_data_content_type_field(): + """Test that content_type accepts both raw strings and enum values.""" + data1 = WebhookData(title="Test", content_type="application/json") + assert data1.content_type == ContentType.JSON + + data2 = WebhookData(title="Test", content_type=ContentType.FORM_DATA) + assert data2.content_type == ContentType.FORM_DATA + + +def test_webhook_parameter_models(): + """Test webhook parameter model validation.""" + # Test WebhookParameter + param = WebhookParameter(name="test_param", required=True) + assert param.name == "test_param" + assert param.required is True + + param_default = WebhookParameter(name="test_param") + assert param_default.required is False + + # Test WebhookBodyParameter + body_param = WebhookBodyParameter(name="test_body", type="string", required=True) + assert body_param.name == "test_body" + assert body_param.type == "string" + assert body_param.required is True + + body_param_default = WebhookBodyParameter(name="test_body") + assert body_param_default.type == "string" # Default type + assert body_param_default.required is False + + +def test_webhook_data_field_defaults(): + """Test webhook data model field defaults.""" + data = WebhookData(title="Minimal Webhook") + + assert data.method == Method.GET + assert data.content_type == ContentType.JSON + assert data.headers == [] + assert data.params == [] + assert data.body == [] + assert data.status_code == 200 + assert data.response_body == "" + assert data.webhook_id is None + assert data.timeout == 30 diff --git a/api/tests/unit_tests/core/workflow/test_enums.py b/api/tests/unit_tests/core/workflow/test_enums.py new file mode 100644 index 0000000000..7cdb2328f2 --- /dev/null +++ b/api/tests/unit_tests/core/workflow/test_enums.py @@ -0,0 +1,32 @@ +"""Tests for workflow pause related enums and constants.""" + +from core.workflow.enums import ( + WorkflowExecutionStatus, +) + + +class TestWorkflowExecutionStatus: + """Test WorkflowExecutionStatus enum.""" + + def test_is_ended_method(self): + """Test is_ended method for different statuses.""" + # Test ended statuses + ended_statuses = [ + WorkflowExecutionStatus.SUCCEEDED, + WorkflowExecutionStatus.FAILED, + WorkflowExecutionStatus.PARTIAL_SUCCEEDED, + WorkflowExecutionStatus.STOPPED, + ] + + for status in ended_statuses: + assert status.is_ended(), f"{status} should be considered ended" + + # Test non-ended statuses + non_ended_statuses = [ + WorkflowExecutionStatus.SCHEDULED, + WorkflowExecutionStatus.RUNNING, + WorkflowExecutionStatus.PAUSED, + ] + + for status in non_ended_statuses: + assert not status.is_ended(), f"{status} should not be considered ended" diff --git a/api/tests/unit_tests/core/workflow/test_system_variable_read_only_view.py b/api/tests/unit_tests/core/workflow/test_system_variable_read_only_view.py new file mode 100644 index 0000000000..57bc96fe71 --- /dev/null +++ b/api/tests/unit_tests/core/workflow/test_system_variable_read_only_view.py @@ -0,0 +1,202 @@ +from typing import cast + +import pytest + +from core.file.models import File, FileTransferMethod, FileType +from core.workflow.system_variable import SystemVariable, SystemVariableReadOnlyView + + +class TestSystemVariableReadOnlyView: + """Test cases for SystemVariableReadOnlyView class.""" + + def test_read_only_property_access(self): + """Test that all properties return correct values from wrapped instance.""" + # Create test data + test_file = File( + id="file-123", + tenant_id="tenant-123", + type=FileType.IMAGE, + transfer_method=FileTransferMethod.LOCAL_FILE, + related_id="related-123", + ) + + datasource_info = {"key": "value", "nested": {"data": 42}} + + # Create SystemVariable with all fields + system_var = SystemVariable( + user_id="user-123", + app_id="app-123", + workflow_id="workflow-123", + files=[test_file], + workflow_execution_id="exec-123", + query="test query", + conversation_id="conv-123", + dialogue_count=5, + document_id="doc-123", + original_document_id="orig-doc-123", + dataset_id="dataset-123", + batch="batch-123", + datasource_type="type-123", + datasource_info=datasource_info, + invoke_from="invoke-123", + ) + + # Create read-only view + read_only_view = SystemVariableReadOnlyView(system_var) + + # Test all properties + assert read_only_view.user_id == "user-123" + assert read_only_view.app_id == "app-123" + assert read_only_view.workflow_id == "workflow-123" + assert read_only_view.workflow_execution_id == "exec-123" + assert read_only_view.query == "test query" + assert read_only_view.conversation_id == "conv-123" + assert read_only_view.dialogue_count == 5 + assert read_only_view.document_id == "doc-123" + assert read_only_view.original_document_id == "orig-doc-123" + assert read_only_view.dataset_id == "dataset-123" + assert read_only_view.batch == "batch-123" + assert read_only_view.datasource_type == "type-123" + assert read_only_view.invoke_from == "invoke-123" + + def test_defensive_copying_of_mutable_objects(self): + """Test that mutable objects are defensively copied.""" + # Create test data + test_file = File( + id="file-123", + tenant_id="tenant-123", + type=FileType.IMAGE, + transfer_method=FileTransferMethod.LOCAL_FILE, + related_id="related-123", + ) + + datasource_info = {"key": "original_value"} + + # Create SystemVariable + system_var = SystemVariable( + files=[test_file], datasource_info=datasource_info, workflow_execution_id="exec-123" + ) + + # Create read-only view + read_only_view = SystemVariableReadOnlyView(system_var) + + # Test files defensive copying + files_copy = read_only_view.files + assert isinstance(files_copy, tuple) # Should be immutable tuple + assert len(files_copy) == 1 + assert files_copy[0].id == "file-123" + + # Verify it's a copy (can't modify original through view) + assert isinstance(files_copy, tuple) + # tuples don't have append method, so they're immutable + + # Test datasource_info defensive copying + datasource_copy = read_only_view.datasource_info + assert datasource_copy is not None + assert datasource_copy["key"] == "original_value" + + datasource_copy = cast(dict, datasource_copy) + with pytest.raises(TypeError): + datasource_copy["key"] = "modified value" + + # Verify original is unchanged + assert system_var.datasource_info is not None + assert system_var.datasource_info["key"] == "original_value" + assert read_only_view.datasource_info is not None + assert read_only_view.datasource_info["key"] == "original_value" + + def test_always_accesses_latest_data(self): + """Test that properties always return the latest data from wrapped instance.""" + # Create SystemVariable + system_var = SystemVariable(user_id="original-user", workflow_execution_id="exec-123") + + # Create read-only view + read_only_view = SystemVariableReadOnlyView(system_var) + + # Verify initial value + assert read_only_view.user_id == "original-user" + + # Modify the wrapped instance + system_var.user_id = "modified-user" + + # Verify view returns the new value + assert read_only_view.user_id == "modified-user" + + def test_repr_method(self): + """Test the __repr__ method.""" + # Create SystemVariable + system_var = SystemVariable(workflow_execution_id="exec-123") + + # Create read-only view + read_only_view = SystemVariableReadOnlyView(system_var) + + # Test repr + repr_str = repr(read_only_view) + assert "SystemVariableReadOnlyView" in repr_str + assert "system_variable=" in repr_str + + def test_none_value_handling(self): + """Test that None values are properly handled.""" + # Create SystemVariable with all None values except workflow_execution_id + system_var = SystemVariable( + user_id=None, + app_id=None, + workflow_id=None, + workflow_execution_id="exec-123", + query=None, + conversation_id=None, + dialogue_count=None, + document_id=None, + original_document_id=None, + dataset_id=None, + batch=None, + datasource_type=None, + datasource_info=None, + invoke_from=None, + ) + + # Create read-only view + read_only_view = SystemVariableReadOnlyView(system_var) + + # Test all None values + assert read_only_view.user_id is None + assert read_only_view.app_id is None + assert read_only_view.workflow_id is None + assert read_only_view.query is None + assert read_only_view.conversation_id is None + assert read_only_view.dialogue_count is None + assert read_only_view.document_id is None + assert read_only_view.original_document_id is None + assert read_only_view.dataset_id is None + assert read_only_view.batch is None + assert read_only_view.datasource_type is None + assert read_only_view.datasource_info is None + assert read_only_view.invoke_from is None + + # files should be empty tuple even when default list is empty + assert read_only_view.files == () + + def test_empty_files_handling(self): + """Test that empty files list is handled correctly.""" + # Create SystemVariable with empty files + system_var = SystemVariable(files=[], workflow_execution_id="exec-123") + + # Create read-only view + read_only_view = SystemVariableReadOnlyView(system_var) + + # Test files handling + assert read_only_view.files == () + assert isinstance(read_only_view.files, tuple) + + def test_empty_datasource_info_handling(self): + """Test that empty datasource_info is handled correctly.""" + # Create SystemVariable with empty datasource_info + system_var = SystemVariable(datasource_info={}, workflow_execution_id="exec-123") + + # Create read-only view + read_only_view = SystemVariableReadOnlyView(system_var) + + # Test datasource_info handling + assert read_only_view.datasource_info == {} + # Should be a copy, not the same object + assert read_only_view.datasource_info is not system_var.datasource_info diff --git a/api/tests/unit_tests/extensions/test_celery_ssl.py b/api/tests/unit_tests/extensions/test_celery_ssl.py index bc46fe8322..fc7a090ef9 100644 --- a/api/tests/unit_tests/extensions/test_celery_ssl.py +++ b/api/tests/unit_tests/extensions/test_celery_ssl.py @@ -131,6 +131,12 @@ class TestCelerySSLConfiguration: mock_config.ENABLE_MAIL_CLEAN_DOCUMENT_NOTIFY_TASK = False mock_config.ENABLE_DATASETS_QUEUE_MONITOR = False mock_config.ENABLE_CHECK_UPGRADABLE_PLUGIN_TASK = False + mock_config.ENABLE_WORKFLOW_SCHEDULE_POLLER_TASK = False + mock_config.WORKFLOW_SCHEDULE_POLLER_INTERVAL = 1 + mock_config.WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE = 100 + mock_config.WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK = 0 + mock_config.ENABLE_TRIGGER_PROVIDER_REFRESH_TASK = False + mock_config.TRIGGER_PROVIDER_REFRESH_INTERVAL = 15 with patch("extensions.ext_celery.dify_config", mock_config): from dify_app import DifyApp diff --git a/api/tests/unit_tests/libs/broadcast_channel/redis/test_channel_unit_tests.py b/api/tests/unit_tests/libs/broadcast_channel/redis/test_channel_unit_tests.py new file mode 100644 index 0000000000..dffad4142c --- /dev/null +++ b/api/tests/unit_tests/libs/broadcast_channel/redis/test_channel_unit_tests.py @@ -0,0 +1,514 @@ +""" +Comprehensive unit tests for Redis broadcast channel implementation. + +This test suite covers all aspects of the Redis broadcast channel including: +- Basic functionality and contract compliance +- Error handling and edge cases +- Thread safety and concurrency +- Resource management and cleanup +- Performance and reliability scenarios +""" + +import dataclasses +import threading +import time +from collections.abc import Generator +from unittest.mock import MagicMock, patch + +import pytest + +from libs.broadcast_channel.exc import BroadcastChannelError, SubscriptionClosedError +from libs.broadcast_channel.redis.channel import ( + BroadcastChannel as RedisBroadcastChannel, +) +from libs.broadcast_channel.redis.channel import ( + Topic, + _RedisSubscription, +) + + +class TestBroadcastChannel: + """Test cases for the main BroadcastChannel class.""" + + @pytest.fixture + def mock_redis_client(self) -> MagicMock: + """Create a mock Redis client for testing.""" + client = MagicMock() + client.pubsub.return_value = MagicMock() + return client + + @pytest.fixture + def broadcast_channel(self, mock_redis_client: MagicMock) -> RedisBroadcastChannel: + """Create a BroadcastChannel instance with mock Redis client.""" + return RedisBroadcastChannel(mock_redis_client) + + def test_topic_creation(self, broadcast_channel: RedisBroadcastChannel, mock_redis_client: MagicMock): + """Test that topic() method returns a Topic instance with correct parameters.""" + topic_name = "test-topic" + topic = broadcast_channel.topic(topic_name) + + assert isinstance(topic, Topic) + assert topic._client == mock_redis_client + assert topic._topic == topic_name + + def test_topic_isolation(self, broadcast_channel: RedisBroadcastChannel): + """Test that different topic names create isolated Topic instances.""" + topic1 = broadcast_channel.topic("topic1") + topic2 = broadcast_channel.topic("topic2") + + assert topic1 is not topic2 + assert topic1._topic == "topic1" + assert topic2._topic == "topic2" + + +class TestTopic: + """Test cases for the Topic class.""" + + @pytest.fixture + def mock_redis_client(self) -> MagicMock: + """Create a mock Redis client for testing.""" + client = MagicMock() + client.pubsub.return_value = MagicMock() + return client + + @pytest.fixture + def topic(self, mock_redis_client: MagicMock) -> Topic: + """Create a Topic instance for testing.""" + return Topic(mock_redis_client, "test-topic") + + def test_as_producer_returns_self(self, topic: Topic): + """Test that as_producer() returns self as Producer interface.""" + producer = topic.as_producer() + assert producer is topic + # Producer is a Protocol, check duck typing instead + assert hasattr(producer, "publish") + + def test_as_subscriber_returns_self(self, topic: Topic): + """Test that as_subscriber() returns self as Subscriber interface.""" + subscriber = topic.as_subscriber() + assert subscriber is topic + # Subscriber is a Protocol, check duck typing instead + assert hasattr(subscriber, "subscribe") + + def test_publish_calls_redis_publish(self, topic: Topic, mock_redis_client: MagicMock): + """Test that publish() calls Redis PUBLISH with correct parameters.""" + payload = b"test message" + topic.publish(payload) + + mock_redis_client.publish.assert_called_once_with("test-topic", payload) + + +@dataclasses.dataclass(frozen=True) +class SubscriptionTestCase: + """Test case data for subscription tests.""" + + name: str + buffer_size: int + payload: bytes + expected_messages: list[bytes] + should_drop: bool = False + description: str = "" + + +class TestRedisSubscription: + """Test cases for the _RedisSubscription class.""" + + @pytest.fixture + def mock_pubsub(self) -> MagicMock: + """Create a mock PubSub instance for testing.""" + pubsub = MagicMock() + pubsub.subscribe = MagicMock() + pubsub.unsubscribe = MagicMock() + pubsub.close = MagicMock() + pubsub.get_message = MagicMock() + return pubsub + + @pytest.fixture + def subscription(self, mock_pubsub: MagicMock) -> Generator[_RedisSubscription, None, None]: + """Create a _RedisSubscription instance for testing.""" + subscription = _RedisSubscription( + pubsub=mock_pubsub, + topic="test-topic", + ) + yield subscription + subscription.close() + + @pytest.fixture + def started_subscription(self, subscription: _RedisSubscription) -> _RedisSubscription: + """Create a subscription that has been started.""" + subscription._start_if_needed() + return subscription + + # ==================== Lifecycle Tests ==================== + + def test_subscription_initialization(self, mock_pubsub: MagicMock): + """Test that subscription is properly initialized.""" + subscription = _RedisSubscription( + pubsub=mock_pubsub, + topic="test-topic", + ) + + assert subscription._pubsub is mock_pubsub + assert subscription._topic == "test-topic" + assert not subscription._closed.is_set() + assert subscription._dropped_count == 0 + assert subscription._listener_thread is None + assert not subscription._started + + def test_start_if_needed_first_call(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that _start_if_needed() properly starts subscription on first call.""" + subscription._start_if_needed() + + mock_pubsub.subscribe.assert_called_once_with("test-topic") + assert subscription._started is True + assert subscription._listener_thread is not None + + def test_start_if_needed_subsequent_calls(self, started_subscription: _RedisSubscription): + """Test that _start_if_needed() doesn't start subscription on subsequent calls.""" + original_thread = started_subscription._listener_thread + started_subscription._start_if_needed() + + # Should not create new thread or generator + assert started_subscription._listener_thread is original_thread + + def test_start_if_needed_when_closed(self, subscription: _RedisSubscription): + """Test that _start_if_needed() raises error when subscription is closed.""" + subscription.close() + + with pytest.raises(SubscriptionClosedError, match="The Redis subscription is closed"): + subscription._start_if_needed() + + def test_start_if_needed_when_cleaned_up(self, subscription: _RedisSubscription): + """Test that _start_if_needed() raises error when pubsub is None.""" + subscription._pubsub = None + + with pytest.raises(SubscriptionClosedError, match="The Redis subscription has been cleaned up"): + subscription._start_if_needed() + + def test_context_manager_usage(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that subscription works as context manager.""" + with subscription as sub: + assert sub is subscription + assert subscription._started is True + mock_pubsub.subscribe.assert_called_once_with("test-topic") + + def test_close_idempotent(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that close() is idempotent and can be called multiple times.""" + subscription._start_if_needed() + + # Close multiple times + subscription.close() + subscription.close() + subscription.close() + + # Should only cleanup once + mock_pubsub.unsubscribe.assert_called_once_with("test-topic") + mock_pubsub.close.assert_called_once() + assert subscription._pubsub is None + assert subscription._closed.is_set() + + def test_close_cleanup(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that close() properly cleans up all resources.""" + subscription._start_if_needed() + thread = subscription._listener_thread + + subscription.close() + + # Verify cleanup + mock_pubsub.unsubscribe.assert_called_once_with("test-topic") + mock_pubsub.close.assert_called_once() + assert subscription._pubsub is None + assert subscription._listener_thread is None + + # Wait for thread to finish (with timeout) + if thread and thread.is_alive(): + thread.join(timeout=1.0) + assert not thread.is_alive() + + # ==================== Message Processing Tests ==================== + + def test_message_iterator_with_messages(self, started_subscription: _RedisSubscription): + """Test message iterator behavior with messages in queue.""" + test_messages = [b"msg1", b"msg2", b"msg3"] + + # Add messages to queue + for msg in test_messages: + started_subscription._queue.put_nowait(msg) + + # Iterate through messages + iterator = iter(started_subscription) + received_messages = [] + + for msg in iterator: + received_messages.append(msg) + if len(received_messages) >= len(test_messages): + break + + assert received_messages == test_messages + + def test_message_iterator_when_closed(self, subscription: _RedisSubscription): + """Test that iterator raises error when subscription is closed.""" + subscription.close() + + with pytest.raises(BroadcastChannelError, match="The Redis subscription is closed"): + iter(subscription) + + # ==================== Message Enqueue Tests ==================== + + def test_enqueue_message_success(self, started_subscription: _RedisSubscription): + """Test successful message enqueue.""" + payload = b"test message" + + started_subscription._enqueue_message(payload) + + assert started_subscription._queue.qsize() == 1 + assert started_subscription._queue.get_nowait() == payload + + def test_enqueue_message_when_closed(self, subscription: _RedisSubscription): + """Test message enqueue when subscription is closed.""" + subscription.close() + payload = b"test message" + + # Should not raise exception, but should not enqueue + subscription._enqueue_message(payload) + + assert subscription._queue.empty() + + def test_enqueue_message_with_full_queue(self, started_subscription: _RedisSubscription): + """Test message enqueue with full queue (dropping behavior).""" + # Fill the queue + for i in range(started_subscription._queue.maxsize): + started_subscription._queue.put_nowait(f"old_msg_{i}".encode()) + + # Try to enqueue new message (should drop oldest) + new_message = b"new_message" + started_subscription._enqueue_message(new_message) + + # Should have dropped one message and added new one + assert started_subscription._dropped_count == 1 + + # New message should be in queue + messages = [] + while not started_subscription._queue.empty(): + messages.append(started_subscription._queue.get_nowait()) + + assert new_message in messages + + # ==================== Listener Thread Tests ==================== + + @patch("time.sleep", side_effect=lambda x: None) # Speed up test + def test_listener_thread_normal_operation( + self, mock_sleep, subscription: _RedisSubscription, mock_pubsub: MagicMock + ): + """Test listener thread normal operation.""" + # Mock message from Redis + mock_message = {"type": "message", "channel": "test-topic", "data": b"test payload"} + mock_pubsub.get_message.return_value = mock_message + + # Start listener + subscription._start_if_needed() + + # Wait a bit for processing + time.sleep(0.1) + + # Verify message was processed + assert not subscription._queue.empty() + assert subscription._queue.get_nowait() == b"test payload" + + def test_listener_thread_ignores_subscribe_messages(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that listener thread ignores subscribe/unsubscribe messages.""" + mock_message = {"type": "subscribe", "channel": "test-topic", "data": 1} + mock_pubsub.get_message.return_value = mock_message + + subscription._start_if_needed() + time.sleep(0.1) + + # Should not enqueue subscribe messages + assert subscription._queue.empty() + + def test_listener_thread_ignores_wrong_channel(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that listener thread ignores messages from wrong channels.""" + mock_message = {"type": "message", "channel": "wrong-topic", "data": b"test payload"} + mock_pubsub.get_message.return_value = mock_message + + subscription._start_if_needed() + time.sleep(0.1) + + # Should not enqueue messages from wrong channels + assert subscription._queue.empty() + + def test_listener_thread_handles_redis_exceptions(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that listener thread handles Redis exceptions gracefully.""" + mock_pubsub.get_message.side_effect = Exception("Redis error") + + subscription._start_if_needed() + + # Wait for thread to handle exception + time.sleep(0.2) + + # Thread should still be alive but not processing + assert subscription._listener_thread is not None + assert not subscription._listener_thread.is_alive() + + def test_listener_thread_stops_when_closed(self, subscription: _RedisSubscription, mock_pubsub: MagicMock): + """Test that listener thread stops when subscription is closed.""" + subscription._start_if_needed() + thread = subscription._listener_thread + + # Close subscription + subscription.close() + + # Wait for thread to finish + if thread is not None and thread.is_alive(): + thread.join(timeout=1.0) + + assert thread is None or not thread.is_alive() + + # ==================== Table-driven Tests ==================== + + @pytest.mark.parametrize( + "test_case", + [ + SubscriptionTestCase( + name="basic_message", + buffer_size=5, + payload=b"hello world", + expected_messages=[b"hello world"], + description="Basic message publishing and receiving", + ), + SubscriptionTestCase( + name="empty_message", + buffer_size=5, + payload=b"", + expected_messages=[b""], + description="Empty message handling", + ), + SubscriptionTestCase( + name="large_message", + buffer_size=5, + payload=b"x" * 10000, + expected_messages=[b"x" * 10000], + description="Large message handling", + ), + SubscriptionTestCase( + name="unicode_message", + buffer_size=5, + payload="你好世界".encode(), + expected_messages=["你好世界".encode()], + description="Unicode message handling", + ), + ], + ) + def test_subscription_scenarios(self, test_case: SubscriptionTestCase, mock_pubsub: MagicMock): + """Test various subscription scenarios using table-driven approach.""" + subscription = _RedisSubscription( + pubsub=mock_pubsub, + topic="test-topic", + ) + + # Simulate receiving message + mock_message = {"type": "message", "channel": "test-topic", "data": test_case.payload} + mock_pubsub.get_message.return_value = mock_message + + try: + with subscription: + # Wait for message processing + time.sleep(0.1) + + # Collect received messages + received = [] + for msg in subscription: + received.append(msg) + if len(received) >= len(test_case.expected_messages): + break + + assert received == test_case.expected_messages, f"Failed: {test_case.description}" + finally: + subscription.close() + + def test_concurrent_close_and_enqueue(self, started_subscription: _RedisSubscription): + """Test concurrent close and enqueue operations.""" + errors = [] + + def close_subscription(): + try: + time.sleep(0.05) # Small delay + started_subscription.close() + except Exception as e: + errors.append(e) + + def enqueue_messages(): + try: + for i in range(50): + started_subscription._enqueue_message(f"msg_{i}".encode()) + time.sleep(0.001) + except Exception as e: + errors.append(e) + + # Start threads + close_thread = threading.Thread(target=close_subscription) + enqueue_thread = threading.Thread(target=enqueue_messages) + + close_thread.start() + enqueue_thread.start() + + # Wait for completion + close_thread.join(timeout=2.0) + enqueue_thread.join(timeout=2.0) + + # Should not have any errors (operations should be safe) + assert len(errors) == 0 + + # ==================== Error Handling Tests ==================== + + def test_iterator_after_close(self, subscription: _RedisSubscription): + """Test iterator behavior after close.""" + subscription.close() + + with pytest.raises(SubscriptionClosedError, match="The Redis subscription is closed"): + iter(subscription) + + def test_start_after_close(self, subscription: _RedisSubscription): + """Test start attempts after close.""" + subscription.close() + + with pytest.raises(SubscriptionClosedError, match="The Redis subscription is closed"): + subscription._start_if_needed() + + def test_pubsub_none_operations(self, subscription: _RedisSubscription): + """Test operations when pubsub is None.""" + subscription._pubsub = None + + with pytest.raises(SubscriptionClosedError, match="The Redis subscription has been cleaned up"): + subscription._start_if_needed() + + # Close should still work + subscription.close() # Should not raise + + def test_channel_name_variations(self, mock_pubsub: MagicMock): + """Test various channel name formats.""" + channel_names = [ + "simple", + "with-dashes", + "with_underscores", + "with.numbers", + "WITH.UPPERCASE", + "mixed-CASE_name", + "very.long.channel.name.with.multiple.parts", + ] + + for channel_name in channel_names: + subscription = _RedisSubscription( + pubsub=mock_pubsub, + topic=channel_name, + ) + + subscription._start_if_needed() + mock_pubsub.subscribe.assert_called_with(channel_name) + subscription.close() + + def test_received_on_closed_subscription(self, subscription: _RedisSubscription): + subscription.close() + + with pytest.raises(SubscriptionClosedError): + subscription.receive() diff --git a/api/tests/unit_tests/libs/test_cron_compatibility.py b/api/tests/unit_tests/libs/test_cron_compatibility.py new file mode 100644 index 0000000000..6f3a94f6dc --- /dev/null +++ b/api/tests/unit_tests/libs/test_cron_compatibility.py @@ -0,0 +1,381 @@ +""" +Enhanced cron syntax compatibility tests for croniter backend. + +This test suite mirrors the frontend cron-parser tests to ensure +complete compatibility between frontend and backend cron processing. +""" + +import unittest +from datetime import UTC, datetime, timedelta + +import pytest +import pytz +from croniter import CroniterBadCronError + +from libs.schedule_utils import calculate_next_run_at + + +class TestCronCompatibility(unittest.TestCase): + """Test enhanced cron syntax compatibility with frontend.""" + + def setUp(self): + """Set up test environment with fixed time.""" + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_enhanced_dayofweek_syntax(self): + """Test enhanced day-of-week syntax compatibility.""" + test_cases = [ + ("0 9 * * 7", 0), # Sunday as 7 + ("0 9 * * 0", 0), # Sunday as 0 + ("0 9 * * MON", 1), # Monday abbreviation + ("0 9 * * TUE", 2), # Tuesday abbreviation + ("0 9 * * WED", 3), # Wednesday abbreviation + ("0 9 * * THU", 4), # Thursday abbreviation + ("0 9 * * FRI", 5), # Friday abbreviation + ("0 9 * * SAT", 6), # Saturday abbreviation + ("0 9 * * SUN", 0), # Sunday abbreviation + ] + + for expr, expected_weekday in test_cases: + with self.subTest(expr=expr): + next_time = calculate_next_run_at(expr, "UTC", self.base_time) + assert next_time is not None + assert (next_time.weekday() + 1 if next_time.weekday() < 6 else 0) == expected_weekday + assert next_time.hour == 9 + assert next_time.minute == 0 + + def test_enhanced_month_syntax(self): + """Test enhanced month syntax compatibility.""" + test_cases = [ + ("0 9 1 JAN *", 1), # January abbreviation + ("0 9 1 FEB *", 2), # February abbreviation + ("0 9 1 MAR *", 3), # March abbreviation + ("0 9 1 APR *", 4), # April abbreviation + ("0 9 1 MAY *", 5), # May abbreviation + ("0 9 1 JUN *", 6), # June abbreviation + ("0 9 1 JUL *", 7), # July abbreviation + ("0 9 1 AUG *", 8), # August abbreviation + ("0 9 1 SEP *", 9), # September abbreviation + ("0 9 1 OCT *", 10), # October abbreviation + ("0 9 1 NOV *", 11), # November abbreviation + ("0 9 1 DEC *", 12), # December abbreviation + ] + + for expr, expected_month in test_cases: + with self.subTest(expr=expr): + next_time = calculate_next_run_at(expr, "UTC", self.base_time) + assert next_time is not None + assert next_time.month == expected_month + assert next_time.day == 1 + assert next_time.hour == 9 + + def test_predefined_expressions(self): + """Test predefined cron expressions compatibility.""" + test_cases = [ + ("@yearly", lambda dt: dt.month == 1 and dt.day == 1 and dt.hour == 0), + ("@annually", lambda dt: dt.month == 1 and dt.day == 1 and dt.hour == 0), + ("@monthly", lambda dt: dt.day == 1 and dt.hour == 0), + ("@weekly", lambda dt: dt.weekday() == 6 and dt.hour == 0), # Sunday = 6 in weekday() + ("@daily", lambda dt: dt.hour == 0 and dt.minute == 0), + ("@midnight", lambda dt: dt.hour == 0 and dt.minute == 0), + ("@hourly", lambda dt: dt.minute == 0), + ] + + for expr, validator in test_cases: + with self.subTest(expr=expr): + next_time = calculate_next_run_at(expr, "UTC", self.base_time) + assert next_time is not None + assert validator(next_time), f"Validator failed for {expr}: {next_time}" + + def test_special_characters(self): + """Test special characters in cron expressions.""" + test_cases = [ + "0 9 ? * 1", # ? wildcard + "0 12 * * 7", # Sunday as 7 + "0 15 L * *", # Last day of month + ] + + for expr in test_cases: + with self.subTest(expr=expr): + try: + next_time = calculate_next_run_at(expr, "UTC", self.base_time) + assert next_time is not None + assert next_time > self.base_time + except Exception as e: + self.fail(f"Expression '{expr}' should be valid but raised: {e}") + + def test_range_and_list_syntax(self): + """Test range and list syntax with abbreviations.""" + test_cases = [ + "0 9 * * MON-FRI", # Weekday range with abbreviations + "0 9 * JAN-MAR *", # Month range with abbreviations + "0 9 * * SUN,WED,FRI", # Weekday list with abbreviations + "0 9 1 JAN,JUN,DEC *", # Month list with abbreviations + ] + + for expr in test_cases: + with self.subTest(expr=expr): + try: + next_time = calculate_next_run_at(expr, "UTC", self.base_time) + assert next_time is not None + assert next_time > self.base_time + except Exception as e: + self.fail(f"Expression '{expr}' should be valid but raised: {e}") + + def test_invalid_enhanced_syntax(self): + """Test that invalid enhanced syntax is properly rejected.""" + invalid_expressions = [ + "0 12 * JANUARY *", # Full month name (not supported) + "0 12 * * MONDAY", # Full day name (not supported) + "0 12 32 JAN *", # Invalid day with valid month + "15 10 1 * 8", # Invalid day of week + "15 10 1 INVALID *", # Invalid month abbreviation + "15 10 1 * INVALID", # Invalid day abbreviation + "@invalid", # Invalid predefined expression + ] + + for expr in invalid_expressions: + with self.subTest(expr=expr): + with pytest.raises((CroniterBadCronError, ValueError)): + calculate_next_run_at(expr, "UTC", self.base_time) + + def test_edge_cases_with_enhanced_syntax(self): + """Test edge cases with enhanced syntax.""" + test_cases = [ + ("0 0 29 FEB *", lambda dt: dt.month == 2 and dt.day == 29), # Feb 29 with month abbreviation + ] + + for expr, validator in test_cases: + with self.subTest(expr=expr): + try: + next_time = calculate_next_run_at(expr, "UTC", self.base_time) + if next_time: # Some combinations might not occur soon + assert validator(next_time), f"Validator failed for {expr}: {next_time}" + except (CroniterBadCronError, ValueError): + # Some edge cases might be valid but not have upcoming occurrences + pass + + # Test complex expressions that have specific constraints + complex_expr = "59 23 31 DEC SAT" # December 31st at 23:59 on Saturday + try: + next_time = calculate_next_run_at(complex_expr, "UTC", self.base_time) + if next_time: + # The next occurrence might not be exactly Dec 31 if it's not a Saturday + # Just verify it's a valid result + assert next_time is not None + assert next_time.hour == 23 + assert next_time.minute == 59 + except Exception: + # Complex date constraints might not have near-future occurrences + pass + + +class TestTimezoneCompatibility(unittest.TestCase): + """Test timezone compatibility between frontend and backend.""" + + def setUp(self): + """Set up test environment.""" + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_timezone_consistency(self): + """Test that calculations are consistent across different timezones.""" + timezones = [ + "UTC", + "America/New_York", + "Europe/London", + "Asia/Tokyo", + "Asia/Kolkata", + "Australia/Sydney", + ] + + expression = "0 12 * * *" # Daily at noon + + for timezone in timezones: + with self.subTest(timezone=timezone): + next_time = calculate_next_run_at(expression, timezone, self.base_time) + assert next_time is not None + + # Convert back to the target timezone to verify it's noon + tz = pytz.timezone(timezone) + local_time = next_time.astimezone(tz) + assert local_time.hour == 12 + assert local_time.minute == 0 + + def test_dst_handling(self): + """Test DST boundary handling.""" + # Test around DST spring forward (March 2024) + dst_base = datetime(2024, 3, 8, 10, 0, 0, tzinfo=UTC) + expression = "0 2 * * *" # 2 AM daily (problematic during DST) + timezone = "America/New_York" + + try: + next_time = calculate_next_run_at(expression, timezone, dst_base) + assert next_time is not None + + # During DST spring forward, 2 AM becomes 3 AM - both are acceptable + tz = pytz.timezone(timezone) + local_time = next_time.astimezone(tz) + assert local_time.hour in [2, 3] # Either 2 AM or 3 AM is acceptable + except Exception as e: + self.fail(f"DST handling failed: {e}") + + def test_half_hour_timezones(self): + """Test timezones with half-hour offsets.""" + timezones_with_offsets = [ + ("Asia/Kolkata", 17, 30), # UTC+5:30 -> 12:00 UTC = 17:30 IST + ("Australia/Adelaide", 22, 30), # UTC+10:30 -> 12:00 UTC = 22:30 ACDT (summer time) + ] + + expression = "0 12 * * *" # Noon UTC + + for timezone, expected_hour, expected_minute in timezones_with_offsets: + with self.subTest(timezone=timezone): + try: + next_time = calculate_next_run_at(expression, timezone, self.base_time) + assert next_time is not None + + tz = pytz.timezone(timezone) + local_time = next_time.astimezone(tz) + assert local_time.hour == expected_hour + assert local_time.minute == expected_minute + except Exception: + # Some complex timezone calculations might vary + pass + + def test_invalid_timezone_handling(self): + """Test handling of invalid timezones.""" + expression = "0 12 * * *" + invalid_timezone = "Invalid/Timezone" + + with pytest.raises((ValueError, Exception)): # Should raise an exception + calculate_next_run_at(expression, invalid_timezone, self.base_time) + + +class TestFrontendBackendIntegration(unittest.TestCase): + """Test integration patterns that mirror frontend usage.""" + + def setUp(self): + """Set up test environment.""" + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_execution_time_calculator_pattern(self): + """Test the pattern used by execution-time-calculator.ts.""" + # This mirrors the exact usage from execution-time-calculator.ts:47 + test_data = { + "cron_expression": "30 14 * * 1-5", # 2:30 PM weekdays + "timezone": "America/New_York", + } + + # Get next 5 execution times (like the frontend does) + execution_times = [] + current_base = self.base_time + + for _ in range(5): + next_time = calculate_next_run_at(test_data["cron_expression"], test_data["timezone"], current_base) + assert next_time is not None + execution_times.append(next_time) + current_base = next_time + timedelta(seconds=1) # Move slightly forward + + assert len(execution_times) == 5 + + # Validate each execution time + for exec_time in execution_times: + # Convert to local timezone + tz = pytz.timezone(test_data["timezone"]) + local_time = exec_time.astimezone(tz) + + # Should be weekdays (1-5) + assert local_time.weekday() in [0, 1, 2, 3, 4] # Mon-Fri in Python weekday + + # Should be 2:30 PM in local time + assert local_time.hour == 14 + assert local_time.minute == 30 + assert local_time.second == 0 + + def test_schedule_service_integration(self): + """Test integration with ScheduleService patterns.""" + from core.workflow.nodes.trigger_schedule.entities import VisualConfig + from services.trigger.schedule_service import ScheduleService + + # Test enhanced syntax through visual config conversion + visual_configs = [ + # Test with month abbreviations + { + "frequency": "monthly", + "config": VisualConfig(time="9:00 AM", monthly_days=[1]), + "expected_cron": "0 9 1 * *", + }, + # Test with weekday abbreviations + { + "frequency": "weekly", + "config": VisualConfig(time="2:30 PM", weekdays=["mon", "wed", "fri"]), + "expected_cron": "30 14 * * 1,3,5", + }, + ] + + for test_case in visual_configs: + with self.subTest(frequency=test_case["frequency"]): + cron_expr = ScheduleService.visual_to_cron(test_case["frequency"], test_case["config"]) + assert cron_expr == test_case["expected_cron"] + + # Verify the generated cron expression is valid + next_time = calculate_next_run_at(cron_expr, "UTC", self.base_time) + assert next_time is not None + + def test_error_handling_consistency(self): + """Test that error handling matches frontend expectations.""" + invalid_expressions = [ + "60 10 1 * *", # Invalid minute + "15 25 1 * *", # Invalid hour + "15 10 32 * *", # Invalid day + "15 10 1 13 *", # Invalid month + "15 10 1", # Too few fields + "15 10 1 * * *", # 6 fields (not supported in frontend) + "0 15 10 1 * * *", # 7 fields (not supported in frontend) + "invalid expression", # Completely invalid + ] + + for expr in invalid_expressions: + with self.subTest(expr=repr(expr)): + with pytest.raises((CroniterBadCronError, ValueError, Exception)): + calculate_next_run_at(expr, "UTC", self.base_time) + + # Note: Empty/whitespace expressions are not tested here as they are + # not expected in normal usage due to database constraints (nullable=False) + + def test_performance_requirements(self): + """Test that complex expressions parse within reasonable time.""" + import time + + complex_expressions = [ + "*/5 9-17 * * 1-5", # Every 5 minutes, weekdays, business hours + "0 */2 1,15 * *", # Every 2 hours on 1st and 15th + "30 14 * * 1,3,5", # Mon, Wed, Fri at 14:30 + "15,45 8-18 * * 1-5", # 15 and 45 minutes past hour, weekdays + "0 9 * JAN-MAR MON-FRI", # Enhanced syntax: Q1 weekdays at 9 AM + "0 12 ? * SUN", # Enhanced syntax: Sundays at noon with ? + ] + + start_time = time.time() + + for expr in complex_expressions: + with self.subTest(expr=expr): + try: + next_time = calculate_next_run_at(expr, "UTC", self.base_time) + assert next_time is not None + except CroniterBadCronError: + # Some enhanced syntax might not be supported, that's OK + pass + + end_time = time.time() + execution_time = (end_time - start_time) * 1000 # Convert to milliseconds + + # Should complete within reasonable time (less than 150ms like frontend) + assert execution_time < 150, "Complex expressions should parse quickly" + + +if __name__ == "__main__": + # Import timedelta for the test + from datetime import timedelta + + unittest.main() diff --git a/api/tests/unit_tests/libs/test_datetime_utils.py b/api/tests/unit_tests/libs/test_datetime_utils.py index e914ca4816..84f5b63fbf 100644 --- a/api/tests/unit_tests/libs/test_datetime_utils.py +++ b/api/tests/unit_tests/libs/test_datetime_utils.py @@ -1,8 +1,10 @@ import datetime +from unittest.mock import patch import pytest +import pytz -from libs.datetime_utils import naive_utc_now +from libs.datetime_utils import naive_utc_now, parse_time_range def test_naive_utc_now(monkeypatch: pytest.MonkeyPatch): @@ -20,3 +22,247 @@ def test_naive_utc_now(monkeypatch: pytest.MonkeyPatch): naive_time = naive_datetime.time() utc_time = tz_aware_utc_now.time() assert naive_time == utc_time + + +class TestParseTimeRange: + """Test cases for parse_time_range function.""" + + def test_parse_time_range_basic(self): + """Test basic time range parsing.""" + start, end = parse_time_range("2024-01-01 10:00", "2024-01-01 18:00", "UTC") + + assert start is not None + assert end is not None + assert start < end + assert start.tzinfo == pytz.UTC + assert end.tzinfo == pytz.UTC + + def test_parse_time_range_start_only(self): + """Test parsing with only start time.""" + start, end = parse_time_range("2024-01-01 10:00", None, "UTC") + + assert start is not None + assert end is None + assert start.tzinfo == pytz.UTC + + def test_parse_time_range_end_only(self): + """Test parsing with only end time.""" + start, end = parse_time_range(None, "2024-01-01 18:00", "UTC") + + assert start is None + assert end is not None + assert end.tzinfo == pytz.UTC + + def test_parse_time_range_both_none(self): + """Test parsing with both times None.""" + start, end = parse_time_range(None, None, "UTC") + + assert start is None + assert end is None + + def test_parse_time_range_different_timezones(self): + """Test parsing with different timezones.""" + # Test with US/Eastern timezone + start, end = parse_time_range("2024-01-01 10:00", "2024-01-01 18:00", "US/Eastern") + + assert start is not None + assert end is not None + assert start.tzinfo == pytz.UTC + assert end.tzinfo == pytz.UTC + # Verify the times are correctly converted to UTC + assert start.hour == 15 # 10 AM EST = 3 PM UTC (in January) + assert end.hour == 23 # 6 PM EST = 11 PM UTC (in January) + + def test_parse_time_range_invalid_start_format(self): + """Test parsing with invalid start time format.""" + with pytest.raises(ValueError, match="time data.*does not match format"): + parse_time_range("invalid-date", "2024-01-01 18:00", "UTC") + + def test_parse_time_range_invalid_end_format(self): + """Test parsing with invalid end time format.""" + with pytest.raises(ValueError, match="time data.*does not match format"): + parse_time_range("2024-01-01 10:00", "invalid-date", "UTC") + + def test_parse_time_range_invalid_timezone(self): + """Test parsing with invalid timezone.""" + with pytest.raises(pytz.exceptions.UnknownTimeZoneError): + parse_time_range("2024-01-01 10:00", "2024-01-01 18:00", "Invalid/Timezone") + + def test_parse_time_range_start_after_end(self): + """Test parsing with start time after end time.""" + with pytest.raises(ValueError, match="start must be earlier than or equal to end"): + parse_time_range("2024-01-01 18:00", "2024-01-01 10:00", "UTC") + + def test_parse_time_range_start_equals_end(self): + """Test parsing with start time equal to end time.""" + start, end = parse_time_range("2024-01-01 10:00", "2024-01-01 10:00", "UTC") + + assert start is not None + assert end is not None + assert start == end + + def test_parse_time_range_dst_ambiguous_time(self): + """Test parsing during DST ambiguous time (fall back).""" + # This test simulates DST fall back where 2:30 AM occurs twice + with patch("pytz.timezone") as mock_timezone: + # Mock timezone that raises AmbiguousTimeError + mock_tz = mock_timezone.return_value + + # Create a mock datetime object for the return value + mock_dt = datetime.datetime(2024, 1, 1, 10, 0, 0) + mock_utc_dt = mock_dt.replace(tzinfo=pytz.UTC) + + # Create a proper mock for the localized datetime + from unittest.mock import MagicMock + + mock_localized_dt = MagicMock() + mock_localized_dt.astimezone.return_value = mock_utc_dt + + # Set up side effects: first call raises exception, second call succeeds + mock_tz.localize.side_effect = [ + pytz.AmbiguousTimeError("Ambiguous time"), # First call for start + mock_localized_dt, # Second call for start (with is_dst=False) + pytz.AmbiguousTimeError("Ambiguous time"), # First call for end + mock_localized_dt, # Second call for end (with is_dst=False) + ] + + start, end = parse_time_range("2024-01-01 10:00", "2024-01-01 18:00", "US/Eastern") + + # Should use is_dst=False for ambiguous times + assert mock_tz.localize.call_count == 4 # 2 calls per time (first fails, second succeeds) + assert start is not None + assert end is not None + + def test_parse_time_range_dst_nonexistent_time(self): + """Test parsing during DST nonexistent time (spring forward).""" + with patch("pytz.timezone") as mock_timezone: + # Mock timezone that raises NonExistentTimeError + mock_tz = mock_timezone.return_value + + # Create a mock datetime object for the return value + mock_dt = datetime.datetime(2024, 1, 1, 10, 0, 0) + mock_utc_dt = mock_dt.replace(tzinfo=pytz.UTC) + + # Create a proper mock for the localized datetime + from unittest.mock import MagicMock + + mock_localized_dt = MagicMock() + mock_localized_dt.astimezone.return_value = mock_utc_dt + + # Set up side effects: first call raises exception, second call succeeds + mock_tz.localize.side_effect = [ + pytz.NonExistentTimeError("Non-existent time"), # First call for start + mock_localized_dt, # Second call for start (with adjusted time) + pytz.NonExistentTimeError("Non-existent time"), # First call for end + mock_localized_dt, # Second call for end (with adjusted time) + ] + + start, end = parse_time_range("2024-01-01 10:00", "2024-01-01 18:00", "US/Eastern") + + # Should adjust time forward by 1 hour for nonexistent times + assert mock_tz.localize.call_count == 4 # 2 calls per time (first fails, second succeeds) + assert start is not None + assert end is not None + + def test_parse_time_range_edge_cases(self): + """Test edge cases for time parsing.""" + # Test with midnight times + start, end = parse_time_range("2024-01-01 00:00", "2024-01-01 23:59", "UTC") + assert start is not None + assert end is not None + assert start.hour == 0 + assert start.minute == 0 + assert end.hour == 23 + assert end.minute == 59 + + def test_parse_time_range_different_dates(self): + """Test parsing with different dates.""" + start, end = parse_time_range("2024-01-01 10:00", "2024-01-02 10:00", "UTC") + assert start is not None + assert end is not None + assert start.date() != end.date() + assert (end - start).days == 1 + + def test_parse_time_range_seconds_handling(self): + """Test that seconds are properly set to 0.""" + start, end = parse_time_range("2024-01-01 10:30", "2024-01-01 18:45", "UTC") + assert start is not None + assert end is not None + assert start.second == 0 + assert end.second == 0 + + def test_parse_time_range_timezone_conversion_accuracy(self): + """Test accurate timezone conversion.""" + # Test with a known timezone conversion + start, end = parse_time_range("2024-01-01 12:00", "2024-01-01 12:00", "Asia/Tokyo") + + assert start is not None + assert end is not None + assert start.tzinfo == pytz.UTC + assert end.tzinfo == pytz.UTC + # Tokyo is UTC+9, so 12:00 JST = 03:00 UTC + assert start.hour == 3 + assert end.hour == 3 + + def test_parse_time_range_summer_time(self): + """Test parsing during summer time (DST).""" + # Test with US/Eastern during summer (EDT = UTC-4) + start, end = parse_time_range("2024-07-01 12:00", "2024-07-01 12:00", "US/Eastern") + + assert start is not None + assert end is not None + assert start.tzinfo == pytz.UTC + assert end.tzinfo == pytz.UTC + # 12:00 EDT = 16:00 UTC + assert start.hour == 16 + assert end.hour == 16 + + def test_parse_time_range_winter_time(self): + """Test parsing during winter time (standard time).""" + # Test with US/Eastern during winter (EST = UTC-5) + start, end = parse_time_range("2024-01-01 12:00", "2024-01-01 12:00", "US/Eastern") + + assert start is not None + assert end is not None + assert start.tzinfo == pytz.UTC + assert end.tzinfo == pytz.UTC + # 12:00 EST = 17:00 UTC + assert start.hour == 17 + assert end.hour == 17 + + def test_parse_time_range_empty_strings(self): + """Test parsing with empty strings.""" + # Empty strings are treated as None, so they should not raise errors + start, end = parse_time_range("", "2024-01-01 18:00", "UTC") + assert start is None + assert end is not None + + start, end = parse_time_range("2024-01-01 10:00", "", "UTC") + assert start is not None + assert end is None + + def test_parse_time_range_malformed_datetime(self): + """Test parsing with malformed datetime strings.""" + with pytest.raises(ValueError, match="time data.*does not match format"): + parse_time_range("2024-13-01 10:00", "2024-01-01 18:00", "UTC") + + with pytest.raises(ValueError, match="time data.*does not match format"): + parse_time_range("2024-01-01 10:00", "2024-01-32 18:00", "UTC") + + def test_parse_time_range_very_long_time_range(self): + """Test parsing with very long time range.""" + start, end = parse_time_range("2020-01-01 00:00", "2030-12-31 23:59", "UTC") + + assert start is not None + assert end is not None + assert start < end + assert (end - start).days > 3000 # More than 8 years + + def test_parse_time_range_negative_timezone(self): + """Test parsing with negative timezone offset.""" + start, end = parse_time_range("2024-01-01 12:00", "2024-01-01 12:00", "America/New_York") + + assert start is not None + assert end is not None + assert start.tzinfo == pytz.UTC + assert end.tzinfo == pytz.UTC diff --git a/api/tests/unit_tests/libs/test_schedule_utils_enhanced.py b/api/tests/unit_tests/libs/test_schedule_utils_enhanced.py new file mode 100644 index 0000000000..9a14cdd0fe --- /dev/null +++ b/api/tests/unit_tests/libs/test_schedule_utils_enhanced.py @@ -0,0 +1,411 @@ +""" +Enhanced schedule_utils tests for new cron syntax support. + +These tests verify that the backend schedule_utils functions properly support +the enhanced cron syntax introduced in the frontend, ensuring full compatibility. +""" + +import unittest +from datetime import UTC, datetime, timedelta + +import pytest +import pytz +from croniter import CroniterBadCronError + +from libs.schedule_utils import calculate_next_run_at, convert_12h_to_24h + + +class TestEnhancedCronSyntax(unittest.TestCase): + """Test enhanced cron syntax in calculate_next_run_at.""" + + def setUp(self): + """Set up test with fixed time.""" + # Monday, January 15, 2024, 10:00 AM UTC + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_month_abbreviations(self): + """Test month abbreviations (JAN, FEB, etc.).""" + test_cases = [ + ("0 12 1 JAN *", 1), # January + ("0 12 1 FEB *", 2), # February + ("0 12 1 MAR *", 3), # March + ("0 12 1 APR *", 4), # April + ("0 12 1 MAY *", 5), # May + ("0 12 1 JUN *", 6), # June + ("0 12 1 JUL *", 7), # July + ("0 12 1 AUG *", 8), # August + ("0 12 1 SEP *", 9), # September + ("0 12 1 OCT *", 10), # October + ("0 12 1 NOV *", 11), # November + ("0 12 1 DEC *", 12), # December + ] + + for expr, expected_month in test_cases: + with self.subTest(expr=expr): + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None, f"Failed to parse: {expr}" + assert result.month == expected_month + assert result.day == 1 + assert result.hour == 12 + assert result.minute == 0 + + def test_weekday_abbreviations(self): + """Test weekday abbreviations (SUN, MON, etc.).""" + test_cases = [ + ("0 9 * * SUN", 6), # Sunday (weekday() = 6) + ("0 9 * * MON", 0), # Monday (weekday() = 0) + ("0 9 * * TUE", 1), # Tuesday + ("0 9 * * WED", 2), # Wednesday + ("0 9 * * THU", 3), # Thursday + ("0 9 * * FRI", 4), # Friday + ("0 9 * * SAT", 5), # Saturday + ] + + for expr, expected_weekday in test_cases: + with self.subTest(expr=expr): + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None, f"Failed to parse: {expr}" + assert result.weekday() == expected_weekday + assert result.hour == 9 + assert result.minute == 0 + + def test_sunday_dual_representation(self): + """Test Sunday as both 0 and 7.""" + base_time = datetime(2024, 1, 14, 10, 0, 0, tzinfo=UTC) # Sunday + + # Both should give the same next Sunday + result_0 = calculate_next_run_at("0 10 * * 0", "UTC", base_time) + result_7 = calculate_next_run_at("0 10 * * 7", "UTC", base_time) + result_SUN = calculate_next_run_at("0 10 * * SUN", "UTC", base_time) + + assert result_0 is not None + assert result_7 is not None + assert result_SUN is not None + + # All should be Sundays + assert result_0.weekday() == 6 # Sunday = 6 in weekday() + assert result_7.weekday() == 6 + assert result_SUN.weekday() == 6 + + # Times should be identical + assert result_0 == result_7 + assert result_0 == result_SUN + + def test_predefined_expressions(self): + """Test predefined expressions (@daily, @weekly, etc.).""" + test_cases = [ + ("@yearly", lambda dt: dt.month == 1 and dt.day == 1 and dt.hour == 0 and dt.minute == 0), + ("@annually", lambda dt: dt.month == 1 and dt.day == 1 and dt.hour == 0 and dt.minute == 0), + ("@monthly", lambda dt: dt.day == 1 and dt.hour == 0 and dt.minute == 0), + ("@weekly", lambda dt: dt.weekday() == 6 and dt.hour == 0 and dt.minute == 0), # Sunday + ("@daily", lambda dt: dt.hour == 0 and dt.minute == 0), + ("@midnight", lambda dt: dt.hour == 0 and dt.minute == 0), + ("@hourly", lambda dt: dt.minute == 0), + ] + + for expr, validator in test_cases: + with self.subTest(expr=expr): + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None, f"Failed to parse: {expr}" + assert validator(result), f"Validator failed for {expr}: {result}" + + def test_question_mark_wildcard(self): + """Test ? wildcard character.""" + # ? in day position with specific weekday + result_question = calculate_next_run_at("0 9 ? * 1", "UTC", self.base_time) # Monday + result_star = calculate_next_run_at("0 9 * * 1", "UTC", self.base_time) # Monday + + assert result_question is not None + assert result_star is not None + + # Both should return Mondays at 9:00 + assert result_question.weekday() == 0 # Monday + assert result_star.weekday() == 0 + assert result_question.hour == 9 + assert result_star.hour == 9 + + # Results should be identical + assert result_question == result_star + + def test_last_day_of_month(self): + """Test 'L' for last day of month.""" + expr = "0 12 L * *" # Last day of month at noon + + # Test for February (28 days in 2024 - not a leap year check) + feb_base = datetime(2024, 2, 15, 10, 0, 0, tzinfo=UTC) + result = calculate_next_run_at(expr, "UTC", feb_base) + assert result is not None + assert result.month == 2 + assert result.day == 29 # 2024 is a leap year + assert result.hour == 12 + + def test_range_with_abbreviations(self): + """Test ranges using abbreviations.""" + test_cases = [ + "0 9 * * MON-FRI", # Weekday range + "0 12 * JAN-MAR *", # Q1 months + "0 15 * APR-JUN *", # Q2 months + ] + + for expr in test_cases: + with self.subTest(expr=expr): + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None, f"Failed to parse range expression: {expr}" + assert result > self.base_time + + def test_list_with_abbreviations(self): + """Test lists using abbreviations.""" + test_cases = [ + ("0 9 * * SUN,WED,FRI", [6, 2, 4]), # Specific weekdays + ("0 12 1 JAN,JUN,DEC *", [1, 6, 12]), # Specific months + ] + + for expr, expected_values in test_cases: + with self.subTest(expr=expr): + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None, f"Failed to parse list expression: {expr}" + + if "* *" in expr: # Weekday test + assert result.weekday() in expected_values + else: # Month test + assert result.month in expected_values + + def test_mixed_syntax(self): + """Test mixed traditional and enhanced syntax.""" + test_cases = [ + "30 14 15 JAN,JUN,DEC *", # Numbers + month abbreviations + "0 9 * JAN-MAR MON-FRI", # Month range + weekday range + "45 8 1,15 * MON", # Numbers + weekday abbreviation + ] + + for expr in test_cases: + with self.subTest(expr=expr): + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None, f"Failed to parse mixed syntax: {expr}" + assert result > self.base_time + + def test_complex_enhanced_expressions(self): + """Test complex expressions with multiple enhanced features.""" + # Note: Some of these might not be supported by croniter, that's OK + complex_expressions = [ + "0 9 L JAN *", # Last day of January + "30 14 * * FRI#1", # First Friday of month (if supported) + "0 12 15 JAN-DEC/3 *", # 15th of every 3rd month (quarterly) + ] + + for expr in complex_expressions: + with self.subTest(expr=expr): + try: + result = calculate_next_run_at(expr, "UTC", self.base_time) + if result: # If supported, should return valid result + assert result > self.base_time + except Exception: + # Some complex expressions might not be supported - that's acceptable + pass + + +class TestTimezoneHandlingEnhanced(unittest.TestCase): + """Test timezone handling with enhanced syntax.""" + + def setUp(self): + """Set up test with fixed time.""" + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_enhanced_syntax_with_timezones(self): + """Test enhanced syntax works correctly across timezones.""" + timezones = ["UTC", "America/New_York", "Asia/Tokyo", "Europe/London"] + expression = "0 12 * * MON" # Monday at noon + + for timezone in timezones: + with self.subTest(timezone=timezone): + result = calculate_next_run_at(expression, timezone, self.base_time) + assert result is not None + + # Convert to local timezone to verify it's Monday at noon + tz = pytz.timezone(timezone) + local_time = result.astimezone(tz) + assert local_time.weekday() == 0 # Monday + assert local_time.hour == 12 + assert local_time.minute == 0 + + def test_predefined_expressions_with_timezones(self): + """Test predefined expressions work with different timezones.""" + expression = "@daily" + timezones = ["UTC", "America/New_York", "Asia/Tokyo"] + + for timezone in timezones: + with self.subTest(timezone=timezone): + result = calculate_next_run_at(expression, timezone, self.base_time) + assert result is not None + + # Should be midnight in the specified timezone + tz = pytz.timezone(timezone) + local_time = result.astimezone(tz) + assert local_time.hour == 0 + assert local_time.minute == 0 + + def test_dst_with_enhanced_syntax(self): + """Test DST handling with enhanced syntax.""" + # DST spring forward date in 2024 + dst_base = datetime(2024, 3, 8, 10, 0, 0, tzinfo=UTC) + expression = "0 2 * * SUN" # Sunday at 2 AM (problematic during DST) + timezone = "America/New_York" + + result = calculate_next_run_at(expression, timezone, dst_base) + assert result is not None + + # Should handle DST transition gracefully + tz = pytz.timezone(timezone) + local_time = result.astimezone(tz) + assert local_time.weekday() == 6 # Sunday + + # During DST spring forward, 2 AM might become 3 AM + assert local_time.hour in [2, 3] + + +class TestErrorHandlingEnhanced(unittest.TestCase): + """Test error handling for enhanced syntax.""" + + def setUp(self): + """Set up test with fixed time.""" + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_invalid_enhanced_syntax(self): + """Test that invalid enhanced syntax raises appropriate errors.""" + invalid_expressions = [ + "0 12 * JANUARY *", # Full month name + "0 12 * * MONDAY", # Full day name + "0 12 32 JAN *", # Invalid day with valid month + "0 12 * * MON-SUN-FRI", # Invalid range syntax + "0 12 * JAN- *", # Incomplete range + "0 12 * * ,MON", # Invalid list syntax + "@INVALID", # Invalid predefined + ] + + for expr in invalid_expressions: + with self.subTest(expr=expr): + with pytest.raises((CroniterBadCronError, ValueError)): + calculate_next_run_at(expr, "UTC", self.base_time) + + def test_boundary_values_with_enhanced_syntax(self): + """Test boundary values work with enhanced syntax.""" + # Valid boundary expressions + valid_expressions = [ + "0 0 1 JAN *", # Minimum: January 1st midnight + "59 23 31 DEC *", # Maximum: December 31st 23:59 + "0 12 29 FEB *", # Leap year boundary + ] + + for expr in valid_expressions: + with self.subTest(expr=expr): + try: + result = calculate_next_run_at(expr, "UTC", self.base_time) + if result: # Some dates might not occur soon + assert result > self.base_time + except Exception as e: + # Some boundary cases might be complex to calculate + self.fail(f"Valid boundary expression failed: {expr} - {e}") + + +class TestPerformanceEnhanced(unittest.TestCase): + """Test performance with enhanced syntax.""" + + def setUp(self): + """Set up test with fixed time.""" + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_complex_expression_performance(self): + """Test that complex enhanced expressions parse within reasonable time.""" + import time + + complex_expressions = [ + "*/5 9-17 * * MON-FRI", # Every 5 min, weekdays, business hours + "0 9 * JAN-MAR MON-FRI", # Q1 weekdays at 9 AM + "30 14 1,15 * * ", # 1st and 15th at 14:30 + "0 12 ? * SUN", # Sundays at noon with ? + "@daily", # Predefined expression + ] + + start_time = time.time() + + for expr in complex_expressions: + with self.subTest(expr=expr): + try: + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None + except Exception: + # Some expressions might not be supported - acceptable + pass + + end_time = time.time() + execution_time = (end_time - start_time) * 1000 # milliseconds + + # Should be fast (less than 100ms for all expressions) + assert execution_time < 100, "Enhanced expressions should parse quickly" + + def test_multiple_calculations_performance(self): + """Test performance when calculating multiple next times.""" + import time + + expression = "0 9 * * MON-FRI" # Weekdays at 9 AM + iterations = 20 + + start_time = time.time() + + current_time = self.base_time + for _ in range(iterations): + result = calculate_next_run_at(expression, "UTC", current_time) + assert result is not None + current_time = result + timedelta(seconds=1) # Move forward slightly + + end_time = time.time() + total_time = (end_time - start_time) * 1000 # milliseconds + avg_time = total_time / iterations + + # Average should be very fast (less than 5ms per calculation) + assert avg_time < 5, f"Average calculation time too slow: {avg_time}ms" + + +class TestRegressionEnhanced(unittest.TestCase): + """Regression tests to ensure enhanced syntax doesn't break existing functionality.""" + + def setUp(self): + """Set up test with fixed time.""" + self.base_time = datetime(2024, 1, 15, 10, 0, 0, tzinfo=UTC) + + def test_traditional_syntax_still_works(self): + """Ensure traditional cron syntax continues to work.""" + traditional_expressions = [ + "15 10 1 * *", # Monthly 1st at 10:15 + "0 0 * * 0", # Weekly Sunday midnight + "*/5 * * * *", # Every 5 minutes + "0 9-17 * * 1-5", # Business hours weekdays + "30 14 * * 1", # Monday 14:30 + "0 0 1,15 * *", # 1st and 15th midnight + ] + + for expr in traditional_expressions: + with self.subTest(expr=expr): + result = calculate_next_run_at(expr, "UTC", self.base_time) + assert result is not None, f"Traditional expression failed: {expr}" + assert result > self.base_time + + def test_convert_12h_to_24h_unchanged(self): + """Ensure convert_12h_to_24h function is unchanged.""" + test_cases = [ + ("12:00 AM", (0, 0)), # Midnight + ("12:00 PM", (12, 0)), # Noon + ("1:30 AM", (1, 30)), # Early morning + ("11:45 PM", (23, 45)), # Late evening + ("6:15 AM", (6, 15)), # Morning + ("3:30 PM", (15, 30)), # Afternoon + ] + + for time_str, expected in test_cases: + with self.subTest(time_str=time_str): + result = convert_12h_to_24h(time_str) + assert result == expected, f"12h conversion failed: {time_str}" + + +if __name__ == "__main__": + unittest.main() diff --git a/api/tests/unit_tests/libs/test_token.py b/api/tests/unit_tests/libs/test_token.py index 22790fa4a6..6a65b5faa0 100644 --- a/api/tests/unit_tests/libs/test_token.py +++ b/api/tests/unit_tests/libs/test_token.py @@ -1,5 +1,10 @@ -from constants import COOKIE_NAME_ACCESS_TOKEN -from libs.token import extract_access_token +from unittest.mock import MagicMock + +from werkzeug.wrappers import Response + +from constants import COOKIE_NAME_ACCESS_TOKEN, COOKIE_NAME_WEBAPP_ACCESS_TOKEN +from libs import token +from libs.token import extract_access_token, extract_webapp_access_token, set_csrf_token_to_cookie class MockRequest: @@ -14,10 +19,44 @@ def test_extract_access_token(): return MockRequest(headers, cookies, args) test_cases = [ - (_mock_request({"Authorization": "Bearer 123"}, {}, {}), "123"), - (_mock_request({}, {COOKIE_NAME_ACCESS_TOKEN: "123"}, {}), "123"), - (_mock_request({}, {}, {}), None), - (_mock_request({"Authorization": "Bearer_aaa 123"}, {}, {}), None), + (_mock_request({"Authorization": "Bearer 123"}, {}, {}), "123", "123"), + (_mock_request({}, {COOKIE_NAME_ACCESS_TOKEN: "123"}, {}), "123", None), + (_mock_request({}, {}, {}), None, None), + (_mock_request({"Authorization": "Bearer_aaa 123"}, {}, {}), None, None), + (_mock_request({}, {COOKIE_NAME_WEBAPP_ACCESS_TOKEN: "123"}, {}), None, "123"), ] - for request, expected in test_cases: - assert extract_access_token(request) == expected # pyright: ignore[reportArgumentType] + for request, expected_console, expected_webapp in test_cases: + assert extract_access_token(request) == expected_console # pyright: ignore[reportArgumentType] + assert extract_webapp_access_token(request) == expected_webapp # pyright: ignore[reportArgumentType] + + +def test_real_cookie_name_uses_host_prefix_without_domain(monkeypatch): + monkeypatch.setattr(token.dify_config, "CONSOLE_WEB_URL", "https://console.example.com", raising=False) + monkeypatch.setattr(token.dify_config, "CONSOLE_API_URL", "https://api.example.com", raising=False) + monkeypatch.setattr(token.dify_config, "COOKIE_DOMAIN", "", raising=False) + + assert token._real_cookie_name("csrf_token") == "__Host-csrf_token" + + +def test_real_cookie_name_without_host_prefix_when_domain_present(monkeypatch): + monkeypatch.setattr(token.dify_config, "CONSOLE_WEB_URL", "https://console.example.com", raising=False) + monkeypatch.setattr(token.dify_config, "CONSOLE_API_URL", "https://api.example.com", raising=False) + monkeypatch.setattr(token.dify_config, "COOKIE_DOMAIN", ".example.com", raising=False) + + assert token._real_cookie_name("csrf_token") == "csrf_token" + + +def test_set_csrf_cookie_includes_domain_when_configured(monkeypatch): + monkeypatch.setattr(token.dify_config, "CONSOLE_WEB_URL", "https://console.example.com", raising=False) + monkeypatch.setattr(token.dify_config, "CONSOLE_API_URL", "https://api.example.com", raising=False) + monkeypatch.setattr(token.dify_config, "COOKIE_DOMAIN", ".example.com", raising=False) + + response = Response() + request = MagicMock() + + set_csrf_token_to_cookie(request, response, "abc123") + + cookies = response.headers.getlist("Set-Cookie") + assert any("csrf_token=abc123" in c for c in cookies) + assert any("Domain=example.com" in c for c in cookies) + assert all("__Host-" not in c for c in cookies) diff --git a/api/tests/unit_tests/models/test_base.py b/api/tests/unit_tests/models/test_base.py new file mode 100644 index 0000000000..e0dda3c1dd --- /dev/null +++ b/api/tests/unit_tests/models/test_base.py @@ -0,0 +1,11 @@ +from models.base import DefaultFieldsMixin + + +class FooModel(DefaultFieldsMixin): + def __init__(self, id: str): + self.id = id + + +def test_repr(): + foo_model = FooModel(id="test-id") + assert repr(foo_model) == "" diff --git a/api/tests/unit_tests/models/test_plugin_entities.py b/api/tests/unit_tests/models/test_plugin_entities.py new file mode 100644 index 0000000000..0c61144deb --- /dev/null +++ b/api/tests/unit_tests/models/test_plugin_entities.py @@ -0,0 +1,22 @@ +import binascii +from collections.abc import Mapping +from typing import Any + +from core.plugin.entities.request import TriggerDispatchResponse + + +def test_trigger_dispatch_response(): + raw_http_response = b'HTTP/1.1 200 OK\r\nContent-Type: application/json\r\n\r\n{"message": "Hello, world!"}' + + data: Mapping[str, Any] = { + "user_id": "123", + "events": ["event1", "event2"], + "response": binascii.hexlify(raw_http_response).decode(), + "payload": {"key": "value"}, + } + + response = TriggerDispatchResponse(**data) + + assert response.response.status_code == 200 + assert response.response.headers["Content-Type"] == "application/json" + assert response.response.get_data(as_text=True) == '{"message": "Hello, world!"}' diff --git a/api/tests/unit_tests/repositories/test_sqlalchemy_api_workflow_run_repository.py b/api/tests/unit_tests/repositories/test_sqlalchemy_api_workflow_run_repository.py new file mode 100644 index 0000000000..73b35b8e63 --- /dev/null +++ b/api/tests/unit_tests/repositories/test_sqlalchemy_api_workflow_run_repository.py @@ -0,0 +1,370 @@ +"""Unit tests for DifyAPISQLAlchemyWorkflowRunRepository implementation.""" + +from datetime import UTC, datetime +from unittest.mock import Mock, patch + +import pytest +from sqlalchemy.orm import Session, sessionmaker + +from core.workflow.entities.workflow_pause import WorkflowPauseEntity +from core.workflow.enums import WorkflowExecutionStatus +from models.workflow import WorkflowPause as WorkflowPauseModel +from models.workflow import WorkflowRun +from repositories.sqlalchemy_api_workflow_run_repository import ( + DifyAPISQLAlchemyWorkflowRunRepository, + _PrivateWorkflowPauseEntity, + _WorkflowRunError, +) + + +class TestDifyAPISQLAlchemyWorkflowRunRepository: + """Test DifyAPISQLAlchemyWorkflowRunRepository implementation.""" + + @pytest.fixture + def mock_session(self): + """Create a mock session.""" + return Mock(spec=Session) + + @pytest.fixture + def mock_session_maker(self, mock_session): + """Create a mock sessionmaker.""" + session_maker = Mock(spec=sessionmaker) + + # Create a context manager mock + context_manager = Mock() + context_manager.__enter__ = Mock(return_value=mock_session) + context_manager.__exit__ = Mock(return_value=None) + session_maker.return_value = context_manager + + # Mock session.begin() context manager + begin_context_manager = Mock() + begin_context_manager.__enter__ = Mock(return_value=None) + begin_context_manager.__exit__ = Mock(return_value=None) + mock_session.begin = Mock(return_value=begin_context_manager) + + # Add missing session methods + mock_session.commit = Mock() + mock_session.rollback = Mock() + mock_session.add = Mock() + mock_session.delete = Mock() + mock_session.get = Mock() + mock_session.scalar = Mock() + mock_session.scalars = Mock() + + # Also support expire_on_commit parameter + def make_session(expire_on_commit=None): + cm = Mock() + cm.__enter__ = Mock(return_value=mock_session) + cm.__exit__ = Mock(return_value=None) + return cm + + session_maker.side_effect = make_session + return session_maker + + @pytest.fixture + def repository(self, mock_session_maker): + """Create repository instance with mocked dependencies.""" + + # Create a testable subclass that implements the save method + class TestableDifyAPISQLAlchemyWorkflowRunRepository(DifyAPISQLAlchemyWorkflowRunRepository): + def __init__(self, session_maker): + # Initialize without calling parent __init__ to avoid any instantiation issues + self._session_maker = session_maker + + def save(self, execution): + """Mock implementation of save method.""" + return None + + # Create repository instance + repo = TestableDifyAPISQLAlchemyWorkflowRunRepository(mock_session_maker) + + return repo + + @pytest.fixture + def sample_workflow_run(self): + """Create a sample WorkflowRun model.""" + workflow_run = Mock(spec=WorkflowRun) + workflow_run.id = "workflow-run-123" + workflow_run.tenant_id = "tenant-123" + workflow_run.app_id = "app-123" + workflow_run.workflow_id = "workflow-123" + workflow_run.status = WorkflowExecutionStatus.RUNNING + return workflow_run + + @pytest.fixture + def sample_workflow_pause(self): + """Create a sample WorkflowPauseModel.""" + pause = Mock(spec=WorkflowPauseModel) + pause.id = "pause-123" + pause.workflow_id = "workflow-123" + pause.workflow_run_id = "workflow-run-123" + pause.state_object_key = "workflow-state-123.json" + pause.resumed_at = None + pause.created_at = datetime.now(UTC) + return pause + + +class TestCreateWorkflowPause(TestDifyAPISQLAlchemyWorkflowRunRepository): + """Test create_workflow_pause method.""" + + def test_create_workflow_pause_success( + self, + repository: DifyAPISQLAlchemyWorkflowRunRepository, + mock_session: Mock, + sample_workflow_run: Mock, + ): + """Test successful workflow pause creation.""" + # Arrange + workflow_run_id = "workflow-run-123" + state_owner_user_id = "user-123" + state = '{"test": "state"}' + + mock_session.get.return_value = sample_workflow_run + + with patch("repositories.sqlalchemy_api_workflow_run_repository.uuidv7") as mock_uuidv7: + mock_uuidv7.side_effect = ["pause-123"] + with patch("repositories.sqlalchemy_api_workflow_run_repository.storage") as mock_storage: + # Act + result = repository.create_workflow_pause( + workflow_run_id=workflow_run_id, + state_owner_user_id=state_owner_user_id, + state=state, + ) + + # Assert + assert isinstance(result, _PrivateWorkflowPauseEntity) + assert result.id == "pause-123" + assert result.workflow_execution_id == workflow_run_id + + # Verify database interactions + mock_session.get.assert_called_once_with(WorkflowRun, workflow_run_id) + mock_storage.save.assert_called_once() + mock_session.add.assert_called() + # When using session.begin() context manager, commit is handled automatically + # No explicit commit call is expected + + def test_create_workflow_pause_not_found( + self, repository: DifyAPISQLAlchemyWorkflowRunRepository, mock_session: Mock + ): + """Test workflow pause creation when workflow run not found.""" + # Arrange + mock_session.get.return_value = None + + # Act & Assert + with pytest.raises(ValueError, match="WorkflowRun not found: workflow-run-123"): + repository.create_workflow_pause( + workflow_run_id="workflow-run-123", + state_owner_user_id="user-123", + state='{"test": "state"}', + ) + + mock_session.get.assert_called_once_with(WorkflowRun, "workflow-run-123") + + def test_create_workflow_pause_invalid_status( + self, repository: DifyAPISQLAlchemyWorkflowRunRepository, mock_session: Mock, sample_workflow_run: Mock + ): + """Test workflow pause creation when workflow not in RUNNING status.""" + # Arrange + sample_workflow_run.status = WorkflowExecutionStatus.PAUSED + mock_session.get.return_value = sample_workflow_run + + # Act & Assert + with pytest.raises(_WorkflowRunError, match="Only WorkflowRun with RUNNING status can be paused"): + repository.create_workflow_pause( + workflow_run_id="workflow-run-123", + state_owner_user_id="user-123", + state='{"test": "state"}', + ) + + +class TestResumeWorkflowPause(TestDifyAPISQLAlchemyWorkflowRunRepository): + """Test resume_workflow_pause method.""" + + def test_resume_workflow_pause_success( + self, + repository: DifyAPISQLAlchemyWorkflowRunRepository, + mock_session: Mock, + sample_workflow_run: Mock, + sample_workflow_pause: Mock, + ): + """Test successful workflow pause resume.""" + # Arrange + workflow_run_id = "workflow-run-123" + pause_entity = Mock(spec=WorkflowPauseEntity) + pause_entity.id = "pause-123" + + # Setup workflow run and pause + sample_workflow_run.status = WorkflowExecutionStatus.PAUSED + sample_workflow_run.pause = sample_workflow_pause + sample_workflow_pause.resumed_at = None + + mock_session.scalar.return_value = sample_workflow_run + + with patch("repositories.sqlalchemy_api_workflow_run_repository.naive_utc_now") as mock_now: + mock_now.return_value = datetime.now(UTC) + + # Act + result = repository.resume_workflow_pause( + workflow_run_id=workflow_run_id, + pause_entity=pause_entity, + ) + + # Assert + assert isinstance(result, _PrivateWorkflowPauseEntity) + assert result.id == "pause-123" + + # Verify state transitions + assert sample_workflow_pause.resumed_at is not None + assert sample_workflow_run.status == WorkflowExecutionStatus.RUNNING + + # Verify database interactions + mock_session.add.assert_called() + # When using session.begin() context manager, commit is handled automatically + # No explicit commit call is expected + + def test_resume_workflow_pause_not_paused( + self, + repository: DifyAPISQLAlchemyWorkflowRunRepository, + mock_session: Mock, + sample_workflow_run: Mock, + ): + """Test resume when workflow is not paused.""" + # Arrange + workflow_run_id = "workflow-run-123" + pause_entity = Mock(spec=WorkflowPauseEntity) + pause_entity.id = "pause-123" + + sample_workflow_run.status = WorkflowExecutionStatus.RUNNING + mock_session.scalar.return_value = sample_workflow_run + + # Act & Assert + with pytest.raises(_WorkflowRunError, match="WorkflowRun is not in PAUSED status"): + repository.resume_workflow_pause( + workflow_run_id=workflow_run_id, + pause_entity=pause_entity, + ) + + def test_resume_workflow_pause_id_mismatch( + self, + repository: DifyAPISQLAlchemyWorkflowRunRepository, + mock_session: Mock, + sample_workflow_run: Mock, + sample_workflow_pause: Mock, + ): + """Test resume when pause ID doesn't match.""" + # Arrange + workflow_run_id = "workflow-run-123" + pause_entity = Mock(spec=WorkflowPauseEntity) + pause_entity.id = "pause-456" # Different ID + + sample_workflow_run.status = WorkflowExecutionStatus.PAUSED + sample_workflow_pause.id = "pause-123" + sample_workflow_run.pause = sample_workflow_pause + mock_session.scalar.return_value = sample_workflow_run + + # Act & Assert + with pytest.raises(_WorkflowRunError, match="different id in WorkflowPause and WorkflowPauseEntity"): + repository.resume_workflow_pause( + workflow_run_id=workflow_run_id, + pause_entity=pause_entity, + ) + + +class TestDeleteWorkflowPause(TestDifyAPISQLAlchemyWorkflowRunRepository): + """Test delete_workflow_pause method.""" + + def test_delete_workflow_pause_success( + self, + repository: DifyAPISQLAlchemyWorkflowRunRepository, + mock_session: Mock, + sample_workflow_pause: Mock, + ): + """Test successful workflow pause deletion.""" + # Arrange + pause_entity = Mock(spec=WorkflowPauseEntity) + pause_entity.id = "pause-123" + + mock_session.get.return_value = sample_workflow_pause + + with patch("repositories.sqlalchemy_api_workflow_run_repository.storage") as mock_storage: + # Act + repository.delete_workflow_pause(pause_entity=pause_entity) + + # Assert + mock_storage.delete.assert_called_once_with(sample_workflow_pause.state_object_key) + mock_session.delete.assert_called_once_with(sample_workflow_pause) + # When using session.begin() context manager, commit is handled automatically + # No explicit commit call is expected + + def test_delete_workflow_pause_not_found( + self, + repository: DifyAPISQLAlchemyWorkflowRunRepository, + mock_session: Mock, + ): + """Test delete when pause not found.""" + # Arrange + pause_entity = Mock(spec=WorkflowPauseEntity) + pause_entity.id = "pause-123" + + mock_session.get.return_value = None + + # Act & Assert + with pytest.raises(_WorkflowRunError, match="WorkflowPause not found: pause-123"): + repository.delete_workflow_pause(pause_entity=pause_entity) + + +class TestPrivateWorkflowPauseEntity(TestDifyAPISQLAlchemyWorkflowRunRepository): + """Test _PrivateWorkflowPauseEntity class.""" + + def test_from_models(self, sample_workflow_pause: Mock): + """Test creating _PrivateWorkflowPauseEntity from models.""" + # Act + entity = _PrivateWorkflowPauseEntity.from_models(sample_workflow_pause) + + # Assert + assert isinstance(entity, _PrivateWorkflowPauseEntity) + assert entity._pause_model == sample_workflow_pause + + def test_properties(self, sample_workflow_pause: Mock): + """Test entity properties.""" + # Arrange + entity = _PrivateWorkflowPauseEntity.from_models(sample_workflow_pause) + + # Act & Assert + assert entity.id == sample_workflow_pause.id + assert entity.workflow_execution_id == sample_workflow_pause.workflow_run_id + assert entity.resumed_at == sample_workflow_pause.resumed_at + + def test_get_state(self, sample_workflow_pause: Mock): + """Test getting state from storage.""" + # Arrange + entity = _PrivateWorkflowPauseEntity.from_models(sample_workflow_pause) + expected_state = b'{"test": "state"}' + + with patch("repositories.sqlalchemy_api_workflow_run_repository.storage") as mock_storage: + mock_storage.load.return_value = expected_state + + # Act + result = entity.get_state() + + # Assert + assert result == expected_state + mock_storage.load.assert_called_once_with(sample_workflow_pause.state_object_key) + + def test_get_state_caching(self, sample_workflow_pause: Mock): + """Test state caching in get_state method.""" + # Arrange + entity = _PrivateWorkflowPauseEntity.from_models(sample_workflow_pause) + expected_state = b'{"test": "state"}' + + with patch("repositories.sqlalchemy_api_workflow_run_repository.storage") as mock_storage: + mock_storage.load.return_value = expected_state + + # Act + result1 = entity.get_state() + result2 = entity.get_state() # Should use cache + + # Assert + assert result1 == expected_state + assert result2 == expected_state + mock_storage.load.assert_called_once() # Only called once due to caching diff --git a/api/tests/unit_tests/services/test_document_indexing_task_proxy.py b/api/tests/unit_tests/services/test_document_indexing_task_proxy.py new file mode 100644 index 0000000000..d9183be9fb --- /dev/null +++ b/api/tests/unit_tests/services/test_document_indexing_task_proxy.py @@ -0,0 +1,317 @@ +from unittest.mock import Mock, patch + +from core.entities.document_task import DocumentTask +from core.rag.pipeline.queue import TenantIsolatedTaskQueue +from enums.cloud_plan import CloudPlan +from services.document_indexing_task_proxy import DocumentIndexingTaskProxy + + +class DocumentIndexingTaskProxyTestDataFactory: + """Factory class for creating test data and mock objects for DocumentIndexingTaskProxy tests.""" + + @staticmethod + def create_mock_features(billing_enabled: bool = False, plan: CloudPlan = CloudPlan.SANDBOX) -> Mock: + """Create mock features with billing configuration.""" + features = Mock() + features.billing = Mock() + features.billing.enabled = billing_enabled + features.billing.subscription = Mock() + features.billing.subscription.plan = plan + return features + + @staticmethod + def create_mock_tenant_queue(has_task_key: bool = False) -> Mock: + """Create mock TenantIsolatedTaskQueue.""" + queue = Mock(spec=TenantIsolatedTaskQueue) + queue.get_task_key.return_value = "task_key" if has_task_key else None + queue.push_tasks = Mock() + queue.set_task_waiting_time = Mock() + return queue + + @staticmethod + def create_document_task_proxy( + tenant_id: str = "tenant-123", dataset_id: str = "dataset-456", document_ids: list[str] | None = None + ) -> DocumentIndexingTaskProxy: + """Create DocumentIndexingTaskProxy instance for testing.""" + if document_ids is None: + document_ids = ["doc-1", "doc-2", "doc-3"] + return DocumentIndexingTaskProxy(tenant_id, dataset_id, document_ids) + + +class TestDocumentIndexingTaskProxy: + """Test cases for DocumentIndexingTaskProxy class.""" + + def test_initialization(self): + """Test DocumentIndexingTaskProxy initialization.""" + # Arrange + tenant_id = "tenant-123" + dataset_id = "dataset-456" + document_ids = ["doc-1", "doc-2", "doc-3"] + + # Act + proxy = DocumentIndexingTaskProxy(tenant_id, dataset_id, document_ids) + + # Assert + assert proxy._tenant_id == tenant_id + assert proxy._dataset_id == dataset_id + assert proxy._document_ids == document_ids + assert isinstance(proxy._tenant_isolated_task_queue, TenantIsolatedTaskQueue) + assert proxy._tenant_isolated_task_queue._tenant_id == tenant_id + assert proxy._tenant_isolated_task_queue._unique_key == "document_indexing" + + @patch("services.document_indexing_task_proxy.FeatureService") + def test_features_property(self, mock_feature_service): + """Test cached_property features.""" + # Arrange + mock_features = DocumentIndexingTaskProxyTestDataFactory.create_mock_features() + mock_feature_service.get_features.return_value = mock_features + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + + # Act + features1 = proxy.features + features2 = proxy.features # Second call should use cached property + + # Assert + assert features1 == mock_features + assert features2 == mock_features + assert features1 is features2 # Should be the same instance due to caching + mock_feature_service.get_features.assert_called_once_with("tenant-123") + + @patch("services.document_indexing_task_proxy.normal_document_indexing_task") + def test_send_to_direct_queue(self, mock_task): + """Test _send_to_direct_queue method.""" + # Arrange + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + mock_task.delay = Mock() + + # Act + proxy._send_to_direct_queue(mock_task) + + # Assert + mock_task.delay.assert_called_once_with( + tenant_id="tenant-123", dataset_id="dataset-456", document_ids=["doc-1", "doc-2", "doc-3"] + ) + + @patch("services.document_indexing_task_proxy.normal_document_indexing_task") + def test_send_to_tenant_queue_with_existing_task_key(self, mock_task): + """Test _send_to_tenant_queue when task key exists.""" + # Arrange + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._tenant_isolated_task_queue = DocumentIndexingTaskProxyTestDataFactory.create_mock_tenant_queue( + has_task_key=True + ) + mock_task.delay = Mock() + + # Act + proxy._send_to_tenant_queue(mock_task) + + # Assert + proxy._tenant_isolated_task_queue.push_tasks.assert_called_once() + pushed_tasks = proxy._tenant_isolated_task_queue.push_tasks.call_args[0][0] + assert len(pushed_tasks) == 1 + assert isinstance(DocumentTask(**pushed_tasks[0]), DocumentTask) + assert pushed_tasks[0]["tenant_id"] == "tenant-123" + assert pushed_tasks[0]["dataset_id"] == "dataset-456" + assert pushed_tasks[0]["document_ids"] == ["doc-1", "doc-2", "doc-3"] + mock_task.delay.assert_not_called() + + @patch("services.document_indexing_task_proxy.normal_document_indexing_task") + def test_send_to_tenant_queue_without_task_key(self, mock_task): + """Test _send_to_tenant_queue when no task key exists.""" + # Arrange + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._tenant_isolated_task_queue = DocumentIndexingTaskProxyTestDataFactory.create_mock_tenant_queue( + has_task_key=False + ) + mock_task.delay = Mock() + + # Act + proxy._send_to_tenant_queue(mock_task) + + # Assert + proxy._tenant_isolated_task_queue.set_task_waiting_time.assert_called_once() + mock_task.delay.assert_called_once_with( + tenant_id="tenant-123", dataset_id="dataset-456", document_ids=["doc-1", "doc-2", "doc-3"] + ) + proxy._tenant_isolated_task_queue.push_tasks.assert_not_called() + + @patch("services.document_indexing_task_proxy.normal_document_indexing_task") + def test_send_to_default_tenant_queue(self, mock_task): + """Test _send_to_default_tenant_queue method.""" + # Arrange + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_tenant_queue = Mock() + + # Act + proxy._send_to_default_tenant_queue() + + # Assert + proxy._send_to_tenant_queue.assert_called_once_with(mock_task) + + @patch("services.document_indexing_task_proxy.priority_document_indexing_task") + def test_send_to_priority_tenant_queue(self, mock_task): + """Test _send_to_priority_tenant_queue method.""" + # Arrange + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_tenant_queue = Mock() + + # Act + proxy._send_to_priority_tenant_queue() + + # Assert + proxy._send_to_tenant_queue.assert_called_once_with(mock_task) + + @patch("services.document_indexing_task_proxy.priority_document_indexing_task") + def test_send_to_priority_direct_queue(self, mock_task): + """Test _send_to_priority_direct_queue method.""" + # Arrange + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_direct_queue = Mock() + + # Act + proxy._send_to_priority_direct_queue() + + # Assert + proxy._send_to_direct_queue.assert_called_once_with(mock_task) + + @patch("services.document_indexing_task_proxy.FeatureService") + def test_dispatch_with_billing_enabled_sandbox_plan(self, mock_feature_service): + """Test _dispatch method when billing is enabled with sandbox plan.""" + # Arrange + mock_features = DocumentIndexingTaskProxyTestDataFactory.create_mock_features( + billing_enabled=True, plan=CloudPlan.SANDBOX + ) + mock_feature_service.get_features.return_value = mock_features + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_default_tenant_queue = Mock() + + # Act + proxy._dispatch() + + # Assert + proxy._send_to_default_tenant_queue.assert_called_once() + + @patch("services.document_indexing_task_proxy.FeatureService") + def test_dispatch_with_billing_enabled_non_sandbox_plan(self, mock_feature_service): + """Test _dispatch method when billing is enabled with non-sandbox plan.""" + # Arrange + mock_features = DocumentIndexingTaskProxyTestDataFactory.create_mock_features( + billing_enabled=True, plan=CloudPlan.TEAM + ) + mock_feature_service.get_features.return_value = mock_features + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_priority_tenant_queue = Mock() + + # Act + proxy._dispatch() + + # If billing enabled with non sandbox plan, should send to priority tenant queue + proxy._send_to_priority_tenant_queue.assert_called_once() + + @patch("services.document_indexing_task_proxy.FeatureService") + def test_dispatch_with_billing_disabled(self, mock_feature_service): + """Test _dispatch method when billing is disabled.""" + # Arrange + mock_features = DocumentIndexingTaskProxyTestDataFactory.create_mock_features(billing_enabled=False) + mock_feature_service.get_features.return_value = mock_features + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_priority_direct_queue = Mock() + + # Act + proxy._dispatch() + + # If billing disabled, for example: self-hosted or enterprise, should send to priority direct queue + proxy._send_to_priority_direct_queue.assert_called_once() + + @patch("services.document_indexing_task_proxy.FeatureService") + def test_delay_method(self, mock_feature_service): + """Test delay method integration.""" + # Arrange + mock_features = DocumentIndexingTaskProxyTestDataFactory.create_mock_features( + billing_enabled=True, plan=CloudPlan.SANDBOX + ) + mock_feature_service.get_features.return_value = mock_features + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_default_tenant_queue = Mock() + + # Act + proxy.delay() + + # Assert + # If billing enabled with sandbox plan, should send to default tenant queue + proxy._send_to_default_tenant_queue.assert_called_once() + + def test_document_task_dataclass(self): + """Test DocumentTask dataclass.""" + # Arrange + tenant_id = "tenant-123" + dataset_id = "dataset-456" + document_ids = ["doc-1", "doc-2"] + + # Act + task = DocumentTask(tenant_id=tenant_id, dataset_id=dataset_id, document_ids=document_ids) + + # Assert + assert task.tenant_id == tenant_id + assert task.dataset_id == dataset_id + assert task.document_ids == document_ids + + @patch("services.document_indexing_task_proxy.FeatureService") + def test_dispatch_edge_case_empty_plan(self, mock_feature_service): + """Test _dispatch method with empty plan string.""" + # Arrange + mock_features = DocumentIndexingTaskProxyTestDataFactory.create_mock_features(billing_enabled=True, plan="") + mock_feature_service.get_features.return_value = mock_features + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_priority_tenant_queue = Mock() + + # Act + proxy._dispatch() + + # Assert + proxy._send_to_priority_tenant_queue.assert_called_once() + + @patch("services.document_indexing_task_proxy.FeatureService") + def test_dispatch_edge_case_none_plan(self, mock_feature_service): + """Test _dispatch method with None plan.""" + # Arrange + mock_features = DocumentIndexingTaskProxyTestDataFactory.create_mock_features(billing_enabled=True, plan=None) + mock_feature_service.get_features.return_value = mock_features + proxy = DocumentIndexingTaskProxyTestDataFactory.create_document_task_proxy() + proxy._send_to_priority_tenant_queue = Mock() + + # Act + proxy._dispatch() + + # Assert + proxy._send_to_priority_tenant_queue.assert_called_once() + + def test_initialization_with_empty_document_ids(self): + """Test initialization with empty document_ids list.""" + # Arrange + tenant_id = "tenant-123" + dataset_id = "dataset-456" + document_ids = [] + + # Act + proxy = DocumentIndexingTaskProxy(tenant_id, dataset_id, document_ids) + + # Assert + assert proxy._tenant_id == tenant_id + assert proxy._dataset_id == dataset_id + assert proxy._document_ids == document_ids + + def test_initialization_with_single_document_id(self): + """Test initialization with single document_id.""" + # Arrange + tenant_id = "tenant-123" + dataset_id = "dataset-456" + document_ids = ["doc-1"] + + # Act + proxy = DocumentIndexingTaskProxy(tenant_id, dataset_id, document_ids) + + # Assert + assert proxy._tenant_id == tenant_id + assert proxy._dataset_id == dataset_id + assert proxy._document_ids == document_ids diff --git a/api/tests/unit_tests/services/test_rag_pipeline_task_proxy.py b/api/tests/unit_tests/services/test_rag_pipeline_task_proxy.py new file mode 100644 index 0000000000..f5a48b1416 --- /dev/null +++ b/api/tests/unit_tests/services/test_rag_pipeline_task_proxy.py @@ -0,0 +1,483 @@ +import json +from unittest.mock import Mock, patch + +import pytest + +from core.app.entities.rag_pipeline_invoke_entities import RagPipelineInvokeEntity +from core.rag.pipeline.queue import TenantIsolatedTaskQueue +from enums.cloud_plan import CloudPlan +from services.rag_pipeline.rag_pipeline_task_proxy import RagPipelineTaskProxy + + +class RagPipelineTaskProxyTestDataFactory: + """Factory class for creating test data and mock objects for RagPipelineTaskProxy tests.""" + + @staticmethod + def create_mock_features(billing_enabled: bool = False, plan: CloudPlan = CloudPlan.SANDBOX) -> Mock: + """Create mock features with billing configuration.""" + features = Mock() + features.billing = Mock() + features.billing.enabled = billing_enabled + features.billing.subscription = Mock() + features.billing.subscription.plan = plan + return features + + @staticmethod + def create_mock_tenant_queue(has_task_key: bool = False) -> Mock: + """Create mock TenantIsolatedTaskQueue.""" + queue = Mock(spec=TenantIsolatedTaskQueue) + queue.get_task_key.return_value = "task_key" if has_task_key else None + queue.push_tasks = Mock() + queue.set_task_waiting_time = Mock() + return queue + + @staticmethod + def create_rag_pipeline_invoke_entity( + pipeline_id: str = "pipeline-123", + user_id: str = "user-456", + tenant_id: str = "tenant-789", + workflow_id: str = "workflow-101", + streaming: bool = True, + workflow_execution_id: str | None = None, + workflow_thread_pool_id: str | None = None, + ) -> RagPipelineInvokeEntity: + """Create RagPipelineInvokeEntity instance for testing.""" + return RagPipelineInvokeEntity( + pipeline_id=pipeline_id, + application_generate_entity={"key": "value"}, + user_id=user_id, + tenant_id=tenant_id, + workflow_id=workflow_id, + streaming=streaming, + workflow_execution_id=workflow_execution_id, + workflow_thread_pool_id=workflow_thread_pool_id, + ) + + @staticmethod + def create_rag_pipeline_task_proxy( + dataset_tenant_id: str = "tenant-123", + user_id: str = "user-456", + rag_pipeline_invoke_entities: list[RagPipelineInvokeEntity] | None = None, + ) -> RagPipelineTaskProxy: + """Create RagPipelineTaskProxy instance for testing.""" + if rag_pipeline_invoke_entities is None: + rag_pipeline_invoke_entities = [RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_invoke_entity()] + return RagPipelineTaskProxy(dataset_tenant_id, user_id, rag_pipeline_invoke_entities) + + @staticmethod + def create_mock_upload_file(file_id: str = "file-123") -> Mock: + """Create mock upload file.""" + upload_file = Mock() + upload_file.id = file_id + return upload_file + + +class TestRagPipelineTaskProxy: + """Test cases for RagPipelineTaskProxy class.""" + + def test_initialization(self): + """Test RagPipelineTaskProxy initialization.""" + # Arrange + dataset_tenant_id = "tenant-123" + user_id = "user-456" + rag_pipeline_invoke_entities = [RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_invoke_entity()] + + # Act + proxy = RagPipelineTaskProxy(dataset_tenant_id, user_id, rag_pipeline_invoke_entities) + + # Assert + assert proxy._dataset_tenant_id == dataset_tenant_id + assert proxy._user_id == user_id + assert proxy._rag_pipeline_invoke_entities == rag_pipeline_invoke_entities + assert isinstance(proxy._tenant_isolated_task_queue, TenantIsolatedTaskQueue) + assert proxy._tenant_isolated_task_queue._tenant_id == dataset_tenant_id + assert proxy._tenant_isolated_task_queue._unique_key == "pipeline" + + def test_initialization_with_empty_entities(self): + """Test initialization with empty rag_pipeline_invoke_entities.""" + # Arrange + dataset_tenant_id = "tenant-123" + user_id = "user-456" + rag_pipeline_invoke_entities = [] + + # Act + proxy = RagPipelineTaskProxy(dataset_tenant_id, user_id, rag_pipeline_invoke_entities) + + # Assert + assert proxy._dataset_tenant_id == dataset_tenant_id + assert proxy._user_id == user_id + assert proxy._rag_pipeline_invoke_entities == [] + + def test_initialization_with_multiple_entities(self): + """Test initialization with multiple rag_pipeline_invoke_entities.""" + # Arrange + dataset_tenant_id = "tenant-123" + user_id = "user-456" + rag_pipeline_invoke_entities = [ + RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_invoke_entity(pipeline_id="pipeline-1"), + RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_invoke_entity(pipeline_id="pipeline-2"), + RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_invoke_entity(pipeline_id="pipeline-3"), + ] + + # Act + proxy = RagPipelineTaskProxy(dataset_tenant_id, user_id, rag_pipeline_invoke_entities) + + # Assert + assert len(proxy._rag_pipeline_invoke_entities) == 3 + assert proxy._rag_pipeline_invoke_entities[0].pipeline_id == "pipeline-1" + assert proxy._rag_pipeline_invoke_entities[1].pipeline_id == "pipeline-2" + assert proxy._rag_pipeline_invoke_entities[2].pipeline_id == "pipeline-3" + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FeatureService") + def test_features_property(self, mock_feature_service): + """Test cached_property features.""" + # Arrange + mock_features = RagPipelineTaskProxyTestDataFactory.create_mock_features() + mock_feature_service.get_features.return_value = mock_features + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + + # Act + features1 = proxy.features + features2 = proxy.features # Second call should use cached property + + # Assert + assert features1 == mock_features + assert features2 == mock_features + assert features1 is features2 # Should be the same instance due to caching + mock_feature_service.get_features.assert_called_once_with("tenant-123") + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_upload_invoke_entities(self, mock_db, mock_file_service_class): + """Test _upload_invoke_entities method.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-123") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + result = proxy._upload_invoke_entities() + + # Assert + assert result == "file-123" + mock_file_service_class.assert_called_once_with(mock_db.engine) + + # Verify upload_text was called with correct parameters + mock_file_service.upload_text.assert_called_once() + call_args = mock_file_service.upload_text.call_args + json_text, name, user_id, tenant_id = call_args[0] + + assert name == "rag_pipeline_invoke_entities.json" + assert user_id == "user-456" + assert tenant_id == "tenant-123" + + # Verify JSON content + parsed_json = json.loads(json_text) + assert len(parsed_json) == 1 + assert parsed_json[0]["pipeline_id"] == "pipeline-123" + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_upload_invoke_entities_with_multiple_entities(self, mock_db, mock_file_service_class): + """Test _upload_invoke_entities method with multiple entities.""" + # Arrange + entities = [ + RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_invoke_entity(pipeline_id="pipeline-1"), + RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_invoke_entity(pipeline_id="pipeline-2"), + ] + proxy = RagPipelineTaskProxy("tenant-123", "user-456", entities) + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-456") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + result = proxy._upload_invoke_entities() + + # Assert + assert result == "file-456" + + # Verify JSON content contains both entities + call_args = mock_file_service.upload_text.call_args + json_text = call_args[0][0] + parsed_json = json.loads(json_text) + assert len(parsed_json) == 2 + assert parsed_json[0]["pipeline_id"] == "pipeline-1" + assert parsed_json[1]["pipeline_id"] == "pipeline-2" + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.rag_pipeline_run_task") + def test_send_to_direct_queue(self, mock_task): + """Test _send_to_direct_queue method.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._tenant_isolated_task_queue = RagPipelineTaskProxyTestDataFactory.create_mock_tenant_queue() + upload_file_id = "file-123" + mock_task.delay = Mock() + + # Act + proxy._send_to_direct_queue(upload_file_id, mock_task) + + # If sent to direct queue, tenant_isolated_task_queue should not be called + proxy._tenant_isolated_task_queue.push_tasks.assert_not_called() + + # Celery should be called directly + mock_task.delay.assert_called_once_with( + rag_pipeline_invoke_entities_file_id=upload_file_id, tenant_id="tenant-123" + ) + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.rag_pipeline_run_task") + def test_send_to_tenant_queue_with_existing_task_key(self, mock_task): + """Test _send_to_tenant_queue when task key exists.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._tenant_isolated_task_queue = RagPipelineTaskProxyTestDataFactory.create_mock_tenant_queue( + has_task_key=True + ) + upload_file_id = "file-123" + mock_task.delay = Mock() + + # Act + proxy._send_to_tenant_queue(upload_file_id, mock_task) + + # If task key exists, should push tasks to the queue + proxy._tenant_isolated_task_queue.push_tasks.assert_called_once_with([upload_file_id]) + # Celery should not be called directly + mock_task.delay.assert_not_called() + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.rag_pipeline_run_task") + def test_send_to_tenant_queue_without_task_key(self, mock_task): + """Test _send_to_tenant_queue when no task key exists.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._tenant_isolated_task_queue = RagPipelineTaskProxyTestDataFactory.create_mock_tenant_queue( + has_task_key=False + ) + upload_file_id = "file-123" + mock_task.delay = Mock() + + # Act + proxy._send_to_tenant_queue(upload_file_id, mock_task) + + # If no task key, should set task waiting time key first + proxy._tenant_isolated_task_queue.set_task_waiting_time.assert_called_once() + mock_task.delay.assert_called_once_with( + rag_pipeline_invoke_entities_file_id=upload_file_id, tenant_id="tenant-123" + ) + + # The first task should be sent to celery directly, so push tasks should not be called + proxy._tenant_isolated_task_queue.push_tasks.assert_not_called() + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.rag_pipeline_run_task") + def test_send_to_default_tenant_queue(self, mock_task): + """Test _send_to_default_tenant_queue method.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_tenant_queue = Mock() + upload_file_id = "file-123" + + # Act + proxy._send_to_default_tenant_queue(upload_file_id) + + # Assert + proxy._send_to_tenant_queue.assert_called_once_with(upload_file_id, mock_task) + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.priority_rag_pipeline_run_task") + def test_send_to_priority_tenant_queue(self, mock_task): + """Test _send_to_priority_tenant_queue method.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_tenant_queue = Mock() + upload_file_id = "file-123" + + # Act + proxy._send_to_priority_tenant_queue(upload_file_id) + + # Assert + proxy._send_to_tenant_queue.assert_called_once_with(upload_file_id, mock_task) + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.priority_rag_pipeline_run_task") + def test_send_to_priority_direct_queue(self, mock_task): + """Test _send_to_priority_direct_queue method.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_direct_queue = Mock() + upload_file_id = "file-123" + + # Act + proxy._send_to_priority_direct_queue(upload_file_id) + + # Assert + proxy._send_to_direct_queue.assert_called_once_with(upload_file_id, mock_task) + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FeatureService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_dispatch_with_billing_enabled_sandbox_plan(self, mock_db, mock_file_service_class, mock_feature_service): + """Test _dispatch method when billing is enabled with sandbox plan.""" + # Arrange + mock_features = RagPipelineTaskProxyTestDataFactory.create_mock_features( + billing_enabled=True, plan=CloudPlan.SANDBOX + ) + mock_feature_service.get_features.return_value = mock_features + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_default_tenant_queue = Mock() + + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-123") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + proxy._dispatch() + + # If billing is enabled with sandbox plan, should send to default tenant queue + proxy._send_to_default_tenant_queue.assert_called_once_with("file-123") + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FeatureService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_dispatch_with_billing_enabled_non_sandbox_plan( + self, mock_db, mock_file_service_class, mock_feature_service + ): + """Test _dispatch method when billing is enabled with non-sandbox plan.""" + # Arrange + mock_features = RagPipelineTaskProxyTestDataFactory.create_mock_features( + billing_enabled=True, plan=CloudPlan.TEAM + ) + mock_feature_service.get_features.return_value = mock_features + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_priority_tenant_queue = Mock() + + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-123") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + proxy._dispatch() + + # If billing is enabled with non-sandbox plan, should send to priority tenant queue + proxy._send_to_priority_tenant_queue.assert_called_once_with("file-123") + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FeatureService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_dispatch_with_billing_disabled(self, mock_db, mock_file_service_class, mock_feature_service): + """Test _dispatch method when billing is disabled.""" + # Arrange + mock_features = RagPipelineTaskProxyTestDataFactory.create_mock_features(billing_enabled=False) + mock_feature_service.get_features.return_value = mock_features + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_priority_direct_queue = Mock() + + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-123") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + proxy._dispatch() + + # If billing is disabled, for example: self-hosted or enterprise, should send to priority direct queue + proxy._send_to_priority_direct_queue.assert_called_once_with("file-123") + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_dispatch_with_empty_upload_file_id(self, mock_db, mock_file_service_class): + """Test _dispatch method when upload_file_id is empty.""" + # Arrange + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = Mock() + mock_upload_file.id = "" # Empty file ID + mock_file_service.upload_text.return_value = mock_upload_file + + # Act & Assert + with pytest.raises(ValueError, match="upload_file_id is empty"): + proxy._dispatch() + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FeatureService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_dispatch_edge_case_empty_plan(self, mock_db, mock_file_service_class, mock_feature_service): + """Test _dispatch method with empty plan string.""" + # Arrange + mock_features = RagPipelineTaskProxyTestDataFactory.create_mock_features(billing_enabled=True, plan="") + mock_feature_service.get_features.return_value = mock_features + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_priority_tenant_queue = Mock() + + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-123") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + proxy._dispatch() + + # Assert + proxy._send_to_priority_tenant_queue.assert_called_once_with("file-123") + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FeatureService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_dispatch_edge_case_none_plan(self, mock_db, mock_file_service_class, mock_feature_service): + """Test _dispatch method with None plan.""" + # Arrange + mock_features = RagPipelineTaskProxyTestDataFactory.create_mock_features(billing_enabled=True, plan=None) + mock_feature_service.get_features.return_value = mock_features + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._send_to_priority_tenant_queue = Mock() + + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-123") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + proxy._dispatch() + + # Assert + proxy._send_to_priority_tenant_queue.assert_called_once_with("file-123") + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FeatureService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.FileService") + @patch("services.rag_pipeline.rag_pipeline_task_proxy.db") + def test_delay_method(self, mock_db, mock_file_service_class, mock_feature_service): + """Test delay method integration.""" + # Arrange + mock_features = RagPipelineTaskProxyTestDataFactory.create_mock_features( + billing_enabled=True, plan=CloudPlan.SANDBOX + ) + mock_feature_service.get_features.return_value = mock_features + proxy = RagPipelineTaskProxyTestDataFactory.create_rag_pipeline_task_proxy() + proxy._dispatch = Mock() + + mock_file_service = Mock() + mock_file_service_class.return_value = mock_file_service + mock_upload_file = RagPipelineTaskProxyTestDataFactory.create_mock_upload_file("file-123") + mock_file_service.upload_text.return_value = mock_upload_file + + # Act + proxy.delay() + + # Assert + proxy._dispatch.assert_called_once() + + @patch("services.rag_pipeline.rag_pipeline_task_proxy.logger") + def test_delay_method_with_empty_entities(self, mock_logger): + """Test delay method with empty rag_pipeline_invoke_entities.""" + # Arrange + proxy = RagPipelineTaskProxy("tenant-123", "user-456", []) + + # Act + proxy.delay() + + # Assert + mock_logger.warning.assert_called_once_with( + "Received empty rag pipeline invoke entities, no tasks delivered: %s %s", "tenant-123", "user-456" + ) diff --git a/api/tests/unit_tests/services/test_schedule_service.py b/api/tests/unit_tests/services/test_schedule_service.py new file mode 100644 index 0000000000..e28965ea2c --- /dev/null +++ b/api/tests/unit_tests/services/test_schedule_service.py @@ -0,0 +1,779 @@ +import unittest +from datetime import UTC, datetime +from unittest.mock import MagicMock, Mock, patch + +import pytest +from sqlalchemy.orm import Session + +from core.workflow.nodes.trigger_schedule.entities import ScheduleConfig, SchedulePlanUpdate, VisualConfig +from core.workflow.nodes.trigger_schedule.exc import ScheduleConfigError +from events.event_handlers.sync_workflow_schedule_when_app_published import ( + sync_schedule_from_workflow, +) +from libs.schedule_utils import calculate_next_run_at, convert_12h_to_24h +from models.account import Account, TenantAccountJoin +from models.trigger import WorkflowSchedulePlan +from models.workflow import Workflow +from services.trigger.schedule_service import ScheduleService + + +class TestScheduleService(unittest.TestCase): + """Test cases for ScheduleService class.""" + + def test_calculate_next_run_at_valid_cron(self): + """Test calculating next run time with valid cron expression.""" + # Test daily cron at 10:30 AM + cron_expr = "30 10 * * *" + timezone = "UTC" + base_time = datetime(2025, 8, 29, 9, 0, 0, tzinfo=UTC) + + next_run = calculate_next_run_at(cron_expr, timezone, base_time) + + assert next_run is not None + assert next_run.hour == 10 + assert next_run.minute == 30 + assert next_run.day == 29 + + def test_calculate_next_run_at_with_timezone(self): + """Test calculating next run time with different timezone.""" + cron_expr = "0 9 * * *" # 9:00 AM + timezone = "America/New_York" + base_time = datetime(2025, 8, 29, 12, 0, 0, tzinfo=UTC) # 8:00 AM EDT + + next_run = calculate_next_run_at(cron_expr, timezone, base_time) + + assert next_run is not None + # 9:00 AM EDT = 13:00 UTC (during EDT) + expected_utc_hour = 13 + assert next_run.hour == expected_utc_hour + + def test_calculate_next_run_at_with_last_day_of_month(self): + """Test calculating next run time with 'L' (last day) syntax.""" + cron_expr = "0 10 L * *" # 10:00 AM on last day of month + timezone = "UTC" + base_time = datetime(2025, 2, 15, 9, 0, 0, tzinfo=UTC) + + next_run = calculate_next_run_at(cron_expr, timezone, base_time) + + assert next_run is not None + # February 2025 has 28 days + assert next_run.day == 28 + assert next_run.month == 2 + + def test_calculate_next_run_at_invalid_cron(self): + """Test calculating next run time with invalid cron expression.""" + cron_expr = "invalid cron" + timezone = "UTC" + + with pytest.raises(ValueError): + calculate_next_run_at(cron_expr, timezone) + + def test_calculate_next_run_at_invalid_timezone(self): + """Test calculating next run time with invalid timezone.""" + from pytz import UnknownTimeZoneError + + cron_expr = "30 10 * * *" + timezone = "Invalid/Timezone" + + with pytest.raises(UnknownTimeZoneError): + calculate_next_run_at(cron_expr, timezone) + + @patch("libs.schedule_utils.calculate_next_run_at") + def test_create_schedule(self, mock_calculate_next_run): + """Test creating a new schedule.""" + mock_session = MagicMock(spec=Session) + mock_calculate_next_run.return_value = datetime(2025, 8, 30, 10, 30, 0, tzinfo=UTC) + + config = ScheduleConfig( + node_id="start", + cron_expression="30 10 * * *", + timezone="UTC", + ) + + schedule = ScheduleService.create_schedule( + session=mock_session, + tenant_id="test-tenant", + app_id="test-app", + config=config, + ) + + assert schedule is not None + assert schedule.tenant_id == "test-tenant" + assert schedule.app_id == "test-app" + assert schedule.node_id == "start" + assert schedule.cron_expression == "30 10 * * *" + assert schedule.timezone == "UTC" + assert schedule.next_run_at is not None + mock_session.add.assert_called_once() + mock_session.flush.assert_called_once() + + @patch("services.trigger.schedule_service.calculate_next_run_at") + def test_update_schedule(self, mock_calculate_next_run): + """Test updating an existing schedule.""" + mock_session = MagicMock(spec=Session) + mock_schedule = Mock(spec=WorkflowSchedulePlan) + mock_schedule.cron_expression = "0 12 * * *" + mock_schedule.timezone = "America/New_York" + mock_session.get.return_value = mock_schedule + mock_calculate_next_run.return_value = datetime(2025, 8, 30, 12, 0, 0, tzinfo=UTC) + + updates = SchedulePlanUpdate( + cron_expression="0 12 * * *", + timezone="America/New_York", + ) + + result = ScheduleService.update_schedule( + session=mock_session, + schedule_id="test-schedule-id", + updates=updates, + ) + + assert result is not None + assert result.cron_expression == "0 12 * * *" + assert result.timezone == "America/New_York" + mock_calculate_next_run.assert_called_once() + mock_session.flush.assert_called_once() + + def test_update_schedule_not_found(self): + """Test updating a non-existent schedule raises exception.""" + from core.workflow.nodes.trigger_schedule.exc import ScheduleNotFoundError + + mock_session = MagicMock(spec=Session) + mock_session.get.return_value = None + + updates = SchedulePlanUpdate( + cron_expression="0 12 * * *", + ) + + with pytest.raises(ScheduleNotFoundError) as context: + ScheduleService.update_schedule( + session=mock_session, + schedule_id="non-existent-id", + updates=updates, + ) + + assert "Schedule not found: non-existent-id" in str(context.value) + mock_session.flush.assert_not_called() + + def test_delete_schedule(self): + """Test deleting a schedule.""" + mock_session = MagicMock(spec=Session) + mock_schedule = Mock(spec=WorkflowSchedulePlan) + mock_session.get.return_value = mock_schedule + + # Should not raise exception and complete successfully + ScheduleService.delete_schedule( + session=mock_session, + schedule_id="test-schedule-id", + ) + + mock_session.delete.assert_called_once_with(mock_schedule) + mock_session.flush.assert_called_once() + + def test_delete_schedule_not_found(self): + """Test deleting a non-existent schedule raises exception.""" + from core.workflow.nodes.trigger_schedule.exc import ScheduleNotFoundError + + mock_session = MagicMock(spec=Session) + mock_session.get.return_value = None + + # Should raise ScheduleNotFoundError + with pytest.raises(ScheduleNotFoundError) as context: + ScheduleService.delete_schedule( + session=mock_session, + schedule_id="non-existent-id", + ) + + assert "Schedule not found: non-existent-id" in str(context.value) + mock_session.delete.assert_not_called() + + @patch("services.trigger.schedule_service.select") + def test_get_tenant_owner(self, mock_select): + """Test getting tenant owner account.""" + mock_session = MagicMock(spec=Session) + mock_account = Mock(spec=Account) + mock_account.id = "owner-account-id" + + # Mock owner query + mock_owner_result = Mock(spec=TenantAccountJoin) + mock_owner_result.account_id = "owner-account-id" + + mock_session.execute.return_value.scalar_one_or_none.return_value = mock_owner_result + mock_session.get.return_value = mock_account + + result = ScheduleService.get_tenant_owner( + session=mock_session, + tenant_id="test-tenant", + ) + + assert result is not None + assert result.id == "owner-account-id" + + @patch("services.trigger.schedule_service.select") + def test_get_tenant_owner_fallback_to_admin(self, mock_select): + """Test getting tenant owner falls back to admin if no owner.""" + mock_session = MagicMock(spec=Session) + mock_account = Mock(spec=Account) + mock_account.id = "admin-account-id" + + # Mock admin query (owner returns None) + mock_admin_result = Mock(spec=TenantAccountJoin) + mock_admin_result.account_id = "admin-account-id" + + mock_session.execute.return_value.scalar_one_or_none.side_effect = [None, mock_admin_result] + mock_session.get.return_value = mock_account + + result = ScheduleService.get_tenant_owner( + session=mock_session, + tenant_id="test-tenant", + ) + + assert result is not None + assert result.id == "admin-account-id" + + @patch("services.trigger.schedule_service.calculate_next_run_at") + def test_update_next_run_at(self, mock_calculate_next_run): + """Test updating next run time after schedule triggered.""" + mock_session = MagicMock(spec=Session) + mock_schedule = Mock(spec=WorkflowSchedulePlan) + mock_schedule.cron_expression = "30 10 * * *" + mock_schedule.timezone = "UTC" + mock_session.get.return_value = mock_schedule + + next_time = datetime(2025, 8, 31, 10, 30, 0, tzinfo=UTC) + mock_calculate_next_run.return_value = next_time + + result = ScheduleService.update_next_run_at( + session=mock_session, + schedule_id="test-schedule-id", + ) + + assert result == next_time + assert mock_schedule.next_run_at == next_time + mock_session.flush.assert_called_once() + + +class TestVisualToCron(unittest.TestCase): + """Test cases for visual configuration to cron conversion.""" + + def test_visual_to_cron_hourly(self): + """Test converting hourly visual config to cron.""" + visual_config = VisualConfig(on_minute=15) + result = ScheduleService.visual_to_cron("hourly", visual_config) + assert result == "15 * * * *" + + def test_visual_to_cron_daily(self): + """Test converting daily visual config to cron.""" + visual_config = VisualConfig(time="2:30 PM") + result = ScheduleService.visual_to_cron("daily", visual_config) + assert result == "30 14 * * *" + + def test_visual_to_cron_weekly(self): + """Test converting weekly visual config to cron.""" + visual_config = VisualConfig( + time="10:00 AM", + weekdays=["mon", "wed", "fri"], + ) + result = ScheduleService.visual_to_cron("weekly", visual_config) + assert result == "0 10 * * 1,3,5" + + def test_visual_to_cron_monthly_with_specific_days(self): + """Test converting monthly visual config with specific days.""" + visual_config = VisualConfig( + time="11:30 AM", + monthly_days=[1, 15], + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + assert result == "30 11 1,15 * *" + + def test_visual_to_cron_monthly_with_last_day(self): + """Test converting monthly visual config with last day using 'L' syntax.""" + visual_config = VisualConfig( + time="11:30 AM", + monthly_days=[1, "last"], + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + assert result == "30 11 1,L * *" + + def test_visual_to_cron_monthly_only_last_day(self): + """Test converting monthly visual config with only last day.""" + visual_config = VisualConfig( + time="9:00 PM", + monthly_days=["last"], + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + assert result == "0 21 L * *" + + def test_visual_to_cron_monthly_with_end_days_and_last(self): + """Test converting monthly visual config with days 29, 30, 31 and 'last'.""" + visual_config = VisualConfig( + time="3:45 PM", + monthly_days=[29, 30, 31, "last"], + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + # Should have 29,30,31,L - the L handles all possible last days + assert result == "45 15 29,30,31,L * *" + + def test_visual_to_cron_invalid_frequency(self): + """Test converting with invalid frequency.""" + with pytest.raises(ScheduleConfigError, match="Unsupported frequency: invalid"): + ScheduleService.visual_to_cron("invalid", VisualConfig()) + + def test_visual_to_cron_weekly_no_weekdays(self): + """Test converting weekly with no weekdays specified.""" + visual_config = VisualConfig(time="10:00 AM") + with pytest.raises(ScheduleConfigError, match="Weekdays are required for weekly schedules"): + ScheduleService.visual_to_cron("weekly", visual_config) + + def test_visual_to_cron_hourly_no_minute(self): + """Test converting hourly with no on_minute specified.""" + visual_config = VisualConfig() # on_minute defaults to 0 + result = ScheduleService.visual_to_cron("hourly", visual_config) + assert result == "0 * * * *" # Should use default value 0 + + def test_visual_to_cron_daily_no_time(self): + """Test converting daily with no time specified.""" + visual_config = VisualConfig(time=None) + with pytest.raises(ScheduleConfigError, match="time is required for daily schedules"): + ScheduleService.visual_to_cron("daily", visual_config) + + def test_visual_to_cron_weekly_no_time(self): + """Test converting weekly with no time specified.""" + visual_config = VisualConfig(weekdays=["mon"]) + visual_config.time = None # Override default + with pytest.raises(ScheduleConfigError, match="time is required for weekly schedules"): + ScheduleService.visual_to_cron("weekly", visual_config) + + def test_visual_to_cron_monthly_no_time(self): + """Test converting monthly with no time specified.""" + visual_config = VisualConfig(monthly_days=[1]) + visual_config.time = None # Override default + with pytest.raises(ScheduleConfigError, match="time is required for monthly schedules"): + ScheduleService.visual_to_cron("monthly", visual_config) + + def test_visual_to_cron_monthly_duplicate_days(self): + """Test monthly with duplicate days should be deduplicated.""" + visual_config = VisualConfig( + time="10:00 AM", + monthly_days=[1, 15, 1, 15, 31], # Duplicates + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + assert result == "0 10 1,15,31 * *" # Should be deduplicated + + def test_visual_to_cron_monthly_unsorted_days(self): + """Test monthly with unsorted days should be sorted.""" + visual_config = VisualConfig( + time="2:30 PM", + monthly_days=[20, 5, 15, 1, 10], # Unsorted + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + assert result == "30 14 1,5,10,15,20 * *" # Should be sorted + + def test_visual_to_cron_weekly_all_weekdays(self): + """Test weekly with all weekdays.""" + visual_config = VisualConfig( + time="8:00 AM", + weekdays=["sun", "mon", "tue", "wed", "thu", "fri", "sat"], + ) + result = ScheduleService.visual_to_cron("weekly", visual_config) + assert result == "0 8 * * 0,1,2,3,4,5,6" + + def test_visual_to_cron_hourly_boundary_values(self): + """Test hourly with boundary minute values.""" + # Minimum value + visual_config = VisualConfig(on_minute=0) + result = ScheduleService.visual_to_cron("hourly", visual_config) + assert result == "0 * * * *" + + # Maximum value + visual_config = VisualConfig(on_minute=59) + result = ScheduleService.visual_to_cron("hourly", visual_config) + assert result == "59 * * * *" + + def test_visual_to_cron_daily_midnight_noon(self): + """Test daily at special times (midnight and noon).""" + # Midnight + visual_config = VisualConfig(time="12:00 AM") + result = ScheduleService.visual_to_cron("daily", visual_config) + assert result == "0 0 * * *" + + # Noon + visual_config = VisualConfig(time="12:00 PM") + result = ScheduleService.visual_to_cron("daily", visual_config) + assert result == "0 12 * * *" + + def test_visual_to_cron_monthly_mixed_with_last_and_duplicates(self): + """Test monthly with mixed days, 'last', and duplicates.""" + visual_config = VisualConfig( + time="11:45 PM", + monthly_days=[15, 1, "last", 15, 30, 1, "last"], # Mixed with duplicates + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + assert result == "45 23 1,15,30,L * *" # Deduplicated and sorted with L at end + + def test_visual_to_cron_weekly_single_day(self): + """Test weekly with single weekday.""" + visual_config = VisualConfig( + time="6:30 PM", + weekdays=["sun"], + ) + result = ScheduleService.visual_to_cron("weekly", visual_config) + assert result == "30 18 * * 0" + + def test_visual_to_cron_monthly_all_possible_days(self): + """Test monthly with all 31 days plus 'last'.""" + all_days = list(range(1, 32)) + ["last"] + visual_config = VisualConfig( + time="12:01 AM", + monthly_days=all_days, + ) + result = ScheduleService.visual_to_cron("monthly", visual_config) + expected_days = ",".join([str(i) for i in range(1, 32)]) + ",L" + assert result == f"1 0 {expected_days} * *" + + def test_visual_to_cron_monthly_no_days(self): + """Test monthly without any days specified should raise error.""" + visual_config = VisualConfig(time="10:00 AM", monthly_days=[]) + with pytest.raises(ScheduleConfigError, match="Monthly days are required for monthly schedules"): + ScheduleService.visual_to_cron("monthly", visual_config) + + def test_visual_to_cron_weekly_empty_weekdays_list(self): + """Test weekly with empty weekdays list should raise error.""" + visual_config = VisualConfig(time="10:00 AM", weekdays=[]) + with pytest.raises(ScheduleConfigError, match="Weekdays are required for weekly schedules"): + ScheduleService.visual_to_cron("weekly", visual_config) + + +class TestParseTime(unittest.TestCase): + """Test cases for time parsing function.""" + + def test_parse_time_am(self): + """Test parsing AM time.""" + hour, minute = convert_12h_to_24h("9:30 AM") + assert hour == 9 + assert minute == 30 + + def test_parse_time_pm(self): + """Test parsing PM time.""" + hour, minute = convert_12h_to_24h("2:45 PM") + assert hour == 14 + assert minute == 45 + + def test_parse_time_noon(self): + """Test parsing 12:00 PM (noon).""" + hour, minute = convert_12h_to_24h("12:00 PM") + assert hour == 12 + assert minute == 0 + + def test_parse_time_midnight(self): + """Test parsing 12:00 AM (midnight).""" + hour, minute = convert_12h_to_24h("12:00 AM") + assert hour == 0 + assert minute == 0 + + def test_parse_time_invalid_format(self): + """Test parsing invalid time format.""" + with pytest.raises(ValueError, match="Invalid time format"): + convert_12h_to_24h("25:00") + + def test_parse_time_invalid_hour(self): + """Test parsing invalid hour.""" + with pytest.raises(ValueError, match="Invalid hour: 13"): + convert_12h_to_24h("13:00 PM") + + def test_parse_time_invalid_minute(self): + """Test parsing invalid minute.""" + with pytest.raises(ValueError, match="Invalid minute: 60"): + convert_12h_to_24h("10:60 AM") + + def test_parse_time_empty_string(self): + """Test parsing empty string.""" + with pytest.raises(ValueError, match="Time string cannot be empty"): + convert_12h_to_24h("") + + def test_parse_time_invalid_period(self): + """Test parsing invalid period.""" + with pytest.raises(ValueError, match="Invalid period"): + convert_12h_to_24h("10:30 XM") + + +class TestExtractScheduleConfig(unittest.TestCase): + """Test cases for extracting schedule configuration from workflow.""" + + def test_extract_schedule_config_with_cron_mode(self): + """Test extracting schedule config in cron mode.""" + workflow = Mock(spec=Workflow) + workflow.graph_dict = { + "nodes": [ + { + "id": "schedule-node", + "data": { + "type": "trigger-schedule", + "mode": "cron", + "cron_expression": "0 10 * * *", + "timezone": "America/New_York", + }, + } + ] + } + + config = ScheduleService.extract_schedule_config(workflow) + + assert config is not None + assert config.node_id == "schedule-node" + assert config.cron_expression == "0 10 * * *" + assert config.timezone == "America/New_York" + + def test_extract_schedule_config_with_visual_mode(self): + """Test extracting schedule config in visual mode.""" + workflow = Mock(spec=Workflow) + workflow.graph_dict = { + "nodes": [ + { + "id": "schedule-node", + "data": { + "type": "trigger-schedule", + "mode": "visual", + "frequency": "daily", + "visual_config": {"time": "10:30 AM"}, + "timezone": "UTC", + }, + } + ] + } + + config = ScheduleService.extract_schedule_config(workflow) + + assert config is not None + assert config.node_id == "schedule-node" + assert config.cron_expression == "30 10 * * *" + assert config.timezone == "UTC" + + def test_extract_schedule_config_no_schedule_node(self): + """Test extracting config when no schedule node exists.""" + workflow = Mock(spec=Workflow) + workflow.graph_dict = { + "nodes": [ + { + "id": "other-node", + "data": {"type": "llm"}, + } + ] + } + + config = ScheduleService.extract_schedule_config(workflow) + assert config is None + + def test_extract_schedule_config_invalid_graph(self): + """Test extracting config with invalid graph data.""" + workflow = Mock(spec=Workflow) + workflow.graph_dict = None + + with pytest.raises(ScheduleConfigError, match="Workflow graph is empty"): + ScheduleService.extract_schedule_config(workflow) + + +class TestScheduleWithTimezone(unittest.TestCase): + """Test cases for schedule with timezone handling.""" + + def test_visual_schedule_with_timezone_integration(self): + """Test complete flow: visual config → cron → execution in different timezones. + + This test verifies that when a user in Shanghai sets a schedule for 10:30 AM, + it runs at 10:30 AM Shanghai time, not 10:30 AM UTC. + """ + # User in Shanghai wants to run a task at 10:30 AM local time + visual_config = VisualConfig( + time="10:30 AM", # This is Shanghai time + monthly_days=[1], + ) + + # Convert to cron expression + cron_expr = ScheduleService.visual_to_cron("monthly", visual_config) + assert cron_expr is not None + + assert cron_expr == "30 10 1 * *" # Direct conversion + + # Now test execution with Shanghai timezone + shanghai_tz = "Asia/Shanghai" + # Base time: 2025-01-01 00:00:00 UTC (08:00:00 Shanghai) + base_time = datetime(2025, 1, 1, 0, 0, 0, tzinfo=UTC) + + next_run = calculate_next_run_at(cron_expr, shanghai_tz, base_time) + + assert next_run is not None + + # Should run at 10:30 AM Shanghai time on Jan 1 + # 10:30 AM Shanghai = 02:30 AM UTC (Shanghai is UTC+8) + assert next_run.year == 2025 + assert next_run.month == 1 + assert next_run.day == 1 + assert next_run.hour == 2 # 02:30 UTC + assert next_run.minute == 30 + + def test_visual_schedule_different_timezones_same_local_time(self): + """Test that same visual config in different timezones runs at different UTC times. + + This verifies that a schedule set for "9:00 AM" runs at 9 AM local time + regardless of the timezone. + """ + visual_config = VisualConfig( + time="9:00 AM", + weekdays=["mon"], + ) + + cron_expr = ScheduleService.visual_to_cron("weekly", visual_config) + assert cron_expr is not None + assert cron_expr == "0 9 * * 1" + + # Base time: Sunday 2025-01-05 12:00:00 UTC + base_time = datetime(2025, 1, 5, 12, 0, 0, tzinfo=UTC) + + # Test New York (UTC-5 in January) + ny_next = calculate_next_run_at(cron_expr, "America/New_York", base_time) + assert ny_next is not None + # Monday 9 AM EST = Monday 14:00 UTC + assert ny_next.day == 6 + assert ny_next.hour == 14 # 9 AM EST = 2 PM UTC + + # Test Tokyo (UTC+9) + tokyo_next = calculate_next_run_at(cron_expr, "Asia/Tokyo", base_time) + assert tokyo_next is not None + # Monday 9 AM JST = Monday 00:00 UTC + assert tokyo_next.day == 6 + assert tokyo_next.hour == 0 # 9 AM JST = 0 AM UTC + + def test_visual_schedule_daily_across_dst_change(self): + """Test that daily schedules adjust correctly during DST changes. + + A schedule set for "10:00 AM" should always run at 10 AM local time, + even when DST changes. + """ + visual_config = VisualConfig( + time="10:00 AM", + ) + + cron_expr = ScheduleService.visual_to_cron("daily", visual_config) + assert cron_expr is not None + + assert cron_expr == "0 10 * * *" + + # Test before DST (EST - UTC-5) + winter_base = datetime(2025, 2, 1, 0, 0, 0, tzinfo=UTC) + winter_next = calculate_next_run_at(cron_expr, "America/New_York", winter_base) + assert winter_next is not None + # 10 AM EST = 15:00 UTC + assert winter_next.hour == 15 + + # Test during DST (EDT - UTC-4) + summer_base = datetime(2025, 6, 1, 0, 0, 0, tzinfo=UTC) + summer_next = calculate_next_run_at(cron_expr, "America/New_York", summer_base) + assert summer_next is not None + # 10 AM EDT = 14:00 UTC + assert summer_next.hour == 14 + + +class TestSyncScheduleFromWorkflow(unittest.TestCase): + """Test cases for syncing schedule from workflow.""" + + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.db") + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.ScheduleService") + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.select") + def test_sync_schedule_create_new(self, mock_select, mock_service, mock_db): + """Test creating new schedule when none exists.""" + mock_session = MagicMock() + mock_db.engine = MagicMock() + mock_session.__enter__ = MagicMock(return_value=mock_session) + mock_session.__exit__ = MagicMock(return_value=None) + Session = MagicMock(return_value=mock_session) + with patch("events.event_handlers.sync_workflow_schedule_when_app_published.Session", Session): + mock_session.scalar.return_value = None # No existing plan + + # Mock extract_schedule_config to return a ScheduleConfig object + mock_config = Mock(spec=ScheduleConfig) + mock_config.node_id = "start" + mock_config.cron_expression = "30 10 * * *" + mock_config.timezone = "UTC" + mock_service.extract_schedule_config.return_value = mock_config + + mock_new_plan = Mock(spec=WorkflowSchedulePlan) + mock_service.create_schedule.return_value = mock_new_plan + + workflow = Mock(spec=Workflow) + result = sync_schedule_from_workflow("tenant-id", "app-id", workflow) + + assert result == mock_new_plan + mock_service.create_schedule.assert_called_once() + mock_session.commit.assert_called_once() + + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.db") + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.ScheduleService") + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.select") + def test_sync_schedule_update_existing(self, mock_select, mock_service, mock_db): + """Test updating existing schedule.""" + mock_session = MagicMock() + mock_db.engine = MagicMock() + mock_session.__enter__ = MagicMock(return_value=mock_session) + mock_session.__exit__ = MagicMock(return_value=None) + Session = MagicMock(return_value=mock_session) + + with patch("events.event_handlers.sync_workflow_schedule_when_app_published.Session", Session): + mock_existing_plan = Mock(spec=WorkflowSchedulePlan) + mock_existing_plan.id = "existing-plan-id" + mock_session.scalar.return_value = mock_existing_plan + + # Mock extract_schedule_config to return a ScheduleConfig object + mock_config = Mock(spec=ScheduleConfig) + mock_config.node_id = "start" + mock_config.cron_expression = "0 12 * * *" + mock_config.timezone = "America/New_York" + mock_service.extract_schedule_config.return_value = mock_config + + mock_updated_plan = Mock(spec=WorkflowSchedulePlan) + mock_service.update_schedule.return_value = mock_updated_plan + + workflow = Mock(spec=Workflow) + result = sync_schedule_from_workflow("tenant-id", "app-id", workflow) + + assert result == mock_updated_plan + mock_service.update_schedule.assert_called_once() + # Verify the arguments passed to update_schedule + call_args = mock_service.update_schedule.call_args + assert call_args.kwargs["session"] == mock_session + assert call_args.kwargs["schedule_id"] == "existing-plan-id" + updates_obj = call_args.kwargs["updates"] + assert isinstance(updates_obj, SchedulePlanUpdate) + assert updates_obj.node_id == "start" + assert updates_obj.cron_expression == "0 12 * * *" + assert updates_obj.timezone == "America/New_York" + mock_session.commit.assert_called_once() + + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.db") + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.ScheduleService") + @patch("events.event_handlers.sync_workflow_schedule_when_app_published.select") + def test_sync_schedule_remove_when_no_config(self, mock_select, mock_service, mock_db): + """Test removing schedule when no schedule config in workflow.""" + mock_session = MagicMock() + mock_db.engine = MagicMock() + mock_session.__enter__ = MagicMock(return_value=mock_session) + mock_session.__exit__ = MagicMock(return_value=None) + Session = MagicMock(return_value=mock_session) + + with patch("events.event_handlers.sync_workflow_schedule_when_app_published.Session", Session): + mock_existing_plan = Mock(spec=WorkflowSchedulePlan) + mock_existing_plan.id = "existing-plan-id" + mock_session.scalar.return_value = mock_existing_plan + + mock_service.extract_schedule_config.return_value = None # No schedule config + + workflow = Mock(spec=Workflow) + result = sync_schedule_from_workflow("tenant-id", "app-id", workflow) + + assert result is None + # Now using ScheduleService.delete_schedule instead of session.delete + mock_service.delete_schedule.assert_called_once_with(session=mock_session, schedule_id="existing-plan-id") + mock_session.commit.assert_called_once() + + +if __name__ == "__main__": + unittest.main() diff --git a/api/tests/unit_tests/services/test_variable_truncator.py b/api/tests/unit_tests/services/test_variable_truncator.py index 6761f939e3..cf6fb25c1c 100644 --- a/api/tests/unit_tests/services/test_variable_truncator.py +++ b/api/tests/unit_tests/services/test_variable_truncator.py @@ -21,6 +21,7 @@ from core.file.enums import FileTransferMethod, FileType from core.file.models import File from core.variables.segments import ( ArrayFileSegment, + ArrayNumberSegment, ArraySegment, FileSegment, FloatSegment, @@ -30,6 +31,7 @@ from core.variables.segments import ( StringSegment, ) from services.variable_truncator import ( + DummyVariableTruncator, MaxDepthExceededError, TruncationResult, UnknownTypeError, @@ -596,3 +598,32 @@ class TestIntegrationScenarios: truncated_mapping, truncated = truncator.truncate_variable_mapping(mapping) assert truncated is False assert truncated_mapping == mapping + + +def test_dummy_variable_truncator_methods(): + """Test DummyVariableTruncator methods work correctly.""" + truncator = DummyVariableTruncator() + + # Test truncate_variable_mapping + test_data: dict[str, Any] = { + "key1": "value1", + "key2": ["item1", "item2"], + "large_array": list(range(2000)), + } + result, is_truncated = truncator.truncate_variable_mapping(test_data) + + assert result == test_data + assert not is_truncated + + # Test truncate method + segment = StringSegment(value="test string") + result = truncator.truncate(segment) + assert isinstance(result, TruncationResult) + assert result.result == segment + assert result.truncated is False + + segment = ArrayNumberSegment(value=list(range(2000))) + result = truncator.truncate(segment) + assert isinstance(result, TruncationResult) + assert result.result == segment + assert result.truncated is False diff --git a/api/tests/unit_tests/services/test_webhook_service.py b/api/tests/unit_tests/services/test_webhook_service.py new file mode 100644 index 0000000000..010295bcd6 --- /dev/null +++ b/api/tests/unit_tests/services/test_webhook_service.py @@ -0,0 +1,482 @@ +from io import BytesIO +from unittest.mock import MagicMock, patch + +import pytest +from flask import Flask +from werkzeug.datastructures import FileStorage + +from services.trigger.webhook_service import WebhookService + + +class TestWebhookServiceUnit: + """Unit tests for WebhookService focusing on business logic without database dependencies.""" + + def test_extract_webhook_data_json(self): + """Test webhook data extraction from JSON request.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json", "Authorization": "Bearer token"}, + query_string="version=1&format=json", + json={"message": "hello", "count": 42}, + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["headers"]["Authorization"] == "Bearer token" + # Query params are now extracted as raw strings + assert webhook_data["query_params"]["version"] == "1" + assert webhook_data["query_params"]["format"] == "json" + assert webhook_data["body"]["message"] == "hello" + assert webhook_data["body"]["count"] == 42 + assert webhook_data["files"] == {} + + def test_extract_webhook_data_query_params_remain_strings(self): + """Query parameters should be extracted as raw strings without automatic conversion.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="GET", + headers={"Content-Type": "application/json"}, + query_string="count=42&threshold=3.14&enabled=true¬e=text", + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + # After refactoring, raw extraction keeps query params as strings + assert webhook_data["query_params"]["count"] == "42" + assert webhook_data["query_params"]["threshold"] == "3.14" + assert webhook_data["query_params"]["enabled"] == "true" + assert webhook_data["query_params"]["note"] == "text" + + def test_extract_webhook_data_form_urlencoded(self): + """Test webhook data extraction from form URL encoded request.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/x-www-form-urlencoded"}, + data={"username": "test", "password": "secret"}, + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["body"]["username"] == "test" + assert webhook_data["body"]["password"] == "secret" + + def test_extract_webhook_data_multipart_with_files(self): + """Test webhook data extraction from multipart form with files.""" + app = Flask(__name__) + + # Create a mock file + file_content = b"test file content" + file_storage = FileStorage(stream=BytesIO(file_content), filename="test.txt", content_type="text/plain") + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "multipart/form-data"}, + data={"message": "test", "upload": file_storage}, + ): + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + with patch.object(WebhookService, "_process_file_uploads") as mock_process_files: + mock_process_files.return_value = {"upload": "mocked_file_obj"} + + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["body"]["message"] == "test" + assert webhook_data["files"]["upload"] == "mocked_file_obj" + mock_process_files.assert_called_once() + + def test_extract_webhook_data_raw_text(self): + """Test webhook data extraction from raw text request.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", method="POST", headers={"Content-Type": "text/plain"}, data="raw text content" + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["body"]["raw"] == "raw text content" + + def test_extract_webhook_data_invalid_json(self): + """Test webhook data extraction with invalid JSON.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", method="POST", headers={"Content-Type": "application/json"}, data="invalid json" + ): + webhook_trigger = MagicMock() + webhook_data = WebhookService.extract_webhook_data(webhook_trigger) + + assert webhook_data["method"] == "POST" + assert webhook_data["body"] == {} # Should default to empty dict + + def test_generate_webhook_response_default(self): + """Test webhook response generation with default values.""" + node_config = {"data": {}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 200 + assert response_data["status"] == "success" + assert "Webhook processed successfully" in response_data["message"] + + def test_generate_webhook_response_custom_json(self): + """Test webhook response generation with custom JSON response.""" + node_config = {"data": {"status_code": 201, "response_body": '{"result": "created", "id": 123}'}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 201 + assert response_data["result"] == "created" + assert response_data["id"] == 123 + + def test_generate_webhook_response_custom_text(self): + """Test webhook response generation with custom text response.""" + node_config = {"data": {"status_code": 202, "response_body": "Request accepted for processing"}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 202 + assert response_data["message"] == "Request accepted for processing" + + def test_generate_webhook_response_invalid_json(self): + """Test webhook response generation with invalid JSON response.""" + node_config = {"data": {"status_code": 400, "response_body": '{"invalid": json}'}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 400 + assert response_data["message"] == '{"invalid": json}' + + def test_generate_webhook_response_empty_response_body(self): + """Test webhook response generation with empty response body.""" + node_config = {"data": {"status_code": 204, "response_body": ""}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 204 + assert response_data["status"] == "success" + assert "Webhook processed successfully" in response_data["message"] + + def test_generate_webhook_response_array_json(self): + """Test webhook response generation with JSON array response.""" + node_config = {"data": {"status_code": 200, "response_body": '[{"id": 1}, {"id": 2}]'}} + + response_data, status_code = WebhookService.generate_webhook_response(node_config) + + assert status_code == 200 + assert isinstance(response_data, list) + assert len(response_data) == 2 + assert response_data[0]["id"] == 1 + assert response_data[1]["id"] == 2 + + @patch("services.trigger.webhook_service.ToolFileManager") + @patch("services.trigger.webhook_service.file_factory") + def test_process_file_uploads_success(self, mock_file_factory, mock_tool_file_manager): + """Test successful file upload processing.""" + # Mock ToolFileManager + mock_tool_file_instance = MagicMock() + mock_tool_file_manager.return_value = mock_tool_file_instance + + # Mock file creation + mock_tool_file = MagicMock() + mock_tool_file.id = "test_file_id" + mock_tool_file_instance.create_file_by_raw.return_value = mock_tool_file + + # Mock file factory + mock_file_obj = MagicMock() + mock_file_factory.build_from_mapping.return_value = mock_file_obj + + # Create mock files + files = { + "file1": MagicMock(filename="test1.txt", content_type="text/plain"), + "file2": MagicMock(filename="test2.jpg", content_type="image/jpeg"), + } + + # Mock file reads + files["file1"].read.return_value = b"content1" + files["file2"].read.return_value = b"content2" + + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + result = WebhookService._process_file_uploads(files, webhook_trigger) + + assert len(result) == 2 + assert "file1" in result + assert "file2" in result + + # Verify file processing was called for each file + assert mock_tool_file_manager.call_count == 2 + assert mock_file_factory.build_from_mapping.call_count == 2 + + @patch("services.trigger.webhook_service.ToolFileManager") + @patch("services.trigger.webhook_service.file_factory") + def test_process_file_uploads_with_errors(self, mock_file_factory, mock_tool_file_manager): + """Test file upload processing with errors.""" + # Mock ToolFileManager + mock_tool_file_instance = MagicMock() + mock_tool_file_manager.return_value = mock_tool_file_instance + + # Mock file creation + mock_tool_file = MagicMock() + mock_tool_file.id = "test_file_id" + mock_tool_file_instance.create_file_by_raw.return_value = mock_tool_file + + # Mock file factory + mock_file_obj = MagicMock() + mock_file_factory.build_from_mapping.return_value = mock_file_obj + + # Create mock files, one will fail + files = { + "good_file": MagicMock(filename="test.txt", content_type="text/plain"), + "bad_file": MagicMock(filename="test.bad", content_type="text/plain"), + } + + files["good_file"].read.return_value = b"content" + files["bad_file"].read.side_effect = Exception("Read error") + + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + result = WebhookService._process_file_uploads(files, webhook_trigger) + + # Should process the good file and skip the bad one + assert len(result) == 1 + assert "good_file" in result + assert "bad_file" not in result + + def test_process_file_uploads_empty_filename(self): + """Test file upload processing with empty filename.""" + files = { + "no_filename": MagicMock(filename="", content_type="text/plain"), + "none_filename": MagicMock(filename=None, content_type="text/plain"), + } + + webhook_trigger = MagicMock() + webhook_trigger.tenant_id = "test_tenant" + + result = WebhookService._process_file_uploads(files, webhook_trigger) + + # Should skip files without filenames + assert len(result) == 0 + + def test_validate_json_value_string(self): + """Test JSON value validation for string type.""" + # Valid string + result = WebhookService._validate_json_value("name", "hello", "string") + assert result == "hello" + + # Invalid string (number) - should raise ValueError + with pytest.raises(ValueError, match="Expected string, got int"): + WebhookService._validate_json_value("name", 123, "string") + + def test_validate_json_value_number(self): + """Test JSON value validation for number type.""" + # Valid integer + result = WebhookService._validate_json_value("count", 42, "number") + assert result == 42 + + # Valid float + result = WebhookService._validate_json_value("price", 19.99, "number") + assert result == 19.99 + + # Invalid number (string) - should raise ValueError + with pytest.raises(ValueError, match="Expected number, got str"): + WebhookService._validate_json_value("count", "42", "number") + + def test_validate_json_value_bool(self): + """Test JSON value validation for boolean type.""" + # Valid boolean + result = WebhookService._validate_json_value("enabled", True, "boolean") + assert result is True + + result = WebhookService._validate_json_value("enabled", False, "boolean") + assert result is False + + # Invalid boolean (string) - should raise ValueError + with pytest.raises(ValueError, match="Expected boolean, got str"): + WebhookService._validate_json_value("enabled", "true", "boolean") + + def test_validate_json_value_object(self): + """Test JSON value validation for object type.""" + # Valid object + result = WebhookService._validate_json_value("user", {"name": "John", "age": 30}, "object") + assert result == {"name": "John", "age": 30} + + # Invalid object (string) - should raise ValueError + with pytest.raises(ValueError, match="Expected object, got str"): + WebhookService._validate_json_value("user", "not_an_object", "object") + + def test_validate_json_value_array_string(self): + """Test JSON value validation for array[string] type.""" + # Valid array of strings + result = WebhookService._validate_json_value("tags", ["tag1", "tag2", "tag3"], "array[string]") + assert result == ["tag1", "tag2", "tag3"] + + # Invalid - not an array + with pytest.raises(ValueError, match="Expected array of strings, got str"): + WebhookService._validate_json_value("tags", "not_an_array", "array[string]") + + # Invalid - array with non-strings + with pytest.raises(ValueError, match="Expected array of strings, got list"): + WebhookService._validate_json_value("tags", ["tag1", 123, "tag3"], "array[string]") + + def test_validate_json_value_array_number(self): + """Test JSON value validation for array[number] type.""" + # Valid array of numbers + result = WebhookService._validate_json_value("scores", [1, 2.5, 3, 4.7], "array[number]") + assert result == [1, 2.5, 3, 4.7] + + # Invalid - array with non-numbers + with pytest.raises(ValueError, match="Expected array of numbers, got list"): + WebhookService._validate_json_value("scores", [1, "2", 3], "array[number]") + + def test_validate_json_value_array_bool(self): + """Test JSON value validation for array[boolean] type.""" + # Valid array of booleans + result = WebhookService._validate_json_value("flags", [True, False, True], "array[boolean]") + assert result == [True, False, True] + + # Invalid - array with non-booleans + with pytest.raises(ValueError, match="Expected array of booleans, got list"): + WebhookService._validate_json_value("flags", [True, "false", True], "array[boolean]") + + def test_validate_json_value_array_object(self): + """Test JSON value validation for array[object] type.""" + # Valid array of objects + result = WebhookService._validate_json_value("users", [{"name": "John"}, {"name": "Jane"}], "array[object]") + assert result == [{"name": "John"}, {"name": "Jane"}] + + # Invalid - array with non-objects + with pytest.raises(ValueError, match="Expected array of objects, got list"): + WebhookService._validate_json_value("users", [{"name": "John"}, "not_object"], "array[object]") + + def test_convert_form_value_string(self): + """Test form value conversion for string type.""" + result = WebhookService._convert_form_value("test", "hello", "string") + assert result == "hello" + + def test_convert_form_value_number(self): + """Test form value conversion for number type.""" + # Integer + result = WebhookService._convert_form_value("count", "42", "number") + assert result == 42 + + # Float + result = WebhookService._convert_form_value("price", "19.99", "number") + assert result == 19.99 + + # Invalid number + with pytest.raises(ValueError, match="Cannot convert 'not_a_number' to number"): + WebhookService._convert_form_value("count", "not_a_number", "number") + + def test_convert_form_value_boolean(self): + """Test form value conversion for boolean type.""" + # True values + assert WebhookService._convert_form_value("flag", "true", "boolean") is True + assert WebhookService._convert_form_value("flag", "1", "boolean") is True + assert WebhookService._convert_form_value("flag", "yes", "boolean") is True + + # False values + assert WebhookService._convert_form_value("flag", "false", "boolean") is False + assert WebhookService._convert_form_value("flag", "0", "boolean") is False + assert WebhookService._convert_form_value("flag", "no", "boolean") is False + + # Invalid boolean + with pytest.raises(ValueError, match="Cannot convert 'maybe' to boolean"): + WebhookService._convert_form_value("flag", "maybe", "boolean") + + def test_extract_and_validate_webhook_data_success(self): + """Test successful unified data extraction and validation.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="POST", + headers={"Content-Type": "application/json"}, + query_string="count=42&enabled=true", + json={"message": "hello", "age": 25}, + ): + webhook_trigger = MagicMock() + node_config = { + "data": { + "method": "post", + "content_type": "application/json", + "params": [ + {"name": "count", "type": "number", "required": True}, + {"name": "enabled", "type": "boolean", "required": True}, + ], + "body": [ + {"name": "message", "type": "string", "required": True}, + {"name": "age", "type": "number", "required": True}, + ], + } + } + + result = WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + # Check that types are correctly converted + assert result["query_params"]["count"] == 42 # Converted to int + assert result["query_params"]["enabled"] is True # Converted to bool + assert result["body"]["message"] == "hello" # Already string + assert result["body"]["age"] == 25 # Already number + + def test_extract_and_validate_webhook_data_validation_error(self): + """Test unified data extraction with validation error.""" + app = Flask(__name__) + + with app.test_request_context( + "/webhook", + method="GET", # Wrong method + headers={"Content-Type": "application/json"}, + ): + webhook_trigger = MagicMock() + node_config = { + "data": { + "method": "post", # Expects POST + "content_type": "application/json", + } + } + + with pytest.raises(ValueError, match="HTTP method mismatch"): + WebhookService.extract_and_validate_webhook_data(webhook_trigger, node_config) + + def test_debug_mode_parameter_handling(self): + """Test that the debug mode parameter is properly handled in _prepare_webhook_execution.""" + from controllers.trigger.webhook import _prepare_webhook_execution + + # Mock the WebhookService methods + with ( + patch.object(WebhookService, "get_webhook_trigger_and_workflow") as mock_get_trigger, + patch.object(WebhookService, "extract_and_validate_webhook_data") as mock_extract, + ): + mock_trigger = MagicMock() + mock_workflow = MagicMock() + mock_config = {"data": {"test": "config"}} + mock_data = {"test": "data"} + + mock_get_trigger.return_value = (mock_trigger, mock_workflow, mock_config) + mock_extract.return_value = mock_data + + result = _prepare_webhook_execution("test_webhook", is_debug=False) + assert result == (mock_trigger, mock_workflow, mock_config, mock_data, None) + + # Reset mock + mock_get_trigger.reset_mock() + + result = _prepare_webhook_execution("test_webhook", is_debug=True) + assert result == (mock_trigger, mock_workflow, mock_config, mock_data, None) diff --git a/api/tests/unit_tests/services/test_workflow_run_service_pause.py b/api/tests/unit_tests/services/test_workflow_run_service_pause.py new file mode 100644 index 0000000000..a062d9444e --- /dev/null +++ b/api/tests/unit_tests/services/test_workflow_run_service_pause.py @@ -0,0 +1,200 @@ +"""Comprehensive unit tests for WorkflowRunService class. + +This test suite covers all pause state management operations including: +- Retrieving pause state for workflow runs +- Saving pause state with file uploads +- Marking paused workflows as resumed +- Error handling and edge cases +- Database transaction management +- Repository-based approach testing +""" + +from datetime import datetime +from unittest.mock import MagicMock, create_autospec, patch + +import pytest +from sqlalchemy import Engine +from sqlalchemy.orm import Session, sessionmaker + +from core.workflow.enums import WorkflowExecutionStatus +from repositories.api_workflow_run_repository import APIWorkflowRunRepository +from repositories.sqlalchemy_api_workflow_run_repository import _PrivateWorkflowPauseEntity +from services.workflow_run_service import ( + WorkflowRunService, +) + + +class TestDataFactory: + """Factory class for creating test data objects.""" + + @staticmethod + def create_workflow_run_mock( + id: str = "workflow-run-123", + tenant_id: str = "tenant-456", + app_id: str = "app-789", + workflow_id: str = "workflow-101", + status: str | WorkflowExecutionStatus = "paused", + pause_id: str | None = None, + **kwargs, + ) -> MagicMock: + """Create a mock WorkflowRun object.""" + mock_run = MagicMock() + mock_run.id = id + mock_run.tenant_id = tenant_id + mock_run.app_id = app_id + mock_run.workflow_id = workflow_id + mock_run.status = status + mock_run.pause_id = pause_id + + for key, value in kwargs.items(): + setattr(mock_run, key, value) + + return mock_run + + @staticmethod + def create_workflow_pause_mock( + id: str = "pause-123", + tenant_id: str = "tenant-456", + app_id: str = "app-789", + workflow_id: str = "workflow-101", + workflow_execution_id: str = "workflow-execution-123", + state_file_id: str = "file-456", + resumed_at: datetime | None = None, + **kwargs, + ) -> MagicMock: + """Create a mock WorkflowPauseModel object.""" + mock_pause = MagicMock() + mock_pause.id = id + mock_pause.tenant_id = tenant_id + mock_pause.app_id = app_id + mock_pause.workflow_id = workflow_id + mock_pause.workflow_execution_id = workflow_execution_id + mock_pause.state_file_id = state_file_id + mock_pause.resumed_at = resumed_at + + for key, value in kwargs.items(): + setattr(mock_pause, key, value) + + return mock_pause + + @staticmethod + def create_upload_file_mock( + id: str = "file-456", + key: str = "upload_files/test/state.json", + name: str = "state.json", + tenant_id: str = "tenant-456", + **kwargs, + ) -> MagicMock: + """Create a mock UploadFile object.""" + mock_file = MagicMock() + mock_file.id = id + mock_file.key = key + mock_file.name = name + mock_file.tenant_id = tenant_id + + for key, value in kwargs.items(): + setattr(mock_file, key, value) + + return mock_file + + @staticmethod + def create_pause_entity_mock( + pause_model: MagicMock | None = None, + upload_file: MagicMock | None = None, + ) -> _PrivateWorkflowPauseEntity: + """Create a mock _PrivateWorkflowPauseEntity object.""" + if pause_model is None: + pause_model = TestDataFactory.create_workflow_pause_mock() + if upload_file is None: + upload_file = TestDataFactory.create_upload_file_mock() + + return _PrivateWorkflowPauseEntity.from_models(pause_model, upload_file) + + +class TestWorkflowRunService: + """Comprehensive unit tests for WorkflowRunService class.""" + + @pytest.fixture + def mock_session_factory(self): + """Create a mock session factory with proper session management.""" + mock_session = create_autospec(Session) + + # Create a mock context manager for the session + mock_session_cm = MagicMock() + mock_session_cm.__enter__ = MagicMock(return_value=mock_session) + mock_session_cm.__exit__ = MagicMock(return_value=None) + + # Create a mock context manager for the transaction + mock_transaction_cm = MagicMock() + mock_transaction_cm.__enter__ = MagicMock(return_value=mock_session) + mock_transaction_cm.__exit__ = MagicMock(return_value=None) + + mock_session.begin = MagicMock(return_value=mock_transaction_cm) + + # Create mock factory that returns the context manager + mock_factory = MagicMock(spec=sessionmaker) + mock_factory.return_value = mock_session_cm + + return mock_factory, mock_session + + @pytest.fixture + def mock_workflow_run_repository(self): + """Create a mock APIWorkflowRunRepository.""" + mock_repo = create_autospec(APIWorkflowRunRepository) + return mock_repo + + @pytest.fixture + def workflow_run_service(self, mock_session_factory, mock_workflow_run_repository): + """Create WorkflowRunService instance with mocked dependencies.""" + session_factory, _ = mock_session_factory + + with patch("services.workflow_run_service.DifyAPIRepositoryFactory") as mock_factory: + mock_factory.create_api_workflow_run_repository.return_value = mock_workflow_run_repository + service = WorkflowRunService(session_factory) + return service + + @pytest.fixture + def workflow_run_service_with_engine(self, mock_session_factory, mock_workflow_run_repository): + """Create WorkflowRunService instance with Engine input.""" + mock_engine = create_autospec(Engine) + session_factory, _ = mock_session_factory + + with patch("services.workflow_run_service.DifyAPIRepositoryFactory") as mock_factory: + mock_factory.create_api_workflow_run_repository.return_value = mock_workflow_run_repository + service = WorkflowRunService(mock_engine) + return service + + # ==================== Initialization Tests ==================== + + def test_init_with_session_factory(self, mock_session_factory, mock_workflow_run_repository): + """Test WorkflowRunService initialization with session_factory.""" + session_factory, _ = mock_session_factory + + with patch("services.workflow_run_service.DifyAPIRepositoryFactory") as mock_factory: + mock_factory.create_api_workflow_run_repository.return_value = mock_workflow_run_repository + service = WorkflowRunService(session_factory) + + assert service._session_factory == session_factory + mock_factory.create_api_workflow_run_repository.assert_called_once_with(session_factory) + + def test_init_with_engine(self, mock_session_factory, mock_workflow_run_repository): + """Test WorkflowRunService initialization with Engine (should convert to sessionmaker).""" + mock_engine = create_autospec(Engine) + session_factory, _ = mock_session_factory + + with patch("services.workflow_run_service.DifyAPIRepositoryFactory") as mock_factory: + mock_factory.create_api_workflow_run_repository.return_value = mock_workflow_run_repository + with patch("services.workflow_run_service.sessionmaker", return_value=session_factory) as mock_sessionmaker: + service = WorkflowRunService(mock_engine) + + mock_sessionmaker.assert_called_once_with(bind=mock_engine, expire_on_commit=False) + assert service._session_factory == session_factory + mock_factory.create_api_workflow_run_repository.assert_called_once_with(session_factory) + + def test_init_with_default_dependencies(self, mock_session_factory): + """Test WorkflowRunService initialization with default dependencies.""" + session_factory, _ = mock_session_factory + + service = WorkflowRunService(session_factory) + + assert service._session_factory == session_factory diff --git a/api/tests/unit_tests/services/tools/test_mcp_tools_transform.py b/api/tests/unit_tests/services/tools/test_mcp_tools_transform.py index fb0139932b..7511fd6f0c 100644 --- a/api/tests/unit_tests/services/tools/test_mcp_tools_transform.py +++ b/api/tests/unit_tests/services/tools/test_mcp_tools_transform.py @@ -180,6 +180,25 @@ class TestMCPToolTransform: # Set tools data with null description mock_provider_full.tools = '[{"name": "tool1", "description": null, "inputSchema": {}}]' + # Mock the to_entity and to_api_response methods + mock_entity = Mock() + mock_entity.to_api_response.return_value = { + "name": "Test MCP Provider", + "type": ToolProviderType.MCP, + "is_team_authorization": True, + "server_url": "https://*****.com/mcp", + "provider_icon": "icon.png", + "masked_headers": {"Authorization": "Bearer *****"}, + "updated_at": 1234567890, + "labels": [], + "author": "Test User", + "description": I18nObject(en_US="Test MCP Provider Description", zh_Hans="Test MCP Provider Description"), + "icon": "icon.png", + "label": I18nObject(en_US="Test MCP Provider", zh_Hans="Test MCP Provider"), + "masked_credentials": {}, + } + mock_provider_full.to_entity.return_value = mock_entity + # Call the method with for_list=True result = ToolTransformService.mcp_provider_to_user_provider(mock_provider_full, for_list=True) @@ -198,6 +217,27 @@ class TestMCPToolTransform: # Set tools data with description mock_provider_full.tools = '[{"name": "tool1", "description": "Tool description", "inputSchema": {}}]' + # Mock the to_entity and to_api_response methods + mock_entity = Mock() + mock_entity.to_api_response.return_value = { + "name": "Test MCP Provider", + "type": ToolProviderType.MCP, + "is_team_authorization": True, + "server_url": "https://*****.com/mcp", + "provider_icon": "icon.png", + "masked_headers": {"Authorization": "Bearer *****"}, + "updated_at": 1234567890, + "labels": [], + "configuration": {"timeout": "30", "sse_read_timeout": "300"}, + "original_headers": {"Authorization": "Bearer secret-token"}, + "author": "Test User", + "description": I18nObject(en_US="Test MCP Provider Description", zh_Hans="Test MCP Provider Description"), + "icon": "icon.png", + "label": I18nObject(en_US="Test MCP Provider", zh_Hans="Test MCP Provider"), + "masked_credentials": {}, + } + mock_provider_full.to_entity.return_value = mock_entity + # Call the method with for_list=False result = ToolTransformService.mcp_provider_to_user_provider(mock_provider_full, for_list=False) @@ -205,8 +245,9 @@ class TestMCPToolTransform: assert isinstance(result, ToolProviderApiEntity) assert result.id == "server-identifier-456" # Should use server_identifier when for_list=False assert result.server_identifier == "server-identifier-456" - assert result.timeout == 30 - assert result.sse_read_timeout == 300 + assert result.configuration is not None + assert result.configuration.timeout == 30 + assert result.configuration.sse_read_timeout == 300 assert result.original_headers == {"Authorization": "Bearer secret-token"} assert len(result.tools) == 1 assert result.tools[0].description.en_US == "Tool description" diff --git a/api/tests/unit_tests/services/workflow/test_workflow_converter.py b/api/tests/unit_tests/services/workflow/test_workflow_converter.py index 63ce4c0c3c..8ea5754363 100644 --- a/api/tests/unit_tests/services/workflow/test_workflow_converter.py +++ b/api/tests/unit_tests/services/workflow/test_workflow_converter.py @@ -199,6 +199,7 @@ def test__convert_to_knowledge_retrieval_node_for_chatbot(): node = WorkflowConverter()._convert_to_knowledge_retrieval_node( new_app_mode=new_app_mode, dataset_config=dataset_config, model_config=model_config ) + assert node is not None assert node["data"]["type"] == "knowledge-retrieval" assert node["data"]["query_variable_selector"] == ["sys", "query"] @@ -231,6 +232,7 @@ def test__convert_to_knowledge_retrieval_node_for_workflow_app(): node = WorkflowConverter()._convert_to_knowledge_retrieval_node( new_app_mode=new_app_mode, dataset_config=dataset_config, model_config=model_config ) + assert node is not None assert node["data"]["type"] == "knowledge-retrieval" assert node["data"]["query_variable_selector"] == ["start", dataset_config.retrieve_config.query_variable] diff --git a/api/uv.lock b/api/uv.lock index 066f9a58a4..6300adae61 100644 --- a/api/uv.lock +++ b/api/uv.lock @@ -48,7 +48,7 @@ wheels = [ [[package]] name = "aiohttp" -version = "3.13.0" +version = "3.13.2" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "aiohappyeyeballs" }, @@ -59,54 +59,54 @@ dependencies = [ { name = "propcache" }, { name = "yarl" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/62/f1/8515650ac3121a9e55c7b217c60e7fae3e0134b5acfe65691781b5356929/aiohttp-3.13.0.tar.gz", hash = "sha256:378dbc57dd8cf341ce243f13fa1fa5394d68e2e02c15cd5f28eae35a70ec7f67", size = 7832348, upload-time = "2025-10-06T19:58:48.089Z" } +sdist = { url = "https://files.pythonhosted.org/packages/1c/ce/3b83ebba6b3207a7135e5fcaba49706f8a4b6008153b4e30540c982fae26/aiohttp-3.13.2.tar.gz", hash = "sha256:40176a52c186aefef6eb3cad2cdd30cd06e3afbe88fe8ab2af9c0b90f228daca", size = 7837994, upload-time = "2025-10-28T20:59:39.937Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/b1/db/df80cacac46cd548a736c5535b13cc18925cf6f9f83cd128cf3839842219/aiohttp-3.13.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:99eb94e97a42367fef5fc11e28cb2362809d3e70837f6e60557816c7106e2e20", size = 741374, upload-time = "2025-10-06T19:55:13.095Z" }, - { url = "https://files.pythonhosted.org/packages/ae/f9/2d6d93fd57ab4726e18a7cdab083772eda8302d682620fbf2aef48322351/aiohttp-3.13.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:4696665b2713021c6eba3e2b882a86013763b442577fe5d2056a42111e732eca", size = 494956, upload-time = "2025-10-06T19:55:14.687Z" }, - { url = "https://files.pythonhosted.org/packages/89/a6/e1c061b079fed04ffd6777950c82f2e8246fd08b7b3c4f56fdd47f697e5a/aiohttp-3.13.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3e6a38366f7f0d0f6ed7a1198055150c52fda552b107dad4785c0852ad7685d1", size = 491154, upload-time = "2025-10-06T19:55:16.661Z" }, - { url = "https://files.pythonhosted.org/packages/fe/4d/ee8913c0d2c7da37fdc98673a342b51611eaa0871682b37b8430084e35b5/aiohttp-3.13.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:aab715b1a0c37f7f11f9f1f579c6fbaa51ef569e47e3c0a4644fba46077a9409", size = 1745707, upload-time = "2025-10-06T19:55:18.376Z" }, - { url = "https://files.pythonhosted.org/packages/f9/70/26b2c97e8fa68644aec43d788940984c5f3b53a8d1468d5baaa328f809c9/aiohttp-3.13.0-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:7972c82bed87d7bd8e374b60a6b6e816d75ba4f7c2627c2d14eed216e62738e1", size = 1702404, upload-time = "2025-10-06T19:55:20.098Z" }, - { url = "https://files.pythonhosted.org/packages/65/1e/c8aa3c293a0e8b18968b1b88e9bd8fb269eb67eb7449f504a4c3e175b159/aiohttp-3.13.0-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:ca8313cb852af788c78d5afdea24c40172cbfff8b35e58b407467732fde20390", size = 1805519, upload-time = "2025-10-06T19:55:21.811Z" }, - { url = "https://files.pythonhosted.org/packages/51/b6/a3753fe86249eb441768658cfc00f8c4e0913b255c13be00ddb8192775e1/aiohttp-3.13.0-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:6c333a2385d2a6298265f4b3e960590f787311b87f6b5e6e21bb8375914ef504", size = 1893904, upload-time = "2025-10-06T19:55:23.462Z" }, - { url = "https://files.pythonhosted.org/packages/51/6d/7b1e020fe1d2a2be7cf0ce5e35922f345e3507cf337faa1a6563c42065c1/aiohttp-3.13.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:cc6d5fc5edbfb8041d9607f6a417997fa4d02de78284d386bea7ab767b5ea4f3", size = 1745043, upload-time = "2025-10-06T19:55:25.208Z" }, - { url = "https://files.pythonhosted.org/packages/e6/df/aad5dce268f9d4f29759c3eeb5fb5995c569d76abb267468dc1075218d5b/aiohttp-3.13.0-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:7ddedba3d0043349edc79df3dc2da49c72b06d59a45a42c1c8d987e6b8d175b8", size = 1604765, upload-time = "2025-10-06T19:55:27.157Z" }, - { url = "https://files.pythonhosted.org/packages/1c/19/a84a0e97b2da2224c8b85e1aef5cac834d07b2903c17bff1a6bdbc7041d2/aiohttp-3.13.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:23ca762140159417a6bbc959ca1927f6949711851e56f2181ddfe8d63512b5ad", size = 1721737, upload-time = "2025-10-06T19:55:28.854Z" }, - { url = "https://files.pythonhosted.org/packages/6c/61/ca6ad390128d964a08554fd63d6df5810fb5fbc7e599cb9e617f1729ae19/aiohttp-3.13.0-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:bfe824d6707a5dc3c5676685f624bc0c63c40d79dc0239a7fd6c034b98c25ebe", size = 1716052, upload-time = "2025-10-06T19:55:30.563Z" }, - { url = "https://files.pythonhosted.org/packages/2a/71/769e249e6625372c7d14be79b8b8c3b0592963a09793fb3d36758e60952c/aiohttp-3.13.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:3c11fa5dd2ef773a8a5a6daa40243d83b450915992eab021789498dc87acc114", size = 1783532, upload-time = "2025-10-06T19:55:32.798Z" }, - { url = "https://files.pythonhosted.org/packages/66/64/b9cd03cdbb629bc492e4a744fbe96550a8340b0cd7a0cc4a9c90cfecd8d3/aiohttp-3.13.0-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:00fdfe370cffede3163ba9d3f190b32c0cfc8c774f6f67395683d7b0e48cdb8a", size = 1593072, upload-time = "2025-10-06T19:55:34.686Z" }, - { url = "https://files.pythonhosted.org/packages/24/0e/87922c8cfdbd09f5e2197e9d87714a98c99c423560d44739e3af55400fe3/aiohttp-3.13.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:6475e42ef92717a678bfbf50885a682bb360a6f9c8819fb1a388d98198fdcb80", size = 1798613, upload-time = "2025-10-06T19:55:36.393Z" }, - { url = "https://files.pythonhosted.org/packages/c5/bb/a3adfe2af76e1ee9e3b5464522004b148b266bc99d7ec424ca7843d64a3c/aiohttp-3.13.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:77da5305a410910218b99f2a963092f4277d8a9c1f429c1ff1b026d1826bd0b6", size = 1737480, upload-time = "2025-10-06T19:55:38.043Z" }, - { url = "https://files.pythonhosted.org/packages/ad/53/e124dcbd64e6365602f3493fe37a11ca5b7ac0a40822a6e2bc8260cd08e0/aiohttp-3.13.0-cp311-cp311-win32.whl", hash = "sha256:2f9d9ea547618d907f2ee6670c9a951f059c5994e4b6de8dcf7d9747b420c820", size = 429824, upload-time = "2025-10-06T19:55:39.595Z" }, - { url = "https://files.pythonhosted.org/packages/3e/bd/485d98b372a2cd6998484a93ddd401ec6b6031657661c36846a10e2a1f6e/aiohttp-3.13.0-cp311-cp311-win_amd64.whl", hash = "sha256:0f19f7798996d4458c669bd770504f710014926e9970f4729cf55853ae200469", size = 454137, upload-time = "2025-10-06T19:55:41.617Z" }, - { url = "https://files.pythonhosted.org/packages/3a/95/7e8bdfa6e79099a086d59d42589492f1fe9d29aae3cefb58b676015ce278/aiohttp-3.13.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:1c272a9a18a5ecc48a7101882230046b83023bb2a662050ecb9bfcb28d9ab53a", size = 735585, upload-time = "2025-10-06T19:55:43.401Z" }, - { url = "https://files.pythonhosted.org/packages/9f/20/2f1d3ee06ee94eafe516810705219bff234d09f135d6951661661d5595ae/aiohttp-3.13.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:97891a23d7fd4e1afe9c2f4473e04595e4acb18e4733b910b6577b74e7e21985", size = 490613, upload-time = "2025-10-06T19:55:45.237Z" }, - { url = "https://files.pythonhosted.org/packages/74/15/ab8600ef6dc1dcd599009a81acfed2ea407037e654d32e47e344e0b08c34/aiohttp-3.13.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:475bd56492ce5f4cffe32b5533c6533ee0c406d1d0e6924879f83adcf51da0ae", size = 489750, upload-time = "2025-10-06T19:55:46.937Z" }, - { url = "https://files.pythonhosted.org/packages/33/59/752640c2b86ca987fe5703a01733b00d375e6cd2392bc7574489934e64e5/aiohttp-3.13.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:c32ada0abb4bc94c30be2b681c42f058ab104d048da6f0148280a51ce98add8c", size = 1736812, upload-time = "2025-10-06T19:55:48.917Z" }, - { url = "https://files.pythonhosted.org/packages/3d/c6/dd6b86ddb852a7fdbcdc7a45b6bdc80178aef713c08279afcaee7a5a9f07/aiohttp-3.13.0-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:4af1f8877ca46ecdd0bc0d4a6b66d4b2bddc84a79e2e8366bc0d5308e76bceb8", size = 1698535, upload-time = "2025-10-06T19:55:50.75Z" }, - { url = "https://files.pythonhosted.org/packages/33/e2/27c92d205b9e8cee7661670e8e9f187931b71e26d42796b153d2a0ba6949/aiohttp-3.13.0-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:e04ab827ec4f775817736b20cdc8350f40327f9b598dec4e18c9ffdcbea88a93", size = 1766573, upload-time = "2025-10-06T19:55:53.106Z" }, - { url = "https://files.pythonhosted.org/packages/df/6a/1fc1ad71d130a30f7a207d8d958a41224c29b834463b5185efb2dbff6ad4/aiohttp-3.13.0-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a6d9487b9471ec36b0faedf52228cd732e89be0a2bbd649af890b5e2ce422353", size = 1865229, upload-time = "2025-10-06T19:55:55.01Z" }, - { url = "https://files.pythonhosted.org/packages/14/51/d0c1701a79fcb0109cff5304da16226581569b89a282d8e7f1549a7e3ec0/aiohttp-3.13.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2e66c57416352f36bf98f6641ddadd47c93740a22af7150d3e9a1ef6e983f9a8", size = 1750379, upload-time = "2025-10-06T19:55:57.219Z" }, - { url = "https://files.pythonhosted.org/packages/ae/3d/2ec4b934f85856de1c0c18e90adc8902adadbfac2b3c0b831bfeb7214fc8/aiohttp-3.13.0-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:469167d5372f5bb3aedff4fc53035d593884fff2617a75317740e885acd48b04", size = 1560798, upload-time = "2025-10-06T19:55:58.888Z" }, - { url = "https://files.pythonhosted.org/packages/38/56/e23d9c3e13006e599fdce3851517c70279e177871e3e567d22cf3baf5d6c/aiohttp-3.13.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:a9f3546b503975a69b547c9fd1582cad10ede1ce6f3e313a2f547c73a3d7814f", size = 1697552, upload-time = "2025-10-06T19:56:01.172Z" }, - { url = "https://files.pythonhosted.org/packages/56/cb/caa32c2ccaeca0a3dc39129079fd2ad02f9406c3a5f7924340435b87d4cd/aiohttp-3.13.0-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:6b4174fcec98601f0cfdf308ee29a6ae53c55f14359e848dab4e94009112ee7d", size = 1718609, upload-time = "2025-10-06T19:56:03.102Z" }, - { url = "https://files.pythonhosted.org/packages/fb/c0/5911856fef9e40fd1ccbb8c54a90116875d5753a92c1cac66ce2059b390d/aiohttp-3.13.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:a533873a7a4ec2270fb362ee5a0d3b98752e4e1dc9042b257cd54545a96bd8ed", size = 1735887, upload-time = "2025-10-06T19:56:04.841Z" }, - { url = "https://files.pythonhosted.org/packages/0e/48/8d6f4757a24c02f0a454c043556593a00645d10583859f7156db44d8b7d3/aiohttp-3.13.0-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:ce887c5e54411d607ee0959cac15bb31d506d86a9bcaddf0b7e9d63325a7a802", size = 1553079, upload-time = "2025-10-06T19:56:07.197Z" }, - { url = "https://files.pythonhosted.org/packages/39/fa/e82c9445e40b50e46770702b5b6ca2f767966d53e1a5eef03583ceac6df6/aiohttp-3.13.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:d871f6a30d43e32fc9252dc7b9febe1a042b3ff3908aa83868d7cf7c9579a59b", size = 1762750, upload-time = "2025-10-06T19:56:09.376Z" }, - { url = "https://files.pythonhosted.org/packages/3d/e6/9d30554e7f1e700bfeae4ab6b153d5dc7441606a9ec5e929288fa93a1477/aiohttp-3.13.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:222c828243b4789d79a706a876910f656fad4381661691220ba57b2ab4547865", size = 1717461, upload-time = "2025-10-06T19:56:11.551Z" }, - { url = "https://files.pythonhosted.org/packages/1f/e5/29cca547990a59ea54f0674fc01de98519fc628cfceeab6175711750eca7/aiohttp-3.13.0-cp312-cp312-win32.whl", hash = "sha256:682d2e434ff2f1108314ff7f056ce44e457f12dbed0249b24e106e385cf154b9", size = 424633, upload-time = "2025-10-06T19:56:13.316Z" }, - { url = "https://files.pythonhosted.org/packages/8b/68/46dd042d7bc62eab30bafdb8569f55ef125c3a88bb174270324224f8df56/aiohttp-3.13.0-cp312-cp312-win_amd64.whl", hash = "sha256:0a2be20eb23888df130214b91c262a90e2de1553d6fb7de9e9010cec994c0ff2", size = 451401, upload-time = "2025-10-06T19:56:15.188Z" }, + { url = "https://files.pythonhosted.org/packages/35/74/b321e7d7ca762638cdf8cdeceb39755d9c745aff7a64c8789be96ddf6e96/aiohttp-3.13.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:4647d02df098f6434bafd7f32ad14942f05a9caa06c7016fdcc816f343997dd0", size = 743409, upload-time = "2025-10-28T20:56:00.354Z" }, + { url = "https://files.pythonhosted.org/packages/99/3d/91524b905ec473beaf35158d17f82ef5a38033e5809fe8742e3657cdbb97/aiohttp-3.13.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:e3403f24bcb9c3b29113611c3c16a2a447c3953ecf86b79775e7be06f7ae7ccb", size = 497006, upload-time = "2025-10-28T20:56:01.85Z" }, + { url = "https://files.pythonhosted.org/packages/eb/d3/7f68bc02a67716fe80f063e19adbd80a642e30682ce74071269e17d2dba1/aiohttp-3.13.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:43dff14e35aba17e3d6d5ba628858fb8cb51e30f44724a2d2f0c75be492c55e9", size = 493195, upload-time = "2025-10-28T20:56:03.314Z" }, + { url = "https://files.pythonhosted.org/packages/98/31/913f774a4708775433b7375c4f867d58ba58ead833af96c8af3621a0d243/aiohttp-3.13.2-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:e2a9ea08e8c58bb17655630198833109227dea914cd20be660f52215f6de5613", size = 1747759, upload-time = "2025-10-28T20:56:04.904Z" }, + { url = "https://files.pythonhosted.org/packages/e8/63/04efe156f4326f31c7c4a97144f82132c3bb21859b7bb84748d452ccc17c/aiohttp-3.13.2-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:53b07472f235eb80e826ad038c9d106c2f653584753f3ddab907c83f49eedead", size = 1704456, upload-time = "2025-10-28T20:56:06.986Z" }, + { url = "https://files.pythonhosted.org/packages/8e/02/4e16154d8e0a9cf4ae76f692941fd52543bbb148f02f098ca73cab9b1c1b/aiohttp-3.13.2-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:e736c93e9c274fce6419af4aac199984d866e55f8a4cec9114671d0ea9688780", size = 1807572, upload-time = "2025-10-28T20:56:08.558Z" }, + { url = "https://files.pythonhosted.org/packages/34/58/b0583defb38689e7f06798f0285b1ffb3a6fb371f38363ce5fd772112724/aiohttp-3.13.2-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:ff5e771f5dcbc81c64898c597a434f7682f2259e0cd666932a913d53d1341d1a", size = 1895954, upload-time = "2025-10-28T20:56:10.545Z" }, + { url = "https://files.pythonhosted.org/packages/6b/f3/083907ee3437425b4e376aa58b2c915eb1a33703ec0dc30040f7ae3368c6/aiohttp-3.13.2-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a3b6fb0c207cc661fa0bf8c66d8d9b657331ccc814f4719468af61034b478592", size = 1747092, upload-time = "2025-10-28T20:56:12.118Z" }, + { url = "https://files.pythonhosted.org/packages/ac/61/98a47319b4e425cc134e05e5f3fc512bf9a04bf65aafd9fdcda5d57ec693/aiohttp-3.13.2-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:97a0895a8e840ab3520e2288db7cace3a1981300d48babeb50e7425609e2e0ab", size = 1606815, upload-time = "2025-10-28T20:56:14.191Z" }, + { url = "https://files.pythonhosted.org/packages/97/4b/e78b854d82f66bb974189135d31fce265dee0f5344f64dd0d345158a5973/aiohttp-3.13.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:9e8f8afb552297aca127c90cb840e9a1d4bfd6a10d7d8f2d9176e1acc69bad30", size = 1723789, upload-time = "2025-10-28T20:56:16.101Z" }, + { url = "https://files.pythonhosted.org/packages/ed/fc/9d2ccc794fc9b9acd1379d625c3a8c64a45508b5091c546dea273a41929e/aiohttp-3.13.2-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:ed2f9c7216e53c3df02264f25d824b079cc5914f9e2deba94155190ef648ee40", size = 1718104, upload-time = "2025-10-28T20:56:17.655Z" }, + { url = "https://files.pythonhosted.org/packages/66/65/34564b8765ea5c7d79d23c9113135d1dd3609173da13084830f1507d56cf/aiohttp-3.13.2-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:99c5280a329d5fa18ef30fd10c793a190d996567667908bef8a7f81f8202b948", size = 1785584, upload-time = "2025-10-28T20:56:19.238Z" }, + { url = "https://files.pythonhosted.org/packages/30/be/f6a7a426e02fc82781afd62016417b3948e2207426d90a0e478790d1c8a4/aiohttp-3.13.2-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:2ca6ffef405fc9c09a746cb5d019c1672cd7f402542e379afc66b370833170cf", size = 1595126, upload-time = "2025-10-28T20:56:20.836Z" }, + { url = "https://files.pythonhosted.org/packages/e5/c7/8e22d5d28f94f67d2af496f14a83b3c155d915d1fe53d94b66d425ec5b42/aiohttp-3.13.2-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:47f438b1a28e926c37632bff3c44df7d27c9b57aaf4e34b1def3c07111fdb782", size = 1800665, upload-time = "2025-10-28T20:56:22.922Z" }, + { url = "https://files.pythonhosted.org/packages/d1/11/91133c8b68b1da9fc16555706aa7276fdf781ae2bb0876c838dd86b8116e/aiohttp-3.13.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:9acda8604a57bb60544e4646a4615c1866ee6c04a8edef9b8ee6fd1d8fa2ddc8", size = 1739532, upload-time = "2025-10-28T20:56:25.924Z" }, + { url = "https://files.pythonhosted.org/packages/17/6b/3747644d26a998774b21a616016620293ddefa4d63af6286f389aedac844/aiohttp-3.13.2-cp311-cp311-win32.whl", hash = "sha256:868e195e39b24aaa930b063c08bb0c17924899c16c672a28a65afded9c46c6ec", size = 431876, upload-time = "2025-10-28T20:56:27.524Z" }, + { url = "https://files.pythonhosted.org/packages/c3/63/688462108c1a00eb9f05765331c107f95ae86f6b197b865d29e930b7e462/aiohttp-3.13.2-cp311-cp311-win_amd64.whl", hash = "sha256:7fd19df530c292542636c2a9a85854fab93474396a52f1695e799186bbd7f24c", size = 456205, upload-time = "2025-10-28T20:56:29.062Z" }, + { url = "https://files.pythonhosted.org/packages/29/9b/01f00e9856d0a73260e86dd8ed0c2234a466c5c1712ce1c281548df39777/aiohttp-3.13.2-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:b1e56bab2e12b2b9ed300218c351ee2a3d8c8fdab5b1ec6193e11a817767e47b", size = 737623, upload-time = "2025-10-28T20:56:30.797Z" }, + { url = "https://files.pythonhosted.org/packages/5a/1b/4be39c445e2b2bd0aab4ba736deb649fabf14f6757f405f0c9685019b9e9/aiohttp-3.13.2-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:364e25edaabd3d37b1db1f0cbcee8c73c9a3727bfa262b83e5e4cf3489a2a9dc", size = 492664, upload-time = "2025-10-28T20:56:32.708Z" }, + { url = "https://files.pythonhosted.org/packages/28/66/d35dcfea8050e131cdd731dff36434390479b4045a8d0b9d7111b0a968f1/aiohttp-3.13.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:c5c94825f744694c4b8db20b71dba9a257cd2ba8e010a803042123f3a25d50d7", size = 491808, upload-time = "2025-10-28T20:56:34.57Z" }, + { url = "https://files.pythonhosted.org/packages/00/29/8e4609b93e10a853b65f8291e64985de66d4f5848c5637cddc70e98f01f8/aiohttp-3.13.2-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ba2715d842ffa787be87cbfce150d5e88c87a98e0b62e0f5aa489169a393dbbb", size = 1738863, upload-time = "2025-10-28T20:56:36.377Z" }, + { url = "https://files.pythonhosted.org/packages/9d/fa/4ebdf4adcc0def75ced1a0d2d227577cd7b1b85beb7edad85fcc87693c75/aiohttp-3.13.2-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:585542825c4bc662221fb257889e011a5aa00f1ae4d75d1d246a5225289183e3", size = 1700586, upload-time = "2025-10-28T20:56:38.034Z" }, + { url = "https://files.pythonhosted.org/packages/da/04/73f5f02ff348a3558763ff6abe99c223381b0bace05cd4530a0258e52597/aiohttp-3.13.2-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:39d02cb6025fe1aabca329c5632f48c9532a3dabccd859e7e2f110668972331f", size = 1768625, upload-time = "2025-10-28T20:56:39.75Z" }, + { url = "https://files.pythonhosted.org/packages/f8/49/a825b79ffec124317265ca7d2344a86bcffeb960743487cb11988ffb3494/aiohttp-3.13.2-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:e67446b19e014d37342f7195f592a2a948141d15a312fe0e700c2fd2f03124f6", size = 1867281, upload-time = "2025-10-28T20:56:41.471Z" }, + { url = "https://files.pythonhosted.org/packages/b9/48/adf56e05f81eac31edcfae45c90928f4ad50ef2e3ea72cb8376162a368f8/aiohttp-3.13.2-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4356474ad6333e41ccefd39eae869ba15a6c5299c9c01dfdcfdd5c107be4363e", size = 1752431, upload-time = "2025-10-28T20:56:43.162Z" }, + { url = "https://files.pythonhosted.org/packages/30/ab/593855356eead019a74e862f21523db09c27f12fd24af72dbc3555b9bfd9/aiohttp-3.13.2-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:eeacf451c99b4525f700f078becff32c32ec327b10dcf31306a8a52d78166de7", size = 1562846, upload-time = "2025-10-28T20:56:44.85Z" }, + { url = "https://files.pythonhosted.org/packages/39/0f/9f3d32271aa8dc35036e9668e31870a9d3b9542dd6b3e2c8a30931cb27ae/aiohttp-3.13.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:d8a9b889aeabd7a4e9af0b7f4ab5ad94d42e7ff679aaec6d0db21e3b639ad58d", size = 1699606, upload-time = "2025-10-28T20:56:46.519Z" }, + { url = "https://files.pythonhosted.org/packages/2c/3c/52d2658c5699b6ef7692a3f7128b2d2d4d9775f2a68093f74bca06cf01e1/aiohttp-3.13.2-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:fa89cb11bc71a63b69568d5b8a25c3ca25b6d54c15f907ca1c130d72f320b76b", size = 1720663, upload-time = "2025-10-28T20:56:48.528Z" }, + { url = "https://files.pythonhosted.org/packages/9b/d4/8f8f3ff1fb7fb9e3f04fcad4e89d8a1cd8fc7d05de67e3de5b15b33008ff/aiohttp-3.13.2-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:8aa7c807df234f693fed0ecd507192fc97692e61fee5702cdc11155d2e5cadc8", size = 1737939, upload-time = "2025-10-28T20:56:50.77Z" }, + { url = "https://files.pythonhosted.org/packages/03/d3/ddd348f8a27a634daae39a1b8e291ff19c77867af438af844bf8b7e3231b/aiohttp-3.13.2-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:9eb3e33fdbe43f88c3c75fa608c25e7c47bbd80f48d012763cb67c47f39a7e16", size = 1555132, upload-time = "2025-10-28T20:56:52.568Z" }, + { url = "https://files.pythonhosted.org/packages/39/b8/46790692dc46218406f94374903ba47552f2f9f90dad554eed61bfb7b64c/aiohttp-3.13.2-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:9434bc0d80076138ea986833156c5a48c9c7a8abb0c96039ddbb4afc93184169", size = 1764802, upload-time = "2025-10-28T20:56:54.292Z" }, + { url = "https://files.pythonhosted.org/packages/ba/e4/19ce547b58ab2a385e5f0b8aa3db38674785085abcf79b6e0edd1632b12f/aiohttp-3.13.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:ff15c147b2ad66da1f2cbb0622313f2242d8e6e8f9b79b5206c84523a4473248", size = 1719512, upload-time = "2025-10-28T20:56:56.428Z" }, + { url = "https://files.pythonhosted.org/packages/70/30/6355a737fed29dcb6dfdd48682d5790cb5eab050f7b4e01f49b121d3acad/aiohttp-3.13.2-cp312-cp312-win32.whl", hash = "sha256:27e569eb9d9e95dbd55c0fc3ec3a9335defbf1d8bc1d20171a49f3c4c607b93e", size = 426690, upload-time = "2025-10-28T20:56:58.736Z" }, + { url = "https://files.pythonhosted.org/packages/0a/0d/b10ac09069973d112de6ef980c1f6bb31cb7dcd0bc363acbdad58f927873/aiohttp-3.13.2-cp312-cp312-win_amd64.whl", hash = "sha256:8709a0f05d59a71f33fd05c17fc11fcb8c30140506e13c2f5e8ee1b8964e1b45", size = 453465, upload-time = "2025-10-28T20:57:00.795Z" }, ] [[package]] name = "aiomysql" -version = "0.2.0" +version = "0.3.2" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "pymysql" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/67/76/2c5b55e4406a1957ffdfd933a94c2517455291c97d2b81cec6813754791a/aiomysql-0.2.0.tar.gz", hash = "sha256:558b9c26d580d08b8c5fd1be23c5231ce3aeff2dadad989540fee740253deb67", size = 114706, upload-time = "2023-06-11T19:57:53.608Z" } +sdist = { url = "https://files.pythonhosted.org/packages/29/e0/302aeffe8d90853556f47f3106b89c16cc2ec2a4d269bdfd82e3f4ae12cc/aiomysql-0.3.2.tar.gz", hash = "sha256:72d15ef5cfc34c03468eb41e1b90adb9fd9347b0b589114bd23ead569a02ac1a", size = 108311, upload-time = "2025-10-22T00:15:21.278Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/42/87/c982ee8b333c85b8ae16306387d703a1fcdfc81a2f3f15a24820ab1a512d/aiomysql-0.2.0-py3-none-any.whl", hash = "sha256:b7c26da0daf23a5ec5e0b133c03d20657276e4eae9b73e040b72787f6f6ade0a", size = 44215, upload-time = "2023-06-11T19:57:51.09Z" }, + { url = "https://files.pythonhosted.org/packages/4c/af/aae0153c3e28712adaf462328f6c7a3c196a1c1c27b491de4377dd3e6b52/aiomysql-0.3.2-py3-none-any.whl", hash = "sha256:c82c5ba04137d7afd5c693a258bea8ead2aad77101668044143a991e04632eb2", size = 71834, upload-time = "2025-10-22T00:15:15.905Z" }, ] [[package]] @@ -124,21 +124,21 @@ wheels = [ [[package]] name = "alembic" -version = "1.17.0" +version = "1.17.1" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "mako" }, { name = "sqlalchemy" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/6b/45/6f4555f2039f364c3ce31399529dcf48dd60726ff3715ad67f547d87dfd2/alembic-1.17.0.tar.gz", hash = "sha256:4652a0b3e19616b57d652b82bfa5e38bf5dbea0813eed971612671cb9e90c0fe", size = 1975526, upload-time = "2025-10-11T18:40:13.585Z" } +sdist = { url = "https://files.pythonhosted.org/packages/6e/b6/2a81d7724c0c124edc5ec7a167e85858b6fd31b9611c6fb8ecf617b7e2d3/alembic-1.17.1.tar.gz", hash = "sha256:8a289f6778262df31571d29cca4c7fbacd2f0f582ea0816f4c399b6da7528486", size = 1981285, upload-time = "2025-10-29T00:23:16.667Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/44/1f/38e29b06bfed7818ebba1f84904afdc8153ef7b6c7e0d8f3bc6643f5989c/alembic-1.17.0-py3-none-any.whl", hash = "sha256:80523bc437d41b35c5db7e525ad9d908f79de65c27d6a5a5eab6df348a352d99", size = 247449, upload-time = "2025-10-11T18:40:16.288Z" }, + { url = "https://files.pythonhosted.org/packages/a5/32/7df1d81ec2e50fb661944a35183d87e62d3f6c6d9f8aff64a4f245226d55/alembic-1.17.1-py3-none-any.whl", hash = "sha256:cbc2386e60f89608bb63f30d2d6cc66c7aaed1fe105bd862828600e5ad167023", size = 247848, upload-time = "2025-10-29T00:23:18.79Z" }, ] [[package]] name = "alibabacloud-credentials" -version = "1.0.2" +version = "1.0.3" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "aiofiles" }, @@ -146,7 +146,10 @@ dependencies = [ { name = "alibabacloud-tea" }, { name = "apscheduler" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/b7/0c/1b0c5f4c2170165719b336616ac0a88f1666fd8690fda41e2e8ae3139fd9/alibabacloud-credentials-1.0.2.tar.gz", hash = "sha256:d2368eb70bd02db9143b2bf531a27a6fecd2cde9601db6e5b48cd6dbe25720ce", size = 30804, upload-time = "2025-05-06T12:30:35.46Z" } +sdist = { url = "https://files.pythonhosted.org/packages/df/82/45ec98bd19387507cf058ce47f62d6fea288bf0511c5a101b832e13d3edd/alibabacloud-credentials-1.0.3.tar.gz", hash = "sha256:9d8707e96afc6f348e23f5677ed15a21c2dfce7cfe6669776548ee4c80e1dfaf", size = 35831, upload-time = "2025-10-14T06:39:58.97Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/88/df/dbd9ae9d531a40d5613573c5a22ef774ecfdcaa0dc43aad42189f89c04ce/alibabacloud_credentials-1.0.3-py3-none-any.whl", hash = "sha256:30c8302f204b663c655d97e1c283ee9f9f84a6257d7901b931477d6cf34445a8", size = 41875, upload-time = "2025-10-14T06:39:58.029Z" }, +] [[package]] name = "alibabacloud-credentials-api" @@ -328,6 +331,15 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/59/75/e0e10dc7ed1408c28e03a6cb2d7a407f99320eb953f229d008a7a6d05546/aniso8601-10.0.1-py2.py3-none-any.whl", hash = "sha256:eb19717fd4e0db6de1aab06f12450ab92144246b257423fe020af5748c0cb89e", size = 52848, upload-time = "2025-04-18T17:29:41.492Z" }, ] +[[package]] +name = "annotated-doc" +version = "0.0.4" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/57/ba/046ceea27344560984e26a590f90bc7f4a75b06701f653222458922b558c/annotated_doc-0.0.4.tar.gz", hash = "sha256:fbcda96e87e9c92ad167c2e53839e57503ecfda18804ea28102353485033faa4", size = 7288, upload-time = "2025-11-10T22:07:42.062Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1e/d3/26bf1008eb3d2daa8ef4cacc7f3bfdc11818d111f7e2d0201bc6e3b49d45/annotated_doc-0.0.4-py3-none-any.whl", hash = "sha256:571ac1dc6991c450b25a9c2d84a3705e2ae7a53467b5d111c24fa8baabbed320", size = 5303, upload-time = "2025-11-10T22:07:40.673Z" }, +] + [[package]] name = "annotated-types" version = "0.7.0" @@ -353,14 +365,14 @@ wheels = [ [[package]] name = "apscheduler" -version = "3.11.0" +version = "3.11.1" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "tzlocal" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/4e/00/6d6814ddc19be2df62c8c898c4df6b5b1914f3bd024b780028caa392d186/apscheduler-3.11.0.tar.gz", hash = "sha256:4c622d250b0955a65d5d0eb91c33e6d43fd879834bf541e0a18661ae60460133", size = 107347, upload-time = "2024-11-24T19:39:26.463Z" } +sdist = { url = "https://files.pythonhosted.org/packages/d0/81/192db4f8471de5bc1f0d098783decffb1e6e69c4f8b4bc6711094691950b/apscheduler-3.11.1.tar.gz", hash = "sha256:0db77af6400c84d1747fe98a04b8b58f0080c77d11d338c4f507a9752880f221", size = 108044, upload-time = "2025-10-31T18:55:42.819Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/d0/ae/9a053dd9229c0fde6b1f1f33f609ccff1ee79ddda364c756a924c6d8563b/APScheduler-3.11.0-py3-none-any.whl", hash = "sha256:fc134ca32e50f5eadcc4938e3a4545ab19131435e851abb40b34d63d5141c6da", size = 64004, upload-time = "2024-11-24T19:39:24.442Z" }, + { url = "https://files.pythonhosted.org/packages/58/9f/d3c76f76c73fcc959d28e9def45b8b1cc3d7722660c5003b19c1022fd7f4/apscheduler-3.11.1-py3-none-any.whl", hash = "sha256:6162cb5683cb09923654fa9bdd3130c4be4bfda6ad8990971c9597ecd52965d2", size = 64278, upload-time = "2025-10-31T18:55:41.186Z" }, ] [[package]] @@ -422,16 +434,15 @@ wheels = [ [[package]] name = "azure-core" -version = "1.35.1" +version = "1.36.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "requests" }, - { name = "six" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/15/6b/2653adc0f33adba8f11b1903701e6b1c10d34ce5d8e25dfa13a422f832b0/azure_core-1.35.1.tar.gz", hash = "sha256:435d05d6df0fff2f73fb3c15493bb4721ede14203f1ff1382aa6b6b2bdd7e562", size = 345290, upload-time = "2025-09-11T22:58:04.481Z" } +sdist = { url = "https://files.pythonhosted.org/packages/0a/c4/d4ff3bc3ddf155156460bff340bbe9533f99fac54ddea165f35a8619f162/azure_core-1.36.0.tar.gz", hash = "sha256:22e5605e6d0bf1d229726af56d9e92bc37b6e726b141a18be0b4d424131741b7", size = 351139, upload-time = "2025-10-15T00:33:49.083Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/27/52/805980aa1ba18282077c484dba634ef0ede1e84eec8be9c92b2e162d0ed6/azure_core-1.35.1-py3-none-any.whl", hash = "sha256:12da0c9e08e48e198f9158b56ddbe33b421477e1dc98c2e1c8f9e254d92c468b", size = 211800, upload-time = "2025-09-11T22:58:06.281Z" }, + { url = "https://files.pythonhosted.org/packages/b1/3c/b90d5afc2e47c4a45f4bba00f9c3193b0417fad5ad3bb07869f9d12832aa/azure_core-1.36.0-py3-none-any.whl", hash = "sha256:fee9923a3a753e94a259563429f3644aaf05c486d45b1215d098115102d91d3b", size = 213302, upload-time = "2025-10-15T00:33:51.058Z" }, ] [[package]] @@ -487,16 +498,16 @@ wheels = [ [[package]] name = "bce-python-sdk" -version = "0.9.46" +version = "0.9.52" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "future" }, { name = "pycryptodome" }, { name = "six" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/53/57/f98bc15c12cc022ef195f689ee57ed61d8a8677bda3089c4d58fb1872d45/bce_python_sdk-0.9.46.tar.gz", hash = "sha256:4bf01b22e6d172ccd94aa201f8bc6f2a98d0da4784160e77cfacfcc71c2686be", size = 253806, upload-time = "2025-09-15T06:51:52.753Z" } +sdist = { url = "https://files.pythonhosted.org/packages/83/0a/e49d7774ce186fd51c611a2533baff8e7db0d22baef12223773f389b06b1/bce_python_sdk-0.9.52.tar.gz", hash = "sha256:dd54213ac25b8b1260fb45f1fbc0f2b1c53bb0f9f594258ca0479f1fc85f7405", size = 275614, upload-time = "2025-11-12T09:09:28.227Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/c0/f5/20e9ab324b22a77970c57bc8267e586e85e2aa1277d80f2c58ca8a39a13e/bce_python_sdk-0.9.46-py3-none-any.whl", hash = "sha256:655074da6592ce8b036f605d9a272bfdcd1f515eb2f8e3f0333bb7cc62f700cb", size = 352622, upload-time = "2025-09-15T06:51:50.811Z" }, + { url = "https://files.pythonhosted.org/packages/a3/d0/f57f75c96e8bb72144845f7208f712a54454f1d063d5ef02f1e9ea476b79/bce_python_sdk-0.9.52-py3-none-any.whl", hash = "sha256:f1ed39aa61c2d4a002cd2345e01dd92ac55c75960440d76163ead419b3b550e7", size = 390401, upload-time = "2025-11-12T09:09:26.663Z" }, ] [[package]] @@ -587,16 +598,16 @@ wheels = [ [[package]] name = "boto3-stubs" -version = "1.40.51" +version = "1.40.72" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "botocore-stubs" }, { name = "types-s3transfer" }, { name = "typing-extensions", marker = "python_full_version < '3.12'" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/82/4d/b07f9ee0fe432fa8ec6dc368ee7a0409e2b6d9df2c5a2a88265c9b6fd878/boto3_stubs-1.40.51.tar.gz", hash = "sha256:0281e820813a310954e15fb7c1d470c24c34c1cccc7b1ddad977fa293a1080a9", size = 100890, upload-time = "2025-10-13T19:25:36.126Z" } +sdist = { url = "https://files.pythonhosted.org/packages/4f/db/90881ac0b8afdfa9b95ae66b4094ed33f88b6086a8945229a95156257ca9/boto3_stubs-1.40.72.tar.gz", hash = "sha256:cbcf7b6e8a7f54e77fcb2b8d00041993fe4f76554c716b1d290e48650d569cd0", size = 99406, upload-time = "2025-11-12T20:36:23.685Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/d3/2e/4476431f11fc3bf7a7e0f4f5c275f17607aa127da7c0d8685a4dc6bf6291/boto3_stubs-1.40.51-py3-none-any.whl", hash = "sha256:896d0ffaa298ce1749eea1a54946320a0f4e07c6912f8e1f8c0744a708ee25a4", size = 69709, upload-time = "2025-10-13T19:25:23.116Z" }, + { url = "https://files.pythonhosted.org/packages/7a/ea/0f2814edc61c2e6fedd9b7a7fbc55149d1ffac7f7cd02d04cc51d1a3b1ca/boto3_stubs-1.40.72-py3-none-any.whl", hash = "sha256:4807f334b87914f75db3c6cd85f7eb706b5777e6ddaf117f8d63219cc01fb4b2", size = 68982, upload-time = "2025-11-12T20:36:12.855Z" }, ] [package.optional-dependencies] @@ -620,14 +631,14 @@ wheels = [ [[package]] name = "botocore-stubs" -version = "1.40.51" +version = "1.40.72" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "types-awscrt" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/55/ca/429fadb6e037cb7b300d508a0b24b59a71961db12539e21749cbec7e7422/botocore_stubs-1.40.51.tar.gz", hash = "sha256:8ddbeb1f68e39382533bb53f3b968d29e640406016af00ad8bbd6e1a2bd59536", size = 42249, upload-time = "2025-10-13T20:26:57.777Z" } +sdist = { url = "https://files.pythonhosted.org/packages/51/c9/17d5337cc81f107fd0a6d04b5b20c75bea0fe8b77bcc644de324487f8310/botocore_stubs-1.40.72.tar.gz", hash = "sha256:6d268d0dd9366dc15e7af52cbd0d3a3f3cd14e2191de0e280badc69f8d34708c", size = 42208, upload-time = "2025-11-12T21:23:53.344Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/c9/b9/5f1296bc46f293f284a1a6259f3c1f21f4161088dc6f70428698841b56a7/botocore_stubs-1.40.51-py3-none-any.whl", hash = "sha256:9a028104979205c9be0b68bb59ba679e4fe452e017eec3d40f6c2b41c590a73c", size = 66541, upload-time = "2025-10-13T20:26:55.559Z" }, + { url = "https://files.pythonhosted.org/packages/3c/99/9387b31ec1d980af83ca097366cc10714757d2c1390b4ac6b692c07a9e7f/botocore_stubs-1.40.72-py3-none-any.whl", hash = "sha256:1166a81074714312d3843be3f879d16966cbffdc440ab61ad6f0cd8922fde679", size = 66542, upload-time = "2025-11-12T21:23:51.018Z" }, ] [[package]] @@ -657,44 +668,30 @@ wheels = [ [[package]] name = "brotli" -version = "1.1.0" +version = "1.2.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/2f/c2/f9e977608bdf958650638c3f1e28f85a1b075f075ebbe77db8555463787b/Brotli-1.1.0.tar.gz", hash = "sha256:81de08ac11bcb85841e440c13611c00b67d3bf82698314928d0b676362546724", size = 7372270, upload-time = "2023-09-07T14:05:41.643Z" } +sdist = { url = "https://files.pythonhosted.org/packages/f7/16/c92ca344d646e71a43b8bb353f0a6490d7f6e06210f8554c8f874e454285/brotli-1.2.0.tar.gz", hash = "sha256:e310f77e41941c13340a95976fe66a8a95b01e783d430eeaf7a2f87e0a57dd0a", size = 7388632, upload-time = "2025-11-05T18:39:42.86Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/96/12/ad41e7fadd5db55459c4c401842b47f7fee51068f86dd2894dd0dcfc2d2a/Brotli-1.1.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:a3daabb76a78f829cafc365531c972016e4aa8d5b4bf60660ad8ecee19df7ccc", size = 873068, upload-time = "2023-09-07T14:03:37.779Z" }, - { url = "https://files.pythonhosted.org/packages/95/4e/5afab7b2b4b61a84e9c75b17814198ce515343a44e2ed4488fac314cd0a9/Brotli-1.1.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c8146669223164fc87a7e3de9f81e9423c67a79d6b3447994dfb9c95da16e2d6", size = 446244, upload-time = "2023-09-07T14:03:39.223Z" }, - { url = "https://files.pythonhosted.org/packages/9d/e6/f305eb61fb9a8580c525478a4a34c5ae1a9bcb12c3aee619114940bc513d/Brotli-1.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:30924eb4c57903d5a7526b08ef4a584acc22ab1ffa085faceb521521d2de32dd", size = 2906500, upload-time = "2023-09-07T14:03:40.858Z" }, - { url = "https://files.pythonhosted.org/packages/3e/4f/af6846cfbc1550a3024e5d3775ede1e00474c40882c7bf5b37a43ca35e91/Brotli-1.1.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ceb64bbc6eac5a140ca649003756940f8d6a7c444a68af170b3187623b43bebf", size = 2943950, upload-time = "2023-09-07T14:03:42.896Z" }, - { url = "https://files.pythonhosted.org/packages/b3/e7/ca2993c7682d8629b62630ebf0d1f3bb3d579e667ce8e7ca03a0a0576a2d/Brotli-1.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a469274ad18dc0e4d316eefa616d1d0c2ff9da369af19fa6f3daa4f09671fd61", size = 2918527, upload-time = "2023-09-07T14:03:44.552Z" }, - { url = "https://files.pythonhosted.org/packages/b3/96/da98e7bedc4c51104d29cc61e5f449a502dd3dbc211944546a4cc65500d3/Brotli-1.1.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:524f35912131cc2cabb00edfd8d573b07f2d9f21fa824bd3fb19725a9cf06327", size = 2845489, upload-time = "2023-09-07T14:03:46.594Z" }, - { url = "https://files.pythonhosted.org/packages/e8/ef/ccbc16947d6ce943a7f57e1a40596c75859eeb6d279c6994eddd69615265/Brotli-1.1.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:5b3cc074004d968722f51e550b41a27be656ec48f8afaeeb45ebf65b561481dd", size = 2914080, upload-time = "2023-09-07T14:03:48.204Z" }, - { url = "https://files.pythonhosted.org/packages/80/d6/0bd38d758d1afa62a5524172f0b18626bb2392d717ff94806f741fcd5ee9/Brotli-1.1.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:19c116e796420b0cee3da1ccec3b764ed2952ccfcc298b55a10e5610ad7885f9", size = 2813051, upload-time = "2023-09-07T14:03:50.348Z" }, - { url = "https://files.pythonhosted.org/packages/14/56/48859dd5d129d7519e001f06dcfbb6e2cf6db92b2702c0c2ce7d97e086c1/Brotli-1.1.0-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:510b5b1bfbe20e1a7b3baf5fed9e9451873559a976c1a78eebaa3b86c57b4265", size = 2938172, upload-time = "2023-09-07T14:03:52.395Z" }, - { url = "https://files.pythonhosted.org/packages/3d/77/a236d5f8cd9e9f4348da5acc75ab032ab1ab2c03cc8f430d24eea2672888/Brotli-1.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:a1fd8a29719ccce974d523580987b7f8229aeace506952fa9ce1d53a033873c8", size = 2933023, upload-time = "2023-09-07T14:03:53.96Z" }, - { url = "https://files.pythonhosted.org/packages/f1/87/3b283efc0f5cb35f7f84c0c240b1e1a1003a5e47141a4881bf87c86d0ce2/Brotli-1.1.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:c247dd99d39e0338a604f8c2b3bc7061d5c2e9e2ac7ba9cc1be5a69cb6cd832f", size = 2935871, upload-time = "2024-10-18T12:32:16.688Z" }, - { url = "https://files.pythonhosted.org/packages/f3/eb/2be4cc3e2141dc1a43ad4ca1875a72088229de38c68e842746b342667b2a/Brotli-1.1.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:1b2c248cd517c222d89e74669a4adfa5577e06ab68771a529060cf5a156e9757", size = 2847784, upload-time = "2024-10-18T12:32:18.459Z" }, - { url = "https://files.pythonhosted.org/packages/66/13/b58ddebfd35edde572ccefe6890cf7c493f0c319aad2a5badee134b4d8ec/Brotli-1.1.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:2a24c50840d89ded6c9a8fdc7b6ed3692ed4e86f1c4a4a938e1e92def92933e0", size = 3034905, upload-time = "2024-10-18T12:32:20.192Z" }, - { url = "https://files.pythonhosted.org/packages/84/9c/bc96b6c7db824998a49ed3b38e441a2cae9234da6fa11f6ed17e8cf4f147/Brotli-1.1.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f31859074d57b4639318523d6ffdca586ace54271a73ad23ad021acd807eb14b", size = 2929467, upload-time = "2024-10-18T12:32:21.774Z" }, - { url = "https://files.pythonhosted.org/packages/e7/71/8f161dee223c7ff7fea9d44893fba953ce97cf2c3c33f78ba260a91bcff5/Brotli-1.1.0-cp311-cp311-win32.whl", hash = "sha256:39da8adedf6942d76dc3e46653e52df937a3c4d6d18fdc94a7c29d263b1f5b50", size = 333169, upload-time = "2023-09-07T14:03:55.404Z" }, - { url = "https://files.pythonhosted.org/packages/02/8a/fece0ee1057643cb2a5bbf59682de13f1725f8482b2c057d4e799d7ade75/Brotli-1.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:aac0411d20e345dc0920bdec5548e438e999ff68d77564d5e9463a7ca9d3e7b1", size = 357253, upload-time = "2023-09-07T14:03:56.643Z" }, - { url = "https://files.pythonhosted.org/packages/5c/d0/5373ae13b93fe00095a58efcbce837fd470ca39f703a235d2a999baadfbc/Brotli-1.1.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:32d95b80260d79926f5fab3c41701dbb818fde1c9da590e77e571eefd14abe28", size = 815693, upload-time = "2024-10-18T12:32:23.824Z" }, - { url = "https://files.pythonhosted.org/packages/8e/48/f6e1cdf86751300c288c1459724bfa6917a80e30dbfc326f92cea5d3683a/Brotli-1.1.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:b760c65308ff1e462f65d69c12e4ae085cff3b332d894637f6273a12a482d09f", size = 422489, upload-time = "2024-10-18T12:32:25.641Z" }, - { url = "https://files.pythonhosted.org/packages/06/88/564958cedce636d0f1bed313381dfc4b4e3d3f6015a63dae6146e1b8c65c/Brotli-1.1.0-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:316cc9b17edf613ac76b1f1f305d2a748f1b976b033b049a6ecdfd5612c70409", size = 873081, upload-time = "2023-09-07T14:03:57.967Z" }, - { url = "https://files.pythonhosted.org/packages/58/79/b7026a8bb65da9a6bb7d14329fd2bd48d2b7f86d7329d5cc8ddc6a90526f/Brotli-1.1.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:caf9ee9a5775f3111642d33b86237b05808dafcd6268faa492250e9b78046eb2", size = 446244, upload-time = "2023-09-07T14:03:59.319Z" }, - { url = "https://files.pythonhosted.org/packages/e5/18/c18c32ecea41b6c0004e15606e274006366fe19436b6adccc1ae7b2e50c2/Brotli-1.1.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:70051525001750221daa10907c77830bc889cb6d865cc0b813d9db7fefc21451", size = 2906505, upload-time = "2023-09-07T14:04:01.327Z" }, - { url = "https://files.pythonhosted.org/packages/08/c8/69ec0496b1ada7569b62d85893d928e865df29b90736558d6c98c2031208/Brotli-1.1.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7f4bf76817c14aa98cc6697ac02f3972cb8c3da93e9ef16b9c66573a68014f91", size = 2944152, upload-time = "2023-09-07T14:04:03.033Z" }, - { url = "https://files.pythonhosted.org/packages/ab/fb/0517cea182219d6768113a38167ef6d4eb157a033178cc938033a552ed6d/Brotli-1.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d0c5516f0aed654134a2fc936325cc2e642f8a0e096d075209672eb321cff408", size = 2919252, upload-time = "2023-09-07T14:04:04.675Z" }, - { url = "https://files.pythonhosted.org/packages/c7/53/73a3431662e33ae61a5c80b1b9d2d18f58dfa910ae8dd696e57d39f1a2f5/Brotli-1.1.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6c3020404e0b5eefd7c9485ccf8393cfb75ec38ce75586e046573c9dc29967a0", size = 2845955, upload-time = "2023-09-07T14:04:06.585Z" }, - { url = "https://files.pythonhosted.org/packages/55/ac/bd280708d9c5ebdbf9de01459e625a3e3803cce0784f47d633562cf40e83/Brotli-1.1.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:4ed11165dd45ce798d99a136808a794a748d5dc38511303239d4e2363c0695dc", size = 2914304, upload-time = "2023-09-07T14:04:08.668Z" }, - { url = "https://files.pythonhosted.org/packages/76/58/5c391b41ecfc4527d2cc3350719b02e87cb424ef8ba2023fb662f9bf743c/Brotli-1.1.0-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:4093c631e96fdd49e0377a9c167bfd75b6d0bad2ace734c6eb20b348bc3ea180", size = 2814452, upload-time = "2023-09-07T14:04:10.736Z" }, - { url = "https://files.pythonhosted.org/packages/c7/4e/91b8256dfe99c407f174924b65a01f5305e303f486cc7a2e8a5d43c8bec3/Brotli-1.1.0-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:7e4c4629ddad63006efa0ef968c8e4751c5868ff0b1c5c40f76524e894c50248", size = 2938751, upload-time = "2023-09-07T14:04:12.875Z" }, - { url = "https://files.pythonhosted.org/packages/5a/a6/e2a39a5d3b412938362bbbeba5af904092bf3f95b867b4a3eb856104074e/Brotli-1.1.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:861bf317735688269936f755fa136a99d1ed526883859f86e41a5d43c61d8966", size = 2933757, upload-time = "2023-09-07T14:04:14.551Z" }, - { url = "https://files.pythonhosted.org/packages/13/f0/358354786280a509482e0e77c1a5459e439766597d280f28cb097642fc26/Brotli-1.1.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:87a3044c3a35055527ac75e419dfa9f4f3667a1e887ee80360589eb8c90aabb9", size = 2936146, upload-time = "2024-10-18T12:32:27.257Z" }, - { url = "https://files.pythonhosted.org/packages/80/f7/daf538c1060d3a88266b80ecc1d1c98b79553b3f117a485653f17070ea2a/Brotli-1.1.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:c5529b34c1c9d937168297f2c1fde7ebe9ebdd5e121297ff9c043bdb2ae3d6fb", size = 2848055, upload-time = "2024-10-18T12:32:29.376Z" }, - { url = "https://files.pythonhosted.org/packages/ad/cf/0eaa0585c4077d3c2d1edf322d8e97aabf317941d3a72d7b3ad8bce004b0/Brotli-1.1.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:ca63e1890ede90b2e4454f9a65135a4d387a4585ff8282bb72964fab893f2111", size = 3035102, upload-time = "2024-10-18T12:32:31.371Z" }, - { url = "https://files.pythonhosted.org/packages/d8/63/1c1585b2aa554fe6dbce30f0c18bdbc877fa9a1bf5ff17677d9cca0ac122/Brotli-1.1.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:e79e6520141d792237c70bcd7a3b122d00f2613769ae0cb61c52e89fd3443839", size = 2930029, upload-time = "2024-10-18T12:32:33.293Z" }, - { url = "https://files.pythonhosted.org/packages/5f/3b/4e3fd1893eb3bbfef8e5a80d4508bec17a57bb92d586c85c12d28666bb13/Brotli-1.1.0-cp312-cp312-win32.whl", hash = "sha256:5f4d5ea15c9382135076d2fb28dde923352fe02951e66935a9efaac8f10e81b0", size = 333276, upload-time = "2023-09-07T14:04:16.49Z" }, - { url = "https://files.pythonhosted.org/packages/3d/d5/942051b45a9e883b5b6e98c041698b1eb2012d25e5948c58d6bf85b1bb43/Brotli-1.1.0-cp312-cp312-win_amd64.whl", hash = "sha256:906bc3a79de8c4ae5b86d3d75a8b77e44404b0f4261714306e3ad248d8ab0951", size = 357255, upload-time = "2023-09-07T14:04:17.83Z" }, + { url = "https://files.pythonhosted.org/packages/7a/ef/f285668811a9e1ddb47a18cb0b437d5fc2760d537a2fe8a57875ad6f8448/brotli-1.2.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:15b33fe93cedc4caaff8a0bd1eb7e3dab1c61bb22a0bf5bdfdfd97cd7da79744", size = 863110, upload-time = "2025-11-05T18:38:12.978Z" }, + { url = "https://files.pythonhosted.org/packages/50/62/a3b77593587010c789a9d6eaa527c79e0848b7b860402cc64bc0bc28a86c/brotli-1.2.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:898be2be399c221d2671d29eed26b6b2713a02c2119168ed914e7d00ceadb56f", size = 445438, upload-time = "2025-11-05T18:38:14.208Z" }, + { url = "https://files.pythonhosted.org/packages/cd/e1/7fadd47f40ce5549dc44493877db40292277db373da5053aff181656e16e/brotli-1.2.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:350c8348f0e76fff0a0fd6c26755d2653863279d086d3aa2c290a6a7251135dd", size = 1534420, upload-time = "2025-11-05T18:38:15.111Z" }, + { url = "https://files.pythonhosted.org/packages/12/8b/1ed2f64054a5a008a4ccd2f271dbba7a5fb1a3067a99f5ceadedd4c1d5a7/brotli-1.2.0-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:2e1ad3fda65ae0d93fec742a128d72e145c9c7a99ee2fcd667785d99eb25a7fe", size = 1632619, upload-time = "2025-11-05T18:38:16.094Z" }, + { url = "https://files.pythonhosted.org/packages/89/5a/7071a621eb2d052d64efd5da2ef55ecdac7c3b0c6e4f9d519e9c66d987ef/brotli-1.2.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:40d918bce2b427a0c4ba189df7a006ac0c7277c180aee4617d99e9ccaaf59e6a", size = 1426014, upload-time = "2025-11-05T18:38:17.177Z" }, + { url = "https://files.pythonhosted.org/packages/26/6d/0971a8ea435af5156acaaccec1a505f981c9c80227633851f2810abd252a/brotli-1.2.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:2a7f1d03727130fc875448b65b127a9ec5d06d19d0148e7554384229706f9d1b", size = 1489661, upload-time = "2025-11-05T18:38:18.41Z" }, + { url = "https://files.pythonhosted.org/packages/f3/75/c1baca8b4ec6c96a03ef8230fab2a785e35297632f402ebb1e78a1e39116/brotli-1.2.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:9c79f57faa25d97900bfb119480806d783fba83cd09ee0b33c17623935b05fa3", size = 1599150, upload-time = "2025-11-05T18:38:19.792Z" }, + { url = "https://files.pythonhosted.org/packages/0d/1a/23fcfee1c324fd48a63d7ebf4bac3a4115bdb1b00e600f80f727d850b1ae/brotli-1.2.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:844a8ceb8483fefafc412f85c14f2aae2fb69567bf2a0de53cdb88b73e7c43ae", size = 1493505, upload-time = "2025-11-05T18:38:20.913Z" }, + { url = "https://files.pythonhosted.org/packages/36/e5/12904bbd36afeef53d45a84881a4810ae8810ad7e328a971ebbfd760a0b3/brotli-1.2.0-cp311-cp311-win32.whl", hash = "sha256:aa47441fa3026543513139cb8926a92a8e305ee9c71a6209ef7a97d91640ea03", size = 334451, upload-time = "2025-11-05T18:38:21.94Z" }, + { url = "https://files.pythonhosted.org/packages/02/8b/ecb5761b989629a4758c394b9301607a5880de61ee2ee5fe104b87149ebc/brotli-1.2.0-cp311-cp311-win_amd64.whl", hash = "sha256:022426c9e99fd65d9475dce5c195526f04bb8be8907607e27e747893f6ee3e24", size = 369035, upload-time = "2025-11-05T18:38:22.941Z" }, + { url = "https://files.pythonhosted.org/packages/11/ee/b0a11ab2315c69bb9b45a2aaed022499c9c24a205c3a49c3513b541a7967/brotli-1.2.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:35d382625778834a7f3061b15423919aa03e4f5da34ac8e02c074e4b75ab4f84", size = 861543, upload-time = "2025-11-05T18:38:24.183Z" }, + { url = "https://files.pythonhosted.org/packages/e1/2f/29c1459513cd35828e25531ebfcbf3e92a5e49f560b1777a9af7203eb46e/brotli-1.2.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:7a61c06b334bd99bc5ae84f1eeb36bfe01400264b3c352f968c6e30a10f9d08b", size = 444288, upload-time = "2025-11-05T18:38:25.139Z" }, + { url = "https://files.pythonhosted.org/packages/3d/6f/feba03130d5fceadfa3a1bb102cb14650798c848b1df2a808356f939bb16/brotli-1.2.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:acec55bb7c90f1dfc476126f9711a8e81c9af7fb617409a9ee2953115343f08d", size = 1528071, upload-time = "2025-11-05T18:38:26.081Z" }, + { url = "https://files.pythonhosted.org/packages/2b/38/f3abb554eee089bd15471057ba85f47e53a44a462cfce265d9bf7088eb09/brotli-1.2.0-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:260d3692396e1895c5034f204f0db022c056f9e2ac841593a4cf9426e2a3faca", size = 1626913, upload-time = "2025-11-05T18:38:27.284Z" }, + { url = "https://files.pythonhosted.org/packages/03/a7/03aa61fbc3c5cbf99b44d158665f9b0dd3d8059be16c460208d9e385c837/brotli-1.2.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:072e7624b1fc4d601036ab3f4f27942ef772887e876beff0301d261210bca97f", size = 1419762, upload-time = "2025-11-05T18:38:28.295Z" }, + { url = "https://files.pythonhosted.org/packages/21/1b/0374a89ee27d152a5069c356c96b93afd1b94eae83f1e004b57eb6ce2f10/brotli-1.2.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:adedc4a67e15327dfdd04884873c6d5a01d3e3b6f61406f99b1ed4865a2f6d28", size = 1484494, upload-time = "2025-11-05T18:38:29.29Z" }, + { url = "https://files.pythonhosted.org/packages/cf/57/69d4fe84a67aef4f524dcd075c6eee868d7850e85bf01d778a857d8dbe0a/brotli-1.2.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:7a47ce5c2288702e09dc22a44d0ee6152f2c7eda97b3c8482d826a1f3cfc7da7", size = 1593302, upload-time = "2025-11-05T18:38:30.639Z" }, + { url = "https://files.pythonhosted.org/packages/d5/3b/39e13ce78a8e9a621c5df3aeb5fd181fcc8caba8c48a194cd629771f6828/brotli-1.2.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:af43b8711a8264bb4e7d6d9a6d004c3a2019c04c01127a868709ec29962b6036", size = 1487913, upload-time = "2025-11-05T18:38:31.618Z" }, + { url = "https://files.pythonhosted.org/packages/62/28/4d00cb9bd76a6357a66fcd54b4b6d70288385584063f4b07884c1e7286ac/brotli-1.2.0-cp312-cp312-win32.whl", hash = "sha256:e99befa0b48f3cd293dafeacdd0d191804d105d279e0b387a32054c1180f3161", size = 334362, upload-time = "2025-11-05T18:38:32.939Z" }, + { url = "https://files.pythonhosted.org/packages/1c/4e/bc1dcac9498859d5e353c9b153627a3752868a9d5f05ce8dedd81a2354ab/brotli-1.2.0-cp312-cp312-win_amd64.whl", hash = "sha256:b35c13ce241abdd44cb8ca70683f20c0c079728a36a996297adb5334adfc1c44", size = 369115, upload-time = "2025-11-05T18:38:33.765Z" }, ] [[package]] @@ -782,11 +779,11 @@ wheels = [ [[package]] name = "certifi" -version = "2025.10.5" +version = "2025.11.12" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/4c/5b/b6ce21586237c77ce67d01dc5507039d444b630dd76611bbca2d8e5dcd91/certifi-2025.10.5.tar.gz", hash = "sha256:47c09d31ccf2acf0be3f701ea53595ee7e0b8fa08801c6624be771df09ae7b43", size = 164519, upload-time = "2025-10-05T04:12:15.808Z" } +sdist = { url = "https://files.pythonhosted.org/packages/a2/8c/58f469717fa48465e4a50c014a0400602d3c437d7c0c468e17ada824da3a/certifi-2025.11.12.tar.gz", hash = "sha256:d8ab5478f2ecd78af242878415affce761ca6bc54a22a27e026d7c25357c3316", size = 160538, upload-time = "2025-11-12T02:54:51.517Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/e4/37/af0d2ef3967ac0d6113837b44a4f0bfe1328c2b9763bd5b1744520e5cfed/certifi-2025.10.5-py3-none-any.whl", hash = "sha256:0f212c2744a9bb6de0c56639a6f68afe01ecd92d91f14ae897c4fe7bbeeef0de", size = 163286, upload-time = "2025-10-05T04:12:14.03Z" }, + { url = "https://files.pythonhosted.org/packages/70/7d/9bc192684cea499815ff478dfcdc13835ddf401365057044fb721ec6bddb/certifi-2025.11.12-py3-none-any.whl", hash = "sha256:97de8790030bbd5c2d96b7ec782fc2f7820ef8dba6db909ccf95449f2d062d4b", size = 159438, upload-time = "2025-11-12T02:54:49.735Z" }, ] [[package]] @@ -836,33 +833,43 @@ wheels = [ [[package]] name = "charset-normalizer" -version = "3.4.3" +version = "3.4.4" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/83/2d/5fd176ceb9b2fc619e63405525573493ca23441330fcdaee6bef9460e924/charset_normalizer-3.4.3.tar.gz", hash = "sha256:6fce4b8500244f6fcb71465d4a4930d132ba9ab8e71a7859e6a5d59851068d14", size = 122371, upload-time = "2025-08-09T07:57:28.46Z" } +sdist = { url = "https://files.pythonhosted.org/packages/13/69/33ddede1939fdd074bce5434295f38fae7136463422fe4fd3e0e89b98062/charset_normalizer-3.4.4.tar.gz", hash = "sha256:94537985111c35f28720e43603b8e7b43a6ecfb2ce1d3058bbe955b73404e21a", size = 129418, upload-time = "2025-10-14T04:42:32.879Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/7f/b5/991245018615474a60965a7c9cd2b4efbaabd16d582a5547c47ee1c7730b/charset_normalizer-3.4.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:b256ee2e749283ef3ddcff51a675ff43798d92d746d1a6e4631bf8c707d22d0b", size = 204483, upload-time = "2025-08-09T07:55:53.12Z" }, - { url = "https://files.pythonhosted.org/packages/c7/2a/ae245c41c06299ec18262825c1569c5d3298fc920e4ddf56ab011b417efd/charset_normalizer-3.4.3-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:13faeacfe61784e2559e690fc53fa4c5ae97c6fcedb8eb6fb8d0a15b475d2c64", size = 145520, upload-time = "2025-08-09T07:55:54.712Z" }, - { url = "https://files.pythonhosted.org/packages/3a/a4/b3b6c76e7a635748c4421d2b92c7b8f90a432f98bda5082049af37ffc8e3/charset_normalizer-3.4.3-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:00237675befef519d9af72169d8604a067d92755e84fe76492fef5441db05b91", size = 158876, upload-time = "2025-08-09T07:55:56.024Z" }, - { url = "https://files.pythonhosted.org/packages/e2/e6/63bb0e10f90a8243c5def74b5b105b3bbbfb3e7bb753915fe333fb0c11ea/charset_normalizer-3.4.3-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:585f3b2a80fbd26b048a0be90c5aae8f06605d3c92615911c3a2b03a8a3b796f", size = 156083, upload-time = "2025-08-09T07:55:57.582Z" }, - { url = "https://files.pythonhosted.org/packages/87/df/b7737ff046c974b183ea9aa111b74185ac8c3a326c6262d413bd5a1b8c69/charset_normalizer-3.4.3-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0e78314bdc32fa80696f72fa16dc61168fda4d6a0c014e0380f9d02f0e5d8a07", size = 150295, upload-time = "2025-08-09T07:55:59.147Z" }, - { url = "https://files.pythonhosted.org/packages/61/f1/190d9977e0084d3f1dc169acd060d479bbbc71b90bf3e7bf7b9927dec3eb/charset_normalizer-3.4.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:96b2b3d1a83ad55310de8c7b4a2d04d9277d5591f40761274856635acc5fcb30", size = 148379, upload-time = "2025-08-09T07:56:00.364Z" }, - { url = "https://files.pythonhosted.org/packages/4c/92/27dbe365d34c68cfe0ca76f1edd70e8705d82b378cb54ebbaeabc2e3029d/charset_normalizer-3.4.3-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:939578d9d8fd4299220161fdd76e86c6a251987476f5243e8864a7844476ba14", size = 160018, upload-time = "2025-08-09T07:56:01.678Z" }, - { url = "https://files.pythonhosted.org/packages/99/04/baae2a1ea1893a01635d475b9261c889a18fd48393634b6270827869fa34/charset_normalizer-3.4.3-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:fd10de089bcdcd1be95a2f73dbe6254798ec1bda9f450d5828c96f93e2536b9c", size = 157430, upload-time = "2025-08-09T07:56:02.87Z" }, - { url = "https://files.pythonhosted.org/packages/2f/36/77da9c6a328c54d17b960c89eccacfab8271fdaaa228305330915b88afa9/charset_normalizer-3.4.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:1e8ac75d72fa3775e0b7cb7e4629cec13b7514d928d15ef8ea06bca03ef01cae", size = 151600, upload-time = "2025-08-09T07:56:04.089Z" }, - { url = "https://files.pythonhosted.org/packages/64/d4/9eb4ff2c167edbbf08cdd28e19078bf195762e9bd63371689cab5ecd3d0d/charset_normalizer-3.4.3-cp311-cp311-win32.whl", hash = "sha256:6cf8fd4c04756b6b60146d98cd8a77d0cdae0e1ca20329da2ac85eed779b6849", size = 99616, upload-time = "2025-08-09T07:56:05.658Z" }, - { url = "https://files.pythonhosted.org/packages/f4/9c/996a4a028222e7761a96634d1820de8a744ff4327a00ada9c8942033089b/charset_normalizer-3.4.3-cp311-cp311-win_amd64.whl", hash = "sha256:31a9a6f775f9bcd865d88ee350f0ffb0e25936a7f930ca98995c05abf1faf21c", size = 107108, upload-time = "2025-08-09T07:56:07.176Z" }, - { url = "https://files.pythonhosted.org/packages/e9/5e/14c94999e418d9b87682734589404a25854d5f5d0408df68bc15b6ff54bb/charset_normalizer-3.4.3-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:e28e334d3ff134e88989d90ba04b47d84382a828c061d0d1027b1b12a62b39b1", size = 205655, upload-time = "2025-08-09T07:56:08.475Z" }, - { url = "https://files.pythonhosted.org/packages/7d/a8/c6ec5d389672521f644505a257f50544c074cf5fc292d5390331cd6fc9c3/charset_normalizer-3.4.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0cacf8f7297b0c4fcb74227692ca46b4a5852f8f4f24b3c766dd94a1075c4884", size = 146223, upload-time = "2025-08-09T07:56:09.708Z" }, - { url = "https://files.pythonhosted.org/packages/fc/eb/a2ffb08547f4e1e5415fb69eb7db25932c52a52bed371429648db4d84fb1/charset_normalizer-3.4.3-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:c6fd51128a41297f5409deab284fecbe5305ebd7e5a1f959bee1c054622b7018", size = 159366, upload-time = "2025-08-09T07:56:11.326Z" }, - { url = "https://files.pythonhosted.org/packages/82/10/0fd19f20c624b278dddaf83b8464dcddc2456cb4b02bb902a6da126b87a1/charset_normalizer-3.4.3-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:3cfb2aad70f2c6debfbcb717f23b7eb55febc0bb23dcffc0f076009da10c6392", size = 157104, upload-time = "2025-08-09T07:56:13.014Z" }, - { url = "https://files.pythonhosted.org/packages/16/ab/0233c3231af734f5dfcf0844aa9582d5a1466c985bbed6cedab85af9bfe3/charset_normalizer-3.4.3-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1606f4a55c0fd363d754049cdf400175ee96c992b1f8018b993941f221221c5f", size = 151830, upload-time = "2025-08-09T07:56:14.428Z" }, - { url = "https://files.pythonhosted.org/packages/ae/02/e29e22b4e02839a0e4a06557b1999d0a47db3567e82989b5bb21f3fbbd9f/charset_normalizer-3.4.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:027b776c26d38b7f15b26a5da1044f376455fb3766df8fc38563b4efbc515154", size = 148854, upload-time = "2025-08-09T07:56:16.051Z" }, - { url = "https://files.pythonhosted.org/packages/05/6b/e2539a0a4be302b481e8cafb5af8792da8093b486885a1ae4d15d452bcec/charset_normalizer-3.4.3-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:42e5088973e56e31e4fa58eb6bd709e42fc03799c11c42929592889a2e54c491", size = 160670, upload-time = "2025-08-09T07:56:17.314Z" }, - { url = "https://files.pythonhosted.org/packages/31/e7/883ee5676a2ef217a40ce0bffcc3d0dfbf9e64cbcfbdf822c52981c3304b/charset_normalizer-3.4.3-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:cc34f233c9e71701040d772aa7490318673aa7164a0efe3172b2981218c26d93", size = 158501, upload-time = "2025-08-09T07:56:18.641Z" }, - { url = "https://files.pythonhosted.org/packages/c1/35/6525b21aa0db614cf8b5792d232021dca3df7f90a1944db934efa5d20bb1/charset_normalizer-3.4.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:320e8e66157cc4e247d9ddca8e21f427efc7a04bbd0ac8a9faf56583fa543f9f", size = 153173, upload-time = "2025-08-09T07:56:20.289Z" }, - { url = "https://files.pythonhosted.org/packages/50/ee/f4704bad8201de513fdc8aac1cabc87e38c5818c93857140e06e772b5892/charset_normalizer-3.4.3-cp312-cp312-win32.whl", hash = "sha256:fb6fecfd65564f208cbf0fba07f107fb661bcd1a7c389edbced3f7a493f70e37", size = 99822, upload-time = "2025-08-09T07:56:21.551Z" }, - { url = "https://files.pythonhosted.org/packages/39/f5/3b3836ca6064d0992c58c7561c6b6eee1b3892e9665d650c803bd5614522/charset_normalizer-3.4.3-cp312-cp312-win_amd64.whl", hash = "sha256:86df271bf921c2ee3818f0522e9a5b8092ca2ad8b065ece5d7d9d0e9f4849bcc", size = 107543, upload-time = "2025-08-09T07:56:23.115Z" }, - { url = "https://files.pythonhosted.org/packages/8a/1f/f041989e93b001bc4e44bb1669ccdcf54d3f00e628229a85b08d330615c5/charset_normalizer-3.4.3-py3-none-any.whl", hash = "sha256:ce571ab16d890d23b5c278547ba694193a45011ff86a9162a71307ed9f86759a", size = 53175, upload-time = "2025-08-09T07:57:26.864Z" }, + { url = "https://files.pythonhosted.org/packages/ed/27/c6491ff4954e58a10f69ad90aca8a1b6fe9c5d3c6f380907af3c37435b59/charset_normalizer-3.4.4-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:6e1fcf0720908f200cd21aa4e6750a48ff6ce4afe7ff5a79a90d5ed8a08296f8", size = 206988, upload-time = "2025-10-14T04:40:33.79Z" }, + { url = "https://files.pythonhosted.org/packages/94/59/2e87300fe67ab820b5428580a53cad894272dbb97f38a7a814a2a1ac1011/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5f819d5fe9234f9f82d75bdfa9aef3a3d72c4d24a6e57aeaebba32a704553aa0", size = 147324, upload-time = "2025-10-14T04:40:34.961Z" }, + { url = "https://files.pythonhosted.org/packages/07/fb/0cf61dc84b2b088391830f6274cb57c82e4da8bbc2efeac8c025edb88772/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:a59cb51917aa591b1c4e6a43c132f0cdc3c76dbad6155df4e28ee626cc77a0a3", size = 142742, upload-time = "2025-10-14T04:40:36.105Z" }, + { url = "https://files.pythonhosted.org/packages/62/8b/171935adf2312cd745d290ed93cf16cf0dfe320863ab7cbeeae1dcd6535f/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:8ef3c867360f88ac904fd3f5e1f902f13307af9052646963ee08ff4f131adafc", size = 160863, upload-time = "2025-10-14T04:40:37.188Z" }, + { url = "https://files.pythonhosted.org/packages/09/73/ad875b192bda14f2173bfc1bc9a55e009808484a4b256748d931b6948442/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d9e45d7faa48ee908174d8fe84854479ef838fc6a705c9315372eacbc2f02897", size = 157837, upload-time = "2025-10-14T04:40:38.435Z" }, + { url = "https://files.pythonhosted.org/packages/6d/fc/de9cce525b2c5b94b47c70a4b4fb19f871b24995c728e957ee68ab1671ea/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:840c25fb618a231545cbab0564a799f101b63b9901f2569faecd6b222ac72381", size = 151550, upload-time = "2025-10-14T04:40:40.053Z" }, + { url = "https://files.pythonhosted.org/packages/55/c2/43edd615fdfba8c6f2dfbd459b25a6b3b551f24ea21981e23fb768503ce1/charset_normalizer-3.4.4-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:ca5862d5b3928c4940729dacc329aa9102900382fea192fc5e52eb69d6093815", size = 149162, upload-time = "2025-10-14T04:40:41.163Z" }, + { url = "https://files.pythonhosted.org/packages/03/86/bde4ad8b4d0e9429a4e82c1e8f5c659993a9a863ad62c7df05cf7b678d75/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d9c7f57c3d666a53421049053eaacdd14bbd0a528e2186fcb2e672effd053bb0", size = 150019, upload-time = "2025-10-14T04:40:42.276Z" }, + { url = "https://files.pythonhosted.org/packages/1f/86/a151eb2af293a7e7bac3a739b81072585ce36ccfb4493039f49f1d3cae8c/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:277e970e750505ed74c832b4bf75dac7476262ee2a013f5574dd49075879e161", size = 143310, upload-time = "2025-10-14T04:40:43.439Z" }, + { url = "https://files.pythonhosted.org/packages/b5/fe/43dae6144a7e07b87478fdfc4dbe9efd5defb0e7ec29f5f58a55aeef7bf7/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:31fd66405eaf47bb62e8cd575dc621c56c668f27d46a61d975a249930dd5e2a4", size = 162022, upload-time = "2025-10-14T04:40:44.547Z" }, + { url = "https://files.pythonhosted.org/packages/80/e6/7aab83774f5d2bca81f42ac58d04caf44f0cc2b65fc6db2b3b2e8a05f3b3/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:0d3d8f15c07f86e9ff82319b3d9ef6f4bf907608f53fe9d92b28ea9ae3d1fd89", size = 149383, upload-time = "2025-10-14T04:40:46.018Z" }, + { url = "https://files.pythonhosted.org/packages/4f/e8/b289173b4edae05c0dde07f69f8db476a0b511eac556dfe0d6bda3c43384/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:9f7fcd74d410a36883701fafa2482a6af2ff5ba96b9a620e9e0721e28ead5569", size = 159098, upload-time = "2025-10-14T04:40:47.081Z" }, + { url = "https://files.pythonhosted.org/packages/d8/df/fe699727754cae3f8478493c7f45f777b17c3ef0600e28abfec8619eb49c/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ebf3e58c7ec8a8bed6d66a75d7fb37b55e5015b03ceae72a8e7c74495551e224", size = 152991, upload-time = "2025-10-14T04:40:48.246Z" }, + { url = "https://files.pythonhosted.org/packages/1a/86/584869fe4ddb6ffa3bd9f491b87a01568797fb9bd8933f557dba9771beaf/charset_normalizer-3.4.4-cp311-cp311-win32.whl", hash = "sha256:eecbc200c7fd5ddb9a7f16c7decb07b566c29fa2161a16cf67b8d068bd21690a", size = 99456, upload-time = "2025-10-14T04:40:49.376Z" }, + { url = "https://files.pythonhosted.org/packages/65/f6/62fdd5feb60530f50f7e38b4f6a1d5203f4d16ff4f9f0952962c044e919a/charset_normalizer-3.4.4-cp311-cp311-win_amd64.whl", hash = "sha256:5ae497466c7901d54b639cf42d5b8c1b6a4fead55215500d2f486d34db48d016", size = 106978, upload-time = "2025-10-14T04:40:50.844Z" }, + { url = "https://files.pythonhosted.org/packages/7a/9d/0710916e6c82948b3be62d9d398cb4fcf4e97b56d6a6aeccd66c4b2f2bd5/charset_normalizer-3.4.4-cp311-cp311-win_arm64.whl", hash = "sha256:65e2befcd84bc6f37095f5961e68a6f077bf44946771354a28ad434c2cce0ae1", size = 99969, upload-time = "2025-10-14T04:40:52.272Z" }, + { url = "https://files.pythonhosted.org/packages/f3/85/1637cd4af66fa687396e757dec650f28025f2a2f5a5531a3208dc0ec43f2/charset_normalizer-3.4.4-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:0a98e6759f854bd25a58a73fa88833fba3b7c491169f86ce1180c948ab3fd394", size = 208425, upload-time = "2025-10-14T04:40:53.353Z" }, + { url = "https://files.pythonhosted.org/packages/9d/6a/04130023fef2a0d9c62d0bae2649b69f7b7d8d24ea5536feef50551029df/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b5b290ccc2a263e8d185130284f8501e3e36c5e02750fc6b6bdeb2e9e96f1e25", size = 148162, upload-time = "2025-10-14T04:40:54.558Z" }, + { url = "https://files.pythonhosted.org/packages/78/29/62328d79aa60da22c9e0b9a66539feae06ca0f5a4171ac4f7dc285b83688/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:74bb723680f9f7a6234dcf67aea57e708ec1fbdf5699fb91dfd6f511b0a320ef", size = 144558, upload-time = "2025-10-14T04:40:55.677Z" }, + { url = "https://files.pythonhosted.org/packages/86/bb/b32194a4bf15b88403537c2e120b817c61cd4ecffa9b6876e941c3ee38fe/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f1e34719c6ed0b92f418c7c780480b26b5d9c50349e9a9af7d76bf757530350d", size = 161497, upload-time = "2025-10-14T04:40:57.217Z" }, + { url = "https://files.pythonhosted.org/packages/19/89/a54c82b253d5b9b111dc74aca196ba5ccfcca8242d0fb64146d4d3183ff1/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:2437418e20515acec67d86e12bf70056a33abdacb5cb1655042f6538d6b085a8", size = 159240, upload-time = "2025-10-14T04:40:58.358Z" }, + { url = "https://files.pythonhosted.org/packages/c0/10/d20b513afe03acc89ec33948320a5544d31f21b05368436d580dec4e234d/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:11d694519d7f29d6cd09f6ac70028dba10f92f6cdd059096db198c283794ac86", size = 153471, upload-time = "2025-10-14T04:40:59.468Z" }, + { url = "https://files.pythonhosted.org/packages/61/fa/fbf177b55bdd727010f9c0a3c49eefa1d10f960e5f09d1d887bf93c2e698/charset_normalizer-3.4.4-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:ac1c4a689edcc530fc9d9aa11f5774b9e2f33f9a0c6a57864e90908f5208d30a", size = 150864, upload-time = "2025-10-14T04:41:00.623Z" }, + { url = "https://files.pythonhosted.org/packages/05/12/9fbc6a4d39c0198adeebbde20b619790e9236557ca59fc40e0e3cebe6f40/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:21d142cc6c0ec30d2efee5068ca36c128a30b0f2c53c1c07bd78cb6bc1d3be5f", size = 150647, upload-time = "2025-10-14T04:41:01.754Z" }, + { url = "https://files.pythonhosted.org/packages/ad/1f/6a9a593d52e3e8c5d2b167daf8c6b968808efb57ef4c210acb907c365bc4/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:5dbe56a36425d26d6cfb40ce79c314a2e4dd6211d51d6d2191c00bed34f354cc", size = 145110, upload-time = "2025-10-14T04:41:03.231Z" }, + { url = "https://files.pythonhosted.org/packages/30/42/9a52c609e72471b0fc54386dc63c3781a387bb4fe61c20231a4ebcd58bdd/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:5bfbb1b9acf3334612667b61bd3002196fe2a1eb4dd74d247e0f2a4d50ec9bbf", size = 162839, upload-time = "2025-10-14T04:41:04.715Z" }, + { url = "https://files.pythonhosted.org/packages/c4/5b/c0682bbf9f11597073052628ddd38344a3d673fda35a36773f7d19344b23/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:d055ec1e26e441f6187acf818b73564e6e6282709e9bcb5b63f5b23068356a15", size = 150667, upload-time = "2025-10-14T04:41:05.827Z" }, + { url = "https://files.pythonhosted.org/packages/e4/24/a41afeab6f990cf2daf6cb8c67419b63b48cf518e4f56022230840c9bfb2/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:af2d8c67d8e573d6de5bc30cdb27e9b95e49115cd9baad5ddbd1a6207aaa82a9", size = 160535, upload-time = "2025-10-14T04:41:06.938Z" }, + { url = "https://files.pythonhosted.org/packages/2a/e5/6a4ce77ed243c4a50a1fecca6aaaab419628c818a49434be428fe24c9957/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:780236ac706e66881f3b7f2f32dfe90507a09e67d1d454c762cf642e6e1586e0", size = 154816, upload-time = "2025-10-14T04:41:08.101Z" }, + { url = "https://files.pythonhosted.org/packages/a8/ef/89297262b8092b312d29cdb2517cb1237e51db8ecef2e9af5edbe7b683b1/charset_normalizer-3.4.4-cp312-cp312-win32.whl", hash = "sha256:5833d2c39d8896e4e19b689ffc198f08ea58116bee26dea51e362ecc7cd3ed26", size = 99694, upload-time = "2025-10-14T04:41:09.23Z" }, + { url = "https://files.pythonhosted.org/packages/3d/2d/1e5ed9dd3b3803994c155cd9aacb60c82c331bad84daf75bcb9c91b3295e/charset_normalizer-3.4.4-cp312-cp312-win_amd64.whl", hash = "sha256:a79cfe37875f822425b89a82333404539ae63dbdddf97f84dcbc3d339aae9525", size = 107131, upload-time = "2025-10-14T04:41:10.467Z" }, + { url = "https://files.pythonhosted.org/packages/d0/d9/0ed4c7098a861482a7b6a95603edce4c0d9db2311af23da1fb2b75ec26fc/charset_normalizer-3.4.4-cp312-cp312-win_arm64.whl", hash = "sha256:376bec83a63b8021bb5c8ea75e21c4ccb86e7e45ca4eb81146091b56599b80c3", size = 100390, upload-time = "2025-10-14T04:41:11.915Z" }, + { url = "https://files.pythonhosted.org/packages/0a/4c/925909008ed5a988ccbb72dcc897407e5d6d3bd72410d69e051fc0c14647/charset_normalizer-3.4.4-py3-none-any.whl", hash = "sha256:7a32c560861a02ff789ad905a2fe94e3f840803362c84fecf1851cb4cf3dc37f", size = 53402, upload-time = "2025-10-14T04:42:31.76Z" }, ] [[package]] @@ -1031,7 +1038,7 @@ wheels = [ [[package]] name = "clickzetta-connector-python" -version = "0.8.104" +version = "0.8.106" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "future" }, @@ -1045,7 +1052,7 @@ dependencies = [ { name = "urllib3" }, ] wheels = [ - { url = "https://files.pythonhosted.org/packages/8f/94/c7eee2224bdab39d16dfe5bb7687f5525c7ed345b7fe8812e18a2d9a6335/clickzetta_connector_python-0.8.104-py3-none-any.whl", hash = "sha256:ae3e466d990677f96c769ec1c29318237df80c80fe9c1e21ba1eaf42bdef0207", size = 79382, upload-time = "2025-09-10T08:46:39.731Z" }, + { url = "https://files.pythonhosted.org/packages/23/38/749c708619f402d4d582dfa73fbeb64ade77b1f250a93bd064d2a1aa3776/clickzetta_connector_python-0.8.106-py3-none-any.whl", hash = "sha256:120d6700051d97609dbd6655c002ab3bc260b7c8e67d39dfc7191e749563f7b4", size = 78121, upload-time = "2025-10-29T02:38:15.014Z" }, ] [[package]] @@ -1153,33 +1160,33 @@ toml = [ [[package]] name = "crc32c" -version = "2.7.1.post0" +version = "2.8" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/8d/02/5e49cc17a5f6f8cb78b55dd57d50b36416e69051c29bba1eab3e86a01927/crc32c-2.7.1.post0.tar.gz", hash = "sha256:dcaa776413af5790cc55561469cd76306e97b325fe4aa195db535fb3f328e709", size = 46574, upload-time = "2025-10-13T02:06:16.898Z" } +sdist = { url = "https://files.pythonhosted.org/packages/e3/66/7e97aa77af7cf6afbff26e3651b564fe41932599bc2d3dce0b2f73d4829a/crc32c-2.8.tar.gz", hash = "sha256:578728964e59c47c356aeeedee6220e021e124b9d3e8631d95d9a5e5f06e261c", size = 48179, upload-time = "2025-10-17T06:20:13.61Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/77/e6/5723311e6320b35d7755fef8ab10c5d4b55173ce11888e8ddaf62b63091f/crc32c-2.7.1.post0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:65124edce1903484b19109da542d8671a814bdd2cc4006847701449a1f137869", size = 64759, upload-time = "2025-10-13T02:05:01.212Z" }, - { url = "https://files.pythonhosted.org/packages/94/ac/f9550d21a4434b5dad9124ccd6b7cee97ce66bc0cb91a605bf01d9c2475d/crc32c-2.7.1.post0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:a177ee47782f9b83002b08c4d4ba57a6e31dcd96be89d1c6b71f599d9c06bba6", size = 61419, upload-time = "2025-10-13T02:05:02.063Z" }, - { url = "https://files.pythonhosted.org/packages/ef/39/82874155870dc42c917ff842073c2714955c3c3d0629579a7ca3db1bbcb1/crc32c-2.7.1.post0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:65ce2c3f1938b1310c1d592ac97f5e32b2cb67b67ae54ec89e710605f01e91cb", size = 59951, upload-time = "2025-10-13T02:05:03.268Z" }, - { url = "https://files.pythonhosted.org/packages/b9/24/42aa97aac254adeafaa44297654a520db1922dcab4a07bbb965b41d52b66/crc32c-2.7.1.post0-cp311-cp311-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:2c57ac2129a4adc56b8898c524a33525f008a346edc5df2b1ab7b7bfc4e80bbe", size = 78633, upload-time = "2025-10-13T02:05:04.429Z" }, - { url = "https://files.pythonhosted.org/packages/ac/76/a0c8683b9c38e260d23eb8d419d3ca39277e5e640521e9cc7ca407633ee4/crc32c-2.7.1.post0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3dcdcc73ea9f5e5e32cf1d0868315c62274f8f8fb2a1356e6bf2e958fc7f5bc9", size = 80187, upload-time = "2025-10-13T02:05:05.254Z" }, - { url = "https://files.pythonhosted.org/packages/16/05/978a558f580294e521dc432656e8d1b955ddc73f22870d7e767ff9c1a8d4/crc32c-2.7.1.post0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:7cc745faf51a57c056021c005766cd8bacd818213ef424064118747c99a16d70", size = 79263, upload-time = "2025-10-13T02:05:06.121Z" }, - { url = "https://files.pythonhosted.org/packages/cd/48/9efdc54175f56b5aea24fbd9076759e052ca828713590a6bf60f822d40f7/crc32c-2.7.1.post0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:1220cf7e97f453e07f998574e663e822e5602ed591b9a2fd436bf65dcae26168", size = 78527, upload-time = "2025-10-13T02:05:06.978Z" }, - { url = "https://files.pythonhosted.org/packages/0f/88/5accf5fb60559df04d0863496a86b912af37e6b7d160d458da25e473a881/crc32c-2.7.1.post0-cp311-cp311-win32.whl", hash = "sha256:d5087f2bc6954b38dc1ceac9b2ea9c9c4956b4e8ce82d965f4c474aefac2d061", size = 63277, upload-time = "2025-10-13T02:05:07.852Z" }, - { url = "https://files.pythonhosted.org/packages/a3/b2/c037161956d00324198a94962788b5e6a6e76b892d96205b15a37bea0c81/crc32c-2.7.1.post0-cp311-cp311-win_amd64.whl", hash = "sha256:feda0b536b1310b0535085835564918df6ba415e0b230734e1386deb7c614c02", size = 65029, upload-time = "2025-10-13T02:05:08.658Z" }, - { url = "https://files.pythonhosted.org/packages/7d/00/243cc1b15bcadf72bd71cf9a33d425715726b95b5f37a85b306d495362f4/crc32c-2.7.1.post0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:4eda225a4c49901b9baf1af2aec19dd614c527bac81e02c52d1b9f1d6f6d244c", size = 64820, upload-time = "2025-10-13T02:05:09.476Z" }, - { url = "https://files.pythonhosted.org/packages/6e/76/e63deacf3e5dcd38764a1a617fd25749ea83fe20ff42a7912a855a975a0f/crc32c-2.7.1.post0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:e304b07182b915fa9ab5340b51a6845d45331974d73b80a1710405ec8f0b4d44", size = 61474, upload-time = "2025-10-13T02:05:10.44Z" }, - { url = "https://files.pythonhosted.org/packages/c2/96/a341802b0a84fc00f9eca4e7dfdc0f41a69fc226b62ea1661468d4812800/crc32c-2.7.1.post0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1bbd4d2272aa7bdc5527fc3130caf31819e5efad19b6abd7158859f1cc808923", size = 59963, upload-time = "2025-10-13T02:05:11.271Z" }, - { url = "https://files.pythonhosted.org/packages/dc/8a/5e1f6789239935a95a6fb579e5f20dc4032265c5de215cec841d369ad188/crc32c-2.7.1.post0-cp312-cp312-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:eea5fe4f477249f19201b2c1ac9f0df70987593b0dd0e0d15521480500d18455", size = 78461, upload-time = "2025-10-13T02:05:12.077Z" }, - { url = "https://files.pythonhosted.org/packages/e8/7a/bf07239d7f55cf94ad6979de1f97105cdcfa1b73cf8818a5461f37043962/crc32c-2.7.1.post0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:dc97ce3c913eded8f4d19d5da7492ebb7ab7de1eb05749c8e5c48f4999e263e0", size = 79963, upload-time = "2025-10-13T02:05:13.343Z" }, - { url = "https://files.pythonhosted.org/packages/e3/17/09a11007d66767a1d339424560386c99323e904e5e7f0e75ff4a13156d3c/crc32c-2.7.1.post0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:c13bdb21cc11fc2e9b7387fe726e65993f79407b3e4b8c107ee345e9c6cfe399", size = 79040, upload-time = "2025-10-13T02:05:14.216Z" }, - { url = "https://files.pythonhosted.org/packages/b2/ca/4f8d8832524a70f39a20302e171782368fd66474e792b2aaf6bc9bb1ba9d/crc32c-2.7.1.post0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:5f9edc07f0617c212d700e31fc6437811b3036f84931e9b837a14169dd0e8d65", size = 78319, upload-time = "2025-10-13T02:05:15.303Z" }, - { url = "https://files.pythonhosted.org/packages/6d/41/63331e510e31928ae5af30fa3d40bca86b8b7c38164b5b59a57cdb8b5a2e/crc32c-2.7.1.post0-cp312-cp312-win32.whl", hash = "sha256:6d205730d184b5ba9a37ee855883b536e40dbf13817d15e4bab4997149c59d82", size = 63286, upload-time = "2025-10-13T02:05:16.181Z" }, - { url = "https://files.pythonhosted.org/packages/ed/3f/05cb1cd66b98f7165b8d181a164ef2c16b7ef0019a191e6ff8defa4df327/crc32c-2.7.1.post0-cp312-cp312-win_amd64.whl", hash = "sha256:f8c1584fe841883300cd3cb0e8341da5a4c954fc2dcf9e0eb15d3b697d90930e", size = 65034, upload-time = "2025-10-13T02:05:17.078Z" }, - { url = "https://files.pythonhosted.org/packages/54/7f/18a4262600e9f772d2b2d10adff4b002d64e5eaa6f0da5e6ded16086e8ad/crc32c-2.7.1.post0-pp311-pypy311_pp73-macosx_10_15_x86_64.whl", hash = "sha256:700d0637f620be903b596fd145d25664c0e821b9b24d462eaa3beeacb906478f", size = 60777, upload-time = "2025-10-13T02:06:10.957Z" }, - { url = "https://files.pythonhosted.org/packages/1b/de/d9a6fdee4b1058b1922b1395814e010e85cb2c1a6ddb1388cbf7523a9f8f/crc32c-2.7.1.post0-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:213aa16faf30c267579f9b76cfc572162fccd537095a5533e329318c2e5da589", size = 59663, upload-time = "2025-10-13T02:06:11.844Z" }, - { url = "https://files.pythonhosted.org/packages/4e/86/9e71dd8847ee075504a7ab69a101ab7dff7fd46cc22dbbef242ceeb187bf/crc32c-2.7.1.post0-pp311-pypy311_pp73-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:0e1b2b232edd75f3281ab059d2811e4ac674931a1889e0070a2fc73d93c0f204", size = 62539, upload-time = "2025-10-13T02:06:13.075Z" }, - { url = "https://files.pythonhosted.org/packages/67/c0/905905212c0aec771d81df4d88f87008dadeecd6ad628d1e17f9a5acd7dd/crc32c-2.7.1.post0-pp311-pypy311_pp73-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b2e76c1e536f2408c5c5ce796e1a89ef252a438aa011c1f31048aa0783b75626", size = 63248, upload-time = "2025-10-13T02:06:14.764Z" }, - { url = "https://files.pythonhosted.org/packages/f8/ba/6bdc8b946c6db999a0318e620a0f50e2099d9cba3d9c9de05932d12795a5/crc32c-2.7.1.post0-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:a1ea03ed177cb022d859ce86bac6044d5cd68dcf7e22f022e288a96f2bd6fa2f", size = 65049, upload-time = "2025-10-13T02:06:15.672Z" }, + { url = "https://files.pythonhosted.org/packages/dc/0b/5e03b22d913698e9cc563f39b9f6bbd508606bf6b8e9122cd6bf196b87ea/crc32c-2.8-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:e560a97fbb96c9897cb1d9b5076ef12fc12e2e25622530a1afd0de4240f17e1f", size = 66329, upload-time = "2025-10-17T06:19:01.771Z" }, + { url = "https://files.pythonhosted.org/packages/6b/38/2fe0051ffe8c6a650c8b1ac0da31b8802d1dbe5fa40a84e4b6b6f5583db5/crc32c-2.8-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6762d276d90331a490ef7e71ffee53b9c0eb053bd75a272d786f3b08d3fe3671", size = 62988, upload-time = "2025-10-17T06:19:02.953Z" }, + { url = "https://files.pythonhosted.org/packages/3e/30/5837a71c014be83aba1469c58820d287fc836512a0cad6b8fdd43868accd/crc32c-2.8-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:60670569f5ede91e39f48fb0cb4060e05b8d8704dd9e17ede930bf441b2f73ef", size = 61522, upload-time = "2025-10-17T06:19:03.796Z" }, + { url = "https://files.pythonhosted.org/packages/ca/29/63972fc1452778e2092ae998c50cbfc2fc93e3fa9798a0278650cd6169c5/crc32c-2.8-cp311-cp311-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:711743da6ccc70b3c6718c328947b0b6f34a1fe6a6c27cc6c1d69cc226bf70e9", size = 80200, upload-time = "2025-10-17T06:19:04.617Z" }, + { url = "https://files.pythonhosted.org/packages/cb/3a/60eb49d7bdada4122b3ffd45b0df54bdc1b8dd092cda4b069a287bdfcff4/crc32c-2.8-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5eb4094a2054774f13b26f21bf56792bb44fa1fcee6c6ad099387a43ffbfb4fa", size = 81757, upload-time = "2025-10-17T06:19:05.496Z" }, + { url = "https://files.pythonhosted.org/packages/f5/63/6efc1b64429ef7d23bd58b75b7ac24d15df327e3ebbe9c247a0f7b1c2ed1/crc32c-2.8-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:fff15bf2bd3e95780516baae935ed12be88deaa5ebe6143c53eb0d26a7bdc7b7", size = 80830, upload-time = "2025-10-17T06:19:06.621Z" }, + { url = "https://files.pythonhosted.org/packages/e1/eb/0ae9f436f8004f1c88f7429e659a7218a3879bd11a6b18ed1257aad7e98b/crc32c-2.8-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:4c0e11e3826668121fa53e0745635baf5e4f0ded437e8ff63ea56f38fc4f970a", size = 80095, upload-time = "2025-10-17T06:19:07.381Z" }, + { url = "https://files.pythonhosted.org/packages/9e/81/4afc9d468977a4cd94a2eb62908553345009a7c0d30e74463a15d4b48ec3/crc32c-2.8-cp311-cp311-win32.whl", hash = "sha256:38f915336715d1f1353ab07d7d786f8a789b119e273aea106ba55355dfc9101d", size = 64886, upload-time = "2025-10-17T06:19:08.497Z" }, + { url = "https://files.pythonhosted.org/packages/d6/e8/94e839c9f7e767bf8479046a207afd440a08f5c59b52586e1af5e64fa4a0/crc32c-2.8-cp311-cp311-win_amd64.whl", hash = "sha256:60e0a765b1caab8d31b2ea80840639253906a9351d4b861551c8c8625ea20f86", size = 66639, upload-time = "2025-10-17T06:19:09.338Z" }, + { url = "https://files.pythonhosted.org/packages/b6/36/fd18ef23c42926b79c7003e16cb0f79043b5b179c633521343d3b499e996/crc32c-2.8-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:572ffb1b78cce3d88e8d4143e154d31044a44be42cb3f6fbbf77f1e7a941c5ab", size = 66379, upload-time = "2025-10-17T06:19:10.115Z" }, + { url = "https://files.pythonhosted.org/packages/7f/b8/c584958e53f7798dd358f5bdb1bbfc97483134f053ee399d3eeb26cca075/crc32c-2.8-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:cf827b3758ee0c4aacd21ceca0e2da83681f10295c38a10bfeb105f7d98f7a68", size = 63042, upload-time = "2025-10-17T06:19:10.946Z" }, + { url = "https://files.pythonhosted.org/packages/62/e6/6f2af0ec64a668a46c861e5bc778ea3ee42171fedfc5440f791f470fd783/crc32c-2.8-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:106fbd79013e06fa92bc3b51031694fcc1249811ed4364ef1554ee3dd2c7f5a2", size = 61528, upload-time = "2025-10-17T06:19:11.768Z" }, + { url = "https://files.pythonhosted.org/packages/17/8b/4a04bd80a024f1a23978f19ae99407783e06549e361ab56e9c08bba3c1d3/crc32c-2.8-cp312-cp312-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:6dde035f91ffbfe23163e68605ee5a4bb8ceebd71ed54bb1fb1d0526cdd125a2", size = 80028, upload-time = "2025-10-17T06:19:12.554Z" }, + { url = "https://files.pythonhosted.org/packages/21/8f/01c7afdc76ac2007d0e6a98e7300b4470b170480f8188475b597d1f4b4c6/crc32c-2.8-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:e41ebe7c2f0fdcd9f3a3fd206989a36b460b4d3f24816d53e5be6c7dba72c5e1", size = 81531, upload-time = "2025-10-17T06:19:13.406Z" }, + { url = "https://files.pythonhosted.org/packages/32/2b/8f78c5a8cc66486be5f51b6f038fc347c3ba748d3ea68be17a014283c331/crc32c-2.8-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:ecf66cf90266d9c15cea597d5cc86c01917cd1a238dc3c51420c7886fa750d7e", size = 80608, upload-time = "2025-10-17T06:19:14.223Z" }, + { url = "https://files.pythonhosted.org/packages/db/86/fad1a94cdeeeb6b6e2323c87f970186e74bfd6fbfbc247bf5c88ad0873d5/crc32c-2.8-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:59eee5f3a69ad0793d5fa9cdc9b9d743b0cd50edf7fccc0a3988a821fef0208c", size = 79886, upload-time = "2025-10-17T06:19:15.345Z" }, + { url = "https://files.pythonhosted.org/packages/d5/db/1a7cb6757a1e32376fa2dfce00c815ea4ee614a94f9bff8228e37420c183/crc32c-2.8-cp312-cp312-win32.whl", hash = "sha256:a73d03ce3604aa5d7a2698e9057a0eef69f529c46497b27ee1c38158e90ceb76", size = 64896, upload-time = "2025-10-17T06:19:16.457Z" }, + { url = "https://files.pythonhosted.org/packages/bf/8e/2024de34399b2e401a37dcb54b224b56c747b0dc46de4966886827b4d370/crc32c-2.8-cp312-cp312-win_amd64.whl", hash = "sha256:56b3b7d015247962cf58186e06d18c3d75a1a63d709d3233509e1c50a2d36aa2", size = 66645, upload-time = "2025-10-17T06:19:17.235Z" }, + { url = "https://files.pythonhosted.org/packages/a7/1d/dd926c68eb8aac8b142a1a10b8eb62d95212c1cf81775644373fe7cceac2/crc32c-2.8-pp311-pypy311_pp73-macosx_10_15_x86_64.whl", hash = "sha256:5833f4071da7ea182c514ba17d1eee8aec3c5be927d798222fbfbbd0f5eea02c", size = 62345, upload-time = "2025-10-17T06:20:09.39Z" }, + { url = "https://files.pythonhosted.org/packages/51/be/803404e5abea2ef2c15042edca04bbb7f625044cca879e47f186b43887c2/crc32c-2.8-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:1dc4da036126ac07b39dd9d03e93e585ec615a2ad28ff12757aef7de175295a8", size = 61229, upload-time = "2025-10-17T06:20:10.236Z" }, + { url = "https://files.pythonhosted.org/packages/fc/3a/00cc578cd27ed0b22c9be25cef2c24539d92df9fa80ebd67a3fc5419724c/crc32c-2.8-pp311-pypy311_pp73-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:15905fa78344654e241371c47e6ed2411f9eeb2b8095311c68c88eccf541e8b4", size = 64108, upload-time = "2025-10-17T06:20:11.072Z" }, + { url = "https://files.pythonhosted.org/packages/6b/bc/0587ef99a1c7629f95dd0c9d4f3d894de383a0df85831eb16c48a6afdae4/crc32c-2.8-pp311-pypy311_pp73-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:c596f918688821f796434e89b431b1698396c38bf0b56de873621528fe3ecb1e", size = 64815, upload-time = "2025-10-17T06:20:11.919Z" }, + { url = "https://files.pythonhosted.org/packages/73/42/94f2b8b92eae9064fcfb8deef2b971514065bd606231f8857ff8ae02bebd/crc32c-2.8-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:8d23c4fe01b3844cb6e091044bc1cebdef7d16472e058ce12d9fadf10d2614af", size = 66659, upload-time = "2025-10-17T06:20:12.766Z" }, ] [[package]] @@ -1188,51 +1195,64 @@ version = "1.7" source = { registry = "https://pypi.org/simple" } sdist = { url = "https://files.pythonhosted.org/packages/6b/b0/e595ce2a2527e169c3bcd6c33d2473c1918e0b7f6826a043ca1245dd4e5b/crcmod-1.7.tar.gz", hash = "sha256:dc7051a0db5f2bd48665a990d3ec1cc305a466a77358ca4492826f41f283601e", size = 89670, upload-time = "2010-06-27T14:35:29.538Z" } +[[package]] +name = "croniter" +version = "6.0.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "python-dateutil" }, + { name = "pytz" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/ad/2f/44d1ae153a0e27be56be43465e5cb39b9650c781e001e7864389deb25090/croniter-6.0.0.tar.gz", hash = "sha256:37c504b313956114a983ece2c2b07790b1f1094fe9d81cc94739214748255577", size = 64481, upload-time = "2024-12-17T17:17:47.32Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/07/4b/290b4c3efd6417a8b0c284896de19b1d5855e6dbdb97d2a35e68fa42de85/croniter-6.0.0-py2.py3-none-any.whl", hash = "sha256:2f878c3856f17896979b2a4379ba1f09c83e374931ea15cc835c5dd2eee9b368", size = 25468, upload-time = "2024-12-17T17:17:45.359Z" }, +] + [[package]] name = "cryptography" -version = "46.0.2" +version = "46.0.3" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "cffi", marker = "platform_python_implementation != 'PyPy'" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/4a/9b/e301418629f7bfdf72db9e80ad6ed9d1b83c487c471803eaa6464c511a01/cryptography-46.0.2.tar.gz", hash = "sha256:21b6fc8c71a3f9a604f028a329e5560009cc4a3a828bfea5fcba8eb7647d88fe", size = 749293, upload-time = "2025-10-01T00:29:11.856Z" } +sdist = { url = "https://files.pythonhosted.org/packages/9f/33/c00162f49c0e2fe8064a62cb92b93e50c74a72bc370ab92f86112b33ff62/cryptography-46.0.3.tar.gz", hash = "sha256:a8b17438104fed022ce745b362294d9ce35b4c2e45c1d958ad4a4b019285f4a1", size = 749258, upload-time = "2025-10-15T23:18:31.74Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/e0/98/7a8df8c19a335c8028414738490fc3955c0cecbfdd37fcc1b9c3d04bd561/cryptography-46.0.2-cp311-abi3-macosx_10_9_universal2.whl", hash = "sha256:f3e32ab7dd1b1ef67b9232c4cf5e2ee4cd517d4316ea910acaaa9c5712a1c663", size = 7261255, upload-time = "2025-10-01T00:27:22.947Z" }, - { url = "https://files.pythonhosted.org/packages/c6/38/b2adb2aa1baa6706adc3eb746691edd6f90a656a9a65c3509e274d15a2b8/cryptography-46.0.2-cp311-abi3-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:1fd1a69086926b623ef8126b4c33d5399ce9e2f3fac07c9c734c2a4ec38b6d02", size = 4297596, upload-time = "2025-10-01T00:27:25.258Z" }, - { url = "https://files.pythonhosted.org/packages/e4/27/0f190ada240003119488ae66c897b5e97149292988f556aef4a6a2a57595/cryptography-46.0.2-cp311-abi3-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:bb7fb9cd44c2582aa5990cf61a4183e6f54eea3172e54963787ba47287edd135", size = 4450899, upload-time = "2025-10-01T00:27:27.458Z" }, - { url = "https://files.pythonhosted.org/packages/85/d5/e4744105ab02fdf6bb58ba9a816e23b7a633255987310b4187d6745533db/cryptography-46.0.2-cp311-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:9066cfd7f146f291869a9898b01df1c9b0e314bfa182cef432043f13fc462c92", size = 4300382, upload-time = "2025-10-01T00:27:29.091Z" }, - { url = "https://files.pythonhosted.org/packages/33/fb/bf9571065c18c04818cb07de90c43fc042c7977c68e5de6876049559c72f/cryptography-46.0.2-cp311-abi3-manylinux_2_28_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:97e83bf4f2f2c084d8dd792d13841d0a9b241643151686010866bbd076b19659", size = 4017347, upload-time = "2025-10-01T00:27:30.767Z" }, - { url = "https://files.pythonhosted.org/packages/35/72/fc51856b9b16155ca071080e1a3ad0c3a8e86616daf7eb018d9565b99baa/cryptography-46.0.2-cp311-abi3-manylinux_2_28_ppc64le.whl", hash = "sha256:4a766d2a5d8127364fd936572c6e6757682fc5dfcbdba1632d4554943199f2fa", size = 4983500, upload-time = "2025-10-01T00:27:32.741Z" }, - { url = "https://files.pythonhosted.org/packages/c1/53/0f51e926799025e31746d454ab2e36f8c3f0d41592bc65cb9840368d3275/cryptography-46.0.2-cp311-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:fab8f805e9675e61ed8538f192aad70500fa6afb33a8803932999b1049363a08", size = 4482591, upload-time = "2025-10-01T00:27:34.869Z" }, - { url = "https://files.pythonhosted.org/packages/86/96/4302af40b23ab8aa360862251fb8fc450b2a06ff24bc5e261c2007f27014/cryptography-46.0.2-cp311-abi3-manylinux_2_34_aarch64.whl", hash = "sha256:1e3b6428a3d56043bff0bb85b41c535734204e599c1c0977e1d0f261b02f3ad5", size = 4300019, upload-time = "2025-10-01T00:27:37.029Z" }, - { url = "https://files.pythonhosted.org/packages/9b/59/0be12c7fcc4c5e34fe2b665a75bc20958473047a30d095a7657c218fa9e8/cryptography-46.0.2-cp311-abi3-manylinux_2_34_ppc64le.whl", hash = "sha256:1a88634851d9b8de8bb53726f4300ab191d3b2f42595e2581a54b26aba71b7cc", size = 4950006, upload-time = "2025-10-01T00:27:40.272Z" }, - { url = "https://files.pythonhosted.org/packages/55/1d/42fda47b0111834b49e31590ae14fd020594d5e4dadd639bce89ad790fba/cryptography-46.0.2-cp311-abi3-manylinux_2_34_x86_64.whl", hash = "sha256:be939b99d4e091eec9a2bcf41aaf8f351f312cd19ff74b5c83480f08a8a43e0b", size = 4482088, upload-time = "2025-10-01T00:27:42.668Z" }, - { url = "https://files.pythonhosted.org/packages/17/50/60f583f69aa1602c2bdc7022dae86a0d2b837276182f8c1ec825feb9b874/cryptography-46.0.2-cp311-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:9f13b040649bc18e7eb37936009b24fd31ca095a5c647be8bb6aaf1761142bd1", size = 4425599, upload-time = "2025-10-01T00:27:44.616Z" }, - { url = "https://files.pythonhosted.org/packages/d1/57/d8d4134cd27e6e94cf44adb3f3489f935bde85f3a5508e1b5b43095b917d/cryptography-46.0.2-cp311-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:9bdc25e4e01b261a8fda4e98618f1c9515febcecebc9566ddf4a70c63967043b", size = 4697458, upload-time = "2025-10-01T00:27:46.209Z" }, - { url = "https://files.pythonhosted.org/packages/d1/2b/531e37408573e1da33adfb4c58875013ee8ac7d548d1548967d94a0ae5c4/cryptography-46.0.2-cp311-abi3-win32.whl", hash = "sha256:8b9bf67b11ef9e28f4d78ff88b04ed0929fcd0e4f70bb0f704cfc32a5c6311ee", size = 3056077, upload-time = "2025-10-01T00:27:48.424Z" }, - { url = "https://files.pythonhosted.org/packages/a8/cd/2f83cafd47ed2dc5a3a9c783ff5d764e9e70d3a160e0df9a9dcd639414ce/cryptography-46.0.2-cp311-abi3-win_amd64.whl", hash = "sha256:758cfc7f4c38c5c5274b55a57ef1910107436f4ae842478c4989abbd24bd5acb", size = 3512585, upload-time = "2025-10-01T00:27:50.521Z" }, - { url = "https://files.pythonhosted.org/packages/00/36/676f94e10bfaa5c5b86c469ff46d3e0663c5dc89542f7afbadac241a3ee4/cryptography-46.0.2-cp311-abi3-win_arm64.whl", hash = "sha256:218abd64a2e72f8472c2102febb596793347a3e65fafbb4ad50519969da44470", size = 2927474, upload-time = "2025-10-01T00:27:52.91Z" }, - { url = "https://files.pythonhosted.org/packages/d5/bb/fa95abcf147a1b0bb94d95f53fbb09da77b24c776c5d87d36f3d94521d2c/cryptography-46.0.2-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:a08e7401a94c002e79dc3bc5231b6558cd4b2280ee525c4673f650a37e2c7685", size = 7248090, upload-time = "2025-10-01T00:28:22.846Z" }, - { url = "https://files.pythonhosted.org/packages/b7/66/f42071ce0e3ffbfa80a88feadb209c779fda92a23fbc1e14f74ebf72ef6b/cryptography-46.0.2-cp38-abi3-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:d30bc11d35743bf4ddf76674a0a369ec8a21f87aaa09b0661b04c5f6c46e8d7b", size = 4293123, upload-time = "2025-10-01T00:28:25.072Z" }, - { url = "https://files.pythonhosted.org/packages/a8/5d/1fdbd2e5c1ba822828d250e5a966622ef00185e476d1cd2726b6dd135e53/cryptography-46.0.2-cp38-abi3-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:bca3f0ce67e5a2a2cf524e86f44697c4323a86e0fd7ba857de1c30d52c11ede1", size = 4439524, upload-time = "2025-10-01T00:28:26.808Z" }, - { url = "https://files.pythonhosted.org/packages/c8/c1/5e4989a7d102d4306053770d60f978c7b6b1ea2ff8c06e0265e305b23516/cryptography-46.0.2-cp38-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:ff798ad7a957a5021dcbab78dfff681f0cf15744d0e6af62bd6746984d9c9e9c", size = 4297264, upload-time = "2025-10-01T00:28:29.327Z" }, - { url = "https://files.pythonhosted.org/packages/28/78/b56f847d220cb1d6d6aef5a390e116ad603ce13a0945a3386a33abc80385/cryptography-46.0.2-cp38-abi3-manylinux_2_28_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:cb5e8daac840e8879407acbe689a174f5ebaf344a062f8918e526824eb5d97af", size = 4011872, upload-time = "2025-10-01T00:28:31.479Z" }, - { url = "https://files.pythonhosted.org/packages/e1/80/2971f214b066b888944f7b57761bf709ee3f2cf805619a18b18cab9b263c/cryptography-46.0.2-cp38-abi3-manylinux_2_28_ppc64le.whl", hash = "sha256:3f37aa12b2d91e157827d90ce78f6180f0c02319468a0aea86ab5a9566da644b", size = 4978458, upload-time = "2025-10-01T00:28:33.267Z" }, - { url = "https://files.pythonhosted.org/packages/a5/84/0cb0a2beaa4f1cbe63ebec4e97cd7e0e9f835d0ba5ee143ed2523a1e0016/cryptography-46.0.2-cp38-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:5e38f203160a48b93010b07493c15f2babb4e0f2319bbd001885adb3f3696d21", size = 4472195, upload-time = "2025-10-01T00:28:36.039Z" }, - { url = "https://files.pythonhosted.org/packages/30/8b/2b542ddbf78835c7cd67b6fa79e95560023481213a060b92352a61a10efe/cryptography-46.0.2-cp38-abi3-manylinux_2_34_aarch64.whl", hash = "sha256:d19f5f48883752b5ab34cff9e2f7e4a7f216296f33714e77d1beb03d108632b6", size = 4296791, upload-time = "2025-10-01T00:28:37.732Z" }, - { url = "https://files.pythonhosted.org/packages/78/12/9065b40201b4f4876e93b9b94d91feb18de9150d60bd842a16a21565007f/cryptography-46.0.2-cp38-abi3-manylinux_2_34_ppc64le.whl", hash = "sha256:04911b149eae142ccd8c9a68892a70c21613864afb47aba92d8c7ed9cc001023", size = 4939629, upload-time = "2025-10-01T00:28:39.654Z" }, - { url = "https://files.pythonhosted.org/packages/f6/9e/6507dc048c1b1530d372c483dfd34e7709fc542765015425f0442b08547f/cryptography-46.0.2-cp38-abi3-manylinux_2_34_x86_64.whl", hash = "sha256:8b16c1ede6a937c291d41176934268e4ccac2c6521c69d3f5961c5a1e11e039e", size = 4471988, upload-time = "2025-10-01T00:28:41.822Z" }, - { url = "https://files.pythonhosted.org/packages/b1/86/d025584a5f7d5c5ec8d3633dbcdce83a0cd579f1141ceada7817a4c26934/cryptography-46.0.2-cp38-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:747b6f4a4a23d5a215aadd1d0b12233b4119c4313df83ab4137631d43672cc90", size = 4422989, upload-time = "2025-10-01T00:28:43.608Z" }, - { url = "https://files.pythonhosted.org/packages/4b/39/536370418b38a15a61bbe413006b79dfc3d2b4b0eafceb5581983f973c15/cryptography-46.0.2-cp38-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:6b275e398ab3a7905e168c036aad54b5969d63d3d9099a0a66cc147a3cc983be", size = 4685578, upload-time = "2025-10-01T00:28:45.361Z" }, - { url = "https://files.pythonhosted.org/packages/15/52/ea7e2b1910f547baed566c866fbb86de2402e501a89ecb4871ea7f169a81/cryptography-46.0.2-cp38-abi3-win32.whl", hash = "sha256:0b507c8e033307e37af61cb9f7159b416173bdf5b41d11c4df2e499a1d8e007c", size = 3036711, upload-time = "2025-10-01T00:28:47.096Z" }, - { url = "https://files.pythonhosted.org/packages/71/9e/171f40f9c70a873e73c2efcdbe91e1d4b1777a03398fa1c4af3c56a2477a/cryptography-46.0.2-cp38-abi3-win_amd64.whl", hash = "sha256:f9b2dc7668418fb6f221e4bf701f716e05e8eadb4f1988a2487b11aedf8abe62", size = 3500007, upload-time = "2025-10-01T00:28:48.967Z" }, - { url = "https://files.pythonhosted.org/packages/3e/7c/15ad426257615f9be8caf7f97990cf3dcbb5b8dd7ed7e0db581a1c4759dd/cryptography-46.0.2-cp38-abi3-win_arm64.whl", hash = "sha256:91447f2b17e83c9e0c89f133119d83f94ce6e0fb55dd47da0a959316e6e9cfa1", size = 2918153, upload-time = "2025-10-01T00:28:51.003Z" }, - { url = "https://files.pythonhosted.org/packages/b7/8c/1aabe338149a7d0f52c3e30f2880b20027ca2a485316756ed6f000462db3/cryptography-46.0.2-pp311-pypy311_pp73-macosx_10_9_x86_64.whl", hash = "sha256:1d3b3edd145953832e09607986f2bd86f85d1dc9c48ced41808b18009d9f30e5", size = 3714495, upload-time = "2025-10-01T00:28:57.222Z" }, - { url = "https://files.pythonhosted.org/packages/e3/0a/0d10eb970fe3e57da9e9ddcfd9464c76f42baf7b3d0db4a782d6746f788f/cryptography-46.0.2-pp311-pypy311_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:fe245cf4a73c20592f0f48da39748b3513db114465be78f0a36da847221bd1b4", size = 4243379, upload-time = "2025-10-01T00:28:58.989Z" }, - { url = "https://files.pythonhosted.org/packages/7d/60/e274b4d41a9eb82538b39950a74ef06e9e4d723cb998044635d9deb1b435/cryptography-46.0.2-pp311-pypy311_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:2b9cad9cf71d0c45566624ff76654e9bae5f8a25970c250a26ccfc73f8553e2d", size = 4409533, upload-time = "2025-10-01T00:29:00.785Z" }, - { url = "https://files.pythonhosted.org/packages/19/9a/fb8548f762b4749aebd13b57b8f865de80258083fe814957f9b0619cfc56/cryptography-46.0.2-pp311-pypy311_pp73-manylinux_2_34_aarch64.whl", hash = "sha256:9bd26f2f75a925fdf5e0a446c0de2714f17819bf560b44b7480e4dd632ad6c46", size = 4243120, upload-time = "2025-10-01T00:29:02.515Z" }, - { url = "https://files.pythonhosted.org/packages/71/60/883f24147fd4a0c5cab74ac7e36a1ff3094a54ba5c3a6253d2ff4b19255b/cryptography-46.0.2-pp311-pypy311_pp73-manylinux_2_34_x86_64.whl", hash = "sha256:7282d8f092b5be7172d6472f29b0631f39f18512a3642aefe52c3c0e0ccfad5a", size = 4408940, upload-time = "2025-10-01T00:29:04.42Z" }, - { url = "https://files.pythonhosted.org/packages/d9/b5/c5e179772ec38adb1c072b3aa13937d2860509ba32b2462bf1dda153833b/cryptography-46.0.2-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:c4b93af7920cdf80f71650769464ccf1fb49a4b56ae0024173c24c48eb6b1612", size = 3438518, upload-time = "2025-10-01T00:29:06.139Z" }, + { url = "https://files.pythonhosted.org/packages/1d/42/9c391dd801d6cf0d561b5890549d4b27bafcc53b39c31a817e69d87c625b/cryptography-46.0.3-cp311-abi3-macosx_10_9_universal2.whl", hash = "sha256:109d4ddfadf17e8e7779c39f9b18111a09efb969a301a31e987416a0191ed93a", size = 7225004, upload-time = "2025-10-15T23:16:52.239Z" }, + { url = "https://files.pythonhosted.org/packages/1c/67/38769ca6b65f07461eb200e85fc1639b438bdc667be02cf7f2cd6a64601c/cryptography-46.0.3-cp311-abi3-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:09859af8466b69bc3c27bdf4f5d84a665e0f7ab5088412e9e2ec49758eca5cbc", size = 4296667, upload-time = "2025-10-15T23:16:54.369Z" }, + { url = "https://files.pythonhosted.org/packages/5c/49/498c86566a1d80e978b42f0d702795f69887005548c041636df6ae1ca64c/cryptography-46.0.3-cp311-abi3-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:01ca9ff2885f3acc98c29f1860552e37f6d7c7d013d7334ff2a9de43a449315d", size = 4450807, upload-time = "2025-10-15T23:16:56.414Z" }, + { url = "https://files.pythonhosted.org/packages/4b/0a/863a3604112174c8624a2ac3c038662d9e59970c7f926acdcfaed8d61142/cryptography-46.0.3-cp311-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:6eae65d4c3d33da080cff9c4ab1f711b15c1d9760809dad6ea763f3812d254cb", size = 4299615, upload-time = "2025-10-15T23:16:58.442Z" }, + { url = "https://files.pythonhosted.org/packages/64/02/b73a533f6b64a69f3cd3872acb6ebc12aef924d8d103133bb3ea750dc703/cryptography-46.0.3-cp311-abi3-manylinux_2_28_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:e5bf0ed4490068a2e72ac03d786693adeb909981cc596425d09032d372bcc849", size = 4016800, upload-time = "2025-10-15T23:17:00.378Z" }, + { url = "https://files.pythonhosted.org/packages/25/d5/16e41afbfa450cde85a3b7ec599bebefaef16b5c6ba4ec49a3532336ed72/cryptography-46.0.3-cp311-abi3-manylinux_2_28_ppc64le.whl", hash = "sha256:5ecfccd2329e37e9b7112a888e76d9feca2347f12f37918facbb893d7bb88ee8", size = 4984707, upload-time = "2025-10-15T23:17:01.98Z" }, + { url = "https://files.pythonhosted.org/packages/c9/56/e7e69b427c3878352c2fb9b450bd0e19ed552753491d39d7d0a2f5226d41/cryptography-46.0.3-cp311-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:a2c0cd47381a3229c403062f764160d57d4d175e022c1df84e168c6251a22eec", size = 4482541, upload-time = "2025-10-15T23:17:04.078Z" }, + { url = "https://files.pythonhosted.org/packages/78/f6/50736d40d97e8483172f1bb6e698895b92a223dba513b0ca6f06b2365339/cryptography-46.0.3-cp311-abi3-manylinux_2_34_aarch64.whl", hash = "sha256:549e234ff32571b1f4076ac269fcce7a808d3bf98b76c8dd560e42dbc66d7d91", size = 4299464, upload-time = "2025-10-15T23:17:05.483Z" }, + { url = "https://files.pythonhosted.org/packages/00/de/d8e26b1a855f19d9994a19c702fa2e93b0456beccbcfe437eda00e0701f2/cryptography-46.0.3-cp311-abi3-manylinux_2_34_ppc64le.whl", hash = "sha256:c0a7bb1a68a5d3471880e264621346c48665b3bf1c3759d682fc0864c540bd9e", size = 4950838, upload-time = "2025-10-15T23:17:07.425Z" }, + { url = "https://files.pythonhosted.org/packages/8f/29/798fc4ec461a1c9e9f735f2fc58741b0daae30688f41b2497dcbc9ed1355/cryptography-46.0.3-cp311-abi3-manylinux_2_34_x86_64.whl", hash = "sha256:10b01676fc208c3e6feeb25a8b83d81767e8059e1fe86e1dc62d10a3018fa926", size = 4481596, upload-time = "2025-10-15T23:17:09.343Z" }, + { url = "https://files.pythonhosted.org/packages/15/8d/03cd48b20a573adfff7652b76271078e3045b9f49387920e7f1f631d125e/cryptography-46.0.3-cp311-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:0abf1ffd6e57c67e92af68330d05760b7b7efb243aab8377e583284dbab72c71", size = 4426782, upload-time = "2025-10-15T23:17:11.22Z" }, + { url = "https://files.pythonhosted.org/packages/fa/b1/ebacbfe53317d55cf33165bda24c86523497a6881f339f9aae5c2e13e57b/cryptography-46.0.3-cp311-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:a04bee9ab6a4da801eb9b51f1b708a1b5b5c9eb48c03f74198464c66f0d344ac", size = 4698381, upload-time = "2025-10-15T23:17:12.829Z" }, + { url = "https://files.pythonhosted.org/packages/96/92/8a6a9525893325fc057a01f654d7efc2c64b9de90413adcf605a85744ff4/cryptography-46.0.3-cp311-abi3-win32.whl", hash = "sha256:f260d0d41e9b4da1ed1e0f1ce571f97fe370b152ab18778e9e8f67d6af432018", size = 3055988, upload-time = "2025-10-15T23:17:14.65Z" }, + { url = "https://files.pythonhosted.org/packages/7e/bf/80fbf45253ea585a1e492a6a17efcb93467701fa79e71550a430c5e60df0/cryptography-46.0.3-cp311-abi3-win_amd64.whl", hash = "sha256:a9a3008438615669153eb86b26b61e09993921ebdd75385ddd748702c5adfddb", size = 3514451, upload-time = "2025-10-15T23:17:16.142Z" }, + { url = "https://files.pythonhosted.org/packages/2e/af/9b302da4c87b0beb9db4e756386a7c6c5b8003cd0e742277888d352ae91d/cryptography-46.0.3-cp311-abi3-win_arm64.whl", hash = "sha256:5d7f93296ee28f68447397bf5198428c9aeeab45705a55d53a6343455dcb2c3c", size = 2928007, upload-time = "2025-10-15T23:17:18.04Z" }, + { url = "https://files.pythonhosted.org/packages/fd/23/45fe7f376a7df8daf6da3556603b36f53475a99ce4faacb6ba2cf3d82021/cryptography-46.0.3-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:cb3d760a6117f621261d662bccc8ef5bc32ca673e037c83fbe565324f5c46936", size = 7218248, upload-time = "2025-10-15T23:17:46.294Z" }, + { url = "https://files.pythonhosted.org/packages/27/32/b68d27471372737054cbd34c84981f9edbc24fe67ca225d389799614e27f/cryptography-46.0.3-cp38-abi3-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:4b7387121ac7d15e550f5cb4a43aef2559ed759c35df7336c402bb8275ac9683", size = 4294089, upload-time = "2025-10-15T23:17:48.269Z" }, + { url = "https://files.pythonhosted.org/packages/26/42/fa8389d4478368743e24e61eea78846a0006caffaf72ea24a15159215a14/cryptography-46.0.3-cp38-abi3-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:15ab9b093e8f09daab0f2159bb7e47532596075139dd74365da52ecc9cb46c5d", size = 4440029, upload-time = "2025-10-15T23:17:49.837Z" }, + { url = "https://files.pythonhosted.org/packages/5f/eb/f483db0ec5ac040824f269e93dd2bd8a21ecd1027e77ad7bdf6914f2fd80/cryptography-46.0.3-cp38-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:46acf53b40ea38f9c6c229599a4a13f0d46a6c3fa9ef19fc1a124d62e338dfa0", size = 4297222, upload-time = "2025-10-15T23:17:51.357Z" }, + { url = "https://files.pythonhosted.org/packages/fd/cf/da9502c4e1912cb1da3807ea3618a6829bee8207456fbbeebc361ec38ba3/cryptography-46.0.3-cp38-abi3-manylinux_2_28_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:10ca84c4668d066a9878890047f03546f3ae0a6b8b39b697457b7757aaf18dbc", size = 4012280, upload-time = "2025-10-15T23:17:52.964Z" }, + { url = "https://files.pythonhosted.org/packages/6b/8f/9adb86b93330e0df8b3dcf03eae67c33ba89958fc2e03862ef1ac2b42465/cryptography-46.0.3-cp38-abi3-manylinux_2_28_ppc64le.whl", hash = "sha256:36e627112085bb3b81b19fed209c05ce2a52ee8b15d161b7c643a7d5a88491f3", size = 4978958, upload-time = "2025-10-15T23:17:54.965Z" }, + { url = "https://files.pythonhosted.org/packages/d1/a0/5fa77988289c34bdb9f913f5606ecc9ada1adb5ae870bd0d1054a7021cc4/cryptography-46.0.3-cp38-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:1000713389b75c449a6e979ffc7dcc8ac90b437048766cef052d4d30b8220971", size = 4473714, upload-time = "2025-10-15T23:17:56.754Z" }, + { url = "https://files.pythonhosted.org/packages/14/e5/fc82d72a58d41c393697aa18c9abe5ae1214ff6f2a5c18ac470f92777895/cryptography-46.0.3-cp38-abi3-manylinux_2_34_aarch64.whl", hash = "sha256:b02cf04496f6576afffef5ddd04a0cb7d49cf6be16a9059d793a30b035f6b6ac", size = 4296970, upload-time = "2025-10-15T23:17:58.588Z" }, + { url = "https://files.pythonhosted.org/packages/78/06/5663ed35438d0b09056973994f1aec467492b33bd31da36e468b01ec1097/cryptography-46.0.3-cp38-abi3-manylinux_2_34_ppc64le.whl", hash = "sha256:71e842ec9bc7abf543b47cf86b9a743baa95f4677d22baa4c7d5c69e49e9bc04", size = 4940236, upload-time = "2025-10-15T23:18:00.897Z" }, + { url = "https://files.pythonhosted.org/packages/fc/59/873633f3f2dcd8a053b8dd1d38f783043b5fce589c0f6988bf55ef57e43e/cryptography-46.0.3-cp38-abi3-manylinux_2_34_x86_64.whl", hash = "sha256:402b58fc32614f00980b66d6e56a5b4118e6cb362ae8f3fda141ba4689bd4506", size = 4472642, upload-time = "2025-10-15T23:18:02.749Z" }, + { url = "https://files.pythonhosted.org/packages/3d/39/8e71f3930e40f6877737d6f69248cf74d4e34b886a3967d32f919cc50d3b/cryptography-46.0.3-cp38-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:ef639cb3372f69ec44915fafcd6698b6cc78fbe0c2ea41be867f6ed612811963", size = 4423126, upload-time = "2025-10-15T23:18:04.85Z" }, + { url = "https://files.pythonhosted.org/packages/cd/c7/f65027c2810e14c3e7268353b1681932b87e5a48e65505d8cc17c99e36ae/cryptography-46.0.3-cp38-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:3b51b8ca4f1c6453d8829e1eb7299499ca7f313900dd4d89a24b8b87c0a780d4", size = 4686573, upload-time = "2025-10-15T23:18:06.908Z" }, + { url = "https://files.pythonhosted.org/packages/0a/6e/1c8331ddf91ca4730ab3086a0f1be19c65510a33b5a441cb334e7a2d2560/cryptography-46.0.3-cp38-abi3-win32.whl", hash = "sha256:6276eb85ef938dc035d59b87c8a7dc559a232f954962520137529d77b18ff1df", size = 3036695, upload-time = "2025-10-15T23:18:08.672Z" }, + { url = "https://files.pythonhosted.org/packages/90/45/b0d691df20633eff80955a0fc7695ff9051ffce8b69741444bd9ed7bd0db/cryptography-46.0.3-cp38-abi3-win_amd64.whl", hash = "sha256:416260257577718c05135c55958b674000baef9a1c7d9e8f306ec60d71db850f", size = 3501720, upload-time = "2025-10-15T23:18:10.632Z" }, + { url = "https://files.pythonhosted.org/packages/e8/cb/2da4cc83f5edb9c3257d09e1e7ab7b23f049c7962cae8d842bbef0a9cec9/cryptography-46.0.3-cp38-abi3-win_arm64.whl", hash = "sha256:d89c3468de4cdc4f08a57e214384d0471911a3830fcdaf7a8cc587e42a866372", size = 2918740, upload-time = "2025-10-15T23:18:12.277Z" }, + { url = "https://files.pythonhosted.org/packages/06/8a/e60e46adab4362a682cf142c7dcb5bf79b782ab2199b0dcb81f55970807f/cryptography-46.0.3-pp311-pypy311_pp73-macosx_10_9_x86_64.whl", hash = "sha256:7ce938a99998ed3c8aa7e7272dca1a610401ede816d36d0693907d863b10d9ea", size = 3698132, upload-time = "2025-10-15T23:18:17.056Z" }, + { url = "https://files.pythonhosted.org/packages/da/38/f59940ec4ee91e93d3311f7532671a5cef5570eb04a144bf203b58552d11/cryptography-46.0.3-pp311-pypy311_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:191bb60a7be5e6f54e30ba16fdfae78ad3a342a0599eb4193ba88e3f3d6e185b", size = 4243992, upload-time = "2025-10-15T23:18:18.695Z" }, + { url = "https://files.pythonhosted.org/packages/b0/0c/35b3d92ddebfdfda76bb485738306545817253d0a3ded0bfe80ef8e67aa5/cryptography-46.0.3-pp311-pypy311_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:c70cc23f12726be8f8bc72e41d5065d77e4515efae3690326764ea1b07845cfb", size = 4409944, upload-time = "2025-10-15T23:18:20.597Z" }, + { url = "https://files.pythonhosted.org/packages/99/55/181022996c4063fc0e7666a47049a1ca705abb9c8a13830f074edb347495/cryptography-46.0.3-pp311-pypy311_pp73-manylinux_2_34_aarch64.whl", hash = "sha256:9394673a9f4de09e28b5356e7fff97d778f8abad85c9d5ac4a4b7e25a0de7717", size = 4242957, upload-time = "2025-10-15T23:18:22.18Z" }, + { url = "https://files.pythonhosted.org/packages/ba/af/72cd6ef29f9c5f731251acadaeb821559fe25f10852f44a63374c9ca08c1/cryptography-46.0.3-pp311-pypy311_pp73-manylinux_2_34_x86_64.whl", hash = "sha256:94cd0549accc38d1494e1f8de71eca837d0509d0d44bf11d158524b0e12cebf9", size = 4409447, upload-time = "2025-10-15T23:18:24.209Z" }, + { url = "https://files.pythonhosted.org/packages/0d/c3/e90f4a4feae6410f914f8ebac129b9ae7a8c92eb60a638012dde42030a9d/cryptography-46.0.3-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:6b5063083824e5509fdba180721d55909ffacccc8adbec85268b48439423d78c", size = 3438528, upload-time = "2025-10-15T23:18:26.227Z" }, ] [[package]] @@ -1268,14 +1288,14 @@ wheels = [ [[package]] name = "deprecated" -version = "1.2.18" +version = "1.3.1" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "wrapt" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/98/97/06afe62762c9a8a86af0cfb7bfdab22a43ad17138b07af5b1a58442690a2/deprecated-1.2.18.tar.gz", hash = "sha256:422b6f6d859da6f2ef57857761bfb392480502a64c3028ca9bbe86085d72115d", size = 2928744, upload-time = "2025-01-27T10:46:25.7Z" } +sdist = { url = "https://files.pythonhosted.org/packages/49/85/12f0a49a7c4ffb70572b6c2ef13c90c88fd190debda93b23f026b25f9634/deprecated-1.3.1.tar.gz", hash = "sha256:b1b50e0ff0c1fddaa5708a2c6b0a6588bb09b892825ab2b214ac9ea9d92a5223", size = 2932523, upload-time = "2025-10-30T08:19:02.757Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/6e/c6/ac0b6c1e2d138f1002bcf799d330bd6d85084fece321e662a14223794041/Deprecated-1.2.18-py2.py3-none-any.whl", hash = "sha256:bd5011788200372a32418f888e326a09ff80d0214bd961147cfed01b5c018eec", size = 9998, upload-time = "2025-01-27T10:46:09.186Z" }, + { url = "https://files.pythonhosted.org/packages/84/d0/205d54408c08b13550c733c4b85429e7ead111c7f0014309637425520a9a/deprecated-1.3.1-py2.py3-none-any.whl", hash = "sha256:597bfef186b6f60181535a29fbe44865ce137a5079f295b479886c82729d5f3f", size = 11298, upload-time = "2025-10-30T08:19:00.758Z" }, ] [[package]] @@ -1292,9 +1312,10 @@ wheels = [ [[package]] name = "dify-api" -version = "1.9.2" +version = "1.10.0" source = { virtual = "." } dependencies = [ + { name = "apscheduler" }, { name = "arize-phoenix-otel" }, { name = "azure-identity" }, { name = "beautifulsoup4" }, @@ -1303,6 +1324,7 @@ dependencies = [ { name = "cachetools" }, { name = "celery" }, { name = "chardet" }, + { name = "croniter" }, { name = "flask" }, { name = "flask-compress" }, { name = "flask-cors" }, @@ -1326,6 +1348,7 @@ dependencies = [ { name = "json-repair" }, { name = "langfuse" }, { name = "langsmith" }, + { name = "litellm" }, { name = "markdown" }, { name = "numpy" }, { name = "openpyxl" }, @@ -1486,6 +1509,7 @@ vdb = [ [package.metadata] requires-dist = [ + { name = "apscheduler", specifier = ">=3.11.0" }, { name = "arize-phoenix-otel", specifier = "~=0.9.2" }, { name = "azure-identity", specifier = "==1.16.1" }, { name = "beautifulsoup4", specifier = "==4.12.2" }, @@ -1494,6 +1518,7 @@ requires-dist = [ { name = "cachetools", specifier = "~=5.3.0" }, { name = "celery", specifier = "~=5.5.2" }, { name = "chardet", specifier = "~=5.1.0" }, + { name = "croniter", specifier = ">=6.0.0" }, { name = "flask", specifier = "~=3.1.2" }, { name = "flask-compress", specifier = ">=1.17,<1.18" }, { name = "flask-cors", specifier = "~=6.0.0" }, @@ -1517,6 +1542,7 @@ requires-dist = [ { name = "json-repair", specifier = ">=0.41.1" }, { name = "langfuse", specifier = "~=2.51.3" }, { name = "langsmith", specifier = "~=0.1.77" }, + { name = "litellm", specifier = "==1.77.1" }, { name = "markdown", specifier = "~=3.5.1" }, { name = "numpy", specifier = "~=1.26.4" }, { name = "openpyxl", specifier = "~=3.1.5" }, @@ -1558,11 +1584,11 @@ requires-dist = [ { name = "sentry-sdk", extras = ["flask"], specifier = "~=2.28.0" }, { name = "sqlalchemy", specifier = "~=2.0.29" }, { name = "sseclient-py", specifier = "~=1.8.0" }, - { name = "starlette", specifier = "==0.47.2" }, + { name = "starlette", specifier = "==0.49.1" }, { name = "tiktoken", specifier = "~=0.9.0" }, { name = "transformers", specifier = "~=4.56.1" }, { name = "unstructured", extras = ["docx", "epub", "md", "ppt", "pptx"], specifier = "~=0.16.1" }, - { name = "weave", specifier = "~=0.51.0" }, + { name = "weave", specifier = ">=0.52.16" }, { name = "weaviate-client", specifier = "==4.17.0" }, { name = "webvtt-py", specifier = "~=0.5.1" }, { name = "yarl", specifier = "~=1.18.3" }, @@ -1590,7 +1616,7 @@ dev = [ { name = "ruff", specifier = "~=0.14.0" }, { name = "scipy-stubs", specifier = ">=1.15.3.0" }, { name = "sseclient-py", specifier = ">=1.8.0" }, - { name = "testcontainers", specifier = "~=4.10.0" }, + { name = "testcontainers", specifier = "~=4.13.2" }, { name = "ty", specifier = "~=0.0.1a19" }, { name = "types-aiofiles", specifier = "~=24.1.0" }, { name = "types-beautifulsoup4", specifier = "~=4.12.0" }, @@ -1664,9 +1690,9 @@ vdb = [ { name = "pgvector", specifier = "==0.2.5" }, { name = "pymilvus", specifier = "~=2.5.0" }, { name = "pymochow", specifier = "==2.2.9" }, - { name = "pyobvector", specifier = "~=0.2.15" }, + { name = "pyobvector", specifier = "~=0.2.17" }, { name = "qdrant-client", specifier = "==1.9.0" }, - { name = "tablestore", specifier = "==6.2.0" }, + { name = "tablestore", specifier = "==6.3.7" }, { name = "tcvectordb", specifier = "~=1.6.4" }, { name = "tidb-vector", specifier = "==0.0.9" }, { name = "upstash-vector", specifier = "==0.6.0" }, @@ -1819,46 +1845,47 @@ wheels = [ [[package]] name = "fastapi" -version = "0.119.0" +version = "0.121.1" source = { registry = "https://pypi.org/simple" } dependencies = [ + { name = "annotated-doc" }, { name = "pydantic" }, { name = "starlette" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/0a/f9/5c5bcce82a7997cc0eb8c47b7800f862f6b56adc40486ed246e5010d443b/fastapi-0.119.0.tar.gz", hash = "sha256:451082403a2c1f0b99c6bd57c09110ed5463856804c8078d38e5a1f1035dbbb7", size = 336756, upload-time = "2025-10-11T17:13:40.53Z" } +sdist = { url = "https://files.pythonhosted.org/packages/6b/a4/29e1b861fc9017488ed02ff1052feffa40940cb355ed632a8845df84ce84/fastapi-0.121.1.tar.gz", hash = "sha256:b6dba0538fd15dab6fe4d3e5493c3957d8a9e1e9257f56446b5859af66f32441", size = 342523, upload-time = "2025-11-08T21:48:14.068Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/ce/70/584c4d7cad80f5e833715c0a29962d7c93b4d18eed522a02981a6d1b6ee5/fastapi-0.119.0-py3-none-any.whl", hash = "sha256:90a2e49ed19515320abb864df570dd766be0662c5d577688f1600170f7f73cf2", size = 107095, upload-time = "2025-10-11T17:13:39.048Z" }, + { url = "https://files.pythonhosted.org/packages/94/fd/2e6f7d706899cc08690c5f6641e2ffbfffe019e8f16ce77104caa5730910/fastapi-0.121.1-py3-none-any.whl", hash = "sha256:2c5c7028bc3a58d8f5f09aecd3fd88a000ccc0c5ad627693264181a3c33aa1fc", size = 109192, upload-time = "2025-11-08T21:48:12.458Z" }, ] [[package]] name = "fastuuid" -version = "0.13.5" +version = "0.14.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/15/80/3c16a1edad2e6cd82fbd15ac998cc1b881f478bf1f80ca717d941c441874/fastuuid-0.13.5.tar.gz", hash = "sha256:d4976821ab424d41542e1ea39bc828a9d454c3f8a04067c06fca123c5b95a1a1", size = 18255, upload-time = "2025-09-26T09:05:38.281Z" } +sdist = { url = "https://files.pythonhosted.org/packages/c3/7d/d9daedf0f2ebcacd20d599928f8913e9d2aea1d56d2d355a93bfa2b611d7/fastuuid-0.14.0.tar.gz", hash = "sha256:178947fc2f995b38497a74172adee64fdeb8b7ec18f2a5934d037641ba265d26", size = 18232, upload-time = "2025-10-19T22:19:22.402Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/04/ab/9351bfc04ff2144115758233130b5469993d3d379323903a4634cb9c78c1/fastuuid-0.13.5-cp311-cp311-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:c122558ca4b5487e2bd0863467e4ccfe636afd1274803741487d48f2e32ea0e1", size = 493910, upload-time = "2025-09-26T09:12:36.995Z" }, - { url = "https://files.pythonhosted.org/packages/b7/ab/84fac529cc12a03d49595e70ac459380f7cb12c70f0fe401781b276f9e94/fastuuid-0.13.5-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:d7abd42a03a17a681abddd19aa4d44ca2747138cf8a48373b395cf1341a10de2", size = 252621, upload-time = "2025-09-26T09:12:22.222Z" }, - { url = "https://files.pythonhosted.org/packages/7f/9d/f4c734d7b74a04ca695781c58a1376f07b206fe2849e58e7778d476a0e94/fastuuid-0.13.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:2705cf7c2d6f7c03053404b75a4c44f872a73f6f9d5ea34f1dc6bba400c4a97c", size = 244269, upload-time = "2025-09-26T09:08:31.921Z" }, - { url = "https://files.pythonhosted.org/packages/5b/da/b42b7eb84523d69cfe9dac82950e105061c8d59f4d4d2cc3e170dbd20937/fastuuid-0.13.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5d220a056fcbad25932c1f25304261198612f271f4d150b2a84e81adb877daf7", size = 271528, upload-time = "2025-09-26T09:12:42.718Z" }, - { url = "https://files.pythonhosted.org/packages/1b/45/6eee36929119e9544b0906fd6591e685d682e4b51cfad4c25d96ccf04009/fastuuid-0.13.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2f29f93b5a0c5f5579f97f77d5319e9bfefd61d8678ec59d850201544faf33bf", size = 272168, upload-time = "2025-09-26T09:07:04.238Z" }, - { url = "https://files.pythonhosted.org/packages/ce/ac/75b70f13515e12194a25b0459dd8a8a33de4ab0a92142f0776d21e41ca84/fastuuid-0.13.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:399d86623fb806151b1feb9fdd818ebfc1d50387199a35f7264f98dfc1540af5", size = 290948, upload-time = "2025-09-26T09:07:53.433Z" }, - { url = "https://files.pythonhosted.org/packages/76/30/1801326a5b433aafc04eae906e6b005e8a3d1120fd996409fe88124edb06/fastuuid-0.13.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:689e8795a1edd573b2c9a455024e4edf605a9690339bba29709857f7180894ea", size = 452932, upload-time = "2025-09-26T09:09:28.017Z" }, - { url = "https://files.pythonhosted.org/packages/61/2a/080b6b2ac4ef2ead54a7463ae4162d66a52867bbd4447ad5354427b82ae2/fastuuid-0.13.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:25e82c4a1734da168b36f7308e397afbe9c9b353799a9c69563a605f11dd4641", size = 468384, upload-time = "2025-09-26T09:08:14.32Z" }, - { url = "https://files.pythonhosted.org/packages/b6/d3/4a3ffcaf8d874f7f208dad7e98ded7c5359b6599073960e3aa0530ca6139/fastuuid-0.13.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:f62299e3cca69aad6a6fb37e26e45055587954d498ad98903fea24382377ea0e", size = 444815, upload-time = "2025-09-26T09:06:38.691Z" }, - { url = "https://files.pythonhosted.org/packages/9d/a0/08dd8663f7bff3e9c0b2416708b01d1fb65f52bcd4bce18760f77c4735fd/fastuuid-0.13.5-cp311-cp311-win32.whl", hash = "sha256:68227f2230381b89fb1ad362ca6e433de85c6c11c36312b41757cad47b8a8e32", size = 144897, upload-time = "2025-09-26T09:14:53.695Z" }, - { url = "https://files.pythonhosted.org/packages/9e/e2/2c2a37dcc56e2323c6214c38c8faac22f9d03d98c481f8a40843e0b9526a/fastuuid-0.13.5-cp311-cp311-win_amd64.whl", hash = "sha256:4a32306982bd031cb20d5d1a726b7b958a55babebd2300ce6c8e352d3496e931", size = 150523, upload-time = "2025-09-26T09:12:24.031Z" }, - { url = "https://files.pythonhosted.org/packages/21/36/434f137c5970cac19e57834e1f7680e85301619d49891618c00666700c61/fastuuid-0.13.5-cp312-cp312-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:35fe8045e866bc6846f8de6fa05acb1de0c32478048484a995e96d31e21dff2a", size = 494638, upload-time = "2025-09-26T09:14:58.695Z" }, - { url = "https://files.pythonhosted.org/packages/ca/3c/083de2ac007b2b305523b9c006dba5051e5afd87a626ef1a39f76e2c6b82/fastuuid-0.13.5-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:02a460333f52d731a006d18a52ef6fcb2d295a1f5b1a5938d30744191b2f77b7", size = 253138, upload-time = "2025-09-26T09:13:33.283Z" }, - { url = "https://files.pythonhosted.org/packages/73/5e/630cffa1c8775db526e39e9e4c5c7db0c27be0786bb21ba82c912ae19f63/fastuuid-0.13.5-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:74b0e4f8c307b9f477a5d7284db4431ce53a3c1e3f4173db7a97db18564a6202", size = 244521, upload-time = "2025-09-26T09:14:40.682Z" }, - { url = "https://files.pythonhosted.org/packages/4d/51/55d78705f4fbdadf88fb40f382f508d6c7a4941ceddd7825fafebb4cc778/fastuuid-0.13.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6955a99ef455c2986f3851f4e0ccc35dec56ac1a7720f2b92e88a75d6684512e", size = 271557, upload-time = "2025-09-26T09:15:09.75Z" }, - { url = "https://files.pythonhosted.org/packages/6a/2b/1b89e90a8635e5587ccdbbeb169c590672ce7637880f2c047482a0359950/fastuuid-0.13.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f10c77b826738c1a27dcdaa92ea4dc1ec9d869748a99e1fde54f1379553d4854", size = 272334, upload-time = "2025-09-26T09:07:48.865Z" }, - { url = "https://files.pythonhosted.org/packages/0c/06/4c8207894eeb30414999e5c3f66ac039bc4003437eb4060d8a1bceb4cc6f/fastuuid-0.13.5-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:bb25dccbeb249d16d5e664f65f17ebec05136821d5ef462c4110e3f76b86fb86", size = 290594, upload-time = "2025-09-26T09:12:54.124Z" }, - { url = "https://files.pythonhosted.org/packages/50/69/96d221931a31d77a47cc2487bdfacfb3091edfc2e7a04b1795df1aec05df/fastuuid-0.13.5-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:a5becc646a3eeafb76ce0a6783ba190cd182e3790a8b2c78ca9db2b5e87af952", size = 452835, upload-time = "2025-09-26T09:14:00.994Z" }, - { url = "https://files.pythonhosted.org/packages/25/ef/bf045f0a47dcec96247497ef3f7a31d86ebc074330e2dccc34b8dbc0468a/fastuuid-0.13.5-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:69b34363752d06e9bb0dbdf02ae391ec56ac948c6f2eb00be90dad68e80774b9", size = 468225, upload-time = "2025-09-26T09:13:38.585Z" }, - { url = "https://files.pythonhosted.org/packages/30/46/4817ab5a3778927155a4bde92540d4c4fa996161ec8b8e080c8928b0984e/fastuuid-0.13.5-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:57d0768afcad0eab8770c9b8cf904716bd3c547e8b9a4e755ee8a673b060a3a3", size = 444907, upload-time = "2025-09-26T09:14:30.163Z" }, - { url = "https://files.pythonhosted.org/packages/80/27/ab284117ce4dc9b356a7196bdbf220510285f201d27f1f078592cdc8187b/fastuuid-0.13.5-cp312-cp312-win32.whl", hash = "sha256:8ac6c6f5129d52eaa6ef9ea4b6e2f7c69468a053f3ab8e439661186b9c06bb85", size = 145415, upload-time = "2025-09-26T09:08:59.494Z" }, - { url = "https://files.pythonhosted.org/packages/f4/0c/f970a4222773b248931819f8940800b760283216ca3dda173ed027e94bdd/fastuuid-0.13.5-cp312-cp312-win_amd64.whl", hash = "sha256:ad630e97715beefef07ec37c9c162336e500400774e2c1cbe1a0df6f80d15b9a", size = 150840, upload-time = "2025-09-26T09:13:46.115Z" }, + { url = "https://files.pythonhosted.org/packages/98/f3/12481bda4e5b6d3e698fbf525df4443cc7dce746f246b86b6fcb2fba1844/fastuuid-0.14.0-cp311-cp311-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:73946cb950c8caf65127d4e9a325e2b6be0442a224fd51ba3b6ac44e1912ce34", size = 516386, upload-time = "2025-10-19T22:42:40.176Z" }, + { url = "https://files.pythonhosted.org/packages/59/19/2fc58a1446e4d72b655648eb0879b04e88ed6fa70d474efcf550f640f6ec/fastuuid-0.14.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:12ac85024637586a5b69645e7ed986f7535106ed3013640a393a03e461740cb7", size = 264569, upload-time = "2025-10-19T22:25:50.977Z" }, + { url = "https://files.pythonhosted.org/packages/78/29/3c74756e5b02c40cfcc8b1d8b5bac4edbd532b55917a6bcc9113550e99d1/fastuuid-0.14.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:05a8dde1f395e0c9b4be515b7a521403d1e8349443e7641761af07c7ad1624b1", size = 254366, upload-time = "2025-10-19T22:29:49.166Z" }, + { url = "https://files.pythonhosted.org/packages/52/96/d761da3fccfa84f0f353ce6e3eb8b7f76b3aa21fd25e1b00a19f9c80a063/fastuuid-0.14.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:09378a05020e3e4883dfdab438926f31fea15fd17604908f3d39cbeb22a0b4dc", size = 278978, upload-time = "2025-10-19T22:35:41.306Z" }, + { url = "https://files.pythonhosted.org/packages/fc/c2/f84c90167cc7765cb82b3ff7808057608b21c14a38531845d933a4637307/fastuuid-0.14.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bbb0c4b15d66b435d2538f3827f05e44e2baafcc003dd7d8472dc67807ab8fd8", size = 279692, upload-time = "2025-10-19T22:25:36.997Z" }, + { url = "https://files.pythonhosted.org/packages/af/7b/4bacd03897b88c12348e7bd77943bac32ccf80ff98100598fcff74f75f2e/fastuuid-0.14.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:cd5a7f648d4365b41dbf0e38fe8da4884e57bed4e77c83598e076ac0c93995e7", size = 303384, upload-time = "2025-10-19T22:29:46.578Z" }, + { url = "https://files.pythonhosted.org/packages/c0/a2/584f2c29641df8bd810d00c1f21d408c12e9ad0c0dafdb8b7b29e5ddf787/fastuuid-0.14.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:c0a94245afae4d7af8c43b3159d5e3934c53f47140be0be624b96acd672ceb73", size = 460921, upload-time = "2025-10-19T22:36:42.006Z" }, + { url = "https://files.pythonhosted.org/packages/24/68/c6b77443bb7764c760e211002c8638c0c7cce11cb584927e723215ba1398/fastuuid-0.14.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:2b29e23c97e77c3a9514d70ce343571e469098ac7f5a269320a0f0b3e193ab36", size = 480575, upload-time = "2025-10-19T22:28:18.975Z" }, + { url = "https://files.pythonhosted.org/packages/5a/87/93f553111b33f9bb83145be12868c3c475bf8ea87c107063d01377cc0e8e/fastuuid-0.14.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:1e690d48f923c253f28151b3a6b4e335f2b06bf669c68a02665bc150b7839e94", size = 452317, upload-time = "2025-10-19T22:25:32.75Z" }, + { url = "https://files.pythonhosted.org/packages/9e/8c/a04d486ca55b5abb7eaa65b39df8d891b7b1635b22db2163734dc273579a/fastuuid-0.14.0-cp311-cp311-win32.whl", hash = "sha256:a6f46790d59ab38c6aa0e35c681c0484b50dc0acf9e2679c005d61e019313c24", size = 154804, upload-time = "2025-10-19T22:24:15.615Z" }, + { url = "https://files.pythonhosted.org/packages/9c/b2/2d40bf00820de94b9280366a122cbaa60090c8cf59e89ac3938cf5d75895/fastuuid-0.14.0-cp311-cp311-win_amd64.whl", hash = "sha256:e150eab56c95dc9e3fefc234a0eedb342fac433dacc273cd4d150a5b0871e1fa", size = 156099, upload-time = "2025-10-19T22:24:31.646Z" }, + { url = "https://files.pythonhosted.org/packages/02/a2/e78fcc5df65467f0d207661b7ef86c5b7ac62eea337c0c0fcedbeee6fb13/fastuuid-0.14.0-cp312-cp312-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:77e94728324b63660ebf8adb27055e92d2e4611645bf12ed9d88d30486471d0a", size = 510164, upload-time = "2025-10-19T22:31:45.635Z" }, + { url = "https://files.pythonhosted.org/packages/2b/b3/c846f933f22f581f558ee63f81f29fa924acd971ce903dab1a9b6701816e/fastuuid-0.14.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:caa1f14d2102cb8d353096bc6ef6c13b2c81f347e6ab9d6fbd48b9dea41c153d", size = 261837, upload-time = "2025-10-19T22:38:38.53Z" }, + { url = "https://files.pythonhosted.org/packages/54/ea/682551030f8c4fa9a769d9825570ad28c0c71e30cf34020b85c1f7ee7382/fastuuid-0.14.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:d23ef06f9e67163be38cece704170486715b177f6baae338110983f99a72c070", size = 251370, upload-time = "2025-10-19T22:40:26.07Z" }, + { url = "https://files.pythonhosted.org/packages/14/dd/5927f0a523d8e6a76b70968e6004966ee7df30322f5fc9b6cdfb0276646a/fastuuid-0.14.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0c9ec605ace243b6dbe3bd27ebdd5d33b00d8d1d3f580b39fdd15cd96fd71796", size = 277766, upload-time = "2025-10-19T22:37:23.779Z" }, + { url = "https://files.pythonhosted.org/packages/16/6e/c0fb547eef61293153348f12e0f75a06abb322664b34a1573a7760501336/fastuuid-0.14.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:808527f2407f58a76c916d6aa15d58692a4a019fdf8d4c32ac7ff303b7d7af09", size = 278105, upload-time = "2025-10-19T22:26:56.821Z" }, + { url = "https://files.pythonhosted.org/packages/2d/b1/b9c75e03b768f61cf2e84ee193dc18601aeaf89a4684b20f2f0e9f52b62c/fastuuid-0.14.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2fb3c0d7fef6674bbeacdd6dbd386924a7b60b26de849266d1ff6602937675c8", size = 301564, upload-time = "2025-10-19T22:30:31.604Z" }, + { url = "https://files.pythonhosted.org/packages/fc/fa/f7395fdac07c7a54f18f801744573707321ca0cee082e638e36452355a9d/fastuuid-0.14.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:ab3f5d36e4393e628a4df337c2c039069344db5f4b9d2a3c9cea48284f1dd741", size = 459659, upload-time = "2025-10-19T22:31:32.341Z" }, + { url = "https://files.pythonhosted.org/packages/66/49/c9fd06a4a0b1f0f048aacb6599e7d96e5d6bc6fa680ed0d46bf111929d1b/fastuuid-0.14.0-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:b9a0ca4f03b7e0b01425281ffd44e99d360e15c895f1907ca105854ed85e2057", size = 478430, upload-time = "2025-10-19T22:26:22.962Z" }, + { url = "https://files.pythonhosted.org/packages/be/9c/909e8c95b494e8e140e8be6165d5fc3f61fdc46198c1554df7b3e1764471/fastuuid-0.14.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:3acdf655684cc09e60fb7e4cf524e8f42ea760031945aa8086c7eae2eeeabeb8", size = 450894, upload-time = "2025-10-19T22:27:01.647Z" }, + { url = "https://files.pythonhosted.org/packages/90/eb/d29d17521976e673c55ef7f210d4cdd72091a9ec6755d0fd4710d9b3c871/fastuuid-0.14.0-cp312-cp312-win32.whl", hash = "sha256:9579618be6280700ae36ac42c3efd157049fe4dd40ca49b021280481c78c3176", size = 154374, upload-time = "2025-10-19T22:29:19.879Z" }, + { url = "https://files.pythonhosted.org/packages/cc/fc/f5c799a6ea6d877faec0472d0b27c079b47c86b1cdc577720a5386483b36/fastuuid-0.14.0-cp312-cp312-win_amd64.whl", hash = "sha256:d9e4332dc4ba054434a9594cbfaf7823b57993d7d8e7267831c3e059857cf397", size = 156550, upload-time = "2025-10-19T22:27:49.658Z" }, ] [[package]] @@ -2059,11 +2086,11 @@ wheels = [ [[package]] name = "fsspec" -version = "2025.9.0" +version = "2025.10.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/de/e0/bab50af11c2d75c9c4a2a26a5254573c0bd97cea152254401510950486fa/fsspec-2025.9.0.tar.gz", hash = "sha256:19fd429483d25d28b65ec68f9f4adc16c17ea2c7c7bf54ec61360d478fb19c19", size = 304847, upload-time = "2025-09-02T19:10:49.215Z" } +sdist = { url = "https://files.pythonhosted.org/packages/24/7f/2747c0d332b9acfa75dc84447a066fdf812b5a6b8d30472b74d309bfe8cb/fsspec-2025.10.0.tar.gz", hash = "sha256:b6789427626f068f9a83ca4e8a3cc050850b6c0f71f99ddb4f542b8266a26a59", size = 309285, upload-time = "2025-10-30T14:58:44.036Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/47/71/70db47e4f6ce3e5c37a607355f80da8860a33226be640226ac52cb05ef2e/fsspec-2025.9.0-py3-none-any.whl", hash = "sha256:530dc2a2af60a414a832059574df4a6e10cce927f6f4a78209390fe38955cfb7", size = 199289, upload-time = "2025-09-02T19:10:47.708Z" }, + { url = "https://files.pythonhosted.org/packages/eb/02/a6b21098b1d5d6249b7c5ab69dde30108a71e4e819d4a9778f1de1d5b70d/fsspec-2025.10.0-py3-none-any.whl", hash = "sha256:7c7712353ae7d875407f97715f0e1ffcc21e33d5b24556cb1e090ae9409ec61d", size = 200966, upload-time = "2025-10-30T14:58:42.53Z" }, ] [[package]] @@ -2269,31 +2296,32 @@ wheels = [ [[package]] name = "google-cloud-core" -version = "2.4.3" +version = "2.5.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "google-api-core" }, { name = "google-auth" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/d6/b8/2b53838d2acd6ec6168fd284a990c76695e84c65deee79c9f3a4276f6b4f/google_cloud_core-2.4.3.tar.gz", hash = "sha256:1fab62d7102844b278fe6dead3af32408b1df3eb06f5c7e8634cbd40edc4da53", size = 35861, upload-time = "2025-03-10T21:05:38.948Z" } +sdist = { url = "https://files.pythonhosted.org/packages/a6/03/ef0bc99d0e0faf4fdbe67ac445e18cdaa74824fd93cd069e7bb6548cb52d/google_cloud_core-2.5.0.tar.gz", hash = "sha256:7c1b7ef5c92311717bd05301aa1a91ffbc565673d3b0b4163a52d8413a186963", size = 36027, upload-time = "2025-10-29T23:17:39.513Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/40/86/bda7241a8da2d28a754aad2ba0f6776e35b67e37c36ae0c45d49370f1014/google_cloud_core-2.4.3-py2.py3-none-any.whl", hash = "sha256:5130f9f4c14b4fafdff75c79448f9495cfade0d8775facf1b09c3bf67e027f6e", size = 29348, upload-time = "2025-03-10T21:05:37.785Z" }, + { url = "https://files.pythonhosted.org/packages/89/20/bfa472e327c8edee00f04beecc80baeddd2ab33ee0e86fd7654da49d45e9/google_cloud_core-2.5.0-py3-none-any.whl", hash = "sha256:67d977b41ae6c7211ee830c7912e41003ea8194bff15ae7d72fd6f51e57acabc", size = 29469, upload-time = "2025-10-29T23:17:38.548Z" }, ] [[package]] name = "google-cloud-resource-manager" -version = "1.14.2" +version = "1.15.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "google-api-core", extra = ["grpc"] }, { name = "google-auth" }, { name = "grpc-google-iam-v1" }, + { name = "grpcio" }, { name = "proto-plus" }, { name = "protobuf" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/6e/ca/a4648f5038cb94af4b3942815942a03aa9398f9fb0bef55b3f1585b9940d/google_cloud_resource_manager-1.14.2.tar.gz", hash = "sha256:962e2d904c550d7bac48372607904ff7bb3277e3bb4a36d80cc9a37e28e6eb74", size = 446370, upload-time = "2025-03-17T11:35:56.343Z" } +sdist = { url = "https://files.pythonhosted.org/packages/fc/19/b95d0e8814ce42522e434cdd85c0cb6236d874d9adf6685fc8e6d1fda9d1/google_cloud_resource_manager-1.15.0.tar.gz", hash = "sha256:3d0b78c3daa713f956d24e525b35e9e9a76d597c438837171304d431084cedaf", size = 449227, upload-time = "2025-10-20T14:57:01.108Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/b1/ea/a92631c358da377af34d3a9682c97af83185c2d66363d5939ab4a1169a7f/google_cloud_resource_manager-1.14.2-py3-none-any.whl", hash = "sha256:d0fa954dedd1d2b8e13feae9099c01b8aac515b648e612834f9942d2795a9900", size = 394344, upload-time = "2025-03-17T11:35:54.722Z" }, + { url = "https://files.pythonhosted.org/packages/8c/93/5aef41a5f146ad4559dd7040ae5fa8e7ddcab4dfadbef6cb4b66d775e690/google_cloud_resource_manager-1.15.0-py3-none-any.whl", hash = "sha256:0ccde5db644b269ddfdf7b407a2c7b60bdbf459f8e666344a5285601d00c7f6d", size = 397151, upload-time = "2025-10-20T14:53:45.409Z" }, ] [[package]] @@ -2388,11 +2416,11 @@ requests = [ [[package]] name = "graphql-core" -version = "3.2.6" +version = "3.2.7" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/c4/16/7574029da84834349b60ed71614d66ca3afe46e9bf9c7b9562102acb7d4f/graphql_core-3.2.6.tar.gz", hash = "sha256:c08eec22f9e40f0bd61d805907e3b3b1b9a320bc606e23dc145eebca07c8fbab", size = 505353, upload-time = "2025-01-26T16:36:27.374Z" } +sdist = { url = "https://files.pythonhosted.org/packages/ac/9b/037a640a2983b09aed4a823f9cf1729e6d780b0671f854efa4727a7affbe/graphql_core-3.2.7.tar.gz", hash = "sha256:27b6904bdd3b43f2a0556dad5d579bdfdeab1f38e8e8788e555bdcb586a6f62c", size = 513484, upload-time = "2025-11-01T22:30:40.436Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/ae/4f/7297663840621022bc73c22d7d9d80dbc78b4db6297f764b545cd5dd462d/graphql_core-3.2.6-py3-none-any.whl", hash = "sha256:78b016718c161a6fb20a7d97bbf107f331cd1afe53e45566c59f776ed7f0b45f", size = 203416, upload-time = "2025-01-26T16:36:24.868Z" }, + { url = "https://files.pythonhosted.org/packages/0a/14/933037032608787fb92e365883ad6a741c235e0ff992865ec5d904a38f1e/graphql_core-3.2.7-py3-none-any.whl", hash = "sha256:17fc8f3ca4a42913d8e24d9ac9f08deddf0a0b2483076575757f6c412ead2ec0", size = 207262, upload-time = "2025-11-01T22:30:38.912Z" }, ] [[package]] @@ -2418,6 +2446,8 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/1f/8e/abdd3f14d735b2929290a018ecf133c901be4874b858dd1c604b9319f064/greenlet-3.2.4-cp311-cp311-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2523e5246274f54fdadbce8494458a2ebdcdbc7b802318466ac5606d3cded1f8", size = 587684, upload-time = "2025-08-07T13:18:25.164Z" }, { url = "https://files.pythonhosted.org/packages/5d/65/deb2a69c3e5996439b0176f6651e0052542bb6c8f8ec2e3fba97c9768805/greenlet-3.2.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:1987de92fec508535687fb807a5cea1560f6196285a4cde35c100b8cd632cc52", size = 1116647, upload-time = "2025-08-07T13:42:38.655Z" }, { url = "https://files.pythonhosted.org/packages/3f/cc/b07000438a29ac5cfb2194bfc128151d52f333cee74dd7dfe3fb733fc16c/greenlet-3.2.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:55e9c5affaa6775e2c6b67659f3a71684de4c549b3dd9afca3bc773533d284fa", size = 1142073, upload-time = "2025-08-07T13:18:21.737Z" }, + { url = "https://files.pythonhosted.org/packages/67/24/28a5b2fa42d12b3d7e5614145f0bd89714c34c08be6aabe39c14dd52db34/greenlet-3.2.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:c9c6de1940a7d828635fbd254d69db79e54619f165ee7ce32fda763a9cb6a58c", size = 1548385, upload-time = "2025-11-04T12:42:11.067Z" }, + { url = "https://files.pythonhosted.org/packages/6a/05/03f2f0bdd0b0ff9a4f7b99333d57b53a7709c27723ec8123056b084e69cd/greenlet-3.2.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:03c5136e7be905045160b1b9fdca93dd6727b180feeafda6818e6496434ed8c5", size = 1613329, upload-time = "2025-11-04T12:42:12.928Z" }, { url = "https://files.pythonhosted.org/packages/d8/0f/30aef242fcab550b0b3520b8e3561156857c94288f0332a79928c31a52cf/greenlet-3.2.4-cp311-cp311-win_amd64.whl", hash = "sha256:9c40adce87eaa9ddb593ccb0fa6a07caf34015a29bf8d344811665b573138db9", size = 299100, upload-time = "2025-08-07T13:44:12.287Z" }, { url = "https://files.pythonhosted.org/packages/44/69/9b804adb5fd0671f367781560eb5eb586c4d495277c93bde4307b9e28068/greenlet-3.2.4-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:3b67ca49f54cede0186854a008109d6ee71f66bd57bb36abd6d0a0267b540cdd", size = 274079, upload-time = "2025-08-07T13:15:45.033Z" }, { url = "https://files.pythonhosted.org/packages/46/e9/d2a80c99f19a153eff70bc451ab78615583b8dac0754cfb942223d2c1a0d/greenlet-3.2.4-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:ddf9164e7a5b08e9d22511526865780a576f19ddd00d62f8a665949327fde8bb", size = 640997, upload-time = "2025-08-07T13:42:56.234Z" }, @@ -2427,101 +2457,103 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/19/0d/6660d55f7373b2ff8152401a83e02084956da23ae58cddbfb0b330978fe9/greenlet-3.2.4-cp312-cp312-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:3b3812d8d0c9579967815af437d96623f45c0f2ae5f04e366de62a12d83a8fb0", size = 607586, upload-time = "2025-08-07T13:18:28.544Z" }, { url = "https://files.pythonhosted.org/packages/8e/1a/c953fdedd22d81ee4629afbb38d2f9d71e37d23caace44775a3a969147d4/greenlet-3.2.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:abbf57b5a870d30c4675928c37278493044d7c14378350b3aa5d484fa65575f0", size = 1123281, upload-time = "2025-08-07T13:42:39.858Z" }, { url = "https://files.pythonhosted.org/packages/3f/c7/12381b18e21aef2c6bd3a636da1088b888b97b7a0362fac2e4de92405f97/greenlet-3.2.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:20fb936b4652b6e307b8f347665e2c615540d4b42b3b4c8a321d8286da7e520f", size = 1151142, upload-time = "2025-08-07T13:18:22.981Z" }, + { url = "https://files.pythonhosted.org/packages/27/45/80935968b53cfd3f33cf99ea5f08227f2646e044568c9b1555b58ffd61c2/greenlet-3.2.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:ee7a6ec486883397d70eec05059353b8e83eca9168b9f3f9a361971e77e0bcd0", size = 1564846, upload-time = "2025-11-04T12:42:15.191Z" }, + { url = "https://files.pythonhosted.org/packages/69/02/b7c30e5e04752cb4db6202a3858b149c0710e5453b71a3b2aec5d78a1aab/greenlet-3.2.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:326d234cbf337c9c3def0676412eb7040a35a768efc92504b947b3e9cfc7543d", size = 1633814, upload-time = "2025-11-04T12:42:17.175Z" }, { url = "https://files.pythonhosted.org/packages/e9/08/b0814846b79399e585f974bbeebf5580fbe59e258ea7be64d9dfb253c84f/greenlet-3.2.4-cp312-cp312-win_amd64.whl", hash = "sha256:a7d4e128405eea3814a12cc2605e0e6aedb4035bf32697f72deca74de4105e02", size = 299899, upload-time = "2025-08-07T13:38:53.448Z" }, ] [[package]] name = "grimp" -version = "3.12" +version = "3.13" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/1b/a4/463903a1cfbc19d3e7125d6614bb900df2b34dd675c7d93544d154819d2b/grimp-3.12.tar.gz", hash = "sha256:1a733b1d719c42bd2fada58240975fa7d09936b57120c34b64cfb31e42701010", size = 845594, upload-time = "2025-10-09T09:51:02.064Z" } +sdist = { url = "https://files.pythonhosted.org/packages/80/b3/ff0d704cdc5cf399d74aabd2bf1694d4c4c3231d4d74b011b8f39f686a86/grimp-3.13.tar.gz", hash = "sha256:759bf6e05186e6473ee71af4119ec181855b2b324f4fcdd78dee9e5b59d87874", size = 847508, upload-time = "2025-10-29T13:04:57.704Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/0f/b5/1c89600bf181d41502aed51b73b3a5889158dee35c534f51df3666779587/grimp-3.12-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:e6c02e51eebfcf71146d42f47c9ce353ac1902ae446e18d0e663ab9fdaa0496c", size = 2062043, upload-time = "2025-10-09T09:49:57.035Z" }, - { url = "https://files.pythonhosted.org/packages/1f/86/bab32c5e26949a82299853ccb28ee30a7899d0355b0d209b535eb03bc04e/grimp-3.12-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:79bc2b0ff6072c43c0ddc4479b25b7a8198795486478cfe3be0503b2c7d32c7f", size = 1981378, upload-time = "2025-10-09T09:49:49.237Z" }, - { url = "https://files.pythonhosted.org/packages/b5/03/b9f7e465488e8593de9a1e88355c3cfba04c02c3a34a6b02cbe946e0d587/grimp-3.12-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3986f11a9dd4167a2943cf6e80b458c0a825b48609713736cc8f2de135000810", size = 2130579, upload-time = "2025-10-09T09:48:36.035Z" }, - { url = "https://files.pythonhosted.org/packages/1b/d0/81c776327354f32f86f321dd8468b32ba6b52dc3511d912d24c4fac96da4/grimp-3.12-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:7a2abe55844f9dad25499ff9456d680496f390d160b6b3a4e5aeabc0183813b4", size = 2091201, upload-time = "2025-10-09T09:48:52.57Z" }, - { url = "https://files.pythonhosted.org/packages/9d/7e/116ac4c1e4407a123fba4bb076b2e880643d70b3f4f1621c3323b5d66e12/grimp-3.12-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e59112d0f557335b619bcf10263d11873579230bd3df4a4b19224ec18e7212d6", size = 2240782, upload-time = "2025-10-09T09:49:30.915Z" }, - { url = "https://files.pythonhosted.org/packages/06/7f/89bbec1241a8504499975f0f08befea0cf3d27c52f9808602fff8075c639/grimp-3.12-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b858e2e5a489c36710322970aa82bfbd3f1c4107c8564960629a59d2f17a53d0", size = 2423143, upload-time = "2025-10-09T09:49:05.18Z" }, - { url = "https://files.pythonhosted.org/packages/86/d7/2f416439b624b2a91bf2e0e456f58d74d51aa7ad239099cf4a8911d952c0/grimp-3.12-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d46cc1222dd301e0be371b97f0cdecae178089704e8a285e3edd4750ec46270a", size = 2303850, upload-time = "2025-10-09T09:49:19.073Z" }, - { url = "https://files.pythonhosted.org/packages/60/bd/8c2f48c26151eb9a65bc41f01004b43cb1b31791ffb61758d40d2f6b485a/grimp-3.12-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6ef06822f75856af28e7fcc580034043c543b1c99b07d2bd467bd173a7f10691", size = 2168571, upload-time = "2025-10-09T09:49:39.844Z" }, - { url = "https://files.pythonhosted.org/packages/5a/45/01a839434ff88be24317aa52cc1ba158833bd1d071efe0da1b14838af024/grimp-3.12-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:4c19f1cba8a95c898473dd18f9c81358019d67f87f140b0b8401550e6d21c5a3", size = 2310869, upload-time = "2025-10-09T09:50:05.153Z" }, - { url = "https://files.pythonhosted.org/packages/ba/7b/0dc45fdc15562c2faf8a95a8685d3805d27decdef6fcfb66d9b577ed2f12/grimp-3.12-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:600e8dbc1cd9c6decbc22089730221c65591b7ba5f89751d07fc7ad014d99aa1", size = 2353397, upload-time = "2025-10-09T09:50:17.755Z" }, - { url = "https://files.pythonhosted.org/packages/a8/ec/07734ecc4f1489ffc071417f7bc881c939bcfdfba10eb585bce510ede1b2/grimp-3.12-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:259ba53b82cfb9c2c2d097b2237970c4e9903fa2d0b664b7e12329d9a64924f9", size = 2350166, upload-time = "2025-10-09T09:50:32.237Z" }, - { url = "https://files.pythonhosted.org/packages/a4/f5/45d80e2fa205066a484f0c1a667a249408a49bb3b665d62677f879920aa0/grimp-3.12-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:a593549b1f66b1c12574e71f9e8c0073b372888c6b6706e2617bba2713ae28c2", size = 2360590, upload-time = "2025-10-09T09:50:49.961Z" }, - { url = "https://files.pythonhosted.org/packages/e6/f2/7ab1bc4d613189183c17741ff0d03490d9749eb5130b8b56e82ed77098b0/grimp-3.12-cp311-cp311-win32.whl", hash = "sha256:356ee969443f06c6c3a270f5a7221f946f0cb135a8b8ece2009990b293504bb3", size = 1748183, upload-time = "2025-10-09T09:51:13.503Z" }, - { url = "https://files.pythonhosted.org/packages/91/62/195f37a68d07fab40c8934ae8e39f9ff1f9a5bf3e375059b9cf14ccba302/grimp-3.12-cp311-cp311-win_amd64.whl", hash = "sha256:75e1f0d74f3a242a1c34e464d775c36b1c8b9d8c92b35f46f221e73e9b2f0065", size = 1851099, upload-time = "2025-10-09T09:51:04.747Z" }, - { url = "https://files.pythonhosted.org/packages/12/ac/0f55980a59c07439a965d3975f1cf3a6574f7d773910b9d6924790e0dddf/grimp-3.12-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:af399fc0ffddfbd7ea6c2e8546be1ab5284ee800f15a445705bdda5d63501b34", size = 2058862, upload-time = "2025-10-09T09:49:58.478Z" }, - { url = "https://files.pythonhosted.org/packages/cc/b1/5fdcb1db7cb3253c78d87a0b8c3f7f9c5214b273861300b51c897c55e6b8/grimp-3.12-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2f08358acbaf9a4b324537bf344fd2d76b5f9b6f1bfaf9a431e9453fc0eaee5f", size = 1977586, upload-time = "2025-10-09T09:49:50.49Z" }, - { url = "https://files.pythonhosted.org/packages/c9/b9/e5f6d265b71430f9641daa9476cde8c23549e396c558b39a0bdc7fee824f/grimp-3.12-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6eeb1616cafe9074fcb390fcfc01e6e5a0e0ddd5acb9dd37579985b2879c239a", size = 2130610, upload-time = "2025-10-09T09:48:38.472Z" }, - { url = "https://files.pythonhosted.org/packages/da/e1/2d0601c9aac2ab7340504e85ca4cd55f2991501a03e421bec78f53a07478/grimp-3.12-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:99e648e299f7cd3daaee2cb745192e7ea159c7d38df76b4dcca12a2ef68a3ede", size = 2092775, upload-time = "2025-10-09T09:48:53.841Z" }, - { url = "https://files.pythonhosted.org/packages/db/a1/e63315477127ed8f31a1a93911d084bf704d6e126ca27650e3c3389701a6/grimp-3.12-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5b24c5ce351030d1f83e69acd76a06863dd87041ceb25572339f7334e210cbc4", size = 2239336, upload-time = "2025-10-09T09:49:32.185Z" }, - { url = "https://files.pythonhosted.org/packages/f2/09/cd76d35121f053a95a58fc5830756c62e5c9de74aa4e16b4dc27ce6ada2c/grimp-3.12-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fd40a5ec09d1dfafaae88b53231ab79378183e2e9a03e7b26b7a30133d027d8a", size = 2421851, upload-time = "2025-10-09T09:49:06.893Z" }, - { url = "https://files.pythonhosted.org/packages/40/46/e8390a7c5ed85b4dbeff4e873f1ece8d9acf72d72f084b397ccc2facfa3b/grimp-3.12-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0aebdfad66d6f4e8b0f7364ce0429d208be3510918097f969428165074d3103e", size = 2304849, upload-time = "2025-10-09T09:49:20.695Z" }, - { url = "https://files.pythonhosted.org/packages/bd/81/f73edbc48a283f634233b6153ac43e4e7b9f58108ffc19da803b0015cb60/grimp-3.12-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:76fd06be98d6bea9ea8a804da22c80accf1d277fe04abd5f3dff05d087f056f7", size = 2168655, upload-time = "2025-10-09T09:49:41.118Z" }, - { url = "https://files.pythonhosted.org/packages/84/1a/8fa5752f725b8872010627bd10e1aedccdb406c3b4118ec3fe127155284e/grimp-3.12-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:a73a42a43e268ac5b196386beae1ec646f4572409e731bccf2a99ab4ed5c46bf", size = 2311124, upload-time = "2025-10-09T09:50:06.477Z" }, - { url = "https://files.pythonhosted.org/packages/83/a0/02d6b2a86289a4ac73f44f59aaee43c1dc936c984204c73d2affe4570eb6/grimp-3.12-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:af990af7d5e64f484d12cdefacfaaed4ea9418ac4d0a5a928953fd91aaf8df80", size = 2354216, upload-time = "2025-10-09T09:50:19.114Z" }, - { url = "https://files.pythonhosted.org/packages/7b/48/0368289f5bbdf943a48305824b30411b35ef2c7cd8edf2bad48d67b3897e/grimp-3.12-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:82ee28c1e9835572af2c733f7e5913a44193c53ae8ca488039164593b4a750fa", size = 2348372, upload-time = "2025-10-09T09:50:37.479Z" }, - { url = "https://files.pythonhosted.org/packages/26/73/b4f90b4926791d720f6069fc8c8b3e204721d1db839a1c00fbcee1e2a36d/grimp-3.12-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:afdceaea00e305909cb30d68e91b94fcf71d1a7234052549ea31148785a03a52", size = 2361167, upload-time = "2025-10-09T09:50:51.733Z" }, - { url = "https://files.pythonhosted.org/packages/0b/ae/94d34c732d531c7165c8942d7995495aac64e9bb5c28cc6751349eacdcde/grimp-3.12-cp312-cp312-win32.whl", hash = "sha256:40f8e048254d2437dffcd383d2301a82c35d9a3082e878b707d87a6e8c539614", size = 1747179, upload-time = "2025-10-09T09:51:15.224Z" }, - { url = "https://files.pythonhosted.org/packages/5b/cd/48bc396ee2f36e72d5c50ba8b4d7f817fc2cdac7b9ab77d2b097f50a4447/grimp-3.12-cp312-cp312-win_amd64.whl", hash = "sha256:199172d17f22199bf400a0bd5c4985784622201e887a023fe799ca3f3437dedf", size = 1850691, upload-time = "2025-10-09T09:51:05.984Z" }, - { url = "https://files.pythonhosted.org/packages/d9/31/c72e53a46692dc8358cff1af1a9494430a0fecd4c3f2d0d8e9c2eb5e828d/grimp-3.12-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:567d037a3db083e54bee621daba59a2e01fd1391364ae0a0c737995f6eed910b", size = 2131392, upload-time = "2025-10-09T09:48:46.857Z" }, - { url = "https://files.pythonhosted.org/packages/39/10/15e43be32734baaebeee090dca16f06ea5ba933b209b8e1c0d5986dabb32/grimp-3.12-pp311-pypy311_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:9b4cc756c91c3d8582ee70b5e013c0e34fdb31c7f808cefe9d15509c45fec31e", size = 2092481, upload-time = "2025-10-09T09:49:00.754Z" }, - { url = "https://files.pythonhosted.org/packages/a1/4a/c9349dee284c2d9384714741896f0f84a1d66011a69cdc364e4d94e188b1/grimp-3.12-pp311-pypy311_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:84bd47f9a8619cb8966f18cb6faf5f6cb8d35ade99312477dd8e9de3a9ae4cb7", size = 2242260, upload-time = "2025-10-09T09:49:37.183Z" }, - { url = "https://files.pythonhosted.org/packages/d8/63/3935823f89c12320840bbf018858eeaca7d5285f9769a48921587a88adeb/grimp-3.12-pp311-pypy311_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7f30e01855c67a39857c87e6c0eafe5e8891010a35e06cf2145f2cfce8ea9780", size = 2422371, upload-time = "2025-10-09T09:49:14.616Z" }, - { url = "https://files.pythonhosted.org/packages/71/8e/5a75c2335a2dc61738b19318dcdd16392015a984211e3d0b9f6679dc6c89/grimp-3.12-pp311-pypy311_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d07e825f6b052186dabd8dbbcc7e008a3b56e551725e2ba47169fe1e4bde76ac", size = 2304257, upload-time = "2025-10-09T09:49:26.908Z" }, - { url = "https://files.pythonhosted.org/packages/40/99/462d86bc9401a39859f272b867331a678f4b5324a539dc771bdae6d36309/grimp-3.12-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5f1a1289d4282be2891ada75ec5d3099e856518c4236b1196e367b630485f8ce", size = 2169360, upload-time = "2025-10-09T09:49:46.575Z" }, - { url = "https://files.pythonhosted.org/packages/d0/07/6d2929f05dae189265633588819d990df35644ad74b6ec74207091dff18d/grimp-3.12-pp311-pypy311_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:85136b555aeb7d3965fdb40af4e4af2011f911b0fde8c20979bf4db7b06455f5", size = 2312280, upload-time = "2025-10-09T09:50:13.491Z" }, - { url = "https://files.pythonhosted.org/packages/5c/47/7e49417e2c496da0b6141e711dca40726d2b30a0adc6db9d04b74c7bafa7/grimp-3.12-pp311-pypy311_pp73-musllinux_1_2_armv7l.whl", hash = "sha256:963efd6ec86e7b47fde835b2526b6be7a3f489857a1cd47a747c94b3e670550a", size = 2354449, upload-time = "2025-10-09T09:50:27.596Z" }, - { url = "https://files.pythonhosted.org/packages/2c/08/2e1db56797e4e26334b3ee4ef1a5fbf56155d74a0318215ed4dcad02ef43/grimp-3.12-pp311-pypy311_pp73-musllinux_1_2_i686.whl", hash = "sha256:c9e2ee478b66f0e20c92af6123142ffd6b604c36e9b3a8d391ea9172cc18b6b3", size = 2350545, upload-time = "2025-10-09T09:50:45.623Z" }, - { url = "https://files.pythonhosted.org/packages/37/78/53594064f11b0ae9e72b3e9df5c055f00c5bff44962f7b777846504fc50d/grimp-3.12-pp311-pypy311_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:e8826362d4e403aa2e03d480e3e4d64284a6b6ccafc2c5777bb2bed2535bdc4e", size = 2361926, upload-time = "2025-10-09T09:50:58.605Z" }, + { url = "https://files.pythonhosted.org/packages/45/cc/d272cf87728a7e6ddb44d3c57c1d3cbe7daf2ffe4dc76e3dc9b953b69ab1/grimp-3.13-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:57745996698932768274a2ed9ba3e5c424f60996c53ecaf1c82b75be9e819ee9", size = 2074518, upload-time = "2025-10-29T13:03:58.51Z" }, + { url = "https://files.pythonhosted.org/packages/06/11/31dc622c5a0d1615b20532af2083f4bba2573aebbba5b9d6911dfd60a37d/grimp-3.13-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ca29f09710342b94fa6441f4d1102a0e49f0b463b1d91e43223baa949c5e9337", size = 1988182, upload-time = "2025-10-29T13:03:50.129Z" }, + { url = "https://files.pythonhosted.org/packages/aa/83/a0e19beb5c42df09e9a60711b227b4f910ba57f46bea258a9e1df883976c/grimp-3.13-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:adda25aa158e11d96dd27166300b955c8ec0c76ce2fd1a13597e9af012aada06", size = 2145832, upload-time = "2025-10-29T13:02:35.218Z" }, + { url = "https://files.pythonhosted.org/packages/bc/f5/13752205e290588e970fdc019b4ab2c063ca8da352295c332e34df5d5842/grimp-3.13-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:03e17029d75500a5282b40cb15cdae030bf14df9dfaa6a2b983f08898dfe74b6", size = 2106762, upload-time = "2025-10-29T13:02:51.681Z" }, + { url = "https://files.pythonhosted.org/packages/ff/30/c4d62543beda4b9a483a6cd5b7dd5e4794aafb511f144d21a452467989a1/grimp-3.13-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6cbfc9d2d0ebc0631fb4012a002f3d8f4e3acb8325be34db525c0392674433b8", size = 2256674, upload-time = "2025-10-29T13:03:27.923Z" }, + { url = "https://files.pythonhosted.org/packages/9b/ea/d07ed41b7121719c3f7bf30c9881dbde69efeacfc2daf4e4a628efe5f123/grimp-3.13-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:161449751a085484608c5b9f863e41e8fb2a98e93f7312ead5d831e487a94518", size = 2442699, upload-time = "2025-10-29T13:03:04.451Z" }, + { url = "https://files.pythonhosted.org/packages/fe/a0/1923f0480756effb53c7e6cef02a3918bb519a86715992720838d44f0329/grimp-3.13-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:119628fbe7f941d1e784edac98e8ced7e78a0b966a4ff2c449e436ee860bd507", size = 2317145, upload-time = "2025-10-29T13:03:15.941Z" }, + { url = "https://files.pythonhosted.org/packages/0d/d9/aef4c8350090653e34bc755a5d9e39cc300f5c46c651c1d50195f69bf9ab/grimp-3.13-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1ca1ac776baf1fa105342b23c72f2e7fdd6771d4cce8d2903d28f92fd34a9e8f", size = 2180288, upload-time = "2025-10-29T13:03:41.023Z" }, + { url = "https://files.pythonhosted.org/packages/9f/2e/a206f76eccffa56310a1c5d5950ed34923a34ae360cb38e297604a288837/grimp-3.13-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:941ff414cc66458f56e6af93c618266091ea70bfdabe7a84039be31d937051ee", size = 2328696, upload-time = "2025-10-29T13:04:06.888Z" }, + { url = "https://files.pythonhosted.org/packages/40/3b/88ff1554409b58faf2673854770e6fc6e90167a182f5166147b7618767d7/grimp-3.13-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:87ad9bcd1caaa2f77c369d61a04b9f2f1b87f4c3b23ae6891b2c943193c4ec62", size = 2367574, upload-time = "2025-10-29T13:04:21.404Z" }, + { url = "https://files.pythonhosted.org/packages/b6/b3/e9c99ecd94567465a0926ae7136e589aed336f6979a4cddcb8dfba16d27c/grimp-3.13-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:751fe37104a4f023d5c6556558b723d843d44361245c20f51a5d196de00e4774", size = 2358842, upload-time = "2025-10-29T13:04:34.26Z" }, + { url = "https://files.pythonhosted.org/packages/74/65/a5fffeeb9273e06dfbe962c8096331ba181ca8415c5f9d110b347f2c0c34/grimp-3.13-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:9b561f79ec0b3a4156937709737191ad57520f2d58fa1fc43cd79f67839a3cd7", size = 2382268, upload-time = "2025-10-29T13:04:46.864Z" }, + { url = "https://files.pythonhosted.org/packages/d9/79/2f3b4323184329b26b46de2b6d1bd64ba1c26e0a9c3cfa0aaecec237b75e/grimp-3.13-cp311-cp311-win32.whl", hash = "sha256:52405ea8c8f20cf5d2d1866c80ee3f0243a38af82bd49d1464c5e254bf2e1f8f", size = 1759345, upload-time = "2025-10-29T13:05:10.435Z" }, + { url = "https://files.pythonhosted.org/packages/b6/ce/e86cf73e412a6bf531cbfa5c733f8ca48b28ebea23a037338be763f24849/grimp-3.13-cp311-cp311-win_amd64.whl", hash = "sha256:6a45d1d3beeefad69717b3718e53680fb3579fe67696b86349d6f39b75e850bf", size = 1859382, upload-time = "2025-10-29T13:05:01.071Z" }, + { url = "https://files.pythonhosted.org/packages/1d/06/ff7e3d72839f46f0fccdc79e1afe332318986751e20f65d7211a5e51366c/grimp-3.13-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:3e715c56ffdd055e5c84d27b4c02d83369b733e6a24579d42bbbc284bd0664a9", size = 2070161, upload-time = "2025-10-29T13:03:59.755Z" }, + { url = "https://files.pythonhosted.org/packages/58/2f/a95bdf8996db9400fd7e288f32628b2177b8840fe5f6b7cd96247b5fa173/grimp-3.13-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f794dea35a4728b948ab8fec970ffbdf2589b34209f3ab902cf8a9148cf1eaad", size = 1984365, upload-time = "2025-10-29T13:03:51.805Z" }, + { url = "https://files.pythonhosted.org/packages/1f/45/cc3d7f3b7b4d93e0b9d747dc45ed73a96203ba083dc857f24159eb6966b4/grimp-3.13-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:69571270f2c27e8a64b968195aa7ecc126797112a9bf1e804ff39ba9f42d6f6d", size = 2145486, upload-time = "2025-10-29T13:02:36.591Z" }, + { url = "https://files.pythonhosted.org/packages/16/92/a6e493b71cb5a9145ad414cc4790c3779853372b840a320f052b22879606/grimp-3.13-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:8f7b226398ae476762ef0afb5ef8f838d39c8e0e2f6d1a4378ce47059b221a4a", size = 2106747, upload-time = "2025-10-29T13:02:53.084Z" }, + { url = "https://files.pythonhosted.org/packages/db/8d/36a09f39fe14ad8843ef3ff81090ef23abbd02984c1fcc1cef30e5713d82/grimp-3.13-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5498aeac4df0131a1787fcbe9bb460b52fc9b781ec6bba607fd6a7d6d3ea6fce", size = 2257027, upload-time = "2025-10-29T13:03:29.44Z" }, + { url = "https://files.pythonhosted.org/packages/a1/7a/90f78787f80504caeef501f1bff47e8b9f6058d45995f1d4c921df17bfef/grimp-3.13-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4be702bb2b5c001a6baf709c452358470881e15e3e074cfc5308903603485dcb", size = 2441208, upload-time = "2025-10-29T13:03:05.733Z" }, + { url = "https://files.pythonhosted.org/packages/61/71/0fbd3a3e914512b9602fa24c8ebc85a8925b101f04f8a8c1d1e220e0a717/grimp-3.13-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9fcf988f3e3d272a88f7be68f0c1d3719fee8624d902e9c0346b9015a0ea6a65", size = 2318758, upload-time = "2025-10-29T13:03:17.454Z" }, + { url = "https://files.pythonhosted.org/packages/34/e9/29c685e88b3b0688f0a2e30c0825e02076ecdf22bc0e37b1468562eaa09a/grimp-3.13-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0ede36d104ff88c208140f978de3345f439345f35b8ef2b4390c59ef6984deba", size = 2180523, upload-time = "2025-10-29T13:03:42.3Z" }, + { url = "https://files.pythonhosted.org/packages/86/bc/7cc09574b287b8850a45051e73272f365259d9b6ca58d7b8773265c6fe35/grimp-3.13-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:b35e44bb8dc80e0bd909a64387f722395453593a1884caca9dc0748efea33764", size = 2328855, upload-time = "2025-10-29T13:04:08.111Z" }, + { url = "https://files.pythonhosted.org/packages/34/86/3b0845900c8f984a57c6afe3409b20638065462d48b6afec0fd409fd6118/grimp-3.13-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:becb88e9405fc40896acd6e2b9bbf6f242a5ae2fd43a1ec0a32319ab6c10a227", size = 2367756, upload-time = "2025-10-29T13:04:22.736Z" }, + { url = "https://files.pythonhosted.org/packages/06/2d/4e70e8c06542db92c3fffaecb43ebfc4114a411505bff574d4da7d82c7db/grimp-3.13-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:a66585b4af46c3fbadbef495483514bee037e8c3075ed179ba4f13e494eb7899", size = 2358595, upload-time = "2025-10-29T13:04:35.595Z" }, + { url = "https://files.pythonhosted.org/packages/dd/06/c511d39eb6c73069af277f4e74991f1f29a05d90cab61f5416b9fc43932f/grimp-3.13-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:29f68c6e2ff70d782ca0e989ec4ec44df73ba847937bcbb6191499224a2f84e2", size = 2381464, upload-time = "2025-10-29T13:04:48.265Z" }, + { url = "https://files.pythonhosted.org/packages/86/f5/42197d69e4c9e2e7eed091d06493da3824e07c37324155569aa895c3b5f7/grimp-3.13-cp312-cp312-win32.whl", hash = "sha256:cc996dcd1a44ae52d257b9a3e98838f8ecfdc42f7c62c8c82c2fcd3828155c98", size = 1758510, upload-time = "2025-10-29T13:05:11.74Z" }, + { url = "https://files.pythonhosted.org/packages/30/dd/59c5f19f51e25f3dbf1c9e88067a88165f649ba1b8e4174dbaf1c950f78b/grimp-3.13-cp312-cp312-win_amd64.whl", hash = "sha256:e2966435947e45b11568f04a65863dcf836343c11ae44aeefdaa7f07eb1a0576", size = 1859530, upload-time = "2025-10-29T13:05:02.638Z" }, + { url = "https://files.pythonhosted.org/packages/e5/81/82de1b5d82701214b1f8e32b2e71fde8e1edbb4f2cdca9beb22ee6c8796d/grimp-3.13-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5a6a3c76525b018c85c0e3a632d94d72be02225f8ada56670f3f213cf0762be4", size = 2145955, upload-time = "2025-10-29T13:02:47.559Z" }, + { url = "https://files.pythonhosted.org/packages/8c/ae/ada18cb73bdf97094af1c60070a5b85549482a57c509ee9a23fdceed4fc3/grimp-3.13-pp311-pypy311_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:239e9b347af4da4cf69465bfa7b2901127f6057bc73416ba8187fb1eabafc6ea", size = 2107150, upload-time = "2025-10-29T13:02:59.891Z" }, + { url = "https://files.pythonhosted.org/packages/10/5e/6d8c65643ad5a1b6e00cc2cd8f56fc063923485f07c59a756fa61eefe7f2/grimp-3.13-pp311-pypy311_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d6db85ce2dc2f804a2edd1c1e9eaa46d282e1f0051752a83ca08ca8b87f87376", size = 2257515, upload-time = "2025-10-29T13:03:36.705Z" }, + { url = "https://files.pythonhosted.org/packages/b2/62/72cbfd7d0f2b95a53edd01d5f6b0d02bde38db739a727e35b76c13e0d0a8/grimp-3.13-pp311-pypy311_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e000f3590bcc6ff7c781ebbc1ac4eb919f97180f13cc4002c868822167bd9aed", size = 2441262, upload-time = "2025-10-29T13:03:12.158Z" }, + { url = "https://files.pythonhosted.org/packages/18/00/b9209ab385567c3bddffb5d9eeecf9cb432b05c30ca8f35904b06e206a89/grimp-3.13-pp311-pypy311_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e2374c217c862c1af933a430192d6a7c6723ed1d90303f1abbc26f709bbb9263", size = 2318557, upload-time = "2025-10-29T13:03:23.925Z" }, + { url = "https://files.pythonhosted.org/packages/11/4d/a3d73c11d09da00a53ceafe2884a71c78f5a76186af6d633cadd6c85d850/grimp-3.13-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3ed0ff17d559ff2e7fa1be8ae086bc4fedcace5d7b12017f60164db8d9a8d806", size = 2180811, upload-time = "2025-10-29T13:03:47.461Z" }, + { url = "https://files.pythonhosted.org/packages/c1/9a/1cdfaa7d7beefd8859b190dfeba11d5ec074e8702b2903e9f182d662ed63/grimp-3.13-pp311-pypy311_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:43960234aabce018c8d796ec8b77c484a1c9cbb6a3bc036a0d307c8dade9874c", size = 2329205, upload-time = "2025-10-29T13:04:15.845Z" }, + { url = "https://files.pythonhosted.org/packages/86/73/b36f86ef98df96e7e8a6166dfa60c8db5d597f051e613a3112f39a870b4c/grimp-3.13-pp311-pypy311_pp73-musllinux_1_2_armv7l.whl", hash = "sha256:44420b638b3e303f32314bd4d309f15de1666629035acd1cdd3720c15917ac85", size = 2368745, upload-time = "2025-10-29T13:04:29.706Z" }, + { url = "https://files.pythonhosted.org/packages/02/2f/0ce37872fad5c4b82d727f6e435fd5bc76f701279bddc9666710318940cf/grimp-3.13-pp311-pypy311_pp73-musllinux_1_2_i686.whl", hash = "sha256:f6127fdb982cf135612504d34aa16b841f421e54751fcd54f80b9531decb2b3f", size = 2358753, upload-time = "2025-10-29T13:04:42.632Z" }, + { url = "https://files.pythonhosted.org/packages/bb/23/935c888ac9ee71184fe5adf5ea86648746739be23c85932857ac19fc1d17/grimp-3.13-pp311-pypy311_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:69893a9ef1edea25226ed17e8e8981e32900c59703972e0780c0e927ce624f75", size = 2383066, upload-time = "2025-10-29T13:04:55.073Z" }, ] [[package]] name = "grpc-google-iam-v1" -version = "0.14.2" +version = "0.14.3" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "googleapis-common-protos", extra = ["grpc"] }, { name = "grpcio" }, { name = "protobuf" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/b9/4e/8d0ca3b035e41fe0b3f31ebbb638356af720335e5a11154c330169b40777/grpc_google_iam_v1-0.14.2.tar.gz", hash = "sha256:b3e1fc387a1a329e41672197d0ace9de22c78dd7d215048c4c78712073f7bd20", size = 16259, upload-time = "2025-03-17T11:40:23.586Z" } +sdist = { url = "https://files.pythonhosted.org/packages/76/1e/1011451679a983f2f5c6771a1682542ecb027776762ad031fd0d7129164b/grpc_google_iam_v1-0.14.3.tar.gz", hash = "sha256:879ac4ef33136c5491a6300e27575a9ec760f6cdf9a2518798c1b8977a5dc389", size = 23745, upload-time = "2025-10-15T21:14:53.318Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/66/6f/dd9b178aee7835b96c2e63715aba6516a9d50f6bebbd1cc1d32c82a2a6c3/grpc_google_iam_v1-0.14.2-py3-none-any.whl", hash = "sha256:a3171468459770907926d56a440b2bb643eec1d7ba215f48f3ecece42b4d8351", size = 19242, upload-time = "2025-03-17T11:40:22.648Z" }, + { url = "https://files.pythonhosted.org/packages/4a/bd/330a1bbdb1afe0b96311249e699b6dc9cfc17916394fd4503ac5aca2514b/grpc_google_iam_v1-0.14.3-py3-none-any.whl", hash = "sha256:7a7f697e017a067206a3dfef44e4c634a34d3dee135fe7d7a4613fe3e59217e6", size = 32690, upload-time = "2025-10-15T21:14:51.72Z" }, ] [[package]] name = "grpcio" -version = "1.75.1" +version = "1.76.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/9d/f7/8963848164c7604efb3a3e6ee457fdb3a469653e19002bd24742473254f8/grpcio-1.75.1.tar.gz", hash = "sha256:3e81d89ece99b9ace23a6916880baca613c03a799925afb2857887efa8b1b3d2", size = 12731327, upload-time = "2025-09-26T09:03:36.887Z" } +sdist = { url = "https://files.pythonhosted.org/packages/b6/e0/318c1ce3ae5a17894d5791e87aea147587c9e702f24122cc7a5c8bbaeeb1/grpcio-1.76.0.tar.gz", hash = "sha256:7be78388d6da1a25c0d5ec506523db58b18be22d9c37d8d3a32c08be4987bd73", size = 12785182, upload-time = "2025-10-21T16:23:12.106Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/0c/3c/35ca9747473a306bfad0cee04504953f7098527cd112a4ab55c55af9e7bd/grpcio-1.75.1-cp311-cp311-linux_armv7l.whl", hash = "sha256:573855ca2e58e35032aff30bfbd1ee103fbcf4472e4b28d4010757700918e326", size = 5709761, upload-time = "2025-09-26T09:01:28.528Z" }, - { url = "https://files.pythonhosted.org/packages/c9/2c/ecbcb4241e4edbe85ac2663f885726fea0e947767401288b50d8fdcb9200/grpcio-1.75.1-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:6a4996a2c8accc37976dc142d5991adf60733e223e5c9a2219e157dc6a8fd3a2", size = 11496691, upload-time = "2025-09-26T09:01:31.214Z" }, - { url = "https://files.pythonhosted.org/packages/81/40/bc07aee2911f0d426fa53fe636216100c31a8ea65a400894f280274cb023/grpcio-1.75.1-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:b1ea1bbe77ecbc1be00af2769f4ae4a88ce93be57a4f3eebd91087898ed749f9", size = 6296084, upload-time = "2025-09-26T09:01:34.596Z" }, - { url = "https://files.pythonhosted.org/packages/b8/d1/10c067f6c67396cbf46448b80f27583b5e8c4b46cdfbe18a2a02c2c2f290/grpcio-1.75.1-cp311-cp311-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:e5b425aee54cc5e3e3c58f00731e8a33f5567965d478d516d35ef99fd648ab68", size = 6950403, upload-time = "2025-09-26T09:01:36.736Z" }, - { url = "https://files.pythonhosted.org/packages/3f/42/5f628abe360b84dfe8dd8f32be6b0606dc31dc04d3358eef27db791ea4d5/grpcio-1.75.1-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:0049a7bf547dafaeeb1db17079ce79596c298bfe308fc084d023c8907a845b9a", size = 6470166, upload-time = "2025-09-26T09:01:39.474Z" }, - { url = "https://files.pythonhosted.org/packages/c3/93/a24035080251324019882ee2265cfde642d6476c0cf8eb207fc693fcebdc/grpcio-1.75.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:5b8ea230c7f77c0a1a3208a04a1eda164633fb0767b4cefd65a01079b65e5b1f", size = 7107828, upload-time = "2025-09-26T09:01:41.782Z" }, - { url = "https://files.pythonhosted.org/packages/e4/f8/d18b984c1c9ba0318e3628dbbeb6af77a5007f02abc378c845070f2d3edd/grpcio-1.75.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:36990d629c3c9fb41e546414e5af52d0a7af37ce7113d9682c46d7e2919e4cca", size = 8045421, upload-time = "2025-09-26T09:01:45.835Z" }, - { url = "https://files.pythonhosted.org/packages/7e/b6/4bf9aacff45deca5eac5562547ed212556b831064da77971a4e632917da3/grpcio-1.75.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:b10ad908118d38c2453ade7ff790e5bce36580c3742919007a2a78e3a1e521ca", size = 7503290, upload-time = "2025-09-26T09:01:49.28Z" }, - { url = "https://files.pythonhosted.org/packages/3b/15/d8d69d10223cb54c887a2180bd29fe5fa2aec1d4995c8821f7aa6eaf72e4/grpcio-1.75.1-cp311-cp311-win32.whl", hash = "sha256:d6be2b5ee7bea656c954dcf6aa8093c6f0e6a3ef9945c99d99fcbfc88c5c0bfe", size = 3950631, upload-time = "2025-09-26T09:01:51.23Z" }, - { url = "https://files.pythonhosted.org/packages/8a/40/7b8642d45fff6f83300c24eaac0380a840e5e7fe0e8d80afd31b99d7134e/grpcio-1.75.1-cp311-cp311-win_amd64.whl", hash = "sha256:61c692fb05956b17dd6d1ab480f7f10ad0536dba3bc8fd4e3c7263dc244ed772", size = 4646131, upload-time = "2025-09-26T09:01:53.266Z" }, - { url = "https://files.pythonhosted.org/packages/3a/81/42be79e73a50aaa20af66731c2defeb0e8c9008d9935a64dd8ea8e8c44eb/grpcio-1.75.1-cp312-cp312-linux_armv7l.whl", hash = "sha256:7b888b33cd14085d86176b1628ad2fcbff94cfbbe7809465097aa0132e58b018", size = 5668314, upload-time = "2025-09-26T09:01:55.424Z" }, - { url = "https://files.pythonhosted.org/packages/c5/a7/3686ed15822fedc58c22f82b3a7403d9faf38d7c33de46d4de6f06e49426/grpcio-1.75.1-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:8775036efe4ad2085975531d221535329f5dac99b6c2a854a995456098f99546", size = 11476125, upload-time = "2025-09-26T09:01:57.927Z" }, - { url = "https://files.pythonhosted.org/packages/14/85/21c71d674f03345ab183c634ecd889d3330177e27baea8d5d247a89b6442/grpcio-1.75.1-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:bb658f703468d7fbb5dcc4037c65391b7dc34f808ac46ed9136c24fc5eeb041d", size = 6246335, upload-time = "2025-09-26T09:02:00.76Z" }, - { url = "https://files.pythonhosted.org/packages/fd/db/3beb661bc56a385ae4fa6b0e70f6b91ac99d47afb726fe76aaff87ebb116/grpcio-1.75.1-cp312-cp312-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:4b7177a1cdb3c51b02b0c0a256b0a72fdab719600a693e0e9037949efffb200b", size = 6916309, upload-time = "2025-09-26T09:02:02.894Z" }, - { url = "https://files.pythonhosted.org/packages/1e/9c/eda9fe57f2b84343d44c1b66cf3831c973ba29b078b16a27d4587a1fdd47/grpcio-1.75.1-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:7d4fa6ccc3ec2e68a04f7b883d354d7fea22a34c44ce535a2f0c0049cf626ddf", size = 6435419, upload-time = "2025-09-26T09:02:05.055Z" }, - { url = "https://files.pythonhosted.org/packages/c3/b8/090c98983e0a9d602e3f919a6e2d4e470a8b489452905f9a0fa472cac059/grpcio-1.75.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:3d86880ecaeb5b2f0a8afa63824de93adb8ebe4e49d0e51442532f4e08add7d6", size = 7064893, upload-time = "2025-09-26T09:02:07.275Z" }, - { url = "https://files.pythonhosted.org/packages/ec/c0/6d53d4dbbd00f8bd81571f5478d8a95528b716e0eddb4217cc7cb45aae5f/grpcio-1.75.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:a8041d2f9e8a742aeae96f4b047ee44e73619f4f9d24565e84d5446c623673b6", size = 8011922, upload-time = "2025-09-26T09:02:09.527Z" }, - { url = "https://files.pythonhosted.org/packages/f2/7c/48455b2d0c5949678d6982c3e31ea4d89df4e16131b03f7d5c590811cbe9/grpcio-1.75.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:3652516048bf4c314ce12be37423c79829f46efffb390ad64149a10c6071e8de", size = 7466181, upload-time = "2025-09-26T09:02:12.279Z" }, - { url = "https://files.pythonhosted.org/packages/fd/12/04a0e79081e3170b6124f8cba9b6275871276be06c156ef981033f691880/grpcio-1.75.1-cp312-cp312-win32.whl", hash = "sha256:44b62345d8403975513af88da2f3d5cc76f73ca538ba46596f92a127c2aea945", size = 3938543, upload-time = "2025-09-26T09:02:14.77Z" }, - { url = "https://files.pythonhosted.org/packages/5f/d7/11350d9d7fb5adc73d2b0ebf6ac1cc70135577701e607407fe6739a90021/grpcio-1.75.1-cp312-cp312-win_amd64.whl", hash = "sha256:b1e191c5c465fa777d4cafbaacf0c01e0d5278022082c0abbd2ee1d6454ed94d", size = 4641938, upload-time = "2025-09-26T09:02:16.927Z" }, + { url = "https://files.pythonhosted.org/packages/a0/00/8163a1beeb6971f66b4bbe6ac9457b97948beba8dd2fc8e1281dce7f79ec/grpcio-1.76.0-cp311-cp311-linux_armv7l.whl", hash = "sha256:2e1743fbd7f5fa713a1b0a8ac8ebabf0ec980b5d8809ec358d488e273b9cf02a", size = 5843567, upload-time = "2025-10-21T16:20:52.829Z" }, + { url = "https://files.pythonhosted.org/packages/10/c1/934202f5cf335e6d852530ce14ddb0fef21be612ba9ecbbcbd4d748ca32d/grpcio-1.76.0-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:a8c2cf1209497cf659a667d7dea88985e834c24b7c3b605e6254cbb5076d985c", size = 11848017, upload-time = "2025-10-21T16:20:56.705Z" }, + { url = "https://files.pythonhosted.org/packages/11/0b/8dec16b1863d74af6eb3543928600ec2195af49ca58b16334972f6775663/grpcio-1.76.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:08caea849a9d3c71a542827d6df9d5a69067b0a1efbea8a855633ff5d9571465", size = 6412027, upload-time = "2025-10-21T16:20:59.3Z" }, + { url = "https://files.pythonhosted.org/packages/d7/64/7b9e6e7ab910bea9d46f2c090380bab274a0b91fb0a2fe9b0cd399fffa12/grpcio-1.76.0-cp311-cp311-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:f0e34c2079d47ae9f6188211db9e777c619a21d4faba6977774e8fa43b085e48", size = 7075913, upload-time = "2025-10-21T16:21:01.645Z" }, + { url = "https://files.pythonhosted.org/packages/68/86/093c46e9546073cefa789bd76d44c5cb2abc824ca62af0c18be590ff13ba/grpcio-1.76.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:8843114c0cfce61b40ad48df65abcfc00d4dba82eae8718fab5352390848c5da", size = 6615417, upload-time = "2025-10-21T16:21:03.844Z" }, + { url = "https://files.pythonhosted.org/packages/f7/b6/5709a3a68500a9c03da6fb71740dcdd5ef245e39266461a03f31a57036d8/grpcio-1.76.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:8eddfb4d203a237da6f3cc8a540dad0517d274b5a1e9e636fd8d2c79b5c1d397", size = 7199683, upload-time = "2025-10-21T16:21:06.195Z" }, + { url = "https://files.pythonhosted.org/packages/91/d3/4b1f2bf16ed52ce0b508161df3a2d186e4935379a159a834cb4a7d687429/grpcio-1.76.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:32483fe2aab2c3794101c2a159070584e5db11d0aa091b2c0ea9c4fc43d0d749", size = 8163109, upload-time = "2025-10-21T16:21:08.498Z" }, + { url = "https://files.pythonhosted.org/packages/5c/61/d9043f95f5f4cf085ac5dd6137b469d41befb04bd80280952ffa2a4c3f12/grpcio-1.76.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:dcfe41187da8992c5f40aa8c5ec086fa3672834d2be57a32384c08d5a05b4c00", size = 7626676, upload-time = "2025-10-21T16:21:10.693Z" }, + { url = "https://files.pythonhosted.org/packages/36/95/fd9a5152ca02d8881e4dd419cdd790e11805979f499a2e5b96488b85cf27/grpcio-1.76.0-cp311-cp311-win32.whl", hash = "sha256:2107b0c024d1b35f4083f11245c0e23846ae64d02f40b2b226684840260ed054", size = 3997688, upload-time = "2025-10-21T16:21:12.746Z" }, + { url = "https://files.pythonhosted.org/packages/60/9c/5c359c8d4c9176cfa3c61ecd4efe5affe1f38d9bae81e81ac7186b4c9cc8/grpcio-1.76.0-cp311-cp311-win_amd64.whl", hash = "sha256:522175aba7af9113c48ec10cc471b9b9bd4f6ceb36aeb4544a8e2c80ed9d252d", size = 4709315, upload-time = "2025-10-21T16:21:15.26Z" }, + { url = "https://files.pythonhosted.org/packages/bf/05/8e29121994b8d959ffa0afd28996d452f291b48cfc0875619de0bde2c50c/grpcio-1.76.0-cp312-cp312-linux_armv7l.whl", hash = "sha256:81fd9652b37b36f16138611c7e884eb82e0cec137c40d3ef7c3f9b3ed00f6ed8", size = 5799718, upload-time = "2025-10-21T16:21:17.939Z" }, + { url = "https://files.pythonhosted.org/packages/d9/75/11d0e66b3cdf998c996489581bdad8900db79ebd83513e45c19548f1cba4/grpcio-1.76.0-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:04bbe1bfe3a68bbfd4e52402ab7d4eb59d72d02647ae2042204326cf4bbad280", size = 11825627, upload-time = "2025-10-21T16:21:20.466Z" }, + { url = "https://files.pythonhosted.org/packages/28/50/2f0aa0498bc188048f5d9504dcc5c2c24f2eb1a9337cd0fa09a61a2e75f0/grpcio-1.76.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:d388087771c837cdb6515539f43b9d4bf0b0f23593a24054ac16f7a960be16f4", size = 6359167, upload-time = "2025-10-21T16:21:23.122Z" }, + { url = "https://files.pythonhosted.org/packages/66/e5/bbf0bb97d29ede1d59d6588af40018cfc345b17ce979b7b45424628dc8bb/grpcio-1.76.0-cp312-cp312-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:9f8f757bebaaea112c00dba718fc0d3260052ce714e25804a03f93f5d1c6cc11", size = 7044267, upload-time = "2025-10-21T16:21:25.995Z" }, + { url = "https://files.pythonhosted.org/packages/f5/86/f6ec2164f743d9609691115ae8ece098c76b894ebe4f7c94a655c6b03e98/grpcio-1.76.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:980a846182ce88c4f2f7e2c22c56aefd515daeb36149d1c897f83cf57999e0b6", size = 6573963, upload-time = "2025-10-21T16:21:28.631Z" }, + { url = "https://files.pythonhosted.org/packages/60/bc/8d9d0d8505feccfdf38a766d262c71e73639c165b311c9457208b56d92ae/grpcio-1.76.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:f92f88e6c033db65a5ae3d97905c8fea9c725b63e28d5a75cb73b49bda5024d8", size = 7164484, upload-time = "2025-10-21T16:21:30.837Z" }, + { url = "https://files.pythonhosted.org/packages/67/e6/5d6c2fc10b95edf6df9b8f19cf10a34263b7fd48493936fffd5085521292/grpcio-1.76.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:4baf3cbe2f0be3289eb68ac8ae771156971848bb8aaff60bad42005539431980", size = 8127777, upload-time = "2025-10-21T16:21:33.577Z" }, + { url = "https://files.pythonhosted.org/packages/3f/c8/dce8ff21c86abe025efe304d9e31fdb0deaaa3b502b6a78141080f206da0/grpcio-1.76.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:615ba64c208aaceb5ec83bfdce7728b80bfeb8be97562944836a7a0a9647d882", size = 7594014, upload-time = "2025-10-21T16:21:41.882Z" }, + { url = "https://files.pythonhosted.org/packages/e0/42/ad28191ebf983a5d0ecef90bab66baa5a6b18f2bfdef9d0a63b1973d9f75/grpcio-1.76.0-cp312-cp312-win32.whl", hash = "sha256:45d59a649a82df5718fd9527ce775fd66d1af35e6d31abdcdc906a49c6822958", size = 3984750, upload-time = "2025-10-21T16:21:44.006Z" }, + { url = "https://files.pythonhosted.org/packages/9e/00/7bd478cbb851c04a48baccaa49b75abaa8e4122f7d86da797500cccdd771/grpcio-1.76.0-cp312-cp312-win_amd64.whl", hash = "sha256:c088e7a90b6017307f423efbb9d1ba97a22aa2170876223f9709e9d1de0b5347", size = 4704003, upload-time = "2025-10-21T16:21:46.244Z" }, ] [[package]] @@ -2603,55 +2635,51 @@ wheels = [ [[package]] name = "hf-xet" -version = "1.1.10" +version = "1.2.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/74/31/feeddfce1748c4a233ec1aa5b7396161c07ae1aa9b7bdbc9a72c3c7dd768/hf_xet-1.1.10.tar.gz", hash = "sha256:408aef343800a2102374a883f283ff29068055c111f003ff840733d3b715bb97", size = 487910, upload-time = "2025-09-12T20:10:27.12Z" } +sdist = { url = "https://files.pythonhosted.org/packages/5e/6e/0f11bacf08a67f7fb5ee09740f2ca54163863b07b70d579356e9222ce5d8/hf_xet-1.2.0.tar.gz", hash = "sha256:a8c27070ca547293b6890c4bf389f713f80e8c478631432962bb7f4bc0bd7d7f", size = 506020, upload-time = "2025-10-24T19:04:32.129Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/f7/a2/343e6d05de96908366bdc0081f2d8607d61200be2ac802769c4284cc65bd/hf_xet-1.1.10-cp37-abi3-macosx_10_12_x86_64.whl", hash = "sha256:686083aca1a6669bc85c21c0563551cbcdaa5cf7876a91f3d074a030b577231d", size = 2761466, upload-time = "2025-09-12T20:10:22.836Z" }, - { url = "https://files.pythonhosted.org/packages/31/f9/6215f948ac8f17566ee27af6430ea72045e0418ce757260248b483f4183b/hf_xet-1.1.10-cp37-abi3-macosx_11_0_arm64.whl", hash = "sha256:71081925383b66b24eedff3013f8e6bbd41215c3338be4b94ba75fd75b21513b", size = 2623807, upload-time = "2025-09-12T20:10:21.118Z" }, - { url = "https://files.pythonhosted.org/packages/15/07/86397573efefff941e100367bbda0b21496ffcdb34db7ab51912994c32a2/hf_xet-1.1.10-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6b6bceb6361c80c1cc42b5a7b4e3efd90e64630bcf11224dcac50ef30a47e435", size = 3186960, upload-time = "2025-09-12T20:10:19.336Z" }, - { url = "https://files.pythonhosted.org/packages/01/a7/0b2e242b918cc30e1f91980f3c4b026ff2eedaf1e2ad96933bca164b2869/hf_xet-1.1.10-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:eae7c1fc8a664e54753ffc235e11427ca61f4b0477d757cc4eb9ae374b69f09c", size = 3087167, upload-time = "2025-09-12T20:10:17.255Z" }, - { url = "https://files.pythonhosted.org/packages/4a/25/3e32ab61cc7145b11eee9d745988e2f0f4fafda81b25980eebf97d8cff15/hf_xet-1.1.10-cp37-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:0a0005fd08f002180f7a12d4e13b22be277725bc23ed0529f8add5c7a6309c06", size = 3248612, upload-time = "2025-09-12T20:10:24.093Z" }, - { url = "https://files.pythonhosted.org/packages/2c/3d/ab7109e607ed321afaa690f557a9ada6d6d164ec852fd6bf9979665dc3d6/hf_xet-1.1.10-cp37-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:f900481cf6e362a6c549c61ff77468bd59d6dd082f3170a36acfef2eb6a6793f", size = 3353360, upload-time = "2025-09-12T20:10:25.563Z" }, - { url = "https://files.pythonhosted.org/packages/ee/0e/471f0a21db36e71a2f1752767ad77e92d8cde24e974e03d662931b1305ec/hf_xet-1.1.10-cp37-abi3-win_amd64.whl", hash = "sha256:5f54b19cc347c13235ae7ee98b330c26dd65ef1df47e5316ffb1e87713ca7045", size = 2804691, upload-time = "2025-09-12T20:10:28.433Z" }, + { url = "https://files.pythonhosted.org/packages/96/2d/22338486473df5923a9ab7107d375dbef9173c338ebef5098ef593d2b560/hf_xet-1.2.0-cp37-abi3-macosx_10_12_x86_64.whl", hash = "sha256:46740d4ac024a7ca9b22bebf77460ff43332868b661186a8e46c227fdae01848", size = 2866099, upload-time = "2025-10-24T19:04:15.366Z" }, + { url = "https://files.pythonhosted.org/packages/7f/8c/c5becfa53234299bc2210ba314eaaae36c2875e0045809b82e40a9544f0c/hf_xet-1.2.0-cp37-abi3-macosx_11_0_arm64.whl", hash = "sha256:27df617a076420d8845bea087f59303da8be17ed7ec0cd7ee3b9b9f579dff0e4", size = 2722178, upload-time = "2025-10-24T19:04:13.695Z" }, + { url = "https://files.pythonhosted.org/packages/9a/92/cf3ab0b652b082e66876d08da57fcc6fa2f0e6c70dfbbafbd470bb73eb47/hf_xet-1.2.0-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3651fd5bfe0281951b988c0facbe726aa5e347b103a675f49a3fa8144c7968fd", size = 3320214, upload-time = "2025-10-24T19:04:03.596Z" }, + { url = "https://files.pythonhosted.org/packages/46/92/3f7ec4a1b6a65bf45b059b6d4a5d38988f63e193056de2f420137e3c3244/hf_xet-1.2.0-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:d06fa97c8562fb3ee7a378dd9b51e343bc5bc8190254202c9771029152f5e08c", size = 3229054, upload-time = "2025-10-24T19:04:01.949Z" }, + { url = "https://files.pythonhosted.org/packages/0b/dd/7ac658d54b9fb7999a0ccb07ad863b413cbaf5cf172f48ebcd9497ec7263/hf_xet-1.2.0-cp37-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:4c1428c9ae73ec0939410ec73023c4f842927f39db09b063b9482dac5a3bb737", size = 3413812, upload-time = "2025-10-24T19:04:24.585Z" }, + { url = "https://files.pythonhosted.org/packages/92/68/89ac4e5b12a9ff6286a12174c8538a5930e2ed662091dd2572bbe0a18c8a/hf_xet-1.2.0-cp37-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:a55558084c16b09b5ed32ab9ed38421e2d87cf3f1f89815764d1177081b99865", size = 3508920, upload-time = "2025-10-24T19:04:26.927Z" }, + { url = "https://files.pythonhosted.org/packages/cb/44/870d44b30e1dcfb6a65932e3e1506c103a8a5aea9103c337e7a53180322c/hf_xet-1.2.0-cp37-abi3-win_amd64.whl", hash = "sha256:e6584a52253f72c9f52f9e549d5895ca7a471608495c4ecaa6cc73dba2b24d69", size = 2905735, upload-time = "2025-10-24T19:04:35.928Z" }, ] [[package]] name = "hiredis" -version = "3.2.1" +version = "3.3.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/f7/08/24b72f425b75e1de7442fb1740f69ca66d5820b9f9c0e2511ff9aadab3b7/hiredis-3.2.1.tar.gz", hash = "sha256:5a5f64479bf04dd829fe7029fad0ea043eac4023abc6e946668cbbec3493a78d", size = 89096, upload-time = "2025-05-23T11:41:57.227Z" } +sdist = { url = "https://files.pythonhosted.org/packages/65/82/d2817ce0653628e0a0cb128533f6af0dd6318a49f3f3a6a7bd1f2f2154af/hiredis-3.3.0.tar.gz", hash = "sha256:105596aad9249634361815c574351f1bd50455dc23b537c2940066c4a9dea685", size = 89048, upload-time = "2025-10-14T16:33:34.263Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/48/84/2ea9636f2ba0811d9eb3bebbbfa84f488238180ddab70c9cb7fa13419d78/hiredis-3.2.1-cp311-cp311-macosx_10_15_universal2.whl", hash = "sha256:e4ae0be44cab5e74e6e4c4a93d04784629a45e781ff483b136cc9e1b9c23975c", size = 82425, upload-time = "2025-05-23T11:39:54.135Z" }, - { url = "https://files.pythonhosted.org/packages/fc/24/b9ebf766a99998fda3975937afa4912e98de9d7f8d0b83f48096bdd961c1/hiredis-3.2.1-cp311-cp311-macosx_10_15_x86_64.whl", hash = "sha256:24647e84c9f552934eb60b7f3d2116f8b64a7020361da9369e558935ca45914d", size = 45231, upload-time = "2025-05-23T11:39:55.455Z" }, - { url = "https://files.pythonhosted.org/packages/68/4c/c009b4d9abeb964d607f0987561892d1589907f770b9e5617552b34a4a4d/hiredis-3.2.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:6fb3e92d1172da8decc5f836bf8b528c0fc9b6d449f1353e79ceeb9dc1801132", size = 43240, upload-time = "2025-05-23T11:39:57.8Z" }, - { url = "https://files.pythonhosted.org/packages/e9/83/d53f3ae9e4ac51b8a35afb7ccd68db871396ed1d7c8ba02ce2c30de0cf17/hiredis-3.2.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:38ba7a32e51e518b6b3e470142e52ed2674558e04d7d73d86eb19ebcb37d7d40", size = 169624, upload-time = "2025-05-23T11:40:00.055Z" }, - { url = "https://files.pythonhosted.org/packages/91/2f/f9f091526e22a45385d45f3870204dc78aee365b6fe32e679e65674da6a7/hiredis-3.2.1-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4fc632be73174891d6bb71480247e57b2fd8f572059f0a1153e4d0339e919779", size = 165799, upload-time = "2025-05-23T11:40:01.194Z" }, - { url = "https://files.pythonhosted.org/packages/1c/cc/e561274438cdb19794f0638136a5a99a9ca19affcb42679b12a78016b8ad/hiredis-3.2.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f03e6839ff21379ad3c195e0700fc9c209e7f344946dea0f8a6d7b5137a2a141", size = 180612, upload-time = "2025-05-23T11:40:02.385Z" }, - { url = "https://files.pythonhosted.org/packages/83/ba/a8a989f465191d55672e57aea2a331bfa3a74b5cbc6f590031c9e11f7491/hiredis-3.2.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:99983873e37c71bb71deb544670ff4f9d6920dab272aaf52365606d87a4d6c73", size = 169934, upload-time = "2025-05-23T11:40:03.524Z" }, - { url = "https://files.pythonhosted.org/packages/52/5f/1148e965df1c67b17bdcaef199f54aec3def0955d19660a39c6ee10a6f55/hiredis-3.2.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ffd982c419f48e3a57f592678c72474429465bb4bfc96472ec805f5d836523f0", size = 170074, upload-time = "2025-05-23T11:40:04.618Z" }, - { url = "https://files.pythonhosted.org/packages/43/5e/e6846ad159a938b539fb8d472e2e68cb6758d7c9454ea0520211f335ea72/hiredis-3.2.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:bc993f4aa4abc029347f309e722f122e05a3b8a0c279ae612849b5cc9dc69f2d", size = 164158, upload-time = "2025-05-23T11:40:05.653Z" }, - { url = "https://files.pythonhosted.org/packages/0a/a1/5891e0615f0993f194c1b51a65aaac063b0db318a70df001b28e49f0579d/hiredis-3.2.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:dde790d420081f18b5949227649ccb3ed991459df33279419a25fcae7f97cd92", size = 162591, upload-time = "2025-05-23T11:40:07.041Z" }, - { url = "https://files.pythonhosted.org/packages/d4/da/8bce52ca81716f53c1014f689aea4c170ba6411e6848f81a1bed1fc375eb/hiredis-3.2.1-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:b0c8cae7edbef860afcf3177b705aef43e10b5628f14d5baf0ec69668247d08d", size = 174808, upload-time = "2025-05-23T11:40:09.146Z" }, - { url = "https://files.pythonhosted.org/packages/84/91/fc1ef444ed4dc432b5da9b48e9bd23266c703528db7be19e2b608d67ba06/hiredis-3.2.1-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:e8a90eaca7e1ce7f175584f07a2cdbbcab13f4863f9f355d7895c4d28805f65b", size = 167060, upload-time = "2025-05-23T11:40:10.757Z" }, - { url = "https://files.pythonhosted.org/packages/66/ad/beebf73a5455f232b97e00564d1e8ad095d4c6e18858c60c6cfdd893ac1e/hiredis-3.2.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:476031958fa44e245e803827e0787d49740daa4de708fe514370293ce519893a", size = 164833, upload-time = "2025-05-23T11:40:12.001Z" }, - { url = "https://files.pythonhosted.org/packages/75/79/a9591bdc0148c0fbdf54cf6f3d449932d3b3b8779e87f33fa100a5a8088f/hiredis-3.2.1-cp311-cp311-win32.whl", hash = "sha256:eb3f5df2a9593b4b4b676dce3cea53b9c6969fc372875188589ddf2bafc7f624", size = 20402, upload-time = "2025-05-23T11:40:13.216Z" }, - { url = "https://files.pythonhosted.org/packages/9f/05/c93cc6fab31e3c01b671126c82f44372fb211facb8bd4571fd372f50898d/hiredis-3.2.1-cp311-cp311-win_amd64.whl", hash = "sha256:1402e763d8a9fdfcc103bbf8b2913971c0a3f7b8a73deacbda3dfe5f3a9d1e0b", size = 22085, upload-time = "2025-05-23T11:40:14.19Z" }, - { url = "https://files.pythonhosted.org/packages/60/a1/6da1578a22df1926497f7a3f6a3d2408fe1d1559f762c1640af5762a8eb6/hiredis-3.2.1-cp312-cp312-macosx_10_15_universal2.whl", hash = "sha256:3742d8b17e73c198cabeab11da35f2e2a81999d406f52c6275234592256bf8e8", size = 82627, upload-time = "2025-05-23T11:40:15.362Z" }, - { url = "https://files.pythonhosted.org/packages/6c/b1/1056558ca8dc330be5bb25162fe5f268fee71571c9a535153df9f871a073/hiredis-3.2.1-cp312-cp312-macosx_10_15_x86_64.whl", hash = "sha256:9c2f3176fb617a79f6cccf22cb7d2715e590acb534af6a82b41f8196ad59375d", size = 45404, upload-time = "2025-05-23T11:40:16.72Z" }, - { url = "https://files.pythonhosted.org/packages/58/4f/13d1fa1a6b02a99e9fed8f546396f2d598c3613c98e6c399a3284fa65361/hiredis-3.2.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:a8bd46189c7fa46174e02670dc44dfecb60f5bd4b67ed88cb050d8f1fd842f09", size = 43299, upload-time = "2025-05-23T11:40:17.697Z" }, - { url = "https://files.pythonhosted.org/packages/c0/25/ddfac123ba5a32eb1f0b40ba1b2ec98a599287f7439def8856c3c7e5dd0d/hiredis-3.2.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f86ee4488c8575b58139cdfdddeae17f91e9a893ffee20260822add443592e2f", size = 172194, upload-time = "2025-05-23T11:40:19.143Z" }, - { url = "https://files.pythonhosted.org/packages/2c/1e/443a3703ce570b631ca43494094fbaeb051578a0ebe4bfcefde351e1ba25/hiredis-3.2.1-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3717832f4a557b2fe7060b9d4a7900e5de287a15595e398c3f04df69019ca69d", size = 168429, upload-time = "2025-05-23T11:40:20.329Z" }, - { url = "https://files.pythonhosted.org/packages/3b/d6/0d8c6c706ed79b2298c001b5458c055615e3166533dcee3900e821a18a3e/hiredis-3.2.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e5cb12c21fb9e2403d28c4e6a38120164973342d34d08120f2d7009b66785644", size = 182967, upload-time = "2025-05-23T11:40:21.921Z" }, - { url = "https://files.pythonhosted.org/packages/da/68/da8dd231fbce858b5a20ab7d7bf558912cd125f08bac4c778865ef5fe2c2/hiredis-3.2.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:080fda1510bbd389af91f919c11a4f2aa4d92f0684afa4709236faa084a42cac", size = 172495, upload-time = "2025-05-23T11:40:23.105Z" }, - { url = "https://files.pythonhosted.org/packages/65/25/83a31420535e2778662caa95533d5c997011fa6a88331f0cdb22afea9ec3/hiredis-3.2.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1252e10a1f3273d1c6bf2021e461652c2e11b05b83e0915d6eb540ec7539afe2", size = 173142, upload-time = "2025-05-23T11:40:24.24Z" }, - { url = "https://files.pythonhosted.org/packages/41/d7/cb907348889eb75e2aa2e6b63e065b611459e0f21fe1e371a968e13f0d55/hiredis-3.2.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:d9e320e99ab7d2a30dc91ff6f745ba38d39b23f43d345cdee9881329d7b511d6", size = 166433, upload-time = "2025-05-23T11:40:25.287Z" }, - { url = "https://files.pythonhosted.org/packages/01/5d/7cbc69d82af7b29a95723d50f5261555ba3d024bfbdc414bdc3d23c0defb/hiredis-3.2.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:641668f385f16550fdd6fdc109b0af6988b94ba2acc06770a5e06a16e88f320c", size = 164883, upload-time = "2025-05-23T11:40:26.454Z" }, - { url = "https://files.pythonhosted.org/packages/f9/00/f995b1296b1d7e0247651347aa230f3225a9800e504fdf553cf7cd001cf7/hiredis-3.2.1-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:1e1f44208c39d6c345ff451f82f21e9eeda6fe9af4ac65972cc3eeb58d41f7cb", size = 177262, upload-time = "2025-05-23T11:40:27.576Z" }, - { url = "https://files.pythonhosted.org/packages/c5/f3/723a67d729e94764ce9e0d73fa5f72a0f87d3ce3c98c9a0b27cbf001cc79/hiredis-3.2.1-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:f882a0d6415fffe1ffcb09e6281d0ba8b1ece470e866612bbb24425bf76cf397", size = 169619, upload-time = "2025-05-23T11:40:29.671Z" }, - { url = "https://files.pythonhosted.org/packages/45/58/f69028df00fb1b223e221403f3be2059ae86031e7885f955d26236bdfc17/hiredis-3.2.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:b4e78719a0730ebffe335528531d154bc8867a246418f74ecd88adbc4d938c49", size = 167303, upload-time = "2025-05-23T11:40:30.902Z" }, - { url = "https://files.pythonhosted.org/packages/2b/7d/567411e65cce76cf265a9a4f837fd2ebc564bef6368dd42ac03f7a517c0a/hiredis-3.2.1-cp312-cp312-win32.whl", hash = "sha256:33c4604d9f79a13b84da79950a8255433fca7edaf292bbd3364fd620864ed7b2", size = 20551, upload-time = "2025-05-23T11:40:32.69Z" }, - { url = "https://files.pythonhosted.org/packages/90/74/b4c291eb4a4a874b3690ff9fc311a65d5292072556421b11b1d786e3e1d0/hiredis-3.2.1-cp312-cp312-win_amd64.whl", hash = "sha256:7b9749375bf9d171aab8813694f379f2cff0330d7424000f5e92890ad4932dc9", size = 22128, upload-time = "2025-05-23T11:40:33.686Z" }, + { url = "https://files.pythonhosted.org/packages/34/0c/be3b1093f93a7c823ca16fbfbb83d3a1de671bbd2add8da1fe2bcfccb2b8/hiredis-3.3.0-cp311-cp311-macosx_10_15_universal2.whl", hash = "sha256:63ee6c1ae6a2462a2439eb93c38ab0315cd5f4b6d769c6a34903058ba538b5d6", size = 81813, upload-time = "2025-10-14T16:32:00.576Z" }, + { url = "https://files.pythonhosted.org/packages/95/2b/ed722d392ac59a7eee548d752506ef32c06ffdd0bce9cf91125a74b8edf9/hiredis-3.3.0-cp311-cp311-macosx_10_15_x86_64.whl", hash = "sha256:31eda3526e2065268a8f97fbe3d0e9a64ad26f1d89309e953c80885c511ea2ae", size = 46049, upload-time = "2025-10-14T16:32:01.319Z" }, + { url = "https://files.pythonhosted.org/packages/e5/61/8ace8027d5b3f6b28e1dc55f4a504be038ba8aa8bf71882b703e8f874c91/hiredis-3.3.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a26bae1b61b7bcafe3d0d0c7d012fb66ab3c95f2121dbea336df67e344e39089", size = 41814, upload-time = "2025-10-14T16:32:02.076Z" }, + { url = "https://files.pythonhosted.org/packages/23/0e/380ade1ffb21034976663a5128f0383533f35caccdba13ff0537dd5ace79/hiredis-3.3.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b9546079f7fd5c50fbff9c791710049b32eebe7f9b94debec1e8b9f4c048cba2", size = 167572, upload-time = "2025-10-14T16:32:03.125Z" }, + { url = "https://files.pythonhosted.org/packages/ca/60/b4a8d2177575b896730f73e6890644591aa56790a75c2b6d6f2302a1dae6/hiredis-3.3.0-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:ae327fc13b1157b694d53f92d50920c0051e30b0c245f980a7036e299d039ab4", size = 179373, upload-time = "2025-10-14T16:32:04.04Z" }, + { url = "https://files.pythonhosted.org/packages/31/53/a473a18d27cfe8afda7772ff9adfba1718fd31d5e9c224589dc17774fa0b/hiredis-3.3.0-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:4016e50a8be5740a59c5af5252e5ad16c395021a999ad24c6604f0d9faf4d346", size = 177504, upload-time = "2025-10-14T16:32:04.934Z" }, + { url = "https://files.pythonhosted.org/packages/7e/0f/f6ee4c26b149063dbf5b1b6894b4a7a1f00a50e3d0cfd30a22d4c3479db3/hiredis-3.3.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c17b473f273465a3d2168a57a5b43846165105ac217d5652a005e14068589ddc", size = 169449, upload-time = "2025-10-14T16:32:05.808Z" }, + { url = "https://files.pythonhosted.org/packages/64/38/e3e113172289e1261ccd43e387a577dd268b0b9270721b5678735803416c/hiredis-3.3.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:9ecd9b09b11bd0b8af87d29c3f5da628d2bdc2a6c23d2dd264d2da082bd4bf32", size = 164010, upload-time = "2025-10-14T16:32:06.695Z" }, + { url = "https://files.pythonhosted.org/packages/8d/9a/ccf4999365691ea73d0dd2ee95ee6ef23ebc9a835a7417f81765bc49eade/hiredis-3.3.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:00fb04eac208cd575d14f246e74a468561081ce235937ab17d77cde73aefc66c", size = 174623, upload-time = "2025-10-14T16:32:07.627Z" }, + { url = "https://files.pythonhosted.org/packages/ed/c7/ee55fa2ade078b7c4f17e8ddc9bc28881d0b71b794ebf9db4cfe4c8f0623/hiredis-3.3.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:60814a7d0b718adf3bfe2c32c6878b0e00d6ae290ad8e47f60d7bba3941234a6", size = 167650, upload-time = "2025-10-14T16:32:08.615Z" }, + { url = "https://files.pythonhosted.org/packages/bf/06/f6cd90275dcb0ba03f69767805151eb60b602bc25830648bd607660e1f97/hiredis-3.3.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:fcbd1a15e935aa323b5b2534b38419511b7909b4b8ee548e42b59090a1b37bb1", size = 165452, upload-time = "2025-10-14T16:32:09.561Z" }, + { url = "https://files.pythonhosted.org/packages/c3/10/895177164a6c4409a07717b5ae058d84a908e1ab629f0401110b02aaadda/hiredis-3.3.0-cp311-cp311-win32.whl", hash = "sha256:73679607c5a19f4bcfc9cf6eb54480bcd26617b68708ac8b1079da9721be5449", size = 20394, upload-time = "2025-10-14T16:32:10.469Z" }, + { url = "https://files.pythonhosted.org/packages/3c/c7/1e8416ae4d4134cb62092c61cabd76b3d720507ee08edd19836cdeea4c7a/hiredis-3.3.0-cp311-cp311-win_amd64.whl", hash = "sha256:30a4df3d48f32538de50648d44146231dde5ad7f84f8f08818820f426840ae97", size = 22336, upload-time = "2025-10-14T16:32:11.221Z" }, + { url = "https://files.pythonhosted.org/packages/48/1c/ed28ae5d704f5c7e85b946fa327f30d269e6272c847fef7e91ba5fc86193/hiredis-3.3.0-cp312-cp312-macosx_10_15_universal2.whl", hash = "sha256:5b8e1d6a2277ec5b82af5dce11534d3ed5dffeb131fd9b210bc1940643b39b5f", size = 82026, upload-time = "2025-10-14T16:32:12.004Z" }, + { url = "https://files.pythonhosted.org/packages/f4/9b/79f30c5c40e248291023b7412bfdef4ad9a8a92d9e9285d65d600817dac7/hiredis-3.3.0-cp312-cp312-macosx_10_15_x86_64.whl", hash = "sha256:c4981de4d335f996822419e8a8b3b87367fcef67dc5fb74d3bff4df9f6f17783", size = 46217, upload-time = "2025-10-14T16:32:13.133Z" }, + { url = "https://files.pythonhosted.org/packages/e7/c3/02b9ed430ad9087aadd8afcdf616717452d16271b701fa47edfe257b681e/hiredis-3.3.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1706480a683e328ae9ba5d704629dee2298e75016aa0207e7067b9c40cecc271", size = 41858, upload-time = "2025-10-14T16:32:13.98Z" }, + { url = "https://files.pythonhosted.org/packages/f1/98/b2a42878b82130a535c7aa20bc937ba2d07d72e9af3ad1ad93e837c419b5/hiredis-3.3.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0a95cef9989736ac313639f8f545b76b60b797e44e65834aabbb54e4fad8d6c8", size = 170195, upload-time = "2025-10-14T16:32:14.728Z" }, + { url = "https://files.pythonhosted.org/packages/66/1d/9dcde7a75115d3601b016113d9b90300726fa8e48aacdd11bf01a453c145/hiredis-3.3.0-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:ca2802934557ccc28a954414c245ba7ad904718e9712cb67c05152cf6b9dd0a3", size = 181808, upload-time = "2025-10-14T16:32:15.622Z" }, + { url = "https://files.pythonhosted.org/packages/56/a1/60f6bda9b20b4e73c85f7f5f046bc2c154a5194fc94eb6861e1fd97ced52/hiredis-3.3.0-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:fe730716775f61e76d75810a38ee4c349d3af3896450f1525f5a4034cf8f2ed7", size = 180578, upload-time = "2025-10-14T16:32:16.514Z" }, + { url = "https://files.pythonhosted.org/packages/d9/01/859d21de65085f323a701824e23ea3330a0ac05f8e184544d7aa5c26128d/hiredis-3.3.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:749faa69b1ce1f741f5eaf743435ac261a9262e2d2d66089192477e7708a9abc", size = 172508, upload-time = "2025-10-14T16:32:17.411Z" }, + { url = "https://files.pythonhosted.org/packages/99/a8/28fd526e554c80853d0fbf57ef2a3235f00e4ed34ce0e622e05d27d0f788/hiredis-3.3.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:95c9427f2ac3f1dd016a3da4e1161fa9d82f221346c8f3fdd6f3f77d4e28946c", size = 166341, upload-time = "2025-10-14T16:32:18.561Z" }, + { url = "https://files.pythonhosted.org/packages/f2/91/ded746b7d2914f557fbbf77be55e90d21f34ba758ae10db6591927c642c8/hiredis-3.3.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:c863ee44fe7bff25e41f3a5105c936a63938b76299b802d758f40994ab340071", size = 176765, upload-time = "2025-10-14T16:32:19.491Z" }, + { url = "https://files.pythonhosted.org/packages/d6/4c/04aa46ff386532cb5f08ee495c2bf07303e93c0acf2fa13850e031347372/hiredis-3.3.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:2213c7eb8ad5267434891f3241c7776e3bafd92b5933fc57d53d4456247dc542", size = 170312, upload-time = "2025-10-14T16:32:20.404Z" }, + { url = "https://files.pythonhosted.org/packages/90/6e/67f9d481c63f542a9cf4c9f0ea4e5717db0312fb6f37fb1f78f3a66de93c/hiredis-3.3.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a172bae3e2837d74530cd60b06b141005075db1b814d966755977c69bd882ce8", size = 167965, upload-time = "2025-10-14T16:32:21.259Z" }, + { url = "https://files.pythonhosted.org/packages/7a/df/dde65144d59c3c0d85e43255798f1fa0c48d413e668cfd92b3d9f87924ef/hiredis-3.3.0-cp312-cp312-win32.whl", hash = "sha256:cb91363b9fd6d41c80df9795e12fffbaf5c399819e6ae8120f414dedce6de068", size = 20533, upload-time = "2025-10-14T16:32:22.192Z" }, + { url = "https://files.pythonhosted.org/packages/f5/a9/55a4ac9c16fdf32e92e9e22c49f61affe5135e177ca19b014484e28950f7/hiredis-3.3.0-cp312-cp312-win_amd64.whl", hash = "sha256:04ec150e95eea3de9ff8bac754978aa17b8bf30a86d4ab2689862020945396b0", size = 22379, upload-time = "2025-10-14T16:32:22.916Z" }, ] [[package]] @@ -2758,7 +2786,7 @@ wheels = [ [[package]] name = "huggingface-hub" -version = "0.35.3" +version = "0.36.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "filelock" }, @@ -2770,9 +2798,9 @@ dependencies = [ { name = "tqdm" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/10/7e/a0a97de7c73671863ca6b3f61fa12518caf35db37825e43d63a70956738c/huggingface_hub-0.35.3.tar.gz", hash = "sha256:350932eaa5cc6a4747efae85126ee220e4ef1b54e29d31c3b45c5612ddf0b32a", size = 461798, upload-time = "2025-09-29T14:29:58.625Z" } +sdist = { url = "https://files.pythonhosted.org/packages/98/63/4910c5fa9128fdadf6a9c5ac138e8b1b6cee4ca44bf7915bbfbce4e355ee/huggingface_hub-0.36.0.tar.gz", hash = "sha256:47b3f0e2539c39bf5cde015d63b72ec49baff67b6931c3d97f3f84532e2b8d25", size = 463358, upload-time = "2025-10-23T12:12:01.413Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/31/a0/651f93d154cb72323358bf2bbae3e642bdb5d2f1bfc874d096f7cb159fa0/huggingface_hub-0.35.3-py3-none-any.whl", hash = "sha256:0e3a01829c19d86d03793e4577816fe3bdfc1602ac62c7fb220d593d351224ba", size = 564262, upload-time = "2025-09-29T14:29:55.813Z" }, + { url = "https://files.pythonhosted.org/packages/cb/bd/1a875e0d592d447cbc02805fd3fe0f497714d6a2583f59d14fa9ebad96eb/huggingface_hub-0.36.0-py3-none-any.whl", hash = "sha256:7bcc9ad17d5b3f07b57c78e79d527102d08313caa278a641993acddcb894548d", size = 566094, upload-time = "2025-10-23T12:11:59.557Z" }, ] [[package]] @@ -2798,15 +2826,14 @@ wheels = [ [[package]] name = "hypothesis" -version = "6.140.3" +version = "6.147.0" source = { registry = "https://pypi.org/simple" } dependencies = [ - { name = "attrs" }, { name = "sortedcontainers" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/18/7f/946343e32881b56adc0eba64e428ad2f85251f9ef16e3e4ec1b6ab80199b/hypothesis-6.140.3.tar.gz", hash = "sha256:4f4a09bf77af21e0cc3dffed1ea639812dc75d38f81308ec9fb0e33f8557b0cb", size = 466925, upload-time = "2025-10-04T22:29:44.499Z" } +sdist = { url = "https://files.pythonhosted.org/packages/f3/53/e19fe74671fd60db86344a4623c818fac58b813cc3efbb7ea3b3074dcb71/hypothesis-6.147.0.tar.gz", hash = "sha256:72e6004ea3bd1460bdb4640b6389df23b87ba7a4851893fd84d1375635d3e507", size = 468587, upload-time = "2025-11-06T20:27:29.682Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/65/2a/0553ac2a8af432df92f2ffc05ca97e7ed64e00c97a371b019ae2690de325/hypothesis-6.140.3-py3-none-any.whl", hash = "sha256:a2cfff51641a58a56081f5c90ae1da6ccf3d043404f411805f7f0e0d75742d0e", size = 534534, upload-time = "2025-10-04T22:29:40.635Z" }, + { url = "https://files.pythonhosted.org/packages/b2/1b/932eddc3d55c4ed6c585006cffe6c6a133b5e1797d873de0bcf5208e4fed/hypothesis-6.147.0-py3-none-any.whl", hash = "sha256:de588807b6da33550d32f47bcd42b1a86d061df85673aa73e6443680249d185e", size = 535595, upload-time = "2025-11-06T20:27:23.536Z" }, ] [[package]] @@ -2820,16 +2847,16 @@ wheels = [ [[package]] name = "import-linter" -version = "2.5.2" +version = "2.6" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "click" }, { name = "grimp" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/6c/fd/49913b98fdeb5a8a120ca756abfc9aa7fdef7c20da1d728173e98ce11160/import_linter-2.5.2.tar.gz", hash = "sha256:d8f2dc6432975cc35edc4cc0bfcf1b811f05500b377ce0c3f62729d68f46c698", size = 159664, upload-time = "2025-10-09T10:53:24.635Z" } +sdist = { url = "https://files.pythonhosted.org/packages/1d/20/37f3661ccbdba41072a74cb7a57a932b6884ab6c489318903d2d870c6c07/import_linter-2.6.tar.gz", hash = "sha256:60429a450eb6ebeed536f6d2b83428b026c5747ca69d029812e2f1360b136f85", size = 161294, upload-time = "2025-11-10T09:59:20.977Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/c1/f4/f20eeb9e6ab178ce011457cd936877202556f14b7af3ef2b3c3e26f3758a/import_linter-2.5.2-py3-none-any.whl", hash = "sha256:a70b64c2451dc6b96ff9ef5af4e3f6a2c8b63532a66a3c96a7c31ca086b10003", size = 44140, upload-time = "2025-10-09T10:53:23.367Z" }, + { url = "https://files.pythonhosted.org/packages/44/df/02389e13d340229baa687bd0b9be4878e13668ce0beadbe531fb2b597386/import_linter-2.6-py3-none-any.whl", hash = "sha256:4e835141294b803325a619b8c789398320b81f0bde7771e0dd36f34524e51b1e", size = 46488, upload-time = "2025-11-10T09:59:19.611Z" }, ] [[package]] @@ -2855,11 +2882,11 @@ wheels = [ [[package]] name = "iniconfig" -version = "2.1.0" +version = "2.3.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/f2/97/ebf4da567aa6827c909642694d71c9fcf53e5b504f2d96afea02718862f3/iniconfig-2.1.0.tar.gz", hash = "sha256:3abbd2e30b36733fee78f9c7f7308f2d0050e88f0087fd25c2645f63c773e1c7", size = 4793, upload-time = "2025-03-19T20:09:59.721Z" } +sdist = { url = "https://files.pythonhosted.org/packages/72/34/14ca021ce8e5dfedc35312d08ba8bf51fdd999c576889fc2c24cb97f4f10/iniconfig-2.3.0.tar.gz", hash = "sha256:c76315c77db068650d49c5b56314774a7804df16fee4402c1f19d6d15d8c4730", size = 20503, upload-time = "2025-10-18T21:55:43.219Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/2c/e1/e6716421ea10d38022b952c159d5161ca1193197fb744506875fbb87ea7b/iniconfig-2.1.0-py3-none-any.whl", hash = "sha256:9deba5723312380e77435581c6bf4935c94cbfab9b1ed33ef8d238ea168eb760", size = 6050, upload-time = "2025-03-19T20:10:01.071Z" }, + { url = "https://files.pythonhosted.org/packages/cb/b1/3846dd7f199d53cb17f49cba7e651e9ce294d8497c8c150530ed11865bb8/iniconfig-2.3.0-py3-none-any.whl", hash = "sha256:f631c04d2c48c52b84d0d0549c99ff3859c98df65b3101406327ecc7d53fbf12", size = 7484, upload-time = "2025-10-18T21:55:41.639Z" }, ] [[package]] @@ -2909,35 +2936,44 @@ wheels = [ [[package]] name = "jiter" -version = "0.11.0" +version = "0.12.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/9d/c0/a3bb4cc13aced219dd18191ea66e874266bd8aa7b96744e495e1c733aa2d/jiter-0.11.0.tar.gz", hash = "sha256:1d9637eaf8c1d6a63d6562f2a6e5ab3af946c66037eb1b894e8fad75422266e4", size = 167094, upload-time = "2025-09-15T09:20:38.212Z" } +sdist = { url = "https://files.pythonhosted.org/packages/45/9d/e0660989c1370e25848bb4c52d061c71837239738ad937e83edca174c273/jiter-0.12.0.tar.gz", hash = "sha256:64dfcd7d5c168b38d3f9f8bba7fc639edb3418abcc74f22fdbe6b8938293f30b", size = 168294, upload-time = "2025-11-09T20:49:23.302Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/38/55/a69fefeef09c2eaabae44b935a1aa81517e49639c0a0c25d861cb18cd7ac/jiter-0.11.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:cb5d9db02979c3f49071fce51a48f4b4e4cf574175fb2b11c7a535fa4867b222", size = 309503, upload-time = "2025-09-15T09:19:08.191Z" }, - { url = "https://files.pythonhosted.org/packages/bd/d5/a6aba9e6551f32f9c127184f398208e4eddb96c59ac065c8a92056089d28/jiter-0.11.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:1dc6a123f3471c4730db7ca8ba75f1bb3dcb6faeb8d46dd781083e7dee88b32d", size = 317688, upload-time = "2025-09-15T09:19:09.918Z" }, - { url = "https://files.pythonhosted.org/packages/bb/f3/5e86f57c1883971cdc8535d0429c2787bf734840a231da30a3be12850562/jiter-0.11.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:09858f8d230f031c7b8e557429102bf050eea29c77ad9c34c8fe253c5329acb7", size = 337418, upload-time = "2025-09-15T09:19:11.078Z" }, - { url = "https://files.pythonhosted.org/packages/5e/4f/a71d8a24c2a70664970574a8e0b766663f5ef788f7fe1cc20ee0c016d488/jiter-0.11.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:dbe2196c4a0ce760925a74ab4456bf644748ab0979762139626ad138f6dac72d", size = 361423, upload-time = "2025-09-15T09:19:13.286Z" }, - { url = "https://files.pythonhosted.org/packages/8f/e5/b09076f4e7fd9471b91e16f9f3dc7330b161b738f3b39b2c37054a36e26a/jiter-0.11.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5beb56d22b63647bafd0b74979216fdee80c580c0c63410be8c11053860ffd09", size = 486367, upload-time = "2025-09-15T09:19:14.546Z" }, - { url = "https://files.pythonhosted.org/packages/fb/f1/98cb3a36f5e62f80cd860f0179f948d9eab5a316d55d3e1bab98d9767af5/jiter-0.11.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:97025d09ef549795d8dc720a824312cee3253c890ac73c621721ddfc75066789", size = 376335, upload-time = "2025-09-15T09:19:15.939Z" }, - { url = "https://files.pythonhosted.org/packages/9f/d8/ec74886497ea393c29dbd7651ddecc1899e86404a6b1f84a3ddab0ab59fd/jiter-0.11.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d50880a6da65d8c23a2cf53c412847d9757e74cc9a3b95c5704a1d1a24667347", size = 348981, upload-time = "2025-09-15T09:19:17.568Z" }, - { url = "https://files.pythonhosted.org/packages/24/93/d22ad7fa3b86ade66c86153ceea73094fc2af8b20c59cb7fceab9fea4704/jiter-0.11.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:452d80a1c86c095a242007bd9fc5d21b8a8442307193378f891cb8727e469648", size = 385797, upload-time = "2025-09-15T09:19:19.121Z" }, - { url = "https://files.pythonhosted.org/packages/c8/bd/e25ff4a4df226e9b885f7cb01ee4b9dc74e3000e612d6f723860d71a1f34/jiter-0.11.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e84e58198d4894668eec2da660ffff60e0f3e60afa790ecc50cb12b0e02ca1d4", size = 516597, upload-time = "2025-09-15T09:19:20.301Z" }, - { url = "https://files.pythonhosted.org/packages/be/fb/beda613db7d93ffa2fdd2683f90f2f5dce8daf4bc2d0d2829e7de35308c6/jiter-0.11.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:df64edcfc5dd5279a791eea52aa113d432c933119a025b0b5739f90d2e4e75f1", size = 508853, upload-time = "2025-09-15T09:19:22.075Z" }, - { url = "https://files.pythonhosted.org/packages/20/64/c5b0d93490634e41e38e2a15de5d54fdbd2c9f64a19abb0f95305b63373c/jiter-0.11.0-cp311-cp311-win32.whl", hash = "sha256:144fc21337d21b1d048f7f44bf70881e1586401d405ed3a98c95a114a9994982", size = 205140, upload-time = "2025-09-15T09:19:23.351Z" }, - { url = "https://files.pythonhosted.org/packages/a1/e6/c347c0e6f5796e97d4356b7e5ff0ce336498b7f4ef848fae621a56f1ccf3/jiter-0.11.0-cp311-cp311-win_amd64.whl", hash = "sha256:b0f32e644d241293b892b1a6dd8f0b9cc029bfd94c97376b2681c36548aabab7", size = 204311, upload-time = "2025-09-15T09:19:24.591Z" }, - { url = "https://files.pythonhosted.org/packages/ba/b5/3009b112b8f673e568ef79af9863d8309a15f0a8cdcc06ed6092051f377e/jiter-0.11.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:2fb7b377688cc3850bbe5c192a6bd493562a0bc50cbc8b047316428fbae00ada", size = 305510, upload-time = "2025-09-15T09:19:25.893Z" }, - { url = "https://files.pythonhosted.org/packages/fe/82/15514244e03b9e71e086bbe2a6de3e4616b48f07d5f834200c873956fb8c/jiter-0.11.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:a1b7cbe3f25bd0d8abb468ba4302a5d45617ee61b2a7a638f63fee1dc086be99", size = 316521, upload-time = "2025-09-15T09:19:27.525Z" }, - { url = "https://files.pythonhosted.org/packages/92/94/7a2e905f40ad2d6d660e00b68d818f9e29fb87ffe82774f06191e93cbe4a/jiter-0.11.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c0a7f0ec81d5b7588c5cade1eb1925b91436ae6726dc2df2348524aeabad5de6", size = 338214, upload-time = "2025-09-15T09:19:28.727Z" }, - { url = "https://files.pythonhosted.org/packages/a8/9c/5791ed5bdc76f12110158d3316a7a3ec0b1413d018b41c5ed399549d3ad5/jiter-0.11.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:07630bb46ea2a6b9c6ed986c6e17e35b26148cce2c535454b26ee3f0e8dcaba1", size = 361280, upload-time = "2025-09-15T09:19:30.013Z" }, - { url = "https://files.pythonhosted.org/packages/d4/7f/b7d82d77ff0d2cb06424141000176b53a9e6b16a1125525bb51ea4990c2e/jiter-0.11.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7764f27d28cd4a9cbc61704dfcd80c903ce3aad106a37902d3270cd6673d17f4", size = 487895, upload-time = "2025-09-15T09:19:31.424Z" }, - { url = "https://files.pythonhosted.org/packages/42/44/10a1475d46f1fc1fd5cc2e82c58e7bca0ce5852208e0fa5df2f949353321/jiter-0.11.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1d4a6c4a737d486f77f842aeb22807edecb4a9417e6700c7b981e16d34ba7c72", size = 378421, upload-time = "2025-09-15T09:19:32.746Z" }, - { url = "https://files.pythonhosted.org/packages/9a/5f/0dc34563d8164d31d07bc09d141d3da08157a68dcd1f9b886fa4e917805b/jiter-0.11.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cf408d2a0abd919b60de8c2e7bc5eeab72d4dafd18784152acc7c9adc3291591", size = 347932, upload-time = "2025-09-15T09:19:34.612Z" }, - { url = "https://files.pythonhosted.org/packages/f7/de/b68f32a4fcb7b4a682b37c73a0e5dae32180140cd1caf11aef6ad40ddbf2/jiter-0.11.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:cdef53eda7d18e799625023e1e250dbc18fbc275153039b873ec74d7e8883e09", size = 386959, upload-time = "2025-09-15T09:19:35.994Z" }, - { url = "https://files.pythonhosted.org/packages/76/0a/c08c92e713b6e28972a846a81ce374883dac2f78ec6f39a0dad9f2339c3a/jiter-0.11.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:53933a38ef7b551dd9c7f1064f9d7bb235bb3168d0fa5f14f0798d1b7ea0d9c5", size = 517187, upload-time = "2025-09-15T09:19:37.426Z" }, - { url = "https://files.pythonhosted.org/packages/89/b5/4a283bec43b15aad54fcae18d951f06a2ec3f78db5708d3b59a48e9c3fbd/jiter-0.11.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:11840d2324c9ab5162fc1abba23bc922124fedcff0d7b7f85fffa291e2f69206", size = 509461, upload-time = "2025-09-15T09:19:38.761Z" }, - { url = "https://files.pythonhosted.org/packages/34/a5/f8bad793010534ea73c985caaeef8cc22dfb1fedb15220ecdf15c623c07a/jiter-0.11.0-cp312-cp312-win32.whl", hash = "sha256:4f01a744d24a5f2bb4a11657a1b27b61dc038ae2e674621a74020406e08f749b", size = 206664, upload-time = "2025-09-15T09:19:40.096Z" }, - { url = "https://files.pythonhosted.org/packages/ed/42/5823ec2b1469395a160b4bf5f14326b4a098f3b6898fbd327366789fa5d3/jiter-0.11.0-cp312-cp312-win_amd64.whl", hash = "sha256:29fff31190ab3a26de026da2f187814f4b9c6695361e20a9ac2123e4d4378a4c", size = 203520, upload-time = "2025-09-15T09:19:41.798Z" }, - { url = "https://files.pythonhosted.org/packages/70/f3/ce100253c80063a7b8b406e1d1562657fd4b9b4e1b562db40e68645342fb/jiter-0.11.0-graalpy311-graalpy242_311_native-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:902b43386c04739229076bd1c4c69de5d115553d982ab442a8ae82947c72ede7", size = 336380, upload-time = "2025-09-15T09:20:36.867Z" }, + { url = "https://files.pythonhosted.org/packages/32/f9/eaca4633486b527ebe7e681c431f529b63fe2709e7c5242fc0f43f77ce63/jiter-0.12.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:d8f8a7e317190b2c2d60eb2e8aa835270b008139562d70fe732e1c0020ec53c9", size = 316435, upload-time = "2025-11-09T20:47:02.087Z" }, + { url = "https://files.pythonhosted.org/packages/10/c1/40c9f7c22f5e6ff715f28113ebaba27ab85f9af2660ad6e1dd6425d14c19/jiter-0.12.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:2218228a077e784c6c8f1a8e5d6b8cb1dea62ce25811c356364848554b2056cd", size = 320548, upload-time = "2025-11-09T20:47:03.409Z" }, + { url = "https://files.pythonhosted.org/packages/6b/1b/efbb68fe87e7711b00d2cfd1f26bb4bfc25a10539aefeaa7727329ffb9cb/jiter-0.12.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9354ccaa2982bf2188fd5f57f79f800ef622ec67beb8329903abf6b10da7d423", size = 351915, upload-time = "2025-11-09T20:47:05.171Z" }, + { url = "https://files.pythonhosted.org/packages/15/2d/c06e659888c128ad1e838123d0638f0efad90cc30860cb5f74dd3f2fc0b3/jiter-0.12.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:8f2607185ea89b4af9a604d4c7ec40e45d3ad03ee66998b031134bc510232bb7", size = 368966, upload-time = "2025-11-09T20:47:06.508Z" }, + { url = "https://files.pythonhosted.org/packages/6b/20/058db4ae5fb07cf6a4ab2e9b9294416f606d8e467fb74c2184b2a1eeacba/jiter-0.12.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3a585a5e42d25f2e71db5f10b171f5e5ea641d3aa44f7df745aa965606111cc2", size = 482047, upload-time = "2025-11-09T20:47:08.382Z" }, + { url = "https://files.pythonhosted.org/packages/49/bb/dc2b1c122275e1de2eb12905015d61e8316b2f888bdaac34221c301495d6/jiter-0.12.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bd9e21d34edff5a663c631f850edcb786719c960ce887a5661e9c828a53a95d9", size = 380835, upload-time = "2025-11-09T20:47:09.81Z" }, + { url = "https://files.pythonhosted.org/packages/23/7d/38f9cd337575349de16da575ee57ddb2d5a64d425c9367f5ef9e4612e32e/jiter-0.12.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4a612534770470686cd5431478dc5a1b660eceb410abade6b1b74e320ca98de6", size = 364587, upload-time = "2025-11-09T20:47:11.529Z" }, + { url = "https://files.pythonhosted.org/packages/f0/a3/b13e8e61e70f0bb06085099c4e2462647f53cc2ca97614f7fedcaa2bb9f3/jiter-0.12.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:3985aea37d40a908f887b34d05111e0aae822943796ebf8338877fee2ab67725", size = 390492, upload-time = "2025-11-09T20:47:12.993Z" }, + { url = "https://files.pythonhosted.org/packages/07/71/e0d11422ed027e21422f7bc1883c61deba2d9752b720538430c1deadfbca/jiter-0.12.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:b1207af186495f48f72529f8d86671903c8c10127cac6381b11dddc4aaa52df6", size = 522046, upload-time = "2025-11-09T20:47:14.6Z" }, + { url = "https://files.pythonhosted.org/packages/9f/59/b968a9aa7102a8375dbbdfbd2aeebe563c7e5dddf0f47c9ef1588a97e224/jiter-0.12.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:ef2fb241de583934c9915a33120ecc06d94aa3381a134570f59eed784e87001e", size = 513392, upload-time = "2025-11-09T20:47:16.011Z" }, + { url = "https://files.pythonhosted.org/packages/ca/e4/7df62002499080dbd61b505c5cb351aa09e9959d176cac2aa8da6f93b13b/jiter-0.12.0-cp311-cp311-win32.whl", hash = "sha256:453b6035672fecce8007465896a25b28a6b59cfe8fbc974b2563a92f5a92a67c", size = 206096, upload-time = "2025-11-09T20:47:17.344Z" }, + { url = "https://files.pythonhosted.org/packages/bb/60/1032b30ae0572196b0de0e87dce3b6c26a1eff71aad5fe43dee3082d32e0/jiter-0.12.0-cp311-cp311-win_amd64.whl", hash = "sha256:ca264b9603973c2ad9435c71a8ec8b49f8f715ab5ba421c85a51cde9887e421f", size = 204899, upload-time = "2025-11-09T20:47:19.365Z" }, + { url = "https://files.pythonhosted.org/packages/49/d5/c145e526fccdb834063fb45c071df78b0cc426bbaf6de38b0781f45d956f/jiter-0.12.0-cp311-cp311-win_arm64.whl", hash = "sha256:cb00ef392e7d684f2754598c02c409f376ddcef857aae796d559e6cacc2d78a5", size = 188070, upload-time = "2025-11-09T20:47:20.75Z" }, + { url = "https://files.pythonhosted.org/packages/92/c9/5b9f7b4983f1b542c64e84165075335e8a236fa9e2ea03a0c79780062be8/jiter-0.12.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:305e061fa82f4680607a775b2e8e0bcb071cd2205ac38e6ef48c8dd5ebe1cf37", size = 314449, upload-time = "2025-11-09T20:47:22.999Z" }, + { url = "https://files.pythonhosted.org/packages/98/6e/e8efa0e78de00db0aee82c0cf9e8b3f2027efd7f8a71f859d8f4be8e98ef/jiter-0.12.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:5c1860627048e302a528333c9307c818c547f214d8659b0705d2195e1a94b274", size = 319855, upload-time = "2025-11-09T20:47:24.779Z" }, + { url = "https://files.pythonhosted.org/packages/20/26/894cd88e60b5d58af53bec5c6759d1292bd0b37a8b5f60f07abf7a63ae5f/jiter-0.12.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:df37577a4f8408f7e0ec3205d2a8f87672af8f17008358063a4d6425b6081ce3", size = 350171, upload-time = "2025-11-09T20:47:26.469Z" }, + { url = "https://files.pythonhosted.org/packages/f5/27/a7b818b9979ac31b3763d25f3653ec3a954044d5e9f5d87f2f247d679fd1/jiter-0.12.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:75fdd787356c1c13a4f40b43c2156276ef7a71eb487d98472476476d803fb2cf", size = 365590, upload-time = "2025-11-09T20:47:27.918Z" }, + { url = "https://files.pythonhosted.org/packages/ba/7e/e46195801a97673a83746170b17984aa8ac4a455746354516d02ca5541b4/jiter-0.12.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1eb5db8d9c65b112aacf14fcd0faae9913d07a8afea5ed06ccdd12b724e966a1", size = 479462, upload-time = "2025-11-09T20:47:29.654Z" }, + { url = "https://files.pythonhosted.org/packages/ca/75/f833bfb009ab4bd11b1c9406d333e3b4357709ed0570bb48c7c06d78c7dd/jiter-0.12.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:73c568cc27c473f82480abc15d1301adf333a7ea4f2e813d6a2c7d8b6ba8d0df", size = 378983, upload-time = "2025-11-09T20:47:31.026Z" }, + { url = "https://files.pythonhosted.org/packages/71/b3/7a69d77943cc837d30165643db753471aff5df39692d598da880a6e51c24/jiter-0.12.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4321e8a3d868919bcb1abb1db550d41f2b5b326f72df29e53b2df8b006eb9403", size = 361328, upload-time = "2025-11-09T20:47:33.286Z" }, + { url = "https://files.pythonhosted.org/packages/b0/ac/a78f90caf48d65ba70d8c6efc6f23150bc39dc3389d65bbec2a95c7bc628/jiter-0.12.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:0a51bad79f8cc9cac2b4b705039f814049142e0050f30d91695a2d9a6611f126", size = 386740, upload-time = "2025-11-09T20:47:34.703Z" }, + { url = "https://files.pythonhosted.org/packages/39/b6/5d31c2cc8e1b6a6bcf3c5721e4ca0a3633d1ab4754b09bc7084f6c4f5327/jiter-0.12.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:2a67b678f6a5f1dd6c36d642d7db83e456bc8b104788262aaefc11a22339f5a9", size = 520875, upload-time = "2025-11-09T20:47:36.058Z" }, + { url = "https://files.pythonhosted.org/packages/30/b5/4df540fae4e9f68c54b8dab004bd8c943a752f0b00efd6e7d64aa3850339/jiter-0.12.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:efe1a211fe1fd14762adea941e3cfd6c611a136e28da6c39272dbb7a1bbe6a86", size = 511457, upload-time = "2025-11-09T20:47:37.932Z" }, + { url = "https://files.pythonhosted.org/packages/07/65/86b74010e450a1a77b2c1aabb91d4a91dd3cd5afce99f34d75fd1ac64b19/jiter-0.12.0-cp312-cp312-win32.whl", hash = "sha256:d779d97c834b4278276ec703dc3fc1735fca50af63eb7262f05bdb4e62203d44", size = 204546, upload-time = "2025-11-09T20:47:40.47Z" }, + { url = "https://files.pythonhosted.org/packages/1c/c7/6659f537f9562d963488e3e55573498a442503ced01f7e169e96a6110383/jiter-0.12.0-cp312-cp312-win_amd64.whl", hash = "sha256:e8269062060212b373316fe69236096aaf4c49022d267c6736eebd66bbbc60bb", size = 205196, upload-time = "2025-11-09T20:47:41.794Z" }, + { url = "https://files.pythonhosted.org/packages/21/f4/935304f5169edadfec7f9c01eacbce4c90bb9a82035ac1de1f3bd2d40be6/jiter-0.12.0-cp312-cp312-win_arm64.whl", hash = "sha256:06cb970936c65de926d648af0ed3d21857f026b1cf5525cb2947aa5e01e05789", size = 186100, upload-time = "2025-11-09T20:47:43.007Z" }, + { url = "https://files.pythonhosted.org/packages/fe/54/5339ef1ecaa881c6948669956567a64d2670941925f245c434f494ffb0e5/jiter-0.12.0-graalpy311-graalpy242_311_native-macosx_10_12_x86_64.whl", hash = "sha256:4739a4657179ebf08f85914ce50332495811004cc1747852e8b2041ed2aab9b8", size = 311144, upload-time = "2025-11-09T20:49:10.503Z" }, + { url = "https://files.pythonhosted.org/packages/27/74/3446c652bffbd5e81ab354e388b1b5fc1d20daac34ee0ed11ff096b1b01a/jiter-0.12.0-graalpy311-graalpy242_311_native-macosx_11_0_arm64.whl", hash = "sha256:41da8def934bf7bec16cb24bd33c0ca62126d2d45d81d17b864bd5ad721393c3", size = 305877, upload-time = "2025-11-09T20:49:12.269Z" }, + { url = "https://files.pythonhosted.org/packages/a1/f4/ed76ef9043450f57aac2d4fbeb27175aa0eb9c38f833be6ef6379b3b9a86/jiter-0.12.0-graalpy311-graalpy242_311_native-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9c44ee814f499c082e69872d426b624987dbc5943ab06e9bbaa4f81989fdb79e", size = 340419, upload-time = "2025-11-09T20:49:13.803Z" }, + { url = "https://files.pythonhosted.org/packages/21/01/857d4608f5edb0664aa791a3d45702e1a5bcfff9934da74035e7b9803846/jiter-0.12.0-graalpy311-graalpy242_311_native-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cd2097de91cf03eaa27b3cbdb969addf83f0179c6afc41bbc4513705e013c65d", size = 347212, upload-time = "2025-11-09T20:49:15.643Z" }, + { url = "https://files.pythonhosted.org/packages/cb/f5/12efb8ada5f5c9edc1d4555fe383c1fb2eac05ac5859258a72d61981d999/jiter-0.12.0-graalpy312-graalpy250_312_native-macosx_10_12_x86_64.whl", hash = "sha256:e8547883d7b96ef2e5fe22b88f8a4c8725a56e7f4abafff20fd5272d634c7ecb", size = 309974, upload-time = "2025-11-09T20:49:17.187Z" }, + { url = "https://files.pythonhosted.org/packages/85/15/d6eb3b770f6a0d332675141ab3962fd4a7c270ede3515d9f3583e1d28276/jiter-0.12.0-graalpy312-graalpy250_312_native-macosx_11_0_arm64.whl", hash = "sha256:89163163c0934854a668ed783a2546a0617f71706a2551a4a0666d91ab365d6b", size = 304233, upload-time = "2025-11-09T20:49:18.734Z" }, + { url = "https://files.pythonhosted.org/packages/8c/3e/e7e06743294eea2cf02ced6aa0ff2ad237367394e37a0e2b4a1108c67a36/jiter-0.12.0-graalpy312-graalpy250_312_native-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d96b264ab7d34bbb2312dedc47ce07cd53f06835eacbc16dde3761f47c3a9e7f", size = 338537, upload-time = "2025-11-09T20:49:20.317Z" }, + { url = "https://files.pythonhosted.org/packages/2f/9c/6753e6522b8d0ef07d3a3d239426669e984fb0eba15a315cdbc1253904e4/jiter-0.12.0-graalpy312-graalpy250_312_native-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c24e864cb30ab82311c6425655b0cdab0a98c5d973b065c66a3f020740c2324c", size = 346110, upload-time = "2025-11-09T20:49:21.817Z" }, ] [[package]] @@ -2960,11 +2996,11 @@ wheels = [ [[package]] name = "json-repair" -version = "0.52.0" +version = "0.53.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/f3/63/2c3c3c8cc1c28a0a20a9ab0eff5439c989ce3cc5956d8a4c7cf1eae0a06e/json_repair-0.52.0.tar.gz", hash = "sha256:0eee59cb3145b462b0734d4cf3246b797686caa669d52eee8dd30e09ea6d7876", size = 35384, upload-time = "2025-10-05T17:18:12.387Z" } +sdist = { url = "https://files.pythonhosted.org/packages/69/9c/be1d84106529aeacbe6151c1e1dc202f5a5cfa0d9bac748d4a1039ebb913/json_repair-0.53.0.tar.gz", hash = "sha256:97fcbf1eea0bbcf6d5cc94befc573623ab4bbba6abdc394cfd3b933a2571266d", size = 36204, upload-time = "2025-11-08T13:45:15.807Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/c6/7f/3a4e456da9a0f9ac54d9842ed51e96960826a98456f0826a9b3e808713c4/json_repair-0.52.0-py3-none-any.whl", hash = "sha256:c783069906a456f62e2a553fbef32a420a4745ff943e2014411728edcc7bf60a", size = 26350, upload-time = "2025-10-05T17:18:10.859Z" }, + { url = "https://files.pythonhosted.org/packages/ba/49/e588ec59b64222c8d38585f9ceffbf71870c3cbfb2873e53297c4f4afd0b/json_repair-0.53.0-py3-none-any.whl", hash = "sha256:17f7439e41ae39964e1d678b1def38cb8ec43d607340564acf3e62d8ce47a727", size = 27404, upload-time = "2025-11-08T13:45:14.464Z" }, ] [[package]] @@ -3084,7 +3120,7 @@ wheels = [ [[package]] name = "litellm" -version = "1.77.4" +version = "1.77.1" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "aiohttp" }, @@ -3095,15 +3131,14 @@ dependencies = [ { name = "jinja2" }, { name = "jsonschema" }, { name = "openai" }, - { name = "pondpond" }, { name = "pydantic" }, { name = "python-dotenv" }, { name = "tiktoken" }, { name = "tokenizers" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/ab/b7/0d3c6dbcff3064238d123f90ae96764a85352f3f5caab6695a55007fd019/litellm-1.77.4.tar.gz", hash = "sha256:ce652e10ecf5b36767bfdf58e53b2802e22c3de383b03554e6ee1a4a66fa743d", size = 10330773, upload-time = "2025-09-24T17:52:44.876Z" } +sdist = { url = "https://files.pythonhosted.org/packages/8c/65/71fe4851709fa4a612e41b80001a9ad803fea979d21b90970093fd65eded/litellm-1.77.1.tar.gz", hash = "sha256:76bab5203115efb9588244e5bafbfc07a800a239be75d8dc6b1b9d17394c6418", size = 10275745, upload-time = "2025-09-13T21:05:21.377Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/3c/32/90f8587818d146d604ed6eec95f96378363fda06b14817399cc68853383e/litellm-1.77.4-py3-none-any.whl", hash = "sha256:66c2bb776f1e19ceddfa977a2bbf7f05e6f26c4b1fec8b2093bd171d842701b8", size = 9138493, upload-time = "2025-09-24T17:52:40.764Z" }, + { url = "https://files.pythonhosted.org/packages/bb/dc/ff4f119cd4d783742c9648a03e0ba5c2b52fc385b2ae9f0d32acf3a78241/litellm-1.77.1-py3-none-any.whl", hash = "sha256:407761dc3c35fbcd41462d3fe65dd3ed70aac705f37cde318006c18940f695a0", size = 9067070, upload-time = "2025-09-13T21:05:18.078Z" }, ] [[package]] @@ -3183,34 +3218,28 @@ wheels = [ [[package]] name = "lz4" -version = "4.4.4" +version = "4.4.5" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/c6/5a/945f5086326d569f14c84ac6f7fcc3229f0b9b1e8cc536b951fd53dfb9e1/lz4-4.4.4.tar.gz", hash = "sha256:070fd0627ec4393011251a094e08ed9fdcc78cb4e7ab28f507638eee4e39abda", size = 171884, upload-time = "2025-04-01T22:55:58.62Z" } +sdist = { url = "https://files.pythonhosted.org/packages/57/51/f1b86d93029f418033dddf9b9f79c8d2641e7454080478ee2aab5123173e/lz4-4.4.5.tar.gz", hash = "sha256:5f0b9e53c1e82e88c10d7c180069363980136b9d7a8306c4dca4f760d60c39f0", size = 172886, upload-time = "2025-11-03T13:02:36.061Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/28/e8/63843dc5ecb1529eb38e1761ceed04a0ad52a9ad8929ab8b7930ea2e4976/lz4-4.4.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ddfc7194cd206496c445e9e5b0c47f970ce982c725c87bd22de028884125b68f", size = 220898, upload-time = "2025-04-01T22:55:23.085Z" }, - { url = "https://files.pythonhosted.org/packages/e4/94/c53de5f07c7dc11cf459aab2a1d754f5df5f693bfacbbe1e4914bfd02f1e/lz4-4.4.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:714f9298c86f8e7278f1c6af23e509044782fa8220eb0260f8f8f1632f820550", size = 189685, upload-time = "2025-04-01T22:55:24.413Z" }, - { url = "https://files.pythonhosted.org/packages/fe/59/c22d516dd0352f2a3415d1f665ccef2f3e74ecec3ca6a8f061a38f97d50d/lz4-4.4.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a8474c91de47733856c6686df3c4aca33753741da7e757979369c2c0d32918ba", size = 1239225, upload-time = "2025-04-01T22:55:25.737Z" }, - { url = "https://files.pythonhosted.org/packages/81/af/665685072e71f3f0e626221b7922867ec249cd8376aca761078c8f11f5da/lz4-4.4.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:80dd27d7d680ea02c261c226acf1d41de2fd77af4fb2da62b278a9376e380de0", size = 1265881, upload-time = "2025-04-01T22:55:26.817Z" }, - { url = "https://files.pythonhosted.org/packages/90/04/b4557ae381d3aa451388a29755cc410066f5e2f78c847f66f154f4520a68/lz4-4.4.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9b7d6dddfd01b49aedb940fdcaf32f41dc58c926ba35f4e31866aeec2f32f4f4", size = 1185593, upload-time = "2025-04-01T22:55:27.896Z" }, - { url = "https://files.pythonhosted.org/packages/7b/e4/03636979f4e8bf92c557f998ca98ee4e6ef92e92eaf0ed6d3c7f2524e790/lz4-4.4.4-cp311-cp311-win32.whl", hash = "sha256:4134b9fd70ac41954c080b772816bb1afe0c8354ee993015a83430031d686a4c", size = 88259, upload-time = "2025-04-01T22:55:29.03Z" }, - { url = "https://files.pythonhosted.org/packages/07/f0/9efe53b4945441a5d2790d455134843ad86739855b7e6199977bf6dc8898/lz4-4.4.4-cp311-cp311-win_amd64.whl", hash = "sha256:f5024d3ca2383470f7c4ef4d0ed8eabad0b22b23eeefde1c192cf1a38d5e9f78", size = 99916, upload-time = "2025-04-01T22:55:29.933Z" }, - { url = "https://files.pythonhosted.org/packages/87/c8/1675527549ee174b9e1db089f7ddfbb962a97314657269b1e0344a5eaf56/lz4-4.4.4-cp311-cp311-win_arm64.whl", hash = "sha256:6ea715bb3357ea1665f77874cf8f55385ff112553db06f3742d3cdcec08633f7", size = 89741, upload-time = "2025-04-01T22:55:31.184Z" }, - { url = "https://files.pythonhosted.org/packages/f7/2d/5523b4fabe11cd98f040f715728d1932eb7e696bfe94391872a823332b94/lz4-4.4.4-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:23ae267494fdd80f0d2a131beff890cf857f1b812ee72dbb96c3204aab725553", size = 220669, upload-time = "2025-04-01T22:55:32.032Z" }, - { url = "https://files.pythonhosted.org/packages/91/06/1a5bbcacbfb48d8ee5b6eb3fca6aa84143a81d92946bdb5cd6b005f1863e/lz4-4.4.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:fff9f3a1ed63d45cb6514bfb8293005dc4141341ce3500abdfeb76124c0b9b2e", size = 189661, upload-time = "2025-04-01T22:55:33.413Z" }, - { url = "https://files.pythonhosted.org/packages/fa/08/39eb7ac907f73e11a69a11576a75a9e36406b3241c0ba41453a7eb842abb/lz4-4.4.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1ea7f07329f85a8eda4d8cf937b87f27f0ac392c6400f18bea2c667c8b7f8ecc", size = 1238775, upload-time = "2025-04-01T22:55:34.835Z" }, - { url = "https://files.pythonhosted.org/packages/e9/26/05840fbd4233e8d23e88411a066ab19f1e9de332edddb8df2b6a95c7fddc/lz4-4.4.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8ccab8f7f7b82f9fa9fc3b0ba584d353bd5aa818d5821d77d5b9447faad2aaad", size = 1265143, upload-time = "2025-04-01T22:55:35.933Z" }, - { url = "https://files.pythonhosted.org/packages/b7/5d/5f2db18c298a419932f3ab2023deb689863cf8fd7ed875b1c43492479af2/lz4-4.4.4-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e43e9d48b2daf80e486213128b0763deed35bbb7a59b66d1681e205e1702d735", size = 1185032, upload-time = "2025-04-01T22:55:37.454Z" }, - { url = "https://files.pythonhosted.org/packages/c4/e6/736ab5f128694b0f6aac58343bcf37163437ac95997276cd0be3ea4c3342/lz4-4.4.4-cp312-cp312-win32.whl", hash = "sha256:33e01e18e4561b0381b2c33d58e77ceee850a5067f0ece945064cbaac2176962", size = 88284, upload-time = "2025-04-01T22:55:38.536Z" }, - { url = "https://files.pythonhosted.org/packages/40/b8/243430cb62319175070e06e3a94c4c7bd186a812e474e22148ae1290d47d/lz4-4.4.4-cp312-cp312-win_amd64.whl", hash = "sha256:d21d1a2892a2dcc193163dd13eaadabb2c1b803807a5117d8f8588b22eaf9f12", size = 99918, upload-time = "2025-04-01T22:55:39.628Z" }, - { url = "https://files.pythonhosted.org/packages/6c/e1/0686c91738f3e6c2e1a243e0fdd4371667c4d2e5009b0a3605806c2aa020/lz4-4.4.4-cp312-cp312-win_arm64.whl", hash = "sha256:2f4f2965c98ab254feddf6b5072854a6935adab7bc81412ec4fe238f07b85f62", size = 89736, upload-time = "2025-04-01T22:55:40.5Z" }, + { url = "https://files.pythonhosted.org/packages/93/5b/6edcd23319d9e28b1bedf32768c3d1fd56eed8223960a2c47dacd2cec2af/lz4-4.4.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:d6da84a26b3aa5da13a62e4b89ab36a396e9327de8cd48b436a3467077f8ccd4", size = 207391, upload-time = "2025-11-03T13:01:36.644Z" }, + { url = "https://files.pythonhosted.org/packages/34/36/5f9b772e85b3d5769367a79973b8030afad0d6b724444083bad09becd66f/lz4-4.4.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:61d0ee03e6c616f4a8b69987d03d514e8896c8b1b7cc7598ad029e5c6aedfd43", size = 207146, upload-time = "2025-11-03T13:01:37.928Z" }, + { url = "https://files.pythonhosted.org/packages/04/f4/f66da5647c0d72592081a37c8775feacc3d14d2625bbdaabd6307c274565/lz4-4.4.5-cp311-cp311-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:33dd86cea8375d8e5dd001e41f321d0a4b1eb7985f39be1b6a4f466cd480b8a7", size = 1292623, upload-time = "2025-11-03T13:01:39.341Z" }, + { url = "https://files.pythonhosted.org/packages/85/fc/5df0f17467cdda0cad464a9197a447027879197761b55faad7ca29c29a04/lz4-4.4.5-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:609a69c68e7cfcfa9d894dc06be13f2e00761485b62df4e2472f1b66f7b405fb", size = 1279982, upload-time = "2025-11-03T13:01:40.816Z" }, + { url = "https://files.pythonhosted.org/packages/25/3b/b55cb577aa148ed4e383e9700c36f70b651cd434e1c07568f0a86c9d5fbb/lz4-4.4.5-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:75419bb1a559af00250b8f1360d508444e80ed4b26d9d40ec5b09fe7875cb989", size = 1368674, upload-time = "2025-11-03T13:01:42.118Z" }, + { url = "https://files.pythonhosted.org/packages/fb/31/e97e8c74c59ea479598e5c55cbe0b1334f03ee74ca97726e872944ed42df/lz4-4.4.5-cp311-cp311-win32.whl", hash = "sha256:12233624f1bc2cebc414f9efb3113a03e89acce3ab6f72035577bc61b270d24d", size = 88168, upload-time = "2025-11-03T13:01:43.282Z" }, + { url = "https://files.pythonhosted.org/packages/18/47/715865a6c7071f417bef9b57c8644f29cb7a55b77742bd5d93a609274e7e/lz4-4.4.5-cp311-cp311-win_amd64.whl", hash = "sha256:8a842ead8ca7c0ee2f396ca5d878c4c40439a527ebad2b996b0444f0074ed004", size = 99491, upload-time = "2025-11-03T13:01:44.167Z" }, + { url = "https://files.pythonhosted.org/packages/14/e7/ac120c2ca8caec5c945e6356ada2aa5cfabd83a01e3170f264a5c42c8231/lz4-4.4.5-cp311-cp311-win_arm64.whl", hash = "sha256:83bc23ef65b6ae44f3287c38cbf82c269e2e96a26e560aa551735883388dcc4b", size = 91271, upload-time = "2025-11-03T13:01:45.016Z" }, + { url = "https://files.pythonhosted.org/packages/1b/ac/016e4f6de37d806f7cc8f13add0a46c9a7cfc41a5ddc2bc831d7954cf1ce/lz4-4.4.5-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:df5aa4cead2044bab83e0ebae56e0944cc7fcc1505c7787e9e1057d6d549897e", size = 207163, upload-time = "2025-11-03T13:01:45.895Z" }, + { url = "https://files.pythonhosted.org/packages/8d/df/0fadac6e5bd31b6f34a1a8dbd4db6a7606e70715387c27368586455b7fc9/lz4-4.4.5-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:6d0bf51e7745484d2092b3a51ae6eb58c3bd3ce0300cf2b2c14f76c536d5697a", size = 207150, upload-time = "2025-11-03T13:01:47.205Z" }, + { url = "https://files.pythonhosted.org/packages/b7/17/34e36cc49bb16ca73fb57fbd4c5eaa61760c6b64bce91fcb4e0f4a97f852/lz4-4.4.5-cp312-cp312-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:7b62f94b523c251cf32aa4ab555f14d39bd1a9df385b72443fd76d7c7fb051f5", size = 1292045, upload-time = "2025-11-03T13:01:48.667Z" }, + { url = "https://files.pythonhosted.org/packages/90/1c/b1d8e3741e9fc89ed3b5f7ef5f22586c07ed6bb04e8343c2e98f0fa7ff04/lz4-4.4.5-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:2c3ea562c3af274264444819ae9b14dbbf1ab070aff214a05e97db6896c7597e", size = 1279546, upload-time = "2025-11-03T13:01:50.159Z" }, + { url = "https://files.pythonhosted.org/packages/55/d9/e3867222474f6c1b76e89f3bd914595af69f55bf2c1866e984c548afdc15/lz4-4.4.5-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:24092635f47538b392c4eaeff14c7270d2c8e806bf4be2a6446a378591c5e69e", size = 1368249, upload-time = "2025-11-03T13:01:51.273Z" }, + { url = "https://files.pythonhosted.org/packages/b2/e7/d667d337367686311c38b580d1ca3d5a23a6617e129f26becd4f5dc458df/lz4-4.4.5-cp312-cp312-win32.whl", hash = "sha256:214e37cfe270948ea7eb777229e211c601a3e0875541c1035ab408fbceaddf50", size = 88189, upload-time = "2025-11-03T13:01:52.605Z" }, + { url = "https://files.pythonhosted.org/packages/a5/0b/a54cd7406995ab097fceb907c7eb13a6ddd49e0b231e448f1a81a50af65c/lz4-4.4.5-cp312-cp312-win_amd64.whl", hash = "sha256:713a777de88a73425cf08eb11f742cd2c98628e79a8673d6a52e3c5f0c116f33", size = 99497, upload-time = "2025-11-03T13:01:53.477Z" }, + { url = "https://files.pythonhosted.org/packages/6a/7e/dc28a952e4bfa32ca16fa2eb026e7a6ce5d1411fcd5986cd08c74ec187b9/lz4-4.4.5-cp312-cp312-win_arm64.whl", hash = "sha256:a88cbb729cc333334ccfb52f070463c21560fca63afcf636a9f160a55fac3301", size = 91279, upload-time = "2025-11-03T13:01:54.419Z" }, ] -[[package]] -name = "madoka" -version = "0.7.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/da/eb/95288b1c4aa541eb296a6271e3f8c7ece03b78923ac47dbe95d2287d9f5e/madoka-0.7.1.tar.gz", hash = "sha256:e258baa84fc0a3764365993b8bf5e1b065383a6ca8c9f862fb3e3e709843fae7", size = 81413, upload-time = "2019-02-10T18:38:01.382Z" } - [[package]] name = "mako" version = "1.3.10" @@ -3471,14 +3500,14 @@ wheels = [ [[package]] name = "mypy-boto3-bedrock-runtime" -version = "1.40.41" +version = "1.40.62" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "typing-extensions", marker = "python_full_version < '3.12'" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/c7/38/79989f7bce998776ed1a01c17f3f58e7bc6f5fc2bcbdff929701526fa2f1/mypy_boto3_bedrock_runtime-1.40.41.tar.gz", hash = "sha256:ee9bda6d6d478c8d0995e84e884bdf1798e150d437974ae27c175774a58ffaa5", size = 28333, upload-time = "2025-09-29T19:26:04.804Z" } +sdist = { url = "https://files.pythonhosted.org/packages/51/d0/ca3c58a1284f9142959fb00889322d4889278c2e4b165350d8e294c07d9c/mypy_boto3_bedrock_runtime-1.40.62.tar.gz", hash = "sha256:5505a60e2b5f9c845ee4778366d49c93c3723f6790d0cec116d8fc5f5609d846", size = 28611, upload-time = "2025-10-29T21:43:02.599Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/3d/6c/d3431dadf473bb76aa590b1ed8cc91726a48b029b542eff9d3024f2d70b9/mypy_boto3_bedrock_runtime-1.40.41-py3-none-any.whl", hash = "sha256:d65dff200986ff06c6b3579ddcea102555f2067c8987fca379bf4f9ed8ba3121", size = 34181, upload-time = "2025-09-29T19:26:01.898Z" }, + { url = "https://files.pythonhosted.org/packages/4b/c5/ad62e5f80684ce5fe878d320634189ef29d00ee294cd62a37f3e51719f47/mypy_boto3_bedrock_runtime-1.40.62-py3-none-any.whl", hash = "sha256:e383e70b5dffb0b335b49fc1b2772f0d35118f99994bc7e731445ba0ab237831", size = 34497, upload-time = "2025-10-29T21:43:01.591Z" }, ] [[package]] @@ -3492,30 +3521,21 @@ wheels = [ [[package]] name = "mysql-connector-python" -version = "9.4.0" +version = "9.5.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/02/77/2b45e6460d05b1f1b7a4c8eb79a50440b4417971973bb78c9ef6cad630a6/mysql_connector_python-9.4.0.tar.gz", hash = "sha256:d111360332ae78933daf3d48ff497b70739aa292ab0017791a33e826234e743b", size = 12185532, upload-time = "2025-07-22T08:02:05.788Z" } +sdist = { url = "https://files.pythonhosted.org/packages/39/33/b332b001bc8c5ee09255a0d4b09a254da674450edd6a3e5228b245ca82a0/mysql_connector_python-9.5.0.tar.gz", hash = "sha256:92fb924285a86d8c146ebd63d94f9eaefa548da7813bc46271508fdc6cc1d596", size = 12251077, upload-time = "2025-10-22T09:05:45.423Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/fe/0c/4365a802129be9fa63885533c38be019f1c6b6f5bcf8844ac53902314028/mysql_connector_python-9.4.0-cp311-cp311-macosx_14_0_arm64.whl", hash = "sha256:7df1a8ddd182dd8adc914f6dc902a986787bf9599705c29aca7b2ce84e79d361", size = 17501627, upload-time = "2025-07-22T07:57:45.416Z" }, - { url = "https://files.pythonhosted.org/packages/c0/bf/ca596c00d7a6eaaf8ef2f66c9b23cd312527f483073c43ffac7843049cb4/mysql_connector_python-9.4.0-cp311-cp311-macosx_14_0_x86_64.whl", hash = "sha256:3892f20472e13e63b1fb4983f454771dd29f211b09724e69a9750e299542f2f8", size = 18369494, upload-time = "2025-07-22T07:57:49.714Z" }, - { url = "https://files.pythonhosted.org/packages/25/14/6510a11ed9f80d77f743dc207773092c4ab78d5efa454b39b48480315d85/mysql_connector_python-9.4.0-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:d3e87142103d71c4df647ece30f98e85e826652272ed1c74822b56f6acdc38e7", size = 33516187, upload-time = "2025-07-22T07:57:55.294Z" }, - { url = "https://files.pythonhosted.org/packages/16/a8/4f99d80f1cf77733ce9a44b6adb7f0dd7079e7afa51ca4826515ef0c3e16/mysql_connector_python-9.4.0-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:b27fcd403436fe83bafb2fe7fcb785891e821e639275c4ad3b3bd1e25f533206", size = 33917818, upload-time = "2025-07-22T07:58:00.523Z" }, - { url = "https://files.pythonhosted.org/packages/15/9c/127f974ca9d5ee25373cb5433da06bb1f36e05f2a6b7436da1fe9c6346b0/mysql_connector_python-9.4.0-cp311-cp311-win_amd64.whl", hash = "sha256:fd6ff5afb9c324b0bbeae958c93156cce4168c743bf130faf224d52818d1f0ee", size = 16392378, upload-time = "2025-07-22T07:58:04.669Z" }, - { url = "https://files.pythonhosted.org/packages/03/7c/a543fb17c2dfa6be8548dfdc5879a0c7924cd5d1c79056c48472bb8fe858/mysql_connector_python-9.4.0-cp312-cp312-macosx_14_0_arm64.whl", hash = "sha256:4efa3898a24aba6a4bfdbf7c1f5023c78acca3150d72cc91199cca2ccd22f76f", size = 17503693, upload-time = "2025-07-22T07:58:08.96Z" }, - { url = "https://files.pythonhosted.org/packages/cb/6e/c22fbee05f5cfd6ba76155b6d45f6261d8d4c1e36e23de04e7f25fbd01a4/mysql_connector_python-9.4.0-cp312-cp312-macosx_14_0_x86_64.whl", hash = "sha256:665c13e7402235162e5b7a2bfdee5895192121b64ea455c90a81edac6a48ede5", size = 18371987, upload-time = "2025-07-22T07:58:13.273Z" }, - { url = "https://files.pythonhosted.org/packages/b4/fd/f426f5f35a3d3180c7f84d1f96b4631be2574df94ca1156adab8618b236c/mysql_connector_python-9.4.0-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:815aa6cad0f351c1223ef345781a538f2e5e44ef405fdb3851eb322bd9c4ca2b", size = 33516214, upload-time = "2025-07-22T07:58:18.967Z" }, - { url = "https://files.pythonhosted.org/packages/45/5a/1b053ae80b43cd3ccebc4bb99a98826969b3b0f8adebdcc2530750ad76ed/mysql_connector_python-9.4.0-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:b3436a2c8c0ec7052932213e8d01882e6eb069dbab33402e685409084b133a1c", size = 33918565, upload-time = "2025-07-22T07:58:25.28Z" }, - { url = "https://files.pythonhosted.org/packages/cb/69/36b989de675d98ba8ff7d45c96c30c699865c657046f2e32db14e78f13d9/mysql_connector_python-9.4.0-cp312-cp312-win_amd64.whl", hash = "sha256:57b0c224676946b70548c56798d5023f65afa1ba5b8ac9f04a143d27976c7029", size = 16392563, upload-time = "2025-07-22T07:58:29.623Z" }, - { url = "https://files.pythonhosted.org/packages/36/34/b6165e15fd45a8deb00932d8e7d823de7650270873b4044c4db6688e1d8f/mysql_connector_python-9.4.0-py2.py3-none-any.whl", hash = "sha256:56e679169c704dab279b176fab2a9ee32d2c632a866c0f7cd48a8a1e2cf802c4", size = 406574, upload-time = "2025-07-22T07:59:08.394Z" }, -] - -[[package]] -name = "nest-asyncio" -version = "1.6.0" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/83/f8/51569ac65d696c8ecbee95938f89d4abf00f47d58d48f6fbabfe8f0baefe/nest_asyncio-1.6.0.tar.gz", hash = "sha256:6f172d5449aca15afd6c646851f4e31e02c598d553a667e38cafa997cfec55fe", size = 7418, upload-time = "2024-01-21T14:25:19.227Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/a0/c4/c2971a3ba4c6103a3d10c4b0f24f461ddc027f0f09763220cf35ca1401b3/nest_asyncio-1.6.0-py3-none-any.whl", hash = "sha256:87af6efd6b5e897c81050477ef65c62e2b2f35d51703cae01aff2905b1852e1c", size = 5195, upload-time = "2024-01-21T14:25:17.223Z" }, + { url = "https://files.pythonhosted.org/packages/05/03/77347d58b0027ce93a41858477e08422e498c6ebc24348b1f725ed7a67ae/mysql_connector_python-9.5.0-cp311-cp311-macosx_14_0_arm64.whl", hash = "sha256:653e70cd10cf2d18dd828fae58dff5f0f7a5cf7e48e244f2093314dddf84a4b9", size = 17578984, upload-time = "2025-10-22T09:01:41.213Z" }, + { url = "https://files.pythonhosted.org/packages/a5/bb/0f45c7ee55ebc56d6731a593d85c0e7f25f83af90a094efebfd5be9fe010/mysql_connector_python-9.5.0-cp311-cp311-macosx_14_0_x86_64.whl", hash = "sha256:5add93f60b3922be71ea31b89bc8a452b876adbb49262561bd559860dae96b3f", size = 18445067, upload-time = "2025-10-22T09:01:43.215Z" }, + { url = "https://files.pythonhosted.org/packages/1c/ec/054de99d4aa50d851a37edca9039280f7194cc1bfd30aab38f5bd6977ebe/mysql_connector_python-9.5.0-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:20950a5e44896c03e3dc93ceb3a5e9b48c9acae18665ca6e13249b3fe5b96811", size = 33668029, upload-time = "2025-10-22T09:01:45.74Z" }, + { url = "https://files.pythonhosted.org/packages/90/a2/e6095dc3a7ad5c959fe4a65681db63af131f572e57cdffcc7816bc84e3ad/mysql_connector_python-9.5.0-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:7fdd3205b9242c284019310fa84437f3357b13f598e3f9b5d80d337d4a6406b8", size = 34101687, upload-time = "2025-10-22T09:01:48.462Z" }, + { url = "https://files.pythonhosted.org/packages/9c/88/bc13c33fca11acaf808bd1809d8602d78f5bb84f7b1e7b1a288c383a14fd/mysql_connector_python-9.5.0-cp311-cp311-win_amd64.whl", hash = "sha256:c021d8b0830958b28712c70c53b206b4cf4766948dae201ea7ca588a186605e0", size = 16511749, upload-time = "2025-10-22T09:01:51.032Z" }, + { url = "https://files.pythonhosted.org/packages/02/89/167ebee82f4b01ba7339c241c3cc2518886a2be9f871770a1efa81b940a0/mysql_connector_python-9.5.0-cp312-cp312-macosx_14_0_arm64.whl", hash = "sha256:a72c2ef9d50b84f3c567c31b3bf30901af740686baa2a4abead5f202e0b7ea61", size = 17581904, upload-time = "2025-10-22T09:01:53.21Z" }, + { url = "https://files.pythonhosted.org/packages/67/46/630ca969ce10b30fdc605d65dab4a6157556d8cc3b77c724f56c2d83cb79/mysql_connector_python-9.5.0-cp312-cp312-macosx_14_0_x86_64.whl", hash = "sha256:bd9ba5a946cfd3b3b2688a75135357e862834b0321ed936fd968049be290872b", size = 18448195, upload-time = "2025-10-22T09:01:55.378Z" }, + { url = "https://files.pythonhosted.org/packages/f6/87/4c421f41ad169d8c9065ad5c46673c7af889a523e4899c1ac1d6bfd37262/mysql_connector_python-9.5.0-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:5ef7accbdf8b5f6ec60d2a1550654b7e27e63bf6f7b04020d5fb4191fb02bc4d", size = 33668638, upload-time = "2025-10-22T09:01:57.896Z" }, + { url = "https://files.pythonhosted.org/packages/a6/01/67cf210d50bfefbb9224b9a5c465857c1767388dade1004c903c8e22a991/mysql_connector_python-9.5.0-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:a6e0a4a0274d15e3d4c892ab93f58f46431222117dba20608178dfb2cc4d5fd8", size = 34102899, upload-time = "2025-10-22T09:02:00.291Z" }, + { url = "https://files.pythonhosted.org/packages/cd/ef/3d1a67d503fff38cc30e11d111cf28f0976987fb175f47b10d44494e1080/mysql_connector_python-9.5.0-cp312-cp312-win_amd64.whl", hash = "sha256:b6c69cb37600b7e22f476150034e2afbd53342a175e20aea887f8158fc5e3ff6", size = 16512684, upload-time = "2025-10-22T09:02:02.411Z" }, + { url = "https://files.pythonhosted.org/packages/95/e1/45373c06781340c7b74fe9b88b85278ac05321889a307eaa5be079a997d4/mysql_connector_python-9.5.0-py2.py3-none-any.whl", hash = "sha256:ace137b88eb6fdafa1e5b2e03ac76ce1b8b1844b3a4af1192a02ae7c1a45bdee", size = 479047, upload-time = "2025-10-22T09:02:27.809Z" }, ] [[package]] @@ -3672,7 +3692,7 @@ wheels = [ [[package]] name = "onnxruntime" -version = "1.23.1" +version = "1.23.2" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "coloredlogs" }, @@ -3683,21 +3703,21 @@ dependencies = [ { name = "sympy" }, ] wheels = [ - { url = "https://files.pythonhosted.org/packages/8a/61/ee52bb2c9402cd1a0d550fc65b826c174f8eed49677dd3833ac1bfc0e35a/onnxruntime-1.23.1-cp311-cp311-macosx_13_0_arm64.whl", hash = "sha256:9ba6e52fb7bc2758a61d1e421d060cf71d5e4259f95ea8a6f72320ae4415f229", size = 17194265, upload-time = "2025-10-08T04:25:24.479Z" }, - { url = "https://files.pythonhosted.org/packages/d3/67/67122b7b4138815090e0d304c8893fefb77370066a847d08e185f04f75fe/onnxruntime-1.23.1-cp311-cp311-macosx_13_0_x86_64.whl", hash = "sha256:7f130f4b0d31ba17c8789053a641958d0d341d96a1bff578d613fb52ded218c2", size = 19150493, upload-time = "2025-10-08T04:24:21.839Z" }, - { url = "https://files.pythonhosted.org/packages/73/e6/66cebc4dcdb217ccb1027cfcbcc01d6399e999c294d986806991c144cbe7/onnxruntime-1.23.1-cp311-cp311-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1b89fd116f20b70e1140a77286954a7715eb9347260ff2008ee7ec94994df039", size = 15216531, upload-time = "2025-10-08T04:24:04.973Z" }, - { url = "https://files.pythonhosted.org/packages/38/47/083847220c4a429e272ce9407bc8c47fa77b62e0c787ef2cc94fe9776c1b/onnxruntime-1.23.1-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:61139a29d536b71db6045c75462e593a53feecc19756dc222531971cd08e5efe", size = 17368047, upload-time = "2025-10-08T04:24:48.426Z" }, - { url = "https://files.pythonhosted.org/packages/ac/8e/b3d861a7d199fd9c6a0b4af9b5d813bcc853d2e4dd4dac2c70b6c23097ed/onnxruntime-1.23.1-cp311-cp311-win_amd64.whl", hash = "sha256:7973186e8eb66e32ea20cb238ae92b604091e4d1df632653ec830abf7584d0b3", size = 13466816, upload-time = "2025-10-08T04:25:15.037Z" }, - { url = "https://files.pythonhosted.org/packages/00/3c/4b4f56b5df4596d1d95aafe13cbc987d050a89364ff5b2f90308376901fb/onnxruntime-1.23.1-cp312-cp312-macosx_13_0_arm64.whl", hash = "sha256:564d6add1688efdb0720cf2158b50314fc35b744ad2623155ee3b805c381d9ce", size = 17194708, upload-time = "2025-10-08T04:25:27.188Z" }, - { url = "https://files.pythonhosted.org/packages/b4/97/05529b97142c1a09bde2caefea4fd29f71329b9275b52bacdbc2c4f9e964/onnxruntime-1.23.1-cp312-cp312-macosx_13_0_x86_64.whl", hash = "sha256:3864c39307714eff1753149215ad86324a9372e3172a0275d5b16ffd296574bf", size = 19152841, upload-time = "2025-10-08T04:24:24.157Z" }, - { url = "https://files.pythonhosted.org/packages/3a/b9/1232fd295fa9c818aa2a7883d87a2f864fb5edee56ec757c6e857fdd1863/onnxruntime-1.23.1-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4e6b6b5ea80a96924f67fe1e5519f6c6f9cd716fdb5a4fd1ecb4f2b0971e8d00", size = 15223749, upload-time = "2025-10-08T04:24:08.088Z" }, - { url = "https://files.pythonhosted.org/packages/c4/b0/4663a333a82c77f159e48fe8639b1f03e4a05036625be9129c20c4d71d12/onnxruntime-1.23.1-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:576502dad714ffe5f3b4e1918c5b3368766b222063c585e5fd88415c063e4c80", size = 17378483, upload-time = "2025-10-08T04:24:50.712Z" }, - { url = "https://files.pythonhosted.org/packages/7c/60/8100d98690cbf1de03e08d1f3eff33ff00c652806c7130658a48a8f60584/onnxruntime-1.23.1-cp312-cp312-win_amd64.whl", hash = "sha256:1b89b7c4d4c00a67debc2b0a1484d7f51b23fef85fbd80ac83ed2d17b2161bd6", size = 13467773, upload-time = "2025-10-08T04:25:17.097Z" }, + { url = "https://files.pythonhosted.org/packages/44/be/467b00f09061572f022ffd17e49e49e5a7a789056bad95b54dfd3bee73ff/onnxruntime-1.23.2-cp311-cp311-macosx_13_0_arm64.whl", hash = "sha256:6f91d2c9b0965e86827a5ba01531d5b669770b01775b23199565d6c1f136616c", size = 17196113, upload-time = "2025-10-22T03:47:33.526Z" }, + { url = "https://files.pythonhosted.org/packages/9f/a8/3c23a8f75f93122d2b3410bfb74d06d0f8da4ac663185f91866b03f7da1b/onnxruntime-1.23.2-cp311-cp311-macosx_13_0_x86_64.whl", hash = "sha256:87d8b6eaf0fbeb6835a60a4265fde7a3b60157cf1b2764773ac47237b4d48612", size = 19153857, upload-time = "2025-10-22T03:46:37.578Z" }, + { url = "https://files.pythonhosted.org/packages/3f/d8/506eed9af03d86f8db4880a4c47cd0dffee973ef7e4f4cff9f1d4bcf7d22/onnxruntime-1.23.2-cp311-cp311-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:bbfd2fca76c855317568c1b36a885ddea2272c13cb0e395002c402f2360429a6", size = 15220095, upload-time = "2025-10-22T03:46:24.769Z" }, + { url = "https://files.pythonhosted.org/packages/e9/80/113381ba832d5e777accedc6cb41d10f9eca82321ae31ebb6bcede530cea/onnxruntime-1.23.2-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:da44b99206e77734c5819aa2142c69e64f3b46edc3bd314f6a45a932defc0b3e", size = 17372080, upload-time = "2025-10-22T03:47:00.265Z" }, + { url = "https://files.pythonhosted.org/packages/3a/db/1b4a62e23183a0c3fe441782462c0ede9a2a65c6bbffb9582fab7c7a0d38/onnxruntime-1.23.2-cp311-cp311-win_amd64.whl", hash = "sha256:902c756d8b633ce0dedd889b7c08459433fbcf35e9c38d1c03ddc020f0648c6e", size = 13468349, upload-time = "2025-10-22T03:47:25.783Z" }, + { url = "https://files.pythonhosted.org/packages/1b/9e/f748cd64161213adeef83d0cb16cb8ace1e62fa501033acdd9f9341fff57/onnxruntime-1.23.2-cp312-cp312-macosx_13_0_arm64.whl", hash = "sha256:b8f029a6b98d3cf5be564d52802bb50a8489ab73409fa9db0bf583eabb7c2321", size = 17195929, upload-time = "2025-10-22T03:47:36.24Z" }, + { url = "https://files.pythonhosted.org/packages/91/9d/a81aafd899b900101988ead7fb14974c8a58695338ab6a0f3d6b0100f30b/onnxruntime-1.23.2-cp312-cp312-macosx_13_0_x86_64.whl", hash = "sha256:218295a8acae83905f6f1aed8cacb8e3eb3bd7513a13fe4ba3b2664a19fc4a6b", size = 19157705, upload-time = "2025-10-22T03:46:40.415Z" }, + { url = "https://files.pythonhosted.org/packages/3c/35/4e40f2fba272a6698d62be2cd21ddc3675edfc1a4b9ddefcc4648f115315/onnxruntime-1.23.2-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:76ff670550dc23e58ea9bc53b5149b99a44e63b34b524f7b8547469aaa0dcb8c", size = 15226915, upload-time = "2025-10-22T03:46:27.773Z" }, + { url = "https://files.pythonhosted.org/packages/ef/88/9cc25d2bafe6bc0d4d3c1db3ade98196d5b355c0b273e6a5dc09c5d5d0d5/onnxruntime-1.23.2-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0f9b4ae77f8e3c9bee50c27bc1beede83f786fe1d52e99ac85aa8d65a01e9b77", size = 17382649, upload-time = "2025-10-22T03:47:02.782Z" }, + { url = "https://files.pythonhosted.org/packages/c0/b4/569d298f9fc4d286c11c45e85d9ffa9e877af12ace98af8cab52396e8f46/onnxruntime-1.23.2-cp312-cp312-win_amd64.whl", hash = "sha256:25de5214923ce941a3523739d34a520aac30f21e631de53bba9174dc9c004435", size = 13470528, upload-time = "2025-10-22T03:47:28.106Z" }, ] [[package]] name = "openai" -version = "2.3.0" +version = "2.7.2" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "anyio" }, @@ -3709,9 +3729,9 @@ dependencies = [ { name = "tqdm" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/de/90/8f26554d24d63ed4f94d33c24271559863223a67e624f4d2e65ba8e48dca/openai-2.3.0.tar.gz", hash = "sha256:8d213ee5aaf91737faea2d7fc1cd608657a5367a18966372a3756ceaabfbd812", size = 589616, upload-time = "2025-10-10T01:12:50.851Z" } +sdist = { url = "https://files.pythonhosted.org/packages/71/e3/cec27fa28ef36c4ccea71e9e8c20be9b8539618732989a82027575aab9d4/openai-2.7.2.tar.gz", hash = "sha256:082ef61163074d8efad0035dd08934cf5e3afd37254f70fc9165dd6a8c67dcbd", size = 595732, upload-time = "2025-11-10T16:42:31.108Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/9c/5b/4be258ff072ed8ee15f6bfd8d5a1a4618aa4704b127c0c5959212ad177d6/openai-2.3.0-py3-none-any.whl", hash = "sha256:a7aa83be6f7b0ab2e4d4d7bcaf36e3d790874c0167380c5d0afd0ed99a86bd7b", size = 999768, upload-time = "2025-10-10T01:12:48.647Z" }, + { url = "https://files.pythonhosted.org/packages/25/66/22cfe4b695b5fd042931b32c67d685e867bfd169ebf46036b95b57314c33/openai-2.7.2-py3-none-any.whl", hash = "sha256:116f522f4427f8a0a59b51655a356da85ce092f3ed6abeca65f03c8be6e073d9", size = 1008375, upload-time = "2025-11-10T16:42:28.574Z" }, ] [[package]] @@ -3732,7 +3752,7 @@ wheels = [ [[package]] name = "openinference-instrumentation" -version = "0.1.40" +version = "0.1.42" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "openinference-semantic-conventions" }, @@ -3740,18 +3760,18 @@ dependencies = [ { name = "opentelemetry-sdk" }, { name = "wrapt" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/37/59/750c25a353260a72287e618b9ccabd57f02db6bfd571c6dbf132202abeff/openinference_instrumentation-0.1.40.tar.gz", hash = "sha256:3080785479793a56023806c71dccbc39418925947407667794c651f992f700a2", size = 23824, upload-time = "2025-10-10T03:48:48.606Z" } +sdist = { url = "https://files.pythonhosted.org/packages/00/d0/b19061a21fd6127d2857c77744a36073bba9c1502d1d5e8517b708eb8b7c/openinference_instrumentation-0.1.42.tar.gz", hash = "sha256:2275babc34022e151b5492cfba41d3b12e28377f8e08cb45e5d64fe2d9d7fe37", size = 23954, upload-time = "2025-11-05T01:37:46.869Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/c2/fd/2b6ea9d95f3eb1deba10975a14b80d7fe79528258111771580a0437d4f44/openinference_instrumentation-0.1.40-py3-none-any.whl", hash = "sha256:d2e894f25addb1dfba563789213139876c5a01fca0a1fa8aa52a455a988a11d4", size = 29967, upload-time = "2025-10-10T03:48:46.518Z" }, + { url = "https://files.pythonhosted.org/packages/c3/71/43ee4616fc95dbd2f560550f199c6652a5eb93f84e8aa0039bc95c19cfe0/openinference_instrumentation-0.1.42-py3-none-any.whl", hash = "sha256:e7521ff90833ef7cc65db526a2f59b76a496180abeaaee30ec6abbbc0b43f8ec", size = 30086, upload-time = "2025-11-05T01:37:43.866Z" }, ] [[package]] name = "openinference-semantic-conventions" -version = "0.1.24" +version = "0.1.25" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/07/15/be7566a4bba4b57f7c70b088f42735f2005e2c0adce646a537f63dcf21de/openinference_semantic_conventions-0.1.24.tar.gz", hash = "sha256:3223b8c3958525457a369d58ebf0c56230a1f00567ae1e99f1c2049a8ac2cacd", size = 12741, upload-time = "2025-10-10T03:49:13.987Z" } +sdist = { url = "https://files.pythonhosted.org/packages/0b/68/81c8a0b90334ff11e4f285e4934c57f30bea3ef0c0b9f99b65e7b80fae3b/openinference_semantic_conventions-0.1.25.tar.gz", hash = "sha256:f0a8c2cfbd00195d1f362b4803518341e80867d446c2959bf1743f1894fce31d", size = 12767, upload-time = "2025-11-05T01:37:45.89Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/9f/c5/fa81b19042b387826151f984a91fa3d0b52b08374e4d5786521ac2d9e704/openinference_semantic_conventions-0.1.24-py3-none-any.whl", hash = "sha256:b2d650ca7e39c5fb02bf908b8049d6ece2a2657757448e1925a38b59548a80b3", size = 10373, upload-time = "2025-10-10T03:49:00.318Z" }, + { url = "https://files.pythonhosted.org/packages/fd/3d/dd14ee2eb8a3f3054249562e76b253a1545c76adbbfd43a294f71acde5c3/openinference_semantic_conventions-0.1.25-py3-none-any.whl", hash = "sha256:3814240f3bd61f05d9562b761de70ee793d55b03bca1634edf57d7a2735af238", size = 10395, upload-time = "2025-11-05T01:37:43.697Z" }, ] [[package]] @@ -4072,7 +4092,7 @@ wheels = [ [[package]] name = "opik" -version = "1.8.74" +version = "1.8.102" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "boto3-stubs", extra = ["bedrock-runtime"] }, @@ -4091,9 +4111,9 @@ dependencies = [ { name = "tqdm" }, { name = "uuid6" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/3b/31/887f133aa82aeb4cb8a01d98ad6ae73cb0580c2c9395d76bae1d67dbb6f6/opik-1.8.74.tar.gz", hash = "sha256:4b18248dbd741dab16dab399c1ab7197f1f6c6775ee06285ff07d3d22e1810de", size = 412504, upload-time = "2025-10-13T13:43:03.117Z" } +sdist = { url = "https://files.pythonhosted.org/packages/30/af/f6382cea86bdfbfd0f9571960a15301da4a6ecd1506070d9252a0c0a7564/opik-1.8.102.tar.gz", hash = "sha256:c836a113e8b7fdf90770a3854dcc859b3c30d6347383d7c11e52971a530ed2c3", size = 490462, upload-time = "2025-11-05T18:54:50.142Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/37/5d/11c12e2471880effa7a597d96bce848271fa93007f7f543ed607fb31822a/opik-1.8.74-py3-none-any.whl", hash = "sha256:34ffbff2c447da117e58bcc2fdf53b3b534dd1ffe9a293eb912f5419fc9904c3", size = 772547, upload-time = "2025-10-13T13:43:01.29Z" }, + { url = "https://files.pythonhosted.org/packages/b9/8b/9b15a01f8360201100b9a5d3e0aeeeda57833fca2b16d34b9fada147fc4b/opik-1.8.102-py3-none-any.whl", hash = "sha256:d8501134bf62bf95443de036f6eaa4f66006f81f9b99e0a8a09e21d8be8c1628", size = 885834, upload-time = "2025-11-05T18:54:48.22Z" }, ] [[package]] @@ -4137,40 +4157,40 @@ wheels = [ [[package]] name = "orjson" -version = "3.11.3" +version = "3.11.4" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/be/4d/8df5f83256a809c22c4d6792ce8d43bb503be0fb7a8e4da9025754b09658/orjson-3.11.3.tar.gz", hash = "sha256:1c0603b1d2ffcd43a411d64797a19556ef76958aef1c182f22dc30860152a98a", size = 5482394, upload-time = "2025-08-26T17:46:43.171Z" } +sdist = { url = "https://files.pythonhosted.org/packages/c6/fe/ed708782d6709cc60eb4c2d8a361a440661f74134675c72990f2c48c785f/orjson-3.11.4.tar.gz", hash = "sha256:39485f4ab4c9b30a3943cfe99e1a213c4776fb69e8abd68f66b83d5a0b0fdc6d", size = 5945188, upload-time = "2025-10-24T15:50:38.027Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/cd/8b/360674cd817faef32e49276187922a946468579fcaf37afdfb6c07046e92/orjson-3.11.3-cp311-cp311-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:9d2ae0cc6aeb669633e0124531f342a17d8e97ea999e42f12a5ad4adaa304c5f", size = 238238, upload-time = "2025-08-26T17:44:54.214Z" }, - { url = "https://files.pythonhosted.org/packages/05/3d/5fa9ea4b34c1a13be7d9046ba98d06e6feb1d8853718992954ab59d16625/orjson-3.11.3-cp311-cp311-macosx_15_0_arm64.whl", hash = "sha256:ba21dbb2493e9c653eaffdc38819b004b7b1b246fb77bfc93dc016fe664eac91", size = 127713, upload-time = "2025-08-26T17:44:55.596Z" }, - { url = "https://files.pythonhosted.org/packages/e5/5f/e18367823925e00b1feec867ff5f040055892fc474bf5f7875649ecfa586/orjson-3.11.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:00f1a271e56d511d1569937c0447d7dce5a99a33ea0dec76673706360a051904", size = 123241, upload-time = "2025-08-26T17:44:57.185Z" }, - { url = "https://files.pythonhosted.org/packages/0f/bd/3c66b91c4564759cf9f473251ac1650e446c7ba92a7c0f9f56ed54f9f0e6/orjson-3.11.3-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:b67e71e47caa6680d1b6f075a396d04fa6ca8ca09aafb428731da9b3ea32a5a6", size = 127895, upload-time = "2025-08-26T17:44:58.349Z" }, - { url = "https://files.pythonhosted.org/packages/82/b5/dc8dcd609db4766e2967a85f63296c59d4722b39503e5b0bf7fd340d387f/orjson-3.11.3-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d7d012ebddffcce8c85734a6d9e5f08180cd3857c5f5a3ac70185b43775d043d", size = 130303, upload-time = "2025-08-26T17:44:59.491Z" }, - { url = "https://files.pythonhosted.org/packages/48/c2/d58ec5fd1270b2aa44c862171891adc2e1241bd7dab26c8f46eb97c6c6f1/orjson-3.11.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:dd759f75d6b8d1b62012b7f5ef9461d03c804f94d539a5515b454ba3a6588038", size = 132366, upload-time = "2025-08-26T17:45:00.654Z" }, - { url = "https://files.pythonhosted.org/packages/73/87/0ef7e22eb8dd1ef940bfe3b9e441db519e692d62ed1aae365406a16d23d0/orjson-3.11.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:6890ace0809627b0dff19cfad92d69d0fa3f089d3e359a2a532507bb6ba34efb", size = 135180, upload-time = "2025-08-26T17:45:02.424Z" }, - { url = "https://files.pythonhosted.org/packages/bb/6a/e5bf7b70883f374710ad74faf99bacfc4b5b5a7797c1d5e130350e0e28a3/orjson-3.11.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f9d4a5e041ae435b815e568537755773d05dac031fee6a57b4ba70897a44d9d2", size = 132741, upload-time = "2025-08-26T17:45:03.663Z" }, - { url = "https://files.pythonhosted.org/packages/bd/0c/4577fd860b6386ffaa56440e792af01c7882b56d2766f55384b5b0e9d39b/orjson-3.11.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:2d68bf97a771836687107abfca089743885fb664b90138d8761cce61d5625d55", size = 131104, upload-time = "2025-08-26T17:45:04.939Z" }, - { url = "https://files.pythonhosted.org/packages/66/4b/83e92b2d67e86d1c33f2ea9411742a714a26de63641b082bdbf3d8e481af/orjson-3.11.3-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:bfc27516ec46f4520b18ef645864cee168d2a027dbf32c5537cb1f3e3c22dac1", size = 403887, upload-time = "2025-08-26T17:45:06.228Z" }, - { url = "https://files.pythonhosted.org/packages/6d/e5/9eea6a14e9b5ceb4a271a1fd2e1dec5f2f686755c0fab6673dc6ff3433f4/orjson-3.11.3-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:f66b001332a017d7945e177e282a40b6997056394e3ed7ddb41fb1813b83e824", size = 145855, upload-time = "2025-08-26T17:45:08.338Z" }, - { url = "https://files.pythonhosted.org/packages/45/78/8d4f5ad0c80ba9bf8ac4d0fc71f93a7d0dc0844989e645e2074af376c307/orjson-3.11.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:212e67806525d2561efbfe9e799633b17eb668b8964abed6b5319b2f1cfbae1f", size = 135361, upload-time = "2025-08-26T17:45:09.625Z" }, - { url = "https://files.pythonhosted.org/packages/0b/5f/16386970370178d7a9b438517ea3d704efcf163d286422bae3b37b88dbb5/orjson-3.11.3-cp311-cp311-win32.whl", hash = "sha256:6e8e0c3b85575a32f2ffa59de455f85ce002b8bdc0662d6b9c2ed6d80ab5d204", size = 136190, upload-time = "2025-08-26T17:45:10.962Z" }, - { url = "https://files.pythonhosted.org/packages/09/60/db16c6f7a41dd8ac9fb651f66701ff2aeb499ad9ebc15853a26c7c152448/orjson-3.11.3-cp311-cp311-win_amd64.whl", hash = "sha256:6be2f1b5d3dc99a5ce5ce162fc741c22ba9f3443d3dd586e6a1211b7bc87bc7b", size = 131389, upload-time = "2025-08-26T17:45:12.285Z" }, - { url = "https://files.pythonhosted.org/packages/3e/2a/bb811ad336667041dea9b8565c7c9faf2f59b47eb5ab680315eea612ef2e/orjson-3.11.3-cp311-cp311-win_arm64.whl", hash = "sha256:fafb1a99d740523d964b15c8db4eabbfc86ff29f84898262bf6e3e4c9e97e43e", size = 126120, upload-time = "2025-08-26T17:45:13.515Z" }, - { url = "https://files.pythonhosted.org/packages/3d/b0/a7edab2a00cdcb2688e1c943401cb3236323e7bfd2839815c6131a3742f4/orjson-3.11.3-cp312-cp312-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:8c752089db84333e36d754c4baf19c0e1437012242048439c7e80eb0e6426e3b", size = 238259, upload-time = "2025-08-26T17:45:15.093Z" }, - { url = "https://files.pythonhosted.org/packages/e1/c6/ff4865a9cc398a07a83342713b5932e4dc3cb4bf4bc04e8f83dedfc0d736/orjson-3.11.3-cp312-cp312-macosx_15_0_arm64.whl", hash = "sha256:9b8761b6cf04a856eb544acdd82fc594b978f12ac3602d6374a7edb9d86fd2c2", size = 127633, upload-time = "2025-08-26T17:45:16.417Z" }, - { url = "https://files.pythonhosted.org/packages/6e/e6/e00bea2d9472f44fe8794f523e548ce0ad51eb9693cf538a753a27b8bda4/orjson-3.11.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b13974dc8ac6ba22feaa867fc19135a3e01a134b4f7c9c28162fed4d615008a", size = 123061, upload-time = "2025-08-26T17:45:17.673Z" }, - { url = "https://files.pythonhosted.org/packages/54/31/9fbb78b8e1eb3ac605467cb846e1c08d0588506028b37f4ee21f978a51d4/orjson-3.11.3-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f83abab5bacb76d9c821fd5c07728ff224ed0e52d7a71b7b3de822f3df04e15c", size = 127956, upload-time = "2025-08-26T17:45:19.172Z" }, - { url = "https://files.pythonhosted.org/packages/36/88/b0604c22af1eed9f98d709a96302006915cfd724a7ebd27d6dd11c22d80b/orjson-3.11.3-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e6fbaf48a744b94091a56c62897b27c31ee2da93d826aa5b207131a1e13d4064", size = 130790, upload-time = "2025-08-26T17:45:20.586Z" }, - { url = "https://files.pythonhosted.org/packages/0e/9d/1c1238ae9fffbfed51ba1e507731b3faaf6b846126a47e9649222b0fd06f/orjson-3.11.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bc779b4f4bba2847d0d2940081a7b6f7b5877e05408ffbb74fa1faf4a136c424", size = 132385, upload-time = "2025-08-26T17:45:22.036Z" }, - { url = "https://files.pythonhosted.org/packages/a3/b5/c06f1b090a1c875f337e21dd71943bc9d84087f7cdf8c6e9086902c34e42/orjson-3.11.3-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bd4b909ce4c50faa2192da6bb684d9848d4510b736b0611b6ab4020ea6fd2d23", size = 135305, upload-time = "2025-08-26T17:45:23.4Z" }, - { url = "https://files.pythonhosted.org/packages/a0/26/5f028c7d81ad2ebbf84414ba6d6c9cac03f22f5cd0d01eb40fb2d6a06b07/orjson-3.11.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:524b765ad888dc5518bbce12c77c2e83dee1ed6b0992c1790cc5fb49bb4b6667", size = 132875, upload-time = "2025-08-26T17:45:25.182Z" }, - { url = "https://files.pythonhosted.org/packages/fe/d4/b8df70d9cfb56e385bf39b4e915298f9ae6c61454c8154a0f5fd7efcd42e/orjson-3.11.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:84fd82870b97ae3cdcea9d8746e592b6d40e1e4d4527835fc520c588d2ded04f", size = 130940, upload-time = "2025-08-26T17:45:27.209Z" }, - { url = "https://files.pythonhosted.org/packages/da/5e/afe6a052ebc1a4741c792dd96e9f65bf3939d2094e8b356503b68d48f9f5/orjson-3.11.3-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:fbecb9709111be913ae6879b07bafd4b0785b44c1eb5cac8ac76da048b3885a1", size = 403852, upload-time = "2025-08-26T17:45:28.478Z" }, - { url = "https://files.pythonhosted.org/packages/f8/90/7bbabafeb2ce65915e9247f14a56b29c9334003536009ef5b122783fe67e/orjson-3.11.3-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:9dba358d55aee552bd868de348f4736ca5a4086d9a62e2bfbbeeb5629fe8b0cc", size = 146293, upload-time = "2025-08-26T17:45:29.86Z" }, - { url = "https://files.pythonhosted.org/packages/27/b3/2d703946447da8b093350570644a663df69448c9d9330e5f1d9cce997f20/orjson-3.11.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:eabcf2e84f1d7105f84580e03012270c7e97ecb1fb1618bda395061b2a84a049", size = 135470, upload-time = "2025-08-26T17:45:31.243Z" }, - { url = "https://files.pythonhosted.org/packages/38/70/b14dcfae7aff0e379b0119c8a812f8396678919c431efccc8e8a0263e4d9/orjson-3.11.3-cp312-cp312-win32.whl", hash = "sha256:3782d2c60b8116772aea8d9b7905221437fdf53e7277282e8d8b07c220f96cca", size = 136248, upload-time = "2025-08-26T17:45:32.567Z" }, - { url = "https://files.pythonhosted.org/packages/35/b8/9e3127d65de7fff243f7f3e53f59a531bf6bb295ebe5db024c2503cc0726/orjson-3.11.3-cp312-cp312-win_amd64.whl", hash = "sha256:79b44319268af2eaa3e315b92298de9a0067ade6e6003ddaef72f8e0bedb94f1", size = 131437, upload-time = "2025-08-26T17:45:34.949Z" }, - { url = "https://files.pythonhosted.org/packages/51/92/a946e737d4d8a7fd84a606aba96220043dcc7d6988b9e7551f7f6d5ba5ad/orjson-3.11.3-cp312-cp312-win_arm64.whl", hash = "sha256:0e92a4e83341ef79d835ca21b8bd13e27c859e4e9e4d7b63defc6e58462a3710", size = 125978, upload-time = "2025-08-26T17:45:36.422Z" }, + { url = "https://files.pythonhosted.org/packages/63/1d/1ea6005fffb56715fd48f632611e163d1604e8316a5bad2288bee9a1c9eb/orjson-3.11.4-cp311-cp311-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:5e59d23cd93ada23ec59a96f215139753fbfe3a4d989549bcb390f8c00370b39", size = 243498, upload-time = "2025-10-24T15:48:48.101Z" }, + { url = "https://files.pythonhosted.org/packages/37/d7/ffed10c7da677f2a9da307d491b9eb1d0125b0307019c4ad3d665fd31f4f/orjson-3.11.4-cp311-cp311-macosx_15_0_arm64.whl", hash = "sha256:5c3aedecfc1beb988c27c79d52ebefab93b6c3921dbec361167e6559aba2d36d", size = 128961, upload-time = "2025-10-24T15:48:49.571Z" }, + { url = "https://files.pythonhosted.org/packages/a2/96/3e4d10a18866d1368f73c8c44b7fe37cc8a15c32f2a7620be3877d4c55a3/orjson-3.11.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da9e5301f1c2caa2a9a4a303480d79c9ad73560b2e7761de742ab39fe59d9175", size = 130321, upload-time = "2025-10-24T15:48:50.713Z" }, + { url = "https://files.pythonhosted.org/packages/eb/1f/465f66e93f434f968dd74d5b623eb62c657bdba2332f5a8be9f118bb74c7/orjson-3.11.4-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:8873812c164a90a79f65368f8f96817e59e35d0cc02786a5356f0e2abed78040", size = 129207, upload-time = "2025-10-24T15:48:52.193Z" }, + { url = "https://files.pythonhosted.org/packages/28/43/d1e94837543321c119dff277ae8e348562fe8c0fafbb648ef7cb0c67e521/orjson-3.11.4-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5d7feb0741ebb15204e748f26c9638e6665a5fa93c37a2c73d64f1669b0ddc63", size = 136323, upload-time = "2025-10-24T15:48:54.806Z" }, + { url = "https://files.pythonhosted.org/packages/bf/04/93303776c8890e422a5847dd012b4853cdd88206b8bbd3edc292c90102d1/orjson-3.11.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:01ee5487fefee21e6910da4c2ee9eef005bee568a0879834df86f888d2ffbdd9", size = 137440, upload-time = "2025-10-24T15:48:56.326Z" }, + { url = "https://files.pythonhosted.org/packages/1e/ef/75519d039e5ae6b0f34d0336854d55544ba903e21bf56c83adc51cd8bf82/orjson-3.11.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3d40d46f348c0321df01507f92b95a377240c4ec31985225a6668f10e2676f9a", size = 136680, upload-time = "2025-10-24T15:48:57.476Z" }, + { url = "https://files.pythonhosted.org/packages/b5/18/bf8581eaae0b941b44efe14fee7b7862c3382fbc9a0842132cfc7cf5ecf4/orjson-3.11.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:95713e5fc8af84d8edc75b785d2386f653b63d62b16d681687746734b4dfc0be", size = 136160, upload-time = "2025-10-24T15:48:59.631Z" }, + { url = "https://files.pythonhosted.org/packages/c4/35/a6d582766d351f87fc0a22ad740a641b0a8e6fc47515e8614d2e4790ae10/orjson-3.11.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:ad73ede24f9083614d6c4ca9a85fe70e33be7bf047ec586ee2363bc7418fe4d7", size = 140318, upload-time = "2025-10-24T15:49:00.834Z" }, + { url = "https://files.pythonhosted.org/packages/76/b3/5a4801803ab2e2e2d703bce1a56540d9f99a9143fbec7bf63d225044fef8/orjson-3.11.4-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:842289889de515421f3f224ef9c1f1efb199a32d76d8d2ca2706fa8afe749549", size = 406330, upload-time = "2025-10-24T15:49:02.327Z" }, + { url = "https://files.pythonhosted.org/packages/80/55/a8f682f64833e3a649f620eafefee175cbfeb9854fc5b710b90c3bca45df/orjson-3.11.4-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:3b2427ed5791619851c52a1261b45c233930977e7de8cf36de05636c708fa905", size = 149580, upload-time = "2025-10-24T15:49:03.517Z" }, + { url = "https://files.pythonhosted.org/packages/ad/e4/c132fa0c67afbb3eb88274fa98df9ac1f631a675e7877037c611805a4413/orjson-3.11.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:3c36e524af1d29982e9b190573677ea02781456b2e537d5840e4538a5ec41907", size = 139846, upload-time = "2025-10-24T15:49:04.761Z" }, + { url = "https://files.pythonhosted.org/packages/54/06/dc3491489efd651fef99c5908e13951abd1aead1257c67f16135f95ce209/orjson-3.11.4-cp311-cp311-win32.whl", hash = "sha256:87255b88756eab4a68ec61837ca754e5d10fa8bc47dc57f75cedfeaec358d54c", size = 135781, upload-time = "2025-10-24T15:49:05.969Z" }, + { url = "https://files.pythonhosted.org/packages/79/b7/5e5e8d77bd4ea02a6ac54c42c818afb01dd31961be8a574eb79f1d2cfb1e/orjson-3.11.4-cp311-cp311-win_amd64.whl", hash = "sha256:e2d5d5d798aba9a0e1fede8d853fa899ce2cb930ec0857365f700dffc2c7af6a", size = 131391, upload-time = "2025-10-24T15:49:07.355Z" }, + { url = "https://files.pythonhosted.org/packages/0f/dc/9484127cc1aa213be398ed735f5f270eedcb0c0977303a6f6ddc46b60204/orjson-3.11.4-cp311-cp311-win_arm64.whl", hash = "sha256:6bb6bb41b14c95d4f2702bce9975fda4516f1db48e500102fc4d8119032ff045", size = 126252, upload-time = "2025-10-24T15:49:08.869Z" }, + { url = "https://files.pythonhosted.org/packages/63/51/6b556192a04595b93e277a9ff71cd0cc06c21a7df98bcce5963fa0f5e36f/orjson-3.11.4-cp312-cp312-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:d4371de39319d05d3f482f372720b841c841b52f5385bd99c61ed69d55d9ab50", size = 243571, upload-time = "2025-10-24T15:49:10.008Z" }, + { url = "https://files.pythonhosted.org/packages/1c/2c/2602392ddf2601d538ff11848b98621cd465d1a1ceb9db9e8043181f2f7b/orjson-3.11.4-cp312-cp312-macosx_15_0_arm64.whl", hash = "sha256:e41fd3b3cac850eaae78232f37325ed7d7436e11c471246b87b2cd294ec94853", size = 128891, upload-time = "2025-10-24T15:49:11.297Z" }, + { url = "https://files.pythonhosted.org/packages/4e/47/bf85dcf95f7a3a12bf223394a4f849430acd82633848d52def09fa3f46ad/orjson-3.11.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:600e0e9ca042878c7fdf189cf1b028fe2c1418cc9195f6cb9824eb6ed99cb938", size = 130137, upload-time = "2025-10-24T15:49:12.544Z" }, + { url = "https://files.pythonhosted.org/packages/b4/4d/a0cb31007f3ab6f1fd2a1b17057c7c349bc2baf8921a85c0180cc7be8011/orjson-3.11.4-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:7bbf9b333f1568ef5da42bc96e18bf30fd7f8d54e9ae066d711056add508e415", size = 129152, upload-time = "2025-10-24T15:49:13.754Z" }, + { url = "https://files.pythonhosted.org/packages/f7/ef/2811def7ce3d8576b19e3929fff8f8f0d44bc5eb2e0fdecb2e6e6cc6c720/orjson-3.11.4-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4806363144bb6e7297b8e95870e78d30a649fdc4e23fc84daa80c8ebd366ce44", size = 136834, upload-time = "2025-10-24T15:49:15.307Z" }, + { url = "https://files.pythonhosted.org/packages/00/d4/9aee9e54f1809cec8ed5abd9bc31e8a9631d19460e3b8470145d25140106/orjson-3.11.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ad355e8308493f527d41154e9053b86a5be892b3b359a5c6d5d95cda23601cb2", size = 137519, upload-time = "2025-10-24T15:49:16.557Z" }, + { url = "https://files.pythonhosted.org/packages/db/ea/67bfdb5465d5679e8ae8d68c11753aaf4f47e3e7264bad66dc2f2249e643/orjson-3.11.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c8a7517482667fb9f0ff1b2f16fe5829296ed7a655d04d68cd9711a4d8a4e708", size = 136749, upload-time = "2025-10-24T15:49:17.796Z" }, + { url = "https://files.pythonhosted.org/packages/01/7e/62517dddcfce6d53a39543cd74d0dccfcbdf53967017c58af68822100272/orjson-3.11.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:97eb5942c7395a171cbfecc4ef6701fc3c403e762194683772df4c54cfbb2210", size = 136325, upload-time = "2025-10-24T15:49:19.347Z" }, + { url = "https://files.pythonhosted.org/packages/18/ae/40516739f99ab4c7ec3aaa5cc242d341fcb03a45d89edeeaabc5f69cb2cf/orjson-3.11.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:149d95d5e018bdd822e3f38c103b1a7c91f88d38a88aada5c4e9b3a73a244241", size = 140204, upload-time = "2025-10-24T15:49:20.545Z" }, + { url = "https://files.pythonhosted.org/packages/82/18/ff5734365623a8916e3a4037fcef1cd1782bfc14cf0992afe7940c5320bf/orjson-3.11.4-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:624f3951181eb46fc47dea3d221554e98784c823e7069edb5dbd0dc826ac909b", size = 406242, upload-time = "2025-10-24T15:49:21.884Z" }, + { url = "https://files.pythonhosted.org/packages/e1/43/96436041f0a0c8c8deca6a05ebeaf529bf1de04839f93ac5e7c479807aec/orjson-3.11.4-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:03bfa548cf35e3f8b3a96c4e8e41f753c686ff3d8e182ce275b1751deddab58c", size = 150013, upload-time = "2025-10-24T15:49:23.185Z" }, + { url = "https://files.pythonhosted.org/packages/1b/48/78302d98423ed8780479a1e682b9aecb869e8404545d999d34fa486e573e/orjson-3.11.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:525021896afef44a68148f6ed8a8bf8375553d6066c7f48537657f64823565b9", size = 139951, upload-time = "2025-10-24T15:49:24.428Z" }, + { url = "https://files.pythonhosted.org/packages/4a/7b/ad613fdcdaa812f075ec0875143c3d37f8654457d2af17703905425981bf/orjson-3.11.4-cp312-cp312-win32.whl", hash = "sha256:b58430396687ce0f7d9eeb3dd47761ca7d8fda8e9eb92b3077a7a353a75efefa", size = 136049, upload-time = "2025-10-24T15:49:25.973Z" }, + { url = "https://files.pythonhosted.org/packages/b9/3c/9cf47c3ff5f39b8350fb21ba65d789b6a1129d4cbb3033ba36c8a9023520/orjson-3.11.4-cp312-cp312-win_amd64.whl", hash = "sha256:c6dbf422894e1e3c80a177133c0dda260f81428f9de16d61041949f6a2e5c140", size = 131461, upload-time = "2025-10-24T15:49:27.259Z" }, + { url = "https://files.pythonhosted.org/packages/c6/3b/e2425f61e5825dc5b08c2a5a2b3af387eaaca22a12b9c8c01504f8614c36/orjson-3.11.4-cp312-cp312-win_arm64.whl", hash = "sha256:d38d2bc06d6415852224fcc9c0bfa834c25431e466dc319f0edd56cca81aa96e", size = 126167, upload-time = "2025-10-24T15:49:28.511Z" }, ] [[package]] @@ -4318,39 +4338,39 @@ wheels = [ [[package]] name = "pillow" -version = "11.3.0" +version = "12.0.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/f3/0d/d0d6dea55cd152ce3d6767bb38a8fc10e33796ba4ba210cbab9354b6d238/pillow-11.3.0.tar.gz", hash = "sha256:3828ee7586cd0b2091b6209e5ad53e20d0649bbe87164a459d0676e035e8f523", size = 47113069, upload-time = "2025-07-01T09:16:30.666Z" } +sdist = { url = "https://files.pythonhosted.org/packages/5a/b0/cace85a1b0c9775a9f8f5d5423c8261c858760e2466c79b2dd184638b056/pillow-12.0.0.tar.gz", hash = "sha256:87d4f8125c9988bfbed67af47dd7a953e2fc7b0cc1e7800ec6d2080d490bb353", size = 47008828, upload-time = "2025-10-15T18:24:14.008Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/db/26/77f8ed17ca4ffd60e1dcd220a6ec6d71210ba398cfa33a13a1cd614c5613/pillow-11.3.0-cp311-cp311-macosx_10_10_x86_64.whl", hash = "sha256:1cd110edf822773368b396281a2293aeb91c90a2db00d78ea43e7e861631b722", size = 5316531, upload-time = "2025-07-01T09:13:59.203Z" }, - { url = "https://files.pythonhosted.org/packages/cb/39/ee475903197ce709322a17a866892efb560f57900d9af2e55f86db51b0a5/pillow-11.3.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:9c412fddd1b77a75aa904615ebaa6001f169b26fd467b4be93aded278266b288", size = 4686560, upload-time = "2025-07-01T09:14:01.101Z" }, - { url = "https://files.pythonhosted.org/packages/d5/90/442068a160fd179938ba55ec8c97050a612426fae5ec0a764e345839f76d/pillow-11.3.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:7d1aa4de119a0ecac0a34a9c8bde33f34022e2e8f99104e47a3ca392fd60e37d", size = 5870978, upload-time = "2025-07-03T13:09:55.638Z" }, - { url = "https://files.pythonhosted.org/packages/13/92/dcdd147ab02daf405387f0218dcf792dc6dd5b14d2573d40b4caeef01059/pillow-11.3.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:91da1d88226663594e3f6b4b8c3c8d85bd504117d043740a8e0ec449087cc494", size = 7641168, upload-time = "2025-07-03T13:10:00.37Z" }, - { url = "https://files.pythonhosted.org/packages/6e/db/839d6ba7fd38b51af641aa904e2960e7a5644d60ec754c046b7d2aee00e5/pillow-11.3.0-cp311-cp311-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:643f189248837533073c405ec2f0bb250ba54598cf80e8c1e043381a60632f58", size = 5973053, upload-time = "2025-07-01T09:14:04.491Z" }, - { url = "https://files.pythonhosted.org/packages/f2/2f/d7675ecae6c43e9f12aa8d58b6012683b20b6edfbdac7abcb4e6af7a3784/pillow-11.3.0-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:106064daa23a745510dabce1d84f29137a37224831d88eb4ce94bb187b1d7e5f", size = 6640273, upload-time = "2025-07-01T09:14:06.235Z" }, - { url = "https://files.pythonhosted.org/packages/45/ad/931694675ede172e15b2ff03c8144a0ddaea1d87adb72bb07655eaffb654/pillow-11.3.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:cd8ff254faf15591e724dc7c4ddb6bf4793efcbe13802a4ae3e863cd300b493e", size = 6082043, upload-time = "2025-07-01T09:14:07.978Z" }, - { url = "https://files.pythonhosted.org/packages/3a/04/ba8f2b11fc80d2dd462d7abec16351b45ec99cbbaea4387648a44190351a/pillow-11.3.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:932c754c2d51ad2b2271fd01c3d121daaa35e27efae2a616f77bf164bc0b3e94", size = 6715516, upload-time = "2025-07-01T09:14:10.233Z" }, - { url = "https://files.pythonhosted.org/packages/48/59/8cd06d7f3944cc7d892e8533c56b0acb68399f640786313275faec1e3b6f/pillow-11.3.0-cp311-cp311-win32.whl", hash = "sha256:b4b8f3efc8d530a1544e5962bd6b403d5f7fe8b9e08227c6b255f98ad82b4ba0", size = 6274768, upload-time = "2025-07-01T09:14:11.921Z" }, - { url = "https://files.pythonhosted.org/packages/f1/cc/29c0f5d64ab8eae20f3232da8f8571660aa0ab4b8f1331da5c2f5f9a938e/pillow-11.3.0-cp311-cp311-win_amd64.whl", hash = "sha256:1a992e86b0dd7aeb1f053cd506508c0999d710a8f07b4c791c63843fc6a807ac", size = 6986055, upload-time = "2025-07-01T09:14:13.623Z" }, - { url = "https://files.pythonhosted.org/packages/c6/df/90bd886fabd544c25addd63e5ca6932c86f2b701d5da6c7839387a076b4a/pillow-11.3.0-cp311-cp311-win_arm64.whl", hash = "sha256:30807c931ff7c095620fe04448e2c2fc673fcbb1ffe2a7da3fb39613489b1ddd", size = 2423079, upload-time = "2025-07-01T09:14:15.268Z" }, - { url = "https://files.pythonhosted.org/packages/40/fe/1bc9b3ee13f68487a99ac9529968035cca2f0a51ec36892060edcc51d06a/pillow-11.3.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:fdae223722da47b024b867c1ea0be64e0df702c5e0a60e27daad39bf960dd1e4", size = 5278800, upload-time = "2025-07-01T09:14:17.648Z" }, - { url = "https://files.pythonhosted.org/packages/2c/32/7e2ac19b5713657384cec55f89065fb306b06af008cfd87e572035b27119/pillow-11.3.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:921bd305b10e82b4d1f5e802b6850677f965d8394203d182f078873851dada69", size = 4686296, upload-time = "2025-07-01T09:14:19.828Z" }, - { url = "https://files.pythonhosted.org/packages/8e/1e/b9e12bbe6e4c2220effebc09ea0923a07a6da1e1f1bfbc8d7d29a01ce32b/pillow-11.3.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:eb76541cba2f958032d79d143b98a3a6b3ea87f0959bbe256c0b5e416599fd5d", size = 5871726, upload-time = "2025-07-03T13:10:04.448Z" }, - { url = "https://files.pythonhosted.org/packages/8d/33/e9200d2bd7ba00dc3ddb78df1198a6e80d7669cce6c2bdbeb2530a74ec58/pillow-11.3.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:67172f2944ebba3d4a7b54f2e95c786a3a50c21b88456329314caaa28cda70f6", size = 7644652, upload-time = "2025-07-03T13:10:10.391Z" }, - { url = "https://files.pythonhosted.org/packages/41/f1/6f2427a26fc683e00d985bc391bdd76d8dd4e92fac33d841127eb8fb2313/pillow-11.3.0-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:97f07ed9f56a3b9b5f49d3661dc9607484e85c67e27f3e8be2c7d28ca032fec7", size = 5977787, upload-time = "2025-07-01T09:14:21.63Z" }, - { url = "https://files.pythonhosted.org/packages/e4/c9/06dd4a38974e24f932ff5f98ea3c546ce3f8c995d3f0985f8e5ba48bba19/pillow-11.3.0-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:676b2815362456b5b3216b4fd5bd89d362100dc6f4945154ff172e206a22c024", size = 6645236, upload-time = "2025-07-01T09:14:23.321Z" }, - { url = "https://files.pythonhosted.org/packages/40/e7/848f69fb79843b3d91241bad658e9c14f39a32f71a301bcd1d139416d1be/pillow-11.3.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:3e184b2f26ff146363dd07bde8b711833d7b0202e27d13540bfe2e35a323a809", size = 6086950, upload-time = "2025-07-01T09:14:25.237Z" }, - { url = "https://files.pythonhosted.org/packages/0b/1a/7cff92e695a2a29ac1958c2a0fe4c0b2393b60aac13b04a4fe2735cad52d/pillow-11.3.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:6be31e3fc9a621e071bc17bb7de63b85cbe0bfae91bb0363c893cbe67247780d", size = 6723358, upload-time = "2025-07-01T09:14:27.053Z" }, - { url = "https://files.pythonhosted.org/packages/26/7d/73699ad77895f69edff76b0f332acc3d497f22f5d75e5360f78cbcaff248/pillow-11.3.0-cp312-cp312-win32.whl", hash = "sha256:7b161756381f0918e05e7cb8a371fff367e807770f8fe92ecb20d905d0e1c149", size = 6275079, upload-time = "2025-07-01T09:14:30.104Z" }, - { url = "https://files.pythonhosted.org/packages/8c/ce/e7dfc873bdd9828f3b6e5c2bbb74e47a98ec23cc5c74fc4e54462f0d9204/pillow-11.3.0-cp312-cp312-win_amd64.whl", hash = "sha256:a6444696fce635783440b7f7a9fc24b3ad10a9ea3f0ab66c5905be1c19ccf17d", size = 6986324, upload-time = "2025-07-01T09:14:31.899Z" }, - { url = "https://files.pythonhosted.org/packages/16/8f/b13447d1bf0b1f7467ce7d86f6e6edf66c0ad7cf44cf5c87a37f9bed9936/pillow-11.3.0-cp312-cp312-win_arm64.whl", hash = "sha256:2aceea54f957dd4448264f9bf40875da0415c83eb85f55069d89c0ed436e3542", size = 2423067, upload-time = "2025-07-01T09:14:33.709Z" }, - { url = "https://files.pythonhosted.org/packages/9e/e3/6fa84033758276fb31da12e5fb66ad747ae83b93c67af17f8c6ff4cc8f34/pillow-11.3.0-pp311-pypy311_pp73-macosx_10_15_x86_64.whl", hash = "sha256:7c8ec7a017ad1bd562f93dbd8505763e688d388cde6e4a010ae1486916e713e6", size = 5270566, upload-time = "2025-07-01T09:16:19.801Z" }, - { url = "https://files.pythonhosted.org/packages/5b/ee/e8d2e1ab4892970b561e1ba96cbd59c0d28cf66737fc44abb2aec3795a4e/pillow-11.3.0-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:9ab6ae226de48019caa8074894544af5b53a117ccb9d3b3dcb2871464c829438", size = 4654618, upload-time = "2025-07-01T09:16:21.818Z" }, - { url = "https://files.pythonhosted.org/packages/f2/6d/17f80f4e1f0761f02160fc433abd4109fa1548dcfdca46cfdadaf9efa565/pillow-11.3.0-pp311-pypy311_pp73-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:fe27fb049cdcca11f11a7bfda64043c37b30e6b91f10cb5bab275806c32f6ab3", size = 4874248, upload-time = "2025-07-03T13:11:20.738Z" }, - { url = "https://files.pythonhosted.org/packages/de/5f/c22340acd61cef960130585bbe2120e2fd8434c214802f07e8c03596b17e/pillow-11.3.0-pp311-pypy311_pp73-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:465b9e8844e3c3519a983d58b80be3f668e2a7a5db97f2784e7079fbc9f9822c", size = 6583963, upload-time = "2025-07-03T13:11:26.283Z" }, - { url = "https://files.pythonhosted.org/packages/31/5e/03966aedfbfcbb4d5f8aa042452d3361f325b963ebbadddac05b122e47dd/pillow-11.3.0-pp311-pypy311_pp73-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5418b53c0d59b3824d05e029669efa023bbef0f3e92e75ec8428f3799487f361", size = 4957170, upload-time = "2025-07-01T09:16:23.762Z" }, - { url = "https://files.pythonhosted.org/packages/cc/2d/e082982aacc927fc2cab48e1e731bdb1643a1406acace8bed0900a61464e/pillow-11.3.0-pp311-pypy311_pp73-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:504b6f59505f08ae014f724b6207ff6222662aab5cc9542577fb084ed0676ac7", size = 5581505, upload-time = "2025-07-01T09:16:25.593Z" }, - { url = "https://files.pythonhosted.org/packages/34/e7/ae39f538fd6844e982063c3a5e4598b8ced43b9633baa3a85ef33af8c05c/pillow-11.3.0-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:c84d689db21a1c397d001aa08241044aa2069e7587b398c8cc63020390b1c1b8", size = 6984598, upload-time = "2025-07-01T09:16:27.732Z" }, + { url = "https://files.pythonhosted.org/packages/0e/5a/a2f6773b64edb921a756eb0729068acad9fc5208a53f4a349396e9436721/pillow-12.0.0-cp311-cp311-macosx_10_10_x86_64.whl", hash = "sha256:0fd00cac9c03256c8b2ff58f162ebcd2587ad3e1f2e397eab718c47e24d231cc", size = 5289798, upload-time = "2025-10-15T18:21:47.763Z" }, + { url = "https://files.pythonhosted.org/packages/2e/05/069b1f8a2e4b5a37493da6c5868531c3f77b85e716ad7a590ef87d58730d/pillow-12.0.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a3475b96f5908b3b16c47533daaa87380c491357d197564e0ba34ae75c0f3257", size = 4650589, upload-time = "2025-10-15T18:21:49.515Z" }, + { url = "https://files.pythonhosted.org/packages/61/e3/2c820d6e9a36432503ead175ae294f96861b07600a7156154a086ba7111a/pillow-12.0.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:110486b79f2d112cf6add83b28b627e369219388f64ef2f960fef9ebaf54c642", size = 6230472, upload-time = "2025-10-15T18:21:51.052Z" }, + { url = "https://files.pythonhosted.org/packages/4f/89/63427f51c64209c5e23d4d52071c8d0f21024d3a8a487737caaf614a5795/pillow-12.0.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:5269cc1caeedb67e6f7269a42014f381f45e2e7cd42d834ede3c703a1d915fe3", size = 8033887, upload-time = "2025-10-15T18:21:52.604Z" }, + { url = "https://files.pythonhosted.org/packages/f6/1b/c9711318d4901093c15840f268ad649459cd81984c9ec9887756cca049a5/pillow-12.0.0-cp311-cp311-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:aa5129de4e174daccbc59d0a3b6d20eaf24417d59851c07ebb37aeb02947987c", size = 6343964, upload-time = "2025-10-15T18:21:54.619Z" }, + { url = "https://files.pythonhosted.org/packages/41/1e/db9470f2d030b4995083044cd8738cdd1bf773106819f6d8ba12597d5352/pillow-12.0.0-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:bee2a6db3a7242ea309aa7ee8e2780726fed67ff4e5b40169f2c940e7eb09227", size = 7034756, upload-time = "2025-10-15T18:21:56.151Z" }, + { url = "https://files.pythonhosted.org/packages/cc/b0/6177a8bdd5ee4ed87cba2de5a3cc1db55ffbbec6176784ce5bb75aa96798/pillow-12.0.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:90387104ee8400a7b4598253b4c406f8958f59fcf983a6cea2b50d59f7d63d0b", size = 6458075, upload-time = "2025-10-15T18:21:57.759Z" }, + { url = "https://files.pythonhosted.org/packages/bc/5e/61537aa6fa977922c6a03253a0e727e6e4a72381a80d63ad8eec350684f2/pillow-12.0.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:bc91a56697869546d1b8f0a3ff35224557ae7f881050e99f615e0119bf934b4e", size = 7125955, upload-time = "2025-10-15T18:21:59.372Z" }, + { url = "https://files.pythonhosted.org/packages/1f/3d/d5033539344ee3cbd9a4d69e12e63ca3a44a739eb2d4c8da350a3d38edd7/pillow-12.0.0-cp311-cp311-win32.whl", hash = "sha256:27f95b12453d165099c84f8a8bfdfd46b9e4bda9e0e4b65f0635430027f55739", size = 6298440, upload-time = "2025-10-15T18:22:00.982Z" }, + { url = "https://files.pythonhosted.org/packages/4d/42/aaca386de5cc8bd8a0254516957c1f265e3521c91515b16e286c662854c4/pillow-12.0.0-cp311-cp311-win_amd64.whl", hash = "sha256:b583dc9070312190192631373c6c8ed277254aa6e6084b74bdd0a6d3b221608e", size = 6999256, upload-time = "2025-10-15T18:22:02.617Z" }, + { url = "https://files.pythonhosted.org/packages/ba/f1/9197c9c2d5708b785f631a6dfbfa8eb3fb9672837cb92ae9af812c13b4ed/pillow-12.0.0-cp311-cp311-win_arm64.whl", hash = "sha256:759de84a33be3b178a64c8ba28ad5c135900359e85fb662bc6e403ad4407791d", size = 2436025, upload-time = "2025-10-15T18:22:04.598Z" }, + { url = "https://files.pythonhosted.org/packages/2c/90/4fcce2c22caf044e660a198d740e7fbc14395619e3cb1abad12192c0826c/pillow-12.0.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:53561a4ddc36facb432fae7a9d8afbfaf94795414f5cdc5fc52f28c1dca90371", size = 5249377, upload-time = "2025-10-15T18:22:05.993Z" }, + { url = "https://files.pythonhosted.org/packages/fd/e0/ed960067543d080691d47d6938ebccbf3976a931c9567ab2fbfab983a5dd/pillow-12.0.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:71db6b4c1653045dacc1585c1b0d184004f0d7e694c7b34ac165ca70c0838082", size = 4650343, upload-time = "2025-10-15T18:22:07.718Z" }, + { url = "https://files.pythonhosted.org/packages/e7/a1/f81fdeddcb99c044bf7d6faa47e12850f13cee0849537a7d27eeab5534d4/pillow-12.0.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:2fa5f0b6716fc88f11380b88b31fe591a06c6315e955c096c35715788b339e3f", size = 6232981, upload-time = "2025-10-15T18:22:09.287Z" }, + { url = "https://files.pythonhosted.org/packages/88/e1/9098d3ce341a8750b55b0e00c03f1630d6178f38ac191c81c97a3b047b44/pillow-12.0.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:82240051c6ca513c616f7f9da06e871f61bfd7805f566275841af15015b8f98d", size = 8041399, upload-time = "2025-10-15T18:22:10.872Z" }, + { url = "https://files.pythonhosted.org/packages/a7/62/a22e8d3b602ae8cc01446d0c57a54e982737f44b6f2e1e019a925143771d/pillow-12.0.0-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:55f818bd74fe2f11d4d7cbc65880a843c4075e0ac7226bc1a23261dbea531953", size = 6347740, upload-time = "2025-10-15T18:22:12.769Z" }, + { url = "https://files.pythonhosted.org/packages/4f/87/424511bdcd02c8d7acf9f65caa09f291a519b16bd83c3fb3374b3d4ae951/pillow-12.0.0-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b87843e225e74576437fd5b6a4c2205d422754f84a06942cfaf1dc32243e45a8", size = 7040201, upload-time = "2025-10-15T18:22:14.813Z" }, + { url = "https://files.pythonhosted.org/packages/dc/4d/435c8ac688c54d11755aedfdd9f29c9eeddf68d150fe42d1d3dbd2365149/pillow-12.0.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:c607c90ba67533e1b2355b821fef6764d1dd2cbe26b8c1005ae84f7aea25ff79", size = 6462334, upload-time = "2025-10-15T18:22:16.375Z" }, + { url = "https://files.pythonhosted.org/packages/2b/f2/ad34167a8059a59b8ad10bc5c72d4d9b35acc6b7c0877af8ac885b5f2044/pillow-12.0.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:21f241bdd5080a15bc86d3466a9f6074a9c2c2b314100dd896ac81ee6db2f1ba", size = 7134162, upload-time = "2025-10-15T18:22:17.996Z" }, + { url = "https://files.pythonhosted.org/packages/0c/b1/a7391df6adacf0a5c2cf6ac1cf1fcc1369e7d439d28f637a847f8803beb3/pillow-12.0.0-cp312-cp312-win32.whl", hash = "sha256:dd333073e0cacdc3089525c7df7d39b211bcdf31fc2824e49d01c6b6187b07d0", size = 6298769, upload-time = "2025-10-15T18:22:19.923Z" }, + { url = "https://files.pythonhosted.org/packages/a2/0b/d87733741526541c909bbf159e338dcace4f982daac6e5a8d6be225ca32d/pillow-12.0.0-cp312-cp312-win_amd64.whl", hash = "sha256:9fe611163f6303d1619bbcb653540a4d60f9e55e622d60a3108be0d5b441017a", size = 7001107, upload-time = "2025-10-15T18:22:21.644Z" }, + { url = "https://files.pythonhosted.org/packages/bc/96/aaa61ce33cc98421fb6088af2a03be4157b1e7e0e87087c888e2370a7f45/pillow-12.0.0-cp312-cp312-win_arm64.whl", hash = "sha256:7dfb439562f234f7d57b1ac6bc8fe7f838a4bd49c79230e0f6a1da93e82f1fad", size = 2436012, upload-time = "2025-10-15T18:22:23.621Z" }, + { url = "https://files.pythonhosted.org/packages/1d/b3/582327e6c9f86d037b63beebe981425d6811104cb443e8193824ef1a2f27/pillow-12.0.0-pp311-pypy311_pp73-macosx_10_15_x86_64.whl", hash = "sha256:b22bd8c974942477156be55a768f7aa37c46904c175be4e158b6a86e3a6b7ca8", size = 5215068, upload-time = "2025-10-15T18:23:59.594Z" }, + { url = "https://files.pythonhosted.org/packages/fd/d6/67748211d119f3b6540baf90f92fae73ae51d5217b171b0e8b5f7e5d558f/pillow-12.0.0-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:805ebf596939e48dbb2e4922a1d3852cfc25c38160751ce02da93058b48d252a", size = 4614994, upload-time = "2025-10-15T18:24:01.669Z" }, + { url = "https://files.pythonhosted.org/packages/2d/e1/f8281e5d844c41872b273b9f2c34a4bf64ca08905668c8ae730eedc7c9fa/pillow-12.0.0-pp311-pypy311_pp73-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:cae81479f77420d217def5f54b5b9d279804d17e982e0f2fa19b1d1e14ab5197", size = 5246639, upload-time = "2025-10-15T18:24:03.403Z" }, + { url = "https://files.pythonhosted.org/packages/94/5a/0d8ab8ffe8a102ff5df60d0de5af309015163bf710c7bb3e8311dd3b3ad0/pillow-12.0.0-pp311-pypy311_pp73-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:aeaefa96c768fc66818730b952a862235d68825c178f1b3ffd4efd7ad2edcb7c", size = 6986839, upload-time = "2025-10-15T18:24:05.344Z" }, + { url = "https://files.pythonhosted.org/packages/20/2e/3434380e8110b76cd9eb00a363c484b050f949b4bbe84ba770bb8508a02c/pillow-12.0.0-pp311-pypy311_pp73-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:09f2d0abef9e4e2f349305a4f8cc784a8a6c2f58a8c4892eea13b10a943bd26e", size = 5313505, upload-time = "2025-10-15T18:24:07.137Z" }, + { url = "https://files.pythonhosted.org/packages/57/ca/5a9d38900d9d74785141d6580950fe705de68af735ff6e727cb911b64740/pillow-12.0.0-pp311-pypy311_pp73-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:bdee52571a343d721fb2eb3b090a82d959ff37fc631e3f70422e0c2e029f3e76", size = 5963654, upload-time = "2025-10-15T18:24:09.579Z" }, + { url = "https://files.pythonhosted.org/packages/95/7e/f896623c3c635a90537ac093c6a618ebe1a90d87206e42309cb5d98a1b9e/pillow-12.0.0-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:b290fd8aa38422444d4b50d579de197557f182ef1068b75f5aa8558638b8d0a5", size = 6997850, upload-time = "2025-10-15T18:24:11.495Z" }, ] [[package]] @@ -4405,18 +4425,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/cd/f6/d1efedc0f9506e47699616e896d8efe39e8f0b6a7d1d590c3e97455ecf4a/polyfile_weave-0.5.7-py3-none-any.whl", hash = "sha256:880454788bc383408bf19eefd6d1c49a18b965d90c99bccb58f4da65870c82dd", size = 1655397, upload-time = "2025-09-22T19:21:09.142Z" }, ] -[[package]] -name = "pondpond" -version = "1.4.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "madoka" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/a6/9b/8411458ca8ce8b5b9b135e4a19823f1caf958ca9985883db104323492982/pondpond-1.4.1.tar.gz", hash = "sha256:8afa34b869d1434d21dd2ec12644abc3b1733fcda8fcf355300338a13a79bb7b", size = 15237, upload-time = "2024-03-01T07:08:06.756Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c4/d4/f18d6985157cc68f76469480182cbee2a03a45858456955acf57f9dcbb4c/pondpond-1.4.1-py3-none-any.whl", hash = "sha256:641028ead4e8018ca6de1220c660ddd6d6fbf62a60e72f410655dd0451d82880", size = 14498, upload-time = "2024-03-01T07:08:04.63Z" }, -] - [[package]] name = "portalocker" version = "2.10.1" @@ -4445,7 +4453,7 @@ wheels = [ [[package]] name = "posthog" -version = "6.7.6" +version = "7.0.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "backoff" }, @@ -4455,9 +4463,9 @@ dependencies = [ { name = "six" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/e2/ce/11d6fa30ab517018796e1d675498992da585479e7079770ec8fa99a61561/posthog-6.7.6.tar.gz", hash = "sha256:ee5c5ad04b857d96d9b7a4f715e23916a2f206bfcf25e5a9d328a3d27664b0d3", size = 119129, upload-time = "2025-09-22T18:11:12.365Z" } +sdist = { url = "https://files.pythonhosted.org/packages/15/4d/16d777528149cd0e06306973081b5b070506abcd0fe831c6cb6966260d59/posthog-7.0.0.tar.gz", hash = "sha256:94973227f5fe5e7d656d305ff48c8bff3d505fd1e78b6fcd7ccc9dfe8d3401c2", size = 126504, upload-time = "2025-11-11T18:13:06.986Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/de/84/586422d8861b5391c8414360b10f603c0b7859bb09ad688e64430ed0df7b/posthog-6.7.6-py3-none-any.whl", hash = "sha256:b09a7e65a042ec416c28874b397d3accae412a80a8b0ef3fa686fbffc99e4d4b", size = 137348, upload-time = "2025-09-22T18:11:10.807Z" }, + { url = "https://files.pythonhosted.org/packages/ca/9a/dc29b9ff4e5233a3c071b6b4c85dba96f4fcb9169c460bc81abd98555fb3/posthog-7.0.0-py3-none-any.whl", hash = "sha256:676d8a5197a17bf7bd00e31020a5f232988f249f57aab532f0d01c6243835934", size = 144727, upload-time = "2025-11-11T18:13:05.444Z" }, ] [[package]] @@ -4539,18 +4547,16 @@ wheels = [ [[package]] name = "psutil" -version = "7.1.0" +version = "7.1.3" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/b3/31/4723d756b59344b643542936e37a31d1d3204bcdc42a7daa8ee9eb06fb50/psutil-7.1.0.tar.gz", hash = "sha256:655708b3c069387c8b77b072fc429a57d0e214221d01c0a772df7dfedcb3bcd2", size = 497660, upload-time = "2025-09-17T20:14:52.902Z" } +sdist = { url = "https://files.pythonhosted.org/packages/e1/88/bdd0a41e5857d5d703287598cbf08dad90aed56774ea52ae071bae9071b6/psutil-7.1.3.tar.gz", hash = "sha256:6c86281738d77335af7aec228328e944b30930899ea760ecf33a4dba66be5e74", size = 489059, upload-time = "2025-11-02T12:25:54.619Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/46/62/ce4051019ee20ce0ed74432dd73a5bb087a6704284a470bb8adff69a0932/psutil-7.1.0-cp36-abi3-macosx_10_9_x86_64.whl", hash = "sha256:76168cef4397494250e9f4e73eb3752b146de1dd950040b29186d0cce1d5ca13", size = 245242, upload-time = "2025-09-17T20:14:56.126Z" }, - { url = "https://files.pythonhosted.org/packages/38/61/f76959fba841bf5b61123fbf4b650886dc4094c6858008b5bf73d9057216/psutil-7.1.0-cp36-abi3-macosx_11_0_arm64.whl", hash = "sha256:5d007560c8c372efdff9e4579c2846d71de737e4605f611437255e81efcca2c5", size = 246682, upload-time = "2025-09-17T20:14:58.25Z" }, - { url = "https://files.pythonhosted.org/packages/88/7a/37c99d2e77ec30d63398ffa6a660450b8a62517cabe44b3e9bae97696e8d/psutil-7.1.0-cp36-abi3-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:22e4454970b32472ce7deaa45d045b34d3648ce478e26a04c7e858a0a6e75ff3", size = 287994, upload-time = "2025-09-17T20:14:59.901Z" }, - { url = "https://files.pythonhosted.org/packages/9d/de/04c8c61232f7244aa0a4b9a9fbd63a89d5aeaf94b2fc9d1d16e2faa5cbb0/psutil-7.1.0-cp36-abi3-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8c70e113920d51e89f212dd7be06219a9b88014e63a4cec69b684c327bc474e3", size = 291163, upload-time = "2025-09-17T20:15:01.481Z" }, - { url = "https://files.pythonhosted.org/packages/f4/58/c4f976234bf6d4737bc8c02a81192f045c307b72cf39c9e5c5a2d78927f6/psutil-7.1.0-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7d4a113425c037300de3ac8b331637293da9be9713855c4fc9d2d97436d7259d", size = 293625, upload-time = "2025-09-17T20:15:04.492Z" }, - { url = "https://files.pythonhosted.org/packages/79/87/157c8e7959ec39ced1b11cc93c730c4fb7f9d408569a6c59dbd92ceb35db/psutil-7.1.0-cp37-abi3-win32.whl", hash = "sha256:09ad740870c8d219ed8daae0ad3b726d3bf9a028a198e7f3080f6a1888b99bca", size = 244812, upload-time = "2025-09-17T20:15:07.462Z" }, - { url = "https://files.pythonhosted.org/packages/bf/e9/b44c4f697276a7a95b8e94d0e320a7bf7f3318521b23de69035540b39838/psutil-7.1.0-cp37-abi3-win_amd64.whl", hash = "sha256:57f5e987c36d3146c0dd2528cd42151cf96cd359b9d67cfff836995cc5df9a3d", size = 247965, upload-time = "2025-09-17T20:15:09.673Z" }, - { url = "https://files.pythonhosted.org/packages/26/65/1070a6e3c036f39142c2820c4b52e9243246fcfc3f96239ac84472ba361e/psutil-7.1.0-cp37-abi3-win_arm64.whl", hash = "sha256:6937cb68133e7c97b6cc9649a570c9a18ba0efebed46d8c5dae4c07fa1b67a07", size = 244971, upload-time = "2025-09-17T20:15:12.262Z" }, + { url = "https://files.pythonhosted.org/packages/ef/94/46b9154a800253e7ecff5aaacdf8ebf43db99de4a2dfa18575b02548654e/psutil-7.1.3-cp36-abi3-macosx_10_9_x86_64.whl", hash = "sha256:2bdbcd0e58ca14996a42adf3621a6244f1bb2e2e528886959c72cf1e326677ab", size = 238359, upload-time = "2025-11-02T12:26:25.284Z" }, + { url = "https://files.pythonhosted.org/packages/68/3a/9f93cff5c025029a36d9a92fef47220ab4692ee7f2be0fba9f92813d0cb8/psutil-7.1.3-cp36-abi3-macosx_11_0_arm64.whl", hash = "sha256:bc31fa00f1fbc3c3802141eede66f3a2d51d89716a194bf2cd6fc68310a19880", size = 239171, upload-time = "2025-11-02T12:26:27.23Z" }, + { url = "https://files.pythonhosted.org/packages/ce/b1/5f49af514f76431ba4eea935b8ad3725cdeb397e9245ab919dbc1d1dc20f/psutil-7.1.3-cp36-abi3-manylinux2010_x86_64.manylinux_2_12_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:3bb428f9f05c1225a558f53e30ccbad9930b11c3fc206836242de1091d3e7dd3", size = 263261, upload-time = "2025-11-02T12:26:29.48Z" }, + { url = "https://files.pythonhosted.org/packages/e0/95/992c8816a74016eb095e73585d747e0a8ea21a061ed3689474fabb29a395/psutil-7.1.3-cp36-abi3-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:56d974e02ca2c8eb4812c3f76c30e28836fffc311d55d979f1465c1feeb2b68b", size = 264635, upload-time = "2025-11-02T12:26:31.74Z" }, + { url = "https://files.pythonhosted.org/packages/55/4c/c3ed1a622b6ae2fd3c945a366e64eb35247a31e4db16cf5095e269e8eb3c/psutil-7.1.3-cp37-abi3-win_amd64.whl", hash = "sha256:f39c2c19fe824b47484b96f9692932248a54c43799a84282cfe58d05a6449efd", size = 247633, upload-time = "2025-11-02T12:26:33.887Z" }, + { url = "https://files.pythonhosted.org/packages/c9/ad/33b2ccec09bf96c2b2ef3f9a6f66baac8253d7565d8839e024a6b905d45d/psutil-7.1.3-cp37-abi3-win_arm64.whl", hash = "sha256:bd0d69cee829226a761e92f28140bec9a5ee9d5b4fb4b0cc589068dbfff559b1", size = 244608, upload-time = "2025-11-02T12:26:36.136Z" }, ] [[package]] @@ -4570,8 +4576,10 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/48/89/3fdb5902bdab8868bbedc1c6e6023a4e08112ceac5db97fc2012060e0c9a/psycopg2_binary-2.9.11-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:2e164359396576a3cc701ba8af4751ae68a07235d7a380c631184a611220d9a4", size = 4410955, upload-time = "2025-10-10T11:11:21.21Z" }, { url = "https://files.pythonhosted.org/packages/ce/24/e18339c407a13c72b336e0d9013fbbbde77b6fd13e853979019a1269519c/psycopg2_binary-2.9.11-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:d57c9c387660b8893093459738b6abddbb30a7eab058b77b0d0d1c7d521ddfd7", size = 4468007, upload-time = "2025-10-10T11:11:24.831Z" }, { url = "https://files.pythonhosted.org/packages/91/7e/b8441e831a0f16c159b5381698f9f7f7ed54b77d57bc9c5f99144cc78232/psycopg2_binary-2.9.11-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:2c226ef95eb2250974bf6fa7a842082b31f68385c4f3268370e3f3870e7859ee", size = 4165012, upload-time = "2025-10-10T11:11:29.51Z" }, + { url = "https://files.pythonhosted.org/packages/0d/61/4aa89eeb6d751f05178a13da95516c036e27468c5d4d2509bb1e15341c81/psycopg2_binary-2.9.11-cp311-cp311-manylinux_2_38_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:a311f1edc9967723d3511ea7d2708e2c3592e3405677bf53d5c7246753591fbb", size = 3981881, upload-time = "2025-10-30T02:55:07.332Z" }, { url = "https://files.pythonhosted.org/packages/76/a1/2f5841cae4c635a9459fe7aca8ed771336e9383b6429e05c01267b0774cf/psycopg2_binary-2.9.11-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:ebb415404821b6d1c47353ebe9c8645967a5235e6d88f914147e7fd411419e6f", size = 3650985, upload-time = "2025-10-10T11:11:34.975Z" }, { url = "https://files.pythonhosted.org/packages/84/74/4defcac9d002bca5709951b975173c8c2fa968e1a95dc713f61b3a8d3b6a/psycopg2_binary-2.9.11-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:f07c9c4a5093258a03b28fab9b4f151aa376989e7f35f855088234e656ee6a94", size = 3296039, upload-time = "2025-10-10T11:11:40.432Z" }, + { url = "https://files.pythonhosted.org/packages/6d/c2/782a3c64403d8ce35b5c50e1b684412cf94f171dc18111be8c976abd2de1/psycopg2_binary-2.9.11-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:00ce1830d971f43b667abe4a56e42c1e2d594b32da4802e44a73bacacb25535f", size = 3043477, upload-time = "2025-10-30T02:55:11.182Z" }, { url = "https://files.pythonhosted.org/packages/c8/31/36a1d8e702aa35c38fc117c2b8be3f182613faa25d794b8aeaab948d4c03/psycopg2_binary-2.9.11-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:cffe9d7697ae7456649617e8bb8d7a45afb71cd13f7ab22af3e5c61f04840908", size = 3345842, upload-time = "2025-10-10T11:11:45.366Z" }, { url = "https://files.pythonhosted.org/packages/6e/b4/a5375cda5b54cb95ee9b836930fea30ae5a8f14aa97da7821722323d979b/psycopg2_binary-2.9.11-cp311-cp311-win_amd64.whl", hash = "sha256:304fd7b7f97eef30e91b8f7e720b3db75fee010b520e434ea35ed1ff22501d03", size = 2713894, upload-time = "2025-10-10T11:11:48.775Z" }, { url = "https://files.pythonhosted.org/packages/d8/91/f870a02f51be4a65987b45a7de4c2e1897dd0d01051e2b559a38fa634e3e/psycopg2_binary-2.9.11-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:be9b840ac0525a283a96b556616f5b4820e0526addb8dcf6525a0fa162730be4", size = 3756603, upload-time = "2025-10-10T11:11:52.213Z" }, @@ -4579,8 +4587,10 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/2d/75/364847b879eb630b3ac8293798e380e441a957c53657995053c5ec39a316/psycopg2_binary-2.9.11-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:ab8905b5dcb05bf3fb22e0cf90e10f469563486ffb6a96569e51f897c750a76a", size = 4411159, upload-time = "2025-10-10T11:12:00.49Z" }, { url = "https://files.pythonhosted.org/packages/6f/a0/567f7ea38b6e1c62aafd58375665a547c00c608a471620c0edc364733e13/psycopg2_binary-2.9.11-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:bf940cd7e7fec19181fdbc29d76911741153d51cab52e5c21165f3262125685e", size = 4468234, upload-time = "2025-10-10T11:12:04.892Z" }, { url = "https://files.pythonhosted.org/packages/30/da/4e42788fb811bbbfd7b7f045570c062f49e350e1d1f3df056c3fb5763353/psycopg2_binary-2.9.11-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:fa0f693d3c68ae925966f0b14b8edda71696608039f4ed61b1fe9ffa468d16db", size = 4166236, upload-time = "2025-10-10T11:12:11.674Z" }, + { url = "https://files.pythonhosted.org/packages/3c/94/c1777c355bc560992af848d98216148be5f1be001af06e06fc49cbded578/psycopg2_binary-2.9.11-cp312-cp312-manylinux_2_38_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:a1cf393f1cdaf6a9b57c0a719a1068ba1069f022a59b8b1fe44b006745b59757", size = 3983083, upload-time = "2025-10-30T02:55:15.73Z" }, { url = "https://files.pythonhosted.org/packages/bd/42/c9a21edf0e3daa7825ed04a4a8588686c6c14904344344a039556d78aa58/psycopg2_binary-2.9.11-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:ef7a6beb4beaa62f88592ccc65df20328029d721db309cb3250b0aae0fa146c3", size = 3652281, upload-time = "2025-10-10T11:12:17.713Z" }, { url = "https://files.pythonhosted.org/packages/12/22/dedfbcfa97917982301496b6b5e5e6c5531d1f35dd2b488b08d1ebc52482/psycopg2_binary-2.9.11-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:31b32c457a6025e74d233957cc9736742ac5a6cb196c6b68499f6bb51390bd6a", size = 3298010, upload-time = "2025-10-10T11:12:22.671Z" }, + { url = "https://files.pythonhosted.org/packages/66/ea/d3390e6696276078bd01b2ece417deac954dfdd552d2edc3d03204416c0c/psycopg2_binary-2.9.11-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:edcb3aeb11cb4bf13a2af3c53a15b3d612edeb6409047ea0b5d6a21a9d744b34", size = 3044641, upload-time = "2025-10-30T02:55:19.929Z" }, { url = "https://files.pythonhosted.org/packages/12/9a/0402ded6cbd321da0c0ba7d34dc12b29b14f5764c2fc10750daa38e825fc/psycopg2_binary-2.9.11-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:62b6d93d7c0b61a1dd6197d208ab613eb7dcfdcca0a49c42ceb082257991de9d", size = 3347940, upload-time = "2025-10-10T11:12:26.529Z" }, { url = "https://files.pythonhosted.org/packages/b1/d2/99b55e85832ccde77b211738ff3925a5d73ad183c0b37bcbbe5a8ff04978/psycopg2_binary-2.9.11-cp312-cp312-win_amd64.whl", hash = "sha256:b33fabeb1fde21180479b2d4667e994de7bbf0eec22832ba5d9b5e4cf65b6c6d", size = 2714147, upload-time = "2025-10-10T11:12:29.535Z" }, ] @@ -4791,7 +4801,7 @@ crypto = [ [[package]] name = "pymilvus" -version = "2.5.16" +version = "2.5.17" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "grpcio" }, @@ -4802,9 +4812,9 @@ dependencies = [ { name = "setuptools" }, { name = "ujson" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/24/e2/5613bc7b2af0ccd760177ca4255243c284cfc0f2cba3f10ff63325c4ca34/pymilvus-2.5.16.tar.gz", hash = "sha256:65f56b81806bc217cca3cf29b70a27d053dea4b1ffada910cf63a38f96381618", size = 1280614, upload-time = "2025-09-19T07:02:14.747Z" } +sdist = { url = "https://files.pythonhosted.org/packages/dc/85/91828a9282bb7f9b210c0a93831979c5829cba5533ac12e87014b6e2208b/pymilvus-2.5.17.tar.gz", hash = "sha256:48ff55db9598e1b4cc25f4fe645b00d64ebcfb03f79f9f741267fc2a35526d43", size = 1281485, upload-time = "2025-11-10T03:24:53.058Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/c6/09/b67a55abee0a53ea50ba0de0cba6e1c0f7ca7ce2c15ffd6f40c059c25e88/pymilvus-2.5.16-py3-none-any.whl", hash = "sha256:76258a324f19c60fee247467e11cd7d6f35a64d2a9c753f5d7b1a5fa15dd6c8a", size = 243272, upload-time = "2025-09-19T07:02:12.443Z" }, + { url = "https://files.pythonhosted.org/packages/59/44/ee0c64617f58c123f570293f36b40f7b56fc123a2aa9573aa22e6ff0fb86/pymilvus-2.5.17-py3-none-any.whl", hash = "sha256:a43d36f2e5f793040917d35858d1ed2532307b7dfb03bc3eaf813aac085bc5a4", size = 244036, upload-time = "2025-11-10T03:24:51.496Z" }, ] [[package]] @@ -4832,7 +4842,7 @@ wheels = [ [[package]] name = "pyobvector" -version = "0.2.16" +version = "0.2.19" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "aiomysql" }, @@ -4842,18 +4852,17 @@ dependencies = [ { name = "sqlalchemy" }, { name = "sqlglot" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/b4/c1/a418b1e10627d3b9d54c7bed460d90bd44c9e9c20be801d6606e9fa3fe01/pyobvector-0.2.16.tar.gz", hash = "sha256:de44588e75de616dee7a9cc5d5c016aeb3390a90fe52f99d9b8ad2476294f6c2", size = 39602, upload-time = "2025-09-03T08:52:23.932Z" } +sdist = { url = "https://files.pythonhosted.org/packages/19/9a/03da0d77f6033694ab7e7214abdd48c372102a185142db880ba00d6a6172/pyobvector-0.2.19.tar.gz", hash = "sha256:5e6847f08679cf6ded800b5b8ae89353173c33f5d90fd1392f55e5fafa4fb886", size = 46314, upload-time = "2025-11-10T08:30:10.186Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/83/7b/c103cca858de87476db5e7c7f0f386b429c3057a7291155c70560b15d951/pyobvector-0.2.16-py3-none-any.whl", hash = "sha256:0710272e5c807a6d0bdeee96972cdc9fdca04fc4b40c2d1260b08ff8b79190ef", size = 52664, upload-time = "2025-09-03T08:52:22.372Z" }, + { url = "https://files.pythonhosted.org/packages/72/48/d6b60ae86a2a2c0c607a33e0c8fc9e469500e06e5bb07ea7e9417910f458/pyobvector-0.2.19-py3-none-any.whl", hash = "sha256:0a6b93c950722ecbab72571e0ab81d0f8f4d1f52df9c25c00693392477e45e4b", size = 59886, upload-time = "2025-11-10T08:30:08.627Z" }, ] [[package]] name = "pypandoc" -version = "1.15" +version = "1.16" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/e1/88/26e650d053df5f3874aa3c05901a14166ce3271f58bfe114fd776987efbd/pypandoc-1.15.tar.gz", hash = "sha256:ea25beebe712ae41d63f7410c08741a3cab0e420f6703f95bc9b3a749192ce13", size = 32940, upload-time = "2025-01-08T17:39:58.705Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/61/06/0763e0ccc81754d3eadb21b2cb86cf21bdedc9b52698c2ad6785db7f0a4e/pypandoc-1.15-py3-none-any.whl", hash = "sha256:4ededcc76c8770f27aaca6dff47724578428eca84212a31479403a9731fc2b16", size = 21321, upload-time = "2025-01-08T17:39:09.928Z" }, + { url = "https://files.pythonhosted.org/packages/24/77/af1fc54740a0712988f9518e629d38edc7b8ffccd7549203f19c3d8a2db6/pypandoc-1.16-py3-none-any.whl", hash = "sha256:868f390d48388743e7a5885915cbbaa005dea36a825ecdfd571f8c523416c822", size = 19425, upload-time = "2025-11-08T15:44:38.429Z" }, ] [[package]] @@ -4867,11 +4876,11 @@ wheels = [ [[package]] name = "pypdf" -version = "6.1.1" +version = "6.2.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/a6/85/4c0f12616db83c2e3ef580c3cfa98bd082e88fc8d02e136bad3bede1e3fa/pypdf-6.1.1.tar.gz", hash = "sha256:10f44d49bf2a82e54c3c5ba3cdcbb118f2a44fc57df8ce51d6fb9b1ed9bfbe8b", size = 5074507, upload-time = "2025-09-28T13:29:16.165Z" } +sdist = { url = "https://files.pythonhosted.org/packages/4e/2b/8795ec0378384000b0a37a2b5e6d67fa3d84802945aa2c612a78a784d7d4/pypdf-6.2.0.tar.gz", hash = "sha256:46b4d8495d68ae9c818e7964853cd9984e6a04c19fe7112760195395992dce48", size = 5272001, upload-time = "2025-11-09T11:10:41.911Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/07/ed/adae13756d9dabdddee483fc7712905bb5585fbf6e922b1a19aca3a29cd1/pypdf-6.1.1-py3-none-any.whl", hash = "sha256:7781f99493208a37a7d4275601d883e19af24e62a525c25844d22157c2e4cde7", size = 323455, upload-time = "2025-09-28T13:29:14.392Z" }, + { url = "https://files.pythonhosted.org/packages/de/ba/743ddcaf1a8fb439342399645921e2cf2c600464cba5531a11f1cc0822b6/pypdf-6.2.0-py3-none-any.whl", hash = "sha256:4c0f3e62677217a777ab79abe22bf1285442d70efabf552f61c7a03b6f5c569f", size = 326592, upload-time = "2025-11-09T11:10:39.941Z" }, ] [[package]] @@ -4997,46 +5006,46 @@ wheels = [ [[package]] name = "python-calamine" -version = "0.5.3" +version = "0.5.4" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/f3/ca/295b37a97275d53f072c7307c9d0c4bfec565d3d74157e7fe336ea18de0a/python_calamine-0.5.3.tar.gz", hash = "sha256:b4529c955fa64444184630d5bc8c82c472d1cf6bfe631f0a7bfc5e4802d4e996", size = 130874, upload-time = "2025-09-08T05:41:27.18Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/fb/e4/bb2c84aee0909868e4cf251a4813d82ba9bcb97e772e28a6746fb7133e15/python_calamine-0.5.3-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:522dcad340efef3114d3bc4081e8f12d3a471455038df6b20f199e14b3f1a1df", size = 847891, upload-time = "2025-09-08T05:38:58.681Z" }, - { url = "https://files.pythonhosted.org/packages/00/aa/7dab22cc2d7aa869e9bce2426fd53cefea19010496116aa0b8a1a658768d/python_calamine-0.5.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e2c667dc044eefc233db115e96f77772c89ec61f054ba94ef2faf71e92ce2b23", size = 820897, upload-time = "2025-09-08T05:39:00.123Z" }, - { url = "https://files.pythonhosted.org/packages/93/95/aa82413e119365fb7a0fd1345879d22982638affab96ff9bbf4f22f6e403/python_calamine-0.5.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7f28cc65ad7da395e0a885c989a1872f9a1939d4c3c846a7bd189b70d7255640", size = 889556, upload-time = "2025-09-08T05:39:01.595Z" }, - { url = "https://files.pythonhosted.org/packages/ae/ab/63bb196a121f6ede57cbb8012e0b642162da088e9e9419531215ab528823/python_calamine-0.5.3-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:8642f3e9b0501e0a639913319107ce6a4fa350919d428c4b06129b1917fa12f8", size = 882632, upload-time = "2025-09-08T05:39:03.426Z" }, - { url = "https://files.pythonhosted.org/packages/6b/60/236db1deecf7a46454c3821b9315a230ad6247f6e823ef948a6b591001cd/python_calamine-0.5.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:88c6b7c9962bec16fcfb326c271077a2a9350b8a08e5cfda2896014d8cd04c84", size = 1032778, upload-time = "2025-09-08T05:39:04.939Z" }, - { url = "https://files.pythonhosted.org/packages/be/18/d143b8c3ee609354859442458e749a0f00086d11b1c003e6d0a61b1f6573/python_calamine-0.5.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:229dd29b0a61990a1c7763a9fadc40a56f8674e6dd5700cb6761cd8e8a731a88", size = 932695, upload-time = "2025-09-08T05:39:06.471Z" }, - { url = "https://files.pythonhosted.org/packages/ee/25/a50886897b6fbf74c550dcaefd9e25487c02514bbdd7ec405fd44c8b52d2/python_calamine-0.5.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:12ac37001bebcb0016770248acfdf3adba2ded352b69ee57924145cb5b6daa0e", size = 905138, upload-time = "2025-09-08T05:39:07.94Z" }, - { url = "https://files.pythonhosted.org/packages/72/37/7f30152f4d5053eb1390fede14c3d8cce6bd6d3383f056a7e14fdf2724b3/python_calamine-0.5.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:1ee817d2d4de7cccf3d50a38a37442af83985cc4a96ca5d511852109c3b71d87", size = 944337, upload-time = "2025-09-08T05:39:09.493Z" }, - { url = "https://files.pythonhosted.org/packages/77/9f/4c44d49ad1177f7730f089bb2e6df555e41319241c90529adb5d5a2bec2e/python_calamine-0.5.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:592a6e15ca1e8cc644bf227f3afa2f6e8ba2eece7d51e6237a84b8269de47734", size = 1067713, upload-time = "2025-09-08T05:39:11.684Z" }, - { url = "https://files.pythonhosted.org/packages/33/b5/bf61a39af88f78562f3a2ca137f7db95d7495e034658f44ee7381014a9a4/python_calamine-0.5.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:51d7f63e4a74fc504398e970a06949f44306078e1cdf112543a60c3745f97f77", size = 1075283, upload-time = "2025-09-08T05:39:13.425Z" }, - { url = "https://files.pythonhosted.org/packages/a4/50/6b96c45c43a7bb78359de9b9ebf78c91148d9448ab3b021a81df4ffdddfe/python_calamine-0.5.3-cp311-cp311-win32.whl", hash = "sha256:54747fd59956cf10e170c85f063be21d1016e85551ba6dea20ac66f21bcb6d1d", size = 669120, upload-time = "2025-09-08T05:39:14.848Z" }, - { url = "https://files.pythonhosted.org/packages/11/3f/ff15f5651bb84199660a4f024b32f9bcb948c1e73d5d533ec58fab31c36d/python_calamine-0.5.3-cp311-cp311-win_amd64.whl", hash = "sha256:49f5f311e4040e251b65f2a2c3493e338f51b1ba30c632f41f8151f95071ed65", size = 713536, upload-time = "2025-09-08T05:39:16.317Z" }, - { url = "https://files.pythonhosted.org/packages/d9/1b/e33ea19a1881934d8dc1c6cbc3dffeef7288cbd2c313fb1249f07bf9c76d/python_calamine-0.5.3-cp311-cp311-win_arm64.whl", hash = "sha256:1201908dc0981e3684ab916bebc83399657a10118f4003310e465ab07dd67d09", size = 679691, upload-time = "2025-09-08T05:39:17.783Z" }, - { url = "https://files.pythonhosted.org/packages/05/24/f6e3369be221baa6a50476b8a02f5100980ae487a630d80d4983b4c73879/python_calamine-0.5.3-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:b9a78e471bc02d3f76c294bf996562a9d0fbf2ad0a49d628330ba247865190f1", size = 844280, upload-time = "2025-09-08T05:39:19.991Z" }, - { url = "https://files.pythonhosted.org/packages/e7/32/f9b689fe40616376457d1a6fd5ab84834066db31fa5ffd10a5b02f996a44/python_calamine-0.5.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:bcbd277a4d0a0108aa2f5126a89ca3f2bb18d0bec7ba7d614da02a4556d18ef2", size = 814054, upload-time = "2025-09-08T05:39:21.888Z" }, - { url = "https://files.pythonhosted.org/packages/f7/26/a07bb6993ae0a524251060397edc710af413dbb175d56f1e1bbc7a2c39c9/python_calamine-0.5.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:04e6b68b26346f559a086bb84c960d4e9ddc79be8c3499752c1ba96051fea98f", size = 889447, upload-time = "2025-09-08T05:39:23.332Z" }, - { url = "https://files.pythonhosted.org/packages/d8/79/5902d00658e2dd4efe3a4062b710a7eaa6082001c199717468fbcd8cef69/python_calamine-0.5.3-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:e60ebeafebf66889753bfad0055edaa38068663961bb9a18e9f89aef2c9cec50", size = 883540, upload-time = "2025-09-08T05:39:25.15Z" }, - { url = "https://files.pythonhosted.org/packages/d0/85/6299c909fcbba0663b527b82c87d204372e6f469b4ed5602f7bc1f7f1103/python_calamine-0.5.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2d9da11edb40e9d2fb214fcf575be8004b44b1b407930eceb2458f1a84be634f", size = 1034891, upload-time = "2025-09-08T05:39:26.666Z" }, - { url = "https://files.pythonhosted.org/packages/65/2c/d0cfd9161b3404528bfba9fe000093be19f2c83ede42c255da4ebfd4da17/python_calamine-0.5.3-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:44d22bc52fe26b72a6dc07ab8a167d5d97aeb28282957f52b930e92106a35e3c", size = 935055, upload-time = "2025-09-08T05:39:28.727Z" }, - { url = "https://files.pythonhosted.org/packages/b8/69/420c382535d1aca9af6bc929c78ad6b9f8416312aa4955b7977f5f864082/python_calamine-0.5.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7b9ace667e04ea6631a0ada0e43dbc796c56e0d021f04bd64cdacb44de4504da", size = 904143, upload-time = "2025-09-08T05:39:30.23Z" }, - { url = "https://files.pythonhosted.org/packages/d8/2b/19cc87654f9c85fbb6265a7ebe92cf0f649c308f0cf8f262b5c3de754d19/python_calamine-0.5.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:7ec0da29de7366258de2eb765a90b9e9fbe9f9865772f3609dacff302b894393", size = 948890, upload-time = "2025-09-08T05:39:31.779Z" }, - { url = "https://files.pythonhosted.org/packages/18/e8/3547cb72d3a0f67c173ca07d9137046f2a6c87fdc31316b10e2d7d851f2a/python_calamine-0.5.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:4bba5adf123200503e6c07c667a8ce82c3b62ba02f9b3e99205be24fc73abc49", size = 1067802, upload-time = "2025-09-08T05:39:33.264Z" }, - { url = "https://files.pythonhosted.org/packages/cb/69/31ab3e8010cbed814b5fcdb2ace43e5b76d6464f8abb1dfab9191416ca3d/python_calamine-0.5.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:f4c49bc58f3cfd1e9595a05cab7e71aa94f6cff5bf3916de2b87cdaa9b4ce9a3", size = 1074607, upload-time = "2025-09-08T05:39:34.803Z" }, - { url = "https://files.pythonhosted.org/packages/c4/40/112d113d974bee5fff564e355b01df5bd524dbd5820c913c9dae574fe80a/python_calamine-0.5.3-cp312-cp312-win32.whl", hash = "sha256:42315463e139f5e44f4dedb9444fa0971c51e82573e872428050914f0dec4194", size = 669578, upload-time = "2025-09-08T05:39:36.305Z" }, - { url = "https://files.pythonhosted.org/packages/3e/87/0af1cf4ad01a2df273cfd3abb7efaba4fba50395b98f5e871cee016d4f09/python_calamine-0.5.3-cp312-cp312-win_amd64.whl", hash = "sha256:8a24bd4c72bd984311f5ebf2e17a8aa3ce4e5ae87eda517c61c3507db8c045de", size = 713021, upload-time = "2025-09-08T05:39:37.942Z" }, - { url = "https://files.pythonhosted.org/packages/5d/4e/6ed2ed3bb4c4c479e85d3444742f101f7b3099db1819e422bf861cf9923b/python_calamine-0.5.3-cp312-cp312-win_arm64.whl", hash = "sha256:e4a713e56d3cca752d1a7d6a00dca81b224e2e1a0567d370bc0db537e042d6b0", size = 679615, upload-time = "2025-09-08T05:39:39.487Z" }, - { url = "https://files.pythonhosted.org/packages/df/d4/fbe043cf6310d831e9af07772be12ec977148e31ec404b37bcb20c471ab0/python_calamine-0.5.3-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:a74fb8379a9caff19c5fe5ac637fcb86ca56698d1e06f5773d5612dea5254c2f", size = 849328, upload-time = "2025-09-08T05:41:10.129Z" }, - { url = "https://files.pythonhosted.org/packages/a4/b3/d1258e3e7f31684421d75f9bde83ccc14064fbfeaf1e26e4f4207f1cf704/python_calamine-0.5.3-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:37efba7ed0234ea73e8d7433c6feabedefdcc4edfdd54546ee28709b950809da", size = 822183, upload-time = "2025-09-08T05:41:11.936Z" }, - { url = "https://files.pythonhosted.org/packages/bb/45/cadba216db106c7de7cd5210efb6e6adbf1c3a5d843ed255e039f3f6d7c7/python_calamine-0.5.3-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3449b4766d19fa33087a4a9eddae097539661f9678ea4160d9c3888d6ba93e01", size = 891063, upload-time = "2025-09-08T05:41:13.644Z" }, - { url = "https://files.pythonhosted.org/packages/ff/a6/d710452f6f32fd2483aaaf3a12fdbb888f7f89d5fcad287eeed6daf0f6c6/python_calamine-0.5.3-pp311-pypy311_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:683f398d800104930345282905088c095969ca26145f86f35681061dee6eb881", size = 884047, upload-time = "2025-09-08T05:41:15.339Z" }, - { url = "https://files.pythonhosted.org/packages/d6/bc/8fead09adbd8069022ae39b97879cb90acbc02d768488ac8d76423a85783/python_calamine-0.5.3-pp311-pypy311_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1b6bfdd64204ad6b9f3132951246b7eb9986a55dc10a805240c7751a1f3bc7d9", size = 1031566, upload-time = "2025-09-08T05:41:17.143Z" }, - { url = "https://files.pythonhosted.org/packages/d0/cd/7259e9a181f31d861cb8e0d98f8e0f17fad2bead885b48a17e8049fcecb5/python_calamine-0.5.3-pp311-pypy311_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81c3654edac2eaf84066a90ea31b544fdeed8847a1ad8a8323118448522b84c9", size = 933438, upload-time = "2025-09-08T05:41:18.822Z" }, - { url = "https://files.pythonhosted.org/packages/39/39/bd737005731591066d6a7d1c4ce1e8d72befe32e028ba11df410937b2aec/python_calamine-0.5.3-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b8ff1a449545d9a4b5a72c4e204d16b26477b82484e9b2010935fa63ad66c607", size = 905036, upload-time = "2025-09-08T05:41:20.555Z" }, - { url = "https://files.pythonhosted.org/packages/b5/20/94a4af86b11ee318770e72081c89545e99b78cdbbe05227e083d92c55c52/python_calamine-0.5.3-pp311-pypy311_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:340046e7c937d02bb314e09fda8c0dc2e11ef2692e60fb5956fbd091b6d82725", size = 946582, upload-time = "2025-09-08T05:41:22.307Z" }, - { url = "https://files.pythonhosted.org/packages/4f/3b/2448580b510a28718802c51f80fbc4d3df668a6824817e7024853b715813/python_calamine-0.5.3-pp311-pypy311_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:421947eef983e0caa245f37ac81234e7e62663bdf423bbee5013a469a3bf632c", size = 1068960, upload-time = "2025-09-08T05:41:23.989Z" }, - { url = "https://files.pythonhosted.org/packages/23/a4/5b13bfaa355d6e20aae87c1230aa5e40403c14386bd9806491ac3a89b840/python_calamine-0.5.3-pp311-pypy311_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:e970101cc4c0e439b14a5f697a43eb508343fd0dc604c5bb5145e5774c4eb0c8", size = 1075022, upload-time = "2025-09-08T05:41:25.697Z" }, + { url = "https://files.pythonhosted.org/packages/25/1a/ff59788a7e8bfeded91a501abdd068dc7e2f5865ee1a55432133b0f7f08c/python_calamine-0.5.4-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:944bcc072aca29d346456b4e42675c4831c52c25641db3e976c6013cdd07d4cd", size = 854308, upload-time = "2025-10-21T07:10:55.17Z" }, + { url = "https://files.pythonhosted.org/packages/24/7d/33fc441a70b771093d10fa5086831be289766535cbcb2b443ff1d5e549d8/python_calamine-0.5.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e637382e50cabc263a37eda7a3cd33f054271e4391a304f68cecb2e490827533", size = 830841, upload-time = "2025-10-21T07:10:57.353Z" }, + { url = "https://files.pythonhosted.org/packages/0f/38/b5b25e6ce0a983c9751fb026bd8c5d77eb81a775948cc3d9ce2b18b2fc91/python_calamine-0.5.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1b2a31d1e711c5661b4f04efd89975d311788bd9a43a111beff74d7c4c8f8d7a", size = 898287, upload-time = "2025-10-21T07:10:58.977Z" }, + { url = "https://files.pythonhosted.org/packages/0f/e9/ab288cd489999f962f791d6c8544803c29dcf24e9b6dde24634c41ec09dd/python_calamine-0.5.4-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2078ede35cbd26cf7186673405ff13321caacd9e45a5e57b54ce7b3ef0eec2ff", size = 886960, upload-time = "2025-10-21T07:11:00.462Z" }, + { url = "https://files.pythonhosted.org/packages/f0/4d/2a261f2ccde7128a683cdb20733f9bc030ab37a90803d8de836bf6113e5b/python_calamine-0.5.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:faab9f59bb9cedba2b35c6e1f5dc72461d8f2837e8f6ab24fafff0d054ddc4b5", size = 1044123, upload-time = "2025-10-21T07:11:02.153Z" }, + { url = "https://files.pythonhosted.org/packages/20/dc/a84c5a5a2c38816570bcc96ae4c9c89d35054e59c4199d3caef9c60b65cf/python_calamine-0.5.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:300d8d5e6c63bdecf79268d3b6d2a84078cda39cb3394ed09c5c00a61ce9ff32", size = 941997, upload-time = "2025-10-21T07:11:03.537Z" }, + { url = "https://files.pythonhosted.org/packages/dd/92/b970d8316c54f274d9060e7c804b79dbfa250edeb6390cd94f5fcfeb5f87/python_calamine-0.5.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0019a74f1c0b1cbf08fee9ece114d310522837cdf63660a46fe46d3688f215ea", size = 905881, upload-time = "2025-10-21T07:11:05.228Z" }, + { url = "https://files.pythonhosted.org/packages/ac/88/9186ac8d3241fc6f90995cc7539bdbd75b770d2dab20978a702c36fbce5f/python_calamine-0.5.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:30b40ffb374f7fb9ce20ca87f43a609288f568e41872f8a72e5af313a9e20af0", size = 947224, upload-time = "2025-10-21T07:11:06.618Z" }, + { url = "https://files.pythonhosted.org/packages/ee/ec/6ac1882dc6b6fa829e2d1d94ffa58bd0c67df3dba074b2e2f3134d7f573a/python_calamine-0.5.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:206242690a5a5dff73a193fb1a1ca3c7a8aed95e2f9f10c875dece5a22068801", size = 1078351, upload-time = "2025-10-21T07:11:08.368Z" }, + { url = "https://files.pythonhosted.org/packages/3e/f1/07aff6966b04b7452c41a802b37199d9e9ac656d66d6092b83ab0937e212/python_calamine-0.5.4-cp311-cp311-musllinux_1_1_armv7l.whl", hash = "sha256:88628e1a17a6f352d6433b0abf6edc4cb2295b8fbb3451392390f3a6a7a8cada", size = 1150148, upload-time = "2025-10-21T07:11:10.18Z" }, + { url = "https://files.pythonhosted.org/packages/4e/be/90aedeb0b77ea592a698a20db09014a5217ce46a55b699121849e239c8e7/python_calamine-0.5.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:22524cfb7720d15894a02392bbd49f8e7a8c173493f0628a45814d78e4243fff", size = 1080101, upload-time = "2025-10-21T07:11:11.489Z" }, + { url = "https://files.pythonhosted.org/packages/30/89/1fadd511d132d5ea9326c003c8753b6d234d61d9a72775fb1632cc94beb9/python_calamine-0.5.4-cp311-cp311-win32.whl", hash = "sha256:d159e98ef3475965555b67354f687257648f5c3686ed08e7faa34d54cc9274e1", size = 679593, upload-time = "2025-10-21T07:11:12.758Z" }, + { url = "https://files.pythonhosted.org/packages/e9/ba/d7324400a02491549ef30e0e480561a3a841aa073ac7c096313bc2cea555/python_calamine-0.5.4-cp311-cp311-win_amd64.whl", hash = "sha256:0d019b082f9a114cf1e130dc52b77f9f881325ab13dc31485d7b4563ad9e0812", size = 721570, upload-time = "2025-10-21T07:11:14.336Z" }, + { url = "https://files.pythonhosted.org/packages/4f/15/8c7895e603b4ae63ff279aae4aa6120658a15f805750ccdb5d8b311df616/python_calamine-0.5.4-cp311-cp311-win_arm64.whl", hash = "sha256:bb20875776e5b4c85134c2bf49fea12288e64448ed49f1d89a3a83f5bb16bd59", size = 685789, upload-time = "2025-10-21T07:11:15.646Z" }, + { url = "https://files.pythonhosted.org/packages/ff/60/b1ace7a0fd636581b3bb27f1011cb7b2fe4d507b58401c4d328cfcb5c849/python_calamine-0.5.4-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:4d711f91283d28f19feb111ed666764de69e6d2a0201df8f84e81a238f68d193", size = 850087, upload-time = "2025-10-21T07:11:17.002Z" }, + { url = "https://files.pythonhosted.org/packages/7f/32/32ca71ce50f9b7c7d6e7ec5fcc579a97ddd8b8ce314fe143ba2a19441dc7/python_calamine-0.5.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:ed67afd3adedb5bcfb428cf1f2d7dfd936dea9fe979ab631194495ab092973ba", size = 825659, upload-time = "2025-10-21T07:11:18.248Z" }, + { url = "https://files.pythonhosted.org/packages/63/c5/27ba71a9da2a09be9ff2f0dac522769956c8c89d6516565b21c9c78bfae6/python_calamine-0.5.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:13662895dac487315ccce25ea272a1ea7e7ac05d899cde4e33d59d6c43274c54", size = 897332, upload-time = "2025-10-21T07:11:19.89Z" }, + { url = "https://files.pythonhosted.org/packages/5a/e7/c4be6ff8e8899ace98cacc9604a2dd1abc4901839b733addfb6ef32c22ba/python_calamine-0.5.4-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:23e354755583cfaa824ddcbe8b099c5c7ac19bf5179320426e7a88eea2f14bc5", size = 886885, upload-time = "2025-10-21T07:11:21.912Z" }, + { url = "https://files.pythonhosted.org/packages/38/24/80258fb041435021efa10d0b528df6842e442585e48cbf130e73fed2529b/python_calamine-0.5.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4e1bc3f22107dcbdeb32d4d3c5c1e8831d3c85d4b004a8606dd779721b29843d", size = 1043907, upload-time = "2025-10-21T07:11:23.3Z" }, + { url = "https://files.pythonhosted.org/packages/f2/20/157340787d03ef6113a967fd8f84218e867ba4c2f7fc58cc645d8665a61a/python_calamine-0.5.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:182b314117e47dbd952adaa2b19c515555083a48d6f9146f46faaabd9dab2f81", size = 942376, upload-time = "2025-10-21T07:11:24.866Z" }, + { url = "https://files.pythonhosted.org/packages/98/f5/aec030f567ee14c60b6fc9028a78767687f484071cb080f7cfa328d6496e/python_calamine-0.5.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b8f882e092ab23f72ea07e2e48f5f2efb1885c1836fb949f22fd4540ae11742e", size = 906455, upload-time = "2025-10-21T07:11:26.203Z" }, + { url = "https://files.pythonhosted.org/packages/29/58/4affc0d1389f837439ad45f400f3792e48030b75868ec757e88cb35d7626/python_calamine-0.5.4-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:62a9b4b7b9bd99d03373e58884dfb60d5a1c292c8e04e11f8b7420b77a46813e", size = 948132, upload-time = "2025-10-21T07:11:27.507Z" }, + { url = "https://files.pythonhosted.org/packages/b4/2e/70ed04f39e682a9116730f56b7fbb54453244ccc1c3dae0662d4819f1c1d/python_calamine-0.5.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:98bb011d33c0e2d183ff30ab3d96792c3493f56f67a7aa2fcadad9a03539e79b", size = 1077436, upload-time = "2025-10-21T07:11:28.801Z" }, + { url = "https://files.pythonhosted.org/packages/cb/ce/806f8ce06b5bb9db33007f85045c304cda410970e7aa07d08f6eaee67913/python_calamine-0.5.4-cp312-cp312-musllinux_1_1_armv7l.whl", hash = "sha256:6b218a95489ff2f1cc1de0bba2a16fcc82981254bbb23f31d41d29191282b9ad", size = 1150570, upload-time = "2025-10-21T07:11:30.237Z" }, + { url = "https://files.pythonhosted.org/packages/18/da/61f13c8d107783128c1063cf52ca9cacdc064c58d58d3cf49c1728ce8296/python_calamine-0.5.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:e8296a4872dbe834205d25d26dd6cfcb33ee9da721668d81b21adc25a07c07e4", size = 1080286, upload-time = "2025-10-21T07:11:31.564Z" }, + { url = "https://files.pythonhosted.org/packages/99/85/c5612a63292eb7d0648b17c5ff32ad5d6c6f3e1d78825f01af5c765f4d3f/python_calamine-0.5.4-cp312-cp312-win32.whl", hash = "sha256:cebb9c88983ae676c60c8c02aa29a9fe13563f240579e66de5c71b969ace5fd9", size = 676617, upload-time = "2025-10-21T07:11:32.833Z" }, + { url = "https://files.pythonhosted.org/packages/bb/18/5a037942de8a8df0c805224b2fba06df6d25c1be3c9484ba9db1ca4f3ee6/python_calamine-0.5.4-cp312-cp312-win_amd64.whl", hash = "sha256:15abd7aff98fde36d7df91ac051e86e66e5d5326a7fa98d54697afe95a613501", size = 721464, upload-time = "2025-10-21T07:11:34.383Z" }, + { url = "https://files.pythonhosted.org/packages/d1/8b/89ca17b44bcd8be5d0e8378d87b880ae17a837573553bd2147cceca7e759/python_calamine-0.5.4-cp312-cp312-win_arm64.whl", hash = "sha256:1cef0d0fc936974020a24acf1509ed2a285b30a4e1adf346c057112072e84251", size = 687268, upload-time = "2025-10-21T07:11:36.324Z" }, + { url = "https://files.pythonhosted.org/packages/ab/a8/0e05992489f8ca99eadfb52e858a7653b01b27a7c66d040abddeb4bdf799/python_calamine-0.5.4-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:8d4be45952555f129584e0ca6ddb442bed5cb97b8d7cd0fd5ae463237b98eb15", size = 856420, upload-time = "2025-10-21T07:13:20.962Z" }, + { url = "https://files.pythonhosted.org/packages/f0/b0/5bbe52c97161acb94066e7020c2fed7eafbca4bf6852a4b02ed80bf0b24b/python_calamine-0.5.4-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:5b387d12cb8cae98c8e0c061c5400f80bad1f43f26fafcf95ff5934df995f50b", size = 833240, upload-time = "2025-10-21T07:13:22.801Z" }, + { url = "https://files.pythonhosted.org/packages/c7/b9/44fa30f6bf479072d9042856d3fab8bdd1532d2d901e479e199bc1de0e6c/python_calamine-0.5.4-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2103714954b7dbed72a0b0eff178b08e854bba130be283e3ae3d7c95521e8f69", size = 899470, upload-time = "2025-10-21T07:13:25.176Z" }, + { url = "https://files.pythonhosted.org/packages/0e/f2/acbb2c1d6acba1eaf6b1efb6485c98995050bddedfb6b93ce05be2753a85/python_calamine-0.5.4-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c09fdebe23a5045d09e12b3366ff8fd45165b6fb56f55e9a12342a5daddbd11a", size = 906108, upload-time = "2025-10-21T07:13:26.709Z" }, + { url = "https://files.pythonhosted.org/packages/77/28/ff007e689539d6924223565995db876ac044466b8859bade371696294659/python_calamine-0.5.4-pp311-pypy311_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:fa992d72fbd38f09107430100b7688c03046d8c1994e4cff9bbbd2a825811796", size = 948580, upload-time = "2025-10-21T07:13:30.816Z" }, + { url = "https://files.pythonhosted.org/packages/a4/06/b423655446fb27e22bfc1ca5e5b11f3449e0350fe8fefa0ebd68675f7e85/python_calamine-0.5.4-pp311-pypy311_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:88e608c7589412d3159be40d270a90994e38c9eafc125bf8ad5a9c92deffd6dd", size = 1079516, upload-time = "2025-10-21T07:13:32.288Z" }, + { url = "https://files.pythonhosted.org/packages/76/f5/c7132088978b712a5eddf1ca6bf64ae81335fbca9443ed486330519954c3/python_calamine-0.5.4-pp311-pypy311_pp73-musllinux_1_1_armv7l.whl", hash = "sha256:51a007801aef12f6bc93a545040a36df48e9af920a7da9ded915584ad9a002b1", size = 1152379, upload-time = "2025-10-21T07:13:33.739Z" }, + { url = "https://files.pythonhosted.org/packages/bd/c8/37a8d80b7e55e7cfbe649f7a92a7e838defc746aac12dca751aad5dd06a6/python_calamine-0.5.4-pp311-pypy311_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:b056db205e45ab9381990a5c15d869f1021c1262d065740c9cd296fc5d3fb248", size = 1080420, upload-time = "2025-10-21T07:13:35.33Z" }, + { url = "https://files.pythonhosted.org/packages/10/52/9a96d06e75862d356dc80a4a465ad88fba544a19823568b4ff484e7a12f2/python_calamine-0.5.4-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:dd8f4123b2403fc22c92ec4f5e51c495427cf3739c5cb614b9829745a80922db", size = 722350, upload-time = "2025-10-21T07:13:37.074Z" }, ] [[package]] @@ -5084,11 +5093,11 @@ wheels = [ [[package]] name = "python-iso639" -version = "2025.2.18" +version = "2025.11.11" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/d5/19/45aa1917c7b1f4eb71104795b9b0cbf97169b99ec46cd303445883536549/python_iso639-2025.2.18.tar.gz", hash = "sha256:34e31e8e76eb3fc839629e257b12bcfd957c6edcbd486bbf66ba5185d1f566e8", size = 173552, upload-time = "2025-02-18T13:48:08.607Z" } +sdist = { url = "https://files.pythonhosted.org/packages/89/6f/45bc5ae1c132ab7852a8642d66d25ffff6e4b398195127ac66158d3b5f4d/python_iso639-2025.11.11.tar.gz", hash = "sha256:75fab30f1a0f46b4e8161eafb84afe4ecd07eaada05e2c5364f14b0f9c864477", size = 173897, upload-time = "2025-11-11T15:23:00.893Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/54/a3/3ceaf89a17a1e1d5e7bbdfe5514aa3055d91285b37a5c8fed662969e3d56/python_iso639-2025.2.18-py3-none-any.whl", hash = "sha256:b2d471c37483a26f19248458b20e7bd96492e15368b01053b540126bcc23152f", size = 167631, upload-time = "2025-02-18T13:48:06.602Z" }, + { url = "https://files.pythonhosted.org/packages/03/69/081960288e4cd541cbdb90e1768373e1198b040bf2ae40cd25b9c9799205/python_iso639-2025.11.11-py3-none-any.whl", hash = "sha256:02ea4cfca2c189b5665e4e8adc8c17c62ab6e4910932541a23baddea33207ea2", size = 167723, upload-time = "2025-11-11T15:22:59.819Z" }, ] [[package]] @@ -5207,43 +5216,37 @@ wheels = [ [[package]] name = "rapidfuzz" -version = "3.14.1" +version = "3.14.3" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/ed/fc/a98b616db9a42dcdda7c78c76bdfdf6fe290ac4c5ffbb186f73ec981ad5b/rapidfuzz-3.14.1.tar.gz", hash = "sha256:b02850e7f7152bd1edff27e9d584505b84968cacedee7a734ec4050c655a803c", size = 57869570, upload-time = "2025-09-08T21:08:15.922Z" } +sdist = { url = "https://files.pythonhosted.org/packages/d3/28/9d808fe62375b9aab5ba92fa9b29371297b067c2790b2d7cda648b1e2f8d/rapidfuzz-3.14.3.tar.gz", hash = "sha256:2491937177868bc4b1e469087601d53f925e8d270ccc21e07404b4b5814b7b5f", size = 57863900, upload-time = "2025-11-01T11:54:52.321Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/5c/c7/c3c860d512606225c11c8ee455b4dc0b0214dbcfac90a2c22dddf55320f3/rapidfuzz-3.14.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:4d976701060886a791c8a9260b1d4139d14c1f1e9a6ab6116b45a1acf3baff67", size = 1938398, upload-time = "2025-09-08T21:05:44.031Z" }, - { url = "https://files.pythonhosted.org/packages/c0/f3/67f5c5cd4d728993c48c1dcb5da54338d77c03c34b4903cc7839a3b89faf/rapidfuzz-3.14.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:5e6ba7e6eb2ab03870dcab441d707513db0b4264c12fba7b703e90e8b4296df2", size = 1392819, upload-time = "2025-09-08T21:05:45.549Z" }, - { url = "https://files.pythonhosted.org/packages/d5/06/400d44842f4603ce1bebeaeabe776f510e329e7dbf6c71b6f2805e377889/rapidfuzz-3.14.1-cp311-cp311-manylinux_2_26_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1e532bf46de5fd3a1efde73a16a4d231d011bce401c72abe3c6ecf9de681003f", size = 1391798, upload-time = "2025-09-08T21:05:47.044Z" }, - { url = "https://files.pythonhosted.org/packages/90/97/a6944955713b47d88e8ca4305ca7484940d808c4e6c4e28b6fa0fcbff97e/rapidfuzz-3.14.1-cp311-cp311-manylinux_2_26_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f9b6a6fb8ed9b951e5f3b82c1ce6b1665308ec1a0da87f799b16e24fc59e4662", size = 1699136, upload-time = "2025-09-08T21:05:48.919Z" }, - { url = "https://files.pythonhosted.org/packages/a8/1e/f311a5c95ddf922db6dd8666efeceb9ac69e1319ed098ac80068a4041732/rapidfuzz-3.14.1-cp311-cp311-manylinux_2_26_s390x.manylinux_2_28_s390x.whl", hash = "sha256:5b6ac3f9810949caef0e63380b11a3c32a92f26bacb9ced5e32c33560fcdf8d1", size = 2236238, upload-time = "2025-09-08T21:05:50.844Z" }, - { url = "https://files.pythonhosted.org/packages/85/27/e14e9830255db8a99200f7111b158ddef04372cf6332a415d053fe57cc9c/rapidfuzz-3.14.1-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:e52e4c34fd567f77513e886b66029c1ae02f094380d10eba18ba1c68a46d8b90", size = 3183685, upload-time = "2025-09-08T21:05:52.362Z" }, - { url = "https://files.pythonhosted.org/packages/61/b2/42850c9616ddd2887904e5dd5377912cbabe2776fdc9fd4b25e6e12fba32/rapidfuzz-3.14.1-cp311-cp311-manylinux_2_31_armv7l.whl", hash = "sha256:2ef72e41b1a110149f25b14637f1cedea6df192462120bea3433980fe9d8ac05", size = 1231523, upload-time = "2025-09-08T21:05:53.927Z" }, - { url = "https://files.pythonhosted.org/packages/de/b5/6b90ed7127a1732efef39db46dd0afc911f979f215b371c325a2eca9cb15/rapidfuzz-3.14.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:fb654a35b373d712a6b0aa2a496b2b5cdd9d32410cfbaecc402d7424a90ba72a", size = 2415209, upload-time = "2025-09-08T21:05:55.422Z" }, - { url = "https://files.pythonhosted.org/packages/70/60/af51c50d238c82f2179edc4b9f799cc5a50c2c0ebebdcfaa97ded7d02978/rapidfuzz-3.14.1-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:2b2c12e5b9eb8fe9a51b92fe69e9ca362c0970e960268188a6d295e1dec91e6d", size = 2532957, upload-time = "2025-09-08T21:05:57.048Z" }, - { url = "https://files.pythonhosted.org/packages/50/92/29811d2ba7c984251a342c4f9ccc7cc4aa09d43d800af71510cd51c36453/rapidfuzz-3.14.1-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:4f069dec5c450bd987481e752f0a9979e8fdf8e21e5307f5058f5c4bb162fa56", size = 2815720, upload-time = "2025-09-08T21:05:58.618Z" }, - { url = "https://files.pythonhosted.org/packages/78/69/cedcdee16a49e49d4985eab73b59447f211736c5953a58f1b91b6c53a73f/rapidfuzz-3.14.1-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:4d0d9163725b7ad37a8c46988cae9ebab255984db95ad01bf1987ceb9e3058dd", size = 3323704, upload-time = "2025-09-08T21:06:00.576Z" }, - { url = "https://files.pythonhosted.org/packages/76/3e/5a3f9a5540f18e0126e36f86ecf600145344acb202d94b63ee45211a18b8/rapidfuzz-3.14.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:db656884b20b213d846f6bc990c053d1f4a60e6d4357f7211775b02092784ca1", size = 4287341, upload-time = "2025-09-08T21:06:02.301Z" }, - { url = "https://files.pythonhosted.org/packages/46/26/45db59195929dde5832852c9de8533b2ac97dcc0d852d1f18aca33828122/rapidfuzz-3.14.1-cp311-cp311-win32.whl", hash = "sha256:4b42f7b9c58cbcfbfaddc5a6278b4ca3b6cd8983e7fd6af70ca791dff7105fb9", size = 1726574, upload-time = "2025-09-08T21:06:04.357Z" }, - { url = "https://files.pythonhosted.org/packages/01/5c/a4caf76535f35fceab25b2aaaed0baecf15b3d1fd40746f71985d20f8c4b/rapidfuzz-3.14.1-cp311-cp311-win_amd64.whl", hash = "sha256:e5847f30d7d4edefe0cb37294d956d3495dd127c1c56e9128af3c2258a520bb4", size = 1547124, upload-time = "2025-09-08T21:06:06.002Z" }, - { url = "https://files.pythonhosted.org/packages/c6/66/aa93b52f95a314584d71fa0b76df00bdd4158aafffa76a350f1ae416396c/rapidfuzz-3.14.1-cp311-cp311-win_arm64.whl", hash = "sha256:5087d8ad453092d80c042a08919b1cb20c8ad6047d772dc9312acd834da00f75", size = 816958, upload-time = "2025-09-08T21:06:07.509Z" }, - { url = "https://files.pythonhosted.org/packages/df/77/2f4887c9b786f203e50b816c1cde71f96642f194e6fa752acfa042cf53fd/rapidfuzz-3.14.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:809515194f628004aac1b1b280c3734c5ea0ccbd45938c9c9656a23ae8b8f553", size = 1932216, upload-time = "2025-09-08T21:06:09.342Z" }, - { url = "https://files.pythonhosted.org/packages/de/bd/b5e445d156cb1c2a87d36d8da53daf4d2a1d1729b4851660017898b49aa0/rapidfuzz-3.14.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:0afcf2d6cb633d0d4260d8df6a40de2d9c93e9546e2c6b317ab03f89aa120ad7", size = 1393414, upload-time = "2025-09-08T21:06:10.959Z" }, - { url = "https://files.pythonhosted.org/packages/de/bd/98d065dd0a4479a635df855616980eaae1a1a07a876db9400d421b5b6371/rapidfuzz-3.14.1-cp312-cp312-manylinux_2_26_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5c1c3d07d53dcafee10599da8988d2b1f39df236aee501ecbd617bd883454fcd", size = 1377194, upload-time = "2025-09-08T21:06:12.471Z" }, - { url = "https://files.pythonhosted.org/packages/d3/8a/1265547b771128b686f3c431377ff1db2fa073397ed082a25998a7b06d4e/rapidfuzz-3.14.1-cp312-cp312-manylinux_2_26_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:6e9ee3e1eb0a027717ee72fe34dc9ac5b3e58119f1bd8dd15bc19ed54ae3e62b", size = 1669573, upload-time = "2025-09-08T21:06:14.016Z" }, - { url = "https://files.pythonhosted.org/packages/a8/57/e73755c52fb451f2054196404ccc468577f8da023b3a48c80bce29ee5d4a/rapidfuzz-3.14.1-cp312-cp312-manylinux_2_26_s390x.manylinux_2_28_s390x.whl", hash = "sha256:70c845b64a033a20c44ed26bc890eeb851215148cc3e696499f5f65529afb6cb", size = 2217833, upload-time = "2025-09-08T21:06:15.666Z" }, - { url = "https://files.pythonhosted.org/packages/20/14/7399c18c460e72d1b754e80dafc9f65cb42a46cc8f29cd57d11c0c4acc94/rapidfuzz-3.14.1-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:26db0e815213d04234298dea0d884d92b9cb8d4ba954cab7cf67a35853128a33", size = 3159012, upload-time = "2025-09-08T21:06:17.631Z" }, - { url = "https://files.pythonhosted.org/packages/f8/5e/24f0226ddb5440cabd88605d2491f99ae3748a6b27b0bc9703772892ced7/rapidfuzz-3.14.1-cp312-cp312-manylinux_2_31_armv7l.whl", hash = "sha256:6ad3395a416f8b126ff11c788531f157c7debeb626f9d897c153ff8980da10fb", size = 1227032, upload-time = "2025-09-08T21:06:21.06Z" }, - { url = "https://files.pythonhosted.org/packages/40/43/1d54a4ad1a5fac2394d5f28a3108e2bf73c26f4f23663535e3139cfede9b/rapidfuzz-3.14.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:61c5b9ab6f730e6478aa2def566223712d121c6f69a94c7cc002044799442afd", size = 2395054, upload-time = "2025-09-08T21:06:23.482Z" }, - { url = "https://files.pythonhosted.org/packages/0c/71/e9864cd5b0f086c4a03791f5dfe0155a1b132f789fe19b0c76fbabd20513/rapidfuzz-3.14.1-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:13e0ea3d0c533969158727d1bb7a08c2cc9a816ab83f8f0dcfde7e38938ce3e6", size = 2524741, upload-time = "2025-09-08T21:06:26.825Z" }, - { url = "https://files.pythonhosted.org/packages/b2/0c/53f88286b912faf4a3b2619a60df4f4a67bd0edcf5970d7b0c1143501f0c/rapidfuzz-3.14.1-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:6325ca435b99f4001aac919ab8922ac464999b100173317defb83eae34e82139", size = 2785311, upload-time = "2025-09-08T21:06:29.471Z" }, - { url = "https://files.pythonhosted.org/packages/53/9a/229c26dc4f91bad323f07304ee5ccbc28f0d21c76047a1e4f813187d0bad/rapidfuzz-3.14.1-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:07a9fad3247e68798424bdc116c1094e88ecfabc17b29edf42a777520347648e", size = 3303630, upload-time = "2025-09-08T21:06:31.094Z" }, - { url = "https://files.pythonhosted.org/packages/05/de/20e330d6d58cbf83da914accd9e303048b7abae2f198886f65a344b69695/rapidfuzz-3.14.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:f8ff5dbe78db0a10c1f916368e21d328935896240f71f721e073cf6c4c8cdedd", size = 4262364, upload-time = "2025-09-08T21:06:32.877Z" }, - { url = "https://files.pythonhosted.org/packages/1f/10/2327f83fad3534a8d69fe9cd718f645ec1fe828b60c0e0e97efc03bf12f8/rapidfuzz-3.14.1-cp312-cp312-win32.whl", hash = "sha256:9c83270e44a6ae7a39fc1d7e72a27486bccc1fa5f34e01572b1b90b019e6b566", size = 1711927, upload-time = "2025-09-08T21:06:34.669Z" }, - { url = "https://files.pythonhosted.org/packages/78/8d/199df0370133fe9f35bc72f3c037b53c93c5c1fc1e8d915cf7c1f6bb8557/rapidfuzz-3.14.1-cp312-cp312-win_amd64.whl", hash = "sha256:e06664c7fdb51c708e082df08a6888fce4c5c416d7e3cc2fa66dd80eb76a149d", size = 1542045, upload-time = "2025-09-08T21:06:36.364Z" }, - { url = "https://files.pythonhosted.org/packages/b3/c6/cc5d4bd1b16ea2657c80b745d8b1c788041a31fad52e7681496197b41562/rapidfuzz-3.14.1-cp312-cp312-win_arm64.whl", hash = "sha256:6c7c26025f7934a169a23dafea6807cfc3fb556f1dd49229faf2171e5d8101cc", size = 813170, upload-time = "2025-09-08T21:06:38.001Z" }, - { url = "https://files.pythonhosted.org/packages/05/c7/1b17347e30f2b50dd976c54641aa12003569acb1bdaabf45a5cc6f471c58/rapidfuzz-3.14.1-pp311-pypy311_pp73-macosx_10_15_x86_64.whl", hash = "sha256:4a21ccdf1bd7d57a1009030527ba8fae1c74bf832d0a08f6b67de8f5c506c96f", size = 1862602, upload-time = "2025-09-08T21:08:09.088Z" }, - { url = "https://files.pythonhosted.org/packages/09/cf/95d0dacac77eda22499991bd5f304c77c5965fb27348019a48ec3fe4a3f6/rapidfuzz-3.14.1-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:589fb0af91d3aff318750539c832ea1100dbac2c842fde24e42261df443845f6", size = 1339548, upload-time = "2025-09-08T21:08:11.059Z" }, - { url = "https://files.pythonhosted.org/packages/b6/58/f515c44ba8c6fa5daa35134b94b99661ced852628c5505ead07b905c3fc7/rapidfuzz-3.14.1-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:a4f18092db4825f2517d135445015b40033ed809a41754918a03ef062abe88a0", size = 1513859, upload-time = "2025-09-08T21:08:13.07Z" }, + { url = "https://files.pythonhosted.org/packages/76/25/5b0a33ad3332ee1213068c66f7c14e9e221be90bab434f0cb4defa9d6660/rapidfuzz-3.14.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:dea2d113e260a5da0c4003e0a5e9fdf24a9dc2bb9eaa43abd030a1e46ce7837d", size = 1953885, upload-time = "2025-11-01T11:52:47.75Z" }, + { url = "https://files.pythonhosted.org/packages/2d/ab/f1181f500c32c8fcf7c966f5920c7e56b9b1d03193386d19c956505c312d/rapidfuzz-3.14.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e6c31a4aa68cfa75d7eede8b0ed24b9e458447db604c2db53f358be9843d81d3", size = 1390200, upload-time = "2025-11-01T11:52:49.491Z" }, + { url = "https://files.pythonhosted.org/packages/14/2a/0f2de974ececad873865c6bb3ea3ad07c976ac293d5025b2d73325aac1d4/rapidfuzz-3.14.3-cp311-cp311-manylinux_2_26_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:02821366d928e68ddcb567fed8723dad7ea3a979fada6283e6914d5858674850", size = 1389319, upload-time = "2025-11-01T11:52:51.224Z" }, + { url = "https://files.pythonhosted.org/packages/ed/69/309d8f3a0bb3031fd9b667174cc4af56000645298af7c2931be5c3d14bb4/rapidfuzz-3.14.3-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:cfe8df315ab4e6db4e1be72c5170f8e66021acde22cd2f9d04d2058a9fd8162e", size = 3178495, upload-time = "2025-11-01T11:52:53.005Z" }, + { url = "https://files.pythonhosted.org/packages/10/b7/f9c44a99269ea5bf6fd6a40b84e858414b6e241288b9f2b74af470d222b1/rapidfuzz-3.14.3-cp311-cp311-manylinux_2_31_armv7l.whl", hash = "sha256:769f31c60cd79420188fcdb3c823227fc4a6deb35cafec9d14045c7f6743acae", size = 1228443, upload-time = "2025-11-01T11:52:54.991Z" }, + { url = "https://files.pythonhosted.org/packages/f2/0a/3b3137abac7f19c9220e14cd7ce993e35071a7655e7ef697785a3edfea1a/rapidfuzz-3.14.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:54fa03062124e73086dae66a3451c553c1e20a39c077fd704dc7154092c34c63", size = 2411998, upload-time = "2025-11-01T11:52:56.629Z" }, + { url = "https://files.pythonhosted.org/packages/f3/b6/983805a844d44670eaae63831024cdc97ada4e9c62abc6b20703e81e7f9b/rapidfuzz-3.14.3-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:834d1e818005ed0d4ae38f6b87b86fad9b0a74085467ece0727d20e15077c094", size = 2530120, upload-time = "2025-11-01T11:52:58.298Z" }, + { url = "https://files.pythonhosted.org/packages/b4/cc/2c97beb2b1be2d7595d805682472f1b1b844111027d5ad89b65e16bdbaaa/rapidfuzz-3.14.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:948b00e8476a91f510dd1ec07272efc7d78c275d83b630455559671d4e33b678", size = 4283129, upload-time = "2025-11-01T11:53:00.188Z" }, + { url = "https://files.pythonhosted.org/packages/4d/03/2f0e5e94941045aefe7eafab72320e61285c07b752df9884ce88d6b8b835/rapidfuzz-3.14.3-cp311-cp311-win32.whl", hash = "sha256:43d0305c36f504232f18ea04e55f2059bb89f169d3119c4ea96a0e15b59e2a91", size = 1724224, upload-time = "2025-11-01T11:53:02.149Z" }, + { url = "https://files.pythonhosted.org/packages/cf/99/5fa23e204435803875daefda73fd61baeabc3c36b8fc0e34c1705aab8c7b/rapidfuzz-3.14.3-cp311-cp311-win_amd64.whl", hash = "sha256:ef6bf930b947bd0735c550683939a032090f1d688dfd8861d6b45307b96fd5c5", size = 1544259, upload-time = "2025-11-01T11:53:03.66Z" }, + { url = "https://files.pythonhosted.org/packages/48/35/d657b85fcc615a42661b98ac90ce8e95bd32af474603a105643963749886/rapidfuzz-3.14.3-cp311-cp311-win_arm64.whl", hash = "sha256:f3eb0ff3b75d6fdccd40b55e7414bb859a1cda77c52762c9c82b85569f5088e7", size = 814734, upload-time = "2025-11-01T11:53:05.008Z" }, + { url = "https://files.pythonhosted.org/packages/fa/8e/3c215e860b458cfbedb3ed73bc72e98eb7e0ed72f6b48099604a7a3260c2/rapidfuzz-3.14.3-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:685c93ea961d135893b5984a5a9851637d23767feabe414ec974f43babbd8226", size = 1945306, upload-time = "2025-11-01T11:53:06.452Z" }, + { url = "https://files.pythonhosted.org/packages/36/d9/31b33512015c899f4a6e6af64df8dfe8acddf4c8b40a4b3e0e6e1bcd00e5/rapidfuzz-3.14.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:fa7c8f26f009f8c673fbfb443792f0cf8cf50c4e18121ff1e285b5e08a94fbdb", size = 1390788, upload-time = "2025-11-01T11:53:08.721Z" }, + { url = "https://files.pythonhosted.org/packages/a9/67/2ee6f8de6e2081ccd560a571d9c9063184fe467f484a17fa90311a7f4a2e/rapidfuzz-3.14.3-cp312-cp312-manylinux_2_26_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:57f878330c8d361b2ce76cebb8e3e1dc827293b6abf404e67d53260d27b5d941", size = 1374580, upload-time = "2025-11-01T11:53:10.164Z" }, + { url = "https://files.pythonhosted.org/packages/30/83/80d22997acd928eda7deadc19ccd15883904622396d6571e935993e0453a/rapidfuzz-3.14.3-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6c5f545f454871e6af05753a0172849c82feaf0f521c5ca62ba09e1b382d6382", size = 3154947, upload-time = "2025-11-01T11:53:12.093Z" }, + { url = "https://files.pythonhosted.org/packages/5b/cf/9f49831085a16384695f9fb096b99662f589e30b89b4a589a1ebc1a19d34/rapidfuzz-3.14.3-cp312-cp312-manylinux_2_31_armv7l.whl", hash = "sha256:07aa0b5d8863e3151e05026a28e0d924accf0a7a3b605da978f0359bb804df43", size = 1223872, upload-time = "2025-11-01T11:53:13.664Z" }, + { url = "https://files.pythonhosted.org/packages/c8/0f/41ee8034e744b871c2e071ef0d360686f5ccfe5659f4fd96c3ec406b3c8b/rapidfuzz-3.14.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:73b07566bc7e010e7b5bd490fb04bb312e820970180df6b5655e9e6224c137db", size = 2392512, upload-time = "2025-11-01T11:53:15.109Z" }, + { url = "https://files.pythonhosted.org/packages/da/86/280038b6b0c2ccec54fb957c732ad6b41cc1fd03b288d76545b9cf98343f/rapidfuzz-3.14.3-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:6de00eb84c71476af7d3110cf25d8fe7c792d7f5fa86764ef0b4ca97e78ca3ed", size = 2521398, upload-time = "2025-11-01T11:53:17.146Z" }, + { url = "https://files.pythonhosted.org/packages/fa/7b/05c26f939607dca0006505e3216248ae2de631e39ef94dd63dbbf0860021/rapidfuzz-3.14.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:d7843a1abf0091773a530636fdd2a49a41bcae22f9910b86b4f903e76ddc82dc", size = 4259416, upload-time = "2025-11-01T11:53:19.34Z" }, + { url = "https://files.pythonhosted.org/packages/40/eb/9e3af4103d91788f81111af1b54a28de347cdbed8eaa6c91d5e98a889aab/rapidfuzz-3.14.3-cp312-cp312-win32.whl", hash = "sha256:dea97ac3ca18cd3ba8f3d04b5c1fe4aa60e58e8d9b7793d3bd595fdb04128d7a", size = 1709527, upload-time = "2025-11-01T11:53:20.949Z" }, + { url = "https://files.pythonhosted.org/packages/b8/63/d06ecce90e2cf1747e29aeab9f823d21e5877a4c51b79720b2d3be7848f8/rapidfuzz-3.14.3-cp312-cp312-win_amd64.whl", hash = "sha256:b5100fd6bcee4d27f28f4e0a1c6b5127bc8ba7c2a9959cad9eab0bf4a7ab3329", size = 1538989, upload-time = "2025-11-01T11:53:22.428Z" }, + { url = "https://files.pythonhosted.org/packages/fc/6d/beee32dcda64af8128aab3ace2ccb33d797ed58c434c6419eea015fec779/rapidfuzz-3.14.3-cp312-cp312-win_arm64.whl", hash = "sha256:4e49c9e992bc5fc873bd0fff7ef16a4405130ec42f2ce3d2b735ba5d3d4eb70f", size = 811161, upload-time = "2025-11-01T11:53:23.811Z" }, + { url = "https://files.pythonhosted.org/packages/c9/33/b5bd6475c7c27164b5becc9b0e3eb978f1e3640fea590dd3dced6006ee83/rapidfuzz-3.14.3-pp311-pypy311_pp73-macosx_10_15_x86_64.whl", hash = "sha256:7cf174b52cb3ef5d49e45d0a1133b7e7d0ecf770ed01f97ae9962c5c91d97d23", size = 1888499, upload-time = "2025-11-01T11:54:42.094Z" }, + { url = "https://files.pythonhosted.org/packages/30/d2/89d65d4db4bb931beade9121bc71ad916b5fa9396e807d11b33731494e8e/rapidfuzz-3.14.3-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:442cba39957a008dfc5bdef21a9c3f4379e30ffb4e41b8555dbaf4887eca9300", size = 1336747, upload-time = "2025-11-01T11:54:43.957Z" }, + { url = "https://files.pythonhosted.org/packages/85/33/cd87d92b23f0b06e8914a61cea6850c6d495ca027f669fab7a379041827a/rapidfuzz-3.14.3-pp311-pypy311_pp73-manylinux_2_26_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1faa0f8f76ba75fd7b142c984947c280ef6558b5067af2ae9b8729b0a0f99ede", size = 1352187, upload-time = "2025-11-01T11:54:45.518Z" }, + { url = "https://files.pythonhosted.org/packages/22/20/9d30b4a1ab26aac22fff17d21dec7e9089ccddfe25151d0a8bb57001dc3d/rapidfuzz-3.14.3-pp311-pypy311_pp73-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1e6eefec45625c634926a9fd46c9e4f31118ac8f3156fff9494422cee45207e6", size = 3101472, upload-time = "2025-11-01T11:54:47.255Z" }, + { url = "https://files.pythonhosted.org/packages/b1/ad/fa2d3e5c29a04ead7eaa731c7cd1f30f9ec3c77b3a578fdf90280797cbcb/rapidfuzz-3.14.3-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:56fefb4382bb12250f164250240b9dd7772e41c5c8ae976fd598a32292449cc5", size = 1511361, upload-time = "2025-11-01T11:54:49.057Z" }, ] [[package]] @@ -5308,38 +5311,38 @@ wheels = [ [[package]] name = "regex" -version = "2025.9.18" +version = "2025.11.3" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/49/d3/eaa0d28aba6ad1827ad1e716d9a93e1ba963ada61887498297d3da715133/regex-2025.9.18.tar.gz", hash = "sha256:c5ba23274c61c6fef447ba6a39333297d0c247f53059dba0bca415cac511edc4", size = 400917, upload-time = "2025-09-19T00:38:35.79Z" } +sdist = { url = "https://files.pythonhosted.org/packages/cc/a9/546676f25e573a4cf00fe8e119b78a37b6a8fe2dc95cda877b30889c9c45/regex-2025.11.3.tar.gz", hash = "sha256:1fedc720f9bb2494ce31a58a1631f9c82df6a09b49c19517ea5cc280b4541e01", size = 414669, upload-time = "2025-11-03T21:34:22.089Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/58/61/80eda662fc4eb32bfedc331f42390974c9e89c7eac1b79cd9eea4d7c458c/regex-2025.9.18-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:51076980cd08cd13c88eb7365427ae27f0d94e7cebe9ceb2bb9ffdae8fc4d82a", size = 484832, upload-time = "2025-09-19T00:35:30.011Z" }, - { url = "https://files.pythonhosted.org/packages/a6/d9/33833d9abddf3f07ad48504ddb53fe3b22f353214bbb878a72eee1e3ddbf/regex-2025.9.18-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:828446870bd7dee4e0cbeed767f07961aa07f0ea3129f38b3ccecebc9742e0b8", size = 288994, upload-time = "2025-09-19T00:35:31.733Z" }, - { url = "https://files.pythonhosted.org/packages/2a/b3/526ee96b0d70ea81980cbc20c3496fa582f775a52e001e2743cc33b2fa75/regex-2025.9.18-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:c28821d5637866479ec4cc23b8c990f5bc6dd24e5e4384ba4a11d38a526e1414", size = 286619, upload-time = "2025-09-19T00:35:33.221Z" }, - { url = "https://files.pythonhosted.org/packages/65/4f/c2c096b02a351b33442aed5895cdd8bf87d372498d2100927c5a053d7ba3/regex-2025.9.18-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:726177ade8e481db669e76bf99de0b278783be8acd11cef71165327abd1f170a", size = 792454, upload-time = "2025-09-19T00:35:35.361Z" }, - { url = "https://files.pythonhosted.org/packages/24/15/b562c9d6e47c403c4b5deb744f8b4bf6e40684cf866c7b077960a925bdff/regex-2025.9.18-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f5cca697da89b9f8ea44115ce3130f6c54c22f541943ac8e9900461edc2b8bd4", size = 858723, upload-time = "2025-09-19T00:35:36.949Z" }, - { url = "https://files.pythonhosted.org/packages/f2/01/dba305409849e85b8a1a681eac4c03ed327d8de37895ddf9dc137f59c140/regex-2025.9.18-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:dfbde38f38004703c35666a1e1c088b778e35d55348da2b7b278914491698d6a", size = 905899, upload-time = "2025-09-19T00:35:38.723Z" }, - { url = "https://files.pythonhosted.org/packages/fe/d0/c51d1e6a80eab11ef96a4cbad17fc0310cf68994fb01a7283276b7e5bbd6/regex-2025.9.18-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:f2f422214a03fab16bfa495cfec72bee4aaa5731843b771860a471282f1bf74f", size = 798981, upload-time = "2025-09-19T00:35:40.416Z" }, - { url = "https://files.pythonhosted.org/packages/c4/5e/72db90970887bbe02296612bd61b0fa31e6d88aa24f6a4853db3e96c575e/regex-2025.9.18-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:a295916890f4df0902e4286bc7223ee7f9e925daa6dcdec4192364255b70561a", size = 781900, upload-time = "2025-09-19T00:35:42.077Z" }, - { url = "https://files.pythonhosted.org/packages/50/ff/596be45eea8e9bc31677fde243fa2904d00aad1b32c31bce26c3dbba0b9e/regex-2025.9.18-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:5db95ff632dbabc8c38c4e82bf545ab78d902e81160e6e455598014f0abe66b9", size = 852952, upload-time = "2025-09-19T00:35:43.751Z" }, - { url = "https://files.pythonhosted.org/packages/e5/1b/2dfa348fa551e900ed3f5f63f74185b6a08e8a76bc62bc9c106f4f92668b/regex-2025.9.18-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:fb967eb441b0f15ae610b7069bdb760b929f267efbf522e814bbbfffdf125ce2", size = 844355, upload-time = "2025-09-19T00:35:45.309Z" }, - { url = "https://files.pythonhosted.org/packages/f4/bf/aefb1def27fe33b8cbbb19c75c13aefccfbef1c6686f8e7f7095705969c7/regex-2025.9.18-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f04d2f20da4053d96c08f7fde6e1419b7ec9dbcee89c96e3d731fca77f411b95", size = 787254, upload-time = "2025-09-19T00:35:46.904Z" }, - { url = "https://files.pythonhosted.org/packages/e3/4e/8ef042e7cf0dbbb401e784e896acfc1b367b95dfbfc9ada94c2ed55a081f/regex-2025.9.18-cp311-cp311-win32.whl", hash = "sha256:895197241fccf18c0cea7550c80e75f185b8bd55b6924fcae269a1a92c614a07", size = 264129, upload-time = "2025-09-19T00:35:48.597Z" }, - { url = "https://files.pythonhosted.org/packages/b4/7d/c4fcabf80dcdd6821c0578ad9b451f8640b9110fb3dcb74793dd077069ff/regex-2025.9.18-cp311-cp311-win_amd64.whl", hash = "sha256:7e2b414deae99166e22c005e154a5513ac31493db178d8aec92b3269c9cce8c9", size = 276160, upload-time = "2025-09-19T00:36:00.45Z" }, - { url = "https://files.pythonhosted.org/packages/64/f8/0e13c8ae4d6df9d128afaba138342d532283d53a4c1e7a8c93d6756c8f4a/regex-2025.9.18-cp311-cp311-win_arm64.whl", hash = "sha256:fb137ec7c5c54f34a25ff9b31f6b7b0c2757be80176435bf367111e3f71d72df", size = 268471, upload-time = "2025-09-19T00:36:02.149Z" }, - { url = "https://files.pythonhosted.org/packages/b0/99/05859d87a66ae7098222d65748f11ef7f2dff51bfd7482a4e2256c90d72b/regex-2025.9.18-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:436e1b31d7efd4dcd52091d076482031c611dde58bf9c46ca6d0a26e33053a7e", size = 486335, upload-time = "2025-09-19T00:36:03.661Z" }, - { url = "https://files.pythonhosted.org/packages/97/7e/d43d4e8b978890932cf7b0957fce58c5b08c66f32698f695b0c2c24a48bf/regex-2025.9.18-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:c190af81e5576b9c5fdc708f781a52ff20f8b96386c6e2e0557a78402b029f4a", size = 289720, upload-time = "2025-09-19T00:36:05.471Z" }, - { url = "https://files.pythonhosted.org/packages/bb/3b/ff80886089eb5dcf7e0d2040d9aaed539e25a94300403814bb24cc775058/regex-2025.9.18-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:e4121f1ce2b2b5eec4b397cc1b277686e577e658d8f5870b7eb2d726bd2300ab", size = 287257, upload-time = "2025-09-19T00:36:07.072Z" }, - { url = "https://files.pythonhosted.org/packages/ee/66/243edf49dd8720cba8d5245dd4d6adcb03a1defab7238598c0c97cf549b8/regex-2025.9.18-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:300e25dbbf8299d87205e821a201057f2ef9aa3deb29caa01cd2cac669e508d5", size = 797463, upload-time = "2025-09-19T00:36:08.399Z" }, - { url = "https://files.pythonhosted.org/packages/df/71/c9d25a1142c70432e68bb03211d4a82299cd1c1fbc41db9409a394374ef5/regex-2025.9.18-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:7b47fcf9f5316c0bdaf449e879407e1b9937a23c3b369135ca94ebc8d74b1742", size = 862670, upload-time = "2025-09-19T00:36:10.101Z" }, - { url = "https://files.pythonhosted.org/packages/f8/8f/329b1efc3a64375a294e3a92d43372bf1a351aa418e83c21f2f01cf6ec41/regex-2025.9.18-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:57a161bd3acaa4b513220b49949b07e252165e6b6dc910ee7617a37ff4f5b425", size = 910881, upload-time = "2025-09-19T00:36:12.223Z" }, - { url = "https://files.pythonhosted.org/packages/35/9e/a91b50332a9750519320ed30ec378b74c996f6befe282cfa6bb6cea7e9fd/regex-2025.9.18-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4f130c3a7845ba42de42f380fff3c8aebe89a810747d91bcf56d40a069f15352", size = 802011, upload-time = "2025-09-19T00:36:13.901Z" }, - { url = "https://files.pythonhosted.org/packages/a4/1d/6be3b8d7856b6e0d7ee7f942f437d0a76e0d5622983abbb6d21e21ab9a17/regex-2025.9.18-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:5f96fa342b6f54dcba928dd452e8d8cb9f0d63e711d1721cd765bb9f73bb048d", size = 786668, upload-time = "2025-09-19T00:36:15.391Z" }, - { url = "https://files.pythonhosted.org/packages/cb/ce/4a60e53df58bd157c5156a1736d3636f9910bdcc271d067b32b7fcd0c3a8/regex-2025.9.18-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:0f0d676522d68c207828dcd01fb6f214f63f238c283d9f01d85fc664c7c85b56", size = 856578, upload-time = "2025-09-19T00:36:16.845Z" }, - { url = "https://files.pythonhosted.org/packages/86/e8/162c91bfe7217253afccde112868afb239f94703de6580fb235058d506a6/regex-2025.9.18-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:40532bff8a1a0621e7903ae57fce88feb2e8a9a9116d341701302c9302aef06e", size = 849017, upload-time = "2025-09-19T00:36:18.597Z" }, - { url = "https://files.pythonhosted.org/packages/35/34/42b165bc45289646ea0959a1bc7531733e90b47c56a72067adfe6b3251f6/regex-2025.9.18-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:039f11b618ce8d71a1c364fdee37da1012f5a3e79b1b2819a9f389cd82fd6282", size = 788150, upload-time = "2025-09-19T00:36:20.464Z" }, - { url = "https://files.pythonhosted.org/packages/79/5d/cdd13b1f3c53afa7191593a7ad2ee24092a5a46417725ffff7f64be8342d/regex-2025.9.18-cp312-cp312-win32.whl", hash = "sha256:e1dd06f981eb226edf87c55d523131ade7285137fbde837c34dc9d1bf309f459", size = 264536, upload-time = "2025-09-19T00:36:21.922Z" }, - { url = "https://files.pythonhosted.org/packages/e0/f5/4a7770c9a522e7d2dc1fa3ffc83ab2ab33b0b22b447e62cffef186805302/regex-2025.9.18-cp312-cp312-win_amd64.whl", hash = "sha256:3d86b5247bf25fa3715e385aa9ff272c307e0636ce0c9595f64568b41f0a9c77", size = 275501, upload-time = "2025-09-19T00:36:23.4Z" }, - { url = "https://files.pythonhosted.org/packages/df/05/9ce3e110e70d225ecbed455b966003a3afda5e58e8aec2964042363a18f4/regex-2025.9.18-cp312-cp312-win_arm64.whl", hash = "sha256:032720248cbeeae6444c269b78cb15664458b7bb9ed02401d3da59fe4d68c3a5", size = 268601, upload-time = "2025-09-19T00:36:25.092Z" }, + { url = "https://files.pythonhosted.org/packages/f7/90/4fb5056e5f03a7048abd2b11f598d464f0c167de4f2a51aa868c376b8c70/regex-2025.11.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:eadade04221641516fa25139273505a1c19f9bf97589a05bc4cfcd8b4a618031", size = 488081, upload-time = "2025-11-03T21:31:11.946Z" }, + { url = "https://files.pythonhosted.org/packages/85/23/63e481293fac8b069d84fba0299b6666df720d875110efd0338406b5d360/regex-2025.11.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:feff9e54ec0dd3833d659257f5c3f5322a12eee58ffa360984b716f8b92983f4", size = 290554, upload-time = "2025-11-03T21:31:13.387Z" }, + { url = "https://files.pythonhosted.org/packages/2b/9d/b101d0262ea293a0066b4522dfb722eb6a8785a8c3e084396a5f2c431a46/regex-2025.11.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3b30bc921d50365775c09a7ed446359e5c0179e9e2512beec4a60cbcef6ddd50", size = 288407, upload-time = "2025-11-03T21:31:14.809Z" }, + { url = "https://files.pythonhosted.org/packages/0c/64/79241c8209d5b7e00577ec9dca35cd493cc6be35b7d147eda367d6179f6d/regex-2025.11.3-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f99be08cfead2020c7ca6e396c13543baea32343b7a9a5780c462e323bd8872f", size = 793418, upload-time = "2025-11-03T21:31:16.556Z" }, + { url = "https://files.pythonhosted.org/packages/3d/e2/23cd5d3573901ce8f9757c92ca4db4d09600b865919b6d3e7f69f03b1afd/regex-2025.11.3-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:6dd329a1b61c0ee95ba95385fb0c07ea0d3fe1a21e1349fa2bec272636217118", size = 860448, upload-time = "2025-11-03T21:31:18.12Z" }, + { url = "https://files.pythonhosted.org/packages/2a/4c/aecf31beeaa416d0ae4ecb852148d38db35391aac19c687b5d56aedf3a8b/regex-2025.11.3-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:4c5238d32f3c5269d9e87be0cf096437b7622b6920f5eac4fd202468aaeb34d2", size = 907139, upload-time = "2025-11-03T21:31:20.753Z" }, + { url = "https://files.pythonhosted.org/packages/61/22/b8cb00df7d2b5e0875f60628594d44dba283e951b1ae17c12f99e332cc0a/regex-2025.11.3-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:10483eefbfb0adb18ee9474498c9a32fcf4e594fbca0543bb94c48bac6183e2e", size = 800439, upload-time = "2025-11-03T21:31:22.069Z" }, + { url = "https://files.pythonhosted.org/packages/02/a8/c4b20330a5cdc7a8eb265f9ce593f389a6a88a0c5f280cf4d978f33966bc/regex-2025.11.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:78c2d02bb6e1da0720eedc0bad578049cad3f71050ef8cd065ecc87691bed2b0", size = 782965, upload-time = "2025-11-03T21:31:23.598Z" }, + { url = "https://files.pythonhosted.org/packages/b4/4c/ae3e52988ae74af4b04d2af32fee4e8077f26e51b62ec2d12d246876bea2/regex-2025.11.3-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:e6b49cd2aad93a1790ce9cffb18964f6d3a4b0b3dbdbd5de094b65296fce6e58", size = 854398, upload-time = "2025-11-03T21:31:25.008Z" }, + { url = "https://files.pythonhosted.org/packages/06/d1/a8b9cf45874eda14b2e275157ce3b304c87e10fb38d9fc26a6e14eb18227/regex-2025.11.3-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:885b26aa3ee56433b630502dc3d36ba78d186a00cc535d3806e6bfd9ed3c70ab", size = 845897, upload-time = "2025-11-03T21:31:26.427Z" }, + { url = "https://files.pythonhosted.org/packages/ea/fe/1830eb0236be93d9b145e0bd8ab499f31602fe0999b1f19e99955aa8fe20/regex-2025.11.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ddd76a9f58e6a00f8772e72cff8ebcff78e022be95edf018766707c730593e1e", size = 788906, upload-time = "2025-11-03T21:31:28.078Z" }, + { url = "https://files.pythonhosted.org/packages/66/47/dc2577c1f95f188c1e13e2e69d8825a5ac582ac709942f8a03af42ed6e93/regex-2025.11.3-cp311-cp311-win32.whl", hash = "sha256:3e816cc9aac1cd3cc9a4ec4d860f06d40f994b5c7b4d03b93345f44e08cc68bf", size = 265812, upload-time = "2025-11-03T21:31:29.72Z" }, + { url = "https://files.pythonhosted.org/packages/50/1e/15f08b2f82a9bbb510621ec9042547b54d11e83cb620643ebb54e4eb7d71/regex-2025.11.3-cp311-cp311-win_amd64.whl", hash = "sha256:087511f5c8b7dfbe3a03f5d5ad0c2a33861b1fc387f21f6f60825a44865a385a", size = 277737, upload-time = "2025-11-03T21:31:31.422Z" }, + { url = "https://files.pythonhosted.org/packages/f4/fc/6500eb39f5f76c5e47a398df82e6b535a5e345f839581012a418b16f9cc3/regex-2025.11.3-cp311-cp311-win_arm64.whl", hash = "sha256:1ff0d190c7f68ae7769cd0313fe45820ba07ffebfddfaa89cc1eb70827ba0ddc", size = 270290, upload-time = "2025-11-03T21:31:33.041Z" }, + { url = "https://files.pythonhosted.org/packages/e8/74/18f04cb53e58e3fb107439699bd8375cf5a835eec81084e0bddbd122e4c2/regex-2025.11.3-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:bc8ab71e2e31b16e40868a40a69007bc305e1109bd4658eb6cad007e0bf67c41", size = 489312, upload-time = "2025-11-03T21:31:34.343Z" }, + { url = "https://files.pythonhosted.org/packages/78/3f/37fcdd0d2b1e78909108a876580485ea37c91e1acf66d3bb8e736348f441/regex-2025.11.3-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:22b29dda7e1f7062a52359fca6e58e548e28c6686f205e780b02ad8ef710de36", size = 291256, upload-time = "2025-11-03T21:31:35.675Z" }, + { url = "https://files.pythonhosted.org/packages/bf/26/0a575f58eb23b7ebd67a45fccbc02ac030b737b896b7e7a909ffe43ffd6a/regex-2025.11.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:3a91e4a29938bc1a082cc28fdea44be420bf2bebe2665343029723892eb073e1", size = 288921, upload-time = "2025-11-03T21:31:37.07Z" }, + { url = "https://files.pythonhosted.org/packages/ea/98/6a8dff667d1af907150432cf5abc05a17ccd32c72a3615410d5365ac167a/regex-2025.11.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:08b884f4226602ad40c5d55f52bf91a9df30f513864e0054bad40c0e9cf1afb7", size = 798568, upload-time = "2025-11-03T21:31:38.784Z" }, + { url = "https://files.pythonhosted.org/packages/64/15/92c1db4fa4e12733dd5a526c2dd2b6edcbfe13257e135fc0f6c57f34c173/regex-2025.11.3-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:3e0b11b2b2433d1c39c7c7a30e3f3d0aeeea44c2a8d0bae28f6b95f639927a69", size = 864165, upload-time = "2025-11-03T21:31:40.559Z" }, + { url = "https://files.pythonhosted.org/packages/f9/e7/3ad7da8cdee1ce66c7cd37ab5ab05c463a86ffeb52b1a25fe7bd9293b36c/regex-2025.11.3-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:87eb52a81ef58c7ba4d45c3ca74e12aa4b4e77816f72ca25258a85b3ea96cb48", size = 912182, upload-time = "2025-11-03T21:31:42.002Z" }, + { url = "https://files.pythonhosted.org/packages/84/bd/9ce9f629fcb714ffc2c3faf62b6766ecb7a585e1e885eb699bcf130a5209/regex-2025.11.3-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a12ab1f5c29b4e93db518f5e3872116b7e9b1646c9f9f426f777b50d44a09e8c", size = 803501, upload-time = "2025-11-03T21:31:43.815Z" }, + { url = "https://files.pythonhosted.org/packages/7c/0f/8dc2e4349d8e877283e6edd6c12bdcebc20f03744e86f197ab6e4492bf08/regex-2025.11.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:7521684c8c7c4f6e88e35ec89680ee1aa8358d3f09d27dfbdf62c446f5d4c695", size = 787842, upload-time = "2025-11-03T21:31:45.353Z" }, + { url = "https://files.pythonhosted.org/packages/f9/73/cff02702960bc185164d5619c0c62a2f598a6abff6695d391b096237d4ab/regex-2025.11.3-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:7fe6e5440584e94cc4b3f5f4d98a25e29ca12dccf8873679a635638349831b98", size = 858519, upload-time = "2025-11-03T21:31:46.814Z" }, + { url = "https://files.pythonhosted.org/packages/61/83/0e8d1ae71e15bc1dc36231c90b46ee35f9d52fab2e226b0e039e7ea9c10a/regex-2025.11.3-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:8e026094aa12b43f4fd74576714e987803a315c76edb6b098b9809db5de58f74", size = 850611, upload-time = "2025-11-03T21:31:48.289Z" }, + { url = "https://files.pythonhosted.org/packages/c8/f5/70a5cdd781dcfaa12556f2955bf170cd603cb1c96a1827479f8faea2df97/regex-2025.11.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:435bbad13e57eb5606a68443af62bed3556de2f46deb9f7d4237bc2f1c9fb3a0", size = 789759, upload-time = "2025-11-03T21:31:49.759Z" }, + { url = "https://files.pythonhosted.org/packages/59/9b/7c29be7903c318488983e7d97abcf8ebd3830e4c956c4c540005fcfb0462/regex-2025.11.3-cp312-cp312-win32.whl", hash = "sha256:3839967cf4dc4b985e1570fd8d91078f0c519f30491c60f9ac42a8db039be204", size = 266194, upload-time = "2025-11-03T21:31:51.53Z" }, + { url = "https://files.pythonhosted.org/packages/1a/67/3b92df89f179d7c367be654ab5626ae311cb28f7d5c237b6bb976cd5fbbb/regex-2025.11.3-cp312-cp312-win_amd64.whl", hash = "sha256:e721d1b46e25c481dc5ded6f4b3f66c897c58d2e8cfdf77bbced84339108b0b9", size = 277069, upload-time = "2025-11-03T21:31:53.151Z" }, + { url = "https://files.pythonhosted.org/packages/d7/55/85ba4c066fe5094d35b249c3ce8df0ba623cfd35afb22d6764f23a52a1c5/regex-2025.11.3-cp312-cp312-win_arm64.whl", hash = "sha256:64350685ff08b1d3a6fff33f45a9ca183dc1d58bbfe4981604e70ec9801bbc26", size = 270330, upload-time = "2025-11-03T21:31:54.514Z" }, ] [[package]] @@ -5423,52 +5426,52 @@ wheels = [ [[package]] name = "rpds-py" -version = "0.27.1" +version = "0.28.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/e9/dd/2c0cbe774744272b0ae725f44032c77bdcab6e8bcf544bffa3b6e70c8dba/rpds_py-0.27.1.tar.gz", hash = "sha256:26a1c73171d10b7acccbded82bf6a586ab8203601e565badc74bbbf8bc5a10f8", size = 27479, upload-time = "2025-08-27T12:16:36.024Z" } +sdist = { url = "https://files.pythonhosted.org/packages/48/dc/95f074d43452b3ef5d06276696ece4b3b5d696e7c9ad7173c54b1390cd70/rpds_py-0.28.0.tar.gz", hash = "sha256:abd4df20485a0983e2ca334a216249b6186d6e3c1627e106651943dbdb791aea", size = 27419, upload-time = "2025-10-22T22:24:29.327Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/b5/c1/7907329fbef97cbd49db6f7303893bd1dd5a4a3eae415839ffdfb0762cae/rpds_py-0.27.1-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:be898f271f851f68b318872ce6ebebbc62f303b654e43bf72683dbdc25b7c881", size = 371063, upload-time = "2025-08-27T12:12:47.856Z" }, - { url = "https://files.pythonhosted.org/packages/11/94/2aab4bc86228bcf7c48760990273653a4900de89c7537ffe1b0d6097ed39/rpds_py-0.27.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:62ac3d4e3e07b58ee0ddecd71d6ce3b1637de2d373501412df395a0ec5f9beb5", size = 353210, upload-time = "2025-08-27T12:12:49.187Z" }, - { url = "https://files.pythonhosted.org/packages/3a/57/f5eb3ecf434342f4f1a46009530e93fd201a0b5b83379034ebdb1d7c1a58/rpds_py-0.27.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4708c5c0ceb2d034f9991623631d3d23cb16e65c83736ea020cdbe28d57c0a0e", size = 381636, upload-time = "2025-08-27T12:12:50.492Z" }, - { url = "https://files.pythonhosted.org/packages/ae/f4/ef95c5945e2ceb5119571b184dd5a1cc4b8541bbdf67461998cfeac9cb1e/rpds_py-0.27.1-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:abfa1171a9952d2e0002aba2ad3780820b00cc3d9c98c6630f2e93271501f66c", size = 394341, upload-time = "2025-08-27T12:12:52.024Z" }, - { url = "https://files.pythonhosted.org/packages/5a/7e/4bd610754bf492d398b61725eb9598ddd5eb86b07d7d9483dbcd810e20bc/rpds_py-0.27.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4b507d19f817ebaca79574b16eb2ae412e5c0835542c93fe9983f1e432aca195", size = 523428, upload-time = "2025-08-27T12:12:53.779Z" }, - { url = "https://files.pythonhosted.org/packages/9f/e5/059b9f65a8c9149361a8b75094864ab83b94718344db511fd6117936ed2a/rpds_py-0.27.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:168b025f8fd8d8d10957405f3fdcef3dc20f5982d398f90851f4abc58c566c52", size = 402923, upload-time = "2025-08-27T12:12:55.15Z" }, - { url = "https://files.pythonhosted.org/packages/f5/48/64cabb7daced2968dd08e8a1b7988bf358d7bd5bcd5dc89a652f4668543c/rpds_py-0.27.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cb56c6210ef77caa58e16e8c17d35c63fe3f5b60fd9ba9d424470c3400bcf9ed", size = 384094, upload-time = "2025-08-27T12:12:57.194Z" }, - { url = "https://files.pythonhosted.org/packages/ae/e1/dc9094d6ff566bff87add8a510c89b9e158ad2ecd97ee26e677da29a9e1b/rpds_py-0.27.1-cp311-cp311-manylinux_2_31_riscv64.whl", hash = "sha256:d252f2d8ca0195faa707f8eb9368955760880b2b42a8ee16d382bf5dd807f89a", size = 401093, upload-time = "2025-08-27T12:12:58.985Z" }, - { url = "https://files.pythonhosted.org/packages/37/8e/ac8577e3ecdd5593e283d46907d7011618994e1d7ab992711ae0f78b9937/rpds_py-0.27.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:6e5e54da1e74b91dbc7996b56640f79b195d5925c2b78efaa8c5d53e1d88edde", size = 417969, upload-time = "2025-08-27T12:13:00.367Z" }, - { url = "https://files.pythonhosted.org/packages/66/6d/87507430a8f74a93556fe55c6485ba9c259949a853ce407b1e23fea5ba31/rpds_py-0.27.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:ffce0481cc6e95e5b3f0a47ee17ffbd234399e6d532f394c8dce320c3b089c21", size = 558302, upload-time = "2025-08-27T12:13:01.737Z" }, - { url = "https://files.pythonhosted.org/packages/3a/bb/1db4781ce1dda3eecc735e3152659a27b90a02ca62bfeea17aee45cc0fbc/rpds_py-0.27.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:a205fdfe55c90c2cd8e540ca9ceba65cbe6629b443bc05db1f590a3db8189ff9", size = 589259, upload-time = "2025-08-27T12:13:03.127Z" }, - { url = "https://files.pythonhosted.org/packages/7b/0e/ae1c8943d11a814d01b482e1f8da903f88047a962dff9bbdadf3bd6e6fd1/rpds_py-0.27.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:689fb5200a749db0415b092972e8eba85847c23885c8543a8b0f5c009b1a5948", size = 554983, upload-time = "2025-08-27T12:13:04.516Z" }, - { url = "https://files.pythonhosted.org/packages/b2/d5/0b2a55415931db4f112bdab072443ff76131b5ac4f4dc98d10d2d357eb03/rpds_py-0.27.1-cp311-cp311-win32.whl", hash = "sha256:3182af66048c00a075010bc7f4860f33913528a4b6fc09094a6e7598e462fe39", size = 217154, upload-time = "2025-08-27T12:13:06.278Z" }, - { url = "https://files.pythonhosted.org/packages/24/75/3b7ffe0d50dc86a6a964af0d1cc3a4a2cdf437cb7b099a4747bbb96d1819/rpds_py-0.27.1-cp311-cp311-win_amd64.whl", hash = "sha256:b4938466c6b257b2f5c4ff98acd8128ec36b5059e5c8f8372d79316b1c36bb15", size = 228627, upload-time = "2025-08-27T12:13:07.625Z" }, - { url = "https://files.pythonhosted.org/packages/8d/3f/4fd04c32abc02c710f09a72a30c9a55ea3cc154ef8099078fd50a0596f8e/rpds_py-0.27.1-cp311-cp311-win_arm64.whl", hash = "sha256:2f57af9b4d0793e53266ee4325535a31ba48e2f875da81a9177c9926dfa60746", size = 220998, upload-time = "2025-08-27T12:13:08.972Z" }, - { url = "https://files.pythonhosted.org/packages/bd/fe/38de28dee5df58b8198c743fe2bea0c785c6d40941b9950bac4cdb71a014/rpds_py-0.27.1-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:ae2775c1973e3c30316892737b91f9283f9908e3cc7625b9331271eaaed7dc90", size = 361887, upload-time = "2025-08-27T12:13:10.233Z" }, - { url = "https://files.pythonhosted.org/packages/7c/9a/4b6c7eedc7dd90986bf0fab6ea2a091ec11c01b15f8ba0a14d3f80450468/rpds_py-0.27.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2643400120f55c8a96f7c9d858f7be0c88d383cd4653ae2cf0d0c88f668073e5", size = 345795, upload-time = "2025-08-27T12:13:11.65Z" }, - { url = "https://files.pythonhosted.org/packages/6f/0e/e650e1b81922847a09cca820237b0edee69416a01268b7754d506ade11ad/rpds_py-0.27.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:16323f674c089b0360674a4abd28d5042947d54ba620f72514d69be4ff64845e", size = 385121, upload-time = "2025-08-27T12:13:13.008Z" }, - { url = "https://files.pythonhosted.org/packages/1b/ea/b306067a712988e2bff00dcc7c8f31d26c29b6d5931b461aa4b60a013e33/rpds_py-0.27.1-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:9a1f4814b65eacac94a00fc9a526e3fdafd78e439469644032032d0d63de4881", size = 398976, upload-time = "2025-08-27T12:13:14.368Z" }, - { url = "https://files.pythonhosted.org/packages/2c/0a/26dc43c8840cb8fe239fe12dbc8d8de40f2365e838f3d395835dde72f0e5/rpds_py-0.27.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7ba32c16b064267b22f1850a34051121d423b6f7338a12b9459550eb2096e7ec", size = 525953, upload-time = "2025-08-27T12:13:15.774Z" }, - { url = "https://files.pythonhosted.org/packages/22/14/c85e8127b573aaf3a0cbd7fbb8c9c99e735a4a02180c84da2a463b766e9e/rpds_py-0.27.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e5c20f33fd10485b80f65e800bbe5f6785af510b9f4056c5a3c612ebc83ba6cb", size = 407915, upload-time = "2025-08-27T12:13:17.379Z" }, - { url = "https://files.pythonhosted.org/packages/ed/7b/8f4fee9ba1fb5ec856eb22d725a4efa3deb47f769597c809e03578b0f9d9/rpds_py-0.27.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:466bfe65bd932da36ff279ddd92de56b042f2266d752719beb97b08526268ec5", size = 386883, upload-time = "2025-08-27T12:13:18.704Z" }, - { url = "https://files.pythonhosted.org/packages/86/47/28fa6d60f8b74fcdceba81b272f8d9836ac0340570f68f5df6b41838547b/rpds_py-0.27.1-cp312-cp312-manylinux_2_31_riscv64.whl", hash = "sha256:41e532bbdcb57c92ba3be62c42e9f096431b4cf478da9bc3bc6ce5c38ab7ba7a", size = 405699, upload-time = "2025-08-27T12:13:20.089Z" }, - { url = "https://files.pythonhosted.org/packages/d0/fd/c5987b5e054548df56953a21fe2ebed51fc1ec7c8f24fd41c067b68c4a0a/rpds_py-0.27.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:f149826d742b406579466283769a8ea448eed82a789af0ed17b0cd5770433444", size = 423713, upload-time = "2025-08-27T12:13:21.436Z" }, - { url = "https://files.pythonhosted.org/packages/ac/ba/3c4978b54a73ed19a7d74531be37a8bcc542d917c770e14d372b8daea186/rpds_py-0.27.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:80c60cfb5310677bd67cb1e85a1e8eb52e12529545441b43e6f14d90b878775a", size = 562324, upload-time = "2025-08-27T12:13:22.789Z" }, - { url = "https://files.pythonhosted.org/packages/b5/6c/6943a91768fec16db09a42b08644b960cff540c66aab89b74be6d4a144ba/rpds_py-0.27.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:7ee6521b9baf06085f62ba9c7a3e5becffbc32480d2f1b351559c001c38ce4c1", size = 593646, upload-time = "2025-08-27T12:13:24.122Z" }, - { url = "https://files.pythonhosted.org/packages/11/73/9d7a8f4be5f4396f011a6bb7a19fe26303a0dac9064462f5651ced2f572f/rpds_py-0.27.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a512c8263249a9d68cac08b05dd59d2b3f2061d99b322813cbcc14c3c7421998", size = 558137, upload-time = "2025-08-27T12:13:25.557Z" }, - { url = "https://files.pythonhosted.org/packages/6e/96/6772cbfa0e2485bcceef8071de7821f81aeac8bb45fbfd5542a3e8108165/rpds_py-0.27.1-cp312-cp312-win32.whl", hash = "sha256:819064fa048ba01b6dadc5116f3ac48610435ac9a0058bbde98e569f9e785c39", size = 221343, upload-time = "2025-08-27T12:13:26.967Z" }, - { url = "https://files.pythonhosted.org/packages/67/b6/c82f0faa9af1c6a64669f73a17ee0eeef25aff30bb9a1c318509efe45d84/rpds_py-0.27.1-cp312-cp312-win_amd64.whl", hash = "sha256:d9199717881f13c32c4046a15f024971a3b78ad4ea029e8da6b86e5aa9cf4594", size = 232497, upload-time = "2025-08-27T12:13:28.326Z" }, - { url = "https://files.pythonhosted.org/packages/e1/96/2817b44bd2ed11aebacc9251da03689d56109b9aba5e311297b6902136e2/rpds_py-0.27.1-cp312-cp312-win_arm64.whl", hash = "sha256:33aa65b97826a0e885ef6e278fbd934e98cdcfed80b63946025f01e2f5b29502", size = 222790, upload-time = "2025-08-27T12:13:29.71Z" }, - { url = "https://files.pythonhosted.org/packages/0c/ed/e1fba02de17f4f76318b834425257c8ea297e415e12c68b4361f63e8ae92/rpds_py-0.27.1-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:cdfe4bb2f9fe7458b7453ad3c33e726d6d1c7c0a72960bcc23800d77384e42df", size = 371402, upload-time = "2025-08-27T12:15:51.561Z" }, - { url = "https://files.pythonhosted.org/packages/af/7c/e16b959b316048b55585a697e94add55a4ae0d984434d279ea83442e460d/rpds_py-0.27.1-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:8fabb8fd848a5f75a2324e4a84501ee3a5e3c78d8603f83475441866e60b94a3", size = 354084, upload-time = "2025-08-27T12:15:53.219Z" }, - { url = "https://files.pythonhosted.org/packages/de/c1/ade645f55de76799fdd08682d51ae6724cb46f318573f18be49b1e040428/rpds_py-0.27.1-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:eda8719d598f2f7f3e0f885cba8646644b55a187762bec091fa14a2b819746a9", size = 383090, upload-time = "2025-08-27T12:15:55.158Z" }, - { url = "https://files.pythonhosted.org/packages/1f/27/89070ca9b856e52960da1472efcb6c20ba27cfe902f4f23ed095b9cfc61d/rpds_py-0.27.1-pp311-pypy311_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:3c64d07e95606ec402a0a1c511fe003873fa6af630bda59bac77fac8b4318ebc", size = 394519, upload-time = "2025-08-27T12:15:57.238Z" }, - { url = "https://files.pythonhosted.org/packages/b3/28/be120586874ef906aa5aeeae95ae8df4184bc757e5b6bd1c729ccff45ed5/rpds_py-0.27.1-pp311-pypy311_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:93a2ed40de81bcff59aabebb626562d48332f3d028ca2036f1d23cbb52750be4", size = 523817, upload-time = "2025-08-27T12:15:59.237Z" }, - { url = "https://files.pythonhosted.org/packages/a8/ef/70cc197bc11cfcde02a86f36ac1eed15c56667c2ebddbdb76a47e90306da/rpds_py-0.27.1-pp311-pypy311_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:387ce8c44ae94e0ec50532d9cb0edce17311024c9794eb196b90e1058aadeb66", size = 403240, upload-time = "2025-08-27T12:16:00.923Z" }, - { url = "https://files.pythonhosted.org/packages/cf/35/46936cca449f7f518f2f4996e0e8344db4b57e2081e752441154089d2a5f/rpds_py-0.27.1-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aaf94f812c95b5e60ebaf8bfb1898a7d7cb9c1af5744d4a67fa47796e0465d4e", size = 385194, upload-time = "2025-08-27T12:16:02.802Z" }, - { url = "https://files.pythonhosted.org/packages/e1/62/29c0d3e5125c3270b51415af7cbff1ec587379c84f55a5761cc9efa8cd06/rpds_py-0.27.1-pp311-pypy311_pp73-manylinux_2_31_riscv64.whl", hash = "sha256:4848ca84d6ded9b58e474dfdbad4b8bfb450344c0551ddc8d958bf4b36aa837c", size = 402086, upload-time = "2025-08-27T12:16:04.806Z" }, - { url = "https://files.pythonhosted.org/packages/8f/66/03e1087679227785474466fdd04157fb793b3b76e3fcf01cbf4c693c1949/rpds_py-0.27.1-pp311-pypy311_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2bde09cbcf2248b73c7c323be49b280180ff39fadcfe04e7b6f54a678d02a7cf", size = 419272, upload-time = "2025-08-27T12:16:06.471Z" }, - { url = "https://files.pythonhosted.org/packages/6a/24/e3e72d265121e00b063aef3e3501e5b2473cf1b23511d56e529531acf01e/rpds_py-0.27.1-pp311-pypy311_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:94c44ee01fd21c9058f124d2d4f0c9dc7634bec93cd4b38eefc385dabe71acbf", size = 560003, upload-time = "2025-08-27T12:16:08.06Z" }, - { url = "https://files.pythonhosted.org/packages/26/ca/f5a344c534214cc2d41118c0699fffbdc2c1bc7046f2a2b9609765ab9c92/rpds_py-0.27.1-pp311-pypy311_pp73-musllinux_1_2_i686.whl", hash = "sha256:df8b74962e35c9249425d90144e721eed198e6555a0e22a563d29fe4486b51f6", size = 590482, upload-time = "2025-08-27T12:16:10.137Z" }, - { url = "https://files.pythonhosted.org/packages/ce/08/4349bdd5c64d9d193c360aa9db89adeee6f6682ab8825dca0a3f535f434f/rpds_py-0.27.1-pp311-pypy311_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:dc23e6820e3b40847e2f4a7726462ba0cf53089512abe9ee16318c366494c17a", size = 556523, upload-time = "2025-08-27T12:16:12.188Z" }, + { url = "https://files.pythonhosted.org/packages/a6/34/058d0db5471c6be7bef82487ad5021ff8d1d1d27794be8730aad938649cf/rpds_py-0.28.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:03065002fd2e287725d95fbc69688e0c6daf6c6314ba38bdbaa3895418e09296", size = 362344, upload-time = "2025-10-22T22:21:39.713Z" }, + { url = "https://files.pythonhosted.org/packages/5d/67/9503f0ec8c055a0782880f300c50a2b8e5e72eb1f94dfc2053da527444dd/rpds_py-0.28.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:28ea02215f262b6d078daec0b45344c89e161eab9526b0d898221d96fdda5f27", size = 348440, upload-time = "2025-10-22T22:21:41.056Z" }, + { url = "https://files.pythonhosted.org/packages/68/2e/94223ee9b32332a41d75b6f94b37b4ce3e93878a556fc5f152cbd856a81f/rpds_py-0.28.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:25dbade8fbf30bcc551cb352376c0ad64b067e4fc56f90e22ba70c3ce205988c", size = 379068, upload-time = "2025-10-22T22:21:42.593Z" }, + { url = "https://files.pythonhosted.org/packages/b4/25/54fd48f9f680cfc44e6a7f39a5fadf1d4a4a1fd0848076af4a43e79f998c/rpds_py-0.28.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:3c03002f54cc855860bfdc3442928ffdca9081e73b5b382ed0b9e8efe6e5e205", size = 390518, upload-time = "2025-10-22T22:21:43.998Z" }, + { url = "https://files.pythonhosted.org/packages/1b/85/ac258c9c27f2ccb1bd5d0697e53a82ebcf8088e3186d5d2bf8498ee7ed44/rpds_py-0.28.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b9699fa7990368b22032baf2b2dce1f634388e4ffc03dfefaaac79f4695edc95", size = 525319, upload-time = "2025-10-22T22:21:45.645Z" }, + { url = "https://files.pythonhosted.org/packages/40/cb/c6734774789566d46775f193964b76627cd5f42ecf246d257ce84d1912ed/rpds_py-0.28.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b9b06fe1a75e05e0713f06ea0c89ecb6452210fd60e2f1b6ddc1067b990e08d9", size = 404896, upload-time = "2025-10-22T22:21:47.544Z" }, + { url = "https://files.pythonhosted.org/packages/1f/53/14e37ce83202c632c89b0691185dca9532288ff9d390eacae3d2ff771bae/rpds_py-0.28.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ac9f83e7b326a3f9ec3ef84cda98fb0a74c7159f33e692032233046e7fd15da2", size = 382862, upload-time = "2025-10-22T22:21:49.176Z" }, + { url = "https://files.pythonhosted.org/packages/6a/83/f3642483ca971a54d60caa4449f9d6d4dbb56a53e0072d0deff51b38af74/rpds_py-0.28.0-cp311-cp311-manylinux_2_31_riscv64.whl", hash = "sha256:0d3259ea9ad8743a75a43eb7819324cdab393263c91be86e2d1901ee65c314e0", size = 398848, upload-time = "2025-10-22T22:21:51.024Z" }, + { url = "https://files.pythonhosted.org/packages/44/09/2d9c8b2f88e399b4cfe86efdf2935feaf0394e4f14ab30c6c5945d60af7d/rpds_py-0.28.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:9a7548b345f66f6695943b4ef6afe33ccd3f1b638bd9afd0f730dd255c249c9e", size = 412030, upload-time = "2025-10-22T22:21:52.665Z" }, + { url = "https://files.pythonhosted.org/packages/dd/f5/e1cec473d4bde6df1fd3738be8e82d64dd0600868e76e92dfeaebbc2d18f/rpds_py-0.28.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:c9a40040aa388b037eb39416710fbcce9443498d2eaab0b9b45ae988b53f5c67", size = 559700, upload-time = "2025-10-22T22:21:54.123Z" }, + { url = "https://files.pythonhosted.org/packages/8d/be/73bb241c1649edbf14e98e9e78899c2c5e52bbe47cb64811f44d2cc11808/rpds_py-0.28.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:8f60c7ea34e78c199acd0d3cda37a99be2c861dd2b8cf67399784f70c9f8e57d", size = 584581, upload-time = "2025-10-22T22:21:56.102Z" }, + { url = "https://files.pythonhosted.org/packages/9c/9c/ffc6e9218cd1eb5c2c7dbd276c87cd10e8c2232c456b554169eb363381df/rpds_py-0.28.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:1571ae4292649100d743b26d5f9c63503bb1fedf538a8f29a98dce2d5ba6b4e6", size = 549981, upload-time = "2025-10-22T22:21:58.253Z" }, + { url = "https://files.pythonhosted.org/packages/5f/50/da8b6d33803a94df0149345ee33e5d91ed4d25fc6517de6a25587eae4133/rpds_py-0.28.0-cp311-cp311-win32.whl", hash = "sha256:5cfa9af45e7c1140af7321fa0bef25b386ee9faa8928c80dc3a5360971a29e8c", size = 214729, upload-time = "2025-10-22T22:21:59.625Z" }, + { url = "https://files.pythonhosted.org/packages/12/fd/b0f48c4c320ee24c8c20df8b44acffb7353991ddf688af01eef5f93d7018/rpds_py-0.28.0-cp311-cp311-win_amd64.whl", hash = "sha256:dd8d86b5d29d1b74100982424ba53e56033dc47720a6de9ba0259cf81d7cecaa", size = 223977, upload-time = "2025-10-22T22:22:01.092Z" }, + { url = "https://files.pythonhosted.org/packages/b4/21/c8e77a2ac66e2ec4e21f18a04b4e9a0417ecf8e61b5eaeaa9360a91713b4/rpds_py-0.28.0-cp311-cp311-win_arm64.whl", hash = "sha256:4e27d3a5709cc2b3e013bf93679a849213c79ae0573f9b894b284b55e729e120", size = 217326, upload-time = "2025-10-22T22:22:02.944Z" }, + { url = "https://files.pythonhosted.org/packages/b8/5c/6c3936495003875fe7b14f90ea812841a08fca50ab26bd840e924097d9c8/rpds_py-0.28.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:6b4f28583a4f247ff60cd7bdda83db8c3f5b05a7a82ff20dd4b078571747708f", size = 366439, upload-time = "2025-10-22T22:22:04.525Z" }, + { url = "https://files.pythonhosted.org/packages/56/f9/a0f1ca194c50aa29895b442771f036a25b6c41a35e4f35b1a0ea713bedae/rpds_py-0.28.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:d678e91b610c29c4b3d52a2c148b641df2b4676ffe47c59f6388d58b99cdc424", size = 348170, upload-time = "2025-10-22T22:22:06.397Z" }, + { url = "https://files.pythonhosted.org/packages/18/ea/42d243d3a586beb72c77fa5def0487daf827210069a95f36328e869599ea/rpds_py-0.28.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e819e0e37a44a78e1383bf1970076e2ccc4dc8c2bbaa2f9bd1dc987e9afff628", size = 378838, upload-time = "2025-10-22T22:22:07.932Z" }, + { url = "https://files.pythonhosted.org/packages/e7/78/3de32e18a94791af8f33601402d9d4f39613136398658412a4e0b3047327/rpds_py-0.28.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5ee514e0f0523db5d3fb171f397c54875dbbd69760a414dccf9d4d7ad628b5bd", size = 393299, upload-time = "2025-10-22T22:22:09.435Z" }, + { url = "https://files.pythonhosted.org/packages/13/7e/4bdb435afb18acea2eb8a25ad56b956f28de7c59f8a1d32827effa0d4514/rpds_py-0.28.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5f3fa06d27fdcee47f07a39e02862da0100cb4982508f5ead53ec533cd5fe55e", size = 518000, upload-time = "2025-10-22T22:22:11.326Z" }, + { url = "https://files.pythonhosted.org/packages/31/d0/5f52a656875cdc60498ab035a7a0ac8f399890cc1ee73ebd567bac4e39ae/rpds_py-0.28.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:46959ef2e64f9e4a41fc89aa20dbca2b85531f9a72c21099a3360f35d10b0d5a", size = 408746, upload-time = "2025-10-22T22:22:13.143Z" }, + { url = "https://files.pythonhosted.org/packages/3e/cd/49ce51767b879cde77e7ad9fae164ea15dce3616fe591d9ea1df51152706/rpds_py-0.28.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8455933b4bcd6e83fde3fefc987a023389c4b13f9a58c8d23e4b3f6d13f78c84", size = 386379, upload-time = "2025-10-22T22:22:14.602Z" }, + { url = "https://files.pythonhosted.org/packages/6a/99/e4e1e1ee93a98f72fc450e36c0e4d99c35370220e815288e3ecd2ec36a2a/rpds_py-0.28.0-cp312-cp312-manylinux_2_31_riscv64.whl", hash = "sha256:ad50614a02c8c2962feebe6012b52f9802deec4263946cddea37aaf28dd25a66", size = 401280, upload-time = "2025-10-22T22:22:16.063Z" }, + { url = "https://files.pythonhosted.org/packages/61/35/e0c6a57488392a8b319d2200d03dad2b29c0db9996f5662c3b02d0b86c02/rpds_py-0.28.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:e5deca01b271492553fdb6c7fd974659dce736a15bae5dad7ab8b93555bceb28", size = 412365, upload-time = "2025-10-22T22:22:17.504Z" }, + { url = "https://files.pythonhosted.org/packages/ff/6a/841337980ea253ec797eb084665436007a1aad0faac1ba097fb906c5f69c/rpds_py-0.28.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:735f8495a13159ce6a0d533f01e8674cec0c57038c920495f87dcb20b3ddb48a", size = 559573, upload-time = "2025-10-22T22:22:19.108Z" }, + { url = "https://files.pythonhosted.org/packages/e7/5e/64826ec58afd4c489731f8b00729c5f6afdb86f1df1df60bfede55d650bb/rpds_py-0.28.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:961ca621ff10d198bbe6ba4957decca61aa2a0c56695384c1d6b79bf61436df5", size = 583973, upload-time = "2025-10-22T22:22:20.768Z" }, + { url = "https://files.pythonhosted.org/packages/b6/ee/44d024b4843f8386a4eeaa4c171b3d31d55f7177c415545fd1a24c249b5d/rpds_py-0.28.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:2374e16cc9131022e7d9a8f8d65d261d9ba55048c78f3b6e017971a4f5e6353c", size = 553800, upload-time = "2025-10-22T22:22:22.25Z" }, + { url = "https://files.pythonhosted.org/packages/7d/89/33e675dccff11a06d4d85dbb4d1865f878d5020cbb69b2c1e7b2d3f82562/rpds_py-0.28.0-cp312-cp312-win32.whl", hash = "sha256:d15431e334fba488b081d47f30f091e5d03c18527c325386091f31718952fe08", size = 216954, upload-time = "2025-10-22T22:22:24.105Z" }, + { url = "https://files.pythonhosted.org/packages/af/36/45f6ebb3210887e8ee6dbf1bc710ae8400bb417ce165aaf3024b8360d999/rpds_py-0.28.0-cp312-cp312-win_amd64.whl", hash = "sha256:a410542d61fc54710f750d3764380b53bf09e8c4edbf2f9141a82aa774a04f7c", size = 227844, upload-time = "2025-10-22T22:22:25.551Z" }, + { url = "https://files.pythonhosted.org/packages/57/91/f3fb250d7e73de71080f9a221d19bd6a1c1eb0d12a1ea26513f6c1052ad6/rpds_py-0.28.0-cp312-cp312-win_arm64.whl", hash = "sha256:1f0cfd1c69e2d14f8c892b893997fa9a60d890a0c8a603e88dca4955f26d1edd", size = 217624, upload-time = "2025-10-22T22:22:26.914Z" }, + { url = "https://files.pythonhosted.org/packages/ae/bc/b43f2ea505f28119bd551ae75f70be0c803d2dbcd37c1b3734909e40620b/rpds_py-0.28.0-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:f5e7101145427087e493b9c9b959da68d357c28c562792300dd21a095118ed16", size = 363913, upload-time = "2025-10-22T22:24:07.129Z" }, + { url = "https://files.pythonhosted.org/packages/28/f2/db318195d324c89a2c57dc5195058cbadd71b20d220685c5bd1da79ee7fe/rpds_py-0.28.0-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:31eb671150b9c62409a888850aaa8e6533635704fe2b78335f9aaf7ff81eec4d", size = 350452, upload-time = "2025-10-22T22:24:08.754Z" }, + { url = "https://files.pythonhosted.org/packages/ae/f2/1391c819b8573a4898cedd6b6c5ec5bc370ce59e5d6bdcebe3c9c1db4588/rpds_py-0.28.0-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:48b55c1f64482f7d8bd39942f376bfdf2f6aec637ee8c805b5041e14eeb771db", size = 380957, upload-time = "2025-10-22T22:24:10.826Z" }, + { url = "https://files.pythonhosted.org/packages/5a/5c/e5de68ee7eb7248fce93269833d1b329a196d736aefb1a7481d1e99d1222/rpds_py-0.28.0-pp311-pypy311_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:24743a7b372e9a76171f6b69c01aedf927e8ac3e16c474d9fe20d552a8cb45c7", size = 391919, upload-time = "2025-10-22T22:24:12.559Z" }, + { url = "https://files.pythonhosted.org/packages/fb/4f/2376336112cbfeb122fd435d608ad8d5041b3aed176f85a3cb32c262eb80/rpds_py-0.28.0-pp311-pypy311_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:389c29045ee8bbb1627ea190b4976a310a295559eaf9f1464a1a6f2bf84dde78", size = 528541, upload-time = "2025-10-22T22:24:14.197Z" }, + { url = "https://files.pythonhosted.org/packages/68/53/5ae232e795853dd20da7225c5dd13a09c0a905b1a655e92bdf8d78a99fd9/rpds_py-0.28.0-pp311-pypy311_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:23690b5827e643150cf7b49569679ec13fe9a610a15949ed48b85eb7f98f34ec", size = 405629, upload-time = "2025-10-22T22:24:16.001Z" }, + { url = "https://files.pythonhosted.org/packages/b9/2d/351a3b852b683ca9b6b8b38ed9efb2347596973849ba6c3a0e99877c10aa/rpds_py-0.28.0-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6f0c9266c26580e7243ad0d72fc3e01d6b33866cfab5084a6da7576bcf1c4f72", size = 384123, upload-time = "2025-10-22T22:24:17.585Z" }, + { url = "https://files.pythonhosted.org/packages/e0/15/870804daa00202728cc91cb8e2385fa9f1f4eb49857c49cfce89e304eae6/rpds_py-0.28.0-pp311-pypy311_pp73-manylinux_2_31_riscv64.whl", hash = "sha256:4c6c4db5d73d179746951486df97fd25e92396be07fc29ee8ff9a8f5afbdfb27", size = 400923, upload-time = "2025-10-22T22:24:19.512Z" }, + { url = "https://files.pythonhosted.org/packages/53/25/3706b83c125fa2a0bccceac951de3f76631f6bd0ee4d02a0ed780712ef1b/rpds_py-0.28.0-pp311-pypy311_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a3b695a8fa799dd2cfdb4804b37096c5f6dba1ac7f48a7fbf6d0485bcd060316", size = 413767, upload-time = "2025-10-22T22:24:21.316Z" }, + { url = "https://files.pythonhosted.org/packages/ef/f9/ce43dbe62767432273ed2584cef71fef8411bddfb64125d4c19128015018/rpds_py-0.28.0-pp311-pypy311_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:6aa1bfce3f83baf00d9c5fcdbba93a3ab79958b4c7d7d1f55e7fe68c20e63912", size = 561530, upload-time = "2025-10-22T22:24:22.958Z" }, + { url = "https://files.pythonhosted.org/packages/46/c9/ffe77999ed8f81e30713dd38fd9ecaa161f28ec48bb80fa1cd9118399c27/rpds_py-0.28.0-pp311-pypy311_pp73-musllinux_1_2_i686.whl", hash = "sha256:7b0f9dceb221792b3ee6acb5438eb1f02b0cb2c247796a72b016dcc92c6de829", size = 585453, upload-time = "2025-10-22T22:24:24.779Z" }, + { url = "https://files.pythonhosted.org/packages/ed/d2/4a73b18821fd4669762c855fd1f4e80ceb66fb72d71162d14da58444a763/rpds_py-0.28.0-pp311-pypy311_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:5d0145edba8abd3db0ab22b5300c99dc152f5c9021fab861be0f0544dc3cbc5f", size = 552199, upload-time = "2025-10-22T22:24:26.54Z" }, ] [[package]] @@ -5485,28 +5488,28 @@ wheels = [ [[package]] name = "ruff" -version = "0.14.0" +version = "0.14.4" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/41/b9/9bd84453ed6dd04688de9b3f3a4146a1698e8faae2ceeccce4e14c67ae17/ruff-0.14.0.tar.gz", hash = "sha256:62ec8969b7510f77945df916de15da55311fade8d6050995ff7f680afe582c57", size = 5452071, upload-time = "2025-10-07T18:21:55.763Z" } +sdist = { url = "https://files.pythonhosted.org/packages/df/55/cccfca45157a2031dcbb5a462a67f7cf27f8b37d4b3b1cd7438f0f5c1df6/ruff-0.14.4.tar.gz", hash = "sha256:f459a49fe1085a749f15414ca76f61595f1a2cc8778ed7c279b6ca2e1fd19df3", size = 5587844, upload-time = "2025-11-06T22:07:45.033Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/3a/4e/79d463a5f80654e93fa653ebfb98e0becc3f0e7cf6219c9ddedf1e197072/ruff-0.14.0-py3-none-linux_armv6l.whl", hash = "sha256:58e15bffa7054299becf4bab8a1187062c6f8cafbe9f6e39e0d5aface455d6b3", size = 12494532, upload-time = "2025-10-07T18:21:00.373Z" }, - { url = "https://files.pythonhosted.org/packages/ee/40/e2392f445ed8e02aa6105d49db4bfff01957379064c30f4811c3bf38aece/ruff-0.14.0-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:838d1b065f4df676b7c9957992f2304e41ead7a50a568185efd404297d5701e8", size = 13160768, upload-time = "2025-10-07T18:21:04.73Z" }, - { url = "https://files.pythonhosted.org/packages/75/da/2a656ea7c6b9bd14c7209918268dd40e1e6cea65f4bb9880eaaa43b055cd/ruff-0.14.0-py3-none-macosx_11_0_arm64.whl", hash = "sha256:703799d059ba50f745605b04638fa7e9682cc3da084b2092feee63500ff3d9b8", size = 12363376, upload-time = "2025-10-07T18:21:07.833Z" }, - { url = "https://files.pythonhosted.org/packages/42/e2/1ffef5a1875add82416ff388fcb7ea8b22a53be67a638487937aea81af27/ruff-0.14.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3ba9a8925e90f861502f7d974cc60e18ca29c72bb0ee8bfeabb6ade35a3abde7", size = 12608055, upload-time = "2025-10-07T18:21:10.72Z" }, - { url = "https://files.pythonhosted.org/packages/4a/32/986725199d7cee510d9f1dfdf95bf1efc5fa9dd714d0d85c1fb1f6be3bc3/ruff-0.14.0-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:e41f785498bd200ffc276eb9e1570c019c1d907b07cfb081092c8ad51975bbe7", size = 12318544, upload-time = "2025-10-07T18:21:13.741Z" }, - { url = "https://files.pythonhosted.org/packages/9a/ed/4969cefd53315164c94eaf4da7cfba1f267dc275b0abdd593d11c90829a3/ruff-0.14.0-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:30a58c087aef4584c193aebf2700f0fbcfc1e77b89c7385e3139956fa90434e2", size = 14001280, upload-time = "2025-10-07T18:21:16.411Z" }, - { url = "https://files.pythonhosted.org/packages/ab/ad/96c1fc9f8854c37681c9613d825925c7f24ca1acfc62a4eb3896b50bacd2/ruff-0.14.0-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:f8d07350bc7af0a5ce8812b7d5c1a7293cf02476752f23fdfc500d24b79b783c", size = 15027286, upload-time = "2025-10-07T18:21:19.577Z" }, - { url = "https://files.pythonhosted.org/packages/b3/00/1426978f97df4fe331074baf69615f579dc4e7c37bb4c6f57c2aad80c87f/ruff-0.14.0-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:eec3bbbf3a7d5482b5c1f42d5fc972774d71d107d447919fca620b0be3e3b75e", size = 14451506, upload-time = "2025-10-07T18:21:22.779Z" }, - { url = "https://files.pythonhosted.org/packages/58/d5/9c1cea6e493c0cf0647674cca26b579ea9d2a213b74b5c195fbeb9678e15/ruff-0.14.0-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:16b68e183a0e28e5c176d51004aaa40559e8f90065a10a559176713fcf435206", size = 13437384, upload-time = "2025-10-07T18:21:25.758Z" }, - { url = "https://files.pythonhosted.org/packages/29/b4/4cd6a4331e999fc05d9d77729c95503f99eae3ba1160469f2b64866964e3/ruff-0.14.0-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:eb732d17db2e945cfcbbc52af0143eda1da36ca8ae25083dd4f66f1542fdf82e", size = 13447976, upload-time = "2025-10-07T18:21:28.83Z" }, - { url = "https://files.pythonhosted.org/packages/3b/c0/ac42f546d07e4f49f62332576cb845d45c67cf5610d1851254e341d563b6/ruff-0.14.0-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:c958f66ab884b7873e72df38dcabee03d556a8f2ee1b8538ee1c2bbd619883dd", size = 13682850, upload-time = "2025-10-07T18:21:31.842Z" }, - { url = "https://files.pythonhosted.org/packages/5f/c4/4b0c9bcadd45b4c29fe1af9c5d1dc0ca87b4021665dfbe1c4688d407aa20/ruff-0.14.0-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:7eb0499a2e01f6e0c285afc5bac43ab380cbfc17cd43a2e1dd10ec97d6f2c42d", size = 12449825, upload-time = "2025-10-07T18:21:35.074Z" }, - { url = "https://files.pythonhosted.org/packages/4b/a8/e2e76288e6c16540fa820d148d83e55f15e994d852485f221b9524514730/ruff-0.14.0-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:4c63b2d99fafa05efca0ab198fd48fa6030d57e4423df3f18e03aa62518c565f", size = 12272599, upload-time = "2025-10-07T18:21:38.08Z" }, - { url = "https://files.pythonhosted.org/packages/18/14/e2815d8eff847391af632b22422b8207704222ff575dec8d044f9ab779b2/ruff-0.14.0-py3-none-musllinux_1_2_i686.whl", hash = "sha256:668fce701b7a222f3f5327f86909db2bbe99c30877c8001ff934c5413812ac02", size = 13193828, upload-time = "2025-10-07T18:21:41.216Z" }, - { url = "https://files.pythonhosted.org/packages/44/c6/61ccc2987cf0aecc588ff8f3212dea64840770e60d78f5606cd7dc34de32/ruff-0.14.0-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:a86bf575e05cb68dcb34e4c7dfe1064d44d3f0c04bbc0491949092192b515296", size = 13628617, upload-time = "2025-10-07T18:21:44.04Z" }, - { url = "https://files.pythonhosted.org/packages/73/e6/03b882225a1b0627e75339b420883dc3c90707a8917d2284abef7a58d317/ruff-0.14.0-py3-none-win32.whl", hash = "sha256:7450a243d7125d1c032cb4b93d9625dea46c8c42b4f06c6b709baac168e10543", size = 12367872, upload-time = "2025-10-07T18:21:46.67Z" }, - { url = "https://files.pythonhosted.org/packages/41/77/56cf9cf01ea0bfcc662de72540812e5ba8e9563f33ef3d37ab2174892c47/ruff-0.14.0-py3-none-win_amd64.whl", hash = "sha256:ea95da28cd874c4d9c922b39381cbd69cb7e7b49c21b8152b014bd4f52acddc2", size = 13464628, upload-time = "2025-10-07T18:21:50.318Z" }, - { url = "https://files.pythonhosted.org/packages/c6/2a/65880dfd0e13f7f13a775998f34703674a4554906167dce02daf7865b954/ruff-0.14.0-py3-none-win_arm64.whl", hash = "sha256:f42c9495f5c13ff841b1da4cb3c2a42075409592825dada7c5885c2c844ac730", size = 12565142, upload-time = "2025-10-07T18:21:53.577Z" }, + { url = "https://files.pythonhosted.org/packages/17/b9/67240254166ae1eaa38dec32265e9153ac53645a6c6670ed36ad00722af8/ruff-0.14.4-py3-none-linux_armv6l.whl", hash = "sha256:e6604613ffbcf2297cd5dcba0e0ac9bd0c11dc026442dfbb614504e87c349518", size = 12606781, upload-time = "2025-11-06T22:07:01.841Z" }, + { url = "https://files.pythonhosted.org/packages/46/c8/09b3ab245d8652eafe5256ab59718641429f68681ee713ff06c5c549f156/ruff-0.14.4-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:d99c0b52b6f0598acede45ee78288e5e9b4409d1ce7f661f0fa36d4cbeadf9a4", size = 12946765, upload-time = "2025-11-06T22:07:05.858Z" }, + { url = "https://files.pythonhosted.org/packages/14/bb/1564b000219144bf5eed2359edc94c3590dd49d510751dad26202c18a17d/ruff-0.14.4-py3-none-macosx_11_0_arm64.whl", hash = "sha256:9358d490ec030f1b51d048a7fd6ead418ed0826daf6149e95e30aa67c168af33", size = 11928120, upload-time = "2025-11-06T22:07:08.023Z" }, + { url = "https://files.pythonhosted.org/packages/a3/92/d5f1770e9988cc0742fefaa351e840d9aef04ec24ae1be36f333f96d5704/ruff-0.14.4-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:81b40d27924f1f02dfa827b9c0712a13c0e4b108421665322218fc38caf615c2", size = 12370877, upload-time = "2025-11-06T22:07:10.015Z" }, + { url = "https://files.pythonhosted.org/packages/e2/29/e9282efa55f1973d109faf839a63235575519c8ad278cc87a182a366810e/ruff-0.14.4-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f5e649052a294fe00818650712083cddc6cc02744afaf37202c65df9ea52efa5", size = 12408538, upload-time = "2025-11-06T22:07:13.085Z" }, + { url = "https://files.pythonhosted.org/packages/8e/01/930ed6ecfce130144b32d77d8d69f5c610e6d23e6857927150adf5d7379a/ruff-0.14.4-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:aa082a8f878deeba955531f975881828fd6afd90dfa757c2b0808aadb437136e", size = 13141942, upload-time = "2025-11-06T22:07:15.386Z" }, + { url = "https://files.pythonhosted.org/packages/6a/46/a9c89b42b231a9f487233f17a89cbef9d5acd538d9488687a02ad288fa6b/ruff-0.14.4-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:1043c6811c2419e39011890f14d0a30470f19d47d197c4858b2787dfa698f6c8", size = 14544306, upload-time = "2025-11-06T22:07:17.631Z" }, + { url = "https://files.pythonhosted.org/packages/78/96/9c6cf86491f2a6d52758b830b89b78c2ae61e8ca66b86bf5a20af73d20e6/ruff-0.14.4-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a9f3a936ac27fb7c2a93e4f4b943a662775879ac579a433291a6f69428722649", size = 14210427, upload-time = "2025-11-06T22:07:19.832Z" }, + { url = "https://files.pythonhosted.org/packages/71/f4/0666fe7769a54f63e66404e8ff698de1dcde733e12e2fd1c9c6efb689cb5/ruff-0.14.4-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:95643ffd209ce78bc113266b88fba3d39e0461f0cbc8b55fb92505030fb4a850", size = 13658488, upload-time = "2025-11-06T22:07:22.32Z" }, + { url = "https://files.pythonhosted.org/packages/ee/79/6ad4dda2cfd55e41ac9ed6d73ef9ab9475b1eef69f3a85957210c74ba12c/ruff-0.14.4-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:456daa2fa1021bc86ca857f43fe29d5d8b3f0e55e9f90c58c317c1dcc2afc7b5", size = 13354908, upload-time = "2025-11-06T22:07:24.347Z" }, + { url = "https://files.pythonhosted.org/packages/b5/60/f0b6990f740bb15c1588601d19d21bcc1bd5de4330a07222041678a8e04f/ruff-0.14.4-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:f911bba769e4a9f51af6e70037bb72b70b45a16db5ce73e1f72aefe6f6d62132", size = 13587803, upload-time = "2025-11-06T22:07:26.327Z" }, + { url = "https://files.pythonhosted.org/packages/c9/da/eaaada586f80068728338e0ef7f29ab3e4a08a692f92eb901a4f06bbff24/ruff-0.14.4-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:76158a7369b3979fa878612c623a7e5430c18b2fd1c73b214945c2d06337db67", size = 12279654, upload-time = "2025-11-06T22:07:28.46Z" }, + { url = "https://files.pythonhosted.org/packages/66/d4/b1d0e82cf9bf8aed10a6d45be47b3f402730aa2c438164424783ac88c0ed/ruff-0.14.4-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:f3b8f3b442d2b14c246e7aeca2e75915159e06a3540e2f4bed9f50d062d24469", size = 12357520, upload-time = "2025-11-06T22:07:31.468Z" }, + { url = "https://files.pythonhosted.org/packages/04/f4/53e2b42cc82804617e5c7950b7079d79996c27e99c4652131c6a1100657f/ruff-0.14.4-py3-none-musllinux_1_2_i686.whl", hash = "sha256:c62da9a06779deecf4d17ed04939ae8b31b517643b26370c3be1d26f3ef7dbde", size = 12719431, upload-time = "2025-11-06T22:07:33.831Z" }, + { url = "https://files.pythonhosted.org/packages/a2/94/80e3d74ed9a72d64e94a7b7706b1c1ebaa315ef2076fd33581f6a1cd2f95/ruff-0.14.4-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:5a443a83a1506c684e98acb8cb55abaf3ef725078be40237463dae4463366349", size = 13464394, upload-time = "2025-11-06T22:07:35.905Z" }, + { url = "https://files.pythonhosted.org/packages/54/1a/a49f071f04c42345c793d22f6cf5e0920095e286119ee53a64a3a3004825/ruff-0.14.4-py3-none-win32.whl", hash = "sha256:643b69cb63cd996f1fc7229da726d07ac307eae442dd8974dbc7cf22c1e18fff", size = 12493429, upload-time = "2025-11-06T22:07:38.43Z" }, + { url = "https://files.pythonhosted.org/packages/bc/22/e58c43e641145a2b670328fb98bc384e20679b5774258b1e540207580266/ruff-0.14.4-py3-none-win_amd64.whl", hash = "sha256:26673da283b96fe35fa0c939bf8411abec47111644aa9f7cfbd3c573fb125d2c", size = 13635380, upload-time = "2025-11-06T22:07:40.496Z" }, + { url = "https://files.pythonhosted.org/packages/30/bd/4168a751ddbbf43e86544b4de8b5c3b7be8d7167a2a5cb977d274e04f0a1/ruff-0.14.4-py3-none-win_arm64.whl", hash = "sha256:dd09c292479596b0e6fec8cd95c65c3a6dc68e9ad17b8f2382130f87ff6a75bb", size = 12663065, upload-time = "2025-11-06T22:07:42.603Z" }, ] [[package]] @@ -5545,14 +5548,14 @@ wheels = [ [[package]] name = "scipy-stubs" -version = "1.16.2.3" +version = "1.16.3.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "optype", extra = ["numpy"] }, ] -sdist = { url = "https://files.pythonhosted.org/packages/6b/47/b165711b36a1afb8d5f408393487586e07f8bdb86f829b5b904c1ddd091f/scipy_stubs-1.16.2.3.tar.gz", hash = "sha256:b1afd21442699b8bdd399508187bddcedc6c29a34b188fd603396cb6754c2a91", size = 355436, upload-time = "2025-10-08T01:38:37.403Z" } +sdist = { url = "https://files.pythonhosted.org/packages/bd/68/c53c3bce6bd069a164015be1be2671c968b526be4af1e85db64c88f04546/scipy_stubs-1.16.3.0.tar.gz", hash = "sha256:d6943c085e47a1ed431309f9ca582b6a206a9db808a036132a0bf01ebc34b506", size = 356462, upload-time = "2025-10-28T22:05:31.198Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/29/9f/3d8f613d0c3be9348cb0c351328249b7a2428f13329447ec6f395628d7b0/scipy_stubs-1.16.2.3-py3-none-any.whl", hash = "sha256:05e93238bdaedb7fa1afedf9c3a2337f94fec3d8c33fb2d403c933e1bcc7412e", size = 556848, upload-time = "2025-10-08T01:38:35.697Z" }, + { url = "https://files.pythonhosted.org/packages/86/1c/0ba7305fa01cfe7a6f1b8c86ccdd1b7a0d43fa9bd769c059995311e291a2/scipy_stubs-1.16.3.0-py3-none-any.whl", hash = "sha256:90e5d82ced2183ef3c5c0a28a77df8cc227458624364fa0ff975ad24fa89d6ad", size = 557713, upload-time = "2025-10-28T22:05:29.454Z" }, ] [[package]] @@ -5719,11 +5722,11 @@ wheels = [ [[package]] name = "sqlglot" -version = "26.33.0" +version = "27.29.0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/25/9d/fcd59b4612d5ad1e2257c67c478107f073b19e1097d3bfde2fb517884416/sqlglot-26.33.0.tar.gz", hash = "sha256:2817278779fa51d6def43aa0d70690b93a25c83eb18ec97130fdaf707abc0d73", size = 5353340, upload-time = "2025-07-01T13:09:06.311Z" } +sdist = { url = "https://files.pythonhosted.org/packages/d1/50/766692a83468adb1bde9e09ea524a01719912f6bc4fdb47ec18368320f6e/sqlglot-27.29.0.tar.gz", hash = "sha256:2270899694663acef94fa93497971837e6fadd712f4a98b32aee1e980bc82722", size = 5503507, upload-time = "2025-10-29T13:50:24.594Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/31/8d/f1d9cb5b18e06aa45689fbeaaea6ebab66d5f01d1e65029a8f7657c06be5/sqlglot-26.33.0-py3-none-any.whl", hash = "sha256:031cee20c0c796a83d26d079a47fdce667604df430598c7eabfa4e4dfd147033", size = 477610, upload-time = "2025-07-01T13:09:03.926Z" }, + { url = "https://files.pythonhosted.org/packages/9b/70/20c1912bc0bfebf516d59d618209443b136c58a7cff141afa7cf30969988/sqlglot-27.29.0-py3-none-any.whl", hash = "sha256:9a5ea8ac61826a7763de10cad45a35f0aa9bfcf7b96ee74afb2314de9089e1cb", size = 526060, upload-time = "2025-10-29T13:50:22.061Z" }, ] [[package]] @@ -5737,15 +5740,15 @@ wheels = [ [[package]] name = "starlette" -version = "0.47.2" +version = "0.49.1" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "anyio" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/04/57/d062573f391d062710d4088fa1369428c38d51460ab6fedff920efef932e/starlette-0.47.2.tar.gz", hash = "sha256:6ae9aa5db235e4846decc1e7b79c4f346adf41e9777aebeb49dfd09bbd7023d8", size = 2583948, upload-time = "2025-07-20T17:31:58.522Z" } +sdist = { url = "https://files.pythonhosted.org/packages/1b/3f/507c21db33b66fb027a332f2cb3abbbe924cc3a79ced12f01ed8645955c9/starlette-0.49.1.tar.gz", hash = "sha256:481a43b71e24ed8c43b11ea02f5353d77840e01480881b8cb5a26b8cae64a8cb", size = 2654703, upload-time = "2025-10-28T17:34:10.928Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/f7/1f/b876b1f83aef204198a42dc101613fefccb32258e5428b5f9259677864b4/starlette-0.47.2-py3-none-any.whl", hash = "sha256:c5847e96134e5c5371ee9fac6fdf1a67336d5815e09eb2a01fdb57a351ef915b", size = 72984, upload-time = "2025-07-20T17:31:56.738Z" }, + { url = "https://files.pythonhosted.org/packages/51/da/545b75d420bb23b5d494b0517757b351963e974e79933f01e05c929f20a6/starlette-0.49.1-py3-none-any.whl", hash = "sha256:d92ce9f07e4a3caa3ac13a79523bd18e3bc0042bb8ff2d759a8e7dd0e1859875", size = 74175, upload-time = "2025-10-28T17:34:09.13Z" }, ] [[package]] @@ -5838,9 +5841,10 @@ wheels = [ [[package]] name = "tablestore" -version = "6.2.0" +version = "6.3.7" source = { registry = "https://pypi.org/simple" } dependencies = [ + { name = "aiohttp" }, { name = "certifi" }, { name = "crc32c" }, { name = "flatbuffers" }, @@ -5850,9 +5854,9 @@ dependencies = [ { name = "six" }, { name = "urllib3" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/a1/58/48d65d181a69f7db19f7cdee01d252168fbfbad2d1bb25abed03e6df3b05/tablestore-6.2.0.tar.gz", hash = "sha256:0773e77c00542be1bfebbc3c7a85f72a881c63e4e7df7c5a9793a54144590e68", size = 85942, upload-time = "2025-04-15T12:11:20.655Z" } +sdist = { url = "https://files.pythonhosted.org/packages/f1/39/47a3ec8e42fe74dd05af1dfed9c3b02b8f8adfdd8656b2c5d4f95f975c9f/tablestore-6.3.7.tar.gz", hash = "sha256:990682dbf6b602f317a2d359b4281dcd054b4326081e7a67b73dbbe95407be51", size = 117440, upload-time = "2025-10-29T02:57:57.415Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/9c/da/30451712a769bcf417add8e81163d478a4d668b0e8d489a9d667260d55df/tablestore-6.2.0-py3-none-any.whl", hash = "sha256:6af496d841ab1ff3f78b46abbd87b95a08d89605c51664d2b30933b1d1c5583a", size = 106297, upload-time = "2025-04-15T12:11:17.476Z" }, + { url = "https://files.pythonhosted.org/packages/fe/55/1b24d8c369204a855ac652712f815e88a4909802094e613fe3742a2d80e3/tablestore-6.3.7-py3-none-any.whl", hash = "sha256:38dcc55085912ab2515e183afd4532a58bb628a763590a99fc1bd2a4aba6855c", size = 139041, upload-time = "2025-10-29T02:57:55.727Z" }, ] [[package]] @@ -5907,7 +5911,7 @@ wheels = [ [[package]] name = "testcontainers" -version = "4.10.0" +version = "4.13.2" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "docker" }, @@ -5916,9 +5920,9 @@ dependencies = [ { name = "urllib3" }, { name = "wrapt" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/a1/49/9c618aff1c50121d183cdfbc3a4a5cf2727a2cde1893efe6ca55c7009196/testcontainers-4.10.0.tar.gz", hash = "sha256:03f85c3e505d8b4edeb192c72a961cebbcba0dd94344ae778b4a159cb6dcf8d3", size = 63327, upload-time = "2025-04-02T16:13:27.582Z" } +sdist = { url = "https://files.pythonhosted.org/packages/18/51/edac83edab339d8b4dce9a7b659163afb1ea7e011bfed1d5573d495a4485/testcontainers-4.13.2.tar.gz", hash = "sha256:2315f1e21b059427a9d11e8921f85fef322fbe0d50749bcca4eaa11271708ba4", size = 78692, upload-time = "2025-10-07T21:53:07.531Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/1c/0a/824b0c1ecf224802125279c3effff2e25ed785ed046e67da6e53d928de4c/testcontainers-4.10.0-py3-none-any.whl", hash = "sha256:31ed1a81238c7e131a2a29df6db8f23717d892b592fa5a1977fd0dcd0c23fc23", size = 107414, upload-time = "2025-04-02T16:13:25.785Z" }, + { url = "https://files.pythonhosted.org/packages/2a/5e/73aa94770f1df0595364aed526f31d54440db5492911e2857318ed326e51/testcontainers-4.13.2-py3-none-any.whl", hash = "sha256:0209baf8f4274b568cde95bef2cadf7b1d33b375321f793790462e235cd684ee", size = 124771, upload-time = "2025-10-07T21:53:05.937Z" }, ] [[package]] @@ -6064,32 +6068,32 @@ wheels = [ [[package]] name = "ty" -version = "0.0.1a22" +version = "0.0.1a26" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/bb/87/eab73cdc990d1141b60237379975efc0e913bfa0d19083daab0f497444a6/ty-0.0.1a22.tar.gz", hash = "sha256:b20ec5362830a1e9e05654c15e88607fdbb45325ec130a9a364c6dd412ecbf55", size = 4312182, upload-time = "2025-10-10T13:07:15.88Z" } +sdist = { url = "https://files.pythonhosted.org/packages/39/39/b4b4ecb6ca6d7e937fa56f0b92a8f48d7719af8fe55bdbf667638e9f93e2/ty-0.0.1a26.tar.gz", hash = "sha256:65143f8efeb2da1644821b710bf6b702a31ddcf60a639d5a576db08bded91db4", size = 4432154, upload-time = "2025-11-10T18:02:30.142Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/4d/30/83e2dbfbc70de8a1932b19daf05ce803d7d76cdc6251de1519a49cf1c27d/ty-0.0.1a22-py3-none-linux_armv6l.whl", hash = "sha256:6efba0c777881d2d072fa7375a64ad20357e825eff2a0b6ff9ec80399a04253b", size = 8581795, upload-time = "2025-10-10T13:06:44.396Z" }, - { url = "https://files.pythonhosted.org/packages/d7/8c/5193534fc4a3569f517408828d077b26d6280fe8c2dd0bdc63db4403dcdb/ty-0.0.1a22-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:2ada020eebe1b44403affdf45cd5c8d3fb8312c3e80469d795690093c0921f55", size = 8682602, upload-time = "2025-10-10T13:06:46.44Z" }, - { url = "https://files.pythonhosted.org/packages/22/4a/7ba53493bf37b61d3e0dfe6df910e6bc74c40d16c3effd84e15c0863d34e/ty-0.0.1a22-py3-none-macosx_11_0_arm64.whl", hash = "sha256:ed4f11f1a5824ea10d3e46b1990d092c3f341b1d492c357d23bed2ac347fd253", size = 8278839, upload-time = "2025-10-10T13:06:48.688Z" }, - { url = "https://files.pythonhosted.org/packages/52/0a/d9862c41b9615de56d2158bfbb5177dbf5a65e94922d3dd13855f48cb91b/ty-0.0.1a22-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:56f48d8f94292909d596dbeb56ff7f9f070bd316aa628b45c02ca2b2f5797f31", size = 8421483, upload-time = "2025-10-10T13:06:50.75Z" }, - { url = "https://files.pythonhosted.org/packages/a5/cb/3ebe0e45b80724d4c2f849fdf304179727fd06df7fee7cd12fe6c3efe49d/ty-0.0.1a22-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:733e9ac22885b6574de26bdbae439c960a06acc825a938d3780c9d498bb65339", size = 8419225, upload-time = "2025-10-10T13:06:52.533Z" }, - { url = "https://files.pythonhosted.org/packages/4f/b5/da65f3f8ad31d881ca9987a3f6f26069a0cc649c9354adb7453ca62116bb/ty-0.0.1a22-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5135d662484e56809c77b3343614005585caadaa5c1cf643ed6a09303497652b", size = 9352336, upload-time = "2025-10-10T13:06:54.476Z" }, - { url = "https://files.pythonhosted.org/packages/a3/24/9c46f2eb16734ab0fcf3291486b1c5c528a1569f94541dc1f19f97dd2a5b/ty-0.0.1a22-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:87f297f99a98154d33a3f21991979418c65d8bf480f6a1bad1e54d46d2dc7df7", size = 9857840, upload-time = "2025-10-10T13:06:56.514Z" }, - { url = "https://files.pythonhosted.org/packages/d8/ae/930c94bbbe5c049eae5355a197c39522844f55c7ab7fccd0ba061f618541/ty-0.0.1a22-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3310217eaa4dccf20b7336fcbeb072097addc6fde0c9d3f791dea437af0aa6dc", size = 9452611, upload-time = "2025-10-10T13:06:58.154Z" }, - { url = "https://files.pythonhosted.org/packages/a2/80/d8f594438465c352cf0ebd4072f5ca3be2871153a3cd279ed2f35ecd487c/ty-0.0.1a22-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:12b032e81012bf5228fd65f01b50e29eb409534b6aac28ee5c48ee3b7b860ddf", size = 9214875, upload-time = "2025-10-10T13:06:59.861Z" }, - { url = "https://files.pythonhosted.org/packages/fd/07/f852fb20ac27707de495c39a02aeb056e3368833b7e12888d43b1f61594d/ty-0.0.1a22-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a3ffda8149cab0000a21e7a078142073e27a1a9ac03b9a0837aa2f53d1fbebcb", size = 8906715, upload-time = "2025-10-10T13:07:01.926Z" }, - { url = "https://files.pythonhosted.org/packages/40/4d/0e0b85b4179891cc3067a6e717f5161921c07873a4f545963fdf1dd3619c/ty-0.0.1a22-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:afa512e7dc78f0cf0b55f87394968ba59c46993c67bc0ef295962144fea85b12", size = 8350873, upload-time = "2025-10-10T13:07:03.999Z" }, - { url = "https://files.pythonhosted.org/packages/a1/1f/e70c63e12b4a0d97d4fd6f872dd199113666ad1b236e18838fa5e5d5502d/ty-0.0.1a22-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:069cdbbea6025f7ebbb5e9043c8d0daf760358df46df8304ef5ca5bb3e320aef", size = 8442568, upload-time = "2025-10-10T13:07:05.745Z" }, - { url = "https://files.pythonhosted.org/packages/de/3b/55518906cb3598f2b99ff1e86c838d77d006cab70cdd2a0a625d02ccb52c/ty-0.0.1a22-py3-none-musllinux_1_2_i686.whl", hash = "sha256:67d31d902e6fd67a4b3523604f635e71d2ec55acfb9118f984600584bfe0ff2a", size = 8896775, upload-time = "2025-10-10T13:07:08.02Z" }, - { url = "https://files.pythonhosted.org/packages/c3/ea/60c654c27931bf84fa9cb463a4c4c49e8869c052fa607a6e930be717b619/ty-0.0.1a22-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:f9e154f262162e6f76b01f318e469ac6c22ffce22b010c396ed34e81d8369821", size = 9054544, upload-time = "2025-10-10T13:07:09.675Z" }, - { url = "https://files.pythonhosted.org/packages/6c/60/9a6d5530d6829ccf656e6ae0fb13d70a4e2514f4fb8910266ebd54286620/ty-0.0.1a22-py3-none-win32.whl", hash = "sha256:37525433ca7b02a8fca4b8fa9dcde818bf3a413b539b9dbc8f7b39d124eb7c49", size = 8165703, upload-time = "2025-10-10T13:07:11.378Z" }, - { url = "https://files.pythonhosted.org/packages/14/9c/ac08c832643850d4e18cbc959abc69cd51d531fe11bdb691098b3cf2f562/ty-0.0.1a22-py3-none-win_amd64.whl", hash = "sha256:75d21cdeba8bcef247af89518d7ce98079cac4a55c4160cb76682ea40a18b92c", size = 8828319, upload-time = "2025-10-10T13:07:12.815Z" }, - { url = "https://files.pythonhosted.org/packages/22/df/38068fc44e3cfb455aeb41d0ff1850a4d3c9988010466d4a8d19860b8b9a/ty-0.0.1a22-py3-none-win_arm64.whl", hash = "sha256:1c7f040fe311e9696917417434c2a0e58402235be842c508002c6a2eff1398b0", size = 8367136, upload-time = "2025-10-10T13:07:14.518Z" }, + { url = "https://files.pythonhosted.org/packages/cc/6a/661833ecacc4d994f7e30a7f1307bfd3a4a91392a6b03fb6a018723e75b8/ty-0.0.1a26-py3-none-linux_armv6l.whl", hash = "sha256:09208dca99bb548e9200136d4d42618476bfe1f4d2066511f2c8e2e4dfeced5e", size = 9173869, upload-time = "2025-11-10T18:01:46.012Z" }, + { url = "https://files.pythonhosted.org/packages/66/a8/32ea50f064342de391a7267f84349287e2f1c2eb0ad4811d6110916179d6/ty-0.0.1a26-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:91d12b66c91a1b82e698a2aa73fe043a1a9da83ff0dfd60b970500bee0963b91", size = 8973420, upload-time = "2025-11-10T18:01:49.32Z" }, + { url = "https://files.pythonhosted.org/packages/d1/f6/6659d55940cd5158a6740ae46a65be84a7ee9167738033a9b1259c36eef5/ty-0.0.1a26-py3-none-macosx_11_0_arm64.whl", hash = "sha256:c5bc6dfcea5477c81ad01d6a29ebc9bfcbdb21c34664f79c9e1b84be7aa8f289", size = 8528888, upload-time = "2025-11-10T18:01:51.511Z" }, + { url = "https://files.pythonhosted.org/packages/79/c9/4cbe7295013cc412b4f100b509aaa21982c08c59764a2efa537ead049345/ty-0.0.1a26-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:40e5d15635e9918924138e8d3fb1cbf80822dfb8dc36ea8f3e72df598c0c4bea", size = 8801867, upload-time = "2025-11-10T18:01:53.888Z" }, + { url = "https://files.pythonhosted.org/packages/ed/b3/25099b219a6444c4b29f175784a275510c1cd85a23a926d687ab56915027/ty-0.0.1a26-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:86dc147ed0790c7c8fd3f0d6c16c3c5135b01e99c440e89c6ca1e0e592bb6682", size = 8975519, upload-time = "2025-11-10T18:01:56.231Z" }, + { url = "https://files.pythonhosted.org/packages/73/3e/3ad570f4f592cb1d11982dd2c426c90d2aa9f3d38bf77a7e2ce8aa614302/ty-0.0.1a26-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:fbe0e07c9d5e624edfc79a468f2ef191f9435581546a5bb6b92713ddc86ad4a6", size = 9331932, upload-time = "2025-11-10T18:01:58.476Z" }, + { url = "https://files.pythonhosted.org/packages/04/fa/62c72eead0302787f9cc0d613fc671107afeecdaf76ebb04db8f91bb9f7e/ty-0.0.1a26-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:0dcebbfe9f24b43d98a078f4a41321ae7b08bea40f5c27d81394b3f54e9f7fb5", size = 9921353, upload-time = "2025-11-10T18:02:00.749Z" }, + { url = "https://files.pythonhosted.org/packages/6c/1f/3b329c4b60d878704e09eb9d05467f911f188e699961c044b75932893e0a/ty-0.0.1a26-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0901b75afc7738224ffc98bbc8ea03a20f167a2a83a4b23a6550115e8b3ddbc6", size = 9700800, upload-time = "2025-11-10T18:02:03.544Z" }, + { url = "https://files.pythonhosted.org/packages/92/24/13fcba20dd86a7c3f83c814279aa3eb6a29c5f1b38a3b3a4a0fd22159189/ty-0.0.1a26-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4788f34d384c132977958d76fef7f274f8d181b22e33933c4d16cff2bb5ca3b9", size = 9728289, upload-time = "2025-11-10T18:02:06.386Z" }, + { url = "https://files.pythonhosted.org/packages/40/7a/798894ff0b948425570b969be35e672693beeb6b852815b7340bc8de1575/ty-0.0.1a26-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b98851c11c560ce63cd972ed9728aa079d9cf40483f2cdcf3626a55849bfe107", size = 9279735, upload-time = "2025-11-10T18:02:09.425Z" }, + { url = "https://files.pythonhosted.org/packages/1a/54/71261cc1b8dc7d3c4ad92a83b4d1681f5cb7ea5965ebcbc53311ae8c6424/ty-0.0.1a26-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:c20b4625a20059adecd86fe2c4df87cd6115fea28caee45d3bdcf8fb83d29510", size = 8767428, upload-time = "2025-11-10T18:02:11.956Z" }, + { url = "https://files.pythonhosted.org/packages/8e/07/b248b73a640badba2b301e6845699b7dd241f40a321b9b1bce684d440f70/ty-0.0.1a26-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:d9909e96276f8d16382d285db92ae902174cae842aa953003ec0c06642db2f8a", size = 9009170, upload-time = "2025-11-10T18:02:14.878Z" }, + { url = "https://files.pythonhosted.org/packages/f8/35/ec8353f2bb7fd2f41bca6070b29ecb58e2de9af043e649678b8c132d5439/ty-0.0.1a26-py3-none-musllinux_1_2_i686.whl", hash = "sha256:a76d649ceefe9baa9bbae97d217bee076fd8eeb2a961f66f1dff73cc70af4ac8", size = 9119215, upload-time = "2025-11-10T18:02:18.329Z" }, + { url = "https://files.pythonhosted.org/packages/70/48/db49fe1b7e66edf90dc285869043f99c12aacf7a99c36ee760e297bac6d5/ty-0.0.1a26-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:a0ee0f6366bcf70fae114e714d45335cacc8daa936037441e02998a9110b7a29", size = 9398655, upload-time = "2025-11-10T18:02:21.031Z" }, + { url = "https://files.pythonhosted.org/packages/10/f8/d869492bdbb21ae8cf4c99b02f20812bbbf49aa187cfeb387dfaa03036a8/ty-0.0.1a26-py3-none-win32.whl", hash = "sha256:86689b90024810cac7750bf0c6e1652e4b4175a9de7b82b8b1583202aeb47287", size = 8645669, upload-time = "2025-11-10T18:02:23.23Z" }, + { url = "https://files.pythonhosted.org/packages/b4/18/8a907575d2b335afee7556cb92233ebb5efcefe17752fc9dcab21cffb23b/ty-0.0.1a26-py3-none-win_amd64.whl", hash = "sha256:829e6e6dbd7d9d370f97b2398b4804552554bdcc2d298114fed5e2ea06cbc05c", size = 9442975, upload-time = "2025-11-10T18:02:25.68Z" }, + { url = "https://files.pythonhosted.org/packages/e9/22/af92dcfdd84b78dd97ac6b7154d6a763781f04a400140444885c297cc213/ty-0.0.1a26-py3-none-win_arm64.whl", hash = "sha256:b8f431c784d4cf5b4195a3521b2eca9c15902f239b91154cb920da33f943c62b", size = 8958958, upload-time = "2025-11-10T18:02:28.071Z" }, ] [[package]] name = "typer" -version = "0.19.2" +version = "0.20.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "click" }, @@ -6097,9 +6101,9 @@ dependencies = [ { name = "shellingham" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/21/ca/950278884e2ca20547ff3eb109478c6baf6b8cf219318e6bc4f666fad8e8/typer-0.19.2.tar.gz", hash = "sha256:9ad824308ded0ad06cc716434705f691d4ee0bfd0fb081839d2e426860e7fdca", size = 104755, upload-time = "2025-09-23T09:47:48.256Z" } +sdist = { url = "https://files.pythonhosted.org/packages/8f/28/7c85c8032b91dbe79725b6f17d2fffc595dff06a35c7a30a37bef73a1ab4/typer-0.20.0.tar.gz", hash = "sha256:1aaf6494031793e4876fb0bacfa6a912b551cf43c1e63c800df8b1a866720c37", size = 106492, upload-time = "2025-10-20T17:03:49.445Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/00/22/35617eee79080a5d071d0f14ad698d325ee6b3bf824fc0467c03b30e7fa8/typer-0.19.2-py3-none-any.whl", hash = "sha256:755e7e19670ffad8283db353267cb81ef252f595aa6834a0d1ca9312d9326cb9", size = 46748, upload-time = "2025-09-23T09:47:46.777Z" }, + { url = "https://files.pythonhosted.org/packages/78/64/7713ffe4b5983314e9d436a90d5bd4f63b6054e2aca783a3cfc44cb95bbf/typer-0.20.0-py3-none-any.whl", hash = "sha256:5b463df6793ec1dca6213a3cf4c0f03bc6e322ac5e16e13ddd622a889489784a", size = 47028, upload-time = "2025-10-20T17:03:47.617Z" }, ] [[package]] @@ -6113,11 +6117,11 @@ wheels = [ [[package]] name = "types-awscrt" -version = "0.28.1" +version = "0.28.4" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/60/19/a3a6377c9e2e389c1421c033a1830c29cac08f2e1e05a082ea84eb22c75f/types_awscrt-0.28.1.tar.gz", hash = "sha256:66d77ec283e1dc907526a44511a12624118723a396c36d3f3dd9855cb614ce14", size = 17410, upload-time = "2025-10-11T21:55:07.443Z" } +sdist = { url = "https://files.pythonhosted.org/packages/f7/6f/d4f2adb086e8f5cd2ae83cf8dbb192057d8b5025120e5b372468292db67f/types_awscrt-0.28.4.tar.gz", hash = "sha256:15929da84802f27019ee8e4484fb1c102e1f6d4cf22eb48688c34a5a86d02eb6", size = 17692, upload-time = "2025-11-11T02:56:53.516Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/ea/c7/0266b797d19b82aebe0e177efe35de7aabdc192bc1605ce3309331f0a505/types_awscrt-0.28.1-py3-none-any.whl", hash = "sha256:d88f43ef779f90b841ba99badb72fe153077225a4e426ae79e943184827b4443", size = 41851, upload-time = "2025-10-11T21:55:06.235Z" }, + { url = "https://files.pythonhosted.org/packages/5e/ae/9acc4adf1d5d7bb7d09b6f9ff5d4d04a72eb64700d104106dd517665cd57/types_awscrt-0.28.4-py3-none-any.whl", hash = "sha256:2d453f9e27583fcc333771b69a5255a5a4e2c52f86e70f65f3c5a6789d3443d0", size = 42307, upload-time = "2025-11-11T02:56:52.231Z" }, ] [[package]] @@ -6238,11 +6242,11 @@ wheels = [ [[package]] name = "types-html5lib" -version = "1.1.11.20250917" +version = "1.1.11.20251014" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/68/4b/a970718e8bd9324ee8fb8eaf02ff069f6d03c20d4523bb4232892ecc3d06/types_html5lib-1.1.11.20250917.tar.gz", hash = "sha256:7b52743377f33f9b4fd7385afbd2d457b8864ee51f90ff2a795ad9e8c053373a", size = 16868, upload-time = "2025-09-17T02:47:41.18Z" } +sdist = { url = "https://files.pythonhosted.org/packages/8a/b8/0ce98d9b20a4e8bdac4f4914054acadf5b3a36a7a832e11e0d1938e4c3ce/types_html5lib-1.1.11.20251014.tar.gz", hash = "sha256:cc628d626e0111a2426a64f5f061ecfd113958b69ff6b3dc0eaaed2347ba9455", size = 16895, upload-time = "2025-10-14T02:54:50.003Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/78/8a/da91a9c64dcb5e69beb567519857411996d8ecae9f6f128bcef8260e7a8d/types_html5lib-1.1.11.20250917-py3-none-any.whl", hash = "sha256:b294fd06d60da205daeb2f615485ca4d475088d2eff1009cf427f4a80fcd5346", size = 22908, upload-time = "2025-09-17T02:47:40.39Z" }, + { url = "https://files.pythonhosted.org/packages/c9/cb/df12640506b8dbd2f2bd0643c5ef4a72fa6285ec4cd7f4b20457842e7fd5/types_html5lib-1.1.11.20251014-py3-none-any.whl", hash = "sha256:4ff2cf18dfc547009ab6fa4190fc3de464ba815c9090c3dd4a5b65f664bfa76c", size = 22916, upload-time = "2025-10-14T02:54:48.686Z" }, ] [[package]] @@ -6331,11 +6335,11 @@ wheels = [ [[package]] name = "types-psutil" -version = "7.0.0.20251001" +version = "7.0.0.20251111" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/9e/91/b020f9100b196a1f247cd12575f68dcdad94f032c1e0c42987d7632142ce/types_psutil-7.0.0.20251001.tar.gz", hash = "sha256:60d696200ddae28677e7d88cdebd6e960294e85adefbaafe0f6e5d0e7b4c1963", size = 20469, upload-time = "2025-10-01T03:04:21.292Z" } +sdist = { url = "https://files.pythonhosted.org/packages/1a/ba/4f48c927f38c7a4d6f7ff65cde91c49d28a95a56e00ec19b2813e1e0b1c1/types_psutil-7.0.0.20251111.tar.gz", hash = "sha256:d109ee2da4c0a9b69b8cefc46e195db8cf0fc0200b6641480df71e7f3f51a239", size = 20287, upload-time = "2025-11-11T03:06:37.482Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/c0/99/50f30e0b648e6f583165cb2e535b0256a02a03efa4868cb2f017ad25b3d8/types_psutil-7.0.0.20251001-py3-none-any.whl", hash = "sha256:adc31de8386d31c61bd4123112fd51e2c700c7502a001cad72a3d56ba6b463d1", size = 23164, upload-time = "2025-10-01T03:04:20.089Z" }, + { url = "https://files.pythonhosted.org/packages/fb/bc/b081d10fbd933cdf839109707a693c668a174e2276d64159a582a9cebd3f/types_psutil-7.0.0.20251111-py3-none-any.whl", hash = "sha256:85ba00205dcfa3c73685122e5a360205d2fbc9b56f942b591027bf401ce0cc47", size = 23052, upload-time = "2025-11-11T03:06:36.011Z" }, ] [[package]] @@ -6383,11 +6387,11 @@ wheels = [ [[package]] name = "types-python-dateutil" -version = "2.9.0.20251008" +version = "2.9.0.20251108" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/fc/83/24ed25dd0c6277a1a170c180ad9eef5879ecc9a4745b58d7905a4588c80d/types_python_dateutil-2.9.0.20251008.tar.gz", hash = "sha256:c3826289c170c93ebd8360c3485311187df740166dbab9dd3b792e69f2bc1f9c", size = 16128, upload-time = "2025-10-08T02:51:34.93Z" } +sdist = { url = "https://files.pythonhosted.org/packages/b0/42/18dff855130c3551d2b5159165bd24466f374dcb78670e5259d2ed51f55c/types_python_dateutil-2.9.0.20251108.tar.gz", hash = "sha256:d8a6687e197f2fa71779ce36176c666841f811368710ab8d274b876424ebfcaa", size = 16220, upload-time = "2025-11-08T02:55:53.393Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/da/af/5d24b8d49ef358468ecfdff5c556adf37f4fd28e336b96f923661a808329/types_python_dateutil-2.9.0.20251008-py3-none-any.whl", hash = "sha256:b9a5232c8921cf7661b29c163ccc56055c418ab2c6eabe8f917cbcc73a4c4157", size = 17934, upload-time = "2025-10-08T02:51:33.55Z" }, + { url = "https://files.pythonhosted.org/packages/25/dd/9fb1f5ef742cab1ea390582f407c967677704d2f5362b48c09de0d0dc8d4/types_python_dateutil-2.9.0.20251108-py3-none-any.whl", hash = "sha256:a4a537f0ea7126f8ccc2763eec9aa31ac8609e3c8e530eb2ddc5ee234b3cd764", size = 18127, upload-time = "2025-11-08T02:55:52.291Z" }, ] [[package]] @@ -6401,11 +6405,11 @@ wheels = [ [[package]] name = "types-pytz" -version = "2025.2.0.20250809" +version = "2025.2.0.20251108" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/07/e2/c774f754de26848f53f05defff5bb21dd9375a059d1ba5b5ea943cf8206e/types_pytz-2025.2.0.20250809.tar.gz", hash = "sha256:222e32e6a29bb28871f8834e8785e3801f2dc4441c715cd2082b271eecbe21e5", size = 10876, upload-time = "2025-08-09T03:14:17.453Z" } +sdist = { url = "https://files.pythonhosted.org/packages/40/ff/c047ddc68c803b46470a357454ef76f4acd8c1088f5cc4891cdd909bfcf6/types_pytz-2025.2.0.20251108.tar.gz", hash = "sha256:fca87917836ae843f07129567b74c1929f1870610681b4c92cb86a3df5817bdb", size = 10961, upload-time = "2025-11-08T02:55:57.001Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/db/d0/91c24fe54e565f2344d7a6821e6c6bb099841ef09007ea6321a0bac0f808/types_pytz-2025.2.0.20250809-py3-none-any.whl", hash = "sha256:4f55ed1b43e925cf851a756fe1707e0f5deeb1976e15bf844bcaa025e8fbd0db", size = 10095, upload-time = "2025-08-09T03:14:16.674Z" }, + { url = "https://files.pythonhosted.org/packages/e7/c1/56ef16bf5dcd255155cc736d276efa6ae0a5c26fd685e28f0412a4013c01/types_pytz-2025.2.0.20251108-py3-none-any.whl", hash = "sha256:0f1c9792cab4eb0e46c52f8845c8f77cf1e313cb3d68bf826aa867fe4717d91c", size = 10116, upload-time = "2025-11-08T02:55:56.194Z" }, ] [[package]] @@ -6734,15 +6738,15 @@ wheels = [ [[package]] name = "uvicorn" -version = "0.37.0" +version = "0.38.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "click" }, { name = "h11" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/71/57/1616c8274c3442d802621abf5deb230771c7a0fec9414cb6763900eb3868/uvicorn-0.37.0.tar.gz", hash = "sha256:4115c8add6d3fd536c8ee77f0e14a7fd2ebba939fed9b02583a97f80648f9e13", size = 80367, upload-time = "2025-09-23T13:33:47.486Z" } +sdist = { url = "https://files.pythonhosted.org/packages/cb/ce/f06b84e2697fef4688ca63bdb2fdf113ca0a3be33f94488f2cadb690b0cf/uvicorn-0.38.0.tar.gz", hash = "sha256:fd97093bdd120a2609fc0d3afe931d4d4ad688b6e75f0f929fde1bc36fe0e91d", size = 80605, upload-time = "2025-10-18T13:46:44.63Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/85/cd/584a2ceb5532af99dd09e50919e3615ba99aa127e9850eafe5f31ddfdb9a/uvicorn-0.37.0-py3-none-any.whl", hash = "sha256:913b2b88672343739927ce381ff9e2ad62541f9f8289664fa1d1d3803fa2ce6c", size = 67976, upload-time = "2025-09-23T13:33:45.842Z" }, + { url = "https://files.pythonhosted.org/packages/ee/d9/d88e73ca598f4f6ff671fb5fde8a32925c2e08a637303a1d12883c7305fa/uvicorn-0.38.0-py3-none-any.whl", hash = "sha256:48c0afd214ceb59340075b4a052ea1ee91c16fbc2a9b1469cca0e54566977b02", size = 68109, upload-time = "2025-10-18T13:46:42.958Z" }, ] [package.optional-dependencies] @@ -6758,22 +6762,22 @@ standard = [ [[package]] name = "uvloop" -version = "0.21.0" +version = "0.22.1" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/af/c0/854216d09d33c543f12a44b393c402e89a920b1a0a7dc634c42de91b9cf6/uvloop-0.21.0.tar.gz", hash = "sha256:3bf12b0fda68447806a7ad847bfa591613177275d35b6724b1ee573faa3704e3", size = 2492741, upload-time = "2024-10-14T23:38:35.489Z" } +sdist = { url = "https://files.pythonhosted.org/packages/06/f0/18d39dbd1971d6d62c4629cc7fa67f74821b0dc1f5a77af43719de7936a7/uvloop-0.22.1.tar.gz", hash = "sha256:6c84bae345b9147082b17371e3dd5d42775bddce91f885499017f4607fdaf39f", size = 2443250, upload-time = "2025-10-16T22:17:19.342Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/57/a7/4cf0334105c1160dd6819f3297f8700fda7fc30ab4f61fbf3e725acbc7cc/uvloop-0.21.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:c0f3fa6200b3108919f8bdabb9a7f87f20e7097ea3c543754cabc7d717d95cf8", size = 1447410, upload-time = "2024-10-14T23:37:33.612Z" }, - { url = "https://files.pythonhosted.org/packages/8c/7c/1517b0bbc2dbe784b563d6ab54f2ef88c890fdad77232c98ed490aa07132/uvloop-0.21.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:0878c2640cf341b269b7e128b1a5fed890adc4455513ca710d77d5e93aa6d6a0", size = 805476, upload-time = "2024-10-14T23:37:36.11Z" }, - { url = "https://files.pythonhosted.org/packages/ee/ea/0bfae1aceb82a503f358d8d2fa126ca9dbdb2ba9c7866974faec1cb5875c/uvloop-0.21.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b9fb766bb57b7388745d8bcc53a359b116b8a04c83a2288069809d2b3466c37e", size = 3960855, upload-time = "2024-10-14T23:37:37.683Z" }, - { url = "https://files.pythonhosted.org/packages/8a/ca/0864176a649838b838f36d44bf31c451597ab363b60dc9e09c9630619d41/uvloop-0.21.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8a375441696e2eda1c43c44ccb66e04d61ceeffcd76e4929e527b7fa401b90fb", size = 3973185, upload-time = "2024-10-14T23:37:40.226Z" }, - { url = "https://files.pythonhosted.org/packages/30/bf/08ad29979a936d63787ba47a540de2132169f140d54aa25bc8c3df3e67f4/uvloop-0.21.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:baa0e6291d91649c6ba4ed4b2f982f9fa165b5bbd50a9e203c416a2797bab3c6", size = 3820256, upload-time = "2024-10-14T23:37:42.839Z" }, - { url = "https://files.pythonhosted.org/packages/da/e2/5cf6ef37e3daf2f06e651aae5ea108ad30df3cb269102678b61ebf1fdf42/uvloop-0.21.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:4509360fcc4c3bd2c70d87573ad472de40c13387f5fda8cb58350a1d7475e58d", size = 3937323, upload-time = "2024-10-14T23:37:45.337Z" }, - { url = "https://files.pythonhosted.org/packages/8c/4c/03f93178830dc7ce8b4cdee1d36770d2f5ebb6f3d37d354e061eefc73545/uvloop-0.21.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:359ec2c888397b9e592a889c4d72ba3d6befba8b2bb01743f72fffbde663b59c", size = 1471284, upload-time = "2024-10-14T23:37:47.833Z" }, - { url = "https://files.pythonhosted.org/packages/43/3e/92c03f4d05e50f09251bd8b2b2b584a2a7f8fe600008bcc4523337abe676/uvloop-0.21.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:f7089d2dc73179ce5ac255bdf37c236a9f914b264825fdaacaded6990a7fb4c2", size = 821349, upload-time = "2024-10-14T23:37:50.149Z" }, - { url = "https://files.pythonhosted.org/packages/a6/ef/a02ec5da49909dbbfb1fd205a9a1ac4e88ea92dcae885e7c961847cd51e2/uvloop-0.21.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:baa4dcdbd9ae0a372f2167a207cd98c9f9a1ea1188a8a526431eef2f8116cc8d", size = 4580089, upload-time = "2024-10-14T23:37:51.703Z" }, - { url = "https://files.pythonhosted.org/packages/06/a7/b4e6a19925c900be9f98bec0a75e6e8f79bb53bdeb891916609ab3958967/uvloop-0.21.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:86975dca1c773a2c9864f4c52c5a55631038e387b47eaf56210f873887b6c8dc", size = 4693770, upload-time = "2024-10-14T23:37:54.122Z" }, - { url = "https://files.pythonhosted.org/packages/ce/0c/f07435a18a4b94ce6bd0677d8319cd3de61f3a9eeb1e5f8ab4e8b5edfcb3/uvloop-0.21.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:461d9ae6660fbbafedd07559c6a2e57cd553b34b0065b6550685f6653a98c1cb", size = 4451321, upload-time = "2024-10-14T23:37:55.766Z" }, - { url = "https://files.pythonhosted.org/packages/8f/eb/f7032be105877bcf924709c97b1bf3b90255b4ec251f9340cef912559f28/uvloop-0.21.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:183aef7c8730e54c9a3ee3227464daed66e37ba13040bb3f350bc2ddc040f22f", size = 4659022, upload-time = "2024-10-14T23:37:58.195Z" }, + { url = "https://files.pythonhosted.org/packages/c7/d5/69900f7883235562f1f50d8184bb7dd84a2fb61e9ec63f3782546fdbd057/uvloop-0.22.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:c60ebcd36f7b240b30788554b6f0782454826a0ed765d8430652621b5de674b9", size = 1352420, upload-time = "2025-10-16T22:16:21.187Z" }, + { url = "https://files.pythonhosted.org/packages/a8/73/c4e271b3bce59724e291465cc936c37758886a4868787da0278b3b56b905/uvloop-0.22.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3b7f102bf3cb1995cfeaee9321105e8f5da76fdb104cdad8986f85461a1b7b77", size = 748677, upload-time = "2025-10-16T22:16:22.558Z" }, + { url = "https://files.pythonhosted.org/packages/86/94/9fb7fad2f824d25f8ecac0d70b94d0d48107ad5ece03769a9c543444f78a/uvloop-0.22.1-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:53c85520781d84a4b8b230e24a5af5b0778efdb39142b424990ff1ef7c48ba21", size = 3753819, upload-time = "2025-10-16T22:16:23.903Z" }, + { url = "https://files.pythonhosted.org/packages/74/4f/256aca690709e9b008b7108bc85fba619a2bc37c6d80743d18abad16ee09/uvloop-0.22.1-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:56a2d1fae65fd82197cb8c53c367310b3eabe1bbb9fb5a04d28e3e3520e4f702", size = 3804529, upload-time = "2025-10-16T22:16:25.246Z" }, + { url = "https://files.pythonhosted.org/packages/7f/74/03c05ae4737e871923d21a76fe28b6aad57f5c03b6e6bfcfa5ad616013e4/uvloop-0.22.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:40631b049d5972c6755b06d0bfe8233b1bd9a8a6392d9d1c45c10b6f9e9b2733", size = 3621267, upload-time = "2025-10-16T22:16:26.819Z" }, + { url = "https://files.pythonhosted.org/packages/75/be/f8e590fe61d18b4a92070905497aec4c0e64ae1761498cad09023f3f4b3e/uvloop-0.22.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:535cc37b3a04f6cd2c1ef65fa1d370c9a35b6695df735fcff5427323f2cd5473", size = 3723105, upload-time = "2025-10-16T22:16:28.252Z" }, + { url = "https://files.pythonhosted.org/packages/3d/ff/7f72e8170be527b4977b033239a83a68d5c881cc4775fca255c677f7ac5d/uvloop-0.22.1-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:fe94b4564e865d968414598eea1a6de60adba0c040ba4ed05ac1300de402cd42", size = 1359936, upload-time = "2025-10-16T22:16:29.436Z" }, + { url = "https://files.pythonhosted.org/packages/c3/c6/e5d433f88fd54d81ef4be58b2b7b0cea13c442454a1db703a1eea0db1a59/uvloop-0.22.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:51eb9bd88391483410daad430813d982010f9c9c89512321f5b60e2cddbdddd6", size = 752769, upload-time = "2025-10-16T22:16:30.493Z" }, + { url = "https://files.pythonhosted.org/packages/24/68/a6ac446820273e71aa762fa21cdcc09861edd3536ff47c5cd3b7afb10eeb/uvloop-0.22.1-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:700e674a166ca5778255e0e1dc4e9d79ab2acc57b9171b79e65feba7184b3370", size = 4317413, upload-time = "2025-10-16T22:16:31.644Z" }, + { url = "https://files.pythonhosted.org/packages/5f/6f/e62b4dfc7ad6518e7eff2516f680d02a0f6eb62c0c212e152ca708a0085e/uvloop-0.22.1-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:7b5b1ac819a3f946d3b2ee07f09149578ae76066d70b44df3fa990add49a82e4", size = 4426307, upload-time = "2025-10-16T22:16:32.917Z" }, + { url = "https://files.pythonhosted.org/packages/90/60/97362554ac21e20e81bcef1150cb2a7e4ffdaf8ea1e5b2e8bf7a053caa18/uvloop-0.22.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:e047cc068570bac9866237739607d1313b9253c3051ad84738cbb095be0537b2", size = 4131970, upload-time = "2025-10-16T22:16:34.015Z" }, + { url = "https://files.pythonhosted.org/packages/99/39/6b3f7d234ba3964c428a6e40006340f53ba37993f46ed6e111c6e9141d18/uvloop-0.22.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:512fec6815e2dd45161054592441ef76c830eddaad55c8aa30952e6fe1ed07c0", size = 4296343, upload-time = "2025-10-16T22:16:35.149Z" }, ] [[package]] @@ -6814,7 +6818,7 @@ wheels = [ [[package]] name = "wandb" -version = "0.22.2" +version = "0.23.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "click" }, @@ -6828,58 +6832,58 @@ dependencies = [ { name = "sentry-sdk" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/c1/a8/680bd77e11a278e6c14a2cb4646e8ab9525b2baaa81c3d12dc0f616aa4aa/wandb-0.22.2.tar.gz", hash = "sha256:510f5a1ac30d16921c36c3b932da852f046641d4aee98a86a7f5ec03a6e95bda", size = 41401439, upload-time = "2025-10-07T19:54:21.88Z" } +sdist = { url = "https://files.pythonhosted.org/packages/ef/8b/db2d44395c967cd452517311fd6ede5d1e07310769f448358d4874248512/wandb-0.23.0.tar.gz", hash = "sha256:e5f98c61a8acc3ee84583ca78057f64344162ce026b9f71cb06eea44aec27c93", size = 44413921, upload-time = "2025-11-11T21:06:30.737Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/e7/b3/8c637fb594cfd574ce9c9f7d0ac2f2d12742eb38ec59dcbb713beae95343/wandb-0.22.2-py3-none-macosx_12_0_arm64.whl", hash = "sha256:2e29c9fa4462b5411b2cd2175ae33eff4309c91de7c426bca6bc8e7abc7e5dec", size = 18677549, upload-time = "2025-10-07T19:54:00.839Z" }, - { url = "https://files.pythonhosted.org/packages/d3/f3/e309a726eaebddad6b8d9a73a50891e5796962ec8a091bb6a61d31692d1e/wandb-0.22.2-py3-none-macosx_12_0_x86_64.whl", hash = "sha256:c42d594cd7a9da4fd39ecdb0abbc081b61f304123277b2b6c4ba84283956fd21", size = 19715188, upload-time = "2025-10-07T19:54:03.805Z" }, - { url = "https://files.pythonhosted.org/packages/f9/73/fad59910215876008f4781b57d828d1b19b3677c9b46af615e7229746435/wandb-0.22.2-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5188d84e66d3fd584f3b3ae4d2a70e78f29403c0528e6aecaa4188a1fcf54d8", size = 18463148, upload-time = "2025-10-07T19:54:05.676Z" }, - { url = "https://files.pythonhosted.org/packages/87/11/572c1913b5b92e4c519f735adfae572b46f2d79d99ede63eec0d6a272d6e/wandb-0.22.2-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:88ccd484af9f21cfc127976793c3cf66cfe1acd75bd8cd650086a64e88bac4bf", size = 19908645, upload-time = "2025-10-07T19:54:07.693Z" }, - { url = "https://files.pythonhosted.org/packages/6d/0d/133aa82f5a505ba638b4fda5014cefddfe7f1f6238ef4afc0871ec61c41f/wandb-0.22.2-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:abf0ed175e791af64110e0a0b99ce02bbbbd1017722bc32d3bc328efb86450cd", size = 18501348, upload-time = "2025-10-07T19:54:10.234Z" }, - { url = "https://files.pythonhosted.org/packages/d0/d5/776203be2601872f01dacc6a5b4274106ec0db7cd3bf2cdb3b741f8fc932/wandb-0.22.2-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:44e77c56403b90bf3473a7ca3bfc4d42c636b7c0e31a5fb9cd0382f08302f74b", size = 20001756, upload-time = "2025-10-07T19:54:12.452Z" }, - { url = "https://files.pythonhosted.org/packages/30/43/ae3fa46e20b1d9a6508dd9abe716d57205c038ed4661c5c98ace48a60eac/wandb-0.22.2-py3-none-win32.whl", hash = "sha256:44d12bd379dbe15be5ceed6bdf23803d42f648ba0dd111297b4c47a3c7be6dbd", size = 19075950, upload-time = "2025-10-07T19:54:14.892Z" }, - { url = "https://files.pythonhosted.org/packages/09/59/c174321e868205f7a659d1e5ec51f546e62267296d6f4179bb9119294964/wandb-0.22.2-py3-none-win_amd64.whl", hash = "sha256:c95eb221bf316c0872f7ac55071856b9f25f95a2de983ada48acf653ce259386", size = 19075953, upload-time = "2025-10-07T19:54:16.837Z" }, - { url = "https://files.pythonhosted.org/packages/7a/a2/c7c24fda78513cab5686949d8cb36459dbbccbbb4b2b6fc67237ece31a00/wandb-0.22.2-py3-none-win_arm64.whl", hash = "sha256:20d2ab9aa10445aab3d60914a980f002a4f66566e28b0cd156b1e462f0080a0d", size = 17383217, upload-time = "2025-10-07T19:54:19.384Z" }, + { url = "https://files.pythonhosted.org/packages/41/61/a3220c7fa4cadfb2b2a5c09e3fa401787326584ade86d7c1f58bf1cd43bd/wandb-0.23.0-py3-none-macosx_12_0_arm64.whl", hash = "sha256:b682ec5e38fc97bd2e868ac7615a0ab4fc6a15220ee1159e87270a5ebb7a816d", size = 18992250, upload-time = "2025-11-11T21:06:03.412Z" }, + { url = "https://files.pythonhosted.org/packages/90/16/e69333cf3d11e7847f424afc6c8ae325e1f6061b2e5118d7a17f41b6525d/wandb-0.23.0-py3-none-macosx_12_0_x86_64.whl", hash = "sha256:ec094eb71b778e77db8c188da19e52c4f96cb9d5b4421d7dc05028afc66fd7e7", size = 20045616, upload-time = "2025-11-11T21:06:07.109Z" }, + { url = "https://files.pythonhosted.org/packages/62/79/42dc6c7bb0b425775fe77f1a3f1a22d75d392841a06b43e150a3a7f2553a/wandb-0.23.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4e43f1f04b98c34f407dcd2744cec0a590abce39bed14a61358287f817514a7b", size = 18758848, upload-time = "2025-11-11T21:06:09.832Z" }, + { url = "https://files.pythonhosted.org/packages/b8/94/d6ddb78334996ccfc1179444bfcfc0f37ffd07ee79bb98940466da6f68f8/wandb-0.23.0-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6e5847f98cbb3175caf5291932374410141f5bb3b7c25f9c5e562c1988ce0bf5", size = 20231493, upload-time = "2025-11-11T21:06:12.323Z" }, + { url = "https://files.pythonhosted.org/packages/52/4d/0ad6df0e750c19dabd24d2cecad0938964f69a072f05fbdab7281bec2b64/wandb-0.23.0-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:6151355fd922539926e870be811474238c9614b96541773b990f1ce53368aef6", size = 18793473, upload-time = "2025-11-11T21:06:14.967Z" }, + { url = "https://files.pythonhosted.org/packages/f8/da/c2ba49c5573dff93dafc0acce691bb1c3d57361bf834b2f2c58e6193439b/wandb-0.23.0-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:df62e426e448ebc44269140deb7240df474e743b12d4b1f53b753afde4aa06d4", size = 20332882, upload-time = "2025-11-11T21:06:17.865Z" }, + { url = "https://files.pythonhosted.org/packages/40/65/21bfb10ee5cd93fbcaf794958863c7e05bac4bbeb1cc1b652094aa3743a5/wandb-0.23.0-py3-none-win32.whl", hash = "sha256:6c21d3eadda17aef7df6febdffdddfb0b4835c7754435fc4fe27631724269f5c", size = 19433198, upload-time = "2025-11-11T21:06:21.913Z" }, + { url = "https://files.pythonhosted.org/packages/f1/33/cbe79e66c171204e32cf940c7fdfb8b5f7d2af7a00f301c632f3a38aa84b/wandb-0.23.0-py3-none-win_amd64.whl", hash = "sha256:b50635fa0e16e528bde25715bf446e9153368428634ca7a5dbd7a22c8ae4e915", size = 19433201, upload-time = "2025-11-11T21:06:24.607Z" }, + { url = "https://files.pythonhosted.org/packages/1c/a0/5ecfae12d78ea036a746c071e4c13b54b28d641efbba61d2947c73b3e6f9/wandb-0.23.0-py3-none-win_arm64.whl", hash = "sha256:fa0181b02ce4d1993588f4a728d8b73ae487eb3cb341e6ce01c156be7a98ec72", size = 17678649, upload-time = "2025-11-11T21:06:27.289Z" }, ] [[package]] name = "watchfiles" -version = "1.1.0" +version = "1.1.1" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "anyio" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/2a/9a/d451fcc97d029f5812e898fd30a53fd8c15c7bbd058fd75cfc6beb9bd761/watchfiles-1.1.0.tar.gz", hash = "sha256:693ed7ec72cbfcee399e92c895362b6e66d63dac6b91e2c11ae03d10d503e575", size = 94406, upload-time = "2025-06-15T19:06:59.42Z" } +sdist = { url = "https://files.pythonhosted.org/packages/c2/c9/8869df9b2a2d6c59d79220a4db37679e74f807c559ffe5265e08b227a210/watchfiles-1.1.1.tar.gz", hash = "sha256:a173cb5c16c4f40ab19cecf48a534c409f7ea983ab8fed0741304a1c0a31b3f2", size = 94440, upload-time = "2025-10-14T15:06:21.08Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/8b/78/7401154b78ab484ccaaeef970dc2af0cb88b5ba8a1b415383da444cdd8d3/watchfiles-1.1.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:c9649dfc57cc1f9835551deb17689e8d44666315f2e82d337b9f07bd76ae3aa2", size = 405751, upload-time = "2025-06-15T19:05:07.679Z" }, - { url = "https://files.pythonhosted.org/packages/76/63/e6c3dbc1f78d001589b75e56a288c47723de28c580ad715eb116639152b5/watchfiles-1.1.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:406520216186b99374cdb58bc48e34bb74535adec160c8459894884c983a149c", size = 397313, upload-time = "2025-06-15T19:05:08.764Z" }, - { url = "https://files.pythonhosted.org/packages/6c/a2/8afa359ff52e99af1632f90cbf359da46184207e893a5f179301b0c8d6df/watchfiles-1.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb45350fd1dc75cd68d3d72c47f5b513cb0578da716df5fba02fff31c69d5f2d", size = 450792, upload-time = "2025-06-15T19:05:09.869Z" }, - { url = "https://files.pythonhosted.org/packages/1d/bf/7446b401667f5c64972a57a0233be1104157fc3abf72c4ef2666c1bd09b2/watchfiles-1.1.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:11ee4444250fcbeb47459a877e5e80ed994ce8e8d20283857fc128be1715dac7", size = 458196, upload-time = "2025-06-15T19:05:11.91Z" }, - { url = "https://files.pythonhosted.org/packages/58/2f/501ddbdfa3fa874ea5597c77eeea3d413579c29af26c1091b08d0c792280/watchfiles-1.1.0-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bda8136e6a80bdea23e5e74e09df0362744d24ffb8cd59c4a95a6ce3d142f79c", size = 484788, upload-time = "2025-06-15T19:05:13.373Z" }, - { url = "https://files.pythonhosted.org/packages/61/1e/9c18eb2eb5c953c96bc0e5f626f0e53cfef4bd19bd50d71d1a049c63a575/watchfiles-1.1.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b915daeb2d8c1f5cee4b970f2e2c988ce6514aace3c9296e58dd64dc9aa5d575", size = 597879, upload-time = "2025-06-15T19:05:14.725Z" }, - { url = "https://files.pythonhosted.org/packages/8b/6c/1467402e5185d89388b4486745af1e0325007af0017c3384cc786fff0542/watchfiles-1.1.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ed8fc66786de8d0376f9f913c09e963c66e90ced9aa11997f93bdb30f7c872a8", size = 477447, upload-time = "2025-06-15T19:05:15.775Z" }, - { url = "https://files.pythonhosted.org/packages/2b/a1/ec0a606bde4853d6c4a578f9391eeb3684a9aea736a8eb217e3e00aa89a1/watchfiles-1.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fe4371595edf78c41ef8ac8df20df3943e13defd0efcb732b2e393b5a8a7a71f", size = 453145, upload-time = "2025-06-15T19:05:17.17Z" }, - { url = "https://files.pythonhosted.org/packages/90/b9/ef6f0c247a6a35d689fc970dc7f6734f9257451aefb30def5d100d6246a5/watchfiles-1.1.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:b7c5f6fe273291f4d414d55b2c80d33c457b8a42677ad14b4b47ff025d0893e4", size = 626539, upload-time = "2025-06-15T19:05:18.557Z" }, - { url = "https://files.pythonhosted.org/packages/34/44/6ffda5537085106ff5aaa762b0d130ac6c75a08015dd1621376f708c94de/watchfiles-1.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:7738027989881e70e3723c75921f1efa45225084228788fc59ea8c6d732eb30d", size = 624472, upload-time = "2025-06-15T19:05:19.588Z" }, - { url = "https://files.pythonhosted.org/packages/c3/e3/71170985c48028fa3f0a50946916a14055e741db11c2e7bc2f3b61f4d0e3/watchfiles-1.1.0-cp311-cp311-win32.whl", hash = "sha256:622d6b2c06be19f6e89b1d951485a232e3b59618def88dbeda575ed8f0d8dbf2", size = 279348, upload-time = "2025-06-15T19:05:20.856Z" }, - { url = "https://files.pythonhosted.org/packages/89/1b/3e39c68b68a7a171070f81fc2561d23ce8d6859659406842a0e4bebf3bba/watchfiles-1.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:48aa25e5992b61debc908a61ab4d3f216b64f44fdaa71eb082d8b2de846b7d12", size = 292607, upload-time = "2025-06-15T19:05:21.937Z" }, - { url = "https://files.pythonhosted.org/packages/61/9f/2973b7539f2bdb6ea86d2c87f70f615a71a1fc2dba2911795cea25968aea/watchfiles-1.1.0-cp311-cp311-win_arm64.whl", hash = "sha256:00645eb79a3faa70d9cb15c8d4187bb72970b2470e938670240c7998dad9f13a", size = 285056, upload-time = "2025-06-15T19:05:23.12Z" }, - { url = "https://files.pythonhosted.org/packages/f6/b8/858957045a38a4079203a33aaa7d23ea9269ca7761c8a074af3524fbb240/watchfiles-1.1.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:9dc001c3e10de4725c749d4c2f2bdc6ae24de5a88a339c4bce32300a31ede179", size = 402339, upload-time = "2025-06-15T19:05:24.516Z" }, - { url = "https://files.pythonhosted.org/packages/80/28/98b222cca751ba68e88521fabd79a4fab64005fc5976ea49b53fa205d1fa/watchfiles-1.1.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:d9ba68ec283153dead62cbe81872d28e053745f12335d037de9cbd14bd1877f5", size = 394409, upload-time = "2025-06-15T19:05:25.469Z" }, - { url = "https://files.pythonhosted.org/packages/86/50/dee79968566c03190677c26f7f47960aff738d32087087bdf63a5473e7df/watchfiles-1.1.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:130fc497b8ee68dce163e4254d9b0356411d1490e868bd8790028bc46c5cc297", size = 450939, upload-time = "2025-06-15T19:05:26.494Z" }, - { url = "https://files.pythonhosted.org/packages/40/45/a7b56fb129700f3cfe2594a01aa38d033b92a33dddce86c8dfdfc1247b72/watchfiles-1.1.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:50a51a90610d0845a5931a780d8e51d7bd7f309ebc25132ba975aca016b576a0", size = 457270, upload-time = "2025-06-15T19:05:27.466Z" }, - { url = "https://files.pythonhosted.org/packages/b5/c8/fa5ef9476b1d02dc6b5e258f515fcaaecf559037edf8b6feffcbc097c4b8/watchfiles-1.1.0-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dc44678a72ac0910bac46fa6a0de6af9ba1355669b3dfaf1ce5f05ca7a74364e", size = 483370, upload-time = "2025-06-15T19:05:28.548Z" }, - { url = "https://files.pythonhosted.org/packages/98/68/42cfcdd6533ec94f0a7aab83f759ec11280f70b11bfba0b0f885e298f9bd/watchfiles-1.1.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a543492513a93b001975ae283a51f4b67973662a375a403ae82f420d2c7205ee", size = 598654, upload-time = "2025-06-15T19:05:29.997Z" }, - { url = "https://files.pythonhosted.org/packages/d3/74/b2a1544224118cc28df7e59008a929e711f9c68ce7d554e171b2dc531352/watchfiles-1.1.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8ac164e20d17cc285f2b94dc31c384bc3aa3dd5e7490473b3db043dd70fbccfd", size = 478667, upload-time = "2025-06-15T19:05:31.172Z" }, - { url = "https://files.pythonhosted.org/packages/8c/77/e3362fe308358dc9f8588102481e599c83e1b91c2ae843780a7ded939a35/watchfiles-1.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f7590d5a455321e53857892ab8879dce62d1f4b04748769f5adf2e707afb9d4f", size = 452213, upload-time = "2025-06-15T19:05:32.299Z" }, - { url = "https://files.pythonhosted.org/packages/6e/17/c8f1a36540c9a1558d4faf08e909399e8133599fa359bf52ec8fcee5be6f/watchfiles-1.1.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:37d3d3f7defb13f62ece99e9be912afe9dd8a0077b7c45ee5a57c74811d581a4", size = 626718, upload-time = "2025-06-15T19:05:33.415Z" }, - { url = "https://files.pythonhosted.org/packages/26/45/fb599be38b4bd38032643783d7496a26a6f9ae05dea1a42e58229a20ac13/watchfiles-1.1.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:7080c4bb3efd70a07b1cc2df99a7aa51d98685be56be6038c3169199d0a1c69f", size = 623098, upload-time = "2025-06-15T19:05:34.534Z" }, - { url = "https://files.pythonhosted.org/packages/a1/e7/fdf40e038475498e160cd167333c946e45d8563ae4dd65caf757e9ffe6b4/watchfiles-1.1.0-cp312-cp312-win32.whl", hash = "sha256:cbcf8630ef4afb05dc30107bfa17f16c0896bb30ee48fc24bf64c1f970f3b1fd", size = 279209, upload-time = "2025-06-15T19:05:35.577Z" }, - { url = "https://files.pythonhosted.org/packages/3f/d3/3ae9d5124ec75143bdf088d436cba39812122edc47709cd2caafeac3266f/watchfiles-1.1.0-cp312-cp312-win_amd64.whl", hash = "sha256:cbd949bdd87567b0ad183d7676feb98136cde5bb9025403794a4c0db28ed3a47", size = 292786, upload-time = "2025-06-15T19:05:36.559Z" }, - { url = "https://files.pythonhosted.org/packages/26/2f/7dd4fc8b5f2b34b545e19629b4a018bfb1de23b3a496766a2c1165ca890d/watchfiles-1.1.0-cp312-cp312-win_arm64.whl", hash = "sha256:0a7d40b77f07be87c6faa93d0951a0fcd8cbca1ddff60a1b65d741bac6f3a9f6", size = 284343, upload-time = "2025-06-15T19:05:37.5Z" }, - { url = "https://files.pythonhosted.org/packages/8c/6b/686dcf5d3525ad17b384fd94708e95193529b460a1b7bf40851f1328ec6e/watchfiles-1.1.0-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:0ece16b563b17ab26eaa2d52230c9a7ae46cf01759621f4fbbca280e438267b3", size = 406910, upload-time = "2025-06-15T19:06:49.335Z" }, - { url = "https://files.pythonhosted.org/packages/f3/d3/71c2dcf81dc1edcf8af9f4d8d63b1316fb0a2dd90cbfd427e8d9dd584a90/watchfiles-1.1.0-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:51b81e55d40c4b4aa8658427a3ee7ea847c591ae9e8b81ef94a90b668999353c", size = 398816, upload-time = "2025-06-15T19:06:50.433Z" }, - { url = "https://files.pythonhosted.org/packages/b8/fa/12269467b2fc006f8fce4cd6c3acfa77491dd0777d2a747415f28ccc8c60/watchfiles-1.1.0-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f2bcdc54ea267fe72bfc7d83c041e4eb58d7d8dc6f578dfddb52f037ce62f432", size = 451584, upload-time = "2025-06-15T19:06:51.834Z" }, - { url = "https://files.pythonhosted.org/packages/bd/d3/254cea30f918f489db09d6a8435a7de7047f8cb68584477a515f160541d6/watchfiles-1.1.0-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:923fec6e5461c42bd7e3fd5ec37492c6f3468be0499bc0707b4bbbc16ac21792", size = 454009, upload-time = "2025-06-15T19:06:52.896Z" }, + { url = "https://files.pythonhosted.org/packages/1f/f8/2c5f479fb531ce2f0564eda479faecf253d886b1ab3630a39b7bf7362d46/watchfiles-1.1.1-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:f57b396167a2565a4e8b5e56a5a1c537571733992b226f4f1197d79e94cf0ae5", size = 406529, upload-time = "2025-10-14T15:04:32.899Z" }, + { url = "https://files.pythonhosted.org/packages/fe/cd/f515660b1f32f65df671ddf6f85bfaca621aee177712874dc30a97397977/watchfiles-1.1.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:421e29339983e1bebc281fab40d812742268ad057db4aee8c4d2bce0af43b741", size = 394384, upload-time = "2025-10-14T15:04:33.761Z" }, + { url = "https://files.pythonhosted.org/packages/7b/c3/28b7dc99733eab43fca2d10f55c86e03bd6ab11ca31b802abac26b23d161/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6e43d39a741e972bab5d8100b5cdacf69db64e34eb19b6e9af162bccf63c5cc6", size = 448789, upload-time = "2025-10-14T15:04:34.679Z" }, + { url = "https://files.pythonhosted.org/packages/4a/24/33e71113b320030011c8e4316ccca04194bf0cbbaeee207f00cbc7d6b9f5/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f537afb3276d12814082a2e9b242bdcf416c2e8fd9f799a737990a1dbe906e5b", size = 460521, upload-time = "2025-10-14T15:04:35.963Z" }, + { url = "https://files.pythonhosted.org/packages/f4/c3/3c9a55f255aa57b91579ae9e98c88704955fa9dac3e5614fb378291155df/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b2cd9e04277e756a2e2d2543d65d1e2166d6fd4c9b183f8808634fda23f17b14", size = 488722, upload-time = "2025-10-14T15:04:37.091Z" }, + { url = "https://files.pythonhosted.org/packages/49/36/506447b73eb46c120169dc1717fe2eff07c234bb3232a7200b5f5bd816e9/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5f3f58818dc0b07f7d9aa7fe9eb1037aecb9700e63e1f6acfed13e9fef648f5d", size = 596088, upload-time = "2025-10-14T15:04:38.39Z" }, + { url = "https://files.pythonhosted.org/packages/82/ab/5f39e752a9838ec4d52e9b87c1e80f1ee3ccdbe92e183c15b6577ab9de16/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9bb9f66367023ae783551042d31b1d7fd422e8289eedd91f26754a66f44d5cff", size = 472923, upload-time = "2025-10-14T15:04:39.666Z" }, + { url = "https://files.pythonhosted.org/packages/af/b9/a419292f05e302dea372fa7e6fda5178a92998411f8581b9830d28fb9edb/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aebfd0861a83e6c3d1110b78ad54704486555246e542be3e2bb94195eabb2606", size = 456080, upload-time = "2025-10-14T15:04:40.643Z" }, + { url = "https://files.pythonhosted.org/packages/b0/c3/d5932fd62bde1a30c36e10c409dc5d54506726f08cb3e1d8d0ba5e2bc8db/watchfiles-1.1.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:5fac835b4ab3c6487b5dbad78c4b3724e26bcc468e886f8ba8cc4306f68f6701", size = 629432, upload-time = "2025-10-14T15:04:41.789Z" }, + { url = "https://files.pythonhosted.org/packages/f7/77/16bddd9779fafb795f1a94319dc965209c5641db5bf1edbbccace6d1b3c0/watchfiles-1.1.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:399600947b170270e80134ac854e21b3ccdefa11a9529a3decc1327088180f10", size = 623046, upload-time = "2025-10-14T15:04:42.718Z" }, + { url = "https://files.pythonhosted.org/packages/46/ef/f2ecb9a0f342b4bfad13a2787155c6ee7ce792140eac63a34676a2feeef2/watchfiles-1.1.1-cp311-cp311-win32.whl", hash = "sha256:de6da501c883f58ad50db3a32ad397b09ad29865b5f26f64c24d3e3281685849", size = 271473, upload-time = "2025-10-14T15:04:43.624Z" }, + { url = "https://files.pythonhosted.org/packages/94/bc/f42d71125f19731ea435c3948cad148d31a64fccde3867e5ba4edee901f9/watchfiles-1.1.1-cp311-cp311-win_amd64.whl", hash = "sha256:35c53bd62a0b885bf653ebf6b700d1bf05debb78ad9292cf2a942b23513dc4c4", size = 287598, upload-time = "2025-10-14T15:04:44.516Z" }, + { url = "https://files.pythonhosted.org/packages/57/c9/a30f897351f95bbbfb6abcadafbaca711ce1162f4db95fc908c98a9165f3/watchfiles-1.1.1-cp311-cp311-win_arm64.whl", hash = "sha256:57ca5281a8b5e27593cb7d82c2ac927ad88a96ed406aa446f6344e4328208e9e", size = 277210, upload-time = "2025-10-14T15:04:45.883Z" }, + { url = "https://files.pythonhosted.org/packages/74/d5/f039e7e3c639d9b1d09b07ea412a6806d38123f0508e5f9b48a87b0a76cc/watchfiles-1.1.1-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:8c89f9f2f740a6b7dcc753140dd5e1ab9215966f7a3530d0c0705c83b401bd7d", size = 404745, upload-time = "2025-10-14T15:04:46.731Z" }, + { url = "https://files.pythonhosted.org/packages/a5/96/a881a13aa1349827490dab2d363c8039527060cfcc2c92cc6d13d1b1049e/watchfiles-1.1.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:bd404be08018c37350f0d6e34676bd1e2889990117a2b90070b3007f172d0610", size = 391769, upload-time = "2025-10-14T15:04:48.003Z" }, + { url = "https://files.pythonhosted.org/packages/4b/5b/d3b460364aeb8da471c1989238ea0e56bec24b6042a68046adf3d9ddb01c/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8526e8f916bb5b9a0a777c8317c23ce65de259422bba5b31325a6fa6029d33af", size = 449374, upload-time = "2025-10-14T15:04:49.179Z" }, + { url = "https://files.pythonhosted.org/packages/b9/44/5769cb62d4ed055cb17417c0a109a92f007114a4e07f30812a73a4efdb11/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2edc3553362b1c38d9f06242416a5d8e9fe235c204a4072e988ce2e5bb1f69f6", size = 459485, upload-time = "2025-10-14T15:04:50.155Z" }, + { url = "https://files.pythonhosted.org/packages/19/0c/286b6301ded2eccd4ffd0041a1b726afda999926cf720aab63adb68a1e36/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:30f7da3fb3f2844259cba4720c3fc7138eb0f7b659c38f3bfa65084c7fc7abce", size = 488813, upload-time = "2025-10-14T15:04:51.059Z" }, + { url = "https://files.pythonhosted.org/packages/c7/2b/8530ed41112dd4a22f4dcfdb5ccf6a1baad1ff6eed8dc5a5f09e7e8c41c7/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f8979280bdafff686ba5e4d8f97840f929a87ed9cdf133cbbd42f7766774d2aa", size = 594816, upload-time = "2025-10-14T15:04:52.031Z" }, + { url = "https://files.pythonhosted.org/packages/ce/d2/f5f9fb49489f184f18470d4f99f4e862a4b3e9ac2865688eb2099e3d837a/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:dcc5c24523771db3a294c77d94771abcfcb82a0e0ee8efd910c37c59ec1b31bb", size = 475186, upload-time = "2025-10-14T15:04:53.064Z" }, + { url = "https://files.pythonhosted.org/packages/cf/68/5707da262a119fb06fbe214d82dd1fe4a6f4af32d2d14de368d0349eb52a/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1db5d7ae38ff20153d542460752ff397fcf5c96090c1230803713cf3147a6803", size = 456812, upload-time = "2025-10-14T15:04:55.174Z" }, + { url = "https://files.pythonhosted.org/packages/66/ab/3cbb8756323e8f9b6f9acb9ef4ec26d42b2109bce830cc1f3468df20511d/watchfiles-1.1.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:28475ddbde92df1874b6c5c8aaeb24ad5be47a11f87cde5a28ef3835932e3e94", size = 630196, upload-time = "2025-10-14T15:04:56.22Z" }, + { url = "https://files.pythonhosted.org/packages/78/46/7152ec29b8335f80167928944a94955015a345440f524d2dfe63fc2f437b/watchfiles-1.1.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:36193ed342f5b9842edd3532729a2ad55c4160ffcfa3700e0d54be496b70dd43", size = 622657, upload-time = "2025-10-14T15:04:57.521Z" }, + { url = "https://files.pythonhosted.org/packages/0a/bf/95895e78dd75efe9a7f31733607f384b42eb5feb54bd2eb6ed57cc2e94f4/watchfiles-1.1.1-cp312-cp312-win32.whl", hash = "sha256:859e43a1951717cc8de7f4c77674a6d389b106361585951d9e69572823f311d9", size = 272042, upload-time = "2025-10-14T15:04:59.046Z" }, + { url = "https://files.pythonhosted.org/packages/87/0a/90eb755f568de2688cb220171c4191df932232c20946966c27a59c400850/watchfiles-1.1.1-cp312-cp312-win_amd64.whl", hash = "sha256:91d4c9a823a8c987cce8fa2690923b069966dabb196dd8d137ea2cede885fde9", size = 288410, upload-time = "2025-10-14T15:05:00.081Z" }, + { url = "https://files.pythonhosted.org/packages/36/76/f322701530586922fbd6723c4f91ace21364924822a8772c549483abed13/watchfiles-1.1.1-cp312-cp312-win_arm64.whl", hash = "sha256:a625815d4a2bdca61953dbba5a39d60164451ef34c88d751f6c368c3ea73d404", size = 278209, upload-time = "2025-10-14T15:05:01.168Z" }, + { url = "https://files.pythonhosted.org/packages/d3/8e/e500f8b0b77be4ff753ac94dc06b33d8f0d839377fee1b78e8c8d8f031bf/watchfiles-1.1.1-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:db476ab59b6765134de1d4fe96a1a9c96ddf091683599be0f26147ea1b2e4b88", size = 408250, upload-time = "2025-10-14T15:06:10.264Z" }, + { url = "https://files.pythonhosted.org/packages/bd/95/615e72cd27b85b61eec764a5ca51bd94d40b5adea5ff47567d9ebc4d275a/watchfiles-1.1.1-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:89eef07eee5e9d1fda06e38822ad167a044153457e6fd997f8a858ab7564a336", size = 396117, upload-time = "2025-10-14T15:06:11.28Z" }, + { url = "https://files.pythonhosted.org/packages/c9/81/e7fe958ce8a7fb5c73cc9fb07f5aeaf755e6aa72498c57d760af760c91f8/watchfiles-1.1.1-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ce19e06cbda693e9e7686358af9cd6f5d61312ab8b00488bc36f5aabbaf77e24", size = 450493, upload-time = "2025-10-14T15:06:12.321Z" }, + { url = "https://files.pythonhosted.org/packages/6e/d4/ed38dd3b1767193de971e694aa544356e63353c33a85d948166b5ff58b9e/watchfiles-1.1.1-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3e6f39af2eab0118338902798b5aa6664f46ff66bc0280de76fca67a7f262a49", size = 457546, upload-time = "2025-10-14T15:06:13.372Z" }, ] [[package]] @@ -6893,7 +6897,7 @@ wheels = [ [[package]] name = "weave" -version = "0.51.59" +version = "0.52.16" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "click" }, @@ -6901,18 +6905,17 @@ dependencies = [ { name = "eval-type-backport" }, { name = "gql", extra = ["aiohttp", "requests"] }, { name = "jsonschema" }, - { name = "nest-asyncio" }, { name = "packaging" }, { name = "polyfile-weave" }, { name = "pydantic" }, - { name = "rich" }, { name = "sentry-sdk" }, { name = "tenacity" }, + { name = "tzdata", marker = "sys_platform == 'win32'" }, { name = "wandb" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/0e/53/1b0350a64837df3e29eda6149a542f3a51e706122086f82547153820e982/weave-0.51.59.tar.gz", hash = "sha256:fad34c0478f3470401274cba8fa2bfd45d14a187db0a5724bd507e356761b349", size = 480572, upload-time = "2025-07-25T22:05:07.458Z" } +sdist = { url = "https://files.pythonhosted.org/packages/be/30/b795b5a857e8a908e68f3ed969587bb2bc63527ef2260f72ac1a6fd983e8/weave-0.52.16.tar.gz", hash = "sha256:7bb8fdce0393007e9c40fb1769d0606bfe55401c4cd13146457ccac4b49c695d", size = 607024, upload-time = "2025-11-07T19:45:30.898Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/1d/bc/fa5ffb887a1ee28109b29c62416c9e0f41da8e75e6871671208b3d42b392/weave-0.51.59-py3-none-any.whl", hash = "sha256:2238578574ecdf6285efdf028c78987769720242ac75b7b84b1dbc59060468ce", size = 612468, upload-time = "2025-07-25T22:05:05.088Z" }, + { url = "https://files.pythonhosted.org/packages/e5/87/a54513796605dfaef2c3c23c2733bcb4b24866a623635c057b2ffdb74052/weave-0.52.16-py3-none-any.whl", hash = "sha256:85985b8cf233032c6d915dfac95b3bcccb1304444d99a6b4a61f3666b58146ce", size = 764366, upload-time = "2025-11-07T19:45:28.878Z" }, ] [[package]] @@ -7130,34 +7133,29 @@ wheels = [ [[package]] name = "zope-event" -version = "6.0" +version = "6.1" source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "setuptools" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/c2/d8/9c8b0c6bb1db09725395618f68d3b8a08089fca0aed28437500caaf713ee/zope_event-6.0.tar.gz", hash = "sha256:0ebac894fa7c5f8b7a89141c272133d8c1de6ddc75ea4b1f327f00d1f890df92", size = 18731, upload-time = "2025-09-12T07:10:13.551Z" } +sdist = { url = "https://files.pythonhosted.org/packages/46/33/d3eeac228fc14de76615612ee208be2d8a5b5b0fada36bf9b62d6b40600c/zope_event-6.1.tar.gz", hash = "sha256:6052a3e0cb8565d3d4ef1a3a7809336ac519bc4fe38398cb8d466db09adef4f0", size = 18739, upload-time = "2025-11-07T08:05:49.934Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/d1/b5/1abb5a8b443314c978617bf46d5d9ad648bdf21058074e817d7efbb257db/zope_event-6.0-py3-none-any.whl", hash = "sha256:6f0922593407cc673e7d8766b492c519f91bdc99f3080fe43dcec0a800d682a3", size = 6409, upload-time = "2025-09-12T07:10:12.316Z" }, + { url = "https://files.pythonhosted.org/packages/c2/b0/956902e5e1302f8c5d124e219c6bf214e2649f92ad5fce85b05c039a04c9/zope_event-6.1-py3-none-any.whl", hash = "sha256:0ca78b6391b694272b23ec1335c0294cc471065ed10f7f606858fc54566c25a0", size = 6414, upload-time = "2025-11-07T08:05:48.874Z" }, ] [[package]] name = "zope-interface" -version = "8.0.1" +version = "8.1" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/88/3a/7fcf02178b8fad0a51e67e32765cd039ae505d054d744d76b8c2bbcba5ba/zope_interface-8.0.1.tar.gz", hash = "sha256:eba5610d042c3704a48222f7f7c6ab5b243ed26f917e2bc69379456b115e02d1", size = 253746, upload-time = "2025-09-25T05:55:51.285Z" } +sdist = { url = "https://files.pythonhosted.org/packages/6a/7d/b5b85e09f87be5f33decde2626347123696fc6d9d655cb16f5a986b60a97/zope_interface-8.1.tar.gz", hash = "sha256:a02ee40770c6a2f3d168a8f71f09b62aec3e4fb366da83f8e849dbaa5b38d12f", size = 253831, upload-time = "2025-11-10T07:56:24.825Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/f2/2f/c10c739bcb9b072090c97c2e08533777497190daa19d190d72b4cce9c7cb/zope_interface-8.0.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:4bd01022d2e1bce4a4a4ed9549edb25393c92e607d7daa6deff843f1f68b479d", size = 207903, upload-time = "2025-09-25T05:58:21.671Z" }, - { url = "https://files.pythonhosted.org/packages/b5/e1/9845ac3697f108d9a1af6912170c59a23732090bbfb35955fe77e5544955/zope_interface-8.0.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:29be8db8b712d94f1c05e24ea230a879271d787205ba1c9a6100d1d81f06c69a", size = 208345, upload-time = "2025-09-25T05:58:24.217Z" }, - { url = "https://files.pythonhosted.org/packages/f2/49/6573bc8b841cfab18e80c8e8259f1abdbbf716140011370de30231be79ad/zope_interface-8.0.1-cp311-cp311-manylinux1_i686.manylinux2014_i686.manylinux_2_17_i686.manylinux_2_5_i686.whl", hash = "sha256:51ae1b856565b30455b7879fdf0a56a88763b401d3f814fa9f9542d7410dbd7e", size = 255027, upload-time = "2025-09-25T05:58:19.975Z" }, - { url = "https://files.pythonhosted.org/packages/e2/fd/908b0fd4b1ab6e412dfac9bd2b606f2893ef9ba3dd36d643f5e5b94c57b3/zope_interface-8.0.1-cp311-cp311-manylinux1_x86_64.manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:d2e7596149cb1acd1d4d41b9f8fe2ffc0e9e29e2e91d026311814181d0d9efaf", size = 259800, upload-time = "2025-09-25T05:58:11.487Z" }, - { url = "https://files.pythonhosted.org/packages/dc/78/8419a2b4e88410520ed4b7f93bbd25a6d4ae66c4e2b131320f2b90f43077/zope_interface-8.0.1-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:b2737c11c34fb9128816759864752d007ec4f987b571c934c30723ed881a7a4f", size = 260978, upload-time = "2025-09-25T06:26:24.483Z" }, - { url = "https://files.pythonhosted.org/packages/e5/90/caf68152c292f1810e2bd3acd2177badf08a740aa8a348714617d6c9ad0b/zope_interface-8.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:cf66e4bf731aa7e0ced855bb3670e8cda772f6515a475c6a107bad5cb6604103", size = 212155, upload-time = "2025-09-25T05:59:40.318Z" }, - { url = "https://files.pythonhosted.org/packages/dc/a6/0f08713ddda834c428ebf97b2a7fd8dea50c0100065a8955924dbd94dae8/zope_interface-8.0.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:115f27c1cc95ce7a517d960ef381beedb0a7ce9489645e80b9ab3cbf8a78799c", size = 208609, upload-time = "2025-09-25T05:58:53.698Z" }, - { url = "https://files.pythonhosted.org/packages/e9/5e/d423045f54dc81e0991ec655041e7a0eccf6b2642535839dd364b35f4d7f/zope_interface-8.0.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:af655c573b84e3cb6a4f6fd3fbe04e4dc91c63c6b6f99019b3713ef964e589bc", size = 208797, upload-time = "2025-09-25T05:58:56.258Z" }, - { url = "https://files.pythonhosted.org/packages/c6/43/39d4bb3f7a80ebd261446792493cfa4e198badd47107224f5b6fe1997ad9/zope_interface-8.0.1-cp312-cp312-manylinux1_i686.manylinux2014_i686.manylinux_2_17_i686.manylinux_2_5_i686.whl", hash = "sha256:23f82ef9b2d5370750cc1bf883c3b94c33d098ce08557922a3fbc7ff3b63dfe1", size = 259242, upload-time = "2025-09-25T05:58:21.602Z" }, - { url = "https://files.pythonhosted.org/packages/da/29/49effcff64ef30731e35520a152a9dfcafec86cf114b4c2aff942e8264ba/zope_interface-8.0.1-cp312-cp312-manylinux1_x86_64.manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:35a1565d5244997f2e629c5c68715b3d9d9036e8df23c4068b08d9316dcb2822", size = 264696, upload-time = "2025-09-25T05:58:13.351Z" }, - { url = "https://files.pythonhosted.org/packages/c7/39/b947673ec9a258eeaa20208dd2f6127d9fbb3e5071272a674ebe02063a78/zope_interface-8.0.1-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:029ea1db7e855a475bf88d9910baab4e94d007a054810e9007ac037a91c67c6f", size = 264229, upload-time = "2025-09-25T06:26:26.226Z" }, - { url = "https://files.pythonhosted.org/packages/8f/ee/eed6efd1fc3788d1bef7a814e0592d8173b7fe601c699b935009df035fc2/zope_interface-8.0.1-cp312-cp312-win_amd64.whl", hash = "sha256:0beb3e7f7dc153944076fcaf717a935f68d39efa9fce96ec97bafcc0c2ea6cab", size = 212270, upload-time = "2025-09-25T05:58:53.584Z" }, + { url = "https://files.pythonhosted.org/packages/dd/a5/92e53d4d67c127d3ed0e002b90e758a28b4dacb9d81da617c3bae28d2907/zope_interface-8.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:db263a60c728c86e6a74945f3f74cfe0ede252e726cf71e05a0c7aca8d9d5432", size = 207891, upload-time = "2025-11-10T07:58:53.189Z" }, + { url = "https://files.pythonhosted.org/packages/b3/76/a100cc050aa76df9bcf8bbd51000724465e2336fd4c786b5904c6c6dfc55/zope_interface-8.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:cfa89e5b05b7a79ab34e368293ad008321231e321b3ce4430487407b4fe3450a", size = 208335, upload-time = "2025-11-10T07:58:54.232Z" }, + { url = "https://files.pythonhosted.org/packages/ab/ae/37c3e964c599c57323e02ca92a6bf81b4bc9848b88fb5eb3f6fc26320af2/zope_interface-8.1-cp311-cp311-manylinux1_i686.manylinux2014_i686.manylinux_2_17_i686.manylinux_2_5_i686.whl", hash = "sha256:87eaf011912a06ef86da70aba2ca0ddb68b8ab84a7d1da6b144a586b70a61bca", size = 255011, upload-time = "2025-11-10T07:58:30.304Z" }, + { url = "https://files.pythonhosted.org/packages/b6/9b/b693b6021d83177db2f5237fc3917921c7f497bac9a062eba422435ee172/zope_interface-8.1-cp311-cp311-manylinux1_x86_64.manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:10f06d128f1c181ded3af08c5004abcb3719c13a976ce9163124e7eeded6899a", size = 259780, upload-time = "2025-11-10T07:58:33.306Z" }, + { url = "https://files.pythonhosted.org/packages/c3/e2/0d1783563892ad46fedd0b1369e8d60ff8fcec0cd6859ab2d07e36f4f0ff/zope_interface-8.1-cp311-cp311-win_amd64.whl", hash = "sha256:17fb5382a4b9bd2ea05648a457c583e5a69f0bfa3076ed1963d48bc42a2da81f", size = 212143, upload-time = "2025-11-10T07:59:56.744Z" }, + { url = "https://files.pythonhosted.org/packages/02/6f/0bfb2beb373b7ca1c3d12807678f20bac1a07f62892f84305a1b544da785/zope_interface-8.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:a8aee385282ab2a9813171b15f41317e22ab0a96cf05e9e9e16b29f4af8b6feb", size = 208596, upload-time = "2025-11-10T07:58:09.945Z" }, + { url = "https://files.pythonhosted.org/packages/49/50/169981a42812a2e21bc33fb48640ad01a790ed93c179a9854fe66f901641/zope_interface-8.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:af651a87f950a13e45fd49510111f582717fb106a63d6a0c2d3ba86b29734f07", size = 208787, upload-time = "2025-11-10T07:58:11.4Z" }, + { url = "https://files.pythonhosted.org/packages/f8/fb/cb9cb9591a7c78d0878b280b3d3cea42ec17c69c2219b655521b9daa36e8/zope_interface-8.1-cp312-cp312-manylinux1_i686.manylinux2014_i686.manylinux_2_17_i686.manylinux_2_5_i686.whl", hash = "sha256:80ed7683cf337f3b295e4b96153e2e87f12595c218323dc237c0147a6cc9da26", size = 259224, upload-time = "2025-11-10T07:58:31.882Z" }, + { url = "https://files.pythonhosted.org/packages/18/28/aa89afcefbb93b26934bb5cf030774804b267de2d9300f8bd8e0c6f20bc4/zope_interface-8.1-cp312-cp312-manylinux1_x86_64.manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:fb9a7a45944b28c16d25df7a91bf2b9bdb919fa2b9e11782366a1e737d266ec1", size = 264671, upload-time = "2025-11-10T07:58:36.283Z" }, + { url = "https://files.pythonhosted.org/packages/de/7a/9cea2b9e64d74f27484c59b9a42d6854506673eb0b90c3b8cd088f652d5b/zope_interface-8.1-cp312-cp312-win_amd64.whl", hash = "sha256:fc5e120e3618741714c474b2427d08d36bd292855208b4397e325bd50d81439d", size = 212257, upload-time = "2025-11-10T07:59:54.691Z" }, ] [[package]] diff --git a/dev/basedpyright-check b/dev/basedpyright-check index 1c87b27d6f..1b3d1df7ad 100755 --- a/dev/basedpyright-check +++ b/dev/basedpyright-check @@ -8,9 +8,14 @@ cd "$SCRIPT_DIR/.." # Get the path argument if provided PATH_TO_CHECK="$1" -# run basedpyright checks -if [ -n "$PATH_TO_CHECK" ]; then - uv run --directory api --dev -- basedpyright --threads $(nproc) "$PATH_TO_CHECK" -else - uv run --directory api --dev -- basedpyright --threads $(nproc) -fi +# Determine CPU core count based on OS +CPU_CORES=$( + if [[ "$(uname -s)" == "Darwin" ]]; then + sysctl -n hw.ncpu 2>/dev/null + else + nproc + fi +) + +# Run basedpyright checks +uv run --directory api --dev -- basedpyright --threads "$CPU_CORES" $PATH_TO_CHECK diff --git a/dev/start-beat b/dev/start-beat new file mode 100755 index 0000000000..e417874b25 --- /dev/null +++ b/dev/start-beat @@ -0,0 +1,60 @@ +#!/bin/bash + +set -x + +# Help function +show_help() { + echo "Usage: $0 [OPTIONS]" + echo "" + echo "Options:" + echo " --loglevel LEVEL Log level (default: INFO)" + echo " --scheduler SCHEDULER Scheduler class (default: celery.beat:PersistentScheduler)" + echo " -h, --help Show this help message" + echo "" + echo "Examples:" + echo " $0" + echo " $0 --loglevel DEBUG" + echo " $0 --scheduler django_celery_beat.schedulers:DatabaseScheduler" + echo "" + echo "Description:" + echo " Starts Celery Beat scheduler for periodic task execution." + echo " Beat sends scheduled tasks to worker queues at specified intervals." +} + +# Parse command line arguments +LOGLEVEL="INFO" +SCHEDULER="celery.beat:PersistentScheduler" + +while [[ $# -gt 0 ]]; do + case $1 in + --loglevel) + LOGLEVEL="$2" + shift 2 + ;; + --scheduler) + SCHEDULER="$2" + shift 2 + ;; + -h|--help) + show_help + exit 0 + ;; + *) + echo "Unknown option: $1" + show_help + exit 1 + ;; + esac +done + +SCRIPT_DIR="$(dirname "$(realpath "$0")")" +cd "$SCRIPT_DIR/.." + +echo "Starting Celery Beat with:" +echo " Log Level: ${LOGLEVEL}" +echo " Scheduler: ${SCHEDULER}" + +uv --directory api run \ + celery -A app.celery beat \ + --loglevel ${LOGLEVEL} \ + --scheduler ${SCHEDULER} \ No newline at end of file diff --git a/dev/start-web b/dev/start-web new file mode 100755 index 0000000000..dc06d6a59f --- /dev/null +++ b/dev/start-web @@ -0,0 +1,8 @@ +#!/bin/bash + +set -x + +SCRIPT_DIR="$(dirname "$(realpath "$0")")" +cd "$SCRIPT_DIR/../web" + +pnpm install && pnpm build && pnpm start diff --git a/dev/start-worker b/dev/start-worker index a7f16b853f..b1e010975b 100755 --- a/dev/start-worker +++ b/dev/start-worker @@ -2,9 +2,106 @@ set -x +# Help function +show_help() { + echo "Usage: $0 [OPTIONS]" + echo "" + echo "Options:" + echo " -q, --queues QUEUES Comma-separated list of queues to process" + echo " -c, --concurrency NUM Number of worker processes (default: 1)" + echo " -P, --pool POOL Pool implementation (default: gevent)" + echo " --loglevel LEVEL Log level (default: INFO)" + echo " -h, --help Show this help message" + echo "" + echo "Examples:" + echo " $0 --queues dataset,workflow" + echo " $0 --queues workflow_professional,workflow_team --concurrency 4" + echo " $0 --queues dataset --concurrency 2 --pool prefork" + echo "" + echo "Available queues:" + echo " dataset - RAG indexing and document processing" + echo " workflow - Workflow triggers (community edition)" + echo " workflow_professional - Professional tier workflows (cloud edition)" + echo " workflow_team - Team tier workflows (cloud edition)" + echo " workflow_sandbox - Sandbox tier workflows (cloud edition)" + echo " schedule_poller - Schedule polling tasks" + echo " schedule_executor - Schedule execution tasks" + echo " mail - Email notifications" + echo " ops_trace - Operations tracing" + echo " app_deletion - Application cleanup" + echo " plugin - Plugin operations" + echo " workflow_storage - Workflow storage tasks" + echo " conversation - Conversation tasks" + echo " priority_pipeline - High priority pipeline tasks" + echo " pipeline - Standard pipeline tasks" + echo " triggered_workflow_dispatcher - Trigger dispatcher tasks" + echo " trigger_refresh_executor - Trigger refresh tasks" +} + +# Parse command line arguments +QUEUES="" +CONCURRENCY=1 +POOL="gevent" +LOGLEVEL="INFO" + +while [[ $# -gt 0 ]]; do + case $1 in + -q|--queues) + QUEUES="$2" + shift 2 + ;; + -c|--concurrency) + CONCURRENCY="$2" + shift 2 + ;; + -P|--pool) + POOL="$2" + shift 2 + ;; + --loglevel) + LOGLEVEL="$2" + shift 2 + ;; + -h|--help) + show_help + exit 0 + ;; + *) + echo "Unknown option: $1" + show_help + exit 1 + ;; + esac +done + SCRIPT_DIR="$(dirname "$(realpath "$0")")" cd "$SCRIPT_DIR/.." +# If no queues specified, use edition-based defaults +if [[ -z "${QUEUES}" ]]; then + # Get EDITION from environment, default to SELF_HOSTED (community edition) + EDITION=${EDITION:-"SELF_HOSTED"} + + # Configure queues based on edition + if [[ "${EDITION}" == "CLOUD" ]]; then + # Cloud edition: separate queues for dataset and trigger tasks + QUEUES="dataset,priority_dataset,priority_pipeline,pipeline,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,workflow_professional,workflow_team,workflow_sandbox,schedule_poller,schedule_executor,triggered_workflow_dispatcher,trigger_refresh_executor" + else + # Community edition (SELF_HOSTED): dataset and workflow have separate queues + QUEUES="dataset,priority_dataset,priority_pipeline,pipeline,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,workflow,schedule_poller,schedule_executor,triggered_workflow_dispatcher,trigger_refresh_executor" + fi + + echo "No queues specified, using edition-based defaults: ${QUEUES}" +else + echo "Using specified queues: ${QUEUES}" +fi + +echo "Starting Celery worker with:" +echo " Queues: ${QUEUES}" +echo " Concurrency: ${CONCURRENCY}" +echo " Pool: ${POOL}" +echo " Log Level: ${LOGLEVEL}" + uv --directory api run \ - celery -A app.celery worker \ - -P gevent -c 1 --loglevel INFO -Q dataset,generation,mail,ops_trace,app_deletion,plugin,workflow_storage,conversation,priority_pipeline,pipeline + celery -A app.celery worker \ + -P ${POOL} -c ${CONCURRENCY} --loglevel ${LOGLEVEL} -Q ${QUEUES} diff --git a/docker/.env.example b/docker/.env.example index ca580dcb79..519f4aa3e0 100644 --- a/docker/.env.example +++ b/docker/.env.example @@ -24,6 +24,11 @@ CONSOLE_WEB_URL= # Example: https://api.dify.ai SERVICE_API_URL= +# Trigger external URL +# used to display trigger endpoint API Base URL to the front-end. +# Example: https://api.dify.ai +TRIGGER_URL=http://localhost + # WebApp API backend Url, # used to declare the back-end URL for the front-end API. # If empty, it is the same domain. @@ -149,6 +154,12 @@ DIFY_PORT=5001 SERVER_WORKER_AMOUNT=1 # Defaults to gevent. If using windows, it can be switched to sync or solo. +# +# Warning: Changing this parameter requires disabling patching for +# psycopg2 and gRPC (see `gunicorn.conf.py` and `celery_entrypoint.py`). +# Modifying it may also decrease throughput. +# +# It is strongly discouraged to change this parameter. SERVER_WORKER_CLASS=gevent # Default number of worker connections, the default is 10. @@ -156,6 +167,12 @@ SERVER_WORKER_CONNECTIONS=10 # Similar to SERVER_WORKER_CLASS. # If using windows, it can be switched to sync or solo. +# +# Warning: Changing this parameter requires disabling patching for +# psycopg2 and gRPC (see `gunicorn_conf.py` and `celery_entrypoint.py`). +# Modifying it may also decrease throughput. +# +# It is strongly discouraged to change this parameter. CELERY_WORKER_CLASS= # Request handling timeout. The default is 200, @@ -201,6 +218,10 @@ ENABLE_WEBSITE_JINAREADER=true ENABLE_WEBSITE_FIRECRAWL=true ENABLE_WEBSITE_WATERCRAWL=true +# Enable inline LaTeX rendering with single dollar signs ($...$) in the web frontend +# Default is false for security reasons to prevent conflicts with regular text +NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX=false + # ------------------------------ # Database Configuration # The database uses PostgreSQL. Please use the public schema. @@ -260,16 +281,18 @@ POSTGRES_MAINTENANCE_WORK_MEM=64MB POSTGRES_EFFECTIVE_CACHE_SIZE=4096MB # Sets the maximum allowed duration of any statement before termination. -# Default is 60000 milliseconds. +# Default is 0 (no timeout). # # Reference: https://www.postgresql.org/docs/current/runtime-config-client.html#GUC-STATEMENT-TIMEOUT -POSTGRES_STATEMENT_TIMEOUT=60000 +# A value of 0 prevents the server from timing out statements. +POSTGRES_STATEMENT_TIMEOUT=0 # Sets the maximum allowed duration of any idle in-transaction session before termination. -# Default is 60000 milliseconds. +# Default is 0 (no timeout). # # Reference: https://www.postgresql.org/docs/current/runtime-config-client.html#GUC-IDLE-IN-TRANSACTION-SESSION-TIMEOUT -POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT=60000 +# A value of 0 prevents the server from terminating idle sessions. +POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT=0 # ------------------------------ # Redis Configuration @@ -314,7 +337,7 @@ REDIS_CLUSTERS_PASSWORD= # Celery Configuration # ------------------------------ -# Use standalone redis as the broker, and redis db 1 for celery broker. (redis_username is usually set by defualt as empty) +# Use standalone redis as the broker, and redis db 1 for celery broker. (redis_username is usually set by default as empty) # Format as follows: `redis://:@:/`. # Example: redis://:difyai123456@redis:6379/1 # If use Redis Sentinel, format as follows: `sentinel://:@:/` @@ -342,6 +365,11 @@ WEB_API_CORS_ALLOW_ORIGINS=* # Specifies the allowed origins for cross-origin requests to the console API, # e.g. https://cloud.dify.ai or * for all origins. CONSOLE_CORS_ALLOW_ORIGINS=* +# Set COOKIE_DOMAIN when the console frontend and API are on different subdomains. +# Provide the registrable domain (e.g. example.com); leading dots are optional. +COOKIE_DOMAIN= +# The frontend reads NEXT_PUBLIC_COOKIE_DOMAIN to align cookie handling with the API. +NEXT_PUBLIC_COOKIE_DOMAIN= # ------------------------------ # File Storage Configuration @@ -469,6 +497,7 @@ VECTOR_INDEX_NAME_PREFIX=Vector_index # The Weaviate endpoint URL. Only available when VECTOR_STORE is `weaviate`. WEAVIATE_ENDPOINT=http://weaviate:8080 WEAVIATE_API_KEY=WVF5YThaHlkYwhGUSmCRgsX3tD5ngdN8pkih +WEAVIATE_GRPC_ENDPOINT=grpc://weaviate:50051 # The Qdrant endpoint URL. Only available when VECTOR_STORE is `qdrant`. QDRANT_URL=http://qdrant:6333 @@ -739,6 +768,12 @@ UPLOAD_FILE_SIZE_LIMIT=15 # The maximum number of files that can be uploaded at a time, default 5. UPLOAD_FILE_BATCH_LIMIT=5 +# Comma-separated list of file extensions blocked from upload for security reasons. +# Extensions should be lowercase without dots (e.g., exe,bat,sh,dll). +# Empty by default to allow all file types. +# Recommended: exe,bat,cmd,com,scr,vbs,ps1,msi,dll +UPLOAD_FILE_EXTENSION_BLACKLIST= + # ETL type, support: `dify`, `Unstructured` # `dify` Dify's proprietary file extraction scheme # `Unstructured` Unstructured.io file extraction scheme @@ -968,6 +1003,9 @@ HTTP_REQUEST_MAX_WRITE_TIMEOUT=600 # Base64 encoded client private key data for mutual TLS authentication (PEM format, optional) # HTTP_REQUEST_NODE_SSL_CLIENT_KEY_DATA=LS0tLS1CRUdJTi... +# Webhook request configuration +WEBHOOK_REQUEST_BODY_MAX_SIZE=10485760 + # Respect X-* headers to redirect clients RESPECT_XFORWARD_HEADERS_ENABLED=false @@ -1244,6 +1282,7 @@ MARKETPLACE_ENABLED=true MARKETPLACE_API_URL=https://marketplace.dify.ai FORCE_VERIFYING_SIGNATURE=true +ENFORCE_LANGGENIUS_PLUGIN_SIGNATURES=true PLUGIN_STDIO_BUFFER_SIZE=1024 PLUGIN_STDIO_MAX_BUFFER_SIZE=5242880 @@ -1327,6 +1366,9 @@ SWAGGER_UI_PATH=/swagger-ui.html # Set to false to export dataset IDs as plain text for easier cross-environment import DSL_EXPORT_ENCRYPT_DATASET_ID=true +# Maximum number of segments for dataset segments API (0 for unlimited) +DATASET_MAX_SEGMENTS_PER_REQUEST=0 + # Celery schedule tasks configuration ENABLE_CLEAN_EMBEDDING_CACHE_TASK=false ENABLE_CLEAN_UNUSED_DATASETS_TASK=false @@ -1336,3 +1378,10 @@ ENABLE_CLEAN_MESSAGES=false ENABLE_MAIL_CLEAN_DOCUMENT_NOTIFY_TASK=false ENABLE_DATASETS_QUEUE_MONITOR=false ENABLE_CHECK_UPGRADABLE_PLUGIN_TASK=true +ENABLE_WORKFLOW_SCHEDULE_POLLER_TASK=true +WORKFLOW_SCHEDULE_POLLER_INTERVAL=1 +WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE=100 +WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK=0 + +# Tenant isolated task queue configuration +TENANT_ISOLATED_TASK_CONCURRENCY=1 diff --git a/docker/docker-compose-template.yaml b/docker/docker-compose-template.yaml index 9650be90db..e01437689d 100644 --- a/docker/docker-compose-template.yaml +++ b/docker/docker-compose-template.yaml @@ -2,7 +2,7 @@ x-shared-env: &shared-api-worker-env services: # API service api: - image: langgenius/dify-api:1.9.2 + image: langgenius/dify-api:1.10.0 restart: always environment: # Use the shared environment variables. @@ -29,14 +29,14 @@ services: - default # worker service - # The Celery worker for processing the queue. + # The Celery worker for processing all queues (dataset, workflow, mail, etc.) worker: - image: langgenius/dify-api:1.9.2 + image: langgenius/dify-api:1.10.0 restart: always environment: # Use the shared environment variables. <<: *shared-api-worker-env - # Startup mode, 'worker' starts the Celery worker for processing the queue. + # Startup mode, 'worker' starts the Celery worker for processing all queues. MODE: worker SENTRY_DSN: ${API_SENTRY_DSN:-} SENTRY_TRACES_SAMPLE_RATE: ${API_SENTRY_TRACES_SAMPLE_RATE:-1.0} @@ -58,7 +58,7 @@ services: # worker_beat service # Celery beat for scheduling periodic tasks. worker_beat: - image: langgenius/dify-api:1.9.2 + image: langgenius/dify-api:1.10.0 restart: always environment: # Use the shared environment variables. @@ -76,11 +76,12 @@ services: # Frontend web application. web: - image: langgenius/dify-web:1.9.2 + image: langgenius/dify-web:1.10.0 restart: always environment: CONSOLE_API_URL: ${CONSOLE_API_URL:-} APP_API_URL: ${APP_API_URL:-} + NEXT_PUBLIC_COOKIE_DOMAIN: ${NEXT_PUBLIC_COOKIE_DOMAIN:-} SENTRY_DSN: ${WEB_SENTRY_DSN:-} NEXT_TELEMETRY_DISABLED: ${NEXT_TELEMETRY_DISABLED:-0} TEXT_GENERATION_TIMEOUT_MS: ${TEXT_GENERATION_TIMEOUT_MS:-60000} @@ -100,6 +101,8 @@ services: ENABLE_WEBSITE_JINAREADER: ${ENABLE_WEBSITE_JINAREADER:-true} ENABLE_WEBSITE_FIRECRAWL: ${ENABLE_WEBSITE_FIRECRAWL:-true} ENABLE_WEBSITE_WATERCRAWL: ${ENABLE_WEBSITE_WATERCRAWL:-true} + NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX: ${NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX:-false} + # The postgres database. db: image: postgres:15-alpine @@ -115,8 +118,8 @@ services: -c 'work_mem=${POSTGRES_WORK_MEM:-4MB}' -c 'maintenance_work_mem=${POSTGRES_MAINTENANCE_WORK_MEM:-64MB}' -c 'effective_cache_size=${POSTGRES_EFFECTIVE_CACHE_SIZE:-4096MB}' - -c 'statement_timeout=${POSTGRES_STATEMENT_TIMEOUT:-60000}' - -c 'idle_in_transaction_session_timeout=${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-60000}' + -c 'statement_timeout=${POSTGRES_STATEMENT_TIMEOUT:-0}' + -c 'idle_in_transaction_session_timeout=${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-0}' volumes: - ./volumes/db/data:/var/lib/postgresql/data healthcheck: @@ -179,7 +182,7 @@ services: # plugin daemon plugin_daemon: - image: langgenius/dify-plugin-daemon:0.3.3-local + image: langgenius/dify-plugin-daemon:0.4.1-local restart: always environment: # Use the shared environment variables. diff --git a/docker/docker-compose.middleware.yaml b/docker/docker-compose.middleware.yaml index 9a1b9b53ba..b93457f8dc 100644 --- a/docker/docker-compose.middleware.yaml +++ b/docker/docker-compose.middleware.yaml @@ -15,8 +15,8 @@ services: -c 'work_mem=${POSTGRES_WORK_MEM:-4MB}' -c 'maintenance_work_mem=${POSTGRES_MAINTENANCE_WORK_MEM:-64MB}' -c 'effective_cache_size=${POSTGRES_EFFECTIVE_CACHE_SIZE:-4096MB}' - -c 'statement_timeout=${POSTGRES_STATEMENT_TIMEOUT:-60000}' - -c 'idle_in_transaction_session_timeout=${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-60000}' + -c 'statement_timeout=${POSTGRES_STATEMENT_TIMEOUT:-0}' + -c 'idle_in_transaction_session_timeout=${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-0}' volumes: - ${PGDATA_HOST_VOLUME:-./volumes/db/data}:/var/lib/postgresql/data ports: @@ -87,7 +87,7 @@ services: # plugin daemon plugin_daemon: - image: langgenius/dify-plugin-daemon:0.3.3-local + image: langgenius/dify-plugin-daemon:0.4.0-local restart: always env_file: - ./middleware.env diff --git a/docker/docker-compose.yaml b/docker/docker-compose.yaml index d2ca6b859e..0117ebce3f 100644 --- a/docker/docker-compose.yaml +++ b/docker/docker-compose.yaml @@ -8,6 +8,7 @@ x-shared-env: &shared-api-worker-env CONSOLE_API_URL: ${CONSOLE_API_URL:-} CONSOLE_WEB_URL: ${CONSOLE_WEB_URL:-} SERVICE_API_URL: ${SERVICE_API_URL:-} + TRIGGER_URL: ${TRIGGER_URL:-http://localhost} APP_API_URL: ${APP_API_URL:-} APP_WEB_URL: ${APP_WEB_URL:-} FILES_URL: ${FILES_URL:-} @@ -51,6 +52,7 @@ x-shared-env: &shared-api-worker-env ENABLE_WEBSITE_JINAREADER: ${ENABLE_WEBSITE_JINAREADER:-true} ENABLE_WEBSITE_FIRECRAWL: ${ENABLE_WEBSITE_FIRECRAWL:-true} ENABLE_WEBSITE_WATERCRAWL: ${ENABLE_WEBSITE_WATERCRAWL:-true} + NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX: ${NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX:-false} DB_USERNAME: ${DB_USERNAME:-postgres} DB_PASSWORD: ${DB_PASSWORD:-difyai123456} DB_HOST: ${DB_HOST:-db} @@ -68,8 +70,8 @@ x-shared-env: &shared-api-worker-env POSTGRES_WORK_MEM: ${POSTGRES_WORK_MEM:-4MB} POSTGRES_MAINTENANCE_WORK_MEM: ${POSTGRES_MAINTENANCE_WORK_MEM:-64MB} POSTGRES_EFFECTIVE_CACHE_SIZE: ${POSTGRES_EFFECTIVE_CACHE_SIZE:-4096MB} - POSTGRES_STATEMENT_TIMEOUT: ${POSTGRES_STATEMENT_TIMEOUT:-60000} - POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT: ${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-60000} + POSTGRES_STATEMENT_TIMEOUT: ${POSTGRES_STATEMENT_TIMEOUT:-0} + POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT: ${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-0} REDIS_HOST: ${REDIS_HOST:-redis} REDIS_PORT: ${REDIS_PORT:-6379} REDIS_USERNAME: ${REDIS_USERNAME:-} @@ -98,6 +100,8 @@ x-shared-env: &shared-api-worker-env CELERY_SENTINEL_SOCKET_TIMEOUT: ${CELERY_SENTINEL_SOCKET_TIMEOUT:-0.1} WEB_API_CORS_ALLOW_ORIGINS: ${WEB_API_CORS_ALLOW_ORIGINS:-*} CONSOLE_CORS_ALLOW_ORIGINS: ${CONSOLE_CORS_ALLOW_ORIGINS:-*} + COOKIE_DOMAIN: ${COOKIE_DOMAIN:-} + NEXT_PUBLIC_COOKIE_DOMAIN: ${NEXT_PUBLIC_COOKIE_DOMAIN:-} STORAGE_TYPE: ${STORAGE_TYPE:-opendal} OPENDAL_SCHEME: ${OPENDAL_SCHEME:-fs} OPENDAL_FS_ROOT: ${OPENDAL_FS_ROOT:-storage} @@ -154,6 +158,7 @@ x-shared-env: &shared-api-worker-env VECTOR_INDEX_NAME_PREFIX: ${VECTOR_INDEX_NAME_PREFIX:-Vector_index} WEAVIATE_ENDPOINT: ${WEAVIATE_ENDPOINT:-http://weaviate:8080} WEAVIATE_API_KEY: ${WEAVIATE_API_KEY:-WVF5YThaHlkYwhGUSmCRgsX3tD5ngdN8pkih} + WEAVIATE_GRPC_ENDPOINT: ${WEAVIATE_GRPC_ENDPOINT:-grpc://weaviate:50051} QDRANT_URL: ${QDRANT_URL:-http://qdrant:6333} QDRANT_API_KEY: ${QDRANT_API_KEY:-difyai123456} QDRANT_CLIENT_TIMEOUT: ${QDRANT_CLIENT_TIMEOUT:-20} @@ -350,6 +355,7 @@ x-shared-env: &shared-api-worker-env CLICKZETTA_VECTOR_DISTANCE_FUNCTION: ${CLICKZETTA_VECTOR_DISTANCE_FUNCTION:-cosine_distance} UPLOAD_FILE_SIZE_LIMIT: ${UPLOAD_FILE_SIZE_LIMIT:-15} UPLOAD_FILE_BATCH_LIMIT: ${UPLOAD_FILE_BATCH_LIMIT:-5} + UPLOAD_FILE_EXTENSION_BLACKLIST: ${UPLOAD_FILE_EXTENSION_BLACKLIST:-} ETL_TYPE: ${ETL_TYPE:-dify} UNSTRUCTURED_API_URL: ${UNSTRUCTURED_API_URL:-} UNSTRUCTURED_API_KEY: ${UNSTRUCTURED_API_KEY:-} @@ -430,6 +436,7 @@ x-shared-env: &shared-api-worker-env HTTP_REQUEST_MAX_CONNECT_TIMEOUT: ${HTTP_REQUEST_MAX_CONNECT_TIMEOUT:-10} HTTP_REQUEST_MAX_READ_TIMEOUT: ${HTTP_REQUEST_MAX_READ_TIMEOUT:-600} HTTP_REQUEST_MAX_WRITE_TIMEOUT: ${HTTP_REQUEST_MAX_WRITE_TIMEOUT:-600} + WEBHOOK_REQUEST_BODY_MAX_SIZE: ${WEBHOOK_REQUEST_BODY_MAX_SIZE:-10485760} RESPECT_XFORWARD_HEADERS_ENABLED: ${RESPECT_XFORWARD_HEADERS_ENABLED:-false} SSRF_PROXY_HTTP_URL: ${SSRF_PROXY_HTTP_URL:-http://ssrf_proxy:3128} SSRF_PROXY_HTTPS_URL: ${SSRF_PROXY_HTTPS_URL:-http://ssrf_proxy:3128} @@ -544,6 +551,7 @@ x-shared-env: &shared-api-worker-env MARKETPLACE_ENABLED: ${MARKETPLACE_ENABLED:-true} MARKETPLACE_API_URL: ${MARKETPLACE_API_URL:-https://marketplace.dify.ai} FORCE_VERIFYING_SIGNATURE: ${FORCE_VERIFYING_SIGNATURE:-true} + ENFORCE_LANGGENIUS_PLUGIN_SIGNATURES: ${ENFORCE_LANGGENIUS_PLUGIN_SIGNATURES:-true} PLUGIN_STDIO_BUFFER_SIZE: ${PLUGIN_STDIO_BUFFER_SIZE:-1024} PLUGIN_STDIO_MAX_BUFFER_SIZE: ${PLUGIN_STDIO_MAX_BUFFER_SIZE:-5242880} PLUGIN_PYTHON_ENV_INIT_TIMEOUT: ${PLUGIN_PYTHON_ENV_INIT_TIMEOUT:-120} @@ -599,6 +607,7 @@ x-shared-env: &shared-api-worker-env SWAGGER_UI_ENABLED: ${SWAGGER_UI_ENABLED:-true} SWAGGER_UI_PATH: ${SWAGGER_UI_PATH:-/swagger-ui.html} DSL_EXPORT_ENCRYPT_DATASET_ID: ${DSL_EXPORT_ENCRYPT_DATASET_ID:-true} + DATASET_MAX_SEGMENTS_PER_REQUEST: ${DATASET_MAX_SEGMENTS_PER_REQUEST:-0} ENABLE_CLEAN_EMBEDDING_CACHE_TASK: ${ENABLE_CLEAN_EMBEDDING_CACHE_TASK:-false} ENABLE_CLEAN_UNUSED_DATASETS_TASK: ${ENABLE_CLEAN_UNUSED_DATASETS_TASK:-false} ENABLE_CREATE_TIDB_SERVERLESS_TASK: ${ENABLE_CREATE_TIDB_SERVERLESS_TASK:-false} @@ -607,11 +616,16 @@ x-shared-env: &shared-api-worker-env ENABLE_MAIL_CLEAN_DOCUMENT_NOTIFY_TASK: ${ENABLE_MAIL_CLEAN_DOCUMENT_NOTIFY_TASK:-false} ENABLE_DATASETS_QUEUE_MONITOR: ${ENABLE_DATASETS_QUEUE_MONITOR:-false} ENABLE_CHECK_UPGRADABLE_PLUGIN_TASK: ${ENABLE_CHECK_UPGRADABLE_PLUGIN_TASK:-true} + ENABLE_WORKFLOW_SCHEDULE_POLLER_TASK: ${ENABLE_WORKFLOW_SCHEDULE_POLLER_TASK:-true} + WORKFLOW_SCHEDULE_POLLER_INTERVAL: ${WORKFLOW_SCHEDULE_POLLER_INTERVAL:-1} + WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE: ${WORKFLOW_SCHEDULE_POLLER_BATCH_SIZE:-100} + WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK: ${WORKFLOW_SCHEDULE_MAX_DISPATCH_PER_TICK:-0} + TENANT_ISOLATED_TASK_CONCURRENCY: ${TENANT_ISOLATED_TASK_CONCURRENCY:-1} services: # API service api: - image: langgenius/dify-api:1.9.2 + image: langgenius/dify-api:1.10.0 restart: always environment: # Use the shared environment variables. @@ -638,14 +652,14 @@ services: - default # worker service - # The Celery worker for processing the queue. + # The Celery worker for processing all queues (dataset, workflow, mail, etc.) worker: - image: langgenius/dify-api:1.9.2 + image: langgenius/dify-api:1.10.0 restart: always environment: # Use the shared environment variables. <<: *shared-api-worker-env - # Startup mode, 'worker' starts the Celery worker for processing the queue. + # Startup mode, 'worker' starts the Celery worker for processing all queues. MODE: worker SENTRY_DSN: ${API_SENTRY_DSN:-} SENTRY_TRACES_SAMPLE_RATE: ${API_SENTRY_TRACES_SAMPLE_RATE:-1.0} @@ -667,7 +681,7 @@ services: # worker_beat service # Celery beat for scheduling periodic tasks. worker_beat: - image: langgenius/dify-api:1.9.2 + image: langgenius/dify-api:1.10.0 restart: always environment: # Use the shared environment variables. @@ -685,11 +699,12 @@ services: # Frontend web application. web: - image: langgenius/dify-web:1.9.2 + image: langgenius/dify-web:1.10.0 restart: always environment: CONSOLE_API_URL: ${CONSOLE_API_URL:-} APP_API_URL: ${APP_API_URL:-} + NEXT_PUBLIC_COOKIE_DOMAIN: ${NEXT_PUBLIC_COOKIE_DOMAIN:-} SENTRY_DSN: ${WEB_SENTRY_DSN:-} NEXT_TELEMETRY_DISABLED: ${NEXT_TELEMETRY_DISABLED:-0} TEXT_GENERATION_TIMEOUT_MS: ${TEXT_GENERATION_TIMEOUT_MS:-60000} @@ -709,6 +724,8 @@ services: ENABLE_WEBSITE_JINAREADER: ${ENABLE_WEBSITE_JINAREADER:-true} ENABLE_WEBSITE_FIRECRAWL: ${ENABLE_WEBSITE_FIRECRAWL:-true} ENABLE_WEBSITE_WATERCRAWL: ${ENABLE_WEBSITE_WATERCRAWL:-true} + NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX: ${NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX:-false} + # The postgres database. db: image: postgres:15-alpine @@ -724,8 +741,8 @@ services: -c 'work_mem=${POSTGRES_WORK_MEM:-4MB}' -c 'maintenance_work_mem=${POSTGRES_MAINTENANCE_WORK_MEM:-64MB}' -c 'effective_cache_size=${POSTGRES_EFFECTIVE_CACHE_SIZE:-4096MB}' - -c 'statement_timeout=${POSTGRES_STATEMENT_TIMEOUT:-60000}' - -c 'idle_in_transaction_session_timeout=${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-60000}' + -c 'statement_timeout=${POSTGRES_STATEMENT_TIMEOUT:-0}' + -c 'idle_in_transaction_session_timeout=${POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT:-0}' volumes: - ./volumes/db/data:/var/lib/postgresql/data healthcheck: @@ -788,7 +805,7 @@ services: # plugin daemon plugin_daemon: - image: langgenius/dify-plugin-daemon:0.3.3-local + image: langgenius/dify-plugin-daemon:0.4.1-local restart: always environment: # Use the shared environment variables. diff --git a/docker/middleware.env.example b/docker/middleware.env.example index c9bb8c0528..24629c2d89 100644 --- a/docker/middleware.env.example +++ b/docker/middleware.env.example @@ -41,16 +41,18 @@ POSTGRES_MAINTENANCE_WORK_MEM=64MB POSTGRES_EFFECTIVE_CACHE_SIZE=4096MB # Sets the maximum allowed duration of any statement before termination. -# Default is 60000 milliseconds. +# Default is 0 (no timeout). # # Reference: https://www.postgresql.org/docs/current/runtime-config-client.html#GUC-STATEMENT-TIMEOUT -POSTGRES_STATEMENT_TIMEOUT=60000 +# A value of 0 prevents the server from timing out statements. +POSTGRES_STATEMENT_TIMEOUT=0 # Sets the maximum allowed duration of any idle in-transaction session before termination. -# Default is 60000 milliseconds. +# Default is 0 (no timeout). # # Reference: https://www.postgresql.org/docs/current/runtime-config-client.html#GUC-IDLE-IN-TRANSACTION-SESSION-TIMEOUT -POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT=60000 +# A value of 0 prevents the server from terminating idle sessions. +POSTGRES_IDLE_IN_TRANSACTION_SESSION_TIMEOUT=0 # ----------------------------- # Environment Variables for redis Service diff --git a/docker/nginx/conf.d/default.conf.template b/docker/nginx/conf.d/default.conf.template index 48d7da8cf5..1d63c1b97d 100644 --- a/docker/nginx/conf.d/default.conf.template +++ b/docker/nginx/conf.d/default.conf.template @@ -39,10 +39,17 @@ server { proxy_pass http://web:3000; include proxy.conf; } + location /mcp { proxy_pass http://api:5001; include proxy.conf; } + + location /triggers { + proxy_pass http://api:5001; + include proxy.conf; + } + # placeholder for acme challenge location ${ACME_CHALLENGE_LOCATION} diff --git a/docs/hi-IN/CONTRIBUTING.md b/docs/hi-IN/CONTRIBUTING.md new file mode 100644 index 0000000000..5c1ea4f8fd --- /dev/null +++ b/docs/hi-IN/CONTRIBUTING.md @@ -0,0 +1,101 @@ +# योगदान (CONTRIBUTING) + +तो आप Dify में योगदान देना चाहते हैं — यह शानदार है, हम उत्सुक हैं यह देखने के लिए कि आप क्या बनाते हैं। सीमित टीम और फंडिंग वाले एक स्टार्टअप के रूप में, हमारा बड़ा लक्ष्य LLM एप्लिकेशनों के निर्माण और प्रबंधन के लिए सबसे सहज वर्कफ़्लो डिज़ाइन करना है। समुदाय से मिलने वाली कोई भी मदद वास्तव में मायने रखती है। + +हमारे वर्तमान चरण को देखते हुए हमें तेज़ी से काम करना और डिलीवर करना होता है, लेकिन हम यह भी सुनिश्चित करना चाहते हैं कि आपके जैसे योगदानकर्ताओं के लिए योगदान देने का अनुभव यथासंभव सरल और सुगम हो।\ +इसी उद्देश्य से हमने यह योगदान गाइड तैयार किया है, ताकि आप कोडबेस से परिचित हो सकें और जान सकें कि हम योगदानकर्ताओं के साथ कैसे काम करते हैं — ताकि आप जल्दी से मज़ेदार हिस्से पर पहुँच सकें। + +यह गाइड, Dify की तरह ही, एक निरंतर विकसित होता दस्तावेज़ है। यदि यह कभी-कभी वास्तविक प्रोजेक्ट से पीछे रह जाए तो हम आपके समझ के लिए आभारी हैं, और सुधार के लिए किसी भी सुझाव का स्वागत करते हैं। + +लाइसेंसिंग के संदर्भ में, कृपया एक मिनट निकालकर हमारा छोटा [License and Contributor Agreement](../../LICENSE) पढ़ें।\ +समुदाय [code of conduct](https://github.com/langgenius/.github/blob/main/CODE_OF_CONDUCT.md) का भी पालन करता है। + +## शुरू करने से पहले + +कुछ योगदान करने की तलाश में हैं? हमारे [good first issues](https://github.com/langgenius/dify/issues?q=is%3Aissue%20state%3Aopen%20label%3A%22good%20first%20issue%22) ब्राउज़ करें और किसी एक को चुनकर शुरुआत करें! + +कोई नया मॉडल रनटाइम या टूल जोड़ना चाहते हैं? हमारे [plugin repo](https://github.com/langgenius/dify-plugins) में एक PR खोलें और हमें दिखाएँ कि आपने क्या बनाया है। + +किसी मौजूदा मॉडल रनटाइम या टूल को अपडेट करना है, या कुछ बग्स को ठीक करना है? हमारे [official plugin repo](https://github.com/langgenius/dify-official-plugins) पर जाएँ और अपना जादू दिखाएँ! + +मज़े में शामिल हों, योगदान दें, और चलिए मिलकर कुछ शानदार बनाते हैं! 💡✨ + +PR के विवरण में मौजूदा issue को लिंक करना या नया issue खोलना न भूलें। + +### बग रिपोर्ट (Bug reports) + +> [!IMPORTANT]\ +> कृपया बग रिपोर्ट सबमिट करते समय निम्नलिखित जानकारी अवश्य शामिल करें: + +- एक स्पष्ट और वर्णनात्मक शीर्षक +- बग का विस्तृत विवरण, जिसमें कोई भी त्रुटि संदेश (error messages) शामिल हो +- बग को पुन: उत्पन्न करने के चरण +- अपेक्षित व्यवहार +- **लॉग्स**, यदि उपलब्ध हों — बैकएंड समस्याओं के लिए यह बहुत महत्वपूर्ण है, आप इन्हें docker-compose logs में पा सकते हैं +- स्क्रीनशॉट या वीडियो (यदि लागू हो) + +हम प्राथमिकता कैसे तय करते हैं: + +| समस्या प्रकार (Issue Type) | प्राथमिकता (Priority) | +| ------------------------------------------------------------ | --------------- | +| मुख्य कार्यों में बग (क्लाउड सेवा, लॉगिन न होना, एप्लिकेशन न चलना, सुरक्षा खामियाँ) | गंभीर (Critical) | +| गैर-गंभीर बग, प्रदर्शन सुधार | मध्यम प्राथमिकता (Medium Priority) | +| छोटे सुधार (टाइपो, भ्रमित करने वाला लेकिन काम करने वाला UI) | निम्न प्राथमिकता (Low Priority) | + +### फ़ीचर अनुरोध (Feature requests) + +> [!NOTE]\ +> कृपया फ़ीचर अनुरोध सबमिट करते समय निम्नलिखित जानकारी अवश्य शामिल करें: + +- एक स्पष्ट और वर्णनात्मक शीर्षक +- फ़ीचर का विस्तृत विवरण +- फ़ीचर के उपयोग का मामला (use case) +- फ़ीचर अनुरोध से संबंधित कोई अन्य संदर्भ या स्क्रीनशॉट + +हम प्राथमिकता कैसे तय करते हैं: + +| फ़ीचर प्रकार (Feature Type) | प्राथमिकता (Priority) | +| ------------------------------------------------------------ | --------------- | +| किसी टीम सदस्य द्वारा उच्च प्राथमिकता (High-Priority) के रूप में चिह्नित फ़ीचर | उच्च प्राथमिकता (High Priority) | +| हमारे [community feedback board](https://github.com/langgenius/dify/discussions/categories/feedbacks) से लोकप्रिय फ़ीचर अनुरोध | मध्यम प्राथमिकता (Medium Priority) | +| गैर-मुख्य फ़ीचर्स और छोटे सुधार | निम्न प्राथमिकता (Low Priority) | +| मूल्यवान लेकिन तात्कालिक नहीं | भविष्य का फ़ीचर (Future-Feature) | + +## अपना PR सबमिट करना (Submitting your PR) + +### पुल रिक्वेस्ट प्रक्रिया (Pull Request Process) + +1. रिपॉज़िटरी को Fork करें +1. PR ड्राफ्ट करने से पहले, कृपया अपने बदलावों पर चर्चा करने के लिए एक issue बनाएँ +1. अपने परिवर्तनों के लिए एक नई शाखा (branch) बनाएँ +1. अपने बदलावों के लिए उपयुक्त टेस्ट जोड़ें +1. सुनिश्चित करें कि आपका कोड मौजूदा टेस्ट पास करता है +1. PR विवरण में issue लिंक करें, जैसे: `fixes #` +1. मर्ज हो जाएँ! 🎉 + +### प्रोजेक्ट सेटअप करें (Setup the project) + +#### फ्रंटएंड (Frontend) + +फ्रंटएंड सेवा सेटअप करने के लिए, कृपया हमारी विस्तृत [guide](https://github.com/langgenius/dify/blob/main/web/README.md) देखें जो `web/README.md` फ़ाइल में उपलब्ध है।\ +यह दस्तावेज़ आपको फ्रंटएंड वातावरण को सही ढंग से सेटअप करने के लिए विस्तृत निर्देश प्रदान करता है। + +#### बैकएंड (Backend) + +बैकएंड सेवा सेटअप करने के लिए, कृपया हमारी विस्तृत [instructions](https://github.com/langgenius/dify/blob/main/api/README.md) देखें जो `api/README.md` फ़ाइल में दी गई हैं।\ +यह दस्तावेज़ चरण-दर-चरण मार्गदर्शन प्रदान करता है जिससे आप बैकएंड को सुचारू रूप से चला सकें। + +#### अन्य महत्वपूर्ण बातें (Other things to note) + +सेटअप शुरू करने से पहले इस दस्तावेज़ की सावधानीपूर्वक समीक्षा करने की अनुशंसा की जाती है, क्योंकि इसमें निम्नलिखित महत्वपूर्ण जानकारी शामिल है: + +- आवश्यक पूर्व-आवश्यकताएँ और निर्भरताएँ +- इंस्टॉलेशन चरण +- कॉन्फ़िगरेशन विवरण +- सामान्य समस्या निवारण सुझाव + +यदि सेटअप प्रक्रिया के दौरान आपको कोई समस्या आती है, तो बेझिझक हमसे संपर्क करें। + +## सहायता प्राप्त करना (Getting Help) + +यदि योगदान करते समय आप कहीं अटक जाएँ या कोई महत्वपूर्ण प्रश्न हो, तो संबंधित GitHub issue के माध्यम से हमें अपने प्रश्न भेजें, या त्वरित बातचीत के लिए हमारे [Discord](https://discord.gg/8Tpq4AcN9c) पर जुड़ें। diff --git a/docs/hi-IN/README.md b/docs/hi-IN/README.md new file mode 100644 index 0000000000..7c4fc70db0 --- /dev/null +++ b/docs/hi-IN/README.md @@ -0,0 +1,224 @@ +![cover-v5-optimized](../../images/GitHub_README_if.png) + +

+ 📌 Dify वर्कफ़्लो फ़ाइल अपलोड पेश है: Google NotebookLM पॉडकास्ट को पुनः बनाएँ +

+ +

+ Dify Cloud · + स्व-होस्टिंग · + दस्तावेज़ीकरण · + Dify संस्करण का अवलोकन +

+ +

+ + Static Badge + + Static Badge + + chat on Discord + + join Reddit + + follow on X(Twitter) + + follow on LinkedIn + + Docker Pulls + + Commits last month + + Issues closed + + Discussion posts +

+ +

+ README in English + 繁體中文文件 + 简体中文文件 + 日本語のREADME + README en Español + README en Français + README tlhIngan Hol + README in Korean + README بالعربية + Türkçe README + README Tiếng Việt + README in Deutsch + README in Italiano + README in বাংলা + README in हिन्दी +

+ +Dify एक मुक्त-स्रोत प्लेटफ़ॉर्म है जो LLM अनुप्रयोगों (एप्लिकेशनों) के विकास के लिए बनाया गया है। इसका सहज इंटरफ़ेस एजेंटिक एआई वर्कफ़्लो, RAG पाइपलाइनों, एजेंट क्षमताओं, मॉडल प्रबंधन, ऑब्ज़र्वेबिलिटी (निगरानी) सुविधाओं और अन्य को एक साथ जोड़ता है — जिससे आप प्रोटोटाइप से उत्पादन (प्रोडक्शन) तक जल्दी पहुँच सकते हैं। + +## त्वरित प्रारंभ + +> Dify स्थापित करने से पहले, सुनिश्चित करें कि आपकी मशीन निम्नलिखित न्यूनतम सिस्टम आवश्यकताओं को पूरा करती है: +> +> - CPU >= 2 Core +> - RAM >= 4 GiB + +
+ +Dify सर्वर शुरू करने का सबसे आसान तरीका [Docker Compose](../..docker/docker-compose.yaml) के माध्यम से है। नीचे दिए गए कमांड्स से Dify चलाने से पहले, सुनिश्चित करें कि आपकी मशीन पर [Docker] (https://docs.docker.com/get-docker/) और [Docker Compose] (https://docs.docker.com/compose/install/) इंस्टॉल हैं।: + +```bash +cd dify +cd docker +cp .env.example .env +docker compose up -d +``` + +रन करने के बाद, आप अपने ब्राउज़र में [http://localhost/install](http://localhost/install) पर Dify डैशबोर्ड एक्सेस कर सकते हैं और प्रारंभिक सेटअप प्रक्रिया शुरू कर सकते हैं। + +#### सहायता प्राप्त करना + +यदि आपको Dify सेटअप करते समय कोई समस्या आती है, तो कृपया हमारे [FAQ](https://docs.dify.ai/getting-started/install-self-hosted/faqs) को देखें। यदि फिर भी समस्या बनी रहती है, तो [the community and us](#community--contact) से संपर्क करें। + +> यदि आप Dify में योगदान देना चाहते हैं या अतिरिक्त विकास करना चाहते हैं, तो हमारे [guide to deploying from source code](https://docs.dify.ai/getting-started/install-self-hosted/local-source-code) को देखें। + +## मुख्य विशेषताएँ + +**1. वर्कफ़्लो**:\ +एक दृश्य कैनवास पर शक्तिशाली एआई वर्कफ़्लो बनाएं और परीक्षण करें, नीचे दी गई सभी सुविधाओं और उससे भी आगे का उपयोग करते हुए। + +**2. व्यापक मॉडल समर्थन**:\ +कई इन्फ़रेंस प्रदाताओं और स्व-होस्टेड समाधानों से सैकड़ों स्वामित्व / मुक्त-स्रोत LLMs के साथ सहज एकीकरण, जिसमें GPT, Mistral, Llama3, और कोई भी OpenAI API-संगत मॉडल शामिल हैं। समर्थित मॉडल प्रदाताओं की पूरी सूची [here](https://docs.dify.ai/getting-started/readme/model-providers) पर पाई जा सकती है। + +![providers-v5](https://github.com/langgenius/dify/assets/13230914/5a17bdbe-097a-4100-8363-40255b70f6e3) + +**3. प्रॉम्प्ट IDE**:\ +प्रॉम्प्ट बनाने, मॉडल प्रदर्शन की तुलना करने, और चैट-आधारित ऐप में टेक्स्ट-टू-स्पीच जैसी अतिरिक्त सुविधाएँ जोड़ने के लिए सहज इंटरफ़ेस। + +**4. RAG पाइपलाइन**:\ +विस्तृत RAG क्षमताएँ जो दस्तावेज़ इनजेशन से लेकर रिट्रीवल तक सब कुछ कवर करती हैं, और PDFs, PPTs, तथा अन्य सामान्य दस्तावेज़ प्रारूपों से टेक्स्ट निकालने के लिए आउट-ऑफ़-द-बॉक्स समर्थन प्रदान करती हैं। + +**5. एजेंट क्षमताएँ**:\ +आप LLM फ़ंक्शन कॉलिंग या ReAct के आधार पर एजेंट परिभाषित कर सकते हैं, और एजेंट के लिए पूर्व-निर्मित या कस्टम टूल जोड़ सकते हैं। Dify एआई एजेंटों के लिए 50+ अंतर्निर्मित टूल प्रदान करता है, जैसे Google Search, DALL·E, Stable Diffusion और WolframAlpha। + +**6. LLMOps**:\ +समय के साथ एप्लिकेशन लॉग्स और प्रदर्शन की निगरानी और विश्लेषण करें। आप उत्पादन डेटा और एनोटेशनों के आधार पर प्रॉम्प्ट्स, डेटासेट्स और मॉडल्स को निरंतर सुधार सकते हैं। + +**7. Backend-as-a-Service**:\ +Dify की सभी सेवाएँ संबंधित APIs के साथ आती हैं, जिससे आप Dify को आसानी से अपने व्यावसायिक लॉजिक में एकीकृत कर सकते हैं। + +## Dify का उपयोग करना + +- **Cloud
**\ + हम [Dify Cloud](https://dify.ai) सेवा प्रदान करते हैं, जिसे कोई भी बिना किसी सेटअप के आज़मा सकता है। यह स्व-परिनियोजित संस्करण की सभी क्षमताएँ प्रदान करता है और सैंडबॉक्स प्लान में 200 निःशुल्क GPT-4 कॉल्स शामिल करता है। + +- **Dify कम्युनिटी संस्करण की स्व-होस्टिंग
**\ + अपने वातावरण में Dify को जल्दी चलाएँ इस [starter guide](#quick-start) की मदद से।\ + आगे के संदर्भों और विस्तृत निर्देशों के लिए हमारी [documentation](https://docs.dify.ai) देखें। + +- **उद्यमों / संगठनों के लिए Dify
**\ + हम अतिरिक्त एंटरप्राइज़-केंद्रित सुविधाएँ प्रदान करते हैं।\ + [इस चैटबॉट के माध्यम से हमें अपने प्रश्न भेजें](https://udify.app/chat/22L1zSxg6yW1cWQg) या [हमें ईमेल भेजें](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) ताकि हम एंटरप्राइज़ आवश्यकताओं पर चर्चा कर सकें।
+ + > AWS का उपयोग करने वाले स्टार्टअप्स और छोटे व्यवसायों के लिए, [AWS Marketplace पर Dify Premium](https://aws.amazon.com/marketplace/pp/prodview-t22mebxzwjhu6) देखें और इसे एक क्लिक में अपने AWS VPC पर डिप्लॉय करें। यह एक किफायती AMI ऑफ़रिंग है, जो आपको कस्टम लोगो और ब्रांडिंग के साथ ऐप्स बनाने की अनुमति देती है। + +## आगे बने रहें + +GitHub पर Dify को स्टार करें और नए रिलीज़ की सूचना तुरंत प्राप्त करें। + +![star-us](https://github.com/langgenius/dify/assets/13230914/b823edc1-6388-4e25-ad45-2f6b187adbb4) + +## उन्नत सेटअप + +### कस्टम कॉन्फ़िगरेशन + +यदि आपको कॉन्फ़िगरेशन को कस्टमाइज़ करने की आवश्यकता है, तो कृपया हमारी [.env.example](../../docker/.env.example) फ़ाइल में दिए गए टिप्पणियों (comments) को देखें और अपने `.env` फ़ाइल में संबंधित मानों को अपडेट करें।\ +इसके अतिरिक्त, आपको अपने विशेष डिप्लॉयमेंट वातावरण और आवश्यकताओं के आधार पर `docker-compose.yaml` फ़ाइल में भी बदलाव करने की आवश्यकता हो सकती है, जैसे इमेज संस्करण, पोर्ट मैपिंग या वॉल्यूम माउंट्स बदलना।\ +कोई भी बदलाव करने के बाद, कृपया `docker-compose up -d` कमांड को पुनः चलाएँ।\ +उपलब्ध सभी environment variables की पूरी सूची [here](https://docs.dify.ai/getting-started/install-self-hosted/environments) पर पाई जा सकती है। + +### Grafana के साथ मेट्रिक्स मॉनिटरिंग + +Grafana में Dify के PostgreSQL डेटाबेस को डेटा स्रोत के रूप में उपयोग करते हुए डैशबोर्ड आयात करें, ताकि आप ऐप्स, टेनेंट्स, संदेशों आदि के स्तर पर मेट्रिक्स की निगरानी कर सकें। + +- [Grafana Dashboard by @bowenliang123](https://github.com/bowenliang123/dify-grafana-dashboard) + +### Kubernetes के साथ डिप्लॉयमेंट + +यदि आप उच्च उपलब्धता (high-availability) सेटअप कॉन्फ़िगर करना चाहते हैं, तो समुदाय द्वारा योगदान किए गए [Helm Charts](https://helm.sh/) और YAML फ़ाइलें उपलब्ध हैं जो Dify को Kubernetes पर डिप्लॉय करने की अनुमति देती हैं। + +- [Helm Chart by @LeoQuote](https://github.com/douban/charts/tree/master/charts/dify) +- [Helm Chart by @BorisPolonsky](https://github.com/BorisPolonsky/dify-helm) +- [Helm Chart by @magicsong](https://github.com/magicsong/ai-charts) +- [YAML file by @Winson-030](https://github.com/Winson-030/dify-kubernetes) +- [YAML file by @wyy-holding](https://github.com/wyy-holding/dify-k8s) +- [🚀 NEW! YAML files (Supports Dify v1.6.0) by @Zhoneym](https://github.com/Zhoneym/DifyAI-Kubernetes) + +#### डिप्लॉयमेंट के लिए Terraform का उपयोग + +[terraform](https://www.terraform.io/) का उपयोग करके एक क्लिक में Dify को क्लाउड प्लेटफ़ॉर्म पर डिप्लॉय करें। + +##### Azure Global + +- [Azure Terraform by @nikawang](https://github.com/nikawang/dify-azure-terraform) + +##### Google Cloud + +- [Google Cloud Terraform by @sotazum](https://github.com/DeNA/dify-google-cloud-terraform) + +#### डिप्लॉयमेंट के लिए AWS CDK का उपयोग + +[CDK](https://aws.amazon.com/cdk/) का उपयोग करके Dify को AWS पर डिप्लॉय करें। + +##### AWS + +- [AWS CDK by @KevinZhao (EKS आधारित)](https://github.com/aws-samples/solution-for-deploying-dify-on-aws) +- [AWS CDK by @tmokmss (ECS आधारित)](https://github.com/aws-samples/dify-self-hosted-on-aws) + +#### Alibaba Cloud Computing Nest का उपयोग + +[Alibaba Cloud Computing Nest](https://computenest.console.aliyun.com/service/instance/create/default?type=user&ServiceName=Dify%E7%A4%BE%E5%8C%BA%E7%89%88) के साथ Dify को Alibaba Cloud पर तेज़ी से डिप्लॉय करें। + +#### Alibaba Cloud Data Management का उपयोग + +[Alibaba Cloud Data Management](https://www.alibabacloud.com/help/en/dms/dify-in-invitational-preview/) के साथ एक क्लिक में Dify को Alibaba Cloud पर डिप्लॉय करें। + +#### Azure Devops Pipeline के साथ AKS पर डिप्लॉय करें + +[Azure Devops Pipeline Helm Chart by @LeoZhang](https://github.com/Ruiruiz30/Dify-helm-chart-AKS) के साथ एक क्लिक में Dify को AKS पर डिप्लॉय करें। + +## योगदान (Contributing) + +जो लोग कोड में योगदान देना चाहते हैं, वे हमारे [Contribution Guide](./CONTRIBUTING.md) को देखें।\ +साथ ही, कृपया Dify को सोशल मीडिया, कार्यक्रमों और सम्मेलनों में साझा करके इसका समर्थन करने पर विचार करें। + +> हम ऐसे योगदानकर्ताओं की तलाश कर रहे हैं जो Dify को मंदारिन या अंग्रेज़ी के अलावा अन्य भाषाओं में अनुवाद करने में मदद कर सकें।\ +> यदि आप सहायता करने में रुचि रखते हैं, तो अधिक जानकारी के लिए [i18n README](https://github.com/langgenius/dify/blob/main/web/i18n-config/README.md) देखें, और हमारे [Discord Community Server](https://discord.gg/8Tpq4AcN9c) के `global-users` चैनल में हमें संदेश दें। + +## समुदाय और संपर्क (Community & contact) + +- [GitHub Discussion](https://github.com/langgenius/dify/discussions) — सर्वोत्तम उपयोग के लिए: प्रतिक्रिया साझा करने और प्रश्न पूछने हेतु। +- [GitHub Issues](https://github.com/langgenius/dify/issues) — सर्वोत्तम उपयोग के लिए: Dify.AI का उपयोग करते समय आने वाली बग्स या फीचर सुझावों के लिए। देखें: [Contribution Guide](../../CONTRIBUTING.md)। +- [Discord](https://discord.gg/FngNHpbcY7) — सर्वोत्तम उपयोग के लिए: अपने एप्लिकेशन साझा करने और समुदाय के साथ जुड़ने के लिए। +- [X(Twitter)](https://twitter.com/dify_ai) — सर्वोत्तम उपयोग के लिए: अपने एप्लिकेशन साझा करने और समुदाय से जुड़े रहने के लिए। + +**योगदानकर्ता** + + + + + +## स्टार इतिहास (Star history) + +[![Star History Chart](https://api.star-history.com/svg?repos=langgenius/dify&type=Date)](https://star-history.com/#langgenius/dify&Date) + +## सुरक्षा प्रकटीकरण (Security disclosure) + +आपकी गोपनीयता की सुरक्षा के लिए, कृपया GitHub पर सुरक्षा संबंधित समस्याएँ पोस्ट करने से बचें।\ +इसके बजाय, समस्याओं की रिपोर्ट security@dify.ai पर करें, और हमारी टीम आपको विस्तृत उत्तर के साथ प्रतिक्रिया देगी। + +## लाइसेंस (License) + +यह रिपॉज़िटरी [Dify Open Source License](../../LICENSE) के अंतर्गत लाइसेंस प्राप्त है, जो Apache 2.0 पर आधारित है और इसमें अतिरिक्त शर्तें शामिल हैं। diff --git a/docs/it-IT/README.md b/docs/it-IT/README.md new file mode 100644 index 0000000000..598e87ec25 --- /dev/null +++ b/docs/it-IT/README.md @@ -0,0 +1,213 @@ +![cover-v5-optimized](../../images/GitHub_README_if.png) + +

+ 📌 Introduzione a Dify Workflow File Upload: ricreando il podcast di Google NotebookLM +

+ +

+ Dify Cloud · + Self-Hosted · + Documentazione · + Panoramica dei prodotti Dify +

+ +

+ + Static Badge + + Static Badge + + chat on Discord + + join Reddit + + follow on X(Twitter) + + follow on LinkedIn + + Docker Pulls + + Commits last month + + Issues closed + + Discussion posts +

+ +

+ README in English + 繁體中文文件 + 简体中文文件 + 日本語のREADME + README en Español + README en Français + README tlhIngan Hol + README in Korean + README بالعربية + Türkçe README + README Tiếng Việt + README in Deutsch + README in Italiano + README in বাংলা +

+ +Dify è una piattaforma open-source per lo sviluppo di applicazioni LLM. La sua interfaccia intuitiva combina flussi di lavoro AI basati su agenti, pipeline RAG, funzionalità di agenti, gestione dei modelli, funzionalità di monitoraggio e altro ancora, permettendovi di passare rapidamente da un prototipo alla produzione. + +## Avvio Rapido + +> Prima di installare Dify, assicuratevi che il vostro sistema soddisfi i seguenti requisiti minimi: +> +> - CPU >= 2 Core +> - RAM >= 4 GiB + +
+ +Il modo più semplice per avviare il server Dify è tramite [docker compose](../../docker/docker-compose.yaml). Prima di eseguire Dify con i seguenti comandi, assicuratevi che [Docker](https://docs.docker.com/get-docker/) e [Docker Compose](https://docs.docker.com/compose/install/) siano installati sul vostro sistema: + +```bash +cd dify +cd docker +cp .env.example .env +docker compose up -d +``` + +Dopo aver avviato il server, potete accedere al dashboard di Dify tramite il vostro browser all'indirizzo [http://localhost/install](http://localhost/install) e avviare il processo di inizializzazione. + +#### Richiedere Aiuto + +Consultate le nostre [FAQ](https://docs.dify.ai/getting-started/install-self-hosted/faqs) se riscontrate problemi durante la configurazione di Dify. Contattateci [tramite la community](#community--contatti) se continuano a verificarsi difficoltà. + +> Se desiderate contribuire a Dify o effettuare ulteriori sviluppi, consultate la nostra [guida al deployment dal codice sorgente](https://docs.dify.ai/getting-started/install-self-hosted/local-source-code). + +## Caratteristiche Principali + +**1. Workflow**: +Create e testate potenti flussi di lavoro AI su un'interfaccia visuale, utilizzando tutte le funzionalità seguenti e oltre. + +**2. Supporto Completo dei Modelli**: +Integrazione perfetta con centinaia di LLM proprietari e open-source di decine di provider di inferenza e soluzioni self-hosted, che coprono GPT, Mistral, Llama3 e tutti i modelli compatibili con l'API OpenAI. L'elenco completo dei provider di modelli supportati è disponibile [qui](https://docs.dify.ai/getting-started/readme/model-providers). + +![providers-v5](https://github.com/langgenius/dify/assets/13230914/5a17bdbe-097a-4100-8363-40255b70f6e3) + +**3. Prompt IDE**: +Interfaccia intuitiva per creare prompt, confrontare le prestazioni dei modelli e aggiungere funzionalità aggiuntive come text-to-speech in un'applicazione basata su chat. + +**4. Pipeline RAG**: +Funzionalità RAG complete che coprono tutto, dall'acquisizione dei documenti alla loro interrogazione, con supporto pronto all'uso per l'estrazione di testo da PDF, PPT e altri formati di documenti comuni. + +**5. Capacità degli Agenti**: +Potete definire agenti basati su LLM Function Calling o ReAct e aggiungere strumenti predefiniti o personalizzati per l'agente. Dify fornisce oltre 50 strumenti integrati per gli agenti AI, come Google Search, DALL·E, Stable Diffusion e WolframAlpha. + +**6. LLMOps**: +Monitorate e analizzate i log delle applicazioni e le prestazioni nel tempo. Potete migliorare continuamente prompt, dataset e modelli basandovi sui dati di produzione e sulle annotazioni. + +**7. Backend-as-a-Service**: +Tutte le offerte di Dify sono dotate di API corrispondenti, permettendovi di integrare facilmente Dify nella vostra logica di business. + +## Utilizzo di Dify + +- **Cloud
** + Ospitiamo un servizio [Dify Cloud](https://dify.ai) che chiunque può provare senza configurazione. Offre tutte le funzionalità della versione self-hosted e include 200 chiamate GPT-4 gratuite nel piano sandbox. + +- **Dify Community Edition Self-Hosted
** + Avviate rapidamente Dify nel vostro ambiente con questa [guida di avvio rapido](#avvio-rapido). Utilizzate la nostra [documentazione](https://docs.dify.ai) per ulteriori informazioni e istruzioni dettagliate. + +- **Dify per Aziende / Organizzazioni
** + Offriamo funzionalità aggiuntive specifiche per le aziende. Potete [scriverci via email](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) per discutere le vostre esigenze aziendali.
+ + > Per startup e piccole imprese che utilizzano AWS, date un'occhiata a [Dify Premium su AWS Marketplace](https://aws.amazon.com/marketplace/pp/prodview-t22mebxzwjhu6) e distribuitelo con un solo clic nel vostro AWS VPC. Si tratta di un'offerta AMI conveniente con l'opzione di creare app con logo e branding personalizzati. + +## Resta Sempre Aggiornato + +Mettete una stella a Dify su GitHub e ricevete notifiche immediate sui nuovi rilasci. + +![star-us](https://github.com/langgenius/dify/assets/13230914/b823edc1-6388-4e25-ad45-2f6b187adbb4) + +## Configurazioni Avanzate + +Se dovete personalizzare la configurazione, leggete i commenti nel nostro file [.env.example](../../docker/.env.example) e aggiornate i valori corrispondenti nel vostro file `.env`. Inoltre, potrebbe essere necessario apportare modifiche al file `docker-compose.yaml`, come cambiare le versioni delle immagini, le mappature delle porte o i mount dei volumi, a seconda del vostro ambiente di distribuzione specifico e dei vostri requisiti. Dopo aver apportato le modifiche, riavviate `docker-compose up -d`. L'elenco completo delle variabili d'ambiente disponibili è disponibile [qui](https://docs.dify.ai/getting-started/install-self-hosted/environments). + +### Monitoraggio delle Metriche con Grafana + +Importate la dashboard in Grafana, utilizzando il database PostgreSQL di Dify come origine dati, per monitorare le metriche a livello di app, tenant, messaggi e altro ancora. + +- [Dashboard Grafana di @bowenliang123](https://github.com/bowenliang123/dify-grafana-dashboard) + +### Distribuzione con Kubernetes + +Se desiderate configurare un'installazione ad alta disponibilità, ci sono [Helm Charts](https://helm.sh/) e file YAML forniti dalla community che consentono di distribuire Dify su Kubernetes. + +- [Helm Chart di @LeoQuote](https://github.com/douban/charts/tree/master/charts/dify) +- [Helm Chart di @BorisPolonsky](https://github.com/BorisPolonsky/dify-helm) +- [Helm Chart di @magicsong](https://github.com/magicsong/ai-charts) +- [File YAML di @Winson-030](https://github.com/Winson-030/dify-kubernetes) +- [File YAML di @wyy-holding](https://github.com/wyy-holding/dify-k8s) +- [🚀 NUOVO! File YAML (Supporta Dify v1.6.0) di @Zhoneym](https://github.com/Zhoneym/DifyAI-Kubernetes) + +#### Utilizzo di Terraform per la Distribuzione + +Distribuite Dify con un solo clic su una piattaforma cloud utilizzando [terraform](https://www.terraform.io/). + +##### Azure Global + +- [Azure Terraform di @nikawang](https://github.com/nikawang/dify-azure-terraform) + +##### Google Cloud + +- [Google Cloud Terraform di @sotazum](https://github.com/DeNA/dify-google-cloud-terraform) + +#### Utilizzo di AWS CDK per la Distribuzione + +Distribuzione di Dify su AWS con [CDK](https://aws.amazon.com/cdk/) + +##### AWS + +- [AWS CDK di @KevinZhao (basato su EKS)](https://github.com/aws-samples/solution-for-deploying-dify-on-aws) +- [AWS CDK di @tmokmss (basato su ECS)](https://github.com/aws-samples/dify-self-hosted-on-aws) + +#### Alibaba Cloud + +[Alibaba Cloud Computing Nest](https://computenest.console.aliyun.com/service/instance/create/default?type=user&ServiceName=Dify%E7%A4%BE%E5%8C%BA%E7%89%88) + +#### Alibaba Cloud Data Management + +Distribuzione con un clic di Dify su Alibaba Cloud con [Alibaba Cloud Data Management](https://www.alibabacloud.com/help/en/dms/dify-in-invitational-preview/) + +#### Utilizzo di Azure DevOps Pipeline per la Distribuzione su AKS + +Distribuite Dify con un clic in AKS utilizzando [Azure DevOps Pipeline Helm Chart di @LeoZhang](https://github.com/Ruiruiz30/Dify-helm-chart-AKS) + +## Contribuire + +Se desiderate contribuire con codice, leggete la nostra [Guida ai Contributi](../../CONTRIBUTING.md). Allo stesso tempo, vi chiediamo di supportare Dify condividendolo sui social media e presentandolo a eventi e conferenze. + +> Cerchiamo collaboratori che aiutino a tradurre Dify in altre lingue oltre al mandarino o all'inglese. Se siete interessati a collaborare, leggete il [README i18n](https://github.com/langgenius/dify/blob/main/web/i18n-config/README.md) per ulteriori informazioni e lasciate un commento nel canale `global-users` del nostro [server della community Discord](https://discord.gg/8Tpq4AcN9c). + +## Community & Contatti + +- [GitHub Discussion](https://github.com/langgenius/dify/discussions). Ideale per: condividere feedback e porre domande. +- [GitHub Issues](https://github.com/langgenius/dify/issues). Ideale per: bug che riscontrate durante l'utilizzo di Dify.AI e proposte di funzionalità. Consultate la nostra [Guida ai Contributi](../../CONTRIBUTING.md). +- [Discord](https://discord.gg/FngNHpbcY7). Ideale per: condividere le vostre applicazioni e interagire con la community. +- [X(Twitter)](https://twitter.com/dify_ai). Ideale per: condividere le vostre applicazioni e interagire con la community. + +**Collaboratori** + + + + + +## Storia delle Stelle + +[![Star History Chart](https://api.star-history.com/svg?repos=langgenius/dify&type=Date)](https://star-history.com/#langgenius/dify&Date) + +## Divulgazione sulla Sicurezza + +Per proteggere la vostra privacy, evitate di pubblicare problemi di sicurezza su GitHub. Inviate invece le vostre domande a security@dify.ai e vi forniremo una risposta più dettagliata. + +## Licenza + +Questo repository è disponibile sotto la [Dify Open Source License](../../LICENSE), che è essenzialmente Apache 2.0 con alcune restrizioni aggiuntive. diff --git a/docs/pt-BR/README.md b/docs/pt-BR/README.md index f96b18eabb..444faa0a67 100644 --- a/docs/pt-BR/README.md +++ b/docs/pt-BR/README.md @@ -91,7 +91,7 @@ Todas os recursos do Dify vêm com APIs correspondentes, permitindo que você in Use nossa [documentação](https://docs.dify.ai) para referências adicionais e instruções mais detalhadas. - **Dify para empresas/organizações
** - Oferecemos recursos adicionais voltados para empresas. [Envie suas perguntas através deste chatbot](https://udify.app/chat/22L1zSxg6yW1cWQg) ou [envie-nos um e-mail](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) para discutir necessidades empresariais.
+ Oferecemos recursos adicionais voltados para empresas. Você pode [falar conosco por e-mail](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) para discutir necessidades empresariais.
> Para startups e pequenas empresas que utilizam AWS, confira o [Dify Premium no AWS Marketplace](https://aws.amazon.com/marketplace/pp/prodview-t22mebxzwjhu6) e implemente no seu próprio AWS VPC com um clique. É uma oferta AMI acessível com a opção de criar aplicativos com logotipo e marca personalizados. diff --git a/docs/vi-VN/README.md b/docs/vi-VN/README.md index 51f7c5d994..07329e84cd 100644 --- a/docs/vi-VN/README.md +++ b/docs/vi-VN/README.md @@ -86,7 +86,7 @@ Tất cả các dịch vụ của Dify đều đi kèm với các API tương Sử dụng [tài liệu](https://docs.dify.ai) của chúng tôi để tham khảo thêm và nhận hướng dẫn chi tiết hơn. - **Dify cho doanh nghiệp / tổ chức
** - Chúng tôi cung cấp các tính năng bổ sung tập trung vào doanh nghiệp. [Ghi lại câu hỏi của bạn cho chúng tôi thông qua chatbot này](https://udify.app/chat/22L1zSxg6yW1cWQg) hoặc [gửi email cho chúng tôi](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) để thảo luận về nhu cầu doanh nghiệp.
+ Chúng tôi cung cấp các tính năng bổ sung tập trung vào doanh nghiệp. [Gửi email cho chúng tôi](mailto:business@dify.ai?subject=%5BGitHub%5DBusiness%20License%20Inquiry) để thảo luận về nhu cầu doanh nghiệp.
> Đối với các công ty khởi nghiệp và doanh nghiệp nhỏ sử dụng AWS, hãy xem [Dify Premium trên AWS Marketplace](https://aws.amazon.com/marketplace/pp/prodview-t22mebxzwjhu6) và triển khai nó vào AWS VPC của riêng bạn chỉ với một cú nhấp chuột. Đây là một AMI giá cả phải chăng với tùy chọn tạo ứng dụng với logo và thương hiệu tùy chỉnh. diff --git a/web/.env.example b/web/.env.example index 23b72b3414..5bfcc9dac0 100644 --- a/web/.env.example +++ b/web/.env.example @@ -34,6 +34,9 @@ NEXT_PUBLIC_CSP_WHITELIST= # Default is not allow to embed into iframe to prevent Clickjacking: https://owasp.org/www-community/attacks/Clickjacking NEXT_PUBLIC_ALLOW_EMBED= +# Shared cookie domain when console UI and API use different subdomains (e.g. example.com) +NEXT_PUBLIC_COOKIE_DOMAIN= + # Allow rendering unsafe URLs which have "data:" scheme. NEXT_PUBLIC_ALLOW_UNSAFE_DATA_SCHEME=false @@ -61,5 +64,9 @@ NEXT_PUBLIC_ENABLE_WEBSITE_JINAREADER=true NEXT_PUBLIC_ENABLE_WEBSITE_FIRECRAWL=true NEXT_PUBLIC_ENABLE_WEBSITE_WATERCRAWL=true +# Enable inline LaTeX rendering with single dollar signs ($...$) +# Default is false for security reasons to prevent conflicts with regular text +NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX=false + # The maximum number of tree node depth for workflow NEXT_PUBLIC_MAX_TREE_DEPTH=50 diff --git a/web/.husky/pre-commit b/web/.husky/pre-commit index 1db4b6dd67..26e9bf69d4 100644 --- a/web/.husky/pre-commit +++ b/web/.husky/pre-commit @@ -44,9 +44,32 @@ fi if $web_modified; then echo "Running ESLint on web module" + + if git diff --cached --quiet -- 'web/**/*.ts' 'web/**/*.tsx'; then + web_ts_modified=false + else + ts_diff_status=$? + if [ $ts_diff_status -eq 1 ]; then + web_ts_modified=true + else + echo "Unable to determine staged TypeScript changes (git exit code: $ts_diff_status)." + exit $ts_diff_status + fi + fi + cd ./web || exit 1 lint-staged + if $web_ts_modified; then + echo "Running TypeScript type-check" + if ! pnpm run type-check; then + echo "Type check failed. Please run 'pnpm run type-check' to fix the errors." + exit 1 + fi + else + echo "No staged TypeScript changes detected, skipping type-check" + fi + echo "Running unit tests check" modified_files=$(git diff --cached --name-only -- utils | grep -v '\.spec\.ts$' || true) diff --git a/web/.storybook/main.ts b/web/.storybook/main.ts index 57abae42ab..ca56261431 100644 --- a/web/.storybook/main.ts +++ b/web/.storybook/main.ts @@ -1,5 +1,8 @@ import type { StorybookConfig } from '@storybook/nextjs' import path from 'node:path' +import { fileURLToPath } from 'node:url' + +const storybookDir = path.dirname(fileURLToPath(import.meta.url)) const config: StorybookConfig = { stories: ['../app/components/**/*.stories.@(js|jsx|mjs|ts|tsx)'], @@ -32,9 +35,9 @@ const config: StorybookConfig = { config.resolve.alias = { ...config.resolve.alias, // Mock the plugin index files to avoid circular dependencies - [path.resolve(__dirname, '../app/components/base/prompt-editor/plugins/context-block/index.tsx')]: path.resolve(__dirname, '__mocks__/context-block.tsx'), - [path.resolve(__dirname, '../app/components/base/prompt-editor/plugins/history-block/index.tsx')]: path.resolve(__dirname, '__mocks__/history-block.tsx'), - [path.resolve(__dirname, '../app/components/base/prompt-editor/plugins/query-block/index.tsx')]: path.resolve(__dirname, '__mocks__/query-block.tsx'), + [path.resolve(storybookDir, '../app/components/base/prompt-editor/plugins/context-block/index.tsx')]: path.resolve(storybookDir, '__mocks__/context-block.tsx'), + [path.resolve(storybookDir, '../app/components/base/prompt-editor/plugins/history-block/index.tsx')]: path.resolve(storybookDir, '__mocks__/history-block.tsx'), + [path.resolve(storybookDir, '../app/components/base/prompt-editor/plugins/query-block/index.tsx')]: path.resolve(storybookDir, '__mocks__/query-block.tsx'), } return config }, diff --git a/web/.storybook/utils/audio-player-manager.mock.ts b/web/.storybook/utils/audio-player-manager.mock.ts new file mode 100644 index 0000000000..aca8b56b76 --- /dev/null +++ b/web/.storybook/utils/audio-player-manager.mock.ts @@ -0,0 +1,64 @@ +import { AudioPlayerManager } from '@/app/components/base/audio-btn/audio.player.manager' + +type PlayerCallback = ((event: string) => void) | null + +class MockAudioPlayer { + private callback: PlayerCallback = null + private finishTimer?: ReturnType + + public setCallback(callback: PlayerCallback) { + this.callback = callback + } + + public playAudio() { + this.clearTimer() + this.callback?.('play') + this.finishTimer = setTimeout(() => { + this.callback?.('ended') + }, 2000) + } + + public pauseAudio() { + this.clearTimer() + this.callback?.('paused') + } + + private clearTimer() { + if (this.finishTimer) + clearTimeout(this.finishTimer) + } +} + +class MockAudioPlayerManager { + private readonly player = new MockAudioPlayer() + + public getAudioPlayer( + _url: string, + _isPublic: boolean, + _id: string | undefined, + _msgContent: string | null | undefined, + _voice: string | undefined, + callback: PlayerCallback, + ) { + this.player.setCallback(callback) + return this.player + } + + public resetMsgId() { + // No-op for the mock + } +} + +export const ensureMockAudioManager = () => { + const managerAny = AudioPlayerManager as unknown as { + getInstance: () => AudioPlayerManager + __isStorybookMockInstalled?: boolean + } + + if (managerAny.__isStorybookMockInstalled) + return + + const mock = new MockAudioPlayerManager() + managerAny.getInstance = () => mock as unknown as AudioPlayerManager + managerAny.__isStorybookMockInstalled = true +} diff --git a/web/.storybook/utils/form-story-wrapper.tsx b/web/.storybook/utils/form-story-wrapper.tsx new file mode 100644 index 0000000000..689c3a20ff --- /dev/null +++ b/web/.storybook/utils/form-story-wrapper.tsx @@ -0,0 +1,83 @@ +import { useState } from 'react' +import type { ReactNode } from 'react' +import { useStore } from '@tanstack/react-form' +import { useAppForm } from '@/app/components/base/form' + +type UseAppFormOptions = Parameters[0] +type AppFormInstance = ReturnType + +type FormStoryWrapperProps = { + options?: UseAppFormOptions + children: (form: AppFormInstance) => ReactNode + title?: string + subtitle?: string +} + +export const FormStoryWrapper = ({ + options, + children, + title, + subtitle, +}: FormStoryWrapperProps) => { + const [lastSubmitted, setLastSubmitted] = useState(null) + const [submitCount, setSubmitCount] = useState(0) + + const form = useAppForm({ + ...options, + onSubmit: (context) => { + setSubmitCount(count => count + 1) + setLastSubmitted(context.value) + options?.onSubmit?.(context) + }, + }) + + const values = useStore(form.store, state => state.values) + const isSubmitting = useStore(form.store, state => state.isSubmitting) + const canSubmit = useStore(form.store, state => state.canSubmit) + + return ( +
+
+ {(title || subtitle) && ( +
+ {title &&

{title}

} + {subtitle &&

{subtitle}

} +
+ )} + {children(form)} +
+ +
+ ) +} + +export type FormStoryRender = (form: AppFormInstance) => ReactNode diff --git a/web/__tests__/check-i18n.test.ts b/web/__tests__/check-i18n.test.ts index b579f22d4b..7773edcdbb 100644 --- a/web/__tests__/check-i18n.test.ts +++ b/web/__tests__/check-i18n.test.ts @@ -759,4 +759,104 @@ export default translation` expect(result).not.toContain('Zbuduj inteligentnego agenta') }) }) + + describe('Performance and Scalability', () => { + it('should handle large translation files efficiently', async () => { + // Create a large translation file with 1000 keys + const largeContent = `const translation = { +${Array.from({ length: 1000 }, (_, i) => ` key${i}: 'value${i}',`).join('\n')} +} + +export default translation` + + fs.writeFileSync(path.join(testEnDir, 'large.ts'), largeContent) + + const startTime = Date.now() + const keys = await getKeysFromLanguage('en-US') + const endTime = Date.now() + + expect(keys.length).toBe(1000) + expect(endTime - startTime).toBeLessThan(1000) // Should complete in under 1 second + }) + + it('should handle multiple translation files concurrently', async () => { + // Create multiple files + for (let i = 0; i < 10; i++) { + const content = `const translation = { + key${i}: 'value${i}', + nested${i}: { + subkey: 'subvalue' + } +} + +export default translation` + fs.writeFileSync(path.join(testEnDir, `file${i}.ts`), content) + } + + const startTime = Date.now() + const keys = await getKeysFromLanguage('en-US') + const endTime = Date.now() + + expect(keys.length).toBe(20) // 10 files * 2 keys each + expect(endTime - startTime).toBeLessThan(500) + }) + }) + + describe('Unicode and Internationalization', () => { + it('should handle Unicode characters in keys and values', async () => { + const unicodeContent = `const translation = { + '中文键': '中文值', + 'العربية': 'قيمة', + 'emoji_😀': 'value with emoji 🎉', + 'mixed_中文_English': 'mixed value' +} + +export default translation` + + fs.writeFileSync(path.join(testEnDir, 'unicode.ts'), unicodeContent) + + const keys = await getKeysFromLanguage('en-US') + + expect(keys).toContain('unicode.中文键') + expect(keys).toContain('unicode.العربية') + expect(keys).toContain('unicode.emoji_😀') + expect(keys).toContain('unicode.mixed_中文_English') + }) + + it('should handle RTL language files', async () => { + const rtlContent = `const translation = { + مرحبا: 'Hello', + العالم: 'World', + nested: { + مفتاح: 'key' + } +} + +export default translation` + + fs.writeFileSync(path.join(testEnDir, 'rtl.ts'), rtlContent) + + const keys = await getKeysFromLanguage('en-US') + + expect(keys).toContain('rtl.مرحبا') + expect(keys).toContain('rtl.العالم') + expect(keys).toContain('rtl.nested.مفتاح') + }) + }) + + describe('Error Recovery', () => { + it('should handle syntax errors in translation files gracefully', async () => { + const invalidContent = `const translation = { + validKey: 'valid value', + invalidKey: 'missing quote, + anotherKey: 'another value' +} + +export default translation` + + fs.writeFileSync(path.join(testEnDir, 'invalid.ts'), invalidContent) + + await expect(getKeysFromLanguage('en-US')).rejects.toThrow() + }) + }) }) diff --git a/web/__tests__/embedded-user-id-auth.test.tsx b/web/__tests__/embedded-user-id-auth.test.tsx new file mode 100644 index 0000000000..5c3c3c943f --- /dev/null +++ b/web/__tests__/embedded-user-id-auth.test.tsx @@ -0,0 +1,132 @@ +import React from 'react' +import { fireEvent, render, screen, waitFor } from '@testing-library/react' + +import MailAndPasswordAuth from '@/app/(shareLayout)/webapp-signin/components/mail-and-password-auth' +import CheckCode from '@/app/(shareLayout)/webapp-signin/check-code/page' + +jest.mock('react-i18next', () => ({ + useTranslation: () => ({ + t: (key: string) => key, + }), +})) + +const replaceMock = jest.fn() +const backMock = jest.fn() + +jest.mock('next/navigation', () => ({ + usePathname: jest.fn(() => '/chatbot/test-app'), + useRouter: jest.fn(() => ({ + replace: replaceMock, + back: backMock, + })), + useSearchParams: jest.fn(), +})) + +const mockStoreState = { + embeddedUserId: 'embedded-user-99', + shareCode: 'test-app', +} + +const useWebAppStoreMock = jest.fn((selector?: (state: typeof mockStoreState) => any) => { + return selector ? selector(mockStoreState) : mockStoreState +}) + +jest.mock('@/context/web-app-context', () => ({ + useWebAppStore: (selector?: (state: typeof mockStoreState) => any) => useWebAppStoreMock(selector), +})) + +const webAppLoginMock = jest.fn() +const webAppEmailLoginWithCodeMock = jest.fn() +const sendWebAppEMailLoginCodeMock = jest.fn() + +jest.mock('@/service/common', () => ({ + webAppLogin: (...args: any[]) => webAppLoginMock(...args), + webAppEmailLoginWithCode: (...args: any[]) => webAppEmailLoginWithCodeMock(...args), + sendWebAppEMailLoginCode: (...args: any[]) => sendWebAppEMailLoginCodeMock(...args), +})) + +const fetchAccessTokenMock = jest.fn() + +jest.mock('@/service/share', () => ({ + fetchAccessToken: (...args: any[]) => fetchAccessTokenMock(...args), +})) + +const setWebAppAccessTokenMock = jest.fn() +const setWebAppPassportMock = jest.fn() + +jest.mock('@/service/webapp-auth', () => ({ + setWebAppAccessToken: (...args: any[]) => setWebAppAccessTokenMock(...args), + setWebAppPassport: (...args: any[]) => setWebAppPassportMock(...args), + webAppLogout: jest.fn(), +})) + +jest.mock('@/app/components/signin/countdown', () => () =>
) + +jest.mock('@remixicon/react', () => ({ + RiMailSendFill: () =>
, + RiArrowLeftLine: () =>
, +})) + +const { useSearchParams } = jest.requireMock('next/navigation') as { + useSearchParams: jest.Mock +} + +beforeEach(() => { + jest.clearAllMocks() +}) + +describe('embedded user id propagation in authentication flows', () => { + it('passes embedded user id when logging in with email and password', async () => { + const params = new URLSearchParams() + params.set('redirect_url', encodeURIComponent('/chatbot/test-app')) + useSearchParams.mockReturnValue(params) + + webAppLoginMock.mockResolvedValue({ result: 'success', data: { access_token: 'login-token' } }) + fetchAccessTokenMock.mockResolvedValue({ access_token: 'passport-token' }) + + render() + + fireEvent.change(screen.getByLabelText('login.email'), { target: { value: 'user@example.com' } }) + fireEvent.change(screen.getByLabelText(/login\.password/), { target: { value: 'strong-password' } }) + fireEvent.click(screen.getByRole('button', { name: 'login.signBtn' })) + + await waitFor(() => { + expect(fetchAccessTokenMock).toHaveBeenCalledWith({ + appCode: 'test-app', + userId: 'embedded-user-99', + }) + }) + expect(setWebAppAccessTokenMock).toHaveBeenCalledWith('login-token') + expect(setWebAppPassportMock).toHaveBeenCalledWith('test-app', 'passport-token') + expect(replaceMock).toHaveBeenCalledWith('/chatbot/test-app') + }) + + it('passes embedded user id when verifying email code', async () => { + const params = new URLSearchParams() + params.set('redirect_url', encodeURIComponent('/chatbot/test-app')) + params.set('email', encodeURIComponent('user@example.com')) + params.set('token', encodeURIComponent('token-abc')) + useSearchParams.mockReturnValue(params) + + webAppEmailLoginWithCodeMock.mockResolvedValue({ result: 'success', data: { access_token: 'code-token' } }) + fetchAccessTokenMock.mockResolvedValue({ access_token: 'passport-token' }) + + render() + + fireEvent.change( + screen.getByPlaceholderText('login.checkCode.verificationCodePlaceholder'), + { target: { value: '123456' } }, + ) + fireEvent.click(screen.getByRole('button', { name: 'login.checkCode.verify' })) + + await waitFor(() => { + expect(fetchAccessTokenMock).toHaveBeenCalledWith({ + appCode: 'test-app', + userId: 'embedded-user-99', + }) + }) + expect(setWebAppAccessTokenMock).toHaveBeenCalledWith('code-token') + expect(setWebAppPassportMock).toHaveBeenCalledWith('test-app', 'passport-token') + expect(replaceMock).toHaveBeenCalledWith('/chatbot/test-app') + }) +}) diff --git a/web/__tests__/embedded-user-id-store.test.tsx b/web/__tests__/embedded-user-id-store.test.tsx new file mode 100644 index 0000000000..24a815222e --- /dev/null +++ b/web/__tests__/embedded-user-id-store.test.tsx @@ -0,0 +1,155 @@ +import React from 'react' +import { render, screen, waitFor } from '@testing-library/react' + +import WebAppStoreProvider, { useWebAppStore } from '@/context/web-app-context' + +jest.mock('next/navigation', () => ({ + usePathname: jest.fn(() => '/chatbot/sample-app'), + useSearchParams: jest.fn(() => { + const params = new URLSearchParams() + return params + }), +})) + +jest.mock('@/service/use-share', () => { + const { AccessMode } = jest.requireActual('@/models/access-control') + return { + useGetWebAppAccessModeByCode: jest.fn(() => ({ + isLoading: false, + data: { accessMode: AccessMode.PUBLIC }, + })), + } +}) + +jest.mock('@/app/components/base/chat/utils', () => ({ + getProcessedSystemVariablesFromUrlParams: jest.fn(), +})) + +const { getProcessedSystemVariablesFromUrlParams: mockGetProcessedSystemVariablesFromUrlParams } + = jest.requireMock('@/app/components/base/chat/utils') as { + getProcessedSystemVariablesFromUrlParams: jest.Mock + } + +jest.mock('@/context/global-public-context', () => { + const mockGlobalStoreState = { + isGlobalPending: false, + setIsGlobalPending: jest.fn(), + systemFeatures: {}, + setSystemFeatures: jest.fn(), + } + const useGlobalPublicStore = Object.assign( + (selector?: (state: typeof mockGlobalStoreState) => any) => + selector ? selector(mockGlobalStoreState) : mockGlobalStoreState, + { + setState: (updater: any) => { + if (typeof updater === 'function') + Object.assign(mockGlobalStoreState, updater(mockGlobalStoreState) ?? {}) + + else + Object.assign(mockGlobalStoreState, updater) + }, + __mockState: mockGlobalStoreState, + }, + ) + return { + useGlobalPublicStore, + } +}) + +const { + useGlobalPublicStore: useGlobalPublicStoreMock, +} = jest.requireMock('@/context/global-public-context') as { + useGlobalPublicStore: ((selector?: (state: any) => any) => any) & { + setState: (updater: any) => void + __mockState: { + isGlobalPending: boolean + setIsGlobalPending: jest.Mock + systemFeatures: Record + setSystemFeatures: jest.Mock + } + } +} +const mockGlobalStoreState = useGlobalPublicStoreMock.__mockState + +const TestConsumer = () => { + const embeddedUserId = useWebAppStore(state => state.embeddedUserId) + const embeddedConversationId = useWebAppStore(state => state.embeddedConversationId) + return ( + <> +
{embeddedUserId ?? 'null'}
+
{embeddedConversationId ?? 'null'}
+ + ) +} + +const initialWebAppStore = (() => { + const snapshot = useWebAppStore.getState() + return { + shareCode: null as string | null, + appInfo: null, + appParams: null, + webAppAccessMode: snapshot.webAppAccessMode, + appMeta: null, + userCanAccessApp: false, + embeddedUserId: null, + embeddedConversationId: null, + updateShareCode: snapshot.updateShareCode, + updateAppInfo: snapshot.updateAppInfo, + updateAppParams: snapshot.updateAppParams, + updateWebAppAccessMode: snapshot.updateWebAppAccessMode, + updateWebAppMeta: snapshot.updateWebAppMeta, + updateUserCanAccessApp: snapshot.updateUserCanAccessApp, + updateEmbeddedUserId: snapshot.updateEmbeddedUserId, + updateEmbeddedConversationId: snapshot.updateEmbeddedConversationId, + } +})() + +beforeEach(() => { + mockGlobalStoreState.isGlobalPending = false + mockGetProcessedSystemVariablesFromUrlParams.mockReset() + useWebAppStore.setState(initialWebAppStore, true) +}) + +describe('WebAppStoreProvider embedded user id handling', () => { + it('hydrates embedded user and conversation ids from system variables', async () => { + mockGetProcessedSystemVariablesFromUrlParams.mockResolvedValue({ + user_id: 'iframe-user-123', + conversation_id: 'conversation-456', + }) + + render( + + + , + ) + + await waitFor(() => { + expect(screen.getByTestId('embedded-user-id')).toHaveTextContent('iframe-user-123') + expect(screen.getByTestId('embedded-conversation-id')).toHaveTextContent('conversation-456') + }) + expect(useWebAppStore.getState().embeddedUserId).toBe('iframe-user-123') + expect(useWebAppStore.getState().embeddedConversationId).toBe('conversation-456') + }) + + it('clears embedded user id when system variable is absent', async () => { + useWebAppStore.setState(state => ({ + ...state, + embeddedUserId: 'previous-user', + embeddedConversationId: 'existing-conversation', + })) + mockGetProcessedSystemVariablesFromUrlParams.mockResolvedValue({}) + + render( + + + , + ) + + await waitFor(() => { + expect(screen.getByTestId('embedded-user-id')).toHaveTextContent('null') + expect(screen.getByTestId('embedded-conversation-id')).toHaveTextContent('null') + }) + expect(useWebAppStore.getState().embeddedUserId).toBeNull() + expect(useWebAppStore.getState().embeddedConversationId).toBeNull() + }) +}) diff --git a/web/__tests__/navigation-utils.test.ts b/web/__tests__/navigation-utils.test.ts index fa4986e63d..3eeba52943 100644 --- a/web/__tests__/navigation-utils.test.ts +++ b/web/__tests__/navigation-utils.test.ts @@ -286,4 +286,116 @@ describe('Navigation Utilities', () => { expect(mockPush).toHaveBeenCalledWith('/datasets/filtered-set/documents?page=1&limit=50&status=active&type=pdf&sort=created_at&order=desc') }) }) + + describe('Edge Cases and Error Handling', () => { + test('handles special characters in query parameters', () => { + Object.defineProperty(window, 'location', { + value: { search: '?keyword=hello%20world&filter=type%3Apdf&tag=%E4%B8%AD%E6%96%87' }, + writable: true, + }) + + const path = createNavigationPath('/datasets/123/documents') + expect(path).toContain('hello+world') + expect(path).toContain('type%3Apdf') + expect(path).toContain('%E4%B8%AD%E6%96%87') + }) + + test('handles duplicate query parameters', () => { + Object.defineProperty(window, 'location', { + value: { search: '?tag=tag1&tag=tag2&tag=tag3' }, + writable: true, + }) + + const params = extractQueryParams(['tag']) + // URLSearchParams.get() returns the first value + expect(params.tag).toBe('tag1') + }) + + test('handles very long query strings', () => { + const longValue = 'a'.repeat(1000) + Object.defineProperty(window, 'location', { + value: { search: `?data=${longValue}` }, + writable: true, + }) + + const path = createNavigationPath('/datasets/123/documents') + expect(path).toContain(longValue) + expect(path.length).toBeGreaterThan(1000) + }) + + test('handles empty string values in query parameters', () => { + const path = createNavigationPathWithParams('/datasets/123/documents', { + page: 1, + keyword: '', + filter: '', + sort: 'name', + }) + + expect(path).toBe('/datasets/123/documents?page=1&sort=name') + expect(path).not.toContain('keyword=') + expect(path).not.toContain('filter=') + }) + + test('handles null and undefined values in mergeQueryParams', () => { + Object.defineProperty(window, 'location', { + value: { search: '?page=1&limit=10&keyword=test' }, + writable: true, + }) + + const merged = mergeQueryParams({ + keyword: null, + filter: undefined, + sort: 'name', + }) + const result = merged.toString() + + expect(result).toContain('page=1') + expect(result).toContain('limit=10') + expect(result).not.toContain('keyword') + expect(result).toContain('sort=name') + }) + + test('handles navigation with hash fragments', () => { + Object.defineProperty(window, 'location', { + value: { search: '?page=1', hash: '#section-2' }, + writable: true, + }) + + const path = createNavigationPath('/datasets/123/documents') + // Should preserve query params but not hash + expect(path).toBe('/datasets/123/documents?page=1') + }) + + test('handles malformed query strings gracefully', () => { + Object.defineProperty(window, 'location', { + value: { search: '?page=1&invalid&limit=10&=value&key=' }, + writable: true, + }) + + const params = extractQueryParams(['page', 'limit', 'invalid', 'key']) + expect(params.page).toBe('1') + expect(params.limit).toBe('10') + // Malformed params should be handled by URLSearchParams + expect(params.invalid).toBe('') // for `&invalid` + expect(params.key).toBe('') // for `&key=` + }) + }) + + describe('Performance Tests', () => { + test('handles large number of query parameters efficiently', () => { + const manyParams = Array.from({ length: 50 }, (_, i) => `param${i}=value${i}`).join('&') + Object.defineProperty(window, 'location', { + value: { search: `?${manyParams}` }, + writable: true, + }) + + const startTime = Date.now() + const path = createNavigationPath('/datasets/123/documents') + const endTime = Date.now() + + expect(endTime - startTime).toBeLessThan(50) // Should be fast + expect(path).toContain('param0=value0') + expect(path).toContain('param49=value49') + }) + }) }) diff --git a/web/__tests__/workflow-onboarding-integration.test.tsx b/web/__tests__/workflow-onboarding-integration.test.tsx new file mode 100644 index 0000000000..c1a922bb1f --- /dev/null +++ b/web/__tests__/workflow-onboarding-integration.test.tsx @@ -0,0 +1,614 @@ +import { BlockEnum } from '@/app/components/workflow/types' +import { useWorkflowStore } from '@/app/components/workflow/store' + +// Mock zustand store +jest.mock('@/app/components/workflow/store') + +// Mock ReactFlow store +const mockGetNodes = jest.fn() +jest.mock('reactflow', () => ({ + useStoreApi: () => ({ + getState: () => ({ + getNodes: mockGetNodes, + }), + }), +})) + +describe('Workflow Onboarding Integration Logic', () => { + const mockSetShowOnboarding = jest.fn() + const mockSetHasSelectedStartNode = jest.fn() + const mockSetHasShownOnboarding = jest.fn() + const mockSetShouldAutoOpenStartNodeSelector = jest.fn() + + beforeEach(() => { + jest.clearAllMocks() + + // Mock store implementation + ;(useWorkflowStore as jest.Mock).mockReturnValue({ + showOnboarding: false, + setShowOnboarding: mockSetShowOnboarding, + hasSelectedStartNode: false, + setHasSelectedStartNode: mockSetHasSelectedStartNode, + hasShownOnboarding: false, + setHasShownOnboarding: mockSetHasShownOnboarding, + notInitialWorkflow: false, + shouldAutoOpenStartNodeSelector: false, + setShouldAutoOpenStartNodeSelector: mockSetShouldAutoOpenStartNodeSelector, + }) + }) + + describe('Onboarding State Management', () => { + it('should initialize onboarding state correctly', () => { + const store = useWorkflowStore() + + expect(store.showOnboarding).toBe(false) + expect(store.hasSelectedStartNode).toBe(false) + expect(store.hasShownOnboarding).toBe(false) + }) + + it('should update onboarding visibility', () => { + const store = useWorkflowStore() + + store.setShowOnboarding(true) + expect(mockSetShowOnboarding).toHaveBeenCalledWith(true) + + store.setShowOnboarding(false) + expect(mockSetShowOnboarding).toHaveBeenCalledWith(false) + }) + + it('should track node selection state', () => { + const store = useWorkflowStore() + + store.setHasSelectedStartNode(true) + expect(mockSetHasSelectedStartNode).toHaveBeenCalledWith(true) + }) + + it('should track onboarding show state', () => { + const store = useWorkflowStore() + + store.setHasShownOnboarding(true) + expect(mockSetHasShownOnboarding).toHaveBeenCalledWith(true) + }) + }) + + describe('Node Validation Logic', () => { + /** + * Test the critical fix in use-nodes-sync-draft.ts + * This ensures trigger nodes are recognized as valid start nodes + */ + it('should validate Start node as valid start node', () => { + const mockNode = { + data: { type: BlockEnum.Start }, + id: 'start-1', + } + + // Simulate the validation logic from use-nodes-sync-draft.ts + const isValidStartNode = mockNode.data.type === BlockEnum.Start + || mockNode.data.type === BlockEnum.TriggerSchedule + || mockNode.data.type === BlockEnum.TriggerWebhook + || mockNode.data.type === BlockEnum.TriggerPlugin + + expect(isValidStartNode).toBe(true) + }) + + it('should validate TriggerSchedule as valid start node', () => { + const mockNode = { + data: { type: BlockEnum.TriggerSchedule }, + id: 'trigger-schedule-1', + } + + const isValidStartNode = mockNode.data.type === BlockEnum.Start + || mockNode.data.type === BlockEnum.TriggerSchedule + || mockNode.data.type === BlockEnum.TriggerWebhook + || mockNode.data.type === BlockEnum.TriggerPlugin + + expect(isValidStartNode).toBe(true) + }) + + it('should validate TriggerWebhook as valid start node', () => { + const mockNode = { + data: { type: BlockEnum.TriggerWebhook }, + id: 'trigger-webhook-1', + } + + const isValidStartNode = mockNode.data.type === BlockEnum.Start + || mockNode.data.type === BlockEnum.TriggerSchedule + || mockNode.data.type === BlockEnum.TriggerWebhook + || mockNode.data.type === BlockEnum.TriggerPlugin + + expect(isValidStartNode).toBe(true) + }) + + it('should validate TriggerPlugin as valid start node', () => { + const mockNode = { + data: { type: BlockEnum.TriggerPlugin }, + id: 'trigger-plugin-1', + } + + const isValidStartNode = mockNode.data.type === BlockEnum.Start + || mockNode.data.type === BlockEnum.TriggerSchedule + || mockNode.data.type === BlockEnum.TriggerWebhook + || mockNode.data.type === BlockEnum.TriggerPlugin + + expect(isValidStartNode).toBe(true) + }) + + it('should reject non-trigger nodes as invalid start nodes', () => { + const mockNode = { + data: { type: BlockEnum.LLM }, + id: 'llm-1', + } + + const isValidStartNode = mockNode.data.type === BlockEnum.Start + || mockNode.data.type === BlockEnum.TriggerSchedule + || mockNode.data.type === BlockEnum.TriggerWebhook + || mockNode.data.type === BlockEnum.TriggerPlugin + + expect(isValidStartNode).toBe(false) + }) + + it('should handle array of nodes with mixed types', () => { + const mockNodes = [ + { data: { type: BlockEnum.LLM }, id: 'llm-1' }, + { data: { type: BlockEnum.TriggerWebhook }, id: 'webhook-1' }, + { data: { type: BlockEnum.Answer }, id: 'answer-1' }, + ] + + // Simulate hasStartNode logic from use-nodes-sync-draft.ts + const hasStartNode = mockNodes.find(node => + node.data.type === BlockEnum.Start + || node.data.type === BlockEnum.TriggerSchedule + || node.data.type === BlockEnum.TriggerWebhook + || node.data.type === BlockEnum.TriggerPlugin, + ) + + expect(hasStartNode).toBeTruthy() + expect(hasStartNode?.id).toBe('webhook-1') + }) + + it('should return undefined when no valid start nodes exist', () => { + const mockNodes = [ + { data: { type: BlockEnum.LLM }, id: 'llm-1' }, + { data: { type: BlockEnum.Answer }, id: 'answer-1' }, + ] + + const hasStartNode = mockNodes.find(node => + node.data.type === BlockEnum.Start + || node.data.type === BlockEnum.TriggerSchedule + || node.data.type === BlockEnum.TriggerWebhook + || node.data.type === BlockEnum.TriggerPlugin, + ) + + expect(hasStartNode).toBeUndefined() + }) + }) + + describe('Auto-open Logic for Node Handles', () => { + /** + * Test the auto-open logic from node-handle.tsx + * This ensures all trigger types auto-open the block selector when flagged + */ + it('should auto-expand for Start node in new workflow', () => { + const shouldAutoOpenStartNodeSelector = true + const nodeType = BlockEnum.Start + const isChatMode = false + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + expect(shouldAutoExpand).toBe(true) + }) + + it('should auto-expand for TriggerSchedule in new workflow', () => { + const shouldAutoOpenStartNodeSelector = true + const nodeType = BlockEnum.TriggerSchedule + const isChatMode = false + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + expect(shouldAutoExpand).toBe(true) + }) + + it('should auto-expand for TriggerWebhook in new workflow', () => { + const shouldAutoOpenStartNodeSelector = true + const nodeType = BlockEnum.TriggerWebhook + const isChatMode = false + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + expect(shouldAutoExpand).toBe(true) + }) + + it('should auto-expand for TriggerPlugin in new workflow', () => { + const shouldAutoOpenStartNodeSelector = true + const nodeType = BlockEnum.TriggerPlugin + const isChatMode = false + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + expect(shouldAutoExpand).toBe(true) + }) + + it('should not auto-expand for non-trigger nodes', () => { + const shouldAutoOpenStartNodeSelector = true + const nodeType = BlockEnum.LLM + const isChatMode = false + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + expect(shouldAutoExpand).toBe(false) + }) + + it('should not auto-expand in chat mode', () => { + const shouldAutoOpenStartNodeSelector = true + const nodeType = BlockEnum.Start + const isChatMode = true + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + expect(shouldAutoExpand).toBe(false) + }) + + it('should not auto-expand for existing workflows', () => { + const shouldAutoOpenStartNodeSelector = false + const nodeType = BlockEnum.Start + const isChatMode = false + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + expect(shouldAutoExpand).toBe(false) + }) + it('should reset auto-open flag after triggering once', () => { + let shouldAutoOpenStartNodeSelector = true + const nodeType = BlockEnum.Start + const isChatMode = false + + const shouldAutoExpand = shouldAutoOpenStartNodeSelector && ( + nodeType === BlockEnum.Start + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + ) && !isChatMode + + if (shouldAutoExpand) + shouldAutoOpenStartNodeSelector = false + + expect(shouldAutoExpand).toBe(true) + expect(shouldAutoOpenStartNodeSelector).toBe(false) + }) + }) + + describe('Node Creation Without Auto-selection', () => { + /** + * Test that nodes are created without the 'selected: true' property + * This prevents auto-opening the properties panel + */ + it('should create Start node without auto-selection', () => { + const nodeData = { type: BlockEnum.Start, title: 'Start' } + + // Simulate node creation logic from workflow-children.tsx + const createdNodeData = { + ...nodeData, + // Note: 'selected: true' should NOT be added + } + + expect(createdNodeData.selected).toBeUndefined() + expect(createdNodeData.type).toBe(BlockEnum.Start) + }) + + it('should create TriggerWebhook node without auto-selection', () => { + const nodeData = { type: BlockEnum.TriggerWebhook, title: 'Webhook Trigger' } + const toolConfig = { webhook_url: 'https://example.com/webhook' } + + const createdNodeData = { + ...nodeData, + ...toolConfig, + // Note: 'selected: true' should NOT be added + } + + expect(createdNodeData.selected).toBeUndefined() + expect(createdNodeData.type).toBe(BlockEnum.TriggerWebhook) + expect(createdNodeData.webhook_url).toBe('https://example.com/webhook') + }) + + it('should preserve other node properties while avoiding auto-selection', () => { + const nodeData = { + type: BlockEnum.TriggerSchedule, + title: 'Schedule Trigger', + config: { interval: '1h' }, + } + + const createdNodeData = { + ...nodeData, + } + + expect(createdNodeData.selected).toBeUndefined() + expect(createdNodeData.type).toBe(BlockEnum.TriggerSchedule) + expect(createdNodeData.title).toBe('Schedule Trigger') + expect(createdNodeData.config).toEqual({ interval: '1h' }) + }) + }) + + describe('Workflow Initialization Logic', () => { + /** + * Test the initialization logic from use-workflow-init.ts + * This ensures onboarding is triggered correctly for new workflows + */ + it('should trigger onboarding for new workflow when draft does not exist', () => { + // Simulate the error handling logic from use-workflow-init.ts + const error = { + json: jest.fn().mockResolvedValue({ code: 'draft_workflow_not_exist' }), + bodyUsed: false, + } + + const mockWorkflowStore = { + setState: jest.fn(), + } + + // Simulate error handling + if (error && error.json && !error.bodyUsed) { + error.json().then((err: any) => { + if (err.code === 'draft_workflow_not_exist') { + mockWorkflowStore.setState({ + notInitialWorkflow: true, + showOnboarding: true, + }) + } + }) + } + + return error.json().then(() => { + expect(mockWorkflowStore.setState).toHaveBeenCalledWith({ + notInitialWorkflow: true, + showOnboarding: true, + }) + }) + }) + + it('should not trigger onboarding for existing workflows', () => { + // Simulate successful draft fetch + const mockWorkflowStore = { + setState: jest.fn(), + } + + // Normal initialization path should not set showOnboarding: true + mockWorkflowStore.setState({ + environmentVariables: [], + conversationVariables: [], + }) + + expect(mockWorkflowStore.setState).not.toHaveBeenCalledWith( + expect.objectContaining({ showOnboarding: true }), + ) + }) + + it('should create empty draft with proper structure', () => { + const mockSyncWorkflowDraft = jest.fn() + const appId = 'test-app-id' + + // Simulate the syncWorkflowDraft call from use-workflow-init.ts + const draftParams = { + url: `/apps/${appId}/workflows/draft`, + params: { + graph: { + nodes: [], // Empty nodes initially + edges: [], + }, + features: { + retriever_resource: { enabled: true }, + }, + environment_variables: [], + conversation_variables: [], + }, + } + + mockSyncWorkflowDraft(draftParams) + + expect(mockSyncWorkflowDraft).toHaveBeenCalledWith({ + url: `/apps/${appId}/workflows/draft`, + params: { + graph: { + nodes: [], + edges: [], + }, + features: { + retriever_resource: { enabled: true }, + }, + environment_variables: [], + conversation_variables: [], + }, + }) + }) + }) + + describe('Auto-Detection for Empty Canvas', () => { + beforeEach(() => { + mockGetNodes.mockClear() + }) + + it('should detect empty canvas and trigger onboarding', () => { + // Mock empty canvas + mockGetNodes.mockReturnValue([]) + + // Mock store with proper state for auto-detection + ;(useWorkflowStore as jest.Mock).mockReturnValue({ + showOnboarding: false, + hasShownOnboarding: false, + notInitialWorkflow: false, + setShowOnboarding: mockSetShowOnboarding, + setHasShownOnboarding: mockSetHasShownOnboarding, + hasSelectedStartNode: false, + setHasSelectedStartNode: mockSetHasSelectedStartNode, + shouldAutoOpenStartNodeSelector: false, + setShouldAutoOpenStartNodeSelector: mockSetShouldAutoOpenStartNodeSelector, + getState: () => ({ + showOnboarding: false, + hasShownOnboarding: false, + notInitialWorkflow: false, + setShowOnboarding: mockSetShowOnboarding, + setHasShownOnboarding: mockSetHasShownOnboarding, + hasSelectedStartNode: false, + setHasSelectedStartNode: mockSetHasSelectedStartNode, + setShouldAutoOpenStartNodeSelector: mockSetShouldAutoOpenStartNodeSelector, + }), + }) + + // Simulate empty canvas check logic + const nodes = mockGetNodes() + const startNodeTypes = [ + BlockEnum.Start, + BlockEnum.TriggerSchedule, + BlockEnum.TriggerWebhook, + BlockEnum.TriggerPlugin, + ] + const hasStartNode = nodes.some(node => startNodeTypes.includes(node.data?.type)) + const isEmpty = nodes.length === 0 || !hasStartNode + + expect(isEmpty).toBe(true) + expect(nodes.length).toBe(0) + }) + + it('should detect canvas with non-start nodes as empty', () => { + // Mock canvas with non-start nodes + mockGetNodes.mockReturnValue([ + { id: '1', data: { type: BlockEnum.LLM } }, + { id: '2', data: { type: BlockEnum.Code } }, + ]) + + const nodes = mockGetNodes() + const startNodeTypes = [ + BlockEnum.Start, + BlockEnum.TriggerSchedule, + BlockEnum.TriggerWebhook, + BlockEnum.TriggerPlugin, + ] + const hasStartNode = nodes.some(node => startNodeTypes.includes(node.data.type)) + const isEmpty = nodes.length === 0 || !hasStartNode + + expect(isEmpty).toBe(true) + expect(hasStartNode).toBe(false) + }) + + it('should not detect canvas with start nodes as empty', () => { + // Mock canvas with start node + mockGetNodes.mockReturnValue([ + { id: '1', data: { type: BlockEnum.Start } }, + ]) + + const nodes = mockGetNodes() + const startNodeTypes = [ + BlockEnum.Start, + BlockEnum.TriggerSchedule, + BlockEnum.TriggerWebhook, + BlockEnum.TriggerPlugin, + ] + const hasStartNode = nodes.some(node => startNodeTypes.includes(node.data.type)) + const isEmpty = nodes.length === 0 || !hasStartNode + + expect(isEmpty).toBe(false) + expect(hasStartNode).toBe(true) + }) + + it('should not trigger onboarding if already shown in session', () => { + // Mock empty canvas + mockGetNodes.mockReturnValue([]) + + // Mock store with hasShownOnboarding = true + ;(useWorkflowStore as jest.Mock).mockReturnValue({ + showOnboarding: false, + hasShownOnboarding: true, // Already shown in this session + notInitialWorkflow: false, + setShowOnboarding: mockSetShowOnboarding, + setHasShownOnboarding: mockSetHasShownOnboarding, + hasSelectedStartNode: false, + setHasSelectedStartNode: mockSetHasSelectedStartNode, + shouldAutoOpenStartNodeSelector: false, + setShouldAutoOpenStartNodeSelector: mockSetShouldAutoOpenStartNodeSelector, + getState: () => ({ + showOnboarding: false, + hasShownOnboarding: true, + notInitialWorkflow: false, + setShowOnboarding: mockSetShowOnboarding, + setHasShownOnboarding: mockSetHasShownOnboarding, + hasSelectedStartNode: false, + setHasSelectedStartNode: mockSetHasSelectedStartNode, + setShouldAutoOpenStartNodeSelector: mockSetShouldAutoOpenStartNodeSelector, + }), + }) + + // Simulate the check logic with hasShownOnboarding = true + const store = useWorkflowStore() + const shouldTrigger = !store.hasShownOnboarding && !store.showOnboarding && !store.notInitialWorkflow + + expect(shouldTrigger).toBe(false) + }) + + it('should not trigger onboarding during initial workflow creation', () => { + // Mock empty canvas + mockGetNodes.mockReturnValue([]) + + // Mock store with notInitialWorkflow = true (initial creation) + ;(useWorkflowStore as jest.Mock).mockReturnValue({ + showOnboarding: false, + hasShownOnboarding: false, + notInitialWorkflow: true, // Initial workflow creation + setShowOnboarding: mockSetShowOnboarding, + setHasShownOnboarding: mockSetHasShownOnboarding, + hasSelectedStartNode: false, + setHasSelectedStartNode: mockSetHasSelectedStartNode, + shouldAutoOpenStartNodeSelector: false, + setShouldAutoOpenStartNodeSelector: mockSetShouldAutoOpenStartNodeSelector, + getState: () => ({ + showOnboarding: false, + hasShownOnboarding: false, + notInitialWorkflow: true, + setShowOnboarding: mockSetShowOnboarding, + setHasShownOnboarding: mockSetHasShownOnboarding, + hasSelectedStartNode: false, + setHasSelectedStartNode: mockSetHasSelectedStartNode, + setShouldAutoOpenStartNodeSelector: mockSetShouldAutoOpenStartNodeSelector, + }), + }) + + // Simulate the check logic with notInitialWorkflow = true + const store = useWorkflowStore() + const shouldTrigger = !store.hasShownOnboarding && !store.showOnboarding && !store.notInitialWorkflow + + expect(shouldTrigger).toBe(false) + }) + }) +}) diff --git a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/layout-main.tsx b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/layout-main.tsx index a36a7e281d..1f836de6e6 100644 --- a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/layout-main.tsx +++ b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/layout-main.tsx @@ -24,7 +24,7 @@ import { fetchAppDetailDirect } from '@/service/apps' import { useAppContext } from '@/context/app-context' import Loading from '@/app/components/base/loading' import useBreakpoints, { MediaType } from '@/hooks/use-breakpoints' -import type { App } from '@/types/app' +import { type App, AppModeEnum } from '@/types/app' import useDocumentTitle from '@/hooks/use-document-title' import { useStore as useTagStore } from '@/app/components/base/tag-management/store' import dynamic from 'next/dynamic' @@ -64,12 +64,12 @@ const AppDetailLayout: FC = (props) => { selectedIcon: NavIcon }>>([]) - const getNavigationConfig = useCallback((appId: string, isCurrentWorkspaceEditor: boolean, mode: string) => { + const getNavigationConfig = useCallback((appId: string, isCurrentWorkspaceEditor: boolean, mode: AppModeEnum) => { const navConfig = [ ...(isCurrentWorkspaceEditor ? [{ name: t('common.appMenus.promptEng'), - href: `/app/${appId}/${(mode === 'workflow' || mode === 'advanced-chat') ? 'workflow' : 'configuration'}`, + href: `/app/${appId}/${(mode === AppModeEnum.WORKFLOW || mode === AppModeEnum.ADVANCED_CHAT) ? 'workflow' : 'configuration'}`, icon: RiTerminalWindowLine, selectedIcon: RiTerminalWindowFill, }] @@ -83,7 +83,7 @@ const AppDetailLayout: FC = (props) => { }, ...(isCurrentWorkspaceEditor ? [{ - name: mode !== 'workflow' + name: mode !== AppModeEnum.WORKFLOW ? t('common.appMenus.logAndAnn') : t('common.appMenus.logs'), href: `/app/${appId}/logs`, @@ -110,7 +110,7 @@ const AppDetailLayout: FC = (props) => { const mode = isMobile ? 'collapse' : 'expand' setAppSidebarExpand(isMobile ? mode : localeMode) // TODO: consider screen size and mode - // if ((appDetail.mode === 'advanced-chat' || appDetail.mode === 'workflow') && (pathname).endsWith('workflow')) + // if ((appDetail.mode === AppModeEnum.ADVANCED_CHAT || appDetail.mode === 'workflow') && (pathname).endsWith('workflow')) // setAppSidebarExpand('collapse') } }, [appDetail, isMobile]) @@ -138,10 +138,10 @@ const AppDetailLayout: FC = (props) => { router.replace(`/app/${appId}/overview`) return } - if ((res.mode === 'workflow' || res.mode === 'advanced-chat') && (pathname).endsWith('configuration')) { + if ((res.mode === AppModeEnum.WORKFLOW || res.mode === AppModeEnum.ADVANCED_CHAT) && (pathname).endsWith('configuration')) { router.replace(`/app/${appId}/workflow`) } - else if ((res.mode !== 'workflow' && res.mode !== 'advanced-chat') && (pathname).endsWith('workflow')) { + else if ((res.mode !== AppModeEnum.WORKFLOW && res.mode !== AppModeEnum.ADVANCED_CHAT) && (pathname).endsWith('workflow')) { router.replace(`/app/${appId}/configuration`) } else { diff --git a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/card-view.tsx b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/card-view.tsx index e58e79918f..57f3ef6881 100644 --- a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/card-view.tsx +++ b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/card-view.tsx @@ -1,11 +1,12 @@ 'use client' import type { FC } from 'react' -import React from 'react' +import React, { useMemo } from 'react' import { useTranslation } from 'react-i18next' import { useContext } from 'use-context-selector' import AppCard from '@/app/components/app/overview/app-card' import Loading from '@/app/components/base/loading' import MCPServiceCard from '@/app/components/tools/mcp/mcp-service-card' +import TriggerCard from '@/app/components/app/overview/trigger-card' import { ToastContext } from '@/app/components/base/toast' import { fetchAppDetail, @@ -14,11 +15,15 @@ import { updateAppSiteStatus, } from '@/service/apps' import type { App } from '@/types/app' +import { AppModeEnum } from '@/types/app' import type { UpdateAppSiteCodeResponse } from '@/models/app' import { asyncRunSafe } from '@/utils' import { NEED_REFRESH_APP_LIST_KEY } from '@/config' import type { IAppCardProps } from '@/app/components/app/overview/app-card' import { useStore as useAppStore } from '@/app/components/app/store' +import { useAppWorkflow } from '@/service/use-workflow' +import type { BlockEnum } from '@/app/components/workflow/types' +import { isTriggerNode } from '@/app/components/workflow/types' export type ICardViewProps = { appId: string @@ -32,7 +37,22 @@ const CardView: FC = ({ appId, isInPanel, className }) => { const appDetail = useAppStore(state => state.appDetail) const setAppDetail = useAppStore(state => state.setAppDetail) + const isWorkflowApp = appDetail?.mode === AppModeEnum.WORKFLOW const showMCPCard = isInPanel + const showTriggerCard = isInPanel && isWorkflowApp + const { data: currentWorkflow } = useAppWorkflow(isWorkflowApp ? appDetail.id : '') + const hasTriggerNode = useMemo(() => { + if (!isWorkflowApp) + return false + if (!currentWorkflow) + return null + const nodes = currentWorkflow.graph?.nodes || [] + return nodes.some((node) => { + const nodeType = node.data?.type as BlockEnum | undefined + return !!nodeType && isTriggerNode(nodeType) + }) + }, [isWorkflowApp, currentWorkflow]) + const shouldRenderAppCards = !isWorkflowApp || hasTriggerNode === false const updateAppDetail = async () => { try { @@ -106,23 +126,35 @@ const CardView: FC = ({ appId, isInPanel, className }) => { return (
- - - {showMCPCard && ( - + + + {showMCPCard && ( + + )} + + ) + } + {showTriggerCard && ( + )}
diff --git a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/chart-view.tsx b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/chart-view.tsx index 847de19165..64cd2fbd28 100644 --- a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/chart-view.tsx +++ b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/chart-view.tsx @@ -5,15 +5,22 @@ import quarterOfYear from 'dayjs/plugin/quarterOfYear' import { useTranslation } from 'react-i18next' import type { PeriodParams } from '@/app/components/app/overview/app-chart' import { AvgResponseTime, AvgSessionInteractions, AvgUserInteractions, ConversationsChart, CostChart, EndUsersChart, MessagesChart, TokenPerSecond, UserSatisfactionRate, WorkflowCostChart, WorkflowDailyTerminalsChart, WorkflowMessagesChart } from '@/app/components/app/overview/app-chart' -import type { Item } from '@/app/components/base/select' -import { SimpleSelect } from '@/app/components/base/select' -import { TIME_PERIOD_MAPPING } from '@/app/components/app/log/filter' import { useStore as useAppStore } from '@/app/components/app/store' +import TimeRangePicker from './time-range-picker' +import { TIME_PERIOD_MAPPING as LONG_TIME_PERIOD_MAPPING } from '@/app/components/app/log/filter' +import { IS_CLOUD_EDITION } from '@/config' +import LongTimeRangePicker from './long-time-range-picker' dayjs.extend(quarterOfYear) const today = dayjs() +const TIME_PERIOD_MAPPING = [ + { value: 0, name: 'today' }, + { value: 7, name: 'last7days' }, + { value: 30, name: 'last30days' }, +] + const queryDateFormat = 'YYYY-MM-DD HH:mm' export type IChartViewProps = { @@ -26,21 +33,10 @@ export default function ChartView({ appId, headerRight }: IChartViewProps) { const appDetail = useAppStore(state => state.appDetail) const isChatApp = appDetail?.mode !== 'completion' && appDetail?.mode !== 'workflow' const isWorkflow = appDetail?.mode === 'workflow' - const [period, setPeriod] = useState({ name: t('appLog.filter.period.last7days'), query: { start: today.subtract(7, 'day').startOf('day').format(queryDateFormat), end: today.endOf('day').format(queryDateFormat) } }) - - const onSelect = (item: Item) => { - if (item.value === -1) { - setPeriod({ name: item.name, query: undefined }) - } - else if (item.value === 0) { - const startOfToday = today.startOf('day').format(queryDateFormat) - const endOfToday = today.endOf('day').format(queryDateFormat) - setPeriod({ name: item.name, query: { start: startOfToday, end: endOfToday } }) - } - else { - setPeriod({ name: item.name, query: { start: today.subtract(item.value as number, 'day').startOf('day').format(queryDateFormat), end: today.endOf('day').format(queryDateFormat) } }) - } - } + const [period, setPeriod] = useState(IS_CLOUD_EDITION + ? { name: t('appLog.filter.period.today'), query: { start: today.startOf('day').format(queryDateFormat), end: today.endOf('day').format(queryDateFormat) } } + : { name: t('appLog.filter.period.last7days'), query: { start: today.subtract(7, 'day').startOf('day').format(queryDateFormat), end: today.endOf('day').format(queryDateFormat) } }, + ) if (!appDetail) return null @@ -50,20 +46,20 @@ export default function ChartView({ appId, headerRight }: IChartViewProps) {
{t('common.appMenus.overview')}
-
- ({ value: k, name: t(`appLog.filter.period.${v.name}`) }))} - className='mt-0 !w-40' - notClearable={true} - onSelect={(item) => { - const id = item.value - const value = TIME_PERIOD_MAPPING[id]?.value ?? '-1' - const name = item.name || t('appLog.filter.period.allTime') - onSelect({ value, name }) - }} - defaultValue={'2'} + {IS_CLOUD_EDITION ? ( + -
+ ) : ( + + )} + {headerRight}
diff --git a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/long-time-range-picker.tsx b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/long-time-range-picker.tsx new file mode 100644 index 0000000000..cad4d41a0e --- /dev/null +++ b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/long-time-range-picker.tsx @@ -0,0 +1,63 @@ +'use client' +import type { PeriodParams } from '@/app/components/app/overview/app-chart' +import type { FC } from 'react' +import React from 'react' +import type { Item } from '@/app/components/base/select' +import { SimpleSelect } from '@/app/components/base/select' +import { useTranslation } from 'react-i18next' +import dayjs from 'dayjs' +type Props = { + periodMapping: { [key: string]: { value: number; name: string } } + onSelect: (payload: PeriodParams) => void + queryDateFormat: string +} + +const today = dayjs() + +const LongTimeRangePicker: FC = ({ + periodMapping, + onSelect, + queryDateFormat, +}) => { + const { t } = useTranslation() + + const handleSelect = React.useCallback((item: Item) => { + const id = item.value + const value = periodMapping[id]?.value ?? '-1' + const name = item.name || t('appLog.filter.period.allTime') + if (value === -1) { + onSelect({ name: t('appLog.filter.period.allTime'), query: undefined }) + } + else if (value === 0) { + const startOfToday = today.startOf('day').format(queryDateFormat) + const endOfToday = today.endOf('day').format(queryDateFormat) + onSelect({ + name, + query: { + start: startOfToday, + end: endOfToday, + }, + }) + } + else { + onSelect({ + name, + query: { + start: today.subtract(value as number, 'day').startOf('day').format(queryDateFormat), + end: today.endOf('day').format(queryDateFormat), + }, + }) + } + }, [onSelect, periodMapping, queryDateFormat, t]) + + return ( + ({ value: k, name: t(`appLog.filter.period.${v.name}`) }))} + className='mt-0 !w-40' + notClearable={true} + onSelect={handleSelect} + defaultValue={'2'} + /> + ) +} +export default React.memo(LongTimeRangePicker) diff --git a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/date-picker.tsx b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/date-picker.tsx new file mode 100644 index 0000000000..2bfdece433 --- /dev/null +++ b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/date-picker.tsx @@ -0,0 +1,80 @@ +'use client' +import { RiCalendarLine } from '@remixicon/react' +import type { Dayjs } from 'dayjs' +import type { FC } from 'react' +import React, { useCallback } from 'react' +import cn from '@/utils/classnames' +import { formatToLocalTime } from '@/utils/format' +import { useI18N } from '@/context/i18n' +import Picker from '@/app/components/base/date-and-time-picker/date-picker' +import type { TriggerProps } from '@/app/components/base/date-and-time-picker/types' +import { noop } from 'lodash-es' +import dayjs from 'dayjs' + +type Props = { + start: Dayjs + end: Dayjs + onStartChange: (date?: Dayjs) => void + onEndChange: (date?: Dayjs) => void +} + +const today = dayjs() +const DatePicker: FC = ({ + start, + end, + onStartChange, + onEndChange, +}) => { + const { locale } = useI18N() + + const renderDate = useCallback(({ value, handleClickTrigger, isOpen }: TriggerProps) => { + return ( +
+ {value ? formatToLocalTime(value, locale, 'MMM D') : ''} +
+ ) + }, [locale]) + + const availableStartDate = end.subtract(30, 'day') + const startDateDisabled = useCallback((date: Dayjs) => { + if (date.isAfter(today, 'date')) + return true + return !((date.isAfter(availableStartDate, 'date') || date.isSame(availableStartDate, 'date')) && (date.isBefore(end, 'date') || date.isSame(end, 'date'))) + }, [availableStartDate, end]) + + const availableEndDate = start.add(30, 'day') + const endDateDisabled = useCallback((date: Dayjs) => { + if (date.isAfter(today, 'date')) + return true + return !((date.isAfter(start, 'date') || date.isSame(start, 'date')) && (date.isBefore(availableEndDate, 'date') || date.isSame(availableEndDate, 'date'))) + }, [availableEndDate, start]) + + return ( +
+
+ +
+ + - + +
+ + ) +} +export default React.memo(DatePicker) diff --git a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/index.tsx b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/index.tsx new file mode 100644 index 0000000000..4738bdeebf --- /dev/null +++ b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/index.tsx @@ -0,0 +1,86 @@ +'use client' +import type { PeriodParams, PeriodParamsWithTimeRange } from '@/app/components/app/overview/app-chart' +import type { FC } from 'react' +import React, { useCallback, useState } from 'react' +import type { Dayjs } from 'dayjs' +import { HourglassShape } from '@/app/components/base/icons/src/vender/other' +import RangeSelector from './range-selector' +import DatePicker from './date-picker' +import dayjs from 'dayjs' +import { useI18N } from '@/context/i18n' +import { formatToLocalTime } from '@/utils/format' + +const today = dayjs() + +type Props = { + ranges: { value: number; name: string }[] + onSelect: (payload: PeriodParams) => void + queryDateFormat: string +} + +const TimeRangePicker: FC = ({ + ranges, + onSelect, + queryDateFormat, +}) => { + const { locale } = useI18N() + + const [isCustomRange, setIsCustomRange] = useState(false) + const [start, setStart] = useState(today) + const [end, setEnd] = useState(today) + + const handleRangeChange = useCallback((payload: PeriodParamsWithTimeRange) => { + setIsCustomRange(false) + setStart(payload.query!.start) + setEnd(payload.query!.end) + onSelect({ + name: payload.name, + query: { + start: payload.query!.start.format(queryDateFormat), + end: payload.query!.end.format(queryDateFormat), + }, + }) + }, [onSelect, queryDateFormat]) + + const handleDateChange = useCallback((type: 'start' | 'end') => { + return (date?: Dayjs) => { + if (!date) return + if (type === 'start' && date.isSame(start)) return + if (type === 'end' && date.isSame(end)) return + if (type === 'start') + setStart(date) + else + setEnd(date) + + const currStart = type === 'start' ? date : start + const currEnd = type === 'end' ? date : end + onSelect({ + name: `${formatToLocalTime(currStart, locale, 'MMM D')} - ${formatToLocalTime(currEnd, locale, 'MMM D')}`, + query: { + start: currStart.format(queryDateFormat), + end: currEnd.format(queryDateFormat), + }, + }) + + setIsCustomRange(true) + } + }, [start, end, onSelect, locale, queryDateFormat]) + + return ( +
+ + + +
+ ) +} +export default React.memo(TimeRangePicker) diff --git a/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/range-selector.tsx b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/range-selector.tsx new file mode 100644 index 0000000000..f99ea52492 --- /dev/null +++ b/web/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/time-range-picker/range-selector.tsx @@ -0,0 +1,81 @@ +'use client' +import type { PeriodParamsWithTimeRange, TimeRange } from '@/app/components/app/overview/app-chart' +import type { FC } from 'react' +import React, { useCallback } from 'react' +import { SimpleSelect } from '@/app/components/base/select' +import type { Item } from '@/app/components/base/select' +import dayjs from 'dayjs' +import { RiArrowDownSLine, RiCheckLine } from '@remixicon/react' +import cn from '@/utils/classnames' +import { useTranslation } from 'react-i18next' + +const today = dayjs() + +type Props = { + isCustomRange: boolean + ranges: { value: number; name: string }[] + onSelect: (payload: PeriodParamsWithTimeRange) => void +} + +const RangeSelector: FC = ({ + isCustomRange, + ranges, + onSelect, +}) => { + const { t } = useTranslation() + + const handleSelectRange = useCallback((item: Item) => { + const { name, value } = item + let period: TimeRange | null = null + if (value === 0) { + const startOfToday = today.startOf('day') + const endOfToday = today.endOf('day') + period = { start: startOfToday, end: endOfToday } + } + else { + period = { start: today.subtract(item.value as number, 'day').startOf('day'), end: today.endOf('day') } + } + onSelect({ query: period!, name }) + }, [onSelect]) + + const renderTrigger = useCallback((item: Item | null, isOpen: boolean) => { + return ( +
+
{isCustomRange ? t('appLog.filter.period.custom') : item?.name}
+ +
+ ) + }, [isCustomRange]) + + const renderOption = useCallback(({ item, selected }: { item: Item; selected: boolean }) => { + return ( + <> + {selected && ( + + + )} + {item.name} + + ) + }, []) + return ( + ({ ...v, name: t(`appLog.filter.period.${v.name}`) }))} + className='mt-0 !w-40' + notClearable={true} + onSelect={handleSelectRange} + defaultValue={0} + wrapperClassName='h-8' + optionWrapClassName='w-[200px] translate-x-[-24px]' + renderTrigger={renderTrigger} + optionClassName='flex items-center py-0 pl-7 pr-2 h-8' + renderOption={renderOption} + /> + ) +} +export default React.memo(RangeSelector) diff --git a/web/app/(commonLayout)/explore/installed/[appId]/page.tsx b/web/app/(commonLayout)/explore/installed/[appId]/page.tsx index e288c62b5d..983fdb9d23 100644 --- a/web/app/(commonLayout)/explore/installed/[appId]/page.tsx +++ b/web/app/(commonLayout)/explore/installed/[appId]/page.tsx @@ -2,14 +2,14 @@ import React from 'react' import Main from '@/app/components/explore/installed-app' export type IInstalledAppProps = { - params: { + params?: Promise<{ appId: string - } + }> } // Using Next.js page convention for async server components async function InstalledApp({ params }: IInstalledAppProps) { - const appId = (await params).appId + const { appId } = await (params ?? Promise.reject(new Error('Missing params'))) return (
) diff --git a/web/app/(commonLayout)/layout.tsx b/web/app/(commonLayout)/layout.tsx index be9c4fe49a..7f6bbb1f52 100644 --- a/web/app/(commonLayout)/layout.tsx +++ b/web/app/(commonLayout)/layout.tsx @@ -10,6 +10,8 @@ import { ProviderContextProvider } from '@/context/provider-context' import { ModalContextProvider } from '@/context/modal-context' import GotoAnything from '@/app/components/goto-anything' import Zendesk from '@/app/components/base/zendesk' +import ReadmePanel from '@/app/components/plugins/readme-panel' +import Splash from '../components/splash' const Layout = ({ children }: { children: ReactNode }) => { return ( @@ -24,7 +26,9 @@ const Layout = ({ children }: { children: ReactNode }) => {
{children} + + diff --git a/web/app/(shareLayout)/components/splash.tsx b/web/app/(shareLayout)/components/splash.tsx index 16d291d4b4..eb9538e49b 100644 --- a/web/app/(shareLayout)/components/splash.tsx +++ b/web/app/(shareLayout)/components/splash.tsx @@ -15,6 +15,7 @@ const Splash: FC = ({ children }) => { const { t } = useTranslation() const shareCode = useWebAppStore(s => s.shareCode) const webAppAccessMode = useWebAppStore(s => s.webAppAccessMode) + const embeddedUserId = useWebAppStore(s => s.embeddedUserId) const searchParams = useSearchParams() const router = useRouter() const redirectUrl = searchParams.get('redirect_url') @@ -69,11 +70,14 @@ const Splash: FC = ({ children }) => { } else if (userLoggedIn && !appLoggedIn) { try { - const { access_token } = await fetchAccessToken({ appCode: shareCode! }) + const { access_token } = await fetchAccessToken({ + appCode: shareCode!, + userId: embeddedUserId || undefined, + }) setWebAppPassport(shareCode!, access_token) redirectOrFinish() } - catch (error) { + catch { await webAppLogout(shareCode!) proceedToAuth() } @@ -85,7 +89,8 @@ const Splash: FC = ({ children }) => { router, message, webAppAccessMode, - tokenFromUrl]) + tokenFromUrl, + embeddedUserId]) if (message) { return
diff --git a/web/app/(shareLayout)/webapp-signin/check-code/page.tsx b/web/app/(shareLayout)/webapp-signin/check-code/page.tsx index 4a1326fedf..69131cdabe 100644 --- a/web/app/(shareLayout)/webapp-signin/check-code/page.tsx +++ b/web/app/(shareLayout)/webapp-signin/check-code/page.tsx @@ -12,6 +12,7 @@ import { sendWebAppEMailLoginCode, webAppEmailLoginWithCode } from '@/service/co import I18NContext from '@/context/i18n' import { setWebAppAccessToken, setWebAppPassport } from '@/service/webapp-auth' import { fetchAccessToken } from '@/service/share' +import { useWebAppStore } from '@/context/web-app-context' export default function CheckCode() { const { t } = useTranslation() @@ -23,6 +24,7 @@ export default function CheckCode() { const [loading, setIsLoading] = useState(false) const { locale } = useContext(I18NContext) const redirectUrl = searchParams.get('redirect_url') + const embeddedUserId = useWebAppStore(s => s.embeddedUserId) const getAppCodeFromRedirectUrl = useCallback(() => { if (!redirectUrl) @@ -63,7 +65,10 @@ export default function CheckCode() { const ret = await webAppEmailLoginWithCode({ email, code, token }) if (ret.result === 'success') { setWebAppAccessToken(ret.data.access_token) - const { access_token } = await fetchAccessToken({ appCode: appCode! }) + const { access_token } = await fetchAccessToken({ + appCode: appCode!, + userId: embeddedUserId || undefined, + }) setWebAppPassport(appCode!, access_token) router.replace(decodeURIComponent(redirectUrl)) } diff --git a/web/app/(shareLayout)/webapp-signin/components/mail-and-password-auth.tsx b/web/app/(shareLayout)/webapp-signin/components/mail-and-password-auth.tsx index ce220b103e..0136445ac9 100644 --- a/web/app/(shareLayout)/webapp-signin/components/mail-and-password-auth.tsx +++ b/web/app/(shareLayout)/webapp-signin/components/mail-and-password-auth.tsx @@ -10,6 +10,7 @@ import { emailRegex } from '@/config' import { webAppLogin } from '@/service/common' import Input from '@/app/components/base/input' import I18NContext from '@/context/i18n' +import { useWebAppStore } from '@/context/web-app-context' import { noop } from 'lodash-es' import { fetchAccessToken } from '@/service/share' import { setWebAppAccessToken, setWebAppPassport } from '@/service/webapp-auth' @@ -30,6 +31,7 @@ export default function MailAndPasswordAuth({ isEmailSetup }: MailAndPasswordAut const [isLoading, setIsLoading] = useState(false) const redirectUrl = searchParams.get('redirect_url') + const embeddedUserId = useWebAppStore(s => s.embeddedUserId) const getAppCodeFromRedirectUrl = useCallback(() => { if (!redirectUrl) @@ -82,7 +84,10 @@ export default function MailAndPasswordAuth({ isEmailSetup }: MailAndPasswordAut if (res.result === 'success') { setWebAppAccessToken(res.data.access_token) - const { access_token } = await fetchAccessToken({ appCode: appCode! }) + const { access_token } = await fetchAccessToken({ + appCode: appCode!, + userId: embeddedUserId || undefined, + }) setWebAppPassport(appCode!, access_token) router.replace(decodeURIComponent(redirectUrl)) } diff --git a/web/app/account/oauth/authorize/constants.ts b/web/app/account/oauth/authorize/constants.ts new file mode 100644 index 0000000000..f1d8b98ef4 --- /dev/null +++ b/web/app/account/oauth/authorize/constants.ts @@ -0,0 +1,3 @@ +export const OAUTH_AUTHORIZE_PENDING_KEY = 'oauth_authorize_pending' +export const REDIRECT_URL_KEY = 'oauth_redirect_url' +export const OAUTH_AUTHORIZE_PENDING_TTL = 60 * 3 diff --git a/web/app/account/oauth/authorize/page.tsx b/web/app/account/oauth/authorize/page.tsx index 4aa5fa0b8e..c9b26b97c1 100644 --- a/web/app/account/oauth/authorize/page.tsx +++ b/web/app/account/oauth/authorize/page.tsx @@ -19,11 +19,11 @@ import { } from '@remixicon/react' import dayjs from 'dayjs' import { useIsLogin } from '@/service/use-common' - -export const OAUTH_AUTHORIZE_PENDING_KEY = 'oauth_authorize_pending' -export const REDIRECT_URL_KEY = 'oauth_redirect_url' - -const OAUTH_AUTHORIZE_PENDING_TTL = 60 * 3 +import { + OAUTH_AUTHORIZE_PENDING_KEY, + OAUTH_AUTHORIZE_PENDING_TTL, + REDIRECT_URL_KEY, +} from './constants' function setItemWithExpiry(key: string, value: string, ttl: number) { const item = { diff --git a/web/app/components/app-sidebar/app-info.tsx b/web/app/components/app-sidebar/app-info.tsx index baf52946df..c2bda8d8fc 100644 --- a/web/app/components/app-sidebar/app-info.tsx +++ b/web/app/components/app-sidebar/app-info.tsx @@ -26,11 +26,11 @@ import { fetchWorkflowDraft } from '@/service/workflow' import ContentDialog from '@/app/components/base/content-dialog' import Button from '@/app/components/base/button' import CardView from '@/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/card-view' -import Divider from '../base/divider' import type { Operation } from './app-operations' import AppOperations from './app-operations' import dynamic from 'next/dynamic' import cn from '@/utils/classnames' +import { AppModeEnum } from '@/types/app' const SwitchAppModal = dynamic(() => import('@/app/components/app/switch-app-modal'), { ssr: false, @@ -158,7 +158,7 @@ const AppInfo = ({ expand, onlyShowDetail = false, openState = false, onDetailEx const exportCheck = async () => { if (!appDetail) return - if (appDetail.mode !== 'workflow' && appDetail.mode !== 'advanced-chat') { + if (appDetail.mode !== AppModeEnum.WORKFLOW && appDetail.mode !== AppModeEnum.ADVANCED_CHAT) { onExport() return } @@ -208,7 +208,7 @@ const AppInfo = ({ expand, onlyShowDetail = false, openState = false, onDetailEx if (!appDetail) return null - const operations = [ + const primaryOperations = [ { id: 'edit', title: t('app.editApp'), @@ -235,7 +235,11 @@ const AppInfo = ({ expand, onlyShowDetail = false, openState = false, onDetailEx icon: , onClick: exportCheck, }, - (appDetail.mode !== 'agent-chat' && (appDetail.mode === 'advanced-chat' || appDetail.mode === 'workflow')) ? { + ] + + const secondaryOperations: Operation[] = [ + // Import DSL (conditional) + ...(appDetail.mode !== AppModeEnum.AGENT_CHAT && (appDetail.mode === AppModeEnum.ADVANCED_CHAT || appDetail.mode === AppModeEnum.WORKFLOW)) ? [{ id: 'import', title: t('workflow.common.importDSL'), icon: , @@ -244,18 +248,39 @@ const AppInfo = ({ expand, onlyShowDetail = false, openState = false, onDetailEx onDetailExpand?.(false) setShowImportDSLModal(true) }, - } : undefined, - (appDetail.mode !== 'agent-chat' && (appDetail.mode === 'completion' || appDetail.mode === 'chat')) ? { - id: 'switch', - title: t('app.switch'), - icon: , + }] : [], + // Divider + { + id: 'divider-1', + title: '', + icon: <>, + onClick: () => { /* divider has no action */ }, + type: 'divider' as const, + }, + // Delete operation + { + id: 'delete', + title: t('common.operation.delete'), + icon: , onClick: () => { setOpen(false) onDetailExpand?.(false) - setShowSwitchModal(true) + setShowConfirmDelete(true) }, - } : undefined, - ].filter((op): op is Operation => Boolean(op)) + }, + ] + + // Keep the switch operation separate as it's not part of the main operations + const switchOperation = (appDetail.mode !== AppModeEnum.AGENT_CHAT && (appDetail.mode === AppModeEnum.COMPLETION || appDetail.mode === AppModeEnum.CHAT)) ? { + id: 'switch', + title: t('app.switch'), + icon: , + onClick: () => { + setOpen(false) + onDetailExpand?.(false) + setShowSwitchModal(true) + }, + } : null return (
@@ -298,7 +323,12 @@ const AppInfo = ({ expand, onlyShowDetail = false, openState = false, onDetailEx
{appDetail.name}
-
{appDetail.mode === 'advanced-chat' ? t('app.types.advanced') : appDetail.mode === 'agent-chat' ? t('app.types.agent') : appDetail.mode === 'chat' ? t('app.types.chatbot') : appDetail.mode === 'completion' ? t('app.types.completion') : t('app.types.workflow')}
+
+ {appDetail.mode === AppModeEnum.ADVANCED_CHAT ? t('app.types.advanced') + : appDetail.mode === AppModeEnum.AGENT_CHAT ? t('app.types.agent') + : appDetail.mode === AppModeEnum.CHAT ? t('app.types.chatbot') + : appDetail.mode === AppModeEnum.COMPLETION ? t('app.types.completion') + : t('app.types.workflow')}
)}
@@ -323,7 +353,7 @@ const AppInfo = ({ expand, onlyShowDetail = false, openState = false, onDetailEx />
{appDetail.name}
-
{appDetail.mode === 'advanced-chat' ? t('app.types.advanced') : appDetail.mode === 'agent-chat' ? t('app.types.agent') : appDetail.mode === 'chat' ? t('app.types.chatbot') : appDetail.mode === 'completion' ? t('app.types.completion') : t('app.types.workflow')}
+
{appDetail.mode === AppModeEnum.ADVANCED_CHAT ? t('app.types.advanced') : appDetail.mode === AppModeEnum.AGENT_CHAT ? t('app.types.agent') : appDetail.mode === AppModeEnum.CHAT ? t('app.types.chatbot') : appDetail.mode === AppModeEnum.COMPLETION ? t('app.types.completion') : t('app.types.workflow')}
{/* description */} @@ -333,7 +363,8 @@ const AppInfo = ({ expand, onlyShowDetail = false, openState = false, onDetailEx {/* operations */}
- -
- -
+ {/* Switch operation (if available) */} + {switchOperation && ( +
+ +
+ )} {showSwitchModal && ( void + id: string + title: string + icon: JSX.Element + onClick: () => void + type?: 'divider' } -const AppOperations = ({ operations, gap }: { - operations: Operation[] +type AppOperationsProps = { gap: number -}) => { + operations?: Operation[] + primaryOperations?: Operation[] + secondaryOperations?: Operation[] +} + +const EMPTY_OPERATIONS: Operation[] = [] + +const AppOperations = ({ + operations, + primaryOperations, + secondaryOperations, + gap, +}: AppOperationsProps) => { const { t } = useTranslation() const [visibleOpreations, setVisibleOperations] = useState([]) const [moreOperations, setMoreOperations] = useState([]) @@ -23,22 +37,59 @@ const AppOperations = ({ operations, gap }: { setShowMore(true) }, [setShowMore]) + const primaryOps = useMemo(() => { + if (operations) + return operations + if (primaryOperations) + return primaryOperations + return EMPTY_OPERATIONS + }, [operations, primaryOperations]) + + const secondaryOps = useMemo(() => { + if (operations) + return EMPTY_OPERATIONS + if (secondaryOperations) + return secondaryOperations + return EMPTY_OPERATIONS + }, [operations, secondaryOperations]) + const inlineOperations = primaryOps.filter(operation => operation.type !== 'divider') + useEffect(() => { - const moreElement = document.getElementById('more') - const navElement = document.getElementById('nav') + const applyState = (visible: Operation[], overflow: Operation[]) => { + const combinedMore = [...overflow, ...secondaryOps] + if (!overflow.length && combinedMore[0]?.type === 'divider') + combinedMore.shift() + setVisibleOperations(visible) + setMoreOperations(combinedMore) + } + + const inline = primaryOps.filter(operation => operation.type !== 'divider') + + if (!inline.length) { + applyState([], []) + return + } + + const navElement = navRef.current + const moreElement = document.getElementById('more-measure') + + if (!navElement || !moreElement) + return + let width = 0 - const containerWidth = navElement?.clientWidth ?? 0 - const moreWidth = moreElement?.clientWidth ?? 0 + const containerWidth = navElement.clientWidth + const moreWidth = moreElement.clientWidth - if (containerWidth === 0 || moreWidth === 0) return + if (containerWidth === 0 || moreWidth === 0) + return - const updatedEntries: Record = operations.reduce((pre, cur) => { + const updatedEntries: Record = inline.reduce((pre, cur) => { pre[cur.id] = false return pre }, {} as Record) - const childrens = Array.from(navRef.current!.children).slice(0, -1) + const childrens = Array.from(navElement.children).slice(0, -1) for (let i = 0; i < childrens.length; i++) { - const child: any = childrens[i] + const child = childrens[i] as HTMLElement const id = child.dataset.targetid if (!id) break const childWidth = child.clientWidth @@ -55,88 +106,106 @@ const AppOperations = ({ operations, gap }: { break } } - setVisibleOperations(operations.filter(item => updatedEntries[item.id])) - setMoreOperations(operations.filter(item => !updatedEntries[item.id])) - }, [operations, gap]) + + const visible = inline.filter(item => updatedEntries[item.id]) + const overflow = inline.filter(item => !updatedEntries[item.id]) + + applyState(visible, overflow) + }, [gap, primaryOps, secondaryOps]) + + const shouldShowMoreButton = moreOperations.length > 0 return ( <> - {!visibleOpreations.length && } -
- {visibleOpreations.map(operation => + {inlineOperations.map(operation => ( , - )} - {visibleOpreations.length < operations.length && - - - - -
- {moreOperations.map(item =>
+ ))} + +
+
+ {visibleOpreations.map(operation => ( + + ))} + {shouldShowMoreButton && ( + + +
)} -
-
-
} + + + {t('common.operation.more')} + + + + +
+ {moreOperations.map(item => item.type === 'divider' + ? ( +
+ ) + : ( +
+ {cloneElement(item.icon, { className: 'h-4 w-4 text-text-tertiary' })} + {item.title} +
+ ))} +
+ + + )}
) diff --git a/web/app/components/app-sidebar/app-sidebar-dropdown.tsx b/web/app/components/app-sidebar/app-sidebar-dropdown.tsx index b1da43ae14..3c5d38dd82 100644 --- a/web/app/components/app-sidebar/app-sidebar-dropdown.tsx +++ b/web/app/components/app-sidebar/app-sidebar-dropdown.tsx @@ -17,6 +17,7 @@ import NavLink from './navLink' import { useStore as useAppStore } from '@/app/components/app/store' import type { NavIcon } from './navLink' import cn from '@/utils/classnames' +import { AppModeEnum } from '@/types/app' type Props = { navigation: Array<{ @@ -97,7 +98,7 @@ const AppSidebarDropdown = ({ navigation }: Props) => {
{appDetail.name}
-
{appDetail.mode === 'advanced-chat' ? t('app.types.advanced') : appDetail.mode === 'agent-chat' ? t('app.types.agent') : appDetail.mode === 'chat' ? t('app.types.chatbot') : appDetail.mode === 'completion' ? t('app.types.completion') : t('app.types.workflow')}
+
{appDetail.mode === AppModeEnum.ADVANCED_CHAT ? t('app.types.advanced') : appDetail.mode === AppModeEnum.AGENT_CHAT ? t('app.types.agent') : appDetail.mode === AppModeEnum.CHAT ? t('app.types.chatbot') : appDetail.mode === AppModeEnum.COMPLETION ? t('app.types.completion') : t('app.types.workflow')}
diff --git a/web/app/components/app-sidebar/basic.tsx b/web/app/components/app-sidebar/basic.tsx index 77a965c03e..da85fb154b 100644 --- a/web/app/components/app-sidebar/basic.tsx +++ b/web/app/components/app-sidebar/basic.tsx @@ -3,7 +3,7 @@ import { useTranslation } from 'react-i18next' import AppIcon from '../base/app-icon' import Tooltip from '@/app/components/base/tooltip' import { - Code, + ApiAggregate, WindowCursor, } from '@/app/components/base/icons/src/vender/workflow' @@ -40,8 +40,8 @@ const NotionSvg = , - api:
- + api:
+
, dataset: , webapp:
@@ -56,12 +56,12 @@ export default function AppBasic({ icon, icon_background, name, isExternal, type return (
{icon && icon_background && iconType === 'app' && ( -
+
)} {iconType !== 'app' - &&
+ &&
{ICON_MAP[iconType]}
diff --git a/web/app/components/app/annotation/index.tsx b/web/app/components/app/annotation/index.tsx index bc63b85f6d..8718890e35 100644 --- a/web/app/components/app/annotation/index.tsx +++ b/web/app/components/app/annotation/index.tsx @@ -24,7 +24,7 @@ import type { AnnotationReplyConfig } from '@/models/debug' import { sleep } from '@/utils' import { useProviderContext } from '@/context/provider-context' import AnnotationFullModal from '@/app/components/billing/annotation-full/modal' -import type { App } from '@/types/app' +import { type App, AppModeEnum } from '@/types/app' import cn from '@/utils/classnames' import { delAnnotations } from '@/service/annotation' @@ -37,7 +37,7 @@ const Annotation: FC = (props) => { const { t } = useTranslation() const [isShowEdit, setIsShowEdit] = useState(false) const [annotationConfig, setAnnotationConfig] = useState(null) - const [isChatApp] = useState(appDetail.mode !== 'completion') + const [isChatApp] = useState(appDetail.mode !== AppModeEnum.COMPLETION) const [controlRefreshSwitch, setControlRefreshSwitch] = useState(() => Date.now()) const { plan, enableBilling } = useProviderContext() const isAnnotationFull = enableBilling && plan.usage.annotatedResponse >= plan.total.annotatedResponse diff --git a/web/app/components/app/app-publisher/features-wrapper.tsx b/web/app/components/app/app-publisher/features-wrapper.tsx index 409c390f4b..4b64558016 100644 --- a/web/app/components/app/app-publisher/features-wrapper.tsx +++ b/web/app/components/app/app-publisher/features-wrapper.tsx @@ -22,37 +22,39 @@ const FeaturesWrappedAppPublisher = (props: Props) => { const features = useFeatures(s => s.features) const featuresStore = useFeaturesStore() const [restoreConfirmOpen, setRestoreConfirmOpen] = useState(false) + const { more_like_this, opening_statement, suggested_questions, sensitive_word_avoidance, speech_to_text, text_to_speech, suggested_questions_after_answer, retriever_resource, annotation_reply, file_upload, resetAppConfig } = props.publishedConfig.modelConfig + const handleConfirm = useCallback(() => { - props.resetAppConfig?.() + resetAppConfig?.() const { features, setFeatures, } = featuresStore!.getState() const newFeatures = produce(features, (draft) => { - draft.moreLikeThis = props.publishedConfig.modelConfig.more_like_this || { enabled: false } + draft.moreLikeThis = more_like_this || { enabled: false } draft.opening = { - enabled: !!props.publishedConfig.modelConfig.opening_statement, - opening_statement: props.publishedConfig.modelConfig.opening_statement || '', - suggested_questions: props.publishedConfig.modelConfig.suggested_questions || [], + enabled: !!opening_statement, + opening_statement: opening_statement || '', + suggested_questions: suggested_questions || [], } - draft.moderation = props.publishedConfig.modelConfig.sensitive_word_avoidance || { enabled: false } - draft.speech2text = props.publishedConfig.modelConfig.speech_to_text || { enabled: false } - draft.text2speech = props.publishedConfig.modelConfig.text_to_speech || { enabled: false } - draft.suggested = props.publishedConfig.modelConfig.suggested_questions_after_answer || { enabled: false } - draft.citation = props.publishedConfig.modelConfig.retriever_resource || { enabled: false } - draft.annotationReply = props.publishedConfig.modelConfig.annotation_reply || { enabled: false } + draft.moderation = sensitive_word_avoidance || { enabled: false } + draft.speech2text = speech_to_text || { enabled: false } + draft.text2speech = text_to_speech || { enabled: false } + draft.suggested = suggested_questions_after_answer || { enabled: false } + draft.citation = retriever_resource || { enabled: false } + draft.annotationReply = annotation_reply || { enabled: false } draft.file = { image: { - detail: props.publishedConfig.modelConfig.file_upload?.image?.detail || Resolution.high, - enabled: !!props.publishedConfig.modelConfig.file_upload?.image?.enabled, - number_limits: props.publishedConfig.modelConfig.file_upload?.image?.number_limits || 3, - transfer_methods: props.publishedConfig.modelConfig.file_upload?.image?.transfer_methods || ['local_file', 'remote_url'], + detail: file_upload?.image?.detail || Resolution.high, + enabled: !!file_upload?.image?.enabled, + number_limits: file_upload?.image?.number_limits || 3, + transfer_methods: file_upload?.image?.transfer_methods || ['local_file', 'remote_url'], }, - enabled: !!(props.publishedConfig.modelConfig.file_upload?.enabled || props.publishedConfig.modelConfig.file_upload?.image?.enabled), - allowed_file_types: props.publishedConfig.modelConfig.file_upload?.allowed_file_types || [SupportUploadFileTypes.image], - allowed_file_extensions: props.publishedConfig.modelConfig.file_upload?.allowed_file_extensions || FILE_EXTS[SupportUploadFileTypes.image].map(ext => `.${ext}`), - allowed_file_upload_methods: props.publishedConfig.modelConfig.file_upload?.allowed_file_upload_methods || props.publishedConfig.modelConfig.file_upload?.image?.transfer_methods || ['local_file', 'remote_url'], - number_limits: props.publishedConfig.modelConfig.file_upload?.number_limits || props.publishedConfig.modelConfig.file_upload?.image?.number_limits || 3, + enabled: !!(file_upload?.enabled || file_upload?.image?.enabled), + allowed_file_types: file_upload?.allowed_file_types || [SupportUploadFileTypes.image], + allowed_file_extensions: file_upload?.allowed_file_extensions || FILE_EXTS[SupportUploadFileTypes.image].map(ext => `.${ext}`), + allowed_file_upload_methods: file_upload?.allowed_file_upload_methods || file_upload?.image?.transfer_methods || ['local_file', 'remote_url'], + number_limits: file_upload?.number_limits || file_upload?.image?.number_limits || 3, } as FileUpload }) setFeatures(newFeatures) @@ -69,7 +71,7 @@ const FeaturesWrappedAppPublisher = (props: Props) => { ...props, onPublish: handlePublish, onRestore: () => setRestoreConfirmOpen(true), - }}/> + }} /> {restoreConfirmOpen && ( = { + [AccessMode.ORGANIZATION]: { + label: 'organization', + icon: RiBuildingLine, + }, + [AccessMode.SPECIFIC_GROUPS_MEMBERS]: { + label: 'specific', + icon: RiLockLine, + }, + [AccessMode.PUBLIC]: { + label: 'anyone', + icon: RiGlobalLine, + }, + [AccessMode.EXTERNAL_MEMBERS]: { + label: 'external', + icon: RiVerifiedBadgeLine, + }, +} + +const AccessModeDisplay: React.FC<{ mode?: AccessMode }> = ({ mode }) => { + const { t } = useTranslation() + + if (!mode || !ACCESS_MODE_MAP[mode]) + return null + + const { icon: Icon, label } = ACCESS_MODE_MAP[mode] + + return ( + <> + +
+ {t(`app.accessControlDialog.accessItems.${label}`)} +
+ + ) +} export type AppPublisherProps = { disabled?: boolean @@ -64,6 +103,9 @@ export type AppPublisherProps = { toolPublished?: boolean inputs?: InputVar[] onRefreshData?: () => void + workflowToolAvailable?: boolean + missingStartNode?: boolean + hasTriggerNode?: boolean // Whether workflow currently contains any trigger nodes (used to hide missing-start CTA when triggers exist). } const PUBLISH_SHORTCUT = ['ctrl', '⇧', 'P'] @@ -82,28 +124,48 @@ const AppPublisher = ({ toolPublished, inputs, onRefreshData, + workflowToolAvailable = true, + missingStartNode = false, + hasTriggerNode = false, }: AppPublisherProps) => { const { t } = useTranslation() + const [published, setPublished] = useState(false) const [open, setOpen] = useState(false) + const [showAppAccessControl, setShowAppAccessControl] = useState(false) + const [isAppAccessSet, setIsAppAccessSet] = useState(true) + const [embeddingModalOpen, setEmbeddingModalOpen] = useState(false) + const appDetail = useAppStore(state => state.appDetail) const setAppDetail = useAppStore(s => s.setAppDetail) const systemFeatures = useGlobalPublicStore(s => s.systemFeatures) const { formatTimeFromNow } = useFormatTimeFromNow() const { app_base_url: appBaseURL = '', access_token: accessToken = '' } = appDetail?.site ?? {} - const appMode = (appDetail?.mode !== 'completion' && appDetail?.mode !== 'workflow') ? 'chat' : appDetail.mode + + const appMode = (appDetail?.mode !== AppModeEnum.COMPLETION && appDetail?.mode !== AppModeEnum.WORKFLOW) ? AppModeEnum.CHAT : appDetail.mode const appURL = `${appBaseURL}${basePath}/${appMode}/${accessToken}` - const isChatApp = ['chat', 'agent-chat', 'completion'].includes(appDetail?.mode || '') + const isChatApp = [AppModeEnum.CHAT, AppModeEnum.AGENT_CHAT, AppModeEnum.COMPLETION].includes(appDetail?.mode || AppModeEnum.CHAT) + const { data: userCanAccessApp, isLoading: isGettingUserCanAccessApp, refetch } = useGetUserCanAccessApp({ appId: appDetail?.id, enabled: false }) const { data: appAccessSubjects, isLoading: isGettingAppWhiteListSubjects } = useAppWhiteListSubjects(appDetail?.id, open && systemFeatures.webapp_auth.enabled && appDetail?.access_mode === AccessMode.SPECIFIC_GROUPS_MEMBERS) + const noAccessPermission = useMemo(() => systemFeatures.webapp_auth.enabled && appDetail && appDetail.access_mode !== AccessMode.EXTERNAL_MEMBERS && !userCanAccessApp?.result, [systemFeatures, appDetail, userCanAccessApp]) + const disabledFunctionButton = useMemo(() => (!publishedAt || missingStartNode || noAccessPermission), [publishedAt, missingStartNode, noAccessPermission]) + + const disabledFunctionTooltip = useMemo(() => { + if (!publishedAt) + return t('app.notPublishedYet') + if (missingStartNode) + return t('app.noUserInputNode') + if (noAccessPermission) + return t('app.noAccessPermission') + }, [missingStartNode, noAccessPermission, publishedAt]) + useEffect(() => { if (systemFeatures.webapp_auth.enabled && open && appDetail) refetch() }, [open, appDetail, refetch, systemFeatures]) - const [showAppAccessControl, setShowAppAccessControl] = useState(false) - const [isAppAccessSet, setIsAppAccessSet] = useState(true) useEffect(() => { if (appDetail && appAccessSubjects) { if (appDetail.access_mode === AccessMode.SPECIFIC_GROUPS_MEMBERS && appAccessSubjects.groups?.length === 0 && appAccessSubjects.members?.length === 0) @@ -162,15 +224,18 @@ const AppPublisher = ({ } }, [appDetail?.id]) - const handleAccessControlUpdate = useCallback(() => { - fetchAppDetail({ url: '/apps', id: appDetail!.id }).then((res) => { + const handleAccessControlUpdate = useCallback(async () => { + if (!appDetail) + return + try { + const res = await fetchAppDetailDirect({ url: '/apps', id: appDetail.id }) setAppDetail(res) + } + finally { setShowAppAccessControl(false) - }) + } }, [appDetail, setAppDetail]) - const [embeddingModalOpen, setEmbeddingModalOpen] = useState(false) - useKeyPress(`${getKeyboardKeyCodeBySystem('ctrl')}.shift.p`, (e) => { e.preventDefault() if (publishDisabled || published) @@ -178,6 +243,10 @@ const AppPublisher = ({ handlePublish() }, { exactMatch: true, useCapture: true }) + const hasPublishedVersion = !!publishedAt + const workflowToolDisabled = !hasPublishedVersion || !workflowToolAvailable + const workflowToolMessage = workflowToolDisabled ? t('workflow.common.workflowAsToolDisabledHint') : undefined + return ( <>
} -
- - } - > - {t('workflow.common.runApp')} - - - {appDetail?.mode === 'workflow' || appDetail?.mode === 'completion' - ? ( - + { + // Hide run/batch run app buttons when there is a trigger node. + !hasTriggerNode && ( +
+ } + disabled={disabledFunctionButton} + link={appURL} + icon={} > - {t('workflow.common.batchRunApp')} + {t('workflow.common.runApp')} - ) - : ( - { - setEmbeddingModalOpen(true) - handleTrigger() - }} - disabled={!publishedAt} - icon={} - > - {t('workflow.common.embedIntoSite')} - - )} - - { - if (publishedAt) - handleOpenInExplore() - }} - disabled={!publishedAt || (systemFeatures.webapp_auth.enabled && !userCanAccessApp?.result)} - icon={} - > - {t('workflow.common.openInExplore')} - - - } - > - {t('workflow.common.accessAPIReference')} - - {appDetail?.mode === 'workflow' && ( - + {appDetail?.mode === AppModeEnum.WORKFLOW || appDetail?.mode === AppModeEnum.COMPLETION + ? ( + + } + > + {t('workflow.common.batchRunApp')} + + + ) + : ( + { + setEmbeddingModalOpen(true) + handleTrigger() + }} + disabled={!publishedAt} + icon={} + > + {t('workflow.common.embedIntoSite')} + + )} + + { + if (publishedAt) + handleOpenInExplore() + }} + disabled={disabledFunctionButton} + icon={} + > + {t('workflow.common.openInExplore')} + + + + } + > + {t('workflow.common.accessAPIReference')} + + + {appDetail?.mode === AppModeEnum.WORKFLOW && ( + + )} +
)} -
}
diff --git a/web/app/components/app/configuration/config-prompt/advanced-prompt-input.tsx b/web/app/components/app/configuration/config-prompt/advanced-prompt-input.tsx index aa8d0f65ca..5bf2f177ff 100644 --- a/web/app/components/app/configuration/config-prompt/advanced-prompt-input.tsx +++ b/web/app/components/app/configuration/config-prompt/advanced-prompt-input.tsx @@ -25,7 +25,7 @@ import Tooltip from '@/app/components/base/tooltip' import PromptEditor from '@/app/components/base/prompt-editor' import ConfigContext from '@/context/debug-configuration' import { getNewVar, getVars } from '@/utils/var' -import { AppType } from '@/types/app' +import { AppModeEnum } from '@/types/app' import { useModalContext } from '@/context/modal-context' import type { ExternalDataTool } from '@/models/common' import { useToastContext } from '@/app/components/base/toast' @@ -102,7 +102,7 @@ const AdvancedPromptInput: FC = ({ }, }) } - const isChatApp = mode !== AppType.completion + const isChatApp = mode !== AppModeEnum.COMPLETION const [isCopied, setIsCopied] = React.useState(false) const promptVariablesObj = (() => { diff --git a/web/app/components/app/configuration/config-prompt/index.tsx b/web/app/components/app/configuration/config-prompt/index.tsx index ec34588e41..416f87e135 100644 --- a/web/app/components/app/configuration/config-prompt/index.tsx +++ b/web/app/components/app/configuration/config-prompt/index.tsx @@ -12,11 +12,13 @@ import Button from '@/app/components/base/button' import AdvancedMessageInput from '@/app/components/app/configuration/config-prompt/advanced-prompt-input' import { PromptRole } from '@/models/debug' import type { PromptItem, PromptVariable } from '@/models/debug' -import { type AppType, ModelModeType } from '@/types/app' +import type { AppModeEnum } from '@/types/app' +import { ModelModeType } from '@/types/app' import ConfigContext from '@/context/debug-configuration' import { MAX_PROMPT_MESSAGE_LENGTH } from '@/config' + export type IPromptProps = { - mode: AppType + mode: AppModeEnum promptTemplate: string promptVariables: PromptVariable[] readonly?: boolean diff --git a/web/app/components/app/configuration/config-prompt/simple-prompt-input.tsx b/web/app/components/app/configuration/config-prompt/simple-prompt-input.tsx index 8634232b2b..68bf6dd7c2 100644 --- a/web/app/components/app/configuration/config-prompt/simple-prompt-input.tsx +++ b/web/app/components/app/configuration/config-prompt/simple-prompt-input.tsx @@ -10,7 +10,7 @@ import PromptEditorHeightResizeWrap from './prompt-editor-height-resize-wrap' import cn from '@/utils/classnames' import type { PromptVariable } from '@/models/debug' import Tooltip from '@/app/components/base/tooltip' -import { AppType } from '@/types/app' +import { AppModeEnum } from '@/types/app' import { getNewVar, getVars } from '@/utils/var' import AutomaticBtn from '@/app/components/app/configuration/config/automatic/automatic-btn' import type { GenRes } from '@/service/debug' @@ -29,7 +29,7 @@ import { useFeaturesStore } from '@/app/components/base/features/hooks' import { noop } from 'lodash-es' export type ISimplePromptInput = { - mode: AppType + mode: AppModeEnum promptTemplate: string promptVariables: PromptVariable[] readonly?: boolean @@ -155,7 +155,7 @@ const Prompt: FC = ({ setModelConfig(newModelConfig) setPrevPromptConfig(modelConfig.configs) - if (mode !== AppType.completion) { + if (mode !== AppModeEnum.COMPLETION) { setIntroduction(res.opening_statement || '') const newFeatures = produce(features, (draft) => { draft.opening = { @@ -177,7 +177,7 @@ const Prompt: FC = ({ {!noTitle && (
-
{mode !== AppType.completion ? t('appDebug.chatSubTitle') : t('appDebug.completionSubTitle')}
+
{mode !== AppModeEnum.COMPLETION ? t('appDebug.chatSubTitle') : t('appDebug.completionSubTitle')}
{!readonly && ( = ({ {showAutomatic && ( = ({ const { type, label, variable, options, max_length } = tempPayload const modalRef = useRef(null) const appDetail = useAppStore(state => state.appDetail) - const isBasicApp = appDetail?.mode !== 'advanced-chat' && appDetail?.mode !== 'workflow' + const isBasicApp = appDetail?.mode !== AppModeEnum.ADVANCED_CHAT && appDetail?.mode !== AppModeEnum.WORKFLOW const isSupportJSON = false const jsonSchemaStr = useMemo(() => { const isJsonObject = type === InputVarType.jsonObject diff --git a/web/app/components/app/configuration/config-var/index.tsx b/web/app/components/app/configuration/config-var/index.tsx index cad7262650..75782ff5c3 100644 --- a/web/app/components/app/configuration/config-var/index.tsx +++ b/web/app/components/app/configuration/config-var/index.tsx @@ -17,7 +17,7 @@ import { getNewVar, hasDuplicateStr } from '@/utils/var' import Toast from '@/app/components/base/toast' import Confirm from '@/app/components/base/confirm' import ConfigContext from '@/context/debug-configuration' -import { AppType } from '@/types/app' +import { AppModeEnum } from '@/types/app' import type { ExternalDataTool } from '@/models/common' import { useModalContext } from '@/context/modal-context' import { useEventEmitterContextContext } from '@/context/event-emitter' @@ -201,7 +201,7 @@ const ConfigVar: FC = ({ promptVariables, readonly, onPromptVar const handleRemoveVar = (index: number) => { const removeVar = promptVariables[index] - if (mode === AppType.completion && dataSets.length > 0 && removeVar.is_context_var) { + if (mode === AppModeEnum.COMPLETION && dataSets.length > 0 && removeVar.is_context_var) { showDeleteContextVarModal() setRemoveIndex(index) return diff --git a/web/app/components/app/configuration/config/agent/agent-tools/setting-built-in-tool.tsx b/web/app/components/app/configuration/config/agent/agent-tools/setting-built-in-tool.tsx index 62bd57c5d1..ef28dd222c 100644 --- a/web/app/components/app/configuration/config/agent/agent-tools/setting-built-in-tool.tsx +++ b/web/app/components/app/configuration/config/agent/agent-tools/setting-built-in-tool.tsx @@ -28,6 +28,7 @@ import { AuthCategory, PluginAuthInAgent, } from '@/app/components/plugins/plugin-auth' +import { ReadmeEntrance } from '@/app/components/plugins/readme-panel/entrance' type Props = { showBackButton?: boolean @@ -193,7 +194,7 @@ const SettingBuiltInTool: FC = ({ onClick={onHide} > - BACK + {t('plugin.detailPanel.operation.back')}
)}
@@ -214,6 +215,8 @@ const SettingBuiltInTool: FC = ({ pluginPayload={{ provider: collection.name, category: AuthCategory.tool, + providerType: collection.type, + detail: collection as any, }} credentialId={credentialId} onAuthorizationItemClick={onAuthorizationItemClick} @@ -243,13 +246,14 @@ const SettingBuiltInTool: FC = ({ )}
{isInfoActive ? infoUI : settingUI} + {!readonly && !isInfoActive && ( +
+ + +
+ )}
- {!readonly && !isInfoActive && ( -
- - -
- )} +
diff --git a/web/app/components/app/configuration/config/automatic/get-automatic-res.tsx b/web/app/components/app/configuration/config/automatic/get-automatic-res.tsx index 5c87eea3ca..dfcaabf017 100644 --- a/web/app/components/app/configuration/config/automatic/get-automatic-res.tsx +++ b/web/app/components/app/configuration/config/automatic/get-automatic-res.tsx @@ -19,8 +19,7 @@ import Modal from '@/app/components/base/modal' import Button from '@/app/components/base/button' import Toast from '@/app/components/base/toast' import { generateBasicAppFirstTimeRule, generateRule } from '@/service/debug' -import type { CompletionParams, Model } from '@/types/app' -import type { AppType } from '@/types/app' +import type { AppModeEnum, CompletionParams, Model } from '@/types/app' import Loading from '@/app/components/base/loading' import Confirm from '@/app/components/base/confirm' @@ -44,7 +43,7 @@ import { useGenerateRuleTemplate } from '@/service/use-apps' const i18nPrefix = 'appDebug.generate' export type IGetAutomaticResProps = { - mode: AppType + mode: AppModeEnum isShow: boolean onClose: () => void onFinished: (res: GenRes) => void @@ -299,7 +298,6 @@ const GetAutomaticRes: FC = ({ portalToFollowElemContentClassName='z-[1000]' isAdvancedMode={true} provider={model.provider} - mode={model.mode} completionParams={model.completion_params} modelId={model.name} setModel={handleModelChange} diff --git a/web/app/components/app/configuration/config/code-generator/get-code-generator-res.tsx b/web/app/components/app/configuration/config/code-generator/get-code-generator-res.tsx index b581da979f..3612f89b02 100644 --- a/web/app/components/app/configuration/config/code-generator/get-code-generator-res.tsx +++ b/web/app/components/app/configuration/config/code-generator/get-code-generator-res.tsx @@ -5,8 +5,8 @@ import { useTranslation } from 'react-i18next' import { languageMap } from '../../../../workflow/nodes/_base/components/editor/code-editor/index' import { generateRule } from '@/service/debug' import type { GenRes } from '@/service/debug' -import type { ModelModeType } from '@/types/app' -import type { AppType, CompletionParams, Model } from '@/types/app' +import type { AppModeEnum, ModelModeType } from '@/types/app' +import type { CompletionParams, Model } from '@/types/app' import Modal from '@/app/components/base/modal' import Button from '@/app/components/base/button' import { Generator } from '@/app/components/base/icons/src/vender/other' @@ -33,7 +33,7 @@ export type IGetCodeGeneratorResProps = { flowId: string nodeId: string currentCode?: string - mode: AppType + mode: AppModeEnum isShow: boolean codeLanguages: CodeLanguage onClose: () => void @@ -142,7 +142,7 @@ export const GetCodeGeneratorResModal: FC = ( ideal_output: ideaOutput, language: languageMap[codeLanguages] || 'javascript', }) - if((res as any).code) // not current or current is the same as the template would return a code field + if ((res as any).code) // not current or current is the same as the template would return a code field res.modified = (res as any).code if (error) { @@ -214,7 +214,6 @@ export const GetCodeGeneratorResModal: FC = ( portalToFollowElemContentClassName='z-[1000]' isAdvancedMode={true} provider={model.provider} - mode={model.mode} completionParams={model.completion_params} modelId={model.name} setModel={handleModelChange} diff --git a/web/app/components/app/configuration/config/index.tsx b/web/app/components/app/configuration/config/index.tsx index 38b72f09cc..1c8b4b344d 100644 --- a/web/app/components/app/configuration/config/index.tsx +++ b/web/app/components/app/configuration/config/index.tsx @@ -14,8 +14,7 @@ import ConfigContext from '@/context/debug-configuration' import ConfigPrompt from '@/app/components/app/configuration/config-prompt' import ConfigVar from '@/app/components/app/configuration/config-var' import type { ModelConfig, PromptVariable } from '@/models/debug' -import type { AppType } from '@/types/app' -import { ModelModeType } from '@/types/app' +import { AppModeEnum, ModelModeType } from '@/types/app' const Config: FC = () => { const { @@ -31,7 +30,7 @@ const Config: FC = () => { setPrevPromptConfig, dataSets, } = useContext(ConfigContext) - const isChatApp = ['advanced-chat', 'agent-chat', 'chat'].includes(mode) + const isChatApp = [AppModeEnum.ADVANCED_CHAT, AppModeEnum.AGENT_CHAT, AppModeEnum.CHAT].includes(mode) const formattingChangedDispatcher = useFormattingChangedDispatcher() const promptTemplate = modelConfig.configs.prompt_template @@ -64,7 +63,7 @@ const Config: FC = () => { > {/* Template */} = ({ readonly, hideMetadataFilter }) => { draft.metadata_model_config = { provider: model.provider, name: model.modelId, - mode: model.mode || 'chat', + mode: model.mode || AppModeEnum.CHAT, completion_params: draft.metadata_model_config?.completion_params || { temperature: 0.7 }, } }) @@ -310,7 +310,7 @@ const DatasetConfig: FC = ({ readonly, hideMetadataFilter }) => {
)} - {!readonly && mode === AppType.completion && dataSet.length > 0 && ( + {!readonly && mode === AppModeEnum.COMPLETION && dataSet.length > 0 && ( = ({ popupClassName='!w-[387px]' portalToFollowElemContentClassName='!z-[1002]' isAdvancedMode={true} - mode={model?.mode} provider={model?.provider} completionParams={model?.completion_params} modelId={model?.name} diff --git a/web/app/components/app/configuration/dataset-config/settings-modal/index.tsx b/web/app/components/app/configuration/dataset-config/settings-modal/index.tsx index 62f1010b54..93d0384aee 100644 --- a/web/app/components/app/configuration/dataset-config/settings-modal/index.tsx +++ b/web/app/components/app/configuration/dataset-config/settings-modal/index.tsx @@ -16,6 +16,7 @@ import { useToastContext } from '@/app/components/base/toast' import { updateDatasetSetting } from '@/service/datasets' import { useAppContext } from '@/context/app-context' import { useModalContext } from '@/context/modal-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' import type { RetrievalConfig } from '@/types/app' import RetrievalSettings from '@/app/components/datasets/external-knowledge-base/create/RetrievalSettings' import RetrievalMethodConfig from '@/app/components/datasets/common/retrieval-method-config' @@ -277,7 +278,7 @@ const SettingsModal: FC = ({
{t('datasetSettings.form.embeddingModelTip')} - setShowAccountSettingModal({ payload: 'provider' })}>{t('datasetSettings.form.embeddingModelTipLink')} + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.PROVIDER })}>{t('datasetSettings.form.embeddingModelTipLink')}
diff --git a/web/app/components/app/configuration/debug/debug-with-multiple-model/debug-item.tsx b/web/app/components/app/configuration/debug/debug-with-multiple-model/debug-item.tsx index 95c43f5101..6148e2e808 100644 --- a/web/app/components/app/configuration/debug/debug-with-multiple-model/debug-item.tsx +++ b/web/app/components/app/configuration/debug/debug-with-multiple-model/debug-item.tsx @@ -11,6 +11,7 @@ import Dropdown from '@/app/components/base/dropdown' import type { Item } from '@/app/components/base/dropdown' import { useProviderContext } from '@/context/provider-context' import { ModelStatusEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' +import { AppModeEnum } from '@/types/app' type DebugItemProps = { modelAndParameter: ModelAndParameter @@ -112,13 +113,13 @@ const DebugItem: FC = ({
{ - (mode === 'chat' || mode === 'agent-chat') && currentProvider && currentModel && currentModel.status === ModelStatusEnum.active && ( + (mode === AppModeEnum.CHAT || mode === AppModeEnum.AGENT_CHAT) && currentProvider && currentModel && currentModel.status === ModelStatusEnum.active && ( ) } { - mode === 'completion' && currentProvider && currentModel && currentModel.status === ModelStatusEnum.active && ( - + mode === AppModeEnum.COMPLETION && currentProvider && currentModel && currentModel.status === ModelStatusEnum.active && ( + ) }
diff --git a/web/app/components/app/configuration/debug/debug-with-multiple-model/index.tsx b/web/app/components/app/configuration/debug/debug-with-multiple-model/index.tsx index b876adfa3d..6c388f5afa 100644 --- a/web/app/components/app/configuration/debug/debug-with-multiple-model/index.tsx +++ b/web/app/components/app/configuration/debug/debug-with-multiple-model/index.tsx @@ -18,6 +18,7 @@ import { useFeatures } from '@/app/components/base/features/hooks' import { useStore as useAppStore } from '@/app/components/app/store' import type { FileEntity } from '@/app/components/base/file-uploader/types' import type { InputForm } from '@/app/components/base/chat/chat/type' +import { AppModeEnum } from '@/types/app' const DebugWithMultipleModel = () => { const { @@ -33,7 +34,7 @@ const DebugWithMultipleModel = () => { } = useDebugWithMultipleModelContext() const { eventEmitter } = useEventEmitterContextContext() - const isChatMode = mode === 'chat' || mode === 'agent-chat' + const isChatMode = mode === AppModeEnum.CHAT || mode === AppModeEnum.AGENT_CHAT const handleSend = useCallback((message: string, files?: FileEntity[]) => { if (checkCanSend && !checkCanSend()) diff --git a/web/app/components/app/configuration/debug/debug-with-multiple-model/model-parameter-trigger.tsx b/web/app/components/app/configuration/debug/debug-with-multiple-model/model-parameter-trigger.tsx index 17d04acdc7..e7c4d98733 100644 --- a/web/app/components/app/configuration/debug/debug-with-multiple-model/model-parameter-trigger.tsx +++ b/web/app/components/app/configuration/debug/debug-with-multiple-model/model-parameter-trigger.tsx @@ -26,7 +26,6 @@ const ModelParameterTrigger: FC = ({ }) => { const { t } = useTranslation() const { - mode, isAdvancedMode, } = useDebugConfigurationContext() const { @@ -57,7 +56,6 @@ const ModelParameterTrigger: FC = ({ return ( = ({ const config: TextGenerationConfig = { pre_prompt: !isAdvancedMode ? modelConfig.configs.prompt_template : '', prompt_type: promptMode, - chat_prompt_config: isAdvancedMode ? chatPromptConfig : {}, - completion_prompt_config: isAdvancedMode ? completionPromptConfig : {}, + chat_prompt_config: isAdvancedMode ? chatPromptConfig : cloneDeep(DEFAULT_CHAT_PROMPT_CONFIG), + completion_prompt_config: isAdvancedMode ? completionPromptConfig : cloneDeep(DEFAULT_COMPLETION_PROMPT_CONFIG), user_input_form: promptVariablesToUserInputsForm(modelConfig.configs.prompt_variables), dataset_query_variable: contextVar || '', // features @@ -74,6 +76,7 @@ const TextGenerationItem: FC = ({ datasets: [...postDatasets], } as any, }, + system_parameters: modelConfig.system_parameters, } const { completion, @@ -127,11 +130,11 @@ const TextGenerationItem: FC = ({ return ( { + const doRegenerate = useCallback((chatItem: ChatItem, editedQuestion?: { message: string, files?: FileEntity[] }) => { const question = editedQuestion ? chatItem : chatList.find(item => item.id === chatItem.parentMessageId)! const parentAnswer = chatList.find(item => item.id === question.parentMessageId) doSend(editedQuestion ? editedQuestion.message : question.content, diff --git a/web/app/components/app/configuration/debug/hooks.tsx b/web/app/components/app/configuration/debug/hooks.tsx index 12022e706a..9f628c46af 100644 --- a/web/app/components/app/configuration/debug/hooks.tsx +++ b/web/app/components/app/configuration/debug/hooks.tsx @@ -12,12 +12,15 @@ import type { ChatConfig, ChatItem, } from '@/app/components/base/chat/types' +import cloneDeep from 'lodash-es/cloneDeep' import { AgentStrategy, } from '@/types/app' +import { SupportUploadFileTypes } from '@/app/components/workflow/types' import { promptVariablesToUserInputsForm } from '@/utils/model-config' import { useDebugConfigurationContext } from '@/context/debug-configuration' import { useEventEmitterContextContext } from '@/context/event-emitter' +import { DEFAULT_CHAT_PROMPT_CONFIG, DEFAULT_COMPLETION_PROMPT_CONFIG } from '@/config' export const useDebugWithSingleOrMultipleModel = (appId: string) => { const localeDebugWithSingleOrMultipleModelConfigs = localStorage.getItem('app-debug-with-single-or-multiple-models') @@ -95,16 +98,14 @@ export const useConfigFromDebugContext = () => { const config: ChatConfig = { pre_prompt: !isAdvancedMode ? modelConfig.configs.prompt_template : '', prompt_type: promptMode, - chat_prompt_config: isAdvancedMode ? chatPromptConfig : {}, - completion_prompt_config: isAdvancedMode ? completionPromptConfig : {}, + chat_prompt_config: isAdvancedMode ? chatPromptConfig : cloneDeep(DEFAULT_CHAT_PROMPT_CONFIG), + completion_prompt_config: isAdvancedMode ? completionPromptConfig : cloneDeep(DEFAULT_COMPLETION_PROMPT_CONFIG), user_input_form: promptVariablesToUserInputsForm(modelConfig.configs.prompt_variables), dataset_query_variable: contextVar || '', opening_statement: introduction, - more_like_this: { - enabled: false, - }, + more_like_this: modelConfig.more_like_this ?? { enabled: false }, suggested_questions: openingSuggestedQuestions, - suggested_questions_after_answer: suggestedQuestionsAfterAnswerConfig, + suggested_questions_after_answer: suggestedQuestionsAfterAnswerConfig ?? { enabled: false }, text_to_speech: textToSpeechConfig, speech_to_text: speechToTextConfig, retriever_resource: citationConfig, @@ -121,8 +122,13 @@ export const useConfigFromDebugContext = () => { }, file_upload: { image: visionConfig, + allowed_file_upload_methods: visionConfig.transfer_methods ?? [], + allowed_file_types: [SupportUploadFileTypes.image], + max_length: visionConfig.number_limits ?? 0, + number_limits: visionConfig.number_limits, }, annotation_reply: annotationConfig, + system_parameters: modelConfig.system_parameters, supportAnnotation: true, appId, diff --git a/web/app/components/app/configuration/debug/index.tsx b/web/app/components/app/configuration/debug/index.tsx index 18631afe30..b5135d2cdb 100644 --- a/web/app/components/app/configuration/debug/index.tsx +++ b/web/app/components/app/configuration/debug/index.tsx @@ -3,6 +3,7 @@ import type { FC } from 'react' import { useTranslation } from 'react-i18next' import React, { useCallback, useEffect, useRef, useState } from 'react' import { produce, setAutoFreeze } from 'immer' +import cloneDeep from 'lodash-es/cloneDeep' import { useBoolean } from 'ahooks' import { RiAddLine, @@ -23,7 +24,7 @@ import { APP_CHAT_WITH_MULTIPLE_MODEL, APP_CHAT_WITH_MULTIPLE_MODEL_RESTART, } from './types' -import { AppType, ModelModeType, TransferMethod } from '@/types/app' +import { AppModeEnum, ModelModeType, TransferMethod } from '@/types/app' import ChatUserInput from '@/app/components/app/configuration/debug/chat-user-input' import PromptValuePanel from '@/app/components/app/configuration/prompt-value-panel' import ConfigContext from '@/context/debug-configuration' @@ -36,7 +37,7 @@ import ActionButton, { ActionButtonState } from '@/app/components/base/action-bu import type { ModelConfig as BackendModelConfig, VisionFile, VisionSettings } from '@/types/app' import { formatBooleanInputs, promptVariablesToUserInputsForm } from '@/utils/model-config' import TextGeneration from '@/app/components/app/text-generate/item' -import { IS_CE_EDITION } from '@/config' +import { DEFAULT_CHAT_PROMPT_CONFIG, DEFAULT_COMPLETION_PROMPT_CONFIG, IS_CE_EDITION } from '@/config' import type { Inputs } from '@/models/debug' import { useDefaultModel } from '@/app/components/header/account-setting/model-provider-page/hooks' import { ModelFeatureEnum, ModelTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' @@ -48,6 +49,7 @@ import PromptLogModal from '@/app/components/base/prompt-log-modal' import { useStore as useAppStore } from '@/app/components/app/store' import { useFeatures, useFeaturesStore } from '@/app/components/base/features/hooks' import { noop } from 'lodash-es' +import { AppSourceType } from '@/service/share' type IDebug = { isAPIKeySet: boolean @@ -91,6 +93,7 @@ const Debug: FC = ({ completionParams, hasSetContextVar, datasetConfigs, + externalDataToolsConfig, } = useContext(ConfigContext) const { eventEmitter } = useEventEmitterContextContext() const { data: text2speechDefaultModel } = useDefaultModel(ModelTypeEnum.textEmbedding) @@ -143,7 +146,7 @@ const Debug: FC = ({ const [completionFiles, setCompletionFiles] = useState([]) const checkCanSend = useCallback(() => { - if (isAdvancedMode && mode !== AppType.completion) { + if (isAdvancedMode && mode !== AppModeEnum.COMPLETION) { if (modelModeType === ModelModeType.completion) { if (!hasSetBlockStatus.history) { notify({ type: 'error', message: t('appDebug.otherError.historyNoBeEmpty') }) @@ -224,8 +227,8 @@ const Debug: FC = ({ const postModelConfig: BackendModelConfig = { pre_prompt: !isAdvancedMode ? modelConfig.configs.prompt_template : '', prompt_type: promptMode, - chat_prompt_config: {}, - completion_prompt_config: {}, + chat_prompt_config: isAdvancedMode ? chatPromptConfig : cloneDeep(DEFAULT_CHAT_PROMPT_CONFIG), + completion_prompt_config: isAdvancedMode ? completionPromptConfig : cloneDeep(DEFAULT_COMPLETION_PROMPT_CONFIG), user_input_form: promptVariablesToUserInputsForm(modelConfig.configs.prompt_variables), dataset_query_variable: contextVar || '', dataset_configs: { @@ -252,11 +255,8 @@ const Debug: FC = ({ suggested_questions_after_answer: suggestedQuestionsAfterAnswerConfig, speech_to_text: speechToTextConfig, retriever_resource: citationConfig, - } - - if (isAdvancedMode) { - postModelConfig.chat_prompt_config = chatPromptConfig - postModelConfig.completion_prompt_config = completionPromptConfig + system_parameters: modelConfig.system_parameters, + external_data_tools: externalDataToolsConfig, } const data: Record = { @@ -412,7 +412,7 @@ const Debug: FC = ({ ) : null } - {mode !== AppType.completion && ( + {mode !== AppModeEnum.COMPLETION && ( <> {!readonly && ( = ({ )} - {mode !== AppType.completion && expanded && ( + {mode !== AppModeEnum.COMPLETION && expanded && (
)} - {mode === AppType.completion && ( + {mode === AppModeEnum.COMPLETION && ( = ({ !debugWithMultipleModel && (
{/* Chat */} - {mode !== AppType.completion && ( + {mode !== AppModeEnum.COMPLETION && (
= ({
)} {/* Text Generation */} - {mode === AppType.completion && ( + {mode === AppModeEnum.COMPLETION && ( <> {(completionRes || isResponding) && ( <>
= ({ )} )} - {mode === AppType.completion && showPromptLogModal && ( + {mode === AppModeEnum.COMPLETION && showPromptLogModal && ( { const mode = modelModeType const toReplacePrePrompt = prePrompt || '' + if (!appMode) + return + if (!isAdvancedPrompt) { const { chat_prompt_config, completion_prompt_config, stop } = await fetchPromptTemplate({ appMode, @@ -122,7 +125,6 @@ const useAdvancedPromptConfig = ({ }) setChatPromptConfig(newPromptConfig) } - else { const newPromptConfig = produce(completion_prompt_config, (draft) => { draft.prompt.text = draft.prompt.text.replace(PRE_PROMPT_PLACEHOLDER_TEXT, toReplacePrePrompt) @@ -152,7 +154,7 @@ const useAdvancedPromptConfig = ({ else draft.prompt.text = completionPromptConfig.prompt?.text.replace(PRE_PROMPT_PLACEHOLDER_TEXT, toReplacePrePrompt) - if (['advanced-chat', 'agent-chat', 'chat'].includes(appMode) && completionPromptConfig.conversation_histories_role.assistant_prefix && completionPromptConfig.conversation_histories_role.user_prefix) + if ([AppModeEnum.ADVANCED_CHAT, AppModeEnum.AGENT_CHAT, AppModeEnum.CHAT].includes(appMode) && completionPromptConfig.conversation_histories_role.assistant_prefix && completionPromptConfig.conversation_histories_role.user_prefix) draft.conversation_histories_role = completionPromptConfig.conversation_histories_role }) setCompletionPromptConfig(newPromptConfig) diff --git a/web/app/components/app/configuration/index.tsx b/web/app/components/app/configuration/index.tsx index a1710c8f39..afe640278e 100644 --- a/web/app/components/app/configuration/index.tsx +++ b/web/app/components/app/configuration/index.tsx @@ -36,22 +36,23 @@ import type { } from '@/models/debug' import type { ExternalDataTool } from '@/models/common' import type { DataSet } from '@/models/datasets' -import type { ModelConfig as BackendModelConfig, VisionSettings } from '@/types/app' +import type { ModelConfig as BackendModelConfig, UserInputFormItem, VisionSettings } from '@/types/app' import ConfigContext from '@/context/debug-configuration' import Config from '@/app/components/app/configuration/config' import Debug from '@/app/components/app/configuration/debug' import Confirm from '@/app/components/base/confirm' import { ModelFeatureEnum, ModelTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' import { ToastContext } from '@/app/components/base/toast' -import { fetchAppDetail, updateAppModelConfig } from '@/service/apps' +import { fetchAppDetailDirect, updateAppModelConfig } from '@/service/apps' import { promptVariablesToUserInputsForm, userInputsFormToPromptVariables } from '@/utils/model-config' import { fetchDatasets } from '@/service/datasets' import { useProviderContext } from '@/context/provider-context' -import { AgentStrategy, AppType, ModelModeType, RETRIEVE_TYPE, Resolution, TransferMethod } from '@/types/app' +import { AgentStrategy, AppModeEnum, ModelModeType, RETRIEVE_TYPE, Resolution, TransferMethod } from '@/types/app' import { PromptMode } from '@/models/debug' import { ANNOTATION_DEFAULT, DATASET_DEFAULT, DEFAULT_AGENT_SETTING, DEFAULT_CHAT_PROMPT_CONFIG, DEFAULT_COMPLETION_PROMPT_CONFIG } from '@/config' import SelectDataSet from '@/app/components/app/configuration/dataset-config/select-dataset' import { useModalContext } from '@/context/modal-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' import useBreakpoints, { MediaType } from '@/hooks/use-breakpoints' import Drawer from '@/app/components/base/drawer' import ModelParameterModal from '@/app/components/header/account-setting/model-provider-page/model-parameter-modal' @@ -110,7 +111,7 @@ const Configuration: FC = () => { const pathname = usePathname() const matched = pathname.match(/\/app\/([^/]+)/) const appId = (matched?.length && matched[1]) ? matched[1] : '' - const [mode, setMode] = useState('') + const [mode, setMode] = useState(AppModeEnum.CHAT) const [publishedConfig, setPublishedConfig] = useState(null) const [conversationId, setConversationId] = useState('') @@ -186,6 +187,8 @@ const Configuration: FC = () => { prompt_template: '', prompt_variables: [] as PromptVariable[], }, + chat_prompt_config: clone(DEFAULT_CHAT_PROMPT_CONFIG), + completion_prompt_config: clone(DEFAULT_COMPLETION_PROMPT_CONFIG), more_like_this: null, opening_statement: '', suggested_questions: [], @@ -196,10 +199,18 @@ const Configuration: FC = () => { suggested_questions_after_answer: null, retriever_resource: null, annotation_reply: null, + external_data_tools: [], + system_parameters: { + audio_file_size_limit: 0, + file_size_limit: 0, + image_file_size_limit: 0, + video_file_size_limit: 0, + workflow_file_upload_limit: 0, + }, dataSets: [], agentConfig: DEFAULT_AGENT_SETTING, }) - const isAgent = mode === 'agent-chat' + const isAgent = mode === AppModeEnum.AGENT_CHAT const isOpenAI = modelConfig.provider === 'langgenius/openai/openai' @@ -441,7 +452,7 @@ const Configuration: FC = () => { const appMode = mode if (modeMode === ModelModeType.completion) { - if (appMode !== AppType.completion) { + if (appMode !== AppModeEnum.COMPLETION) { if (!completionPromptConfig.prompt?.text || !completionPromptConfig.conversation_histories_role.assistant_prefix || !completionPromptConfig.conversation_histories_role.user_prefix) await migrateToDefaultPrompt(true, ModelModeType.completion) } @@ -543,174 +554,176 @@ const Configuration: FC = () => { }) } setCollectionList(collectionList) - fetchAppDetail({ url: '/apps', id: appId }).then(async (res: any) => { - setMode(res.mode) - const modelConfig = res.model_config - const promptMode = modelConfig.prompt_type === PromptMode.advanced ? PromptMode.advanced : PromptMode.simple - doSetPromptMode(promptMode) - if (promptMode === PromptMode.advanced) { - if (modelConfig.chat_prompt_config && modelConfig.chat_prompt_config.prompt.length > 0) - setChatPromptConfig(modelConfig.chat_prompt_config) - else - setChatPromptConfig(clone(DEFAULT_CHAT_PROMPT_CONFIG)) - setCompletionPromptConfig(modelConfig.completion_prompt_config || clone(DEFAULT_COMPLETION_PROMPT_CONFIG) as any) - setCanReturnToSimpleMode(false) - } + const res = await fetchAppDetailDirect({ url: '/apps', id: appId }) + setMode(res.mode as AppModeEnum) + const modelConfig = res.model_config as BackendModelConfig + const promptMode = modelConfig.prompt_type === PromptMode.advanced ? PromptMode.advanced : PromptMode.simple + doSetPromptMode(promptMode) + if (promptMode === PromptMode.advanced) { + if (modelConfig.chat_prompt_config && modelConfig.chat_prompt_config.prompt.length > 0) + setChatPromptConfig(modelConfig.chat_prompt_config) + else + setChatPromptConfig(clone(DEFAULT_CHAT_PROMPT_CONFIG)) + setCompletionPromptConfig(modelConfig.completion_prompt_config || clone(DEFAULT_COMPLETION_PROMPT_CONFIG) as any) + setCanReturnToSimpleMode(false) + } - const model = res.model_config.model + const model = modelConfig.model - let datasets: any = null + let datasets: any = null // old dataset struct - if (modelConfig.agent_mode?.tools?.find(({ dataset }: any) => dataset?.enabled)) - datasets = modelConfig.agent_mode?.tools.filter(({ dataset }: any) => dataset?.enabled) + if (modelConfig.agent_mode?.tools?.find(({ dataset }: any) => dataset?.enabled)) + datasets = modelConfig.agent_mode?.tools.filter(({ dataset }: any) => dataset?.enabled) // new dataset struct - else if (modelConfig.dataset_configs.datasets?.datasets?.length > 0) - datasets = modelConfig.dataset_configs?.datasets?.datasets + else if (modelConfig.dataset_configs.datasets?.datasets?.length > 0) + datasets = modelConfig.dataset_configs?.datasets?.datasets - if (dataSets && datasets?.length && datasets?.length > 0) { - const { data: dataSetsWithDetail } = await fetchDatasets({ url: '/datasets', params: { page: 1, ids: datasets.map(({ dataset }: any) => dataset.id) } }) - datasets = dataSetsWithDetail - setDataSets(datasets) - } + if (dataSets && datasets?.length && datasets?.length > 0) { + const { data: dataSetsWithDetail } = await fetchDatasets({ url: '/datasets', params: { page: 1, ids: datasets.map(({ dataset }: any) => dataset.id) } }) + datasets = dataSetsWithDetail + setDataSets(datasets) + } - setIntroduction(modelConfig.opening_statement) - setSuggestedQuestions(modelConfig.suggested_questions || []) - if (modelConfig.more_like_this) - setMoreLikeThisConfig(modelConfig.more_like_this) + setIntroduction(modelConfig.opening_statement) + setSuggestedQuestions(modelConfig.suggested_questions || []) + if (modelConfig.more_like_this) + setMoreLikeThisConfig(modelConfig.more_like_this) - if (modelConfig.suggested_questions_after_answer) - setSuggestedQuestionsAfterAnswerConfig(modelConfig.suggested_questions_after_answer) + if (modelConfig.suggested_questions_after_answer) + setSuggestedQuestionsAfterAnswerConfig(modelConfig.suggested_questions_after_answer) - if (modelConfig.speech_to_text) - setSpeechToTextConfig(modelConfig.speech_to_text) + if (modelConfig.speech_to_text) + setSpeechToTextConfig(modelConfig.speech_to_text) - if (modelConfig.text_to_speech) - setTextToSpeechConfig(modelConfig.text_to_speech) + if (modelConfig.text_to_speech) + setTextToSpeechConfig(modelConfig.text_to_speech) - if (modelConfig.retriever_resource) - setCitationConfig(modelConfig.retriever_resource) + if (modelConfig.retriever_resource) + setCitationConfig(modelConfig.retriever_resource) - if (modelConfig.annotation_reply) { - let annotationConfig = modelConfig.annotation_reply - if (modelConfig.annotation_reply.enabled) { - annotationConfig = { - ...modelConfig.annotation_reply, - embedding_model: { - ...modelConfig.annotation_reply.embedding_model, - embedding_provider_name: correctModelProvider(modelConfig.annotation_reply.embedding_model.embedding_provider_name), - }, - } + if (modelConfig.annotation_reply) { + let annotationConfig = modelConfig.annotation_reply + if (modelConfig.annotation_reply.enabled) { + annotationConfig = { + ...modelConfig.annotation_reply, + embedding_model: { + ...modelConfig.annotation_reply.embedding_model, + embedding_provider_name: correctModelProvider(modelConfig.annotation_reply.embedding_model.embedding_provider_name), + }, } - setAnnotationConfig(annotationConfig, true) } + setAnnotationConfig(annotationConfig, true) + } - if (modelConfig.sensitive_word_avoidance) - setModerationConfig(modelConfig.sensitive_word_avoidance) + if (modelConfig.sensitive_word_avoidance) + setModerationConfig(modelConfig.sensitive_word_avoidance) - if (modelConfig.external_data_tools) - setExternalDataToolsConfig(modelConfig.external_data_tools) + if (modelConfig.external_data_tools) + setExternalDataToolsConfig(modelConfig.external_data_tools) - const config = { - modelConfig: { - provider: correctModelProvider(model.provider), - model_id: model.name, - mode: model.mode, - configs: { - prompt_template: modelConfig.pre_prompt || '', - prompt_variables: userInputsFormToPromptVariables( - [ - ...modelConfig.user_input_form, - ...( - modelConfig.external_data_tools?.length - ? modelConfig.external_data_tools.map((item: any) => { - return { - external_data_tool: { - variable: item.variable as string, - label: item.label as string, - enabled: item.enabled, - type: item.type as string, - config: item.config, - required: true, - icon: item.icon, - icon_background: item.icon_background, - }, - } - }) - : [] - ), - ], - modelConfig.dataset_query_variable, - ), - }, - more_like_this: modelConfig.more_like_this, - opening_statement: modelConfig.opening_statement, - suggested_questions: modelConfig.suggested_questions, - sensitive_word_avoidance: modelConfig.sensitive_word_avoidance, - speech_to_text: modelConfig.speech_to_text, - text_to_speech: modelConfig.text_to_speech, - file_upload: modelConfig.file_upload, - suggested_questions_after_answer: modelConfig.suggested_questions_after_answer, - retriever_resource: modelConfig.retriever_resource, - annotation_reply: modelConfig.annotation_reply, - external_data_tools: modelConfig.external_data_tools, - dataSets: datasets || [], - agentConfig: res.mode === 'agent-chat' ? { - max_iteration: DEFAULT_AGENT_SETTING.max_iteration, - ...modelConfig.agent_mode, - // remove dataset - enabled: true, // modelConfig.agent_mode?.enabled is not correct. old app: the value of app with dataset's is always true - tools: modelConfig.agent_mode?.tools.filter((tool: any) => { - return !tool.dataset - }).map((tool: any) => { - const toolInCollectionList = collectionList.find(c => tool.provider_id === c.id) - return { - ...tool, - isDeleted: res.deleted_tools?.some((deletedTool: any) => deletedTool.id === tool.id && deletedTool.tool_name === tool.tool_name), - notAuthor: toolInCollectionList?.is_team_authorization === false, - ...(tool.provider_type === 'builtin' ? { - provider_id: correctToolProvider(tool.provider_name, !!toolInCollectionList), - provider_name: correctToolProvider(tool.provider_name, !!toolInCollectionList), - } : {}), - } - }), - } : DEFAULT_AGENT_SETTING, + const config: PublishConfig = { + modelConfig: { + provider: correctModelProvider(model.provider), + model_id: model.name, + mode: model.mode, + configs: { + prompt_template: modelConfig.pre_prompt || '', + prompt_variables: userInputsFormToPromptVariables( + ([ + ...modelConfig.user_input_form, + ...( + modelConfig.external_data_tools?.length + ? modelConfig.external_data_tools.map((item: any) => { + return { + external_data_tool: { + variable: item.variable as string, + label: item.label as string, + enabled: item.enabled, + type: item.type as string, + config: item.config, + required: true, + icon: item.icon, + icon_background: item.icon_background, + }, + } + }) + : [] + ), + ]) as unknown as UserInputFormItem[], + modelConfig.dataset_query_variable, + ), }, - completionParams: model.completion_params, - } + more_like_this: modelConfig.more_like_this ?? { enabled: false }, + opening_statement: modelConfig.opening_statement, + suggested_questions: modelConfig.suggested_questions ?? [], + sensitive_word_avoidance: modelConfig.sensitive_word_avoidance, + speech_to_text: modelConfig.speech_to_text, + text_to_speech: modelConfig.text_to_speech, + file_upload: modelConfig.file_upload ?? null, + suggested_questions_after_answer: modelConfig.suggested_questions_after_answer ?? { enabled: false }, + retriever_resource: modelConfig.retriever_resource, + annotation_reply: modelConfig.annotation_reply ?? null, + external_data_tools: modelConfig.external_data_tools ?? [], + system_parameters: modelConfig.system_parameters, + dataSets: datasets || [], + agentConfig: res.mode === AppModeEnum.AGENT_CHAT ? { + max_iteration: DEFAULT_AGENT_SETTING.max_iteration, + ...modelConfig.agent_mode, + // remove dataset + enabled: true, // modelConfig.agent_mode?.enabled is not correct. old app: the value of app with dataset's is always true + tools: (modelConfig.agent_mode?.tools ?? []).filter((tool: any) => { + return !tool.dataset + }).map((tool: any) => { + const toolInCollectionList = collectionList.find(c => tool.provider_id === c.id) + return { + ...tool, + isDeleted: res.deleted_tools?.some((deletedTool: any) => deletedTool.id === tool.id && deletedTool.tool_name === tool.tool_name) ?? false, + notAuthor: toolInCollectionList?.is_team_authorization === false, + ...(tool.provider_type === 'builtin' ? { + provider_id: correctToolProvider(tool.provider_name, !!toolInCollectionList), + provider_name: correctToolProvider(tool.provider_name, !!toolInCollectionList), + } : {}), + } + }), + strategy: modelConfig.agent_mode?.strategy ?? AgentStrategy.react, + } : DEFAULT_AGENT_SETTING, + }, + completionParams: model.completion_params, + } - if (modelConfig.file_upload) - handleSetVisionConfig(modelConfig.file_upload.image, true) + if (modelConfig.file_upload) + handleSetVisionConfig(modelConfig.file_upload.image, true) - syncToPublishedConfig(config) - setPublishedConfig(config) - const retrievalConfig = getMultipleRetrievalConfig({ - ...modelConfig.dataset_configs, - reranking_model: modelConfig.dataset_configs.reranking_model && { - provider: modelConfig.dataset_configs.reranking_model.reranking_provider_name, - model: modelConfig.dataset_configs.reranking_model.reranking_model_name, - }, - }, datasets, datasets, { - provider: currentRerankProvider?.provider, - model: currentRerankModel?.model, - }) - setDatasetConfigs({ - retrieval_model: RETRIEVE_TYPE.multiWay, - ...modelConfig.dataset_configs, - ...retrievalConfig, - ...(retrievalConfig.reranking_model ? { - reranking_model: { - reranking_model_name: retrievalConfig.reranking_model.model, - reranking_provider_name: correctModelProvider(retrievalConfig.reranking_model.provider), - }, - } : {}), - }) - setHasFetchedDetail(true) + syncToPublishedConfig(config) + setPublishedConfig(config) + const retrievalConfig = getMultipleRetrievalConfig({ + ...modelConfig.dataset_configs, + reranking_model: modelConfig.dataset_configs.reranking_model && { + provider: modelConfig.dataset_configs.reranking_model.reranking_provider_name, + model: modelConfig.dataset_configs.reranking_model.reranking_model_name, + }, + }, datasets, datasets, { + provider: currentRerankProvider?.provider, + model: currentRerankModel?.model, }) + const datasetConfigsToSet = { + ...modelConfig.dataset_configs, + ...retrievalConfig, + ...(retrievalConfig.reranking_model ? { + reranking_model: { + reranking_model_name: retrievalConfig.reranking_model.model, + reranking_provider_name: correctModelProvider(retrievalConfig.reranking_model.provider), + }, + } : {}), + } as DatasetConfigs + datasetConfigsToSet.retrieval_model = datasetConfigsToSet.retrieval_model ?? RETRIEVE_TYPE.multiWay + setDatasetConfigs(datasetConfigsToSet) + setHasFetchedDetail(true) })() }, [appId]) const promptEmpty = (() => { - if (mode !== AppType.completion) + if (mode !== AppModeEnum.COMPLETION) return false if (isAdvancedMode) { @@ -724,7 +737,7 @@ const Configuration: FC = () => { else { return !modelConfig.configs.prompt_template } })() const cannotPublish = (() => { - if (mode !== AppType.completion) { + if (mode !== AppModeEnum.COMPLETION) { if (!isAdvancedMode) return false @@ -739,7 +752,7 @@ const Configuration: FC = () => { } else { return promptEmpty } })() - const contextVarEmpty = mode === AppType.completion && dataSets.length > 0 && !hasSetContextVar + const contextVarEmpty = mode === AppModeEnum.COMPLETION && dataSets.length > 0 && !hasSetContextVar const onPublish = async (modelAndParameter?: ModelAndParameter, features?: FeaturesData) => { const modelId = modelAndParameter?.model || modelConfig.model_id const promptTemplate = modelConfig.configs.prompt_template @@ -749,7 +762,7 @@ const Configuration: FC = () => { notify({ type: 'error', message: t('appDebug.otherError.promptNoBeEmpty') }) return } - if (isAdvancedMode && mode !== AppType.completion) { + if (isAdvancedMode && mode !== AppModeEnum.COMPLETION) { if (modelModeType === ModelModeType.completion) { if (!hasSetBlockStatus.history) { notify({ type: 'error', message: t('appDebug.otherError.historyNoBeEmpty') }) @@ -780,8 +793,8 @@ const Configuration: FC = () => { // Simple Mode prompt pre_prompt: !isAdvancedMode ? promptTemplate : '', prompt_type: promptMode, - chat_prompt_config: {}, - completion_prompt_config: {}, + chat_prompt_config: isAdvancedMode ? chatPromptConfig : clone(DEFAULT_CHAT_PROMPT_CONFIG), + completion_prompt_config: isAdvancedMode ? completionPromptConfig : clone(DEFAULT_COMPLETION_PROMPT_CONFIG), user_input_form: promptVariablesToUserInputsForm(promptVariables), dataset_query_variable: contextVar || '', // features @@ -798,6 +811,7 @@ const Configuration: FC = () => { ...modelConfig.agentConfig, strategy: isFunctionCall ? AgentStrategy.functionCall : AgentStrategy.react, }, + external_data_tools: externalDataToolsConfig, model: { provider: modelAndParameter?.provider || modelConfig.provider, name: modelId, @@ -810,11 +824,7 @@ const Configuration: FC = () => { datasets: [...postDatasets], } as any, }, - } - - if (isAdvancedMode) { - data.chat_prompt_config = chatPromptConfig - data.completion_prompt_config = completionPromptConfig + system_parameters: modelConfig.system_parameters, } await updateAppModelConfig({ url: `/apps/${appId}/model-config`, body: data }) @@ -974,7 +984,6 @@ const Configuration: FC = () => { <> {
setShowAccountSettingModal({ payload: 'provider' })} + onSetting={() => setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.PROVIDER })} inputs={inputs} modelParameterParams={{ setModel: setModel as any, @@ -1033,7 +1042,7 @@ const Configuration: FC = () => { content={t('appDebug.trailUseGPT4Info.description')} isShow={showUseGPT4Confirm} onConfirm={() => { - setShowAccountSettingModal({ payload: 'provider' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.PROVIDER }) setShowUseGPT4Confirm(false) }} onCancel={() => setShowUseGPT4Confirm(false)} @@ -1065,7 +1074,7 @@ const Configuration: FC = () => { setShowAccountSettingModal({ payload: 'provider' })} + onSetting={() => setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.PROVIDER })} inputs={inputs} modelParameterParams={{ setModel: setModel as any, @@ -1082,7 +1091,7 @@ const Configuration: FC = () => { show inWorkflow={false} showFileUpload={false} - isChatMode={mode !== 'completion'} + isChatMode={mode !== AppModeEnum.COMPLETION} disabled={false} onChange={handleFeaturesChange} onClose={() => setShowAppConfigureFeaturesModal(false)} diff --git a/web/app/components/app/configuration/prompt-value-panel/index.tsx b/web/app/components/app/configuration/prompt-value-panel/index.tsx index 9b9b5fd66e..3d4c097212 100644 --- a/web/app/components/app/configuration/prompt-value-panel/index.tsx +++ b/web/app/components/app/configuration/prompt-value-panel/index.tsx @@ -10,7 +10,7 @@ import { } from '@remixicon/react' import ConfigContext from '@/context/debug-configuration' import type { Inputs } from '@/models/debug' -import { AppType, ModelModeType } from '@/types/app' +import { AppModeEnum, ModelModeType } from '@/types/app' import Select from '@/app/components/base/select' import Button from '@/app/components/base/button' import Input from '@/app/components/base/input' @@ -25,7 +25,7 @@ import cn from '@/utils/classnames' import BoolInput from '@/app/components/workflow/nodes/_base/components/before-run-form/bool-input' export type IPromptValuePanelProps = { - appType: AppType + appType: AppModeEnum onSend?: () => void inputs: Inputs visionConfig: VisionSettings @@ -55,7 +55,7 @@ const PromptValuePanel: FC = ({ }, [promptVariables]) const canNotRun = useMemo(() => { - if (mode !== AppType.completion) + if (mode !== AppModeEnum.COMPLETION) return true if (isAdvancedMode) { @@ -95,8 +95,8 @@ const PromptValuePanel: FC = ({
setUserInputFieldCollapse(!userInputFieldCollapse)}>
{t('appDebug.inputs.userInputField')}
- {userInputFieldCollapse && } - {!userInputFieldCollapse && } + {userInputFieldCollapse && } + {!userInputFieldCollapse && }
{!userInputFieldCollapse && (
{t('appDebug.inputs.completionVarTip')}
@@ -221,7 +221,7 @@ const PromptValuePanel: FC = ({
([]) + const [currentType, setCurrentType] = useState([]) const [currCategory, setCurrCategory] = useTabSearchParams({ defaultTab: allCategoriesEn, disableSearchParams: true, @@ -93,15 +93,15 @@ const Apps = ({ if (currentType.length === 0) return filteredByCategory return filteredByCategory.filter((item) => { - if (currentType.includes('chat') && item.app.mode === 'chat') + if (currentType.includes(AppModeEnum.CHAT) && item.app.mode === AppModeEnum.CHAT) return true - if (currentType.includes('advanced-chat') && item.app.mode === 'advanced-chat') + if (currentType.includes(AppModeEnum.ADVANCED_CHAT) && item.app.mode === AppModeEnum.ADVANCED_CHAT) return true - if (currentType.includes('agent-chat') && item.app.mode === 'agent-chat') + if (currentType.includes(AppModeEnum.AGENT_CHAT) && item.app.mode === AppModeEnum.AGENT_CHAT) return true - if (currentType.includes('completion') && item.app.mode === 'completion') + if (currentType.includes(AppModeEnum.COMPLETION) && item.app.mode === AppModeEnum.COMPLETION) return true - if (currentType.includes('workflow') && item.app.mode === 'workflow') + if (currentType.includes(AppModeEnum.WORKFLOW) && item.app.mode === AppModeEnum.WORKFLOW) return true return false }) diff --git a/web/app/components/app/create-app-modal/index.tsx b/web/app/components/app/create-app-modal/index.tsx index 3a07e6e0a1..10fc099f9f 100644 --- a/web/app/components/app/create-app-modal/index.tsx +++ b/web/app/components/app/create-app-modal/index.tsx @@ -18,7 +18,7 @@ import { basePath } from '@/utils/var' import { useAppContext } from '@/context/app-context' import { useProviderContext } from '@/context/provider-context' import { ToastContext } from '@/app/components/base/toast' -import type { AppMode } from '@/types/app' +import { AppModeEnum } from '@/types/app' import { createApp } from '@/service/apps' import Input from '@/app/components/base/input' import Textarea from '@/app/components/base/textarea' @@ -35,7 +35,7 @@ type CreateAppProps = { onSuccess: () => void onClose: () => void onCreateFromTemplate?: () => void - defaultAppMode?: AppMode + defaultAppMode?: AppModeEnum } function CreateApp({ onClose, onSuccess, onCreateFromTemplate, defaultAppMode }: CreateAppProps) { @@ -43,7 +43,7 @@ function CreateApp({ onClose, onSuccess, onCreateFromTemplate, defaultAppMode }: const { push } = useRouter() const { notify } = useContext(ToastContext) - const [appMode, setAppMode] = useState(defaultAppMode || 'advanced-chat') + const [appMode, setAppMode] = useState(defaultAppMode || AppModeEnum.ADVANCED_CHAT) const [appIcon, setAppIcon] = useState({ type: 'emoji', icon: '🤖', background: '#FFEAD5' }) const [showAppIconPicker, setShowAppIconPicker] = useState(false) const [name, setName] = useState('') @@ -57,7 +57,7 @@ function CreateApp({ onClose, onSuccess, onCreateFromTemplate, defaultAppMode }: const isCreatingRef = useRef(false) useEffect(() => { - if (appMode === 'chat' || appMode === 'agent-chat' || appMode === 'completion') + if (appMode === AppModeEnum.CHAT || appMode === AppModeEnum.AGENT_CHAT || appMode === AppModeEnum.COMPLETION) setIsAppTypeExpanded(true) }, [appMode]) @@ -118,24 +118,24 @@ function CreateApp({ onClose, onSuccess, onCreateFromTemplate, defaultAppMode }:
} onClick={() => { - setAppMode('workflow') + setAppMode(AppModeEnum.WORKFLOW) }} />
} onClick={() => { - setAppMode('advanced-chat') + setAppMode(AppModeEnum.ADVANCED_CHAT) }} />
@@ -152,34 +152,34 @@ function CreateApp({ onClose, onSuccess, onCreateFromTemplate, defaultAppMode }: {isAppTypeExpanded && (
} onClick={() => { - setAppMode('chat') + setAppMode(AppModeEnum.CHAT) }} />
} onClick={() => { - setAppMode('agent-chat') + setAppMode(AppModeEnum.AGENT_CHAT) }} />
} onClick={() => { - setAppMode('completion') + setAppMode(AppModeEnum.COMPLETION) }} />
)} @@ -255,11 +255,11 @@ function CreateApp({ onClose, onSuccess, onCreateFromTemplate, defaultAppMode }:
- - - - - + + + + +
@@ -309,16 +309,16 @@ function AppTypeCard({ icon, title, description, active, onClick }: AppTypeCardP } -function AppPreview({ mode }: { mode: AppMode }) { +function AppPreview({ mode }: { mode: AppModeEnum }) { const { t } = useTranslation() const docLink = useDocLink() const modeToPreviewInfoMap = { - 'chat': { + [AppModeEnum.CHAT]: { title: t('app.types.chatbot'), description: t('app.newApp.chatbotUserDescription'), link: docLink('/guides/application-orchestrate/chatbot-application'), }, - 'advanced-chat': { + [AppModeEnum.ADVANCED_CHAT]: { title: t('app.types.advanced'), description: t('app.newApp.advancedUserDescription'), link: docLink('/guides/workflow/README', { @@ -326,12 +326,12 @@ function AppPreview({ mode }: { mode: AppMode }) { 'ja-JP': '/guides/workflow/concepts', }), }, - 'agent-chat': { + [AppModeEnum.AGENT_CHAT]: { title: t('app.types.agent'), description: t('app.newApp.agentUserDescription'), link: docLink('/guides/application-orchestrate/agent'), }, - 'completion': { + [AppModeEnum.COMPLETION]: { title: t('app.newApp.completeApp'), description: t('app.newApp.completionUserDescription'), link: docLink('/guides/application-orchestrate/text-generator', { @@ -339,7 +339,7 @@ function AppPreview({ mode }: { mode: AppMode }) { 'ja-JP': '/guides/application-orchestrate/README', }), }, - 'workflow': { + [AppModeEnum.WORKFLOW]: { title: t('app.types.workflow'), description: t('app.newApp.workflowUserDescription'), link: docLink('/guides/workflow/README', { @@ -358,14 +358,14 @@ function AppPreview({ mode }: { mode: AppMode }) { } -function AppScreenShot({ mode, show }: { mode: AppMode; show: boolean }) { +function AppScreenShot({ mode, show }: { mode: AppModeEnum; show: boolean }) { const { theme } = useTheme() const modeToImageMap = { - 'chat': 'Chatbot', - 'advanced-chat': 'Chatflow', - 'agent-chat': 'Agent', - 'completion': 'TextGenerator', - 'workflow': 'Workflow', + [AppModeEnum.CHAT]: 'Chatbot', + [AppModeEnum.ADVANCED_CHAT]: 'Chatflow', + [AppModeEnum.AGENT_CHAT]: 'Agent', + [AppModeEnum.COMPLETION]: 'TextGenerator', + [AppModeEnum.WORKFLOW]: 'Workflow', } return diff --git a/web/app/components/app/create-from-dsl-modal/index.tsx b/web/app/components/app/create-from-dsl-modal/index.tsx index e1a556a709..0c137abb71 100644 --- a/web/app/components/app/create-from-dsl-modal/index.tsx +++ b/web/app/components/app/create-from-dsl-modal/index.tsx @@ -132,8 +132,6 @@ const CreateFromDSLModal = ({ show, onSuccess, onClose, activeTab = CreateFromDS importedVersion: imported_dsl_version ?? '', systemVersion: current_dsl_version ?? '', }) - if (onClose) - onClose() setTimeout(() => { setShowErrorModal(true) }, 300) diff --git a/web/app/components/app/log-annotation/index.tsx b/web/app/components/app/log-annotation/index.tsx index 12a611eea8..c0b0854b29 100644 --- a/web/app/components/app/log-annotation/index.tsx +++ b/web/app/components/app/log-annotation/index.tsx @@ -11,6 +11,7 @@ import Loading from '@/app/components/base/loading' import { PageType } from '@/app/components/base/features/new-feature-panel/annotation-reply/type' import TabSlider from '@/app/components/base/tab-slider-plain' import { useStore as useAppStore } from '@/app/components/app/store' +import { AppModeEnum } from '@/types/app' type Props = { pageType: PageType @@ -24,7 +25,7 @@ const LogAnnotation: FC = ({ const appDetail = useAppStore(state => state.appDetail) const options = useMemo(() => { - if (appDetail?.mode === 'completion') + if (appDetail?.mode === AppModeEnum.COMPLETION) return [{ value: PageType.log, text: t('appLog.title') }] return [ { value: PageType.log, text: t('appLog.title') }, @@ -42,7 +43,7 @@ const LogAnnotation: FC = ({ return (
- {appDetail.mode !== 'workflow' && ( + {appDetail.mode !== AppModeEnum.WORKFLOW && ( = ({ options={options} /> )} -
- {pageType === PageType.log && appDetail.mode !== 'workflow' && ()} +
+ {pageType === PageType.log && appDetail.mode !== AppModeEnum.WORKFLOW && ()} {pageType === PageType.annotation && ()} - {pageType === PageType.log && appDetail.mode === 'workflow' && ()} + {pageType === PageType.log && appDetail.mode === AppModeEnum.WORKFLOW && ()}
) diff --git a/web/app/components/app/log/empty-element.tsx b/web/app/components/app/log/empty-element.tsx new file mode 100644 index 0000000000..ddddacd873 --- /dev/null +++ b/web/app/components/app/log/empty-element.tsx @@ -0,0 +1,42 @@ +'use client' +import type { FC, SVGProps } from 'react' +import React from 'react' +import Link from 'next/link' +import { Trans, useTranslation } from 'react-i18next' +import { basePath } from '@/utils/var' +import { getRedirectionPath } from '@/utils/app-redirection' +import type { App } from '@/types/app' +import { AppModeEnum } from '@/types/app' + +const ThreeDotsIcon = ({ className }: SVGProps) => { + return + + +} + +const EmptyElement: FC<{ appDetail: App }> = ({ appDetail }) => { + const { t } = useTranslation() + + const getWebAppType = (appType: AppModeEnum) => { + if (appType !== AppModeEnum.COMPLETION && appType !== AppModeEnum.WORKFLOW) + return AppModeEnum.CHAT + return appType + } + + return
+
+ {t('appLog.table.empty.element.title')} +
+ , + testLink: , + }} + /> +
+
+
+} + +export default React.memo(EmptyElement) diff --git a/web/app/components/app/log/index.tsx b/web/app/components/app/log/index.tsx index 13be294bef..55a3f7d12d 100644 --- a/web/app/components/app/log/index.tsx +++ b/web/app/components/app/log/index.tsx @@ -1,21 +1,20 @@ 'use client' -import type { FC, SVGProps } from 'react' +import type { FC } from 'react' import React, { useState } from 'react' import useSWR from 'swr' -import Link from 'next/link' -import { usePathname } from 'next/navigation' import { useDebounce } from 'ahooks' import { omit } from 'lodash-es' import dayjs from 'dayjs' -import { basePath } from '@/utils/var' -import { Trans, useTranslation } from 'react-i18next' +import { useTranslation } from 'react-i18next' import List from './list' import Filter, { TIME_PERIOD_MAPPING } from './filter' +import EmptyElement from './empty-element' import Pagination from '@/app/components/base/pagination' import Loading from '@/app/components/base/loading' import { fetchChatConversations, fetchCompletionConversations } from '@/service/log' import { APP_PAGE_LIMIT } from '@/config' -import type { App, AppMode } from '@/types/app' +import type { App } from '@/types/app' +import { AppModeEnum } from '@/types/app' export type ILogsProps = { appDetail: App } @@ -27,30 +26,6 @@ export type QueryParam = { sort_by?: string } -const ThreeDotsIcon = ({ className }: SVGProps) => { - return - - -} - -const EmptyElement: FC<{ appUrl: string }> = ({ appUrl }) => { - const { t } = useTranslation() - const pathname = usePathname() - const pathSegments = pathname.split('/') - pathSegments.pop() - return
-
- {t('appLog.table.empty.element.title')} -
- , testLink: }} - /> -
-
-
-} - const Logs: FC = ({ appDetail }) => { const { t } = useTranslation() const [queryParams, setQueryParams] = useState({ @@ -63,7 +38,7 @@ const Logs: FC = ({ appDetail }) => { const debouncedQueryParams = useDebounce(queryParams, { wait: 500 }) // Get the app type first - const isChatMode = appDetail.mode !== 'completion' + const isChatMode = appDetail.mode !== AppModeEnum.COMPLETION const query = { page: currPage + 1, @@ -78,12 +53,6 @@ const Logs: FC = ({ appDetail }) => { ...omit(debouncedQueryParams, ['period']), } - const getWebAppType = (appType: AppMode) => { - if (appType !== 'completion' && appType !== 'workflow') - return 'chat' - return appType - } - // When the details are obtained, proceed to the next request const { data: chatConversations, mutate: mutateChatList } = useSWR(() => isChatMode ? { @@ -110,7 +79,7 @@ const Logs: FC = ({ appDetail }) => { ? : total > 0 ? - : + : } {/* Show Pagination only if the total is more than the limit */} {(total && total > APP_PAGE_LIMIT) diff --git a/web/app/components/app/log/list.tsx b/web/app/components/app/log/list.tsx index 8b3370b678..10b78a043e 100644 --- a/web/app/components/app/log/list.tsx +++ b/web/app/components/app/log/list.tsx @@ -14,12 +14,13 @@ import timezone from 'dayjs/plugin/timezone' import { createContext, useContext } from 'use-context-selector' import { useShallow } from 'zustand/react/shallow' import { useTranslation } from 'react-i18next' +import { usePathname, useRouter, useSearchParams } from 'next/navigation' import type { ChatItemInTree } from '../../base/chat/types' import Indicator from '../../header/indicator' import VarPanel from './var-panel' import type { FeedbackFunc, FeedbackType, IChatItem, SubmitAnnotationFunc } from '@/app/components/base/chat/chat/type' import type { Annotation, ChatConversationGeneralDetail, ChatConversationsResponse, ChatMessage, ChatMessagesRequest, CompletionConversationGeneralDetail, CompletionConversationsResponse, LogAnnotation } from '@/models/log' -import type { App } from '@/types/app' +import { type App, AppModeEnum } from '@/types/app' import ActionButton from '@/app/components/base/action-button' import Loading from '@/app/components/base/loading' import Drawer from '@/app/components/base/drawer' @@ -41,6 +42,12 @@ import { getProcessedFilesFromResponse } from '@/app/components/base/file-upload import cn from '@/utils/classnames' import { noop } from 'lodash-es' import PromptLogModal from '../../base/prompt-log-modal' +import { WorkflowContextProvider } from '@/app/components/workflow/context' +import { AppSourceType } from '@/service/share' + +type AppStoreState = ReturnType +type ConversationListItem = ChatConversationGeneralDetail | CompletionConversationGeneralDetail +type ConversationSelection = ConversationListItem | { id: string; isPlaceholder?: true } dayjs.extend(utc) dayjs.extend(timezone) @@ -201,7 +208,7 @@ function DetailPanel({ detail, onFeedback }: IDetailPanel) { const { formatTime } = useTimestamp() const { onClose, appDetail } = useContext(DrawerContext) const { notify } = useContext(ToastContext) - const { currentLogItem, setCurrentLogItem, showMessageLogModal, setShowMessageLogModal, showPromptLogModal, setShowPromptLogModal, currentLogModalActiveTab } = useAppStore(useShallow(state => ({ + const { currentLogItem, setCurrentLogItem, showMessageLogModal, setShowMessageLogModal, showPromptLogModal, setShowPromptLogModal, currentLogModalActiveTab } = useAppStore(useShallow((state: AppStoreState) => ({ currentLogItem: state.currentLogItem, setCurrentLogItem: state.setCurrentLogItem, showMessageLogModal: state.showMessageLogModal, @@ -369,7 +376,7 @@ function DetailPanel({ detail, onFeedback }: IDetailPanel) { // Only load initial messages, don't auto-load more useEffect(() => { - if (appDetail?.id && detail.id && appDetail?.mode !== 'completion' && !fetchInitiated.current) { + if (appDetail?.id && detail.id && appDetail?.mode !== AppModeEnum.COMPLETION && !fetchInitiated.current) { // Mark as initialized, but don't auto-load more messages fetchInitiated.current = true // Still call fetchData to get initial messages @@ -578,8 +585,8 @@ function DetailPanel({ detail, onFeedback }: IDetailPanel) { } }, [hasMore, isLoading, loadMoreMessages]) - const isChatMode = appDetail?.mode !== 'completion' - const isAdvanced = appDetail?.mode === 'advanced-chat' + const isChatMode = appDetail?.mode !== AppModeEnum.COMPLETION + const isAdvanced = appDetail?.mode === AppModeEnum.ADVANCED_CHAT const varList = (detail.model_config as any).user_input_form?.map((item: any) => { const itemContent = item[Object.keys(item)[0]] @@ -683,12 +690,12 @@ function DetailPanel({ detail, onFeedback }: IDetailPanel) { }}>
item.from_source === 'admin')} onFeedback={feedback => onFeedback(detail.message.id, feedback)} @@ -774,15 +781,17 @@ function DetailPanel({ detail, onFeedback }: IDetailPanel) { } {showMessageLogModal && ( - { - setCurrentLogItem() - setShowMessageLogModal(false) - }} - defaultTab={currentLogModalActiveTab} - /> + + { + setCurrentLogItem() + setShowMessageLogModal(false) + }} + defaultTab={currentLogModalActiveTab} + /> + )} {!isChatMode && showPromptLogModal && ( = ({ logs, appDetail, onRefresh }) => { const { t } = useTranslation() const { formatTime } = useTimestamp() + const router = useRouter() + const pathname = usePathname() + const searchParams = useSearchParams() + const conversationIdInUrl = searchParams.get('conversation_id') ?? undefined const media = useBreakpoints() const isMobile = media === MediaType.mobile const [showDrawer, setShowDrawer] = useState(false) // Whether to display the chat details drawer - const [currentConversation, setCurrentConversation] = useState() // Currently selected conversation - const isChatMode = appDetail.mode !== 'completion' // Whether the app is a chat app - const isChatflow = appDetail.mode === 'advanced-chat' // Whether the app is a chatflow app - const { setShowPromptLogModal, setShowAgentLogModal, setShowMessageLogModal } = useAppStore(useShallow(state => ({ + const [currentConversation, setCurrentConversation] = useState() // Currently selected conversation + const closingConversationIdRef = useRef(null) + const pendingConversationIdRef = useRef(null) + const pendingConversationCacheRef = useRef(undefined) + const isChatMode = appDetail.mode !== AppModeEnum.COMPLETION // Whether the app is a chat app + const isChatflow = appDetail.mode === AppModeEnum.ADVANCED_CHAT // Whether the app is a chatflow app + const { setShowPromptLogModal, setShowAgentLogModal, setShowMessageLogModal } = useAppStore(useShallow((state: AppStoreState) => ({ setShowPromptLogModal: state.setShowPromptLogModal, setShowAgentLogModal: state.setShowAgentLogModal, setShowMessageLogModal: state.setShowMessageLogModal, }))) + const activeConversationId = conversationIdInUrl ?? pendingConversationIdRef.current ?? currentConversation?.id + + const buildUrlWithConversation = useCallback((conversationId?: string) => { + const params = new URLSearchParams(searchParams.toString()) + if (conversationId) + params.set('conversation_id', conversationId) + else + params.delete('conversation_id') + + const queryString = params.toString() + return queryString ? `${pathname}?${queryString}` : pathname + }, [pathname, searchParams]) + + const handleRowClick = useCallback((log: ConversationListItem) => { + if (conversationIdInUrl === log.id) { + if (!showDrawer) + setShowDrawer(true) + + if (!currentConversation || currentConversation.id !== log.id) + setCurrentConversation(log) + return + } + + pendingConversationIdRef.current = log.id + pendingConversationCacheRef.current = log + if (!showDrawer) + setShowDrawer(true) + + if (currentConversation?.id !== log.id) + setCurrentConversation(undefined) + + router.push(buildUrlWithConversation(log.id), { scroll: false }) + }, [buildUrlWithConversation, conversationIdInUrl, currentConversation, router, showDrawer]) + + const currentConversationId = currentConversation?.id + + useEffect(() => { + if (!conversationIdInUrl) { + if (pendingConversationIdRef.current) + return + + if (showDrawer || currentConversationId) { + setShowDrawer(false) + setCurrentConversation(undefined) + } + closingConversationIdRef.current = null + pendingConversationCacheRef.current = undefined + return + } + + if (closingConversationIdRef.current === conversationIdInUrl) + return + + if (pendingConversationIdRef.current === conversationIdInUrl) + pendingConversationIdRef.current = null + + const matchedConversation = logs?.data?.find((item: ConversationListItem) => item.id === conversationIdInUrl) + const nextConversation: ConversationSelection = matchedConversation + ?? pendingConversationCacheRef.current + ?? { id: conversationIdInUrl, isPlaceholder: true } + + if (!showDrawer) + setShowDrawer(true) + + if (!currentConversation || currentConversation.id !== conversationIdInUrl || (!('created_at' in currentConversation) && matchedConversation)) + setCurrentConversation(nextConversation) + + if (pendingConversationCacheRef.current?.id === conversationIdInUrl || matchedConversation) + pendingConversationCacheRef.current = undefined + }, [conversationIdInUrl, currentConversation, isChatMode, logs?.data, showDrawer]) + + const onCloseDrawer = useCallback(() => { + onRefresh() + setShowDrawer(false) + setCurrentConversation(undefined) + setShowPromptLogModal(false) + setShowAgentLogModal(false) + setShowMessageLogModal(false) + pendingConversationIdRef.current = null + pendingConversationCacheRef.current = undefined + closingConversationIdRef.current = conversationIdInUrl ?? null + + if (conversationIdInUrl) + router.replace(buildUrlWithConversation(), { scroll: false }) + }, [buildUrlWithConversation, conversationIdInUrl, onRefresh, router, setShowAgentLogModal, setShowMessageLogModal, setShowPromptLogModal]) + // Annotated data needs to be highlighted const renderTdValue = (value: string | number | null, isEmptyStyle: boolean, isHighlight = false, annotation?: LogAnnotation) => { return ( @@ -925,15 +1027,6 @@ const ConversationList: FC = ({ logs, appDetail, onRefresh }) ) } - const onCloseDrawer = () => { - onRefresh() - setShowDrawer(false) - setCurrentConversation(undefined) - setShowPromptLogModal(false) - setShowAgentLogModal(false) - setShowMessageLogModal(false) - } - if (!logs) return @@ -960,11 +1053,8 @@ const ConversationList: FC = ({ logs, appDetail, onRefresh }) const rightValue = get(log, isChatMode ? 'message_count' : 'message.answer') return { - setShowDrawer(true) - setCurrentConversation(log) - }}> + className={cn('cursor-pointer border-b border-divider-subtle hover:bg-background-default-hover', activeConversationId !== log.id ? '' : 'bg-background-default-hover')} + onClick={() => handleRowClick(log)}> {!log.read_at && (
diff --git a/web/app/components/app/overview/__tests__/toggle-logic.test.ts b/web/app/components/app/overview/__tests__/toggle-logic.test.ts new file mode 100644 index 0000000000..0c1e1ea0d3 --- /dev/null +++ b/web/app/components/app/overview/__tests__/toggle-logic.test.ts @@ -0,0 +1,228 @@ +import { getWorkflowEntryNode } from '@/app/components/workflow/utils/workflow-entry' + +// Mock the getWorkflowEntryNode function +jest.mock('@/app/components/workflow/utils/workflow-entry', () => ({ + getWorkflowEntryNode: jest.fn(), +})) + +const mockGetWorkflowEntryNode = getWorkflowEntryNode as jest.MockedFunction + +describe('App Card Toggle Logic', () => { + beforeEach(() => { + jest.clearAllMocks() + }) + + // Helper function that mirrors the actual logic from app-card.tsx + const calculateToggleState = ( + appMode: string, + currentWorkflow: any, + isCurrentWorkspaceEditor: boolean, + isCurrentWorkspaceManager: boolean, + cardType: 'webapp' | 'api', + ) => { + const isWorkflowApp = appMode === 'workflow' + const appUnpublished = isWorkflowApp && !currentWorkflow?.graph + const hasEntryNode = mockGetWorkflowEntryNode(currentWorkflow?.graph?.nodes || []) + const missingEntryNode = isWorkflowApp && !hasEntryNode + const hasInsufficientPermissions = cardType === 'webapp' ? !isCurrentWorkspaceEditor : !isCurrentWorkspaceManager + const toggleDisabled = hasInsufficientPermissions || appUnpublished || missingEntryNode + const isMinimalState = appUnpublished || missingEntryNode + + return { + toggleDisabled, + isMinimalState, + appUnpublished, + missingEntryNode, + hasInsufficientPermissions, + } + } + + describe('Entry Node Detection Logic', () => { + it('should disable toggle when workflow missing entry node', () => { + mockGetWorkflowEntryNode.mockReturnValue(false) + + const result = calculateToggleState( + 'workflow', + { graph: { nodes: [] } }, + true, + true, + 'webapp', + ) + + expect(result.toggleDisabled).toBe(true) + expect(result.missingEntryNode).toBe(true) + expect(result.isMinimalState).toBe(true) + }) + + it('should enable toggle when workflow has entry node', () => { + mockGetWorkflowEntryNode.mockReturnValue(true) + + const result = calculateToggleState( + 'workflow', + { graph: { nodes: [{ data: { type: 'start' } }] } }, + true, + true, + 'webapp', + ) + + expect(result.toggleDisabled).toBe(false) + expect(result.missingEntryNode).toBe(false) + expect(result.isMinimalState).toBe(false) + }) + }) + + describe('Published State Logic', () => { + it('should disable toggle when workflow unpublished (no graph)', () => { + const result = calculateToggleState( + 'workflow', + null, // No workflow data = unpublished + true, + true, + 'webapp', + ) + + expect(result.toggleDisabled).toBe(true) + expect(result.appUnpublished).toBe(true) + expect(result.isMinimalState).toBe(true) + }) + + it('should disable toggle when workflow unpublished (empty graph)', () => { + const result = calculateToggleState( + 'workflow', + {}, // No graph property = unpublished + true, + true, + 'webapp', + ) + + expect(result.toggleDisabled).toBe(true) + expect(result.appUnpublished).toBe(true) + expect(result.isMinimalState).toBe(true) + }) + + it('should consider published state when workflow has graph', () => { + mockGetWorkflowEntryNode.mockReturnValue(true) + + const result = calculateToggleState( + 'workflow', + { graph: { nodes: [] } }, + true, + true, + 'webapp', + ) + + expect(result.appUnpublished).toBe(false) + }) + }) + + describe('Permissions Logic', () => { + it('should disable webapp toggle when user lacks editor permissions', () => { + mockGetWorkflowEntryNode.mockReturnValue(true) + + const result = calculateToggleState( + 'workflow', + { graph: { nodes: [] } }, + false, // No editor permission + true, + 'webapp', + ) + + expect(result.toggleDisabled).toBe(true) + expect(result.hasInsufficientPermissions).toBe(true) + }) + + it('should disable api toggle when user lacks manager permissions', () => { + mockGetWorkflowEntryNode.mockReturnValue(true) + + const result = calculateToggleState( + 'workflow', + { graph: { nodes: [] } }, + true, + false, // No manager permission + 'api', + ) + + expect(result.toggleDisabled).toBe(true) + expect(result.hasInsufficientPermissions).toBe(true) + }) + + it('should enable toggle when user has proper permissions', () => { + mockGetWorkflowEntryNode.mockReturnValue(true) + + const webappResult = calculateToggleState( + 'workflow', + { graph: { nodes: [] } }, + true, // Has editor permission + false, + 'webapp', + ) + + const apiResult = calculateToggleState( + 'workflow', + { graph: { nodes: [] } }, + false, + true, // Has manager permission + 'api', + ) + + expect(webappResult.toggleDisabled).toBe(false) + expect(apiResult.toggleDisabled).toBe(false) + }) + }) + + describe('Combined Conditions Logic', () => { + it('should handle multiple disable conditions correctly', () => { + mockGetWorkflowEntryNode.mockReturnValue(false) + + const result = calculateToggleState( + 'workflow', + null, // Unpublished + false, // No permissions + false, + 'webapp', + ) + + // All three conditions should be true + expect(result.appUnpublished).toBe(true) + expect(result.missingEntryNode).toBe(true) + expect(result.hasInsufficientPermissions).toBe(true) + expect(result.toggleDisabled).toBe(true) + expect(result.isMinimalState).toBe(true) + }) + + it('should enable when all conditions are satisfied', () => { + mockGetWorkflowEntryNode.mockReturnValue(true) + + const result = calculateToggleState( + 'workflow', + { graph: { nodes: [{ data: { type: 'start' } }] } }, // Published + true, // Has permissions + true, + 'webapp', + ) + + expect(result.appUnpublished).toBe(false) + expect(result.missingEntryNode).toBe(false) + expect(result.hasInsufficientPermissions).toBe(false) + expect(result.toggleDisabled).toBe(false) + expect(result.isMinimalState).toBe(false) + }) + }) + + describe('Non-Workflow Apps', () => { + it('should not check workflow-specific conditions for non-workflow apps', () => { + const result = calculateToggleState( + 'chat', // Non-workflow mode + null, + true, + true, + 'webapp', + ) + + expect(result.appUnpublished).toBe(false) // isWorkflowApp is false + expect(result.missingEntryNode).toBe(false) // isWorkflowApp is false + expect(result.toggleDisabled).toBe(false) + expect(result.isMinimalState).toBe(false) + }) + }) +}) diff --git a/web/app/components/app/overview/apikey-info-panel/index.tsx b/web/app/components/app/overview/apikey-info-panel/index.tsx index 7654d49e99..b50b0077cb 100644 --- a/web/app/components/app/overview/apikey-info-panel/index.tsx +++ b/web/app/components/app/overview/apikey-info-panel/index.tsx @@ -9,6 +9,7 @@ import { LinkExternal02 } from '@/app/components/base/icons/src/vender/line/gene import { IS_CE_EDITION } from '@/config' import { useProviderContext } from '@/context/provider-context' import { useModalContext } from '@/context/modal-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const APIKeyInfoPanel: FC = () => { const isCloud = !IS_CE_EDITION @@ -47,7 +48,7 @@ const APIKeyInfoPanel: FC = () => { - ) - })} -
+ +
+ +
{op.opName}
+
+
+ + ) + })} + + )} {isApp ? ( <> setShowSettingsModal(false)} diff --git a/web/app/components/app/overview/app-chart.tsx b/web/app/components/app/overview/app-chart.tsx index c550f0b23f..8f28e16402 100644 --- a/web/app/components/app/overview/app-chart.tsx +++ b/web/app/components/app/overview/app-chart.tsx @@ -4,6 +4,7 @@ import React from 'react' import ReactECharts from 'echarts-for-react' import type { EChartsOption } from 'echarts' import useSWR from 'swr' +import type { Dayjs } from 'dayjs' import dayjs from 'dayjs' import { get } from 'lodash-es' import Decimal from 'decimal.js' @@ -78,6 +79,16 @@ export type PeriodParams = { } } +export type TimeRange = { + start: Dayjs + end: Dayjs +} + +export type PeriodParamsWithTimeRange = { + name: string + query?: TimeRange +} + export type IBizChartProps = { period: PeriodParams id: string @@ -215,9 +226,7 @@ const Chart: React.FC = ({ formatter(params) { return `
${params.name}
${valueFormatter((params.data as any)[yField])} - ${!CHART_TYPE_CONFIG[chartType].showTokens - ? '' - : ` + ${!CHART_TYPE_CONFIG[chartType].showTokens ? '' : ` ( ~$${get(params.data, 'total_price', 0)} ) diff --git a/web/app/components/app/overview/customize/index.tsx b/web/app/components/app/overview/customize/index.tsx index 11d29bb0c8..e440a8cf26 100644 --- a/web/app/components/app/overview/customize/index.tsx +++ b/web/app/components/app/overview/customize/index.tsx @@ -4,7 +4,7 @@ import React from 'react' import { ArrowTopRightOnSquareIcon } from '@heroicons/react/24/outline' import { useTranslation } from 'react-i18next' import { useDocLink } from '@/context/i18n' -import type { AppMode } from '@/types/app' +import { AppModeEnum } from '@/types/app' import Button from '@/app/components/base/button' import Modal from '@/app/components/base/modal' import Tag from '@/app/components/base/tag' @@ -15,7 +15,7 @@ type IShareLinkProps = { linkUrl: string api_base_url: string appId: string - mode: AppMode + mode: AppModeEnum } const StepNum: FC<{ children: React.ReactNode }> = ({ children }) => @@ -42,7 +42,7 @@ const CustomizeModal: FC = ({ }) => { const { t } = useTranslation() const docLink = useDocLink() - const isChatApp = mode === 'chat' || mode === 'advanced-chat' + const isChatApp = mode === AppModeEnum.CHAT || mode === AppModeEnum.ADVANCED_CHAT return = ({ if (isFreePlan) setShowPricingModal() else - setShowAccountSettingModal({ payload: 'billing' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.BILLING }) }, [isFreePlan, setShowAccountSettingModal, setShowPricingModal]) useEffect(() => { @@ -328,7 +329,7 @@ const SettingsModal: FC = ({
{t(`${prefixSettings}.workflow.subTitle`)}
setInputInfo({ ...inputInfo, show_workflow_steps: v })} /> diff --git a/web/app/components/app/overview/trigger-card.tsx b/web/app/components/app/overview/trigger-card.tsx new file mode 100644 index 0000000000..5a0e387ba2 --- /dev/null +++ b/web/app/components/app/overview/trigger-card.tsx @@ -0,0 +1,224 @@ +'use client' +import React from 'react' +import { useTranslation } from 'react-i18next' +import Link from 'next/link' +import { TriggerAll } from '@/app/components/base/icons/src/vender/workflow' +import Switch from '@/app/components/base/switch' +import type { AppDetailResponse } from '@/models/app' +import type { AppSSO } from '@/types/app' +import { useAppContext } from '@/context/app-context' +import { + type AppTrigger, + useAppTriggers, + useInvalidateAppTriggers, + useUpdateTriggerStatus, +} from '@/service/use-tools' +import { useAllTriggerPlugins } from '@/service/use-triggers' +import { canFindTool } from '@/utils' +import { useTriggerStatusStore } from '@/app/components/workflow/store/trigger-status' +import BlockIcon from '@/app/components/workflow/block-icon' +import { BlockEnum } from '@/app/components/workflow/types' +import { useDocLink } from '@/context/i18n' + +export type ITriggerCardProps = { + appInfo: AppDetailResponse & Partial + onToggleResult?: (err: Error | null, message?: string) => void +} + +const getTriggerIcon = (trigger: AppTrigger, triggerPlugins: any[]) => { + const { trigger_type, status, provider_name } = trigger + + // Status dot styling based on trigger status + const getStatusDot = () => { + if (status === 'enabled') { + return ( +
+ ) + } + else { + return ( +
+ ) + } + } + + // Get BlockEnum type from trigger_type + let blockType: BlockEnum + switch (trigger_type) { + case 'trigger-webhook': + blockType = BlockEnum.TriggerWebhook + break + case 'trigger-schedule': + blockType = BlockEnum.TriggerSchedule + break + case 'trigger-plugin': + blockType = BlockEnum.TriggerPlugin + break + default: + blockType = BlockEnum.TriggerWebhook + } + + let triggerIcon: string | undefined + if (trigger_type === 'trigger-plugin' && provider_name) { + const targetTriggers = triggerPlugins || [] + const foundTrigger = targetTriggers.find(triggerWithProvider => + canFindTool(triggerWithProvider.id, provider_name) + || triggerWithProvider.id.includes(provider_name) + || triggerWithProvider.name === provider_name, + ) + triggerIcon = foundTrigger?.icon + } + + return ( +
+ + {getStatusDot()} +
+ ) +} + +function TriggerCard({ appInfo, onToggleResult }: ITriggerCardProps) { + const { t } = useTranslation() + const docLink = useDocLink() + const appId = appInfo.id + const { isCurrentWorkspaceEditor } = useAppContext() + const { data: triggersResponse, isLoading } = useAppTriggers(appId) + const { mutateAsync: updateTriggerStatus } = useUpdateTriggerStatus() + const invalidateAppTriggers = useInvalidateAppTriggers() + const { data: triggerPlugins } = useAllTriggerPlugins() + + // Zustand store for trigger status sync + const { setTriggerStatus, setTriggerStatuses } = useTriggerStatusStore() + + const triggers = triggersResponse?.data || [] + const triggerCount = triggers.length + + // Sync trigger statuses to Zustand store when data loads initially or after API calls + React.useEffect(() => { + if (triggers.length > 0) { + const statusMap = triggers.reduce((acc, trigger) => { + // Map API status to EntryNodeStatus: only 'enabled' shows green, others show gray + acc[trigger.node_id] = trigger.status === 'enabled' ? 'enabled' : 'disabled' + return acc + }, {} as Record) + + // Only update if there are actual changes to prevent overriding optimistic updates + setTriggerStatuses(statusMap) + } + }, [triggers, setTriggerStatuses]) + + const onToggleTrigger = async (trigger: AppTrigger, enabled: boolean) => { + try { + // Immediately update Zustand store for real-time UI sync + const newStatus = enabled ? 'enabled' : 'disabled' + setTriggerStatus(trigger.node_id, newStatus) + + await updateTriggerStatus({ + appId, + triggerId: trigger.id, + enableTrigger: enabled, + }) + invalidateAppTriggers(appId) + + // Success toast notification + onToggleResult?.(null) + } + catch (error) { + // Rollback Zustand store state on error + const rollbackStatus = enabled ? 'disabled' : 'enabled' + setTriggerStatus(trigger.node_id, rollbackStatus) + + // Error toast notification + onToggleResult?.(error as Error) + } + } + + if (isLoading) { + return ( +
+
+
+
+
+
+
+ ) + } + + return ( +
+
+
+
+
+
+ +
+
+
+ {triggerCount > 0 + ? t('appOverview.overview.triggerInfo.triggersAdded', { count: triggerCount }) + : t('appOverview.overview.triggerInfo.noTriggerAdded') + } +
+
+
+
+
+ + {triggerCount > 0 && ( +
+ {triggers.map(trigger => ( +
+
+
+ {getTriggerIcon(trigger, triggerPlugins || [])} +
+
+ {trigger.title} +
+
+
+
+ {trigger.status === 'enabled' + ? t('appOverview.overview.status.running') + : t('appOverview.overview.status.disable')} +
+
+
+ onToggleTrigger(trigger, enabled)} + disabled={!isCurrentWorkspaceEditor} + /> +
+
+ ))} +
+ )} + + {triggerCount === 0 && ( +
+
+ {t('appOverview.overview.triggerInfo.triggerStatusDescription')}{' '} + + {t('appOverview.overview.triggerInfo.learnAboutTriggers')} + +
+
+ )} +
+
+ ) +} + +export default TriggerCard diff --git a/web/app/components/app/switch-app-modal/index.tsx b/web/app/components/app/switch-app-modal/index.tsx index f1654eb65e..a7e1cea429 100644 --- a/web/app/components/app/switch-app-modal/index.tsx +++ b/web/app/components/app/switch-app-modal/index.tsx @@ -24,6 +24,7 @@ import { AlertTriangle } from '@/app/components/base/icons/src/vender/solid/aler import AppIcon from '@/app/components/base/app-icon' import { useStore as useAppStore } from '@/app/components/app/store' import { noop } from 'lodash-es' +import { AppModeEnum } from '@/types/app' type SwitchAppModalProps = { show: boolean @@ -77,7 +78,7 @@ const SwitchAppModal = ({ show, appDetail, inAppDetail = false, onSuccess, onClo isCurrentWorkspaceEditor, { id: newAppID, - mode: appDetail.mode === 'completion' ? 'workflow' : 'advanced-chat', + mode: appDetail.mode === AppModeEnum.COMPLETION ? AppModeEnum.WORKFLOW : AppModeEnum.ADVANCED_CHAT, }, removeOriginal ? replace : push, ) diff --git a/web/app/components/app/type-selector/index.tsx b/web/app/components/app/type-selector/index.tsx index f8432ceab6..0f6f050953 100644 --- a/web/app/components/app/type-selector/index.tsx +++ b/web/app/components/app/type-selector/index.tsx @@ -9,13 +9,14 @@ import { PortalToFollowElemTrigger, } from '@/app/components/base/portal-to-follow-elem' import { BubbleTextMod, ChatBot, ListSparkle, Logic } from '@/app/components/base/icons/src/vender/solid/communication' -import type { AppMode } from '@/types/app' +import { AppModeEnum } from '@/types/app' + export type AppSelectorProps = { - value: Array + value: Array onChange: (value: AppSelectorProps['value']) => void } -const allTypes: AppMode[] = ['workflow', 'advanced-chat', 'chat', 'agent-chat', 'completion'] +const allTypes: AppModeEnum[] = [AppModeEnum.WORKFLOW, AppModeEnum.ADVANCED_CHAT, AppModeEnum.CHAT, AppModeEnum.AGENT_CHAT, AppModeEnum.COMPLETION] const AppTypeSelector = ({ value, onChange }: AppSelectorProps) => { const [open, setOpen] = useState(false) @@ -66,7 +67,7 @@ const AppTypeSelector = ({ value, onChange }: AppSelectorProps) => { export default AppTypeSelector type AppTypeIconProps = { - type: AppMode + type: AppModeEnum style?: React.CSSProperties className?: string wrapperClassName?: string @@ -75,27 +76,27 @@ type AppTypeIconProps = { export const AppTypeIcon = React.memo(({ type, className, wrapperClassName, style }: AppTypeIconProps) => { const wrapperClassNames = cn('inline-flex h-5 w-5 items-center justify-center rounded-md border border-divider-regular', wrapperClassName) const iconClassNames = cn('h-3.5 w-3.5 text-components-avatar-shape-fill-stop-100', className) - if (type === 'chat') { + if (type === AppModeEnum.CHAT) { return
} - if (type === 'agent-chat') { + if (type === AppModeEnum.AGENT_CHAT) { return
} - if (type === 'advanced-chat') { + if (type === AppModeEnum.ADVANCED_CHAT) { return
} - if (type === 'workflow') { + if (type === AppModeEnum.WORKFLOW) { return
} - if (type === 'completion') { + if (type === AppModeEnum.COMPLETION) { return
@@ -133,7 +134,7 @@ function AppTypeSelectTrigger({ values }: { readonly values: AppSelectorProps['v type AppTypeSelectorItemProps = { checked: boolean - type: AppMode + type: AppModeEnum onClick: () => void } function AppTypeSelectorItem({ checked, type, onClick }: AppTypeSelectorItemProps) { @@ -147,21 +148,21 @@ function AppTypeSelectorItem({ checked, type, onClick }: AppTypeSelectorItemProp } type AppTypeLabelProps = { - type: AppMode + type: AppModeEnum className?: string } export function AppTypeLabel({ type, className }: AppTypeLabelProps) { const { t } = useTranslation() let label = '' - if (type === 'chat') + if (type === AppModeEnum.CHAT) label = t('app.typeSelector.chatbot') - if (type === 'agent-chat') + if (type === AppModeEnum.AGENT_CHAT) label = t('app.typeSelector.agent') - if (type === 'completion') + if (type === AppModeEnum.COMPLETION) label = t('app.typeSelector.completion') - if (type === 'advanced-chat') + if (type === AppModeEnum.ADVANCED_CHAT) label = t('app.typeSelector.advanced') - if (type === 'workflow') + if (type === AppModeEnum.WORKFLOW) label = t('app.typeSelector.workflow') return {label} diff --git a/web/app/components/app/workflow-log/detail.tsx b/web/app/components/app/workflow-log/detail.tsx index 7ce701dd68..1c1ed75e80 100644 --- a/web/app/components/app/workflow-log/detail.tsx +++ b/web/app/components/app/workflow-log/detail.tsx @@ -3,6 +3,7 @@ import type { FC } from 'react' import { useTranslation } from 'react-i18next' import { RiCloseLine, RiPlayLargeLine } from '@remixicon/react' import Run from '@/app/components/workflow/run' +import { WorkflowContextProvider } from '@/app/components/workflow/context' import { useStore } from '@/app/components/app/store' import TooltipPlus from '@/app/components/base/tooltip' import { useRouter } from 'next/navigation' @@ -10,9 +11,10 @@ import { useRouter } from 'next/navigation' type ILogDetail = { runID: string onClose: () => void + canReplay?: boolean } -const DetailPanel: FC = ({ runID, onClose }) => { +const DetailPanel: FC = ({ runID, onClose, canReplay = false }) => { const { t } = useTranslation() const appDetail = useStore(state => state.appDetail) const router = useRouter() @@ -29,24 +31,28 @@ const DetailPanel: FC = ({ runID, onClose }) => {

{t('appLog.runDetail.workflowTitle')}

- - - + + + )}
- + + +
) } diff --git a/web/app/components/app/workflow-log/index.tsx b/web/app/components/app/workflow-log/index.tsx index f58d387d68..30a1974347 100644 --- a/web/app/components/app/workflow-log/index.tsx +++ b/web/app/components/app/workflow-log/index.tsx @@ -1,23 +1,21 @@ 'use client' -import type { FC, SVGProps } from 'react' +import type { FC } from 'react' import React, { useState } from 'react' import useSWR from 'swr' -import { usePathname } from 'next/navigation' import { useDebounce } from 'ahooks' import { omit } from 'lodash-es' import dayjs from 'dayjs' import utc from 'dayjs/plugin/utc' import timezone from 'dayjs/plugin/timezone' -import { Trans, useTranslation } from 'react-i18next' -import Link from 'next/link' +import { useTranslation } from 'react-i18next' import List from './list' -import { basePath } from '@/utils/var' import Filter, { TIME_PERIOD_MAPPING } from './filter' +import EmptyElement from '@/app/components/app/log/empty-element' import Pagination from '@/app/components/base/pagination' import Loading from '@/app/components/base/loading' import { fetchWorkflowLogs } from '@/service/log' import { APP_PAGE_LIMIT } from '@/config' -import type { App, AppMode } from '@/types/app' +import type { App } from '@/types/app' import { useAppContext } from '@/context/app-context' dayjs.extend(utc) @@ -33,29 +31,6 @@ export type QueryParam = { keyword?: string } -const ThreeDotsIcon = ({ className }: SVGProps) => { - return - - -} -const EmptyElement: FC<{ appUrl: string }> = ({ appUrl }) => { - const { t } = useTranslation() - const pathname = usePathname() - const pathSegments = pathname.split('/') - pathSegments.pop() - return
-
- {t('appLog.table.empty.element.title')} -
- , testLink: }} - /> -
-
-
-} - const Logs: FC = ({ appDetail }) => { const { t } = useTranslation() const { userProfile: { timezone } } = useAppContext() @@ -66,6 +41,7 @@ const Logs: FC = ({ appDetail }) => { const query = { page: currPage + 1, + detail: true, limit, ...(debouncedQueryParams.status !== 'all' ? { status: debouncedQueryParams.status } : {}), ...(debouncedQueryParams.keyword ? { keyword: debouncedQueryParams.keyword } : {}), @@ -78,12 +54,6 @@ const Logs: FC = ({ appDetail }) => { ...omit(debouncedQueryParams, ['period', 'status']), } - const getWebAppType = (appType: AppMode) => { - if (appType !== 'completion' && appType !== 'workflow') - return 'chat' - return appType - } - const { data: workflowLogs, mutate } = useSWR({ url: `/apps/${appDetail.id}/workflow-app-logs`, params: query, @@ -101,7 +71,7 @@ const Logs: FC = ({ appDetail }) => { ? : total > 0 ? - : + : } {/* Show Pagination only if the total is more than the limit */} {(total && total > APP_PAGE_LIMIT) diff --git a/web/app/components/app/workflow-log/list.tsx b/web/app/components/app/workflow-log/list.tsx index 395df5da2b..0e9b5dd67f 100644 --- a/web/app/components/app/workflow-log/list.tsx +++ b/web/app/components/app/workflow-log/list.tsx @@ -1,16 +1,19 @@ 'use client' import type { FC } from 'react' -import React, { useState } from 'react' +import React, { useEffect, useState } from 'react' import { useTranslation } from 'react-i18next' +import { ArrowDownIcon } from '@heroicons/react/24/outline' import DetailPanel from './detail' +import TriggerByDisplay from './trigger-by-display' import type { WorkflowAppLogDetail, WorkflowLogsResponse } from '@/models/log' -import type { App } from '@/types/app' +import { type App, AppModeEnum } from '@/types/app' import Loading from '@/app/components/base/loading' import Drawer from '@/app/components/base/drawer' import Indicator from '@/app/components/header/indicator' import useBreakpoints, { MediaType } from '@/hooks/use-breakpoints' import useTimestamp from '@/hooks/use-timestamp' import cn from '@/utils/classnames' +import type { WorkflowRunTriggeredFrom } from '@/models/log' type ILogs = { logs?: WorkflowLogsResponse @@ -29,6 +32,28 @@ const WorkflowAppLogList: FC = ({ logs, appDetail, onRefresh }) => { const [showDrawer, setShowDrawer] = useState(false) const [currentLog, setCurrentLog] = useState() + const [sortOrder, setSortOrder] = useState<'asc' | 'desc'>('desc') + const [localLogs, setLocalLogs] = useState(logs?.data || []) + + useEffect(() => { + if (!logs?.data) { + setLocalLogs([]) + return + } + + const sortedLogs = [...logs.data].sort((a, b) => { + const result = a.created_at - b.created_at + return sortOrder === 'asc' ? result : -result + }) + + setLocalLogs(sortedLogs) + }, [logs?.data, sortOrder]) + + const handleSort = () => { + setSortOrder(sortOrder === 'asc' ? 'desc' : 'asc') + } + + const isWorkflow = appDetail?.mode === AppModeEnum.WORKFLOW const statusTdRender = (status: string) => { if (status === 'succeeded') { @@ -43,7 +68,7 @@ const WorkflowAppLogList: FC = ({ logs, appDetail, onRefresh }) => { return (
- Fail + Failure
) } @@ -88,15 +113,26 @@ const WorkflowAppLogList: FC = ({ logs, appDetail, onRefresh }) => { - {t('appLog.table.header.startTime')} + +
+ {t('appLog.table.header.startTime')} + +
+ {t('appLog.table.header.status')} {t('appLog.table.header.runtime')} {t('appLog.table.header.tokens')} - {t('appLog.table.header.user')} + {t('appLog.table.header.user')} + {isWorkflow && {t('appLog.table.header.triggered_from')}} - {logs.data.map((log: WorkflowAppLogDetail) => { + {localLogs.map((log: WorkflowAppLogDetail) => { const endUser = log.created_by_end_user ? log.created_by_end_user.session_id : log.created_by_account ? log.created_by_account.name : defaultValue return = ({ logs, appDetail, onRefresh }) => { {endUser}
+ {isWorkflow && ( + + + + )} })} @@ -136,7 +177,11 @@ const WorkflowAppLogList: FC = ({ logs, appDetail, onRefresh }) => { footer={null} panelClassName='mt-16 mx-2 sm:mr-2 mb-3 !p-0 !max-w-[600px] rounded-xl border border-components-panel-border' > - +
) diff --git a/web/app/components/app/workflow-log/trigger-by-display.tsx b/web/app/components/app/workflow-log/trigger-by-display.tsx new file mode 100644 index 0000000000..1411503cc2 --- /dev/null +++ b/web/app/components/app/workflow-log/trigger-by-display.tsx @@ -0,0 +1,134 @@ +'use client' +import type { FC } from 'react' +import React from 'react' +import { useTranslation } from 'react-i18next' +import { + Code, + KnowledgeRetrieval, + Schedule, + WebhookLine, + WindowCursor, +} from '@/app/components/base/icons/src/vender/workflow' +import BlockIcon from '@/app/components/workflow/block-icon' +import { BlockEnum } from '@/app/components/workflow/types' +import useTheme from '@/hooks/use-theme' +import type { TriggerMetadata } from '@/models/log' +import { WorkflowRunTriggeredFrom } from '@/models/log' +import { Theme } from '@/types/app' + +type TriggerByDisplayProps = { + triggeredFrom: WorkflowRunTriggeredFrom + className?: string + showText?: boolean + triggerMetadata?: TriggerMetadata +} + +const getTriggerDisplayName = (triggeredFrom: WorkflowRunTriggeredFrom, t: any, metadata?: TriggerMetadata) => { + if (triggeredFrom === WorkflowRunTriggeredFrom.PLUGIN && metadata?.event_name) + return metadata.event_name + + const nameMap: Record = { + 'debugging': t('appLog.triggerBy.debugging'), + 'app-run': t('appLog.triggerBy.appRun'), + 'webhook': t('appLog.triggerBy.webhook'), + 'schedule': t('appLog.triggerBy.schedule'), + 'plugin': t('appLog.triggerBy.plugin'), + 'rag-pipeline-run': t('appLog.triggerBy.ragPipelineRun'), + 'rag-pipeline-debugging': t('appLog.triggerBy.ragPipelineDebugging'), + } + + return nameMap[triggeredFrom] || triggeredFrom +} + +const getPluginIcon = (metadata: TriggerMetadata | undefined, theme: Theme) => { + if (!metadata) + return null + + const icon = theme === Theme.dark + ? metadata.icon_dark || metadata.icon + : metadata.icon || metadata.icon_dark + + if (!icon) + return null + + return ( + + ) +} + +const getTriggerIcon = (triggeredFrom: WorkflowRunTriggeredFrom, metadata: TriggerMetadata | undefined, theme: Theme) => { + switch (triggeredFrom) { + case 'webhook': + return ( +
+ +
+ ) + case 'schedule': + return ( +
+ +
+ ) + case 'plugin': + return getPluginIcon(metadata, theme) || ( + + ) + case 'debugging': + return ( +
+ +
+ ) + case 'rag-pipeline-run': + case 'rag-pipeline-debugging': + return ( +
+ +
+ ) + case 'app-run': + default: + // For user input types (app-run, etc.), use webapp icon + return ( +
+ +
+ ) + } +} + +const TriggerByDisplay: FC = ({ + triggeredFrom, + className = '', + showText = true, + triggerMetadata, +}) => { + const { t } = useTranslation() + const { theme } = useTheme() + + const displayName = getTriggerDisplayName(triggeredFrom, t, triggerMetadata) + const icon = getTriggerIcon(triggeredFrom, triggerMetadata, theme) + + return ( +
+
+ {icon} +
+ {showText && ( + + {displayName} + + )} +
+ ) +} + +export default TriggerByDisplay diff --git a/web/app/components/apps/app-card.tsx b/web/app/components/apps/app-card.tsx index cd3495e3c6..8356cfd31c 100644 --- a/web/app/components/apps/app-card.tsx +++ b/web/app/components/apps/app-card.tsx @@ -6,7 +6,7 @@ import { useRouter } from 'next/navigation' import { useTranslation } from 'react-i18next' import { RiBuildingLine, RiGlobalLine, RiLockLine, RiMoreFill, RiVerifiedBadgeLine } from '@remixicon/react' import cn from '@/utils/classnames' -import type { App } from '@/types/app' +import { type App, AppModeEnum } from '@/types/app' import Toast, { ToastContext } from '@/app/components/base/toast' import { copyApp, deleteApp, exportAppConfig, updateAppInfo } from '@/service/apps' import type { DuplicateAppModalProps } from '@/app/components/app/duplicate-modal' @@ -171,7 +171,7 @@ const AppCard = ({ app, onRefresh }: AppCardProps) => { } const exportCheck = async () => { - if (app.mode !== 'workflow' && app.mode !== 'advanced-chat') { + if (app.mode !== AppModeEnum.WORKFLOW && app.mode !== AppModeEnum.ADVANCED_CHAT) { onExport() return } @@ -269,7 +269,7 @@ const AppCard = ({ app, onRefresh }: AppCardProps) => { - {(app.mode === 'completion' || app.mode === 'chat') && ( + {(app.mode === AppModeEnum.COMPLETION || app.mode === AppModeEnum.CHAT) && ( <> - - : !(isGettingUserCanAccessApp || !userCanAccessApp?.result) && ( - <> + !app.has_draft_trigger && ( + (!systemFeatures.webapp_auth.enabled) + ? <> - ) + : !(isGettingUserCanAccessApp || !userCanAccessApp?.result) && ( + <> + + + + ) + ) } { @@ -425,7 +427,7 @@ const AppCard = ({ app, onRefresh }: AppCardProps) => { ) } popupClassName={ - (app.mode === 'completion' || app.mode === 'chat') + (app.mode === AppModeEnum.COMPLETION || app.mode === AppModeEnum.CHAT) ? '!w-[256px] translate-x-[-224px]' : '!w-[216px] translate-x-[-128px]' } diff --git a/web/app/components/apps/footer.tsx b/web/app/components/apps/footer.tsx index 9fed4c8757..16411ffc57 100644 --- a/web/app/components/apps/footer.tsx +++ b/web/app/components/apps/footer.tsx @@ -1,6 +1,6 @@ import React from 'react' import Link from 'next/link' -import { RiDiscordFill, RiGithubFill } from '@remixicon/react' +import { RiDiscordFill, RiDiscussLine, RiGithubFill } from '@remixicon/react' import { useTranslation } from 'react-i18next' type CustomLinkProps = { @@ -38,6 +38,9 @@ const Footer = () => { + + +
) diff --git a/web/app/components/apps/list.tsx b/web/app/components/apps/list.tsx index 49cba78c5b..b4388195f3 100644 --- a/web/app/components/apps/list.tsx +++ b/web/app/components/apps/list.tsx @@ -35,6 +35,7 @@ import dynamic from 'next/dynamic' import Empty from './empty' import Footer from './footer' import { useGlobalPublicStore } from '@/context/global-public-context' +import { AppModeEnum } from '@/types/app' const TagManagementModal = dynamic(() => import('@/app/components/base/tag-management'), { ssr: false, @@ -126,11 +127,11 @@ const List: FC = ({ const anchorRef = useRef(null) const options = [ { value: 'all', text: t('app.types.all'), icon: }, - { value: 'workflow', text: t('app.types.workflow'), icon: }, - { value: 'advanced-chat', text: t('app.types.advanced'), icon: }, - { value: 'chat', text: t('app.types.chatbot'), icon: }, - { value: 'agent-chat', text: t('app.types.agent'), icon: }, - { value: 'completion', text: t('app.types.completion'), icon: }, + { value: AppModeEnum.WORKFLOW, text: t('app.types.workflow'), icon: }, + { value: AppModeEnum.ADVANCED_CHAT, text: t('app.types.advanced'), icon: }, + { value: AppModeEnum.CHAT, text: t('app.types.chatbot'), icon: }, + { value: AppModeEnum.AGENT_CHAT, text: t('app.types.agent'), icon: }, + { value: AppModeEnum.COMPLETION, text: t('app.types.completion'), icon: }, ] useEffect(() => { @@ -155,15 +156,23 @@ const List: FC = ({ return } - if (anchorRef.current) { + if (anchorRef.current && containerRef.current) { + // Calculate dynamic rootMargin: clamps to 100-200px range, using 20% of container height as the base value for better responsiveness + const containerHeight = containerRef.current.clientHeight + const dynamicMargin = Math.max(100, Math.min(containerHeight * 0.2, 200)) // Clamps to 100-200px range, using 20% of container height as the base value + observer = new IntersectionObserver((entries) => { if (entries[0].isIntersecting && !isLoading && !error && hasMore) setSize((size: number) => size + 1) - }, { rootMargin: '100px' }) + }, { + root: containerRef.current, + rootMargin: `${dynamicMargin}px`, + threshold: 0.1, // Trigger when 10% of the anchor element is visible + }) observer.observe(anchorRef.current) } return () => observer?.disconnect() - }, [isLoading, setSize, anchorRef, mutate, data, error]) + }, [isLoading, setSize, data, error]) const { run: handleSearch } = useDebounceFn(() => { setSearchKeywords(keywords) diff --git a/web/app/components/base/action-button/index.stories.tsx b/web/app/components/base/action-button/index.stories.tsx index c174adbc73..07e0592374 100644 --- a/web/app/components/base/action-button/index.stories.tsx +++ b/web/app/components/base/action-button/index.stories.tsx @@ -3,7 +3,7 @@ import { RiAddLine, RiDeleteBinLine, RiEditLine, RiMore2Fill, RiSaveLine, RiShar import ActionButton, { ActionButtonState } from '.' const meta = { - title: 'Base/ActionButton', + title: 'Base/General/ActionButton', component: ActionButton, parameters: { layout: 'centered', diff --git a/web/app/components/base/agent-log-modal/index.stories.tsx b/web/app/components/base/agent-log-modal/index.stories.tsx new file mode 100644 index 0000000000..b512c8c581 --- /dev/null +++ b/web/app/components/base/agent-log-modal/index.stories.tsx @@ -0,0 +1,146 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect, useRef } from 'react' +import AgentLogModal from '.' +import { ToastProvider } from '@/app/components/base/toast' +import { useStore as useAppStore } from '@/app/components/app/store' +import type { IChatItem } from '@/app/components/base/chat/chat/type' +import type { AgentLogDetailResponse } from '@/models/log' + +const MOCK_RESPONSE: AgentLogDetailResponse = { + meta: { + status: 'finished', + executor: 'Agent Runner', + start_time: '2024-03-12T10:00:00Z', + elapsed_time: 12.45, + total_tokens: 2589, + agent_mode: 'ReACT', + iterations: 2, + error: undefined, + }, + iterations: [ + { + created_at: '2024-03-12T10:00:05Z', + files: [], + thought: JSON.stringify({ reasoning: 'Summarise conversation' }, null, 2), + tokens: 934, + tool_calls: [ + { + status: 'success', + tool_icon: null, + tool_input: { query: 'Latest revenue numbers' }, + tool_output: { answer: 'Revenue up 12% QoQ' }, + tool_name: 'search', + tool_label: { + 'en-US': 'Revenue Search', + }, + time_cost: 1.8, + }, + ], + tool_raw: { + inputs: JSON.stringify({ context: 'Summaries' }, null, 2), + outputs: JSON.stringify({ observation: 'Revenue up 12% QoQ' }, null, 2), + }, + }, + { + created_at: '2024-03-12T10:00:09Z', + files: [], + thought: JSON.stringify({ final: 'Revenue increased 12% quarter-over-quarter.' }, null, 2), + tokens: 642, + tool_calls: [], + tool_raw: { + inputs: JSON.stringify({ context: 'Compose summary' }, null, 2), + outputs: JSON.stringify({ observation: 'Final answer ready' }, null, 2), + }, + }, + ], + files: [], +} + +const MOCK_CHAT_ITEM: IChatItem = { + id: 'message-1', + content: JSON.stringify({ answer: 'Revenue grew 12% QoQ.' }, null, 2), + input: JSON.stringify({ question: 'Summarise revenue trends.' }, null, 2), + isAnswer: true, + conversationId: 'conv-123', +} + +const AgentLogModalDemo = ({ + width = 960, +}: { + width?: number +}) => { + const originalFetchRef = useRef(null) + const setAppDetail = useAppStore(state => state.setAppDetail) + + useEffect(() => { + setAppDetail({ + id: 'app-1', + name: 'Analytics Agent', + mode: 'agent-chat', + } as any) + + originalFetchRef.current = globalThis.fetch?.bind(globalThis) + + const handler = async (input: RequestInfo | URL, init?: RequestInit) => { + const request = input instanceof Request ? input : new Request(input, init) + const url = request.url + const parsed = new URL(url, window.location.origin) + + if (parsed.pathname.endsWith('/apps/app-1/agent/logs')) { + return new Response(JSON.stringify(MOCK_RESPONSE), { + status: 200, + headers: { 'Content-Type': 'application/json' }, + }) + } + + if (originalFetchRef.current) + return originalFetchRef.current(request) + + throw new Error(`Unhandled request: ${url}`) + } + + globalThis.fetch = handler as typeof globalThis.fetch + + return () => { + if (originalFetchRef.current) + globalThis.fetch = originalFetchRef.current + setAppDetail(undefined) + } + }, [setAppDetail]) + + return ( + +
+ { + console.log('Agent log modal closed') + }} + /> +
+
+ ) +} + +const meta = { + title: 'Base/Other/AgentLogModal', + component: AgentLogModalDemo, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Agent execution viewer showing iterations, tool calls, and metadata. Fetch responses are mocked for Storybook.', + }, + }, + }, + args: { + width: 960, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/answer-icon/index.stories.tsx b/web/app/components/base/answer-icon/index.stories.tsx new file mode 100644 index 0000000000..0928d9cda6 --- /dev/null +++ b/web/app/components/base/answer-icon/index.stories.tsx @@ -0,0 +1,107 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import type { ReactNode } from 'react' +import AnswerIcon from '.' + +const SAMPLE_IMAGE = 'data:image/svg+xml;utf8,AI' + +const meta = { + title: 'Base/General/AnswerIcon', + component: AnswerIcon, + parameters: { + docs: { + description: { + component: 'Circular avatar used for assistant answers. Supports emoji, solid background colour, or uploaded imagery.', + }, + }, + }, + tags: ['autodocs'], + args: { + icon: '🤖', + background: '#D5F5F6', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const StoryWrapper = (children: ReactNode) => ( +
+ {children} +
+) + +export const Default: Story = { + render: args => StoryWrapper( +
+ +
, + ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +
+ +
+ `.trim(), + }, + }, + }, +} + +export const CustomEmoji: Story = { + render: args => StoryWrapper( + <> +
+ +
+
+ +
+ , + ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +
+
+ +
+
+ +
+
+ `.trim(), + }, + }, + }, +} + +export const ImageIcon: Story = { + render: args => StoryWrapper( +
+ +
, + ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/app-icon-picker/index.stories.tsx b/web/app/components/base/app-icon-picker/index.stories.tsx new file mode 100644 index 0000000000..bd0ec0e200 --- /dev/null +++ b/web/app/components/base/app-icon-picker/index.stories.tsx @@ -0,0 +1,91 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import AppIconPicker, { type AppIconSelection } from '.' + +const meta = { + title: 'Base/Data Entry/AppIconPicker', + component: AppIconPicker, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Modal workflow for choosing an application avatar. Users can switch between emoji selections and image uploads (when enabled).', + }, + }, + nextjs: { + appDirectory: true, + navigation: { + pathname: '/apps/demo-app/icon-picker', + params: { appId: 'demo-app' }, + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const AppIconPickerDemo = () => { + const [open, setOpen] = useState(false) + const [selection, setSelection] = useState(null) + + return ( +
+ + +
+
Selection preview
+
+          {selection ? JSON.stringify(selection, null, 2) : 'No icon selected yet.'}
+        
+
+ + {open && ( + { + setSelection(result) + setOpen(false) + }} + onClose={() => setOpen(false)} + /> + )} +
+ ) +} + +export const Playground: Story = { + render: () => , + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +const [open, setOpen] = useState(false) +const [selection, setSelection] = useState(null) + +return ( + <> + + {open && ( + { + setSelection(result) + setOpen(false) + }} + onClose={() => setOpen(false)} + /> + )} + +) + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/app-icon/index.stories.tsx b/web/app/components/base/app-icon/index.stories.tsx new file mode 100644 index 0000000000..9fdffb54b0 --- /dev/null +++ b/web/app/components/base/app-icon/index.stories.tsx @@ -0,0 +1,108 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import type { ComponentProps } from 'react' +import AppIcon from '.' + +const meta = { + title: 'Base/General/AppIcon', + component: AppIcon, + parameters: { + docs: { + description: { + component: 'Reusable avatar for applications and workflows. Supports emoji or uploaded imagery, rounded mode, edit overlays, and multiple sizes.', + }, + }, + }, + tags: ['autodocs'], + args: { + icon: '🧭', + background: '#FFEAD5', + size: 'medium', + rounded: false, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = { + render: args => ( +
+ + +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + + `.trim(), + }, + }, + }, +} + +export const Sizes: Story = { + render: (args) => { + const sizes: Array['size']> = ['xs', 'tiny', 'small', 'medium', 'large', 'xl', 'xxl'] + return ( +
+ {sizes.map(size => ( +
+ + {size} +
+ ))} +
+ ) + }, + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +{(['xs','tiny','small','medium','large','xl','xxl'] as const).map(size => ( + +))} + `.trim(), + }, + }, + }, +} + +export const WithEditOverlay: Story = { + render: args => ( +
+ + +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/audio-btn/index.stories.tsx b/web/app/components/base/audio-btn/index.stories.tsx new file mode 100644 index 0000000000..1c989b80a6 --- /dev/null +++ b/web/app/components/base/audio-btn/index.stories.tsx @@ -0,0 +1,75 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect } from 'react' +import type { ComponentProps } from 'react' +import AudioBtn from '.' +import { ensureMockAudioManager } from '../../../../.storybook/utils/audio-player-manager.mock' + +ensureMockAudioManager() + +const StoryWrapper = (props: ComponentProps) => { + useEffect(() => { + ensureMockAudioManager() + }, []) + + return ( +
+ + Click to toggle playback +
+ ) +} + +const meta = { + title: 'Base/General/AudioBtn', + component: AudioBtn, + tags: ['autodocs'], + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Audio playback toggle that streams assistant responses. The story uses a mocked audio player so you can inspect loading and playback states without calling the real API.', + }, + }, + nextjs: { + appDirectory: true, + navigation: { + pathname: '/apps/demo-app/text-to-audio', + params: { appId: 'demo-app' }, + }, + }, + }, + argTypes: { + id: { + control: 'text', + description: 'Message identifier used to scope the audio stream.', + }, + value: { + control: 'text', + description: 'Text content that would be converted to speech.', + }, + voice: { + control: 'text', + description: 'Voice profile used for playback.', + }, + isAudition: { + control: 'boolean', + description: 'Switches to the audition style with minimal padding.', + }, + className: { + control: 'text', + description: 'Optional custom class for the wrapper.', + }, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = { + render: args => , + args: { + id: 'message-1', + value: 'This is an audio preview for the current assistant response.', + voice: 'alloy', + }, +} diff --git a/web/app/components/base/audio-gallery/index.stories.tsx b/web/app/components/base/audio-gallery/index.stories.tsx new file mode 100644 index 0000000000..539ab9e332 --- /dev/null +++ b/web/app/components/base/audio-gallery/index.stories.tsx @@ -0,0 +1,37 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import AudioGallery from '.' + +const AUDIO_SOURCES = [ + 'https://interactive-examples.mdn.mozilla.net/media/cc0-audio/t-rex-roar.mp3', +] + +const meta = { + title: 'Base/Data Display/AudioGallery', + component: AudioGallery, + parameters: { + docs: { + description: { + component: 'List of audio players that render waveform previews and playback controls for each source.', + }, + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + srcs: AUDIO_SOURCES, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} diff --git a/web/app/components/base/auto-height-textarea/index.stories.tsx b/web/app/components/base/auto-height-textarea/index.stories.tsx index dcbcb253c6..d0f36e4736 100644 --- a/web/app/components/base/auto-height-textarea/index.stories.tsx +++ b/web/app/components/base/auto-height-textarea/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import AutoHeightTextarea from '.' const meta = { - title: 'Base/AutoHeightTextarea', + title: 'Base/Data Entry/AutoHeightTextarea', component: AutoHeightTextarea, parameters: { layout: 'centered', diff --git a/web/app/components/base/avatar/index.stories.tsx b/web/app/components/base/avatar/index.stories.tsx new file mode 100644 index 0000000000..1b3dc3eb3b --- /dev/null +++ b/web/app/components/base/avatar/index.stories.tsx @@ -0,0 +1,73 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import Avatar from '.' + +const meta = { + title: 'Base/Data Display/Avatar', + component: Avatar, + parameters: { + docs: { + description: { + component: 'Initials or image-based avatar used across contacts and member lists. Falls back to the first letter when the image fails to load.', + }, + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + name: 'Alex Doe', + avatar: 'https://cloud.dify.ai/logo/logo.svg', + size: 40, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} + +export const WithFallback: Story = { + args: { + avatar: null, + name: 'Fallback', + }, + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, +} + +export const CustomSizes: Story = { + render: args => ( +
+ {[24, 32, 48, 64].map(size => ( +
+ + {size}px +
+ ))} +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +{[24, 32, 48, 64].map(size => ( + +))} + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/badge.tsx b/web/app/components/base/badge.tsx index 0362d8d6c4..9a0c93118a 100644 --- a/web/app/components/base/badge.tsx +++ b/web/app/components/base/badge.tsx @@ -20,7 +20,7 @@ const Badge = ({ return (
+ `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + text: 'beta', + uppercase: true, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} + +export const WithCornerMark: Story = { + args: { + text: 'new', + hasRedCornerMark: true, + }, + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, +} + +export const CustomContent: Story = { + render: args => ( + + + + Production + + + ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + + + Production + + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/block-input/index.stories.tsx b/web/app/components/base/block-input/index.stories.tsx index 0685f4150f..d05cc221b6 100644 --- a/web/app/components/base/block-input/index.stories.tsx +++ b/web/app/components/base/block-input/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import BlockInput from '.' const meta = { - title: 'Base/BlockInput', + title: 'Base/Data Entry/BlockInput', component: BlockInput, parameters: { layout: 'centered', diff --git a/web/app/components/base/button/add-button.stories.tsx b/web/app/components/base/button/add-button.stories.tsx new file mode 100644 index 0000000000..edd52b2b78 --- /dev/null +++ b/web/app/components/base/button/add-button.stories.tsx @@ -0,0 +1,52 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import AddButton from './add-button' + +const meta = { + title: 'Base/General/AddButton', + component: AddButton, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Compact icon-only button used for inline “add” actions in lists, cards, and modals.', + }, + }, + }, + tags: ['autodocs'], + argTypes: { + className: { + control: 'text', + description: 'Extra classes appended to the clickable container.', + }, + onClick: { + control: false, + description: 'Triggered when the add button is pressed.', + }, + }, + args: { + onClick: () => console.log('Add button clicked'), + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = { + args: { + className: 'bg-white/80 shadow-sm backdrop-blur-sm', + }, +} + +export const InToolbar: Story = { + render: args => ( +
+ Attachments +
+ +
+
+ ), + args: { + className: 'border border-dashed border-primary-200', + }, +} diff --git a/web/app/components/base/button/index.stories.tsx b/web/app/components/base/button/index.stories.tsx index e51b928e5e..02d20b4af4 100644 --- a/web/app/components/base/button/index.stories.tsx +++ b/web/app/components/base/button/index.stories.tsx @@ -4,7 +4,7 @@ import { RocketLaunchIcon } from '@heroicons/react/20/solid' import { Button } from '.' const meta = { - title: 'Base/Button', + title: 'Base/General/Button', component: Button, parameters: { layout: 'centered', diff --git a/web/app/components/base/button/sync-button.stories.tsx b/web/app/components/base/button/sync-button.stories.tsx new file mode 100644 index 0000000000..dcfbf6daf3 --- /dev/null +++ b/web/app/components/base/button/sync-button.stories.tsx @@ -0,0 +1,57 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import SyncButton from './sync-button' + +const meta = { + title: 'Base/General/SyncButton', + component: SyncButton, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Icon-only refresh button that surfaces a tooltip and is used for manual sync actions across the UI.', + }, + }, + }, + tags: ['autodocs'], + argTypes: { + className: { + control: 'text', + description: 'Additional classes appended to the clickable container.', + }, + popupContent: { + control: 'text', + description: 'Tooltip text shown on hover.', + }, + onClick: { + control: false, + description: 'Triggered when the sync button is pressed.', + }, + }, + args: { + popupContent: 'Sync now', + onClick: () => console.log('Sync button clicked'), + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = { + args: { + className: 'bg-white/80 shadow-sm backdrop-blur-sm', + }, +} + +export const InHeader: Story = { + render: args => ( +
+ Logs +
+ +
+
+ ), + args: { + popupContent: 'Refresh logs', + }, +} diff --git a/web/app/components/base/chat/chat-with-history/chat-wrapper.tsx b/web/app/components/base/chat/chat-with-history/chat-wrapper.tsx index e8ed5dfc95..26b873d4dd 100644 --- a/web/app/components/base/chat/chat-with-history/chat-wrapper.tsx +++ b/web/app/components/base/chat/chat-with-history/chat-wrapper.tsx @@ -3,7 +3,6 @@ import Chat from '../chat' import type { ChatConfig, ChatItem, - ChatItemInTree, OnSend, } from '../types' import { useChat } from '../chat/hooks' @@ -155,7 +154,7 @@ const ChatWrapper = () => { ) }, [chatList, handleNewConversationCompleted, handleSend, currentConversationId, currentConversationInputs, newConversationInputs, isInstalledApp, appId]) - const doRegenerate = useCallback((chatItem: ChatItemInTree, editedQuestion?: { message: string, files?: FileEntity[] }) => { + const doRegenerate = useCallback((chatItem: ChatItem, editedQuestion?: { message: string, files?: FileEntity[] }) => { const question = editedQuestion ? chatItem : chatList.find(item => item.id === chatItem.parentMessageId)! const parentAnswer = chatList.find(item => item.id === question.parentMessageId) doSend(editedQuestion ? editedQuestion.message : question.content, diff --git a/web/app/components/base/chat/chat-with-history/inputs-form/content.tsx b/web/app/components/base/chat/chat-with-history/inputs-form/content.tsx index 392bdf2b77..c7785ebd89 100644 --- a/web/app/components/base/chat/chat-with-history/inputs-form/content.tsx +++ b/web/app/components/base/chat/chat-with-history/inputs-form/content.tsx @@ -49,7 +49,7 @@ const InputsFormContent = ({ showTip }: Props) => {
{form.label}
{!form.required && ( -
{t('appDebug.variableTable.optional')}
+
{t('workflow.panel.optional')}
)}
)} diff --git a/web/app/components/base/chat/chat/answer/index.stories.tsx b/web/app/components/base/chat/chat/answer/index.stories.tsx index 02d0f015b5..95bc3bd5c0 100644 --- a/web/app/components/base/chat/chat/answer/index.stories.tsx +++ b/web/app/components/base/chat/chat/answer/index.stories.tsx @@ -6,7 +6,7 @@ import { markdownContentSVG } from './__mocks__/markdownContentSVG' import Answer from '.' const meta = { - title: 'Base/Chat Answer', + title: 'Base/Other/Chat Answer', component: Answer, parameters: { layout: 'fullscreen', diff --git a/web/app/components/base/chat/chat/question.stories.tsx b/web/app/components/base/chat/chat/question.stories.tsx index 6474add9df..f0ee860c89 100644 --- a/web/app/components/base/chat/chat/question.stories.tsx +++ b/web/app/components/base/chat/chat/question.stories.tsx @@ -5,7 +5,7 @@ import Question from './question' import { User } from '@/app/components/base/icons/src/public/avatar' const meta = { - title: 'Base/Chat Question', + title: 'Base/Other/Chat Question', component: Question, parameters: { layout: 'centered', diff --git a/web/app/components/base/chat/embedded-chatbot/chat-wrapper.tsx b/web/app/components/base/chat/embedded-chatbot/chat-wrapper.tsx index 6abf251a3e..58f438c409 100644 --- a/web/app/components/base/chat/embedded-chatbot/chat-wrapper.tsx +++ b/web/app/components/base/chat/embedded-chatbot/chat-wrapper.tsx @@ -3,7 +3,6 @@ import Chat from '../chat' import type { ChatConfig, ChatItem, - ChatItemInTree, OnSend, } from '../types' import { useChat } from '../chat/hooks' @@ -150,7 +149,7 @@ const ChatWrapper = () => { ) }, [currentConversationId, currentConversationInputs, newConversationInputs, chatList, handleSend, isInstalledApp, appId, handleNewConversationCompleted]) - const doRegenerate = useCallback((chatItem: ChatItemInTree, editedQuestion?: { message: string, files?: FileEntity[] }) => { + const doRegenerate = useCallback((chatItem: ChatItem, editedQuestion?: { message: string, files?: FileEntity[] }) => { const question = editedQuestion ? chatItem : chatList.find(item => item.id === chatItem.parentMessageId)! const parentAnswer = chatList.find(item => item.id === question.parentMessageId) doSend(editedQuestion ? editedQuestion.message : question.content, diff --git a/web/app/components/base/chat/embedded-chatbot/hooks.tsx b/web/app/components/base/chat/embedded-chatbot/hooks.tsx index 500860e740..ea1f4c54f5 100644 --- a/web/app/components/base/chat/embedded-chatbot/hooks.tsx +++ b/web/app/components/base/chat/embedded-chatbot/hooks.tsx @@ -19,18 +19,14 @@ import { buildChatItemTree, getProcessedInputsFromUrlParams, getProcessedSystemV import { getProcessedFilesFromResponse } from '../../file-uploader/utils' import { AppSourceType, - fetchAppInfo, - fetchAppMeta, - fetchAppParams, fetchChatList, fetchConversations, generationConversationName, updateFeedback, } from '@/service/share' -import { - fetchTryAppInfo, -} from '@/service/try-app' + import type { + AppData, // AppData, ConversationItem, } from '@/models/share' @@ -40,7 +36,7 @@ import { InputVarType } from '@/app/components/workflow/types' import { TransferMethod } from '@/types/app' import { addFileInfos, sortAgentSorts } from '@/app/components/tools/utils' import { noop } from 'lodash-es' -import { useGlobalPublicStore } from '@/context/global-public-context' +import { useWebAppStore } from '@/context/web-app-context' function getFormattedChatList(messages: any[]) { const newChatList: ChatItem[] = [] @@ -70,17 +66,21 @@ function getFormattedChatList(messages: any[]) { export const useEmbeddedChatbot = (appSourceType: AppSourceType, tryAppId?: string) => { const isInstalledApp = false // just can be webapp and try app - const systemFeatures = useGlobalPublicStore(s => s.systemFeatures) const isTryApp = appSourceType === AppSourceType.tryApp - const { data: appInfo, isLoading: appInfoLoading, error: appInfoError } = useSWR('appInfo', () => { - return isTryApp ? () => fetchTryAppInfo(tryAppId!) : fetchAppInfo - }) + const appInfo = useWebAppStore(s => s.appInfo) + const appMeta = useWebAppStore(s => s.appMeta) + const appParams = useWebAppStore(s => s.appParams) + const appId = useMemo(() => { return isTryApp ? tryAppId : (appInfo as any)?.app_id }, [appInfo]) + const embeddedConversationId = useWebAppStore(s => s.embeddedConversationId) + const embeddedUserId = useWebAppStore(s => s.embeddedUserId) + const [userId, setUserId] = useState() const [conversationId, setConversationId] = useState() + useEffect(() => { if (isTryApp) return getProcessedSystemVariablesFromUrlParams().then(({ user_id, conversation_id }) => { @@ -89,6 +89,14 @@ export const useEmbeddedChatbot = (appSourceType: AppSourceType, tryAppId?: stri }) }, []) + useEffect(() => { + setUserId(embeddedUserId || undefined) + }, [embeddedUserId]) + + useEffect(() => { + setConversationId(embeddedConversationId || undefined) + }, [embeddedConversationId]) + useEffect(() => { if (isTryApp) return const setLanguageFromParams = async () => { @@ -108,9 +116,9 @@ export const useEmbeddedChatbot = (appSourceType: AppSourceType, tryAppId?: stri // If locale is set as a system variable, use that await changeLanguage(localeFromSysVar) } - else if (appInfo?.site.default_language) { + else if ((appInfo as unknown as AppData)?.site?.default_language) { // Otherwise use the default from app config - await changeLanguage(appInfo.site.default_language) + await changeLanguage((appInfo as unknown as AppData).site?.default_language) } } @@ -152,9 +160,9 @@ export const useEmbeddedChatbot = (appSourceType: AppSourceType, tryAppId?: stri return currentConversationId }, [currentConversationId, newConversationId]) - - const { data: appParams } = useSWR(['appParams', appSourceType, appId], () => fetchAppParams(appSourceType, appId)) - const { data: appMeta } = useSWR(isTryApp ? null : ['appMeta', appSourceType, appId], () => fetchAppMeta(appSourceType, appId)) + // todo app params + // const { data: appParams } = useSWR(['appParams', appSourceType, appId], () => fetchAppParams(appSourceType, appId)) + // const { data: appMeta } = useSWR(isTryApp ? null : ['appMeta', appSourceType, appId], () => fetchAppMeta(appSourceType, appId)) const { data: appPinnedConversationData } = useSWR(isTryApp ? null : ['appConversationData', appSourceType, appId, true], () => fetchConversations(appSourceType, appId, undefined, true, 100)) const { data: appConversationData, isLoading: appConversationDataLoading, mutate: mutateAppConversationData } = useSWR(isTryApp ? null : ['appConversationData', appSourceType, appId, false], () => fetchConversations(appSourceType, appId, undefined, false, 100)) const { data: appChatListData, isLoading: appChatListDataLoading } = useSWR((chatShouldReloadKey && !isTryApp) ? ['appChatList', chatShouldReloadKey, appSourceType, appId] : null, () => fetchChatList(chatShouldReloadKey, appSourceType, appId)) @@ -413,9 +421,6 @@ export const useEmbeddedChatbot = (appSourceType: AppSourceType, tryAppId?: stri }, [appSourceType, appId, t, notify]) return { - appInfoError, - appInfoLoading: appInfoLoading || (systemFeatures.webapp_auth.enabled && isCheckingPermission), - userCanAccess: isTryApp || (systemFeatures.webapp_auth.enabled ? (userCanAccessResult as { result: boolean })?.result : true), appSourceType, isInstalledApp, allowResetChat, diff --git a/web/app/components/base/chat/embedded-chatbot/index.tsx b/web/app/components/base/chat/embedded-chatbot/index.tsx index 1553d1f153..a0192419c2 100644 --- a/web/app/components/base/chat/embedded-chatbot/index.tsx +++ b/web/app/components/base/chat/embedded-chatbot/index.tsx @@ -20,6 +20,7 @@ import DifyLogo from '@/app/components/base/logo/dify-logo' import cn from '@/utils/classnames' import useDocumentTitle from '@/hooks/use-document-title' import { useGlobalPublicStore } from '@/context/global-public-context' +import { AppSourceType } from '@/service/share' const Chatbot = () => { const { @@ -131,9 +132,10 @@ const EmbeddedChatbotWrapper = () => { setCurrentConversationInputs, allInputsHidden, initUserVariables, - } = useEmbeddedChatbot() + } = useEmbeddedChatbot(AppSourceType.webApp) return {
{form.label}
{!form.required && ( -
{t('appDebug.variableTable.optional')}
+
{t('workflow.panel.optional')}
)}
)} diff --git a/web/app/components/base/chat/types.ts b/web/app/components/base/chat/types.ts index f7f7aa4dce..5b0fe1f248 100644 --- a/web/app/components/base/chat/types.ts +++ b/web/app/components/base/chat/types.ts @@ -85,7 +85,7 @@ export type OnSend = { (message: string, files: FileEntity[] | undefined, isRegenerate: boolean, lastAnswer?: ChatItem | null): void } -export type OnRegenerate = (chatItem: ChatItem) => void +export type OnRegenerate = (chatItem: ChatItem, editedQuestion?: { message: string; files?: FileEntity[] }) => void export type Callback = { onSuccess: () => void diff --git a/web/app/components/base/checkbox-list/index.tsx b/web/app/components/base/checkbox-list/index.tsx new file mode 100644 index 0000000000..ca8333a200 --- /dev/null +++ b/web/app/components/base/checkbox-list/index.tsx @@ -0,0 +1,203 @@ +'use client' +import Badge from '@/app/components/base/badge' +import Checkbox from '@/app/components/base/checkbox' +import SearchInput from '@/app/components/base/search-input' +import SearchMenu from '@/assets/search-menu.svg' +import cn from '@/utils/classnames' +import Image from 'next/image' +import type { FC } from 'react' +import { useCallback, useMemo, useState } from 'react' +import { useTranslation } from 'react-i18next' +import Button from '../button' + +export type CheckboxListOption = { + label: string + value: string + disabled?: boolean +} + +export type CheckboxListProps = { + title?: string + label?: string + description?: string + options: CheckboxListOption[] + value?: string[] + onChange?: (value: string[]) => void + disabled?: boolean + containerClassName?: string + showSelectAll?: boolean + showCount?: boolean + showSearch?: boolean + maxHeight?: string | number +} + +const CheckboxList: FC = ({ + title = '', + label, + description, + options, + value = [], + onChange, + disabled = false, + containerClassName, + showSelectAll = true, + showCount = true, + showSearch = true, + maxHeight, +}) => { + const { t } = useTranslation() + const [searchQuery, setSearchQuery] = useState('') + + const filteredOptions = useMemo(() => { + if (!searchQuery?.trim()) + return options + + const query = searchQuery.toLowerCase() + return options.filter(option => + option.label.toLowerCase().includes(query) || option.value.toLowerCase().includes(query), + ) + }, [options, searchQuery]) + + const selectedCount = value.length + + const isAllSelected = useMemo(() => { + const selectableOptions = options.filter(option => !option.disabled) + return selectableOptions.length > 0 && selectableOptions.every(option => value.includes(option.value)) + }, [options, value]) + + const isIndeterminate = useMemo(() => { + const selectableOptions = options.filter(option => !option.disabled) + const selectedCount = selectableOptions.filter(option => value.includes(option.value)).length + return selectedCount > 0 && selectedCount < selectableOptions.length + }, [options, value]) + + const handleSelectAll = useCallback(() => { + if (disabled) + return + + if (isAllSelected) { + // Deselect all + onChange?.([]) + } + else { + // Select all non-disabled options + const allValues = options + .filter(option => !option.disabled) + .map(option => option.value) + onChange?.(allValues) + } + }, [isAllSelected, options, onChange, disabled]) + + const handleToggleOption = useCallback((optionValue: string) => { + if (disabled) + return + + const newValue = value.includes(optionValue) + ? value.filter(v => v !== optionValue) + : [...value, optionValue] + onChange?.(newValue) + }, [value, onChange, disabled]) + + return ( +
+ {label && ( +
+ {label} +
+ )} + {description && ( +
+ {description} +
+ )} + +
+ {(showSelectAll || title || showSearch) && ( +
+ {!searchQuery && showSelectAll && ( + + )} + {!searchQuery ?
+ {title && ( + + {title} + + )} + {showCount && selectedCount > 0 && ( + + {t('common.operation.selectCount', { count: selectedCount })} + + )} +
:
{ + filteredOptions.length > 0 + ? t('common.operation.searchCount', { count: filteredOptions.length, content: title }) + : t('common.operation.noSearchCount', { content: title })}
} + {showSearch && ( + + )} +
+ )} + +
+ {!filteredOptions.length ? ( +
+ {searchQuery ?
+ search menu + {t('common.operation.noSearchResults', { content: title })} + +
: t('common.noData')} +
+ ) : ( + filteredOptions.map((option) => { + const selected = value.includes(option.value) + + return ( +
{ + if (!option.disabled && !disabled) + handleToggleOption(option.value) + }} + > + { + if (!option.disabled && !disabled) + handleToggleOption(option.value) + }} + disabled={option.disabled || disabled} + /> +
+ {option.label} +
+
+ ) + }) + )} +
+
+
+ ) +} + +export default CheckboxList diff --git a/web/app/components/base/checkbox/index.stories.tsx b/web/app/components/base/checkbox/index.stories.tsx index 65fa8e1b97..3f8d4606eb 100644 --- a/web/app/components/base/checkbox/index.stories.tsx +++ b/web/app/components/base/checkbox/index.stories.tsx @@ -13,7 +13,7 @@ const createToggleItem = ( } const meta = { - title: 'Base/Checkbox', + title: 'Base/Data Entry/Checkbox', component: Checkbox, parameters: { layout: 'centered', diff --git a/web/app/components/base/checkbox/index.tsx b/web/app/components/base/checkbox/index.tsx index 2411d98966..9495292ea6 100644 --- a/web/app/components/base/checkbox/index.tsx +++ b/web/app/components/base/checkbox/index.tsx @@ -30,7 +30,7 @@ const Checkbox = ({
{}, + // eslint-disable-next-line no-empty-function + onClear: () => {}, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const ChipDemo = (props: React.ComponentProps) => { + const [selection, setSelection] = useState(props.value) + + return ( +
+ setSelection(item.value)} + onClear={() => setSelection('all')} + /> +
+ Current value: {selection} +
+
+ ) +} + +export const Playground: Story = { + render: args => , + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +const [selection, setSelection] = useState('all') + + setSelection(item.value)} + onClear={() => setSelection('all')} +/> + `.trim(), + }, + }, + }, +} + +export const WithoutLeftIcon: Story = { + args: { + showLeftIcon: false, + // eslint-disable-next-line no-empty-function + onSelect: () => {}, + // eslint-disable-next-line no-empty-function + onClear: () => {}, + }, + render: args => ( + + ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/confirm/index.stories.tsx b/web/app/components/base/confirm/index.stories.tsx index a524137b79..12cb46d9e4 100644 --- a/web/app/components/base/confirm/index.stories.tsx +++ b/web/app/components/base/confirm/index.stories.tsx @@ -4,7 +4,7 @@ import Confirm from '.' import Button from '../button' const meta = { - title: 'Base/Confirm', + title: 'Base/Feedback/Confirm', component: Confirm, parameters: { layout: 'centered', diff --git a/web/app/components/base/confirm/index.tsx b/web/app/components/base/confirm/index.tsx index 9771c0e43c..b2aa774cc2 100644 --- a/web/app/components/base/confirm/index.tsx +++ b/web/app/components/base/confirm/index.tsx @@ -2,6 +2,7 @@ import React, { useEffect, useRef, useState } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import Button from '../button' +import Tooltip from '../tooltip' export type IConfirm = { className?: string @@ -37,7 +38,9 @@ function Confirm({ }: IConfirm) { const { t } = useTranslation() const dialogRef = useRef(null) + const titleRef = useRef(null) const [isVisible, setIsVisible] = useState(isShow) + const [isTitleTruncated, setIsTitleTruncated] = useState(false) const confirmTxt = confirmText || `${t('common.operation.confirm')}` const cancelTxt = cancelText || `${t('common.operation.cancel')}` @@ -80,6 +83,13 @@ function Confirm({ } }, [isShow]) + useEffect(() => { + if (titleRef.current) { + const isOverflowing = titleRef.current.scrollWidth > titleRef.current.clientWidth + setIsTitleTruncated(isOverflowing) + } + }, [title, isVisible]) + if (!isVisible) return null @@ -92,8 +102,18 @@ function Confirm({
-
{title}
-
{content}
+ +
+ {title} +
+
+
{content}
{showCancel && } diff --git a/web/app/components/base/content-dialog/index.stories.tsx b/web/app/components/base/content-dialog/index.stories.tsx new file mode 100644 index 0000000000..aaebcad1b7 --- /dev/null +++ b/web/app/components/base/content-dialog/index.stories.tsx @@ -0,0 +1,119 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect, useState } from 'react' +import ContentDialog from '.' + +type Props = React.ComponentProps + +const meta = { + title: 'Base/Feedback/ContentDialog', + component: ContentDialog, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Sliding panel overlay used in the app detail view. Includes dimmed backdrop and animated entrance/exit transitions.', + }, + }, + }, + tags: ['autodocs'], + argTypes: { + className: { + control: 'text', + description: 'Additional classes applied to the sliding panel container.', + }, + show: { + control: 'boolean', + description: 'Controls visibility of the dialog.', + }, + onClose: { + control: false, + description: 'Invoked when the overlay/backdrop is clicked.', + }, + children: { + control: false, + table: { disable: true }, + }, + }, + args: { + show: false, + children: null, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const DemoWrapper = (props: Props) => { + const [open, setOpen] = useState(props.show) + + useEffect(() => { + setOpen(props.show) + }, [props.show]) + + return ( +
+
+ +
+ + { + props.onClose?.() + setOpen(false) + }} + > +
+

Plan summary

+

+ Use this area to present rich content for the selected run, configuration details, or + any supporting context. +

+
+ Scrollable placeholder content. Add domain-specific information, activity logs, or + editors in the real application. +
+
+ + +
+
+
+
+ ) +} + +export const Default: Story = { + args: { + children: null, + }, + render: args => , +} + +export const NarrowPanel: Story = { + render: args => , + args: { + className: 'max-w-[420px]', + children: null, + }, + parameters: { + docs: { + description: { + story: 'Applies a custom width class to show the dialog as a narrower information panel.', + }, + }, + }, +} diff --git a/web/app/components/base/copy-feedback/index.stories.tsx b/web/app/components/base/copy-feedback/index.stories.tsx new file mode 100644 index 0000000000..3bab620aec --- /dev/null +++ b/web/app/components/base/copy-feedback/index.stories.tsx @@ -0,0 +1,54 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import CopyFeedback, { CopyFeedbackNew } from '.' + +const meta = { + title: 'Base/Feedback/CopyFeedback', + component: CopyFeedback, + parameters: { + docs: { + description: { + component: 'Copy-to-clipboard button that shows instant feedback and a tooltip. Includes the original ActionButton wrapper and the newer ghost-button variant.', + }, + }, + }, + tags: ['autodocs'], + args: { + content: 'acc-3f92fa', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const CopyDemo = ({ content }: { content: string }) => { + const [value] = useState(content) + return ( +
+
+ Client ID: + {value} + +
+
+ Use the new ghost variant: + +
+
+ ) +} + +export const Playground: Story = { + render: args => , + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/copy-icon/index.stories.tsx b/web/app/components/base/copy-icon/index.stories.tsx new file mode 100644 index 0000000000..5962773792 --- /dev/null +++ b/web/app/components/base/copy-icon/index.stories.tsx @@ -0,0 +1,68 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import CopyIcon from '.' + +const meta = { + title: 'Base/General/CopyIcon', + component: CopyIcon, + parameters: { + docs: { + description: { + component: 'Interactive copy-to-clipboard glyph that swaps to a checkmark once the content has been copied. Tooltips rely on the app locale.', + }, + }, + }, + tags: ['autodocs'], + args: { + content: 'https://console.dify.ai/apps/12345', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = { + render: args => ( +
+ Hover or click to copy the app link: + +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +
+ Hover or click to copy the app link: + +
+ `.trim(), + }, + }, + }, +} + +export const InlineUsage: Story = { + render: args => ( +
+

+ Use the copy icon inline with labels or metadata. Clicking the icon copies the value to the clipboard and shows a success tooltip. +

+
+ Client ID + acc-3f92fa + +
+
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/corner-label/index.stories.tsx b/web/app/components/base/corner-label/index.stories.tsx new file mode 100644 index 0000000000..1592f94259 --- /dev/null +++ b/web/app/components/base/corner-label/index.stories.tsx @@ -0,0 +1,53 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import CornerLabel from '.' + +const meta = { + title: 'Base/Data Display/CornerLabel', + component: CornerLabel, + parameters: { + docs: { + description: { + component: 'Decorative label that anchors to card corners. Useful for marking “beta”, “deprecated”, or similar callouts.', + }, + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + label: 'beta', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} + +export const OnCard: Story = { + render: args => ( +
+ +
+ Showcase how the label sits on a card header. Pair with contextual text or status information. +
+
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +
+ + ...card content... +
+ `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/date-and-time-picker/calendar/index.tsx b/web/app/components/base/date-and-time-picker/calendar/index.tsx index 00612fcb37..03dcb0eda3 100644 --- a/web/app/components/base/date-and-time-picker/calendar/index.tsx +++ b/web/app/components/base/date-and-time-picker/calendar/index.tsx @@ -8,9 +8,10 @@ const Calendar: FC = ({ selectedDate, onDateClick, wrapperClassName, + getIsDateDisabled, }) => { return
- +
{ days.map(day => = ({ day={day} selectedDate={selectedDate} onClick={onDateClick} + isDisabled={getIsDateDisabled ? getIsDateDisabled(day.date) : false} />) }
diff --git a/web/app/components/base/date-and-time-picker/calendar/item.tsx b/web/app/components/base/date-and-time-picker/calendar/item.tsx index 1da8b9b3b5..7132d7bdfb 100644 --- a/web/app/components/base/date-and-time-picker/calendar/item.tsx +++ b/web/app/components/base/date-and-time-picker/calendar/item.tsx @@ -7,6 +7,7 @@ const Item: FC = ({ day, selectedDate, onClick, + isDisabled, }) => { const { date, isCurrentMonth } = day const isSelected = selectedDate?.isSame(date, 'date') @@ -14,11 +15,12 @@ const Item: FC = ({ return ( - {/* Confirm Button */} - -
+
+ {/* Now Button */} + + {/* Confirm Button */} +
) } diff --git a/web/app/components/base/date-and-time-picker/time-picker/index.spec.tsx b/web/app/components/base/date-and-time-picker/time-picker/index.spec.tsx index 40bc2928c8..24c7fff52f 100644 --- a/web/app/components/base/date-and-time-picker/time-picker/index.spec.tsx +++ b/web/app/components/base/date-and-time-picker/time-picker/index.spec.tsx @@ -3,6 +3,7 @@ import { fireEvent, render, screen } from '@testing-library/react' import TimePicker from './index' import dayjs from '../utils/dayjs' import { isDayjsObject } from '../utils/dayjs' +import type { TimePickerProps } from '../types' jest.mock('react-i18next', () => ({ useTranslation: () => ({ @@ -28,11 +29,21 @@ jest.mock('@/app/components/base/portal-to-follow-elem', () => ({ jest.mock('./options', () => () =>
) jest.mock('./header', () => () =>
) +jest.mock('@/app/components/base/timezone-label', () => { + return function MockTimezoneLabel({ timezone, inline, className }: { timezone: string, inline?: boolean, className?: string }) { + return ( + + UTC+8 + + ) + } +}) describe('TimePicker', () => { - const baseProps = { + const baseProps: Pick = { onChange: jest.fn(), onClear: jest.fn(), + value: undefined, } beforeEach(() => { @@ -92,4 +103,86 @@ describe('TimePicker', () => { expect(isDayjsObject(emitted)).toBe(true) expect(emitted?.utcOffset()).toBe(dayjs().tz('America/New_York').utcOffset()) }) + + describe('Timezone Label Integration', () => { + test('should not display timezone label by default', () => { + render( + , + ) + + expect(screen.queryByTestId('timezone-label')).not.toBeInTheDocument() + }) + + test('should not display timezone label when showTimezone is false', () => { + render( + , + ) + + expect(screen.queryByTestId('timezone-label')).not.toBeInTheDocument() + }) + + test('should display timezone label when showTimezone is true', () => { + render( + , + ) + + const timezoneLabel = screen.getByTestId('timezone-label') + expect(timezoneLabel).toBeInTheDocument() + expect(timezoneLabel).toHaveAttribute('data-timezone', 'Asia/Shanghai') + }) + + test('should pass inline prop to timezone label', () => { + render( + , + ) + + const timezoneLabel = screen.getByTestId('timezone-label') + expect(timezoneLabel).toHaveAttribute('data-inline', 'true') + }) + + test('should not display timezone label when showTimezone is true but timezone is not provided', () => { + render( + , + ) + + expect(screen.queryByTestId('timezone-label')).not.toBeInTheDocument() + }) + + test('should apply shrink-0 and text-xs classes to timezone label', () => { + render( + , + ) + + const timezoneLabel = screen.getByTestId('timezone-label') + expect(timezoneLabel).toHaveClass('shrink-0', 'text-xs') + }) + }) }) diff --git a/web/app/components/base/date-and-time-picker/time-picker/index.tsx b/web/app/components/base/date-and-time-picker/time-picker/index.tsx index f23fcf8f4e..9577a107e5 100644 --- a/web/app/components/base/date-and-time-picker/time-picker/index.tsx +++ b/web/app/components/base/date-and-time-picker/time-picker/index.tsx @@ -19,6 +19,7 @@ import Header from './header' import { useTranslation } from 'react-i18next' import { RiCloseCircleFill, RiTimeLine } from '@remixicon/react' import cn from '@/utils/classnames' +import TimezoneLabel from '@/app/components/base/timezone-label' const to24Hour = (hour12: string, period: Period) => { const normalized = Number.parseInt(hour12, 10) % 12 @@ -35,6 +36,10 @@ const TimePicker = ({ title, minuteFilter, popupClassName, + notClearable = false, + triggerFullWidth = false, + showTimezone = false, + placement = 'bottom-start', }: TimePickerProps) => { const { t } = useTranslation() const [isOpen, setIsOpen] = useState(false) @@ -189,7 +194,7 @@ const TimePicker = ({ const inputElem = ( - + {renderTrigger ? (renderTrigger({ inputElem, onClick: handleClickTrigger, isOpen, })) : (
{inputElem} + {showTimezone && timezone && ( + + )} React.ReactNode minuteFilter?: (minutes: string[]) => string[] popupZIndexClassname?: string + noConfirm?: boolean + getIsDateDisabled?: (date: Dayjs) => boolean } export type DatePickerHeaderProps = { @@ -63,6 +66,10 @@ export type TimePickerProps = { title?: string minuteFilter?: (minutes: string[]) => string[] popupClassName?: string + notClearable?: boolean + triggerFullWidth?: boolean + showTimezone?: boolean + placement?: Placement } export type TimePickerFooterProps = { @@ -80,12 +87,14 @@ export type CalendarProps = { selectedDate: Dayjs | undefined onDateClick: (date: Dayjs) => void wrapperClassName?: string + getIsDateDisabled?: (date: Dayjs) => boolean } export type CalendarItemProps = { day: Day selectedDate: Dayjs | undefined onClick: (date: Dayjs) => void + isDisabled: boolean } export type TimeOptionsProps = { diff --git a/web/app/components/base/date-and-time-picker/utils/dayjs.spec.ts b/web/app/components/base/date-and-time-picker/utils/dayjs.spec.ts index 549ab01029..5c891126b5 100644 --- a/web/app/components/base/date-and-time-picker/utils/dayjs.spec.ts +++ b/web/app/components/base/date-and-time-picker/utils/dayjs.spec.ts @@ -1,5 +1,6 @@ import dayjs from './dayjs' import { + convertTimezoneToOffsetStr, getDateWithTimezone, isDayjsObject, toDayjs, @@ -65,3 +66,50 @@ describe('dayjs utilities', () => { expect(result?.minute()).toBe(0) }) }) + +describe('convertTimezoneToOffsetStr', () => { + test('should return default UTC+0 for undefined timezone', () => { + expect(convertTimezoneToOffsetStr(undefined)).toBe('UTC+0') + }) + + test('should return default UTC+0 for invalid timezone', () => { + expect(convertTimezoneToOffsetStr('Invalid/Timezone')).toBe('UTC+0') + }) + + test('should handle whole hour positive offsets without leading zeros', () => { + expect(convertTimezoneToOffsetStr('Asia/Shanghai')).toBe('UTC+8') + expect(convertTimezoneToOffsetStr('Pacific/Auckland')).toBe('UTC+12') + expect(convertTimezoneToOffsetStr('Pacific/Apia')).toBe('UTC+13') + }) + + test('should handle whole hour negative offsets without leading zeros', () => { + expect(convertTimezoneToOffsetStr('Pacific/Niue')).toBe('UTC-11') + expect(convertTimezoneToOffsetStr('Pacific/Honolulu')).toBe('UTC-10') + expect(convertTimezoneToOffsetStr('America/New_York')).toBe('UTC-5') + }) + + test('should handle zero offset', () => { + expect(convertTimezoneToOffsetStr('Europe/London')).toBe('UTC+0') + expect(convertTimezoneToOffsetStr('UTC')).toBe('UTC+0') + }) + + test('should handle half-hour offsets (30 minutes)', () => { + // India Standard Time: UTC+5:30 + expect(convertTimezoneToOffsetStr('Asia/Kolkata')).toBe('UTC+5:30') + // Australian Central Time: UTC+9:30 + expect(convertTimezoneToOffsetStr('Australia/Adelaide')).toBe('UTC+9:30') + expect(convertTimezoneToOffsetStr('Australia/Darwin')).toBe('UTC+9:30') + }) + + test('should handle 45-minute offsets', () => { + // Chatham Time: UTC+12:45 + expect(convertTimezoneToOffsetStr('Pacific/Chatham')).toBe('UTC+12:45') + }) + + test('should preserve leading zeros in minute part for non-zero minutes', () => { + // Ensure +05:30 is displayed as "UTC+5:30", not "UTC+5:3" + const result = convertTimezoneToOffsetStr('Asia/Kolkata') + expect(result).toMatch(/UTC[+-]\d+:30/) + expect(result).not.toMatch(/UTC[+-]\d+:3[^0]/) + }) +}) diff --git a/web/app/components/base/date-and-time-picker/utils/dayjs.ts b/web/app/components/base/date-and-time-picker/utils/dayjs.ts index 808b50247a..b05e725985 100644 --- a/web/app/components/base/date-and-time-picker/utils/dayjs.ts +++ b/web/app/components/base/date-and-time-picker/utils/dayjs.ts @@ -107,7 +107,18 @@ export const convertTimezoneToOffsetStr = (timezone?: string) => { const tzItem = tz.find(item => item.value === timezone) if (!tzItem) return DEFAULT_OFFSET_STR - return `UTC${tzItem.name.charAt(0)}${tzItem.name.charAt(2)}` + // Extract offset from name format like "-11:00 Niue Time" or "+05:30 India Time" + // Name format is always "{offset}:{minutes} {timezone name}" + const offsetMatch = tzItem.name.match(/^([+-]?\d{1,2}):(\d{2})/) + if (!offsetMatch) + return DEFAULT_OFFSET_STR + // Parse hours and minutes separately + const hours = Number.parseInt(offsetMatch[1], 10) + const minutes = Number.parseInt(offsetMatch[2], 10) + const sign = hours >= 0 ? '+' : '' + // If minutes are non-zero, include them in the output (e.g., "UTC+5:30") + // Otherwise, only show hours (e.g., "UTC+8") + return minutes !== 0 ? `UTC${sign}${hours}:${offsetMatch[2]}` : `UTC${sign}${hours}` } export const isDayjsObject = (value: unknown): value is Dayjs => dayjs.isDayjs(value) @@ -150,7 +161,7 @@ export const toDayjs = (value: string | Dayjs | undefined, options: ToDayjsOptio if (format) { const parsedWithFormat = tzName - ? dayjs.tz(trimmed, format, tzName, true) + ? dayjs(trimmed, format, true).tz(tzName, true) : dayjs(trimmed, format, true) if (parsedWithFormat.isValid()) return parsedWithFormat @@ -191,7 +202,7 @@ export const toDayjs = (value: string | Dayjs | undefined, options: ToDayjsOptio const candidateFormats = formats ?? COMMON_PARSE_FORMATS for (const fmt of candidateFormats) { const parsed = tzName - ? dayjs.tz(trimmed, fmt, tzName, true) + ? dayjs(trimmed, fmt, true).tz(tzName, true) : dayjs(trimmed, fmt, true) if (parsed.isValid()) return parsed diff --git a/web/app/components/base/dialog/index.stories.tsx b/web/app/components/base/dialog/index.stories.tsx new file mode 100644 index 0000000000..f573b856d3 --- /dev/null +++ b/web/app/components/base/dialog/index.stories.tsx @@ -0,0 +1,152 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect, useState } from 'react' +import Dialog from '.' + +const meta = { + title: 'Base/Feedback/Dialog', + component: Dialog, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Modal dialog built on Headless UI. Provides animated overlay, title slot, and optional footer region.', + }, + }, + }, + tags: ['autodocs'], + argTypes: { + className: { + control: 'text', + description: 'Additional classes applied to the panel.', + }, + titleClassName: { + control: 'text', + description: 'Extra classes for the title element.', + }, + bodyClassName: { + control: 'text', + description: 'Extra classes for the content area.', + }, + footerClassName: { + control: 'text', + description: 'Extra classes for the footer container.', + }, + title: { + control: 'text', + description: 'Dialog title.', + }, + show: { + control: 'boolean', + description: 'Controls visibility of the dialog.', + }, + onClose: { + control: false, + description: 'Called when the dialog backdrop or close handler fires.', + }, + }, + args: { + title: 'Manage API Keys', + show: false, + children: null, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const DialogDemo = (props: React.ComponentProps) => { + const [open, setOpen] = useState(props.show) + useEffect(() => { + setOpen(props.show) + }, [props.show]) + + return ( +
+ + + { + props.onClose?.() + setOpen(false) + }} + > +
+

+ Centralize API key management for collaborators. You can revoke, rotate, or generate new keys directly from this dialog. +

+
+ This placeholder area represents a form or table that would live inside the dialog body. +
+
+
+
+ ) +} + +export const Default: Story = { + render: args => , + args: { + footer: ( + <> + + + + ), + }, +} + +export const WithoutFooter: Story = { + render: args => , + args: { + footer: undefined, + title: 'Read-only summary', + }, + parameters: { + docs: { + description: { + story: 'Demonstrates the dialog when no footer actions are provided.', + }, + }, + }, +} + +export const CustomStyling: Story = { + render: args => , + args: { + className: 'max-w-[560px] bg-white/95 backdrop-blur', + bodyClassName: 'bg-gray-50 rounded-xl p-5', + footerClassName: 'justify-between px-4 pb-4 pt-4', + titleClassName: 'text-lg text-primary-600', + footer: ( + <> + Last synced 2 minutes ago +
+ + +
+ + ), + }, + parameters: { + docs: { + description: { + story: 'Applies custom classes to the panel, body, title, and footer to match different surfaces.', + }, + }, + }, +} diff --git a/web/app/components/base/divider/index.stories.tsx b/web/app/components/base/divider/index.stories.tsx new file mode 100644 index 0000000000..c634173202 --- /dev/null +++ b/web/app/components/base/divider/index.stories.tsx @@ -0,0 +1,46 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import Divider from '.' + +const meta = { + title: 'Base/Layout/Divider', + component: Divider, + parameters: { + docs: { + description: { + component: 'Lightweight separator supporting horizontal and vertical orientations with gradient or solid backgrounds.', + }, + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Horizontal: Story = {} + +export const Vertical: Story = { + render: args => ( +
+ Filters + + Tags +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/divider/index.tsx b/web/app/components/base/divider/index.tsx index 6fe16b95a2..387f24a5e9 100644 --- a/web/app/components/base/divider/index.tsx +++ b/web/app/components/base/divider/index.tsx @@ -29,7 +29,7 @@ export type DividerProps = { const Divider: FC = ({ type, bgStyle, className = '', style }) => { return ( -
+
) } diff --git a/web/app/components/base/drawer-plus/index.stories.tsx b/web/app/components/base/drawer-plus/index.stories.tsx new file mode 100644 index 0000000000..ddb39f2d63 --- /dev/null +++ b/web/app/components/base/drawer-plus/index.stories.tsx @@ -0,0 +1,124 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { fn } from 'storybook/test' +import { useState } from 'react' +import DrawerPlus from '.' + +const meta = { + title: 'Base/Feedback/DrawerPlus', + component: DrawerPlus, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Enhanced drawer built atop the base drawer component. Provides header/foot slots, mask control, and mobile breakpoints.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +type DrawerPlusProps = React.ComponentProps + +const storyBodyElement: React.JSX.Element = ( +
+

+ DrawerPlus allows rich content with sticky header/footer and responsive masking on mobile. Great for editing flows or showing execution logs. +

+
+ Body content scrolls if it exceeds the allotted height. +
+
+) + +const DrawerPlusDemo = (props: Partial) => { + const [open, setOpen] = useState(false) + + const { + body, + title, + foot, + isShow: _isShow, + onHide: _onHide, + ...rest + } = props + + const resolvedBody: React.JSX.Element = body ?? storyBodyElement + + return ( +
+ + + } + isShow={open} + onHide={() => setOpen(false)} + title={title ?? 'Workflow execution details'} + body={resolvedBody} + foot={foot} + /> +
+ ) +} + +export const Playground: Story = { + render: args => , + args: { + isShow: false, + onHide: fn(), + title: 'Edit configuration', + body: storyBodyElement, + }, +} + +export const WithFooter: Story = { + render: (args) => { + const FooterDemo = () => { + const [open, setOpen] = useState(false) + return ( +
+ + + setOpen(false)} + title={args.title ?? 'Workflow execution details'} + body={args.body ?? ( +
+

Populate the body with scrollable content. Footer stays pinned.

+
+ )} + foot={ +
+ + +
+ } + /> +
+ ) + } + return + }, + args: { + isShow: false, + onHide: fn(), + title: 'Edit configuration!', + body: storyBodyElement, + }, +} diff --git a/web/app/components/base/drawer/index.stories.tsx b/web/app/components/base/drawer/index.stories.tsx new file mode 100644 index 0000000000..e7711bc1a2 --- /dev/null +++ b/web/app/components/base/drawer/index.stories.tsx @@ -0,0 +1,114 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { fn } from 'storybook/test' +import { useState } from 'react' +import Drawer from '.' + +const meta = { + title: 'Base/Feedback/Drawer', + component: Drawer, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Sliding panel built on Headless UI dialog primitives. Supports optional mask, custom footer, and close behaviour.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const DrawerDemo = (props: React.ComponentProps) => { + const [open, setOpen] = useState(false) + + return ( +
+ + + setOpen(false)} + title={props.title ?? 'Edit configuration'} + description={props.description ?? 'Adjust settings in the side panel and save.'} + footer={props.footer ?? undefined} + > +
+

+ This example renders arbitrary content inside the drawer body. Use it for contextual forms, settings, or informational panels. +

+
+ Content area +
+
+
+
+ ) +} + +export const Playground: Story = { + render: args => , + args: { + children: null, + isOpen: false, + onClose: fn(), + }, + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +const [open, setOpen] = useState(false) + + setOpen(false)} + title="Edit configuration" + description="Adjust settings in the side panel and save." +> + ... + + `.trim(), + }, + }, + }, +} + +export const CustomFooter: Story = { + render: args => ( + + + +
+ } + /> + ), + args: { + children: null, + isOpen: false, + onClose: fn(), + }, + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +}> + ... + + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/drawer/index.tsx b/web/app/components/base/drawer/index.tsx index c35acbeac7..101ac22b6c 100644 --- a/web/app/components/base/drawer/index.tsx +++ b/web/app/components/base/drawer/index.tsx @@ -10,6 +10,7 @@ export type IDrawerProps = { description?: string dialogClassName?: string dialogBackdropClassName?: string + containerClassName?: string panelClassName?: string children: React.ReactNode footer?: React.ReactNode @@ -22,6 +23,7 @@ export type IDrawerProps = { onCancel?: () => void onOk?: () => void unmount?: boolean + noOverlay?: boolean } export default function Drawer({ @@ -29,6 +31,7 @@ export default function Drawer({ description = '', dialogClassName = '', dialogBackdropClassName = '', + containerClassName = '', panelClassName = '', children, footer, @@ -41,6 +44,7 @@ export default function Drawer({ onCancel, onOk, unmount = false, + noOverlay = false, }: IDrawerProps) { const { t } = useTranslation() return ( @@ -53,15 +57,15 @@ export default function Drawer({ }} className={cn('fixed inset-0 z-[30] overflow-y-auto', dialogClassName)} > -
+
{/* mask */} - { if (!clickOutsideNotOpen) onClose() }} - /> + />}
<>
diff --git a/web/app/components/base/dropdown/index.stories.tsx b/web/app/components/base/dropdown/index.stories.tsx new file mode 100644 index 0000000000..da70730744 --- /dev/null +++ b/web/app/components/base/dropdown/index.stories.tsx @@ -0,0 +1,85 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { fn } from 'storybook/test' +import { useState } from 'react' +import Dropdown, { type Item } from '.' + +const PRIMARY_ITEMS: Item[] = [ + { value: 'rename', text: 'Rename' }, + { value: 'duplicate', text: 'Duplicate' }, +] + +const SECONDARY_ITEMS: Item[] = [ + { value: 'archive', text: Archive }, + { value: 'delete', text: Delete }, +] + +const meta = { + title: 'Base/Navigation/Dropdown', + component: Dropdown, + parameters: { + docs: { + description: { + component: 'Small contextual menu with optional destructive section. Uses portal positioning utilities for precise placement.', + }, + }, + }, + tags: ['autodocs'], + args: { + items: PRIMARY_ITEMS, + secondItems: SECONDARY_ITEMS, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const DropdownDemo = (props: React.ComponentProps) => { + const [lastAction, setLastAction] = useState('None') + + return ( +
+ { + setLastAction(String(item.value)) + props.onSelect?.(item) + }} + /> +
+ Last action: {lastAction} +
+
+ ) +} + +export const Playground: Story = { + render: args => , + args: { + items: PRIMARY_ITEMS, + secondItems: SECONDARY_ITEMS, + onSelect: fn(), + }, +} + +export const CustomTrigger: Story = { + render: args => ( + ( + + )} + /> + ), + args: { + items: PRIMARY_ITEMS, + onSelect: fn(), + }, +} diff --git a/web/app/components/base/effect/index.stories.tsx b/web/app/components/base/effect/index.stories.tsx new file mode 100644 index 0000000000..a7f316fe7e --- /dev/null +++ b/web/app/components/base/effect/index.stories.tsx @@ -0,0 +1,39 @@ +/* eslint-disable tailwindcss/classnames-order */ +import type { Meta, StoryObj } from '@storybook/nextjs' +import Effect from '.' + +const meta = { + title: 'Base/Other/Effect', + component: Effect, + parameters: { + docs: { + description: { + component: 'Blurred circular glow used as a decorative background accent. Combine with relatively positioned containers.', + }, + source: { + language: 'tsx', + code: ` +
+ +
+ `.trim(), + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = { + render: () => ( +
+ + +
+ Accent glow +
+
+ ), +} diff --git a/web/app/components/base/emoji-picker/Inner.stories.tsx b/web/app/components/base/emoji-picker/Inner.stories.tsx new file mode 100644 index 0000000000..5341d63ee3 --- /dev/null +++ b/web/app/components/base/emoji-picker/Inner.stories.tsx @@ -0,0 +1,57 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import EmojiPickerInner from './Inner' + +const meta = { + title: 'Base/Data Entry/EmojiPickerInner', + component: EmojiPickerInner, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Core emoji grid with search and style swatches. Use this when embedding the selector inline without a modal frame.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const InnerDemo = () => { + const [selection, setSelection] = useState<{ emoji: string; background: string } | null>(null) + + return ( +
+ setSelection({ emoji, background })} + className="flex-1 overflow-hidden rounded-xl border border-divider-subtle bg-white" + /> +
+
Latest selection
+
+          {selection ? JSON.stringify(selection, null, 2) : 'Tap an emoji to set background options.'}
+        
+
+
+ ) +} + +export const Playground: Story = { + render: () => , + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +const [selection, setSelection] = useState<{ emoji: string; background: string } | null>(null) + +return ( + setSelection({ emoji, background })} /> +) + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/emoji-picker/index.stories.tsx b/web/app/components/base/emoji-picker/index.stories.tsx new file mode 100644 index 0000000000..7c9b07f138 --- /dev/null +++ b/web/app/components/base/emoji-picker/index.stories.tsx @@ -0,0 +1,91 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import EmojiPicker from '.' + +const meta = { + title: 'Base/Data Entry/EmojiPicker', + component: EmojiPicker, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Modal-based emoji selector that powers the icon picker. Supports search, background swatches, and confirmation callbacks.', + }, + }, + nextjs: { + appDirectory: true, + navigation: { + pathname: '/apps/demo-app/emoji-picker', + params: { appId: 'demo-app' }, + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const EmojiPickerDemo = () => { + const [open, setOpen] = useState(false) + const [selection, setSelection] = useState<{ emoji: string; background: string } | null>(null) + + return ( +
+ + +
+
Selection preview
+
+          {selection ? JSON.stringify(selection, null, 2) : 'No emoji selected yet.'}
+        
+
+ + {open && ( + { + setSelection({ emoji, background }) + setOpen(false) + }} + onClose={() => setOpen(false)} + /> + )} +
+ ) +} + +export const Playground: Story = { + render: () => , + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +const [open, setOpen] = useState(false) +const [selection, setSelection] = useState<{ emoji: string; background: string } | null>(null) + +return ( + <> + + {open && ( + { + setSelection({ emoji, background }) + setOpen(false) + }} + onClose={() => setOpen(false)} + /> + )} + +) + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/encrypted-bottom/index.tsx b/web/app/components/base/encrypted-bottom/index.tsx new file mode 100644 index 0000000000..8416217517 --- /dev/null +++ b/web/app/components/base/encrypted-bottom/index.tsx @@ -0,0 +1,30 @@ +import cn from '@/utils/classnames' +import { RiLock2Fill } from '@remixicon/react' +import Link from 'next/link' +import { useTranslation } from 'react-i18next' + +type Props = { + className?: string + frontTextKey?: string + backTextKey?: string +} + +export const EncryptedBottom = (props: Props) => { + const { t } = useTranslation() + const { frontTextKey, backTextKey, className } = props + + return ( +
+ + {t(frontTextKey || 'common.provider.encrypted.front')} + + PKCS1_OAEP + + {t(backTextKey || 'common.provider.encrypted.back')} +
+ ) +} diff --git a/web/app/components/base/error-boundary/index.tsx b/web/app/components/base/error-boundary/index.tsx new file mode 100644 index 0000000000..e3df2c2ca8 --- /dev/null +++ b/web/app/components/base/error-boundary/index.tsx @@ -0,0 +1,273 @@ +'use client' +import type { ErrorInfo, ReactNode } from 'react' +import React, { useCallback, useEffect, useRef, useState } from 'react' +import { RiAlertLine, RiBugLine } from '@remixicon/react' +import Button from '@/app/components/base/button' +import cn from '@/utils/classnames' + +type ErrorBoundaryState = { + hasError: boolean + error: Error | null + errorInfo: ErrorInfo | null + errorCount: number +} + +type ErrorBoundaryProps = { + children: ReactNode + fallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode) + onError?: (error: Error, errorInfo: ErrorInfo) => void + onReset?: () => void + showDetails?: boolean + className?: string + resetKeys?: Array + resetOnPropsChange?: boolean + isolate?: boolean + enableRecovery?: boolean + customTitle?: string + customMessage?: string +} + +// Internal class component for error catching +class ErrorBoundaryInner extends React.Component< + ErrorBoundaryProps & { + resetErrorBoundary: () => void + onResetKeysChange: (prevResetKeys?: Array) => void + }, + ErrorBoundaryState +> { + constructor(props: any) { + super(props) + this.state = { + hasError: false, + error: null, + errorInfo: null, + errorCount: 0, + } + } + + static getDerivedStateFromError(error: Error): Partial { + return { + hasError: true, + error, + } + } + + componentDidCatch(error: Error, errorInfo: ErrorInfo) { + if (process.env.NODE_ENV === 'development') { + console.error('ErrorBoundary caught an error:', error) + console.error('Error Info:', errorInfo) + } + + this.setState(prevState => ({ + errorInfo, + errorCount: prevState.errorCount + 1, + })) + + if (this.props.onError) + this.props.onError(error, errorInfo) + } + + componentDidUpdate(prevProps: any) { + const { resetKeys, resetOnPropsChange } = this.props + const { hasError } = this.state + + if (hasError && prevProps.resetKeys !== resetKeys) { + if (resetKeys?.some((key, idx) => key !== prevProps.resetKeys?.[idx])) + this.props.resetErrorBoundary() + } + + if (hasError && resetOnPropsChange && prevProps.children !== this.props.children) + this.props.resetErrorBoundary() + + if (prevProps.resetKeys !== resetKeys) + this.props.onResetKeysChange(prevProps.resetKeys) + } + + render() { + const { hasError, error, errorInfo, errorCount } = this.state + const { + fallback, + children, + showDetails = false, + className, + isolate = true, + enableRecovery = true, + customTitle, + customMessage, + resetErrorBoundary, + } = this.props + + if (hasError && error) { + if (fallback) { + if (typeof fallback === 'function') + return fallback(error, resetErrorBoundary) + + return fallback + } + + return ( +
+
+ +

+ {customTitle || 'Something went wrong'} +

+
+ +

+ {customMessage || 'An unexpected error occurred while rendering this component.'} +

+ + {showDetails && errorInfo && ( +
+ + + + Error Details (Development Only) + + +
+
+ Error: +
+                    {error.toString()}
+                  
+
+ {errorInfo && ( +
+ Component Stack: +
+                      {errorInfo.componentStack}
+                    
+
+ )} + {errorCount > 1 && ( +
+ This error has occurred {errorCount} times +
+ )} +
+
+ )} + + {enableRecovery && ( +
+ + +
+ )} +
+ ) + } + + return children + } +} + +// Main functional component wrapper +const ErrorBoundary: React.FC = (props) => { + const [errorBoundaryKey, setErrorBoundaryKey] = useState(0) + const resetKeysRef = useRef(props.resetKeys) + const prevResetKeysRef = useRef | undefined>(undefined) + + const resetErrorBoundary = useCallback(() => { + setErrorBoundaryKey(prev => prev + 1) + props.onReset?.() + }, [props]) + + const onResetKeysChange = useCallback((prevResetKeys?: Array) => { + prevResetKeysRef.current = prevResetKeys + }, []) + + useEffect(() => { + if (prevResetKeysRef.current !== props.resetKeys) + resetKeysRef.current = props.resetKeys + }, [props.resetKeys]) + + return ( + + ) +} + +// Hook for imperative error handling +export function useErrorHandler() { + const [error, setError] = useState(null) + + useEffect(() => { + if (error) + throw error + }, [error]) + + return setError +} + +// Hook for catching async errors +export function useAsyncError() { + const [, setError] = useState() + + return useCallback( + (error: Error) => { + setError(() => { + throw error + }) + }, + [setError], + ) +} + +// HOC for wrapping components with error boundary +export function withErrorBoundary

( + Component: React.ComponentType

, + errorBoundaryProps?: Omit, +): React.ComponentType

{ + const WrappedComponent = (props: P) => ( + + + + ) + + WrappedComponent.displayName = `withErrorBoundary(${Component.displayName || Component.name || 'Component'})` + + return WrappedComponent +} + +// Simple error fallback component +export const ErrorFallback: React.FC<{ + error: Error + resetErrorBoundary: () => void +}> = ({ error, resetErrorBoundary }) => { + return ( +

+

Oops! Something went wrong

+

{error.message}

+ +
+ ) +} + +export default ErrorBoundary diff --git a/web/app/components/base/features/index.stories.tsx b/web/app/components/base/features/index.stories.tsx new file mode 100644 index 0000000000..f1eaf048b8 --- /dev/null +++ b/web/app/components/base/features/index.stories.tsx @@ -0,0 +1,73 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import { FeaturesProvider } from '.' +import NewFeaturePanel from './new-feature-panel' +import type { Features } from './types' + +const DEFAULT_FEATURES: Features = { + moreLikeThis: { enabled: false }, + opening: { enabled: false }, + suggested: { enabled: false }, + text2speech: { enabled: false }, + speech2text: { enabled: false }, + citation: { enabled: false }, + moderation: { enabled: false }, + file: { enabled: false }, + annotationReply: { enabled: false }, +} + +const meta = { + title: 'Base/Other/FeaturesProvider', + component: FeaturesProvider, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Zustand-backed provider used for feature toggles. Paired with `NewFeaturePanel` for workflow settings.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const FeaturesDemo = () => { + const [show, setShow] = useState(true) + const [features, setFeatures] = useState(DEFAULT_FEATURES) + + return ( + +
+
+
Feature toggles preview
+
+ +
+
+
+ + setFeatures(prev => ({ ...prev, ...next }))} + onClose={() => setShow(false)} + /> +
+ ) +} + +export const Playground: Story = { + render: () => , + args: { + children: null, + }, +} diff --git a/web/app/components/base/features/new-feature-panel/conversation-opener/modal.tsx b/web/app/components/base/features/new-feature-panel/conversation-opener/modal.tsx index f0af893f0d..8ab007e66b 100644 --- a/web/app/components/base/features/new-feature-panel/conversation-opener/modal.tsx +++ b/web/app/components/base/features/new-feature-panel/conversation-opener/modal.tsx @@ -1,4 +1,4 @@ -import React, { useCallback, useEffect, useState } from 'react' +import React, { useCallback, useEffect, useMemo, useState } from 'react' import { useTranslation } from 'react-i18next' import { useBoolean } from 'ahooks' import { produce } from 'immer' @@ -45,7 +45,13 @@ const OpeningSettingModal = ({ const [isShowConfirmAddVar, { setTrue: showConfirmAddVar, setFalse: hideConfirmAddVar }] = useBoolean(false) const [notIncludeKeys, setNotIncludeKeys] = useState([]) + const isSaveDisabled = useMemo(() => !tempValue.trim(), [tempValue]) + const handleSave = useCallback((ignoreVariablesCheck?: boolean) => { + // Prevent saving if opening statement is empty + if (isSaveDisabled) + return + if (!ignoreVariablesCheck) { const keys = getInputKeys(tempValue) const promptKeys = promptVariables.map(item => item.key) @@ -75,7 +81,7 @@ const OpeningSettingModal = ({ } }) onSave(newOpening) - }, [data, onSave, promptVariables, workflowVariables, showConfirmAddVar, tempSuggestedQuestions, tempValue]) + }, [data, onSave, promptVariables, workflowVariables, showConfirmAddVar, tempSuggestedQuestions, tempValue, isSaveDisabled]) const cancelAutoAddVar = useCallback(() => { hideConfirmAddVar() @@ -217,6 +223,7 @@ const OpeningSettingModal = ({ diff --git a/web/app/components/base/features/new-feature-panel/file-upload/setting-modal.tsx b/web/app/components/base/features/new-feature-panel/file-upload/setting-modal.tsx index 92f93b8819..6ebbc05ae5 100644 --- a/web/app/components/base/features/new-feature-panel/file-upload/setting-modal.tsx +++ b/web/app/components/base/features/new-feature-panel/file-upload/setting-modal.tsx @@ -37,7 +37,7 @@ const FileUploadSettings = ({ {children} -
+
onOpen(false)} diff --git a/web/app/components/base/features/new-feature-panel/moderation/moderation-setting-modal.tsx b/web/app/components/base/features/new-feature-panel/moderation/moderation-setting-modal.tsx index 095137203b..ff45a7ea4c 100644 --- a/web/app/components/base/features/new-feature-panel/moderation/moderation-setting-modal.tsx +++ b/web/app/components/base/features/new-feature-panel/moderation/moderation-setting-modal.tsx @@ -26,6 +26,7 @@ import { CustomConfigurationStatusEnum } from '@/app/components/header/account-s import cn from '@/utils/classnames' import { noop } from 'lodash-es' import { useDocLink } from '@/context/i18n' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const systemTypes = ['openai_moderation', 'keywords', 'api'] @@ -55,7 +56,7 @@ const ModerationSettingModal: FC = ({ const { setShowAccountSettingModal } = useModalContext() const handleOpenSettingsModal = () => { setShowAccountSettingModal({ - payload: 'provider', + payload: ACCOUNT_SETTING_TAB.PROVIDER, onCancelCallback: () => { mutate() }, diff --git a/web/app/components/base/file-icon/index.stories.tsx b/web/app/components/base/file-icon/index.stories.tsx new file mode 100644 index 0000000000..dbd3e13fea --- /dev/null +++ b/web/app/components/base/file-icon/index.stories.tsx @@ -0,0 +1,79 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import FileIcon from '.' + +const meta = { + title: 'Base/General/FileIcon', + component: FileIcon, + parameters: { + docs: { + description: { + component: 'Maps a file extension to the appropriate SVG icon used across upload and attachment surfaces.', + }, + }, + }, + tags: ['autodocs'], + argTypes: { + type: { + control: 'text', + description: 'File extension or identifier used to resolve the icon.', + }, + className: { + control: 'text', + description: 'Custom classes passed to the SVG wrapper.', + }, + }, + args: { + type: 'pdf', + className: 'h-10 w-10', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = { + render: args => ( +
+ + Extension: {args.type} +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, +} + +export const Gallery: Story = { + render: () => { + const examples = ['pdf', 'docx', 'xlsx', 'csv', 'json', 'md', 'txt', 'html', 'notion', 'unknown'] + return ( +
+ {examples.map(type => ( +
+ + {type} +
+ ))} +
+ ) + }, + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +{['pdf','docx','xlsx','csv','json','md','txt','html','notion','unknown'].map(type => ( + +))} + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/file-uploader/file-image-render.stories.tsx b/web/app/components/base/file-uploader/file-image-render.stories.tsx new file mode 100644 index 0000000000..132c0b61a3 --- /dev/null +++ b/web/app/components/base/file-uploader/file-image-render.stories.tsx @@ -0,0 +1,32 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import FileImageRender from './file-image-render' + +const SAMPLE_IMAGE = 'data:image/svg+xml;utf8,Preview' + +const meta = { + title: 'Base/General/FileImageRender', + component: FileImageRender, + parameters: { + docs: { + description: { + component: 'Renders image previews inside a bordered frame. Often used in upload galleries and logs.', + }, + source: { + language: 'tsx', + code: ` + + `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + imageUrl: SAMPLE_IMAGE, + className: 'h-32 w-52', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/file-uploader/file-list.stories.tsx b/web/app/components/base/file-uploader/file-list.stories.tsx new file mode 100644 index 0000000000..89c0568735 --- /dev/null +++ b/web/app/components/base/file-uploader/file-list.stories.tsx @@ -0,0 +1,96 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import { FileList } from './file-uploader-in-chat-input/file-list' +import type { FileEntity } from './types' +import { SupportUploadFileTypes } from '@/app/components/workflow/types' +import { TransferMethod } from '@/types/app' + +const SAMPLE_IMAGE = 'data:image/svg+xml;utf8,IMG' + +const filesSample: FileEntity[] = [ + { + id: '1', + name: 'Project Brief.pdf', + size: 256000, + type: 'application/pdf', + progress: 100, + transferMethod: TransferMethod.local_file, + supportFileType: SupportUploadFileTypes.document, + url: '', + }, + { + id: '2', + name: 'Design.png', + size: 128000, + type: 'image/png', + progress: 100, + transferMethod: TransferMethod.local_file, + supportFileType: SupportUploadFileTypes.image, + base64Url: SAMPLE_IMAGE, + }, + { + id: '3', + name: 'Voiceover.mp3', + size: 512000, + type: 'audio/mpeg', + progress: 45, + transferMethod: TransferMethod.remote_url, + supportFileType: SupportUploadFileTypes.audio, + url: '', + }, +] + +const meta = { + title: 'Base/Data Display/FileList', + component: FileList, + parameters: { + docs: { + description: { + component: 'Renders a responsive gallery of uploaded files, handling icons, previews, and progress states.', + }, + }, + }, + tags: ['autodocs'], + args: { + files: filesSample, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const FileListPlayground = (args: React.ComponentProps) => { + const [items, setItems] = useState(args.files || []) + + return ( +
+ setItems(list => list.filter(file => file.id !== fileId))} + /> +
+ ) +} + +export const Playground: Story = { + render: args => , + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +const [files, setFiles] = useState(initialFiles) + + setFiles(list => list.filter(file => file.id !== id))} /> + `.trim(), + }, + }, + }, +} + +export const UploadStates: Story = { + args: { + files: filesSample.map(file => ({ ...file, progress: file.id === '3' ? 45 : 100 })), + }, +} diff --git a/web/app/components/base/file-uploader/file-type-icon.stories.tsx b/web/app/components/base/file-uploader/file-type-icon.stories.tsx new file mode 100644 index 0000000000..c317afab68 --- /dev/null +++ b/web/app/components/base/file-uploader/file-type-icon.stories.tsx @@ -0,0 +1,38 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import FileTypeIcon from './file-type-icon' +import { FileAppearanceTypeEnum } from './types' + +const meta = { + title: 'Base/General/FileTypeIcon', + component: FileTypeIcon, + parameters: { + docs: { + description: { + component: 'Displays the appropriate icon and accent colour for a file appearance type. Useful in lists and attachments.', + }, + }, + }, + tags: ['autodocs'], + args: { + type: FileAppearanceTypeEnum.document, + size: 'md', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const Gallery: Story = { + render: () => ( +
+ {Object.values(FileAppearanceTypeEnum).map(type => ( +
+ + {type} +
+ ))} +
+ ), +} diff --git a/web/app/components/base/file-uploader/file-uploader-in-attachment/index.stories.tsx b/web/app/components/base/file-uploader/file-uploader-in-attachment/index.stories.tsx new file mode 100644 index 0000000000..dabb8b6615 --- /dev/null +++ b/web/app/components/base/file-uploader/file-uploader-in-attachment/index.stories.tsx @@ -0,0 +1,110 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { fn } from 'storybook/test' +import { useState } from 'react' +import FileUploaderInAttachmentWrapper from './index' +import type { FileEntity } from '../types' +import type { FileUpload } from '@/app/components/base/features/types' +import { PreviewMode } from '@/app/components/base/features/types' +import { TransferMethod } from '@/types/app' +import { ToastProvider } from '@/app/components/base/toast' +import { SupportUploadFileTypes } from '@/app/components/workflow/types' + +const SAMPLE_IMAGE = 'data:image/svg+xml;utf8,IMG' + +const mockFiles: FileEntity[] = [ + { + id: 'file-1', + name: 'Requirements.pdf', + size: 256000, + type: 'application/pdf', + progress: 100, + transferMethod: TransferMethod.local_file, + supportFileType: SupportUploadFileTypes.document, + url: '', + }, + { + id: 'file-2', + name: 'Interface.png', + size: 128000, + type: 'image/png', + progress: 100, + transferMethod: TransferMethod.local_file, + supportFileType: SupportUploadFileTypes.image, + base64Url: SAMPLE_IMAGE, + }, + { + id: 'file-3', + name: 'Voiceover.mp3', + size: 512000, + type: 'audio/mpeg', + progress: 35, + transferMethod: TransferMethod.remote_url, + supportFileType: SupportUploadFileTypes.audio, + url: '', + }, +] + +const fileConfig: FileUpload = { + enabled: true, + allowed_file_upload_methods: [TransferMethod.local_file, TransferMethod.remote_url], + allowed_file_types: ['document', 'image', 'audio'], + number_limits: 5, + preview_config: { mode: PreviewMode.NewPage, file_type_list: ['pdf', 'png'] }, +} + +const meta = { + title: 'Base/Data Entry/FileUploaderInAttachment', + component: FileUploaderInAttachmentWrapper, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Attachment-style uploader that supports local files and remote links. Demonstrates upload progress, re-upload, and preview actions.', + }, + }, + nextjs: { + appDirectory: true, + navigation: { + pathname: '/apps/demo-app/uploads', + params: { appId: 'demo-app' }, + }, + }, + }, + tags: ['autodocs'], + args: { + fileConfig, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const AttachmentDemo = (props: React.ComponentProps) => { + const [files, setFiles] = useState(mockFiles) + + return ( + +
+ +
+
+ ) +} + +export const Playground: Story = { + render: args => , + args: { + onChange: fn(), + }, +} + +export const Disabled: Story = { + render: args => , + args: { + onChange: fn(), + }, +} diff --git a/web/app/components/base/file-uploader/file-uploader-in-chat-input/index.stories.tsx b/web/app/components/base/file-uploader/file-uploader-in-chat-input/index.stories.tsx new file mode 100644 index 0000000000..f4165f64cb --- /dev/null +++ b/web/app/components/base/file-uploader/file-uploader-in-chat-input/index.stories.tsx @@ -0,0 +1,95 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import FileUploaderInChatInput from '.' +import { FileContextProvider } from '../store' +import type { FileEntity } from '../types' +import type { FileUpload } from '@/app/components/base/features/types' +import { SupportUploadFileTypes } from '@/app/components/workflow/types' +import { TransferMethod } from '@/types/app' +import { FileList } from '../file-uploader-in-chat-input/file-list' +import { ToastProvider } from '@/app/components/base/toast' + +const mockFiles: FileEntity[] = [ + { + id: '1', + name: 'Dataset.csv', + size: 64000, + type: 'text/csv', + progress: 100, + transferMethod: TransferMethod.local_file, + supportFileType: SupportUploadFileTypes.document, + }, +] + +const chatUploadConfig: FileUpload = { + enabled: true, + allowed_file_upload_methods: [TransferMethod.local_file, TransferMethod.remote_url], + allowed_file_types: ['image', 'document'], + number_limits: 3, +} + +type ChatInputDemoProps = React.ComponentProps & { + initialFiles?: FileEntity[] +} + +const ChatInputDemo = ({ initialFiles = mockFiles, ...props }: ChatInputDemoProps) => { + const [files, setFiles] = useState(initialFiles) + + return ( + + +
+
Simulated chat input
+
+ +
Type a message...
+
+
+ +
+
+
+
+ ) +} + +const meta = { + title: 'Base/Data Entry/FileUploaderInChatInput', + component: ChatInputDemo, + parameters: { + docs: { + description: { + component: 'Attachment trigger suited for chat inputs. Demonstrates integration with the shared file store and preview list.', + }, + }, + nextjs: { + appDirectory: true, + navigation: { + pathname: '/chats/demo', + params: { appId: 'demo-app' }, + }, + }, + }, + tags: ['autodocs'], + args: { + fileConfig: chatUploadConfig, + initialFiles: mockFiles, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = { + render: args => , +} + +export const RemoteOnly: Story = { + args: { + fileConfig: { + ...chatUploadConfig, + allowed_file_upload_methods: [TransferMethod.remote_url], + }, + initialFiles: [], + }, +} diff --git a/web/app/components/base/file-uploader/hooks.ts b/web/app/components/base/file-uploader/hooks.ts index 3f4d4a6b06..521ecdbafd 100644 --- a/web/app/components/base/file-uploader/hooks.ts +++ b/web/app/components/base/file-uploader/hooks.ts @@ -11,6 +11,7 @@ import type { FileEntity } from './types' import { useFileStore } from './store' import { fileUpload, + getFileUploadErrorMessage, getSupportFileType, isAllowedFileExtension, } from './utils' @@ -172,8 +173,9 @@ export const useFile = (fileConfig: FileUpload) => { onSuccessCallback: (res) => { handleUpdateFile({ ...uploadingFile, uploadedId: res.id, progress: 100 }) }, - onErrorCallback: () => { - notify({ type: 'error', message: t('common.fileUploader.uploadFromComputerUploadError') }) + onErrorCallback: (error?: any) => { + const errorMessage = getFileUploadErrorMessage(error, t('common.fileUploader.uploadFromComputerUploadError'), t) + notify({ type: 'error', message: errorMessage }) handleUpdateFile({ ...uploadingFile, progress: -1 }) }, }, !!params.token) @@ -279,8 +281,9 @@ export const useFile = (fileConfig: FileUpload) => { onSuccessCallback: (res) => { handleUpdateFile({ ...uploadingFile, uploadedId: res.id, progress: 100 }) }, - onErrorCallback: () => { - notify({ type: 'error', message: t('common.fileUploader.uploadFromComputerUploadError') }) + onErrorCallback: (error?: any) => { + const errorMessage = getFileUploadErrorMessage(error, t('common.fileUploader.uploadFromComputerUploadError'), t) + notify({ type: 'error', message: errorMessage }) handleUpdateFile({ ...uploadingFile, progress: -1 }) }, }, !!params.token) @@ -302,9 +305,23 @@ export const useFile = (fileConfig: FileUpload) => { const text = e.clipboardData?.getData('text/plain') if (file && !text) { e.preventDefault() + + const allowedFileTypes = fileConfig.allowed_file_types || [] + const fileType = getSupportFileType(file.name, file.type, allowedFileTypes?.includes(SupportUploadFileTypes.custom)) + const isFileTypeAllowed = allowedFileTypes.includes(fileType) + + // Check if file type is in allowed list + if (!isFileTypeAllowed || !fileConfig.enabled) { + notify({ + type: 'error', + message: t('common.fileUploader.fileExtensionNotSupport'), + }) + return + } + handleLocalFileUpload(file) } - }, [handleLocalFileUpload]) + }, [handleLocalFileUpload, fileConfig, notify, t]) const [isDragActive, setIsDragActive] = useState(false) const handleDragFileEnter = useCallback((e: React.DragEvent) => { diff --git a/web/app/components/base/file-uploader/utils.ts b/web/app/components/base/file-uploader/utils.ts index 9c217646ca..e0a1a0250f 100644 --- a/web/app/components/base/file-uploader/utils.ts +++ b/web/app/components/base/file-uploader/utils.ts @@ -7,11 +7,30 @@ import { SupportUploadFileTypes } from '@/app/components/workflow/types' import type { FileResponse } from '@/types/workflow' import { TransferMethod } from '@/types/app' +/** + * Get appropriate error message for file upload errors + * @param error - The error object from upload failure + * @param defaultMessage - Default error message to use if no specific error is matched + * @param t - Translation function + * @returns Localized error message + */ +export const getFileUploadErrorMessage = (error: any, defaultMessage: string, t: (key: string) => string): string => { + const errorCode = error?.response?.code + + if (errorCode === 'forbidden') + return error?.response?.message + + if (errorCode === 'file_extension_blocked') + return t('common.fileUploader.fileExtensionBlocked') + + return defaultMessage +} + type FileUploadParams = { file: File onProgressCallback: (progress: number) => void onSuccessCallback: (res: { id: string }) => void - onErrorCallback: () => void + onErrorCallback: (error?: any) => void } type FileUpload = (v: FileUploadParams, isPublic?: boolean, url?: string) => void export const fileUpload: FileUpload = ({ @@ -37,8 +56,8 @@ export const fileUpload: FileUpload = ({ .then((res: { id: string }) => { onSuccessCallback(res) }) - .catch(() => { - onErrorCallback() + .catch((error) => { + onErrorCallback(error) }) } diff --git a/web/app/components/base/float-right-container/index.stories.tsx b/web/app/components/base/float-right-container/index.stories.tsx new file mode 100644 index 0000000000..18173f086d --- /dev/null +++ b/web/app/components/base/float-right-container/index.stories.tsx @@ -0,0 +1,74 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { fn } from 'storybook/test' +import { useState } from 'react' +import FloatRightContainer from '.' + +const meta = { + title: 'Base/Feedback/FloatRightContainer', + component: FloatRightContainer, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Wrapper that renders content in a drawer on mobile and inline on desktop. Useful for responsive settings panels.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const ContainerDemo = () => { + const [open, setOpen] = useState(false) + const [isMobile, setIsMobile] = useState(false) + + return ( +
+
+ + +
+ + setOpen(false)} + title="Responsive panel" + description="Switch the toggle to see drawer vs inline behaviour." + mask + > +
+

Panel Content

+

+ On desktop, this block renders inline when `isOpen` is true. On mobile it appears inside the drawer wrapper. +

+
+
+
+ ) +} + +export const Playground: Story = { + render: () => , + args: { + isMobile: false, + isOpen: false, + onClose: fn(), + children: null, + }, +} diff --git a/web/app/components/base/form/components/base/base-field.tsx b/web/app/components/base/form/components/base/base-field.tsx index bf415e08a8..db57059b82 100644 --- a/web/app/components/base/form/components/base/base-field.tsx +++ b/web/app/components/base/form/components/base/base-field.tsx @@ -1,20 +1,71 @@ +import CheckboxList from '@/app/components/base/checkbox-list' +import type { FieldState, FormSchema, TypeWithI18N } from '@/app/components/base/form/types' +import { FormItemValidateStatusEnum, FormTypeEnum } from '@/app/components/base/form/types' +import Input from '@/app/components/base/input' +import Radio from '@/app/components/base/radio' +import RadioE from '@/app/components/base/radio/ui' +import PureSelect from '@/app/components/base/select/pure' +import Tooltip from '@/app/components/base/tooltip' +import { useRenderI18nObject } from '@/hooks/use-i18n' +import { useTriggerPluginDynamicOptions } from '@/service/use-triggers' +import cn from '@/utils/classnames' +import { RiExternalLinkLine } from '@remixicon/react' +import type { AnyFieldApi } from '@tanstack/react-form' +import { useStore } from '@tanstack/react-form' import { isValidElement, memo, useCallback, useMemo, } from 'react' -import { RiExternalLinkLine } from '@remixicon/react' -import type { AnyFieldApi } from '@tanstack/react-form' -import { useStore } from '@tanstack/react-form' -import cn from '@/utils/classnames' -import Input from '@/app/components/base/input' -import PureSelect from '@/app/components/base/select/pure' -import type { FormSchema } from '@/app/components/base/form/types' -import { FormTypeEnum } from '@/app/components/base/form/types' -import { useRenderI18nObject } from '@/hooks/use-i18n' -import Radio from '@/app/components/base/radio' -import RadioE from '@/app/components/base/radio/ui' +import { useTranslation } from 'react-i18next' + +const getExtraProps = (type: FormTypeEnum) => { + switch (type) { + case FormTypeEnum.secretInput: + return { type: 'password', autoComplete: 'new-password' } + case FormTypeEnum.textNumber: + return { type: 'number' } + default: + return { type: 'text' } + } +} + +const getTranslatedContent = ({ content, render }: { + content: React.ReactNode | string | null | undefined | TypeWithI18N | Record + render: (content: TypeWithI18N | Record) => string +}): string => { + if (isValidElement(content) || typeof content === 'string') + return content as string + + if (typeof content === 'object' && content !== null) + return render(content as TypeWithI18N) + + return '' +} + +const VALIDATE_STATUS_STYLE_MAP: Record = { + [FormItemValidateStatusEnum.Error]: { + componentClassName: 'border-components-input-border-destructive focus:border-components-input-border-destructive', + textClassName: 'text-text-destructive', + infoFieldName: 'errors', + }, + [FormItemValidateStatusEnum.Warning]: { + componentClassName: 'border-components-input-border-warning focus:border-components-input-border-warning', + textClassName: 'text-text-warning', + infoFieldName: 'warnings', + }, + [FormItemValidateStatusEnum.Success]: { + componentClassName: '', + textClassName: '', + infoFieldName: '', + }, + [FormItemValidateStatusEnum.Validating]: { + componentClassName: '', + textClassName: '', + infoFieldName: '', + }, +} export type BaseFieldProps = { fieldClassName?: string @@ -25,7 +76,9 @@ export type BaseFieldProps = { field: AnyFieldApi disabled?: boolean onChange?: (field: string, value: any) => void + fieldState?: FieldState } + const BaseField = ({ fieldClassName, labelClassName, @@ -35,204 +88,259 @@ const BaseField = ({ field, disabled: propsDisabled, onChange, + fieldState, }: BaseFieldProps) => { const renderI18nObject = useRenderI18nObject() + const { t } = useTranslation() const { + name, label, required, placeholder, options, labelClassName: formLabelClassName, disabled: formSchemaDisabled, + type: formItemType, + dynamicSelectParams, + multiple = false, + tooltip, + showCopy, + description, + url, + help, } = formSchema const disabled = propsDisabled || formSchemaDisabled - const memorizedLabel = useMemo(() => { - if (isValidElement(label)) - return label + const [translatedLabel, translatedPlaceholder, translatedTooltip, translatedDescription, translatedHelp] = useMemo(() => { + const results = [ + label, + placeholder, + tooltip, + description, + help, + ].map(v => getTranslatedContent({ content: v, render: renderI18nObject })) + if (!results[1]) results[1] = t('common.placeholder.input') + return results + }, [label, placeholder, tooltip, description, help, renderI18nObject]) - if (typeof label === 'string') - return label + const watchedVariables = useMemo(() => { + const variables = new Set() - if (typeof label === 'object' && label !== null) - return renderI18nObject(label as Record) - }, [label, renderI18nObject]) - const memorizedPlaceholder = useMemo(() => { - if (typeof placeholder === 'string') - return placeholder + for (const option of options || []) { + for (const condition of option.show_on || []) + variables.add(condition.variable) + } - if (typeof placeholder === 'object' && placeholder !== null) - return renderI18nObject(placeholder as Record) - }, [placeholder, renderI18nObject]) - const optionValues = useStore(field.form.store, (s) => { + return Array.from(variables) + }, [options]) + + const watchedValues = useStore(field.form.store, (s) => { const result: Record = {} - options?.forEach((option) => { - if (option.show_on?.length) { - option.show_on.forEach((condition) => { - result[condition.variable] = s.values[condition.variable] - }) - } - }) + for (const variable of watchedVariables) + result[variable] = s.values[variable] + return result }) + const memorizedOptions = useMemo(() => { return options?.filter((option) => { - if (!option.show_on || option.show_on.length === 0) + if (!option.show_on?.length) return true return option.show_on.every((condition) => { - const conditionValue = optionValues[condition.variable] - return conditionValue === condition.value + return watchedValues[condition.variable] === condition.value }) }).map((option) => { return { - label: typeof option.label === 'string' ? option.label : renderI18nObject(option.label), + label: getTranslatedContent({ content: option.label, render: renderI18nObject }), value: option.value, } }) || [] - }, [options, renderI18nObject, optionValues]) + }, [options, renderI18nObject, watchedValues]) + const value = useStore(field.form.store, s => s.values[field.name]) + const { data: dynamicOptionsData, isLoading: isDynamicOptionsLoading, error: dynamicOptionsError } = useTriggerPluginDynamicOptions( + dynamicSelectParams || { + plugin_id: '', + provider: '', + action: '', + parameter: '', + credential_id: '', + }, + formItemType === FormTypeEnum.dynamicSelect, + ) + + const dynamicOptions = useMemo(() => { + if (!dynamicOptionsData?.options) + return [] + return dynamicOptionsData.options.map(option => ({ + label: getTranslatedContent({ content: option.label, render: renderI18nObject }), + value: option.value, + })) + }, [dynamicOptionsData, renderI18nObject]) + const handleChange = useCallback((value: any) => { field.handleChange(value) onChange?.(field.name, value) }, [field, onChange]) return ( -
-
- {memorizedLabel} - { - required && !isValidElement(label) && ( - * - ) - } -
-
- { - formSchema.type === FormTypeEnum.textInput && ( - { - handleChange(e.target.value) - }} - onBlur={field.handleBlur} - disabled={disabled} - placeholder={memorizedPlaceholder} + <> +
+
+ {translatedLabel} + { + required && !isValidElement(label) && ( + * + ) + } + {tooltip && ( + {translatedTooltip}
} + triggerClassName='ml-0.5 w-4 h-4' /> - ) - } - { - formSchema.type === FormTypeEnum.secretInput && ( - handleChange(e.target.value)} - onBlur={field.handleBlur} - disabled={disabled} - placeholder={memorizedPlaceholder} - autoComplete={'new-password'} - /> - ) - } - { - formSchema.type === FormTypeEnum.textNumber && ( - handleChange(e.target.value)} - onBlur={field.handleBlur} - disabled={disabled} - placeholder={memorizedPlaceholder} - /> - ) - } - { - formSchema.type === FormTypeEnum.select && ( - handleChange(v)} - disabled={disabled} - placeholder={memorizedPlaceholder} - options={memorizedOptions} - triggerPopupSameWidth - popupProps={{ - className: 'max-h-[320px] overflow-y-auto', - }} - /> - ) - } - { - formSchema.type === FormTypeEnum.radio && ( + )} +
+
+ { + [FormTypeEnum.textInput, FormTypeEnum.secretInput, FormTypeEnum.textNumber].includes(formItemType) && ( + { + handleChange(e.target.value) + }} + onBlur={field.handleBlur} + disabled={disabled} + placeholder={translatedPlaceholder} + {...getExtraProps(formItemType)} + showCopyIcon={showCopy} + /> + ) + } + { + formItemType === FormTypeEnum.select && !multiple && ( + handleChange(v)} + disabled={disabled} + placeholder={translatedPlaceholder} + options={memorizedOptions} + triggerPopupSameWidth + popupProps={{ + className: 'max-h-[320px] overflow-y-auto', + }} + /> + ) + } + { + formItemType === FormTypeEnum.checkbox /* && multiple */ && ( + field.handleChange(v)} + options={memorizedOptions} + maxHeight='200px' + /> + ) + } + { + formItemType === FormTypeEnum.dynamicSelect && ( + + ) + } + { + formItemType === FormTypeEnum.radio && ( +
+ { + memorizedOptions.map(option => ( +
!disabled && handleChange(option.value)} + > + { + formSchema.showRadioUI && ( + + ) + } + {option.label} +
+ )) + } +
+ ) + } + { + formItemType === FormTypeEnum.boolean && ( + field.handleChange(v)} + > + True + False + + ) + } + {fieldState?.validateStatus && [FormItemValidateStatusEnum.Error, FormItemValidateStatusEnum.Warning].includes(fieldState?.validateStatus) && (
- { - memorizedOptions.map(option => ( -
!disabled && handleChange(option.value)} - > - { - formSchema.showRadioUI && ( - - ) - } - {option.label} -
- )) - } + {fieldState?.[VALIDATE_STATUS_STYLE_MAP[fieldState?.validateStatus].infoFieldName as keyof FieldState]}
- ) - } - { - formSchema.type === FormTypeEnum.boolean && ( - field.handleChange(v)} - > - True - False - - ) - } - { - formSchema.url && ( - - - {renderI18nObject(formSchema?.help as any)} - - { - - } - - ) - } + )} +
-
+ {description && ( +
+ {translatedDescription} +
+ )} + { + url && ( + + + {translatedHelp} + + + + ) + } + + ) } diff --git a/web/app/components/base/form/components/base/base-form.tsx b/web/app/components/base/form/components/base/base-form.tsx index 6b7e992510..0d35380523 100644 --- a/web/app/components/base/form/components/base/base-form.tsx +++ b/web/app/components/base/form/components/base/base-form.tsx @@ -3,6 +3,7 @@ import { useCallback, useImperativeHandle, useMemo, + useState, } from 'react' import type { AnyFieldApi, @@ -12,9 +13,12 @@ import { useForm, useStore, } from '@tanstack/react-form' -import type { - FormRef, - FormSchema, +import { + type FieldState, + FormItemValidateStatusEnum, + type FormRef, + type FormSchema, + type SetFieldsParam, } from '@/app/components/base/form/types' import { BaseField, @@ -36,6 +40,8 @@ export type BaseFormProps = { disabled?: boolean formFromProps?: AnyFormApi onChange?: (field: string, value: any) => void + onSubmit?: (e: React.FormEvent) => void + preventDefaultSubmit?: boolean } & Pick const BaseForm = ({ @@ -50,6 +56,8 @@ const BaseForm = ({ disabled, formFromProps, onChange, + onSubmit, + preventDefaultSubmit = false, }: BaseFormProps) => { const initialDefaultValues = useMemo(() => { if (defaultValues) @@ -68,6 +76,8 @@ const BaseForm = ({ const { getFormValues } = useGetFormValues(form, formSchemas) const { getValidators } = useGetValidators() + const [fieldStates, setFieldStates] = useState>({}) + const showOnValues = useStore(form.store, (s: any) => { const result: Record = {} formSchemas.forEach((schema) => { @@ -81,6 +91,34 @@ const BaseForm = ({ return result }) + const setFields = useCallback((fields: SetFieldsParam[]) => { + const newFieldStates: Record = { ...fieldStates } + + for (const field of fields) { + const { name, value, errors, warnings, validateStatus, help } = field + + if (value !== undefined) + form.setFieldValue(name, value) + + let finalValidateStatus = validateStatus + if (!finalValidateStatus) { + if (errors && errors.length > 0) + finalValidateStatus = FormItemValidateStatusEnum.Error + else if (warnings && warnings.length > 0) + finalValidateStatus = FormItemValidateStatusEnum.Warning + } + + newFieldStates[name] = { + validateStatus: finalValidateStatus, + help, + errors, + warnings, + } + } + + setFieldStates(newFieldStates) + }, [form, fieldStates]) + useImperativeHandle(ref, () => { return { getForm() { @@ -89,8 +127,9 @@ const BaseForm = ({ getFormValues: (option) => { return getFormValues(option) }, + setFields, } - }, [form, getFormValues]) + }, [form, getFormValues, setFields]) const renderField = useCallback((field: AnyFieldApi) => { const formSchema = formSchemas?.find(schema => schema.name === field.name) @@ -100,18 +139,19 @@ const BaseForm = ({ ) } return null - }, [formSchemas, fieldClassName, labelClassName, inputContainerClassName, inputClassName, disabled, onChange]) + }, [formSchemas, fieldClassName, labelClassName, inputContainerClassName, inputClassName, disabled, onChange, fieldStates]) const renderFieldWrapper = useCallback((formSchema: FormSchema) => { const validators = getValidators(formSchema) @@ -142,9 +182,18 @@ const BaseForm = ({ if (!formSchemas?.length) return null + const handleSubmit = (e: React.FormEvent) => { + if (preventDefaultSubmit) { + e.preventDefault() + e.stopPropagation() + } + onSubmit?.(e) + } + return (
{formSchemas.map(renderFieldWrapper)}
diff --git a/web/app/components/base/form/components/field/select.tsx b/web/app/components/base/form/components/field/select.tsx index dee047e2eb..8a36a49510 100644 --- a/web/app/components/base/form/components/field/select.tsx +++ b/web/app/components/base/form/components/field/select.tsx @@ -11,7 +11,9 @@ type SelectFieldProps = { options: Option[] onChange?: (value: string) => void className?: string -} & Omit +} & Omit & { + multiple?: false +} const SelectField = ({ label, diff --git a/web/app/components/base/form/components/field/variable-or-constant-input.tsx b/web/app/components/base/form/components/field/variable-or-constant-input.tsx index a07e356fa2..b8a96c5401 100644 --- a/web/app/components/base/form/components/field/variable-or-constant-input.tsx +++ b/web/app/components/base/form/components/field/variable-or-constant-input.tsx @@ -1,5 +1,5 @@ import type { ChangeEvent } from 'react' -import { useState } from 'react' +import { useCallback, useState } from 'react' import { RiEditLine } from '@remixicon/react' import cn from '@/utils/classnames' import SegmentedControl from '@/app/components/base/segmented-control' @@ -33,9 +33,9 @@ const VariableOrConstantInputField = ({ }, ] - const handleVariableOrConstantChange = (value: string) => { + const handleVariableOrConstantChange = useCallback((value: string) => { setVariableType(value) - } + }, [setVariableType]) const handleVariableValueChange = () => { console.log('Variable value changed') diff --git a/web/app/components/base/form/hooks/use-get-form-values.ts b/web/app/components/base/form/hooks/use-get-form-values.ts index 36100a724a..b7d08cc005 100644 --- a/web/app/components/base/form/hooks/use-get-form-values.ts +++ b/web/app/components/base/form/hooks/use-get-form-values.ts @@ -12,7 +12,7 @@ export const useGetFormValues = (form: AnyFormApi, formSchemas: FormSchema[]) => const getFormValues = useCallback(( { - needCheckValidatedValues, + needCheckValidatedValues = true, needTransformWhenSecretFieldIsPristine, }: GetValuesOptions, ) => { @@ -20,7 +20,7 @@ export const useGetFormValues = (form: AnyFormApi, formSchemas: FormSchema[]) => if (!needCheckValidatedValues) { return { values, - isCheckValidated: false, + isCheckValidated: true, } } diff --git a/web/app/components/base/form/index.stories.tsx b/web/app/components/base/form/index.stories.tsx new file mode 100644 index 0000000000..f170cb4771 --- /dev/null +++ b/web/app/components/base/form/index.stories.tsx @@ -0,0 +1,559 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useMemo, useState } from 'react' +import { useStore } from '@tanstack/react-form' +import ContactFields from './form-scenarios/demo/contact-fields' +import { demoFormOpts } from './form-scenarios/demo/shared-options' +import { ContactMethods, UserSchema } from './form-scenarios/demo/types' +import BaseForm from './components/base/base-form' +import type { FormSchema } from './types' +import { FormTypeEnum } from './types' +import { type FormStoryRender, FormStoryWrapper } from '../../../../.storybook/utils/form-story-wrapper' +import Button from '../button' +import { TransferMethod } from '@/types/app' +import { PreviewMode } from '@/app/components/base/features/types' + +const FormStoryHost = () => null + +const meta = { + title: 'Base/Data Entry/AppForm', + component: FormStoryHost, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Helper utilities built on top of `@tanstack/react-form` that power form rendering across Dify. These stories demonstrate the `useAppForm` hook, field primitives, conditional visibility, and custom actions.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +type AppFormInstance = Parameters[0] +type ContactFieldsProps = React.ComponentProps +type ContactFieldsFormApi = ContactFieldsProps['form'] + +type PlaygroundFormFieldsProps = { + form: AppFormInstance + status: string +} + +const PlaygroundFormFields = ({ form, status }: PlaygroundFormFieldsProps) => { + type PlaygroundFormValues = typeof demoFormOpts.defaultValues + const name = useStore(form.store, state => (state.values as PlaygroundFormValues).name) + const contactFormApi = form as ContactFieldsFormApi + + return ( +
{ + event.preventDefault() + event.stopPropagation() + form.handleSubmit() + }} + > + ( + + )} + /> + ( + + )} + /> + ( + + )} + /> + + {!!name && } + + + + + +

{status}

+ + ) +} + +const FormPlayground = () => { + const [status, setStatus] = useState('Fill in the form and submit to see results.') + + return ( + { + const result = UserSchema.safeParse(formValue as typeof demoFormOpts.defaultValues) + if (!result.success) + return result.error.issues[0].message + return undefined + }, + }, + onSubmit: () => { + setStatus('Successfully saved profile.') + }, + }} + > + {form => } + + ) +} + +const mockFileUploadConfig = { + enabled: true, + allowed_file_extensions: ['pdf', 'png'], + allowed_file_upload_methods: [TransferMethod.local_file, TransferMethod.remote_url], + number_limits: 3, + preview_config: { + mode: PreviewMode.CurrentPage, + file_type_list: ['pdf', 'png'], + }, +} + +const mockFieldDefaults = { + headline: 'Dify App', + description: 'Streamline your AI workflows with configurable building blocks.', + category: 'workbench', + allowNotifications: true, + dailyLimit: 40, + attachment: [], +} + +const FieldGallery = () => { + const selectOptions = useMemo(() => [ + { value: 'workbench', label: 'Workbench' }, + { value: 'playground', label: 'Playground' }, + { value: 'production', label: 'Production' }, + ], []) + + return ( + + {form => ( +
{ + event.preventDefault() + event.stopPropagation() + form.handleSubmit() + }} + > + ( + + )} + /> + ( + + )} + /> + ( + + )} + /> + ( + + )} + /> + ( + + )} + /> + ( + + )} + /> +
+ + + +
+ + )} +
+ ) +} + +const conditionalSchemas: FormSchema[] = [ + { + type: FormTypeEnum.select, + name: 'channel', + label: 'Preferred channel', + required: true, + default: 'email', + options: ContactMethods, + }, + { + type: FormTypeEnum.textInput, + name: 'contactEmail', + label: 'Email address', + required: true, + placeholder: 'user@example.com', + show_on: [{ variable: 'channel', value: 'email' }], + }, + { + type: FormTypeEnum.textInput, + name: 'contactPhone', + label: 'Phone number', + required: true, + placeholder: '+1 555 123 4567', + show_on: [{ variable: 'channel', value: 'phone' }], + }, + { + type: FormTypeEnum.boolean, + name: 'optIn', + label: 'Opt in to marketing messages', + required: false, + }, +] + +const ConditionalFieldsStory = () => { + const [values, setValues] = useState>({ + channel: 'email', + optIn: false, + }) + + return ( +
+
+ { + setValues(prev => ({ + ...prev, + [field]: value, + })) + }} + /> +
+ +
+ ) +} + +const CustomActionsStory = () => { + return ( + { + const nextValues = value as { datasetName?: string } + if (!nextValues.datasetName || nextValues.datasetName.length < 3) + return 'Dataset name must contain at least 3 characters.' + return undefined + }, + }, + }} + > + {form => ( +
{ + event.preventDefault() + event.stopPropagation() + form.handleSubmit() + }} + > + ( + + )} + /> + ( + + )} + /> + + ( +
+ + + +
+ )} + /> +
+ + )} +
+ ) +} + +export const Playground: Story = { + render: () => , + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +const form = useAppForm({ + ...demoFormOpts, + validators: { + onSubmit: ({ value }) => UserSchema.safeParse(value).success ? undefined : 'Validation failed', + }, + onSubmit: ({ value }) => { + setStatus(\`Successfully saved profile for \${value.name}\`) + }, +}) + +return ( +
+ + {field => } + + + {field => } + + + {field => } + + {!!form.store.state.values.name && } + + + + +) + `.trim(), + }, + }, + }, +} + +export const FieldExplorer: Story = { + render: () => , + parameters: { + nextjs: { + appDirectory: true, + navigation: { + pathname: '/apps/demo-app/form', + params: { appId: 'demo-app' }, + }, + }, + docs: { + source: { + language: 'tsx', + code: ` +const form = useAppForm({ + defaultValues: { + headline: 'Dify App', + description: 'Streamline your AI workflows', + category: 'workbench', + allowNotifications: true, + dailyLimit: 40, + attachment: [], + }, +}) + +return ( +
+ + {field => } + + + {field => } + + + {field => } + + + {field => } + + + {field => } + + + {field => } + + + + +
+) + `.trim(), + }, + }, + }, +} + +export const ConditionalVisibility: Story = { + render: () => , + parameters: { + docs: { + description: { + story: 'Demonstrates schema-driven visibility using `show_on` conditions rendered through the reusable `BaseForm` component.', + }, + source: { + language: 'tsx', + code: ` +const conditionalSchemas: FormSchema[] = [ + { type: FormTypeEnum.select, name: 'channel', label: 'Preferred channel', options: ContactMethods }, + { type: FormTypeEnum.textInput, name: 'contactEmail', label: 'Email', show_on: [{ variable: 'channel', value: 'email' }] }, + { type: FormTypeEnum.textInput, name: 'contactPhone', label: 'Phone', show_on: [{ variable: 'channel', value: 'phone' }] }, + { type: FormTypeEnum.boolean, name: 'optIn', label: 'Opt in to marketing messages' }, +] + +return ( + setValues(prev => ({ ...prev, [field]: value }))} + /> +) + `.trim(), + }, + }, + }, +} + +export const CustomActions: Story = { + render: () => , + parameters: { + docs: { + description: { + story: 'Shows how to replace the default submit button with a fully custom footer leveraging contextual form state.', + }, + source: { + language: 'tsx', + code: ` +const form = useAppForm({ + defaultValues: { + datasetName: 'Support FAQ', + datasetDescription: 'Knowledge base snippets sourced from Zendesk exports.', + }, + validators: { + onChange: ({ value }) => value.datasetName?.length >= 3 ? undefined : 'Dataset name must contain at least 3 characters.', + }, +}) + +return ( +
+ + {field => } + + + {field => } + + + ( +
+ + + +
+ )} + /> +
+
+) + `.trim(), + }, + }, + }, +} diff --git a/web/app/components/base/form/types.ts b/web/app/components/base/form/types.ts index d18c166186..268f9db89a 100644 --- a/web/app/components/base/form/types.ts +++ b/web/app/components/base/form/types.ts @@ -6,6 +6,7 @@ import type { AnyFormApi, FieldValidators, } from '@tanstack/react-form' +import type { Locale } from '@/i18n-config' export type TypeWithI18N = { en_US: T @@ -36,31 +37,49 @@ export enum FormTypeEnum { } export type FormOption = { - label: TypeWithI18N | string + label: string | TypeWithI18N | Record value: string show_on?: FormShowOnObject[] icon?: string } -export type AnyValidators = FieldValidators +export type AnyValidators = FieldValidators + +export enum FormItemValidateStatusEnum { + Success = 'success', + Warning = 'warning', + Error = 'error', + Validating = 'validating', +} export type FormSchema = { type: FormTypeEnum name: string - label: string | ReactNode | TypeWithI18N + label: string | ReactNode | TypeWithI18N | Record required: boolean + multiple?: boolean default?: any - tooltip?: string | TypeWithI18N + description?: string | TypeWithI18N | Record + tooltip?: string | TypeWithI18N | Record show_on?: FormShowOnObject[] url?: string scope?: string - help?: string | TypeWithI18N - placeholder?: string | TypeWithI18N + help?: string | TypeWithI18N | Record + placeholder?: string | TypeWithI18N | Record options?: FormOption[] labelClassName?: string + fieldClassName?: string validators?: AnyValidators showRadioUI?: boolean disabled?: boolean + showCopy?: boolean + dynamicSelectParams?: { + plugin_id: string + provider: string + action: string + parameter: string + credential_id: string + } } export type FormValues = Record @@ -69,11 +88,25 @@ export type GetValuesOptions = { needTransformWhenSecretFieldIsPristine?: boolean needCheckValidatedValues?: boolean } + +export type FieldState = { + validateStatus?: FormItemValidateStatusEnum + help?: string | ReactNode + errors?: string[] + warnings?: string[] +} + +export type SetFieldsParam = { + name: string + value?: any +} & FieldState + export type FormRefObject = { getForm: () => AnyFormApi getFormValues: (obj: GetValuesOptions) => { values: Record isCheckValidated: boolean } + setFields: (fields: SetFieldsParam[]) => void } export type FormRef = ForwardedRef diff --git a/web/app/components/base/fullscreen-modal/index.stories.tsx b/web/app/components/base/fullscreen-modal/index.stories.tsx new file mode 100644 index 0000000000..72fd28df66 --- /dev/null +++ b/web/app/components/base/fullscreen-modal/index.stories.tsx @@ -0,0 +1,59 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import FullScreenModal from '.' + +const meta = { + title: 'Base/Feedback/FullScreenModal', + component: FullScreenModal, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Backdrop-blurred fullscreen modal. Supports close button, custom content, and optional overflow visibility.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const ModalDemo = (props: React.ComponentProps) => { + const [open, setOpen] = useState(false) + + return ( +
+ + + setOpen(false)} + closable + > +
+
+ Full-screen experience +
+
+ Place dashboards, flow builders, or immersive previews here. +
+
+
+
+ ) +} + +export const Playground: Story = { + render: args => , + args: { + open: false, + }, +} diff --git a/web/app/components/base/grid-mask/index.stories.tsx b/web/app/components/base/grid-mask/index.stories.tsx new file mode 100644 index 0000000000..1b67a1510d --- /dev/null +++ b/web/app/components/base/grid-mask/index.stories.tsx @@ -0,0 +1,51 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import GridMask from '.' + +const meta = { + title: 'Base/Layout/GridMask', + component: GridMask, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Displays a soft grid overlay with gradient mask, useful for framing hero sections or marketing callouts.', + }, + }, + }, + args: { + wrapperClassName: 'rounded-2xl p-10', + canvasClassName: '', + gradientClassName: '', + children: ( +
+ Grid Mask Demo + Beautiful backgrounds for feature highlights +

+ Place any content inside the mask. On dark backgrounds the grid and soft gradient add depth without distracting from the main message. +

+
+ ), + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const CustomBackground: Story = { + args: { + wrapperClassName: 'rounded-3xl p-10 bg-[#0A0A1A]', + gradientClassName: 'bg-gradient-to-r from-[#0A0A1A]/90 via-[#101030]/60 to-[#05050A]/90', + children: ( +
+ Custom gradient + Use your own colors +

+ Override gradient and canvas classes to match brand palettes while keeping the grid texture. +

+
+ ), + }, +} diff --git a/web/app/components/base/icons/assets/public/tracing/tencent-icon-big.svg b/web/app/components/base/icons/assets/public/tracing/tencent-icon-big.svg new file mode 100644 index 0000000000..b38316f3b6 --- /dev/null +++ b/web/app/components/base/icons/assets/public/tracing/tencent-icon-big.svg @@ -0,0 +1,23 @@ + + + logo + + + + diff --git a/web/app/components/base/icons/assets/public/tracing/tencent-icon.svg b/web/app/components/base/icons/assets/public/tracing/tencent-icon.svg new file mode 100644 index 0000000000..53347bf23c --- /dev/null +++ b/web/app/components/base/icons/assets/public/tracing/tencent-icon.svg @@ -0,0 +1,23 @@ + + + logo + + + + \ No newline at end of file diff --git a/web/app/components/base/icons/assets/vender/line/alertsAndFeedback/warning.svg b/web/app/components/base/icons/assets/vender/line/alertsAndFeedback/warning.svg new file mode 100644 index 0000000000..8174878acb --- /dev/null +++ b/web/app/components/base/icons/assets/vender/line/alertsAndFeedback/warning.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/line/arrows/IconR.svg b/web/app/components/base/icons/assets/vender/line/arrows/IconR.svg new file mode 100644 index 0000000000..7ff1df98e2 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/line/arrows/IconR.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/line/others/apps-02.svg b/web/app/components/base/icons/assets/vender/line/others/apps-02.svg deleted file mode 100644 index 8e1fec9ecc..0000000000 --- a/web/app/components/base/icons/assets/vender/line/others/apps-02.svg +++ /dev/null @@ -1,5 +0,0 @@ - - - - - diff --git a/web/app/components/base/icons/assets/vender/line/others/exchange-02.svg b/web/app/components/base/icons/assets/vender/line/others/exchange-02.svg deleted file mode 100644 index 45d2770277..0000000000 --- a/web/app/components/base/icons/assets/vender/line/others/exchange-02.svg +++ /dev/null @@ -1,3 +0,0 @@ - - - diff --git a/web/app/components/base/icons/assets/vender/line/others/file-code.svg b/web/app/components/base/icons/assets/vender/line/others/file-code.svg deleted file mode 100644 index eb77033a0a..0000000000 --- a/web/app/components/base/icons/assets/vender/line/others/file-code.svg +++ /dev/null @@ -1,3 +0,0 @@ - - - diff --git a/web/app/components/base/icons/assets/vender/other/hourglass-shape.svg b/web/app/components/base/icons/assets/vender/other/hourglass-shape.svg new file mode 100644 index 0000000000..150630f460 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/other/hourglass-shape.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/plugin/trigger.svg b/web/app/components/base/icons/assets/vender/plugin/trigger.svg new file mode 100644 index 0000000000..261fcd02b7 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/plugin/trigger.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/web/app/components/base/icons/assets/vender/solid/arrows/arrow-down-double-line.svg b/web/app/components/base/icons/assets/vender/solid/arrows/arrow-down-double-line.svg new file mode 100644 index 0000000000..56caa01c59 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/solid/arrows/arrow-down-double-line.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/solid/arrows/arrow-down-round-fill.svg b/web/app/components/base/icons/assets/vender/solid/arrows/arrow-down-round-fill.svg new file mode 100644 index 0000000000..48e70bcb51 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/solid/arrows/arrow-down-round-fill.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/solid/arrows/arrow-up-double-line.svg b/web/app/components/base/icons/assets/vender/solid/arrows/arrow-up-double-line.svg new file mode 100644 index 0000000000..1f0b9858e1 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/solid/arrows/arrow-up-double-line.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/workflow/api-aggregate.svg b/web/app/components/base/icons/assets/vender/workflow/api-aggregate.svg new file mode 100644 index 0000000000..aaf2206d21 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/workflow/api-aggregate.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/workflow/asterisk.svg b/web/app/components/base/icons/assets/vender/workflow/asterisk.svg new file mode 100644 index 0000000000..d273c7e3d5 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/workflow/asterisk.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/workflow/calendar-check-line.svg b/web/app/components/base/icons/assets/vender/workflow/calendar-check-line.svg new file mode 100644 index 0000000000..2c7f148c71 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/workflow/calendar-check-line.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/assets/vender/workflow/schedule.svg b/web/app/components/base/icons/assets/vender/workflow/schedule.svg new file mode 100644 index 0000000000..69977c4c7f --- /dev/null +++ b/web/app/components/base/icons/assets/vender/workflow/schedule.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/web/app/components/base/icons/assets/vender/workflow/trigger-all.svg b/web/app/components/base/icons/assets/vender/workflow/trigger-all.svg new file mode 100644 index 0000000000..dedcc0ad3c --- /dev/null +++ b/web/app/components/base/icons/assets/vender/workflow/trigger-all.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/web/app/components/base/icons/assets/vender/workflow/webhook-line.svg b/web/app/components/base/icons/assets/vender/workflow/webhook-line.svg new file mode 100644 index 0000000000..16fd30a961 --- /dev/null +++ b/web/app/components/base/icons/assets/vender/workflow/webhook-line.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/app/components/base/icons/icon-gallery.stories.tsx b/web/app/components/base/icons/icon-gallery.stories.tsx new file mode 100644 index 0000000000..7da71b3b0b --- /dev/null +++ b/web/app/components/base/icons/icon-gallery.stories.tsx @@ -0,0 +1,258 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import React from 'react' + +declare const require: any + +type IconComponent = React.ComponentType> + +type IconEntry = { + name: string + category: string + path: string + Component: IconComponent +} + +const iconContext = require.context('./src', true, /\.tsx$/) + +const iconEntries: IconEntry[] = iconContext + .keys() + .filter((key: string) => !key.endsWith('.stories.tsx') && !key.endsWith('.spec.tsx')) + .map((key: string) => { + const mod = iconContext(key) + const Component = mod.default as IconComponent | undefined + if (!Component) + return null + + const relativePath = key.replace(/^\.\//, '') + const path = `app/components/base/icons/src/${relativePath}` + const parts = relativePath.split('/') + const fileName = parts.pop() || '' + const category = parts.length ? parts.join('/') : '(root)' + const name = Component.displayName || fileName.replace(/\.tsx$/, '') + + return { + name, + category, + path, + Component, + } + }) + .filter(Boolean) as IconEntry[] + +const sortedEntries = [...iconEntries].sort((a, b) => { + if (a.category === b.category) + return a.name.localeCompare(b.name) + return a.category.localeCompare(b.category) +}) + +const filterEntries = (entries: IconEntry[], query: string) => { + const normalized = query.trim().toLowerCase() + if (!normalized) + return entries + + return entries.filter(entry => + entry.name.toLowerCase().includes(normalized) + || entry.path.toLowerCase().includes(normalized) + || entry.category.toLowerCase().includes(normalized), + ) +} + +const groupByCategory = (entries: IconEntry[]) => entries.reduce((acc, entry) => { + if (!acc[entry.category]) + acc[entry.category] = [] + + acc[entry.category].push(entry) + return acc +}, {} as Record) + +const containerStyle: React.CSSProperties = { + padding: 24, + display: 'flex', + flexDirection: 'column', + gap: 24, +} + +const headerStyle: React.CSSProperties = { + display: 'flex', + flexDirection: 'column', + gap: 8, +} + +const controlsStyle: React.CSSProperties = { + display: 'flex', + alignItems: 'center', + gap: 12, + flexWrap: 'wrap', +} + +const searchInputStyle: React.CSSProperties = { + padding: '8px 12px', + minWidth: 280, + borderRadius: 6, + border: '1px solid #d0d0d5', +} + +const toggleButtonStyle: React.CSSProperties = { + padding: '8px 12px', + borderRadius: 6, + border: '1px solid #d0d0d5', + background: '#fff', + cursor: 'pointer', +} + +const emptyTextStyle: React.CSSProperties = { color: '#5f5f66' } + +const sectionStyle: React.CSSProperties = { + display: 'flex', + flexDirection: 'column', + gap: 12, +} + +const gridStyle: React.CSSProperties = { + display: 'grid', + gap: 12, + gridTemplateColumns: 'repeat(auto-fill, minmax(200px, 1fr))', +} + +const cardStyle: React.CSSProperties = { + border: '1px solid #e1e1e8', + borderRadius: 8, + padding: 12, + display: 'flex', + flexDirection: 'column', + gap: 8, + minHeight: 140, +} + +const previewBaseStyle: React.CSSProperties = { + display: 'flex', + justifyContent: 'center', + alignItems: 'center', + minHeight: 48, + borderRadius: 6, +} + +const nameButtonBaseStyle: React.CSSProperties = { + display: 'inline-flex', + padding: 0, + border: 'none', + background: 'transparent', + font: 'inherit', + cursor: 'pointer', + textAlign: 'left', + fontWeight: 600, +} + +const PREVIEW_SIZE = 40 + +const IconGalleryStory = () => { + const [query, setQuery] = React.useState('') + const [copiedPath, setCopiedPath] = React.useState(null) + const [previewTheme, setPreviewTheme] = React.useState<'light' | 'dark'>('light') + + const filtered = React.useMemo(() => filterEntries(sortedEntries, query), [query]) + + const grouped = React.useMemo(() => groupByCategory(filtered), [filtered]) + + const categoryOrder = React.useMemo( + () => Object.keys(grouped).sort((a, b) => a.localeCompare(b)), + [grouped], + ) + + React.useEffect(() => { + if (!copiedPath) + return undefined + + const timerId = window.setTimeout(() => { + setCopiedPath(null) + }, 1200) + + return () => window.clearTimeout(timerId) + }, [copiedPath]) + + const handleCopy = React.useCallback((text: string) => { + navigator.clipboard?.writeText(text) + .then(() => { + setCopiedPath(text) + }) + .catch((err) => { + console.error('Failed to copy icon path:', err) + }) + }, []) + + return ( +
+
+

Icon Gallery

+

+ Browse all icon components sourced from app/components/base/icons/src. Use the search bar + to filter by name or path. +

+
+ setQuery(event.target.value)} + /> + {filtered.length} icons + +
+
+ {categoryOrder.length === 0 && ( +

No icons match the current filter.

+ )} + {categoryOrder.map(category => ( +
+

{category}

+
+ {grouped[category].map(entry => ( +
+
+ +
+ +
+ ))} +
+
+ ))} +
+ ) +} + +const meta: Meta = { + title: 'Base/Icons/Icon Gallery', + component: IconGalleryStory, + parameters: { + layout: 'fullscreen', + }, +} + +export default meta + +type Story = StoryObj + +export const All: Story = { + render: () => , +} diff --git a/web/app/components/base/icons/src/image/llm/BaichuanTextCn.module.css b/web/app/components/base/icons/src/image/llm/BaichuanTextCn.module.css new file mode 100644 index 0000000000..97ab9b22f9 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/BaichuanTextCn.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/baichuan-text-cn.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/BaichuanTextCn.tsx b/web/app/components/base/icons/src/image/llm/BaichuanTextCn.tsx new file mode 100644 index 0000000000..be9a407eb2 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/BaichuanTextCn.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './BaichuanTextCn.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'BaichuanTextCn' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/Minimax.module.css b/web/app/components/base/icons/src/image/llm/Minimax.module.css new file mode 100644 index 0000000000..551ecc3c62 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/Minimax.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/minimax.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/Minimax.tsx b/web/app/components/base/icons/src/image/llm/Minimax.tsx new file mode 100644 index 0000000000..7df7e3fcbc --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/Minimax.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './Minimax.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'Minimax' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/MinimaxText.module.css b/web/app/components/base/icons/src/image/llm/MinimaxText.module.css new file mode 100644 index 0000000000..a63be49e8b --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/MinimaxText.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/minimax-text.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/MinimaxText.tsx b/web/app/components/base/icons/src/image/llm/MinimaxText.tsx new file mode 100644 index 0000000000..840e8cb439 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/MinimaxText.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './MinimaxText.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'MinimaxText' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/Tongyi.module.css b/web/app/components/base/icons/src/image/llm/Tongyi.module.css new file mode 100644 index 0000000000..3ca440768c --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/Tongyi.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/tongyi.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/Tongyi.tsx b/web/app/components/base/icons/src/image/llm/Tongyi.tsx new file mode 100644 index 0000000000..2f62f1a355 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/Tongyi.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './Tongyi.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'Tongyi' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/TongyiText.module.css b/web/app/components/base/icons/src/image/llm/TongyiText.module.css new file mode 100644 index 0000000000..f713671808 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/TongyiText.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/tongyi-text.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/TongyiText.tsx b/web/app/components/base/icons/src/image/llm/TongyiText.tsx new file mode 100644 index 0000000000..a52f63c248 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/TongyiText.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './TongyiText.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'TongyiText' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/TongyiTextCn.module.css b/web/app/components/base/icons/src/image/llm/TongyiTextCn.module.css new file mode 100644 index 0000000000..d07e6e8bc4 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/TongyiTextCn.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/tongyi-text-cn.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/TongyiTextCn.tsx b/web/app/components/base/icons/src/image/llm/TongyiTextCn.tsx new file mode 100644 index 0000000000..c982c73aed --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/TongyiTextCn.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './TongyiTextCn.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'TongyiTextCn' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/Wxyy.module.css b/web/app/components/base/icons/src/image/llm/Wxyy.module.css new file mode 100644 index 0000000000..44344a495f --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/Wxyy.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/wxyy.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/Wxyy.tsx b/web/app/components/base/icons/src/image/llm/Wxyy.tsx new file mode 100644 index 0000000000..a3c494811e --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/Wxyy.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './Wxyy.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'Wxyy' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/WxyyText.module.css b/web/app/components/base/icons/src/image/llm/WxyyText.module.css new file mode 100644 index 0000000000..58a0c62047 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/WxyyText.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/wxyy-text.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/WxyyText.tsx b/web/app/components/base/icons/src/image/llm/WxyyText.tsx new file mode 100644 index 0000000000..e5dd6e8803 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/WxyyText.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './WxyyText.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'WxyyText' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/WxyyTextCn.module.css b/web/app/components/base/icons/src/image/llm/WxyyTextCn.module.css new file mode 100644 index 0000000000..fb5839ab07 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/WxyyTextCn.module.css @@ -0,0 +1,5 @@ +.wrapper { + display: inline-flex; + background: url(~@/app/components/base/icons/assets/image/llm/wxyy-text-cn.png) center center no-repeat; + background-size: contain; +} diff --git a/web/app/components/base/icons/src/image/llm/WxyyTextCn.tsx b/web/app/components/base/icons/src/image/llm/WxyyTextCn.tsx new file mode 100644 index 0000000000..32108adab4 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/WxyyTextCn.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import cn from '@/utils/classnames' +import s from './WxyyTextCn.module.css' + +const Icon = ( + { + ref, + className, + ...restProps + }: React.DetailedHTMLProps, HTMLSpanElement> & { + ref?: React.RefObject; + }, +) => + +Icon.displayName = 'WxyyTextCn' + +export default Icon diff --git a/web/app/components/base/icons/src/image/llm/index.ts b/web/app/components/base/icons/src/image/llm/index.ts new file mode 100644 index 0000000000..3a4e64ac18 --- /dev/null +++ b/web/app/components/base/icons/src/image/llm/index.ts @@ -0,0 +1,9 @@ +export { default as BaichuanTextCn } from './BaichuanTextCn' +export { default as MinimaxText } from './MinimaxText' +export { default as Minimax } from './Minimax' +export { default as TongyiTextCn } from './TongyiTextCn' +export { default as TongyiText } from './TongyiText' +export { default as Tongyi } from './Tongyi' +export { default as WxyyTextCn } from './WxyyTextCn' +export { default as WxyyText } from './WxyyText' +export { default as Wxyy } from './Wxyy' diff --git a/web/app/components/base/icons/src/public/billing/AwsMarketplaceDark.tsx b/web/app/components/base/icons/src/public/billing/AwsMarketplaceDark.tsx index 5aa2d6c430..7096a4d2eb 100644 --- a/web/app/components/base/icons/src/public/billing/AwsMarketplaceDark.tsx +++ b/web/app/components/base/icons/src/public/billing/AwsMarketplaceDark.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlue.tsx b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlue.tsx index 85697f9dae..8d3e6a8a8a 100644 --- a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlue.tsx +++ b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlue.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlueLight.tsx b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlueLight.tsx index bf4264f1bd..f44856be61 100644 --- a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlueLight.tsx +++ b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectBlueLight.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectOrange.tsx b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectOrange.tsx index bd6cda4470..fe76f5917f 100644 --- a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectOrange.tsx +++ b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectOrange.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectPurple.tsx b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectPurple.tsx index b70808ef8c..f5c5e7ba3a 100644 --- a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectPurple.tsx +++ b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectPurple.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectTeal.tsx b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectTeal.tsx index ddd04a1911..0d2a07e405 100644 --- a/web/app/components/base/icons/src/public/knowledge/OptionCardEffectTeal.tsx +++ b/web/app/components/base/icons/src/public/knowledge/OptionCardEffectTeal.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/dataset-card/ExternalKnowledgeBase.tsx b/web/app/components/base/icons/src/public/knowledge/dataset-card/ExternalKnowledgeBase.tsx index ea6ce30704..06bb8086bc 100644 --- a/web/app/components/base/icons/src/public/knowledge/dataset-card/ExternalKnowledgeBase.tsx +++ b/web/app/components/base/icons/src/public/knowledge/dataset-card/ExternalKnowledgeBase.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/dataset-card/General.tsx b/web/app/components/base/icons/src/public/knowledge/dataset-card/General.tsx index 6508ed57c6..6665039002 100644 --- a/web/app/components/base/icons/src/public/knowledge/dataset-card/General.tsx +++ b/web/app/components/base/icons/src/public/knowledge/dataset-card/General.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/dataset-card/Graph.tsx b/web/app/components/base/icons/src/public/knowledge/dataset-card/Graph.tsx index c1360c52ca..127367f873 100644 --- a/web/app/components/base/icons/src/public/knowledge/dataset-card/Graph.tsx +++ b/web/app/components/base/icons/src/public/knowledge/dataset-card/Graph.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/dataset-card/ParentChild.tsx b/web/app/components/base/icons/src/public/knowledge/dataset-card/ParentChild.tsx index 7c6c3baa7b..922cb2c825 100644 --- a/web/app/components/base/icons/src/public/knowledge/dataset-card/ParentChild.tsx +++ b/web/app/components/base/icons/src/public/knowledge/dataset-card/ParentChild.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/dataset-card/Qa.tsx b/web/app/components/base/icons/src/public/knowledge/dataset-card/Qa.tsx index 34ef88141e..ac41a8b153 100644 --- a/web/app/components/base/icons/src/public/knowledge/dataset-card/Qa.tsx +++ b/web/app/components/base/icons/src/public/knowledge/dataset-card/Qa.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsBlue.tsx b/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsBlue.tsx index 9fd923458e..cfd9570081 100644 --- a/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsBlue.tsx +++ b/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsBlue.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsGray.tsx b/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsGray.tsx index a646251629..2e40a70367 100644 --- a/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsGray.tsx +++ b/web/app/components/base/icons/src/public/knowledge/online-drive/BucketsGray.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/public/knowledge/online-drive/Folder.tsx b/web/app/components/base/icons/src/public/knowledge/online-drive/Folder.tsx index e7a3fdf167..c5c3ea5b72 100644 --- a/web/app/components/base/icons/src/public/knowledge/online-drive/Folder.tsx +++ b/web/app/components/base/icons/src/public/knowledge/online-drive/Folder.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/line/others/FileCode.tsx b/web/app/components/base/icons/src/public/model/Checked.tsx similarity index 87% rename from web/app/components/base/icons/src/vender/line/others/FileCode.tsx rename to web/app/components/base/icons/src/public/model/Checked.tsx index 3660aad794..7854479cd2 100644 --- a/web/app/components/base/icons/src/vender/line/others/FileCode.tsx +++ b/web/app/components/base/icons/src/public/model/Checked.tsx @@ -2,7 +2,7 @@ // DON NOT EDIT IT MANUALLY import * as React from 'react' -import data from './FileCode.json' +import data from './Checked.json' import IconBase from '@/app/components/base/icons/IconBase' import type { IconData } from '@/app/components/base/icons/IconBase' @@ -15,6 +15,6 @@ const Icon = ( }, ) => -Icon.displayName = 'FileCode' +Icon.displayName = 'Checked' export default Icon diff --git a/web/app/components/base/icons/src/public/model/index.ts b/web/app/components/base/icons/src/public/model/index.ts new file mode 100644 index 0000000000..719a6f0309 --- /dev/null +++ b/web/app/components/base/icons/src/public/model/index.ts @@ -0,0 +1 @@ +export { default as Checked } from './Checked' diff --git a/web/app/components/base/icons/src/vender/line/others/Apps02.tsx b/web/app/components/base/icons/src/public/plugins/Google.tsx similarity index 87% rename from web/app/components/base/icons/src/vender/line/others/Apps02.tsx rename to web/app/components/base/icons/src/public/plugins/Google.tsx index 3236059d8d..3e19ecd2f8 100644 --- a/web/app/components/base/icons/src/vender/line/others/Apps02.tsx +++ b/web/app/components/base/icons/src/public/plugins/Google.tsx @@ -2,7 +2,7 @@ // DON NOT EDIT IT MANUALLY import * as React from 'react' -import data from './Apps02.json' +import data from './Google.json' import IconBase from '@/app/components/base/icons/IconBase' import type { IconData } from '@/app/components/base/icons/IconBase' @@ -15,6 +15,6 @@ const Icon = ( }, ) => -Icon.displayName = 'Apps02' +Icon.displayName = 'Google' export default Icon diff --git a/web/app/components/base/icons/src/vender/line/others/Exchange02.tsx b/web/app/components/base/icons/src/public/plugins/WebReader.tsx similarity index 86% rename from web/app/components/base/icons/src/vender/line/others/Exchange02.tsx rename to web/app/components/base/icons/src/public/plugins/WebReader.tsx index 4f58de3619..5606e32f88 100644 --- a/web/app/components/base/icons/src/vender/line/others/Exchange02.tsx +++ b/web/app/components/base/icons/src/public/plugins/WebReader.tsx @@ -2,7 +2,7 @@ // DON NOT EDIT IT MANUALLY import * as React from 'react' -import data from './Exchange02.json' +import data from './WebReader.json' import IconBase from '@/app/components/base/icons/IconBase' import type { IconData } from '@/app/components/base/icons/IconBase' @@ -15,6 +15,6 @@ const Icon = ( }, ) => -Icon.displayName = 'Exchange02' +Icon.displayName = 'WebReader' export default Icon diff --git a/web/app/components/base/icons/src/public/plugins/Wikipedia.tsx b/web/app/components/base/icons/src/public/plugins/Wikipedia.tsx new file mode 100644 index 0000000000..c2fde5c1f8 --- /dev/null +++ b/web/app/components/base/icons/src/public/plugins/Wikipedia.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Wikipedia.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Wikipedia' + +export default Icon diff --git a/web/app/components/base/icons/src/public/plugins/index.ts b/web/app/components/base/icons/src/public/plugins/index.ts new file mode 100644 index 0000000000..87dc37167c --- /dev/null +++ b/web/app/components/base/icons/src/public/plugins/index.ts @@ -0,0 +1,7 @@ +export { default as Google } from './Google' +export { default as PartnerDark } from './PartnerDark' +export { default as PartnerLight } from './PartnerLight' +export { default as VerifiedDark } from './VerifiedDark' +export { default as VerifiedLight } from './VerifiedLight' +export { default as WebReader } from './WebReader' +export { default as Wikipedia } from './Wikipedia' diff --git a/web/app/components/base/icons/src/public/thought/DataSet.tsx b/web/app/components/base/icons/src/public/thought/DataSet.tsx index e279c77ec7..f35ff4efbc 100644 --- a/web/app/components/base/icons/src/public/thought/DataSet.tsx +++ b/web/app/components/base/icons/src/public/thought/DataSet.tsx @@ -18,4 +18,3 @@ const Icon = ( Icon.displayName = 'DataSet' export default Icon - diff --git a/web/app/components/base/icons/src/public/thought/Loading.tsx b/web/app/components/base/icons/src/public/thought/Loading.tsx new file mode 100644 index 0000000000..af959fba40 --- /dev/null +++ b/web/app/components/base/icons/src/public/thought/Loading.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Loading.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Loading' + +export default Icon diff --git a/web/app/components/base/icons/src/public/thought/Search.tsx b/web/app/components/base/icons/src/public/thought/Search.tsx new file mode 100644 index 0000000000..ecd98048d5 --- /dev/null +++ b/web/app/components/base/icons/src/public/thought/Search.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Search.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Search' + +export default Icon diff --git a/web/app/components/base/icons/src/public/thought/ThoughtList.tsx b/web/app/components/base/icons/src/public/thought/ThoughtList.tsx new file mode 100644 index 0000000000..e7f0e312ef --- /dev/null +++ b/web/app/components/base/icons/src/public/thought/ThoughtList.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './ThoughtList.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'ThoughtList' + +export default Icon diff --git a/web/app/components/base/icons/src/public/thought/WebReader.tsx b/web/app/components/base/icons/src/public/thought/WebReader.tsx new file mode 100644 index 0000000000..5606e32f88 --- /dev/null +++ b/web/app/components/base/icons/src/public/thought/WebReader.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './WebReader.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'WebReader' + +export default Icon diff --git a/web/app/components/base/icons/src/public/thought/index.ts b/web/app/components/base/icons/src/public/thought/index.ts index 35adcb50bb..8a45489dbf 100644 --- a/web/app/components/base/icons/src/public/thought/index.ts +++ b/web/app/components/base/icons/src/public/thought/index.ts @@ -1,2 +1,5 @@ export { default as DataSet } from './DataSet' - +export { default as Loading } from './Loading' +export { default as Search } from './Search' +export { default as ThoughtList } from './ThoughtList' +export { default as WebReader } from './WebReader' diff --git a/web/app/components/base/icons/src/public/tracing/index.ts b/web/app/components/base/icons/src/public/tracing/index.ts index 9eaf42b7e0..8911798b56 100644 --- a/web/app/components/base/icons/src/public/tracing/index.ts +++ b/web/app/components/base/icons/src/public/tracing/index.ts @@ -8,10 +8,10 @@ export { default as LangsmithIconBig } from './LangsmithIconBig' export { default as LangsmithIcon } from './LangsmithIcon' export { default as OpikIconBig } from './OpikIconBig' export { default as OpikIcon } from './OpikIcon' -export { default as PhoenixIconBig } from './PhoenixIconBig' -export { default as PhoenixIcon } from './PhoenixIcon' export { default as TencentIconBig } from './TencentIconBig' export { default as TencentIcon } from './TencentIcon' +export { default as PhoenixIconBig } from './PhoenixIconBig' +export { default as PhoenixIcon } from './PhoenixIcon' export { default as TracingIcon } from './TracingIcon' export { default as WeaveIconBig } from './WeaveIconBig' export { default as WeaveIcon } from './WeaveIcon' diff --git a/web/app/components/base/icons/src/vender/knowledge/AddChunks.tsx b/web/app/components/base/icons/src/vender/knowledge/AddChunks.tsx index fc1270ae66..8068f7113c 100644 --- a/web/app/components/base/icons/src/vender/knowledge/AddChunks.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/AddChunks.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/ArrowShape.tsx b/web/app/components/base/icons/src/vender/knowledge/ArrowShape.tsx index 72ae12c7dd..b93cd2a325 100644 --- a/web/app/components/base/icons/src/vender/knowledge/ArrowShape.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/ArrowShape.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/Divider.tsx b/web/app/components/base/icons/src/vender/knowledge/Divider.tsx index 56606448be..8f7537b0db 100644 --- a/web/app/components/base/icons/src/vender/knowledge/Divider.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/Divider.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/Economic.tsx b/web/app/components/base/icons/src/vender/knowledge/Economic.tsx index c69560689e..52e2262fc1 100644 --- a/web/app/components/base/icons/src/vender/knowledge/Economic.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/Economic.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/FullTextSearch.tsx b/web/app/components/base/icons/src/vender/knowledge/FullTextSearch.tsx index 0e36656343..714e63ecc0 100644 --- a/web/app/components/base/icons/src/vender/knowledge/FullTextSearch.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/FullTextSearch.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/GeneralChunk.tsx b/web/app/components/base/icons/src/vender/knowledge/GeneralChunk.tsx index 6e75ed920a..e269f3ad91 100644 --- a/web/app/components/base/icons/src/vender/knowledge/GeneralChunk.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/GeneralChunk.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/HighQuality.tsx b/web/app/components/base/icons/src/vender/knowledge/HighQuality.tsx index 880e63a003..964e4f1a2b 100644 --- a/web/app/components/base/icons/src/vender/knowledge/HighQuality.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/HighQuality.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/HybridSearch.tsx b/web/app/components/base/icons/src/vender/knowledge/HybridSearch.tsx index 45d76c2fd1..b9a83245ee 100644 --- a/web/app/components/base/icons/src/vender/knowledge/HybridSearch.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/HybridSearch.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/ParentChildChunk.tsx b/web/app/components/base/icons/src/vender/knowledge/ParentChildChunk.tsx index 949cd508de..87664b706a 100644 --- a/web/app/components/base/icons/src/vender/knowledge/ParentChildChunk.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/ParentChildChunk.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/QuestionAndAnswer.tsx b/web/app/components/base/icons/src/vender/knowledge/QuestionAndAnswer.tsx index 6ebc279a15..2492e63710 100644 --- a/web/app/components/base/icons/src/vender/knowledge/QuestionAndAnswer.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/QuestionAndAnswer.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/SearchMenu.tsx b/web/app/components/base/icons/src/vender/knowledge/SearchMenu.tsx index 4826abb20f..497f24a984 100644 --- a/web/app/components/base/icons/src/vender/knowledge/SearchMenu.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/SearchMenu.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/knowledge/VectorSearch.tsx b/web/app/components/base/icons/src/vender/knowledge/VectorSearch.tsx index 2346033f89..fa22a54587 100644 --- a/web/app/components/base/icons/src/vender/knowledge/VectorSearch.tsx +++ b/web/app/components/base/icons/src/vender/knowledge/VectorSearch.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/line/alertsAndFeedback/Warning.json b/web/app/components/base/icons/src/vender/line/alertsAndFeedback/Warning.json new file mode 100644 index 0000000000..e131493a55 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/alertsAndFeedback/Warning.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "12", + "height": "12", + "viewBox": "0 0 12 12", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M6.43295 1.50009L11.1961 9.7501C11.3342 9.98925 11.2523 10.295 11.0131 10.4331C10.9371 10.477 10.8509 10.5001 10.7631 10.5001H1.23682C0.960676 10.5001 0.736816 10.2762 0.736816 10.0001C0.736816 9.9123 0.759921 9.8261 0.803806 9.7501L5.56695 1.50009C5.705 1.26094 6.0108 1.179 6.24995 1.31707C6.32595 1.36096 6.3891 1.42408 6.43295 1.50009ZM5.49995 8.0001V9.0001H6.49995V8.0001H5.49995ZM5.49995 4.50008V7.0001H6.49995V4.50008H5.49995Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "Warning" +} diff --git a/web/app/components/base/icons/src/vender/line/alertsAndFeedback/Warning.tsx b/web/app/components/base/icons/src/vender/line/alertsAndFeedback/Warning.tsx new file mode 100644 index 0000000000..b73363b2c2 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/alertsAndFeedback/Warning.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Warning.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Warning' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/alertsAndFeedback/index.ts b/web/app/components/base/icons/src/vender/line/alertsAndFeedback/index.ts index f0a0faf74d..4e721d70eb 100644 --- a/web/app/components/base/icons/src/vender/line/alertsAndFeedback/index.ts +++ b/web/app/components/base/icons/src/vender/line/alertsAndFeedback/index.ts @@ -1,3 +1,4 @@ export { default as AlertTriangle } from './AlertTriangle' export { default as ThumbsDown } from './ThumbsDown' export { default as ThumbsUp } from './ThumbsUp' +export { default as Warning } from './Warning' diff --git a/web/app/components/base/icons/src/vender/line/arrows/IconR.json b/web/app/components/base/icons/src/vender/line/arrows/IconR.json new file mode 100644 index 0000000000..31624cf04f --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/arrows/IconR.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "14", + "height": "14", + "viewBox": "0 0 14 14", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M9.43341 6.41661L6.30441 3.2876L7.12936 2.46265L11.6666 6.99994L7.12936 11.5372L6.30441 10.7122L9.43341 7.58327H2.33331V6.41661H9.43341Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "IconR" +} diff --git a/web/app/components/base/icons/src/vender/line/arrows/IconR.tsx b/web/app/components/base/icons/src/vender/line/arrows/IconR.tsx new file mode 100644 index 0000000000..0546223e95 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/arrows/IconR.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './IconR.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'IconR' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/arrows/index.ts b/web/app/components/base/icons/src/vender/line/arrows/index.ts index c329b3636e..78554c86f1 100644 --- a/web/app/components/base/icons/src/vender/line/arrows/index.ts +++ b/web/app/components/base/icons/src/vender/line/arrows/index.ts @@ -1,3 +1,4 @@ +export { default as IconR } from './IconR' export { default as ArrowNarrowLeft } from './ArrowNarrowLeft' export { default as ArrowUpRight } from './ArrowUpRight' export { default as ChevronDownDouble } from './ChevronDownDouble' diff --git a/web/app/components/base/icons/src/vender/line/communication/AiText.json b/web/app/components/base/icons/src/vender/line/communication/AiText.json new file mode 100644 index 0000000000..2473c64c22 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/communication/AiText.json @@ -0,0 +1,39 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "14", + "height": "14", + "viewBox": "0 0 14 14", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "g", + "attributes": { + "id": "ai-text" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "id": "Vector", + "d": "M2.33301 10.5H4.08301M2.33301 7H5.24967M2.33301 3.5H11.6663M9.91634 5.83333L10.7913 7.875L12.833 8.75L10.7913 9.625L9.91634 11.6667L9.04134 9.625L6.99967 8.75L9.04134 7.875L9.91634 5.83333Z", + "stroke": "currentColor", + "stroke-width": "1.25", + "stroke-linecap": "round", + "stroke-linejoin": "round" + }, + "children": [] + } + ] + } + ] + }, + "name": "AiText" +} diff --git a/web/app/components/base/icons/src/vender/line/communication/AiText.tsx b/web/app/components/base/icons/src/vender/line/communication/AiText.tsx new file mode 100644 index 0000000000..7d5a860038 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/communication/AiText.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './AiText.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'AiText' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/communication/index.ts b/web/app/components/base/icons/src/vender/line/communication/index.ts index 27118f1dde..3ab20e8bb4 100644 --- a/web/app/components/base/icons/src/vender/line/communication/index.ts +++ b/web/app/components/base/icons/src/vender/line/communication/index.ts @@ -1,3 +1,4 @@ +export { default as AiText } from './AiText' export { default as ChatBotSlim } from './ChatBotSlim' export { default as ChatBot } from './ChatBot' export { default as CuteRobot } from './CuteRobot' diff --git a/web/app/components/base/icons/src/vender/line/layout/AlignLeft01.tsx b/web/app/components/base/icons/src/vender/line/layout/AlignLeft01.tsx new file mode 100644 index 0000000000..0761e89f56 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/layout/AlignLeft01.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './AlignLeft01.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'AlignLeft01' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/layout/AlignRight01.tsx b/web/app/components/base/icons/src/vender/line/layout/AlignRight01.tsx new file mode 100644 index 0000000000..ffe1889ff8 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/layout/AlignRight01.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './AlignRight01.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'AlignRight01' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/layout/Grid01.tsx b/web/app/components/base/icons/src/vender/line/layout/Grid01.tsx new file mode 100644 index 0000000000..bc9b6115be --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/layout/Grid01.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Grid01.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Grid01' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/layout/index.ts b/web/app/components/base/icons/src/vender/line/layout/index.ts index a6aa205faa..7c12b1f58f 100644 --- a/web/app/components/base/icons/src/vender/line/layout/index.ts +++ b/web/app/components/base/icons/src/vender/line/layout/index.ts @@ -1 +1,4 @@ +export { default as AlignLeft01 } from './AlignLeft01' +export { default as AlignRight01 } from './AlignRight01' +export { default as Grid01 } from './Grid01' export { default as LayoutGrid02 } from './LayoutGrid02' diff --git a/web/app/components/base/icons/src/vender/line/others/Apps02.json b/web/app/components/base/icons/src/vender/line/others/Apps02.json deleted file mode 100644 index 31378e175d..0000000000 --- a/web/app/components/base/icons/src/vender/line/others/Apps02.json +++ /dev/null @@ -1,36 +0,0 @@ -{ - "icon": { - "type": "element", - "isRootNode": true, - "name": "svg", - "attributes": { - "width": "16", - "height": "16", - "viewBox": "0 0 16 16", - "fill": "none", - "xmlns": "http://www.w3.org/2000/svg" - }, - "children": [ - { - "type": "element", - "name": "g", - "attributes": { - "id": "apps-2-line" - }, - "children": [ - { - "type": "element", - "name": "path", - "attributes": { - "id": "Vector", - "d": "M4.66602 7.6665C3.00916 7.6665 1.66602 6.32336 1.66602 4.6665C1.66602 3.00965 3.00916 1.6665 4.66602 1.6665C6.32287 1.6665 7.66602 3.00965 7.66602 4.6665C7.66602 6.32336 6.32287 7.6665 4.66602 7.6665ZM4.66602 14.3332C3.00916 14.3332 1.66602 12.99 1.66602 11.3332C1.66602 9.6763 3.00916 8.33317 4.66602 8.33317C6.32287 8.33317 7.66602 9.6763 7.66602 11.3332C7.66602 12.99 6.32287 14.3332 4.66602 14.3332ZM11.3327 7.6665C9.67582 7.6665 8.33268 6.32336 8.33268 4.6665C8.33268 3.00965 9.67582 1.6665 11.3327 1.6665C12.9895 1.6665 14.3327 3.00965 14.3327 4.6665C14.3327 6.32336 12.9895 7.6665 11.3327 7.6665ZM11.3327 14.3332C9.67582 14.3332 8.33268 12.99 8.33268 11.3332C8.33268 9.6763 9.67582 8.33317 11.3327 8.33317C12.9895 8.33317 14.3327 9.6763 14.3327 11.3332C14.3327 12.99 12.9895 14.3332 11.3327 14.3332ZM4.66602 6.33317C5.58649 6.33317 6.33268 5.58698 6.33268 4.6665C6.33268 3.74603 5.58649 2.99984 4.66602 2.99984C3.74554 2.99984 2.99935 3.74603 2.99935 4.6665C2.99935 5.58698 3.74554 6.33317 4.66602 6.33317ZM4.66602 12.9998C5.58649 12.9998 6.33268 12.2536 6.33268 11.3332C6.33268 10.4127 5.58649 9.6665 4.66602 9.6665C3.74554 9.6665 2.99935 10.4127 2.99935 11.3332C2.99935 12.2536 3.74554 12.9998 4.66602 12.9998ZM11.3327 6.33317C12.2531 6.33317 12.9993 5.58698 12.9993 4.6665C12.9993 3.74603 12.2531 2.99984 11.3327 2.99984C10.4122 2.99984 9.66602 3.74603 9.66602 4.6665C9.66602 5.58698 10.4122 6.33317 11.3327 6.33317ZM11.3327 12.9998C12.2531 12.9998 12.9993 12.2536 12.9993 11.3332C12.9993 10.4127 12.2531 9.6665 11.3327 9.6665C10.4122 9.6665 9.66602 10.4127 9.66602 11.3332C9.66602 12.2536 10.4122 12.9998 11.3327 12.9998Z", - "fill": "currentColor" - }, - "children": [] - } - ] - } - ] - }, - "name": "Apps02" -} diff --git a/web/app/components/base/icons/src/vender/line/others/Exchange02.json b/web/app/components/base/icons/src/vender/line/others/Exchange02.json deleted file mode 100644 index 3672d8b88b..0000000000 --- a/web/app/components/base/icons/src/vender/line/others/Exchange02.json +++ /dev/null @@ -1,26 +0,0 @@ -{ - "icon": { - "type": "element", - "isRootNode": true, - "name": "svg", - "attributes": { - "width": "16", - "height": "16", - "viewBox": "0 0 16 16", - "fill": "none", - "xmlns": "http://www.w3.org/2000/svg" - }, - "children": [ - { - "type": "element", - "name": "path", - "attributes": { - "d": "M4.66602 14.3334C3.00916 14.3334 1.66602 12.9903 1.66602 11.3334C1.66602 9.67655 3.00916 8.33342 4.66602 8.33342C6.32287 8.33342 7.66602 9.67655 7.66602 11.3334C7.66602 12.9903 6.32287 14.3334 4.66602 14.3334ZM11.3327 7.66675C9.67582 7.66675 8.33268 6.3236 8.33268 4.66675C8.33268 3.00989 9.67582 1.66675 11.3327 1.66675C12.9895 1.66675 14.3327 3.00989 14.3327 4.66675C14.3327 6.3236 12.9895 7.66675 11.3327 7.66675ZM4.66602 13.0001C5.58649 13.0001 6.33268 12.2539 6.33268 11.3334C6.33268 10.4129 5.58649 9.66675 4.66602 9.66675C3.74554 9.66675 2.99935 10.4129 2.99935 11.3334C2.99935 12.2539 3.74554 13.0001 4.66602 13.0001ZM11.3327 6.33342C12.2531 6.33342 12.9993 5.58722 12.9993 4.66675C12.9993 3.74627 12.2531 3.00008 11.3327 3.00008C10.4122 3.00008 9.66602 3.74627 9.66602 4.66675C9.66602 5.58722 10.4122 6.33342 11.3327 6.33342ZM1.99935 5.33341C1.99935 3.49247 3.49174 2.00008 5.33268 2.00008H7.33268V3.33341H5.33268C4.22812 3.33341 3.33268 4.22885 3.33268 5.33341V7.33342H1.99935V5.33341ZM13.9993 8.66675H12.666V10.6667C12.666 11.7713 11.7706 12.6667 10.666 12.6667H8.66602V14.0001H10.666C12.5069 14.0001 13.9993 12.5077 13.9993 10.6667V8.66675Z", - "fill": "currentColor" - }, - "children": [] - } - ] - }, - "name": "Exchange02" -} diff --git a/web/app/components/base/icons/src/vender/line/others/FileCode.json b/web/app/components/base/icons/src/vender/line/others/FileCode.json deleted file mode 100644 index d61af3fdb3..0000000000 --- a/web/app/components/base/icons/src/vender/line/others/FileCode.json +++ /dev/null @@ -1,26 +0,0 @@ -{ - "icon": { - "type": "element", - "isRootNode": true, - "name": "svg", - "attributes": { - "width": "16", - "height": "16", - "viewBox": "0 0 16 16", - "fill": "none", - "xmlns": "http://www.w3.org/2000/svg" - }, - "children": [ - { - "type": "element", - "name": "path", - "attributes": { - "d": "M10 2.66659H3.33333V13.3333H12.6667V5.33325H10V2.66659ZM2 1.99445C2 1.62929 2.29833 1.33325 2.66567 1.33325H10.6667L13.9998 4.66658L14 13.9949C14 14.3659 13.7034 14.6666 13.3377 14.6666H2.66227C2.29651 14.6666 2 14.3631 2 14.0054V1.99445ZM11.7713 7.99992L9.4142 10.3569L8.4714 9.41412L9.8856 7.99992L8.4714 6.58571L9.4142 5.6429L11.7713 7.99992ZM4.22877 7.99992L6.58579 5.6429L7.5286 6.58571L6.11438 7.99992L7.5286 9.41412L6.58579 10.3569L4.22877 7.99992Z", - "fill": "currentColor" - }, - "children": [] - } - ] - }, - "name": "FileCode" -} diff --git a/web/app/components/base/icons/src/vender/line/others/index.ts b/web/app/components/base/icons/src/vender/line/others/index.ts index 2322e9d9f1..99db66b397 100644 --- a/web/app/components/base/icons/src/vender/line/others/index.ts +++ b/web/app/components/base/icons/src/vender/line/others/index.ts @@ -1,10 +1,7 @@ -export { default as Apps02 } from './Apps02' export { default as BubbleX } from './BubbleX' export { default as Colors } from './Colors' export { default as DragHandle } from './DragHandle' export { default as Env } from './Env' -export { default as Exchange02 } from './Exchange02' -export { default as FileCode } from './FileCode' export { default as GlobalVariable } from './GlobalVariable' export { default as Icon3Dots } from './Icon3Dots' export { default as LongArrowLeft } from './LongArrowLeft' diff --git a/web/app/components/base/icons/src/vender/line/users/User01.tsx b/web/app/components/base/icons/src/vender/line/users/User01.tsx new file mode 100644 index 0000000000..42f2144b97 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/users/User01.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './User01.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'User01' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/users/Users01.tsx b/web/app/components/base/icons/src/vender/line/users/Users01.tsx new file mode 100644 index 0000000000..b63daf7242 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/users/Users01.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Users01.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Users01' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/users/index.ts b/web/app/components/base/icons/src/vender/line/users/index.ts new file mode 100644 index 0000000000..9f8a35152f --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/users/index.ts @@ -0,0 +1,2 @@ +export { default as User01 } from './User01' +export { default as Users01 } from './Users01' diff --git a/web/app/components/base/icons/src/vender/line/weather/Stars02.tsx b/web/app/components/base/icons/src/vender/line/weather/Stars02.tsx new file mode 100644 index 0000000000..8a42448c70 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/weather/Stars02.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Stars02.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Stars02' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/line/weather/index.ts b/web/app/components/base/icons/src/vender/line/weather/index.ts new file mode 100644 index 0000000000..1a68bce765 --- /dev/null +++ b/web/app/components/base/icons/src/vender/line/weather/index.ts @@ -0,0 +1 @@ +export { default as Stars02 } from './Stars02' diff --git a/web/app/components/base/icons/src/vender/other/HourglassShape.json b/web/app/components/base/icons/src/vender/other/HourglassShape.json new file mode 100644 index 0000000000..27f31bbe28 --- /dev/null +++ b/web/app/components/base/icons/src/vender/other/HourglassShape.json @@ -0,0 +1,27 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "8", + "height": "14", + "viewBox": "0 0 8 14", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M8 14C8 11.7909 6.20914 10 4 10C1.79086 10 0 11.7909 0 14V0C8.05332e-08 2.20914 1.79086 4 4 4C6.20914 4 8 2.20914 8 0V14Z", + "fill": "currentColor", + "fill-opacity": "1" + }, + "children": [] + } + ] + }, + "name": "HourglassShape" +} diff --git a/web/app/components/base/icons/src/vender/other/HourglassShape.tsx b/web/app/components/base/icons/src/vender/other/HourglassShape.tsx new file mode 100644 index 0000000000..a1ef8c8d5f --- /dev/null +++ b/web/app/components/base/icons/src/vender/other/HourglassShape.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './HourglassShape.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'HourglassShape' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/other/index.ts b/web/app/components/base/icons/src/vender/other/index.ts index 8a7bb7ae28..89cbe9033d 100644 --- a/web/app/components/base/icons/src/vender/other/index.ts +++ b/web/app/components/base/icons/src/vender/other/index.ts @@ -1,6 +1,7 @@ export { default as AnthropicText } from './AnthropicText' export { default as Generator } from './Generator' export { default as Group } from './Group' +export { default as HourglassShape } from './HourglassShape' export { default as Mcp } from './Mcp' export { default as NoToolPlaceholder } from './NoToolPlaceholder' export { default as Openai } from './Openai' diff --git a/web/app/components/base/icons/src/vender/pipeline/InputField.tsx b/web/app/components/base/icons/src/vender/pipeline/InputField.tsx index 4c224844d0..981b2d38d2 100644 --- a/web/app/components/base/icons/src/vender/pipeline/InputField.tsx +++ b/web/app/components/base/icons/src/vender/pipeline/InputField.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/pipeline/PipelineFill.tsx b/web/app/components/base/icons/src/vender/pipeline/PipelineFill.tsx index e0c2cc5386..2a31601cb3 100644 --- a/web/app/components/base/icons/src/vender/pipeline/PipelineFill.tsx +++ b/web/app/components/base/icons/src/vender/pipeline/PipelineFill.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/pipeline/PipelineLine.tsx b/web/app/components/base/icons/src/vender/pipeline/PipelineLine.tsx index e18df7af48..5f37828ed5 100644 --- a/web/app/components/base/icons/src/vender/pipeline/PipelineLine.tsx +++ b/web/app/components/base/icons/src/vender/pipeline/PipelineLine.tsx @@ -11,7 +11,7 @@ const Icon = ( ref, ...props }: React.SVGProps & { - ref?: React.RefObject>; + ref?: React.RefObject>; }, ) => diff --git a/web/app/components/base/icons/src/vender/plugin/Trigger.json b/web/app/components/base/icons/src/vender/plugin/Trigger.json new file mode 100644 index 0000000000..409ef0e478 --- /dev/null +++ b/web/app/components/base/icons/src/vender/plugin/Trigger.json @@ -0,0 +1,73 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "16", + "height": "16", + "viewBox": "0 0 16 16", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "fill-rule": "evenodd", + "clip-rule": "evenodd", + "d": "M7.1499 6.35213L7.25146 6.38208L14.2248 9.03898L14.3172 9.08195C14.7224 9.30788 14.778 9.87906 14.424 10.179L14.342 10.2389L11.8172 11.817L10.2391 14.3417C9.96271 14.7839 9.32424 14.751 9.08219 14.317L9.03923 14.2245L6.38232 7.25122C6.18829 6.74188 6.64437 6.24196 7.1499 6.35213ZM9.81201 12.5084L10.7671 10.981L10.8114 10.9185C10.8589 10.8589 10.9163 10.8075 10.9813 10.7668L12.5086 9.81177L8.15251 8.15226L9.81201 12.5084Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M5.2124 10.3977L3.56266 12.0474L2.61995 11.1047L4.26969 9.455L5.2124 10.3977Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M3.66683 7.99992H1.3335V6.66659H3.66683V7.99992Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M5.2124 4.2688L4.26969 5.21151L2.61995 3.56177L3.56266 2.61906L5.2124 4.2688Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M12.0477 3.56177L10.3979 5.21151L9.45524 4.2688L11.105 2.61906L12.0477 3.56177Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M8.00016 3.66659H6.66683V1.33325H8.00016V3.66659Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "Trigger" +} diff --git a/web/app/components/base/icons/src/vender/plugin/Trigger.tsx b/web/app/components/base/icons/src/vender/plugin/Trigger.tsx new file mode 100644 index 0000000000..b8f6a56ca7 --- /dev/null +++ b/web/app/components/base/icons/src/vender/plugin/Trigger.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Trigger.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Trigger' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/plugin/index.ts b/web/app/components/base/icons/src/vender/plugin/index.ts index 943c764116..b345526eb7 100644 --- a/web/app/components/base/icons/src/vender/plugin/index.ts +++ b/web/app/components/base/icons/src/vender/plugin/index.ts @@ -1,2 +1,3 @@ export { default as BoxSparkleFill } from './BoxSparkleFill' export { default as LeftCorner } from './LeftCorner' +export { default as Trigger } from './Trigger' diff --git a/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownDoubleLine.json b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownDoubleLine.json new file mode 100644 index 0000000000..17bc271b9e --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownDoubleLine.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "16", + "height": "16", + "viewBox": "0 0 16 16", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M8.00001 12.7761L12.1381 8.63804L11.1953 7.69524L8.00001 10.8905L4.80475 7.69524L3.86194 8.63804L8.00001 12.7761ZM8.00001 9.00951L12.1381 4.87146L11.1953 3.92865L8.00001 7.12391L4.80475 3.92865L3.86194 4.87146L8.00001 9.00951Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "ArrowDownDoubleLine" +} diff --git a/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownDoubleLine.tsx b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownDoubleLine.tsx new file mode 100644 index 0000000000..166a5b624b --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownDoubleLine.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './ArrowDownDoubleLine.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'ArrowDownDoubleLine' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownRoundFill.json b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownRoundFill.json new file mode 100644 index 0000000000..b150caf879 --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownRoundFill.json @@ -0,0 +1,27 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "16", + "height": "16", + "viewBox": "0 0 16 16", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M6.02888 6.23572C5.08558 6.23572 4.56458 7.33027 5.15943 8.06239L7.13069 10.4885C7.57898 11.0403 8.42124 11.0403 8.86962 10.4885L10.8408 8.06239C11.4357 7.33027 10.9147 6.23572 9.97134 6.23572H6.02888Z", + "fill": "currentColor", + "fill-opacity": "0.3" + }, + "children": [] + } + ] + }, + "name": "ArrowDownRoundFill" +} diff --git a/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownRoundFill.tsx b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownRoundFill.tsx new file mode 100644 index 0000000000..24a1ea53fd --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/ArrowDownRoundFill.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './ArrowDownRoundFill.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'ArrowDownRoundFill' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/solid/arrows/ArrowUpDoubleLine.json b/web/app/components/base/icons/src/vender/solid/arrows/ArrowUpDoubleLine.json new file mode 100644 index 0000000000..b76fc3e80c --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/ArrowUpDoubleLine.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "16", + "height": "16", + "viewBox": "0 0 16 16", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M8 3.22388L3.86194 7.36193L4.80475 8.30473L8 5.10949L11.1953 8.30473L12.1381 7.36193L8 3.22388ZM8 6.99046L3.86194 11.1285L4.80475 12.0713L8 8.87606L11.1953 12.0713L12.1381 11.1285L8 6.99046Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "ArrowUpDoubleLine" +} diff --git a/web/app/components/base/icons/src/vender/solid/arrows/ArrowUpDoubleLine.tsx b/web/app/components/base/icons/src/vender/solid/arrows/ArrowUpDoubleLine.tsx new file mode 100644 index 0000000000..06ba38ec70 --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/ArrowUpDoubleLine.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './ArrowUpDoubleLine.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'ArrowUpDoubleLine' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/solid/arrows/ChevronDown.tsx b/web/app/components/base/icons/src/vender/solid/arrows/ChevronDown.tsx new file mode 100644 index 0000000000..643ddfbf79 --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/ChevronDown.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './ChevronDown.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'ChevronDown' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/solid/arrows/HighPriority.tsx b/web/app/components/base/icons/src/vender/solid/arrows/HighPriority.tsx new file mode 100644 index 0000000000..af6fa05e5c --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/HighPriority.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './HighPriority.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'HighPriority' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/solid/arrows/index.ts b/web/app/components/base/icons/src/vender/solid/arrows/index.ts new file mode 100644 index 0000000000..58ce9aa8ac --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/arrows/index.ts @@ -0,0 +1,5 @@ +export { default as ArrowDownDoubleLine } from './ArrowDownDoubleLine' +export { default as ArrowDownRoundFill } from './ArrowDownRoundFill' +export { default as ArrowUpDoubleLine } from './ArrowUpDoubleLine' +export { default as ChevronDown } from './ChevronDown' +export { default as HighPriority } from './HighPriority' diff --git a/web/app/components/base/icons/src/vender/solid/communication/AiText.json b/web/app/components/base/icons/src/vender/solid/communication/AiText.json new file mode 100644 index 0000000000..65860e58b9 --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/communication/AiText.json @@ -0,0 +1,53 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "24", + "height": "24", + "viewBox": "0 0 24 24", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M4 5C3.44772 5 3 5.44772 3 6C3 6.55228 3.44772 7 4 7H20C20.5523 7 21 6.55228 21 6C21 5.44772 20.5523 5 20 5H4Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M17.9191 9.60608C17.7616 9.2384 17.4 9 17 9C16.6 9 16.2384 9.2384 16.0809 9.60608L14.7384 12.7384L11.6061 14.0809C11.2384 14.2384 11 14.6 11 15C11 15.4 11.2384 15.7616 11.6061 15.9191L14.7384 17.2616L16.0809 20.3939C16.2384 20.7616 16.6 21 17 21C17.4 21 17.7616 20.7616 17.9191 20.3939L19.2616 17.2616L22.3939 15.9191C22.7616 15.7616 23 15.4 23 15C23 14.6 22.7616 14.2384 22.3939 14.0809L19.2616 12.7384L17.9191 9.60608Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M4 11C3.44772 11 3 11.4477 3 12C3 12.5523 3.44772 13 4 13H9C9.55228 13 10 12.5523 10 12C10 11.4477 9.55228 11 9 11H4Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M4 17C3.44772 17 3 17.4477 3 18C3 18.5523 3.44772 19 4 19H7C7.55228 19 8 18.5523 8 18C8 17.4477 7.55228 17 7 17H4Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "AiText" +} diff --git a/web/app/components/base/icons/src/vender/solid/communication/AiText.tsx b/web/app/components/base/icons/src/vender/solid/communication/AiText.tsx new file mode 100644 index 0000000000..7d5a860038 --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/communication/AiText.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './AiText.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'AiText' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/solid/communication/index.ts b/web/app/components/base/icons/src/vender/solid/communication/index.ts index a1659b7b18..7d2a3a5a95 100644 --- a/web/app/components/base/icons/src/vender/solid/communication/index.ts +++ b/web/app/components/base/icons/src/vender/solid/communication/index.ts @@ -1,3 +1,4 @@ +export { default as AiText } from './AiText' export { default as BubbleTextMod } from './BubbleTextMod' export { default as ChatBot } from './ChatBot' export { default as CuteRobot } from './CuteRobot' diff --git a/web/app/components/base/icons/src/vender/solid/layout/Grid01.tsx b/web/app/components/base/icons/src/vender/solid/layout/Grid01.tsx new file mode 100644 index 0000000000..bc9b6115be --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/layout/Grid01.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Grid01.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Grid01' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/solid/layout/index.ts b/web/app/components/base/icons/src/vender/solid/layout/index.ts new file mode 100644 index 0000000000..73a2513d51 --- /dev/null +++ b/web/app/components/base/icons/src/vender/solid/layout/index.ts @@ -0,0 +1 @@ +export { default as Grid01 } from './Grid01' diff --git a/web/app/components/base/icons/src/vender/workflow/ApiAggregate.json b/web/app/components/base/icons/src/vender/workflow/ApiAggregate.json new file mode 100644 index 0000000000..1057842352 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/ApiAggregate.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "16", + "height": "16", + "viewBox": "0 0 16 16", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M5.92578 11.0094C5.92578 10.0174 5.12163 9.21256 4.12956 9.21256C3.13752 9.2126 2.33333 10.0174 2.33333 11.0094C2.33349 12.0014 3.13762 12.8056 4.12956 12.8057C5.12153 12.8057 5.92562 12.0014 5.92578 11.0094ZM13.6667 11.0094C13.6667 10.0174 12.8625 9.2126 11.8704 9.21256C10.8784 9.21256 10.0742 10.0174 10.0742 11.0094C10.0744 12.0014 10.8785 12.8057 11.8704 12.8057C12.8624 12.8056 13.6665 12.0014 13.6667 11.0094ZM9.79622 4.32389C9.79619 3.33186 8.99205 2.52767 8 2.52767C7.00796 2.52767 6.20382 3.33186 6.20378 4.32389C6.20378 5.31596 7.00793 6.12012 8 6.12012C8.99207 6.12012 9.79622 5.31596 9.79622 4.32389ZM11.1296 4.32389C11.1296 5.82351 10.0748 7.07628 8.66667 7.38184V7.9196L9.74284 8.71387C10.3012 8.19607 11.0489 7.87923 11.8704 7.87923C13.5989 7.87927 15 9.28101 15 11.0094C14.9998 12.7377 13.5988 14.139 11.8704 14.139C10.1421 14.139 8.74104 12.7378 8.74089 11.0094C8.74089 10.5837 8.82585 10.1776 8.97982 9.80762L8 9.08366L7.01953 9.80762C7.17356 10.1777 7.25911 10.5836 7.25911 11.0094C7.25896 12.7378 5.85791 14.139 4.12956 14.139C2.40124 14.139 1.00016 12.7377 1 11.0094C1 9.28101 2.40114 7.87927 4.12956 7.87923C4.95094 7.87923 5.69819 8.19627 6.25651 8.71387L7.33333 7.9196V7.38184C5.92523 7.07628 4.87044 5.82351 4.87044 4.32389C4.87048 2.59548 6.27158 1.19434 8 1.19434C9.72843 1.19434 11.1295 2.59548 11.1296 4.32389Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "ApiAggregate" +} diff --git a/web/app/components/base/icons/src/vender/workflow/ApiAggregate.tsx b/web/app/components/base/icons/src/vender/workflow/ApiAggregate.tsx new file mode 100644 index 0000000000..64193e900b --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/ApiAggregate.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './ApiAggregate.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'ApiAggregate' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/workflow/Asterisk.json b/web/app/components/base/icons/src/vender/workflow/Asterisk.json new file mode 100644 index 0000000000..d7fa156d99 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/Asterisk.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "14", + "height": "14", + "viewBox": "0 0 14 14", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M7.58325 1.75L7.58314 5.98908L11.2549 3.86982L11.8382 4.88018L8.16705 6.99942L11.8382 9.11983L11.2549 10.1302L7.58314 8.01033L7.58325 12.25H6.41659L6.41647 8.01033L2.74495 10.1302L2.16162 9.11983L5.83254 7L2.16162 4.88018L2.74495 3.86982L6.41647 5.98908L6.41659 1.75H7.58325Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "Asterisk" +} diff --git a/web/app/components/base/icons/src/vender/workflow/Asterisk.tsx b/web/app/components/base/icons/src/vender/workflow/Asterisk.tsx new file mode 100644 index 0000000000..916b90429c --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/Asterisk.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Asterisk.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Asterisk' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/workflow/CalendarCheckLine.json b/web/app/components/base/icons/src/vender/workflow/CalendarCheckLine.json new file mode 100644 index 0000000000..8f77528653 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/CalendarCheckLine.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "14", + "height": "14", + "viewBox": "0 0 14 14", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M5.24984 0.583252V1.74992H8.74984V0.583252H9.9165V1.74992H12.2498C12.572 1.74992 12.8332 2.01109 12.8332 2.33325V11.6666C12.8332 11.9888 12.572 12.2499 12.2498 12.2499H1.74984C1.42767 12.2499 1.1665 11.9888 1.1665 11.6666V2.33325C1.1665 2.01109 1.42767 1.74992 1.74984 1.74992H4.08317V0.583252H5.24984ZM11.6665 5.83325H2.33317V11.0833H11.6665V5.83325ZM8.77055 6.49592L9.5955 7.32093L6.70817 10.2083L4.64578 8.14588L5.47073 7.32093L6.70817 8.55835L8.77055 6.49592ZM4.08317 2.91659H2.33317V4.66659H11.6665V2.91659H9.9165V3.49992H8.74984V2.91659H5.24984V3.49992H4.08317V2.91659Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "CalendarCheckLine" +} diff --git a/web/app/components/base/icons/src/vender/workflow/CalendarCheckLine.tsx b/web/app/components/base/icons/src/vender/workflow/CalendarCheckLine.tsx new file mode 100644 index 0000000000..e480da2f04 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/CalendarCheckLine.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './CalendarCheckLine.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'CalendarCheckLine' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/workflow/Schedule.json b/web/app/components/base/icons/src/vender/workflow/Schedule.json new file mode 100644 index 0000000000..1c2d181dc4 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/Schedule.json @@ -0,0 +1,46 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "16", + "height": "16", + "viewBox": "0 0 16 16", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M11.3333 9.33337C11.7015 9.33337 11.9999 9.63193 12 10V11.0573L12.8047 11.862L12.8503 11.9128C13.0638 12.1746 13.0487 12.5607 12.8047 12.8047C12.5606 13.0488 12.1746 13.0639 11.9128 12.8503L11.862 12.8047L10.862 11.8047C10.7371 11.6798 10.6667 11.5101 10.6667 11.3334V10C10.6668 9.63193 10.9652 9.33337 11.3333 9.33337Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "fill-rule": "evenodd", + "clip-rule": "evenodd", + "d": "M11.3333 7.33337C13.5425 7.33337 15.3333 9.12424 15.3333 11.3334C15.3333 13.5425 13.5425 15.3334 11.3333 15.3334C9.12419 15.3334 7.33333 13.5425 7.33333 11.3334C7.33333 9.12424 9.12419 7.33337 11.3333 7.33337ZM11.3333 8.66671C9.86057 8.66671 8.66667 9.86061 8.66667 11.3334C8.66667 12.8061 9.86057 14 11.3333 14C12.8061 14 14 12.8061 14 11.3334C14 9.86061 12.8061 8.66671 11.3333 8.66671Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M10.6667 1.33337C11.0349 1.33337 11.3333 1.63185 11.3333 2.00004V2.66671H12.6667C13.4031 2.66671 14 3.26367 14 4.00004V5.66671C14 6.0349 13.7015 6.33337 13.3333 6.33337C12.9651 6.33337 12.6667 6.0349 12.6667 5.66671V4.00004H3.33333V12.6667H5.66667C6.03486 12.6667 6.33333 12.9652 6.33333 13.3334C6.33333 13.7016 6.03486 14 5.66667 14H3.33333C2.59697 14 2 13.4031 2 12.6667V4.00004C2 3.26366 2.59696 2.66671 3.33333 2.66671H4.66667V2.00004C4.66667 1.63185 4.96514 1.33337 5.33333 1.33337C5.70152 1.33337 6 1.63185 6 2.00004V2.66671H10V2.00004C10 1.63185 10.2985 1.33337 10.6667 1.33337Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "Schedule" +} diff --git a/web/app/components/base/icons/src/vender/workflow/Schedule.tsx b/web/app/components/base/icons/src/vender/workflow/Schedule.tsx new file mode 100644 index 0000000000..71205efd0b --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/Schedule.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './Schedule.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'Schedule' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/workflow/TriggerAll.json b/web/app/components/base/icons/src/vender/workflow/TriggerAll.json new file mode 100644 index 0000000000..c324e8be04 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/TriggerAll.json @@ -0,0 +1,73 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "14", + "height": "14", + "viewBox": "0 0 14 14", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "fill-rule": "evenodd", + "clip-rule": "evenodd", + "d": "M5.34698 6.42505C5.10275 5.79268 5.67045 5.17005 6.29816 5.30916L6.42446 5.34758L13.0846 7.92049L13.1999 7.97518C13.7051 8.26089 13.7647 8.9802 13.3118 9.34432L13.207 9.41659L10.8196 10.8202L9.416 13.2076C9.08465 13.7711 8.28069 13.742 7.97459 13.2004L7.9199 13.0852L5.34698 6.42505ZM8.791 11.6392L9.73631 10.0325L9.73696 10.0318L9.7962 9.94458C9.86055 9.86164 9.94031 9.79125 10.0312 9.73755L10.0319 9.7369L11.6387 8.79159L6.99738 6.99797L8.791 11.6392Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M2.79751 8.9257C3.05781 8.66539 3.47985 8.66547 3.74021 8.9257C4.00057 9.18604 4.00056 9.60805 3.74021 9.86841L3.03318 10.5754C2.77283 10.8356 2.35078 10.8357 2.09047 10.5754C1.83032 10.3151 1.83033 9.89305 2.09047 9.63273L2.79751 8.9257Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M1.99998 5.66659C2.36817 5.66659 2.66665 5.96506 2.66665 6.33325C2.66665 6.70144 2.36817 6.99992 1.99998 6.99992H0.99998C0.63179 6.99992 0.333313 6.70144 0.333313 6.33325C0.333313 5.96506 0.63179 5.66659 0.99998 5.66659H1.99998Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M9.63279 2.09106C9.8931 1.83077 10.3151 1.83086 10.5755 2.09106C10.8358 2.35142 10.8359 2.77343 10.5755 3.03377L9.86847 3.7408C9.6081 4.00098 9.18605 4.0011 8.92576 3.7408C8.66559 3.4805 8.66562 3.05841 8.92576 2.7981L9.63279 2.09106Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M2.09113 2.09041C2.33521 1.84649 2.72126 1.83132 2.98305 2.04484L3.03383 2.09041L3.74087 2.79744L3.78644 2.84823C3.9999 3.11002 3.98476 3.49609 3.74087 3.74015C3.49682 3.9842 3.11079 3.9992 2.84894 3.78573L2.79816 3.74015L2.09113 3.03312L2.04555 2.98234C1.83199 2.72049 1.84705 2.33449 2.09113 2.09041Z", + "fill": "currentColor" + }, + "children": [] + }, + { + "type": "element", + "name": "path", + "attributes": { + "d": "M6.33331 0.333252C6.7015 0.333252 6.99998 0.631729 6.99998 0.999919V1.99992C6.99998 2.36811 6.7015 2.66659 6.33331 2.66659C5.96512 2.66659 5.66665 2.36811 5.66665 1.99992V0.999919C5.66665 0.631729 5.96512 0.333252 6.33331 0.333252Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "TriggerAll" +} diff --git a/web/app/components/base/icons/src/vender/workflow/TriggerAll.tsx b/web/app/components/base/icons/src/vender/workflow/TriggerAll.tsx new file mode 100644 index 0000000000..71f2dbdb36 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/TriggerAll.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './TriggerAll.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'TriggerAll' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/workflow/WebhookLine.json b/web/app/components/base/icons/src/vender/workflow/WebhookLine.json new file mode 100644 index 0000000000..8319fd25f3 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/WebhookLine.json @@ -0,0 +1,26 @@ +{ + "icon": { + "type": "element", + "isRootNode": true, + "name": "svg", + "attributes": { + "width": "16", + "height": "16", + "viewBox": "0 0 16 16", + "fill": "none", + "xmlns": "http://www.w3.org/2000/svg" + }, + "children": [ + { + "type": "element", + "name": "path", + "attributes": { + "d": "M5.91246 9.42618C5.77036 9.66084 5.70006 9.85191 5.81358 10.1502C6.12696 10.9742 5.68488 11.776 4.85394 11.9937C4.07033 12.199 3.30686 11.684 3.15138 10.8451C3.01362 10.1025 3.58988 9.37451 4.40859 9.25851C4.45305 9.25211 4.49808 9.24938 4.55563 9.24591C4.58692 9.24404 4.62192 9.24191 4.66252 9.23884L5.90792 7.15051C5.12463 6.37166 4.65841 5.46114 4.7616 4.33295C4.83455 3.53543 5.14813 2.84626 5.72135 2.28138C6.81916 1.19968 8.49403 1.02449 9.78663 1.85479C11.0282 2.65232 11.5967 4.20582 11.112 5.53545L9.97403 5.22671C10.1263 4.48748 10.0137 3.82362 9.5151 3.25494C9.1857 2.87947 8.76303 2.68267 8.28236 2.61015C7.31883 2.46458 6.37278 3.08364 6.09207 4.02937C5.77342 5.10275 6.25566 5.97954 7.5735 6.64023C7.0207 7.56944 6.47235 8.50124 5.91246 9.42618ZM9.18916 5.51562C9.5877 6.2187 9.99236 6.93244 10.3934 7.63958C12.4206 7.01244 13.9491 8.13458 14.4974 9.33604C15.1597 10.7873 14.707 12.5062 13.4062 13.4016C12.0711 14.3207 10.3827 14.1636 9.19976 12.983L10.1279 12.2063C11.2962 12.963 12.3181 12.9274 13.0767 12.0314C13.7236 11.2669 13.7096 10.1271 13.0439 9.37871C12.2757 8.51511 11.2467 8.48878 10.0029 9.31784C9.48696 8.40251 8.96196 7.49424 8.46236 6.57234C8.2939 6.2616 8.10783 6.08135 7.72816 6.01558C7.09403 5.90564 6.68463 5.36109 6.66007 4.75099C6.63593 4.14763 6.99136 3.60224 7.54696 3.38974C8.0973 3.17924 8.74316 3.34916 9.11336 3.81707C9.4159 4.19938 9.51203 4.62966 9.35283 5.10116C9.32283 5.19018 9.28689 5.27727 9.2475 5.37261C9.22869 5.418 9.20916 5.46538 9.18916 5.51562ZM7.7013 11.2634H10.1417C10.1757 11.3087 10.2075 11.3536 10.2386 11.3973C10.3034 11.4887 10.3649 11.5755 10.4367 11.6526C10.9536 12.2052 11.8263 12.2326 12.3788 11.7197C12.9514 11.1881 12.9773 10.2951 12.4362 9.74011C11.9068 9.19704 11.0019 9.14518 10.5103 9.72018C10.2117 10.0696 9.9057 10.1107 9.50936 10.1045C8.49423 10.0888 7.47843 10.0994 6.46346 10.0994C6.52934 11.5273 5.98953 12.417 4.9189 12.6283C3.87051 12.8352 2.90496 12.3003 2.56502 11.3243C2.17891 10.2153 2.65641 9.32838 4.0361 8.62444C3.93228 8.24838 3.8274 7.86778 3.72357 7.49071C2.21981 7.81844 1.09162 9.27738 1.20809 10.9187C1.31097 12.3676 2.47975 13.6544 3.90909 13.8849C4.68542 14.0102 5.41485 13.88 6.09157 13.4962C6.96216 13.0022 7.46736 12.2254 7.7013 11.2634Z", + "fill": "currentColor" + }, + "children": [] + } + ] + }, + "name": "WebhookLine" +} diff --git a/web/app/components/base/icons/src/vender/workflow/WebhookLine.tsx b/web/app/components/base/icons/src/vender/workflow/WebhookLine.tsx new file mode 100644 index 0000000000..0379692808 --- /dev/null +++ b/web/app/components/base/icons/src/vender/workflow/WebhookLine.tsx @@ -0,0 +1,20 @@ +// GENERATE BY script +// DON NOT EDIT IT MANUALLY + +import * as React from 'react' +import data from './WebhookLine.json' +import IconBase from '@/app/components/base/icons/IconBase' +import type { IconData } from '@/app/components/base/icons/IconBase' + +const Icon = ( + { + ref, + ...props + }: React.SVGProps & { + ref?: React.RefObject>; + }, +) => + +Icon.displayName = 'WebhookLine' + +export default Icon diff --git a/web/app/components/base/icons/src/vender/workflow/index.ts b/web/app/components/base/icons/src/vender/workflow/index.ts index 37b7306f7b..26ee3e4126 100644 --- a/web/app/components/base/icons/src/vender/workflow/index.ts +++ b/web/app/components/base/icons/src/vender/workflow/index.ts @@ -1,6 +1,9 @@ export { default as Agent } from './Agent' export { default as Answer } from './Answer' +export { default as ApiAggregate } from './ApiAggregate' export { default as Assigner } from './Assigner' +export { default as Asterisk } from './Asterisk' +export { default as CalendarCheckLine } from './CalendarCheckLine' export { default as Code } from './Code' export { default as Datasource } from './Datasource' export { default as DocsExtractor } from './DocsExtractor' @@ -19,6 +22,9 @@ export { default as LoopEnd } from './LoopEnd' export { default as Loop } from './Loop' export { default as ParameterExtractor } from './ParameterExtractor' export { default as QuestionClassifier } from './QuestionClassifier' +export { default as Schedule } from './Schedule' export { default as TemplatingTransform } from './TemplatingTransform' +export { default as TriggerAll } from './TriggerAll' export { default as VariableX } from './VariableX' +export { default as WebhookLine } from './WebhookLine' export { default as WindowCursor } from './WindowCursor' diff --git a/web/app/components/base/image-gallery/index.stories.tsx b/web/app/components/base/image-gallery/index.stories.tsx new file mode 100644 index 0000000000..c1b463170c --- /dev/null +++ b/web/app/components/base/image-gallery/index.stories.tsx @@ -0,0 +1,39 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import ImageGallery from '.' + +const IMAGE_SOURCES = [ + 'data:image/svg+xml;utf8,Dataset', + 'data:image/svg+xml;utf8,Playground', + 'data:image/svg+xml;utf8,Workflow', + 'data:image/svg+xml;utf8,Prompts', +] + +const meta = { + title: 'Base/Data Display/ImageGallery', + component: ImageGallery, + parameters: { + docs: { + description: { + component: 'Responsive thumbnail grid with lightbox preview for larger imagery.', + }, + source: { + language: 'tsx', + code: ` +', + 'data:image/svg+xml;utf8,', +]} /> + `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + srcs: IMAGE_SOURCES, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} diff --git a/web/app/components/base/image-gallery/index.tsx b/web/app/components/base/image-gallery/index.tsx index 0f9061fdb6..fdb9711292 100644 --- a/web/app/components/base/image-gallery/index.tsx +++ b/web/app/components/base/image-gallery/index.tsx @@ -1,9 +1,9 @@ 'use client' +import ImagePreview from '@/app/components/base/image-uploader/image-preview' +import cn from '@/utils/classnames' import type { FC } from 'react' import React, { useState } from 'react' import s from './style.module.css' -import cn from '@/utils/classnames' -import ImagePreview from '@/app/components/base/image-uploader/image-preview' type Props = { srcs: string[] @@ -36,10 +36,8 @@ const ImageGallery: FC = ({ const imgStyle = getWidthStyle(imgNum) return (
- {/* TODO: support preview */} {srcs.map((src, index) => ( - - = ({ imagePreviewUrl && ( setImagePreviewUrl('')} title={''} /> + onCancel={() => setImagePreviewUrl('')} + title={''} + /> ) }
diff --git a/web/app/components/base/image-uploader/hooks.ts b/web/app/components/base/image-uploader/hooks.ts index 41074000a2..524e86cc1b 100644 --- a/web/app/components/base/image-uploader/hooks.ts +++ b/web/app/components/base/image-uploader/hooks.ts @@ -2,7 +2,7 @@ import { useCallback, useMemo, useRef, useState } from 'react' import type { ClipboardEvent } from 'react' import { useParams } from 'next/navigation' import { useTranslation } from 'react-i18next' -import { imageUpload } from './utils' +import { getImageUploadErrorMessage, imageUpload } from './utils' import { useToastContext } from '@/app/components/base/toast' import { ALLOW_FILE_EXTENSIONS, TransferMethod } from '@/types/app' import type { ImageFile, VisionSettings } from '@/types/app' @@ -81,8 +81,9 @@ export const useImageFiles = () => { filesRef.current = newFiles setFiles(newFiles) }, - onErrorCallback: () => { - notify({ type: 'error', message: t('common.imageUploader.uploadFromComputerUploadError') }) + onErrorCallback: (error?: any) => { + const errorMessage = getImageUploadErrorMessage(error, t('common.imageUploader.uploadFromComputerUploadError'), t) + notify({ type: 'error', message: errorMessage }) const newFiles = [...files.slice(0, index), { ...currentImageFile, progress: -1 }, ...files.slice(index + 1)] filesRef.current = newFiles setFiles(newFiles) @@ -158,8 +159,9 @@ export const useLocalFileUploader = ({ limit, disabled = false, onUpload }: useL onSuccessCallback: (res) => { onUpload({ ...imageFile, fileId: res.id, progress: 100 }) }, - onErrorCallback: () => { - notify({ type: 'error', message: t('common.imageUploader.uploadFromComputerUploadError') }) + onErrorCallback: (error?: any) => { + const errorMessage = getImageUploadErrorMessage(error, t('common.imageUploader.uploadFromComputerUploadError'), t) + notify({ type: 'error', message: errorMessage }) onUpload({ ...imageFile, progress: -1 }) }, }, !!params.token) diff --git a/web/app/components/base/image-uploader/image-list.stories.tsx b/web/app/components/base/image-uploader/image-list.stories.tsx new file mode 100644 index 0000000000..530ef69556 --- /dev/null +++ b/web/app/components/base/image-uploader/image-list.stories.tsx @@ -0,0 +1,182 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useMemo, useState } from 'react' +import ImageList from './image-list' +import ImageLinkInput from './image-link-input' +import type { ImageFile } from '@/types/app' +import { TransferMethod } from '@/types/app' + +const SAMPLE_BASE64 + = '' + +const createRemoteImage = ( + id: string, + progress: number, + url: string, +): ImageFile => ({ + type: TransferMethod.remote_url, + _id: id, + fileId: `remote-${id}`, + progress, + url, +}) + +const createLocalImage = (id: string, progress: number): ImageFile => ({ + type: TransferMethod.local_file, + _id: id, + fileId: `local-${id}`, + progress, + url: SAMPLE_BASE64, + base64Url: SAMPLE_BASE64, +}) + +const initialImages: ImageFile[] = [ + createLocalImage('local-initial', 100), + createRemoteImage( + 'remote-loading', + 40, + 'https://images.unsplash.com/photo-1500530855697-b586d89ba3ee?auto=format&fit=crop&w=300&q=80', + ), + { + ...createRemoteImage( + 'remote-error', + -1, + 'https://example.com/not-an-image.jpg', + ), + url: 'https://example.com/not-an-image.jpg', + }, +] + +const meta = { + title: 'Base/Data Entry/ImageList', + component: ImageList, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Renders thumbnails for uploaded images and manages their states like uploading, error, and deletion.', + }, + }, + }, + argTypes: { + list: { control: false }, + onRemove: { control: false }, + onReUpload: { control: false }, + onImageLinkLoadError: { control: false }, + onImageLinkLoadSuccess: { control: false }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const ImageUploaderPlayground = ({ readonly }: Story['args']) => { + const [images, setImages] = useState(() => initialImages) + + const activeImages = useMemo(() => images.filter(item => !item.deleted), [images]) + + const handleRemove = (id: string) => { + setImages(prev => prev.map(item => (item._id === id ? { ...item, deleted: true } : item))) + } + + const handleReUpload = (id: string) => { + setImages(prev => prev.map((item) => { + if (item._id !== id) + return item + + return { + ...item, + progress: 60, + } + })) + + setTimeout(() => { + setImages(prev => prev.map((item) => { + if (item._id !== id) + return item + + return { + ...item, + progress: 100, + } + })) + }, 1200) + } + + const handleImageLinkLoadSuccess = (id: string) => { + setImages(prev => prev.map(item => (item._id === id ? { ...item, progress: 100 } : item))) + } + + const handleImageLinkLoadError = (id: string) => { + setImages(prev => prev.map(item => (item._id === id ? { ...item, progress: -1 } : item))) + } + + const handleUploadFromLink = (imageFile: ImageFile) => { + setImages(prev => [ + ...prev, + { + ...imageFile, + fileId: `remote-${imageFile._id}`, + }, + ]) + } + + const handleAddLocalImage = () => { + const id = `local-${Date.now()}` + setImages(prev => [ + ...prev, + createLocalImage(id, 100), + ]) + } + + return ( +
+
+ Add images +
+ + +
+
+ + + +
+ + Files state + +
+          {JSON.stringify(activeImages, null, 2)}
+        
+
+
+ ) +} + +export const Playground: Story = { + render: args => , + args: { + list: [], + }, +} + +export const ReadonlyList: Story = { + render: args => , + args: { + list: [], + }, +} diff --git a/web/app/components/base/image-uploader/utils.ts b/web/app/components/base/image-uploader/utils.ts index 0c1ada747d..3579d0541e 100644 --- a/web/app/components/base/image-uploader/utils.ts +++ b/web/app/components/base/image-uploader/utils.ts @@ -1,10 +1,29 @@ import { upload } from '@/service/base' +/** + * Get appropriate error message for image upload errors + * @param error - The error object from upload failure + * @param defaultMessage - Default error message to use if no specific error is matched + * @param t - Translation function + * @returns Localized error message + */ +export const getImageUploadErrorMessage = (error: any, defaultMessage: string, t: (key: string) => string): string => { + const errorCode = error?.response?.code + + if (errorCode === 'forbidden') + return error?.response?.message + + if (errorCode === 'file_extension_blocked') + return t('common.fileUploader.fileExtensionBlocked') + + return defaultMessage +} + type ImageUploadParams = { file: File onProgressCallback: (progress: number) => void onSuccessCallback: (res: { id: string }) => void - onErrorCallback: () => void + onErrorCallback: (error?: any) => void } type ImageUpload = (v: ImageUploadParams, isPublic?: boolean, url?: string) => void export const imageUpload: ImageUpload = ({ @@ -30,7 +49,7 @@ export const imageUpload: ImageUpload = ({ .then((res: { id: string }) => { onSuccessCallback(res) }) - .catch(() => { - onErrorCallback() + .catch((error) => { + onErrorCallback(error) }) } diff --git a/web/app/components/base/inline-delete-confirm/index.stories.tsx b/web/app/components/base/inline-delete-confirm/index.stories.tsx new file mode 100644 index 0000000000..e0b0757718 --- /dev/null +++ b/web/app/components/base/inline-delete-confirm/index.stories.tsx @@ -0,0 +1,87 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { fn } from 'storybook/test' +import { useState } from 'react' +import InlineDeleteConfirm from '.' + +const meta = { + title: 'Base/Feedback/InlineDeleteConfirm', + component: InlineDeleteConfirm, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Compact confirmation prompt that appears inline, commonly used near delete buttons or destructive controls.', + }, + }, + }, + argTypes: { + variant: { + control: 'select', + options: ['delete', 'warning', 'info'], + }, + }, + args: { + title: 'Delete this item?', + confirmText: 'Delete', + cancelText: 'Cancel', + onConfirm: fn(), + onCancel: fn(), + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const InlineDeleteConfirmDemo = (args: Story['args']) => { + const [visible, setVisible] = useState(true) + + return ( +
+ + {visible && ( + { + console.log('✅ Confirm clicked') + setVisible(false) + }} + onCancel={() => { + console.log('❎ Cancel clicked') + setVisible(false) + }} + /> + )} +
+ ) +} + +export const Playground: Story = { + render: args => , +} + +export const WarningVariant: Story = { + render: args => , + args: { + variant: 'warning', + title: 'Archive conversation?', + confirmText: 'Archive', + cancelText: 'Keep', + }, +} + +export const InfoVariant: Story = { + render: args => , + args: { + variant: 'info', + title: 'Remove collaborator?', + confirmText: 'Remove', + cancelText: 'Keep', + }, +} diff --git a/web/app/components/base/input-number/index.stories.tsx b/web/app/components/base/input-number/index.stories.tsx index 9bb3ec1f8c..88999af9e0 100644 --- a/web/app/components/base/input-number/index.stories.tsx +++ b/web/app/components/base/input-number/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import { InputNumber } from '.' const meta = { - title: 'Base/InputNumber', + title: 'Base/Data Entry/InputNumber', component: InputNumber, parameters: { layout: 'centered', diff --git a/web/app/components/base/input-with-copy/index.spec.tsx b/web/app/components/base/input-with-copy/index.spec.tsx new file mode 100644 index 0000000000..f302f1715a --- /dev/null +++ b/web/app/components/base/input-with-copy/index.spec.tsx @@ -0,0 +1,150 @@ +import React from 'react' +import { fireEvent, render, screen, waitFor } from '@testing-library/react' +import '@testing-library/jest-dom' +import InputWithCopy from './index' + +// Mock the copy-to-clipboard library +jest.mock('copy-to-clipboard', () => jest.fn(() => true)) + +// Mock the i18n hook +jest.mock('react-i18next', () => ({ + useTranslation: () => ({ + t: (key: string) => { + const translations: Record = { + 'common.operation.copy': 'Copy', + 'common.operation.copied': 'Copied', + 'appOverview.overview.appInfo.embedded.copy': 'Copy', + 'appOverview.overview.appInfo.embedded.copied': 'Copied', + } + return translations[key] || key + }, + }), +})) + +// Mock lodash-es debounce +jest.mock('lodash-es', () => ({ + debounce: (fn: any) => fn, +})) + +describe('InputWithCopy component', () => { + beforeEach(() => { + jest.clearAllMocks() + }) + + it('renders correctly with default props', () => { + const mockOnChange = jest.fn() + render() + const input = screen.getByDisplayValue('test value') + const copyButton = screen.getByRole('button') + expect(input).toBeInTheDocument() + expect(copyButton).toBeInTheDocument() + }) + + it('hides copy button when showCopyButton is false', () => { + const mockOnChange = jest.fn() + render() + const input = screen.getByDisplayValue('test value') + const copyButton = screen.queryByRole('button') + expect(input).toBeInTheDocument() + expect(copyButton).not.toBeInTheDocument() + }) + + it('copies input value when copy button is clicked', async () => { + const copyToClipboard = require('copy-to-clipboard') + const mockOnChange = jest.fn() + render() + + const copyButton = screen.getByRole('button') + fireEvent.click(copyButton) + + expect(copyToClipboard).toHaveBeenCalledWith('test value') + }) + + it('copies custom value when copyValue prop is provided', async () => { + const copyToClipboard = require('copy-to-clipboard') + const mockOnChange = jest.fn() + render() + + const copyButton = screen.getByRole('button') + fireEvent.click(copyButton) + + expect(copyToClipboard).toHaveBeenCalledWith('custom copy value') + }) + + it('calls onCopy callback when copy button is clicked', async () => { + const onCopyMock = jest.fn() + const mockOnChange = jest.fn() + render() + + const copyButton = screen.getByRole('button') + fireEvent.click(copyButton) + + expect(onCopyMock).toHaveBeenCalledWith('test value') + }) + + it('shows copied state after successful copy', async () => { + const mockOnChange = jest.fn() + render() + + const copyButton = screen.getByRole('button') + fireEvent.click(copyButton) + + // Hover over the button to trigger tooltip + fireEvent.mouseEnter(copyButton) + + // Check if the tooltip shows "Copied" state + await waitFor(() => { + expect(screen.getByText('Copied')).toBeInTheDocument() + }, { timeout: 2000 }) + }) + + it('passes through all input props correctly', () => { + const mockOnChange = jest.fn() + render( + , + ) + + const input = screen.getByDisplayValue('test value') + expect(input).toHaveAttribute('placeholder', 'Custom placeholder') + expect(input).toBeDisabled() + expect(input).toHaveAttribute('readonly') + expect(input).toHaveClass('custom-class') + }) + + it('handles empty value correctly', () => { + const copyToClipboard = require('copy-to-clipboard') + const mockOnChange = jest.fn() + render() + const input = screen.getByRole('textbox') + const copyButton = screen.getByRole('button') + + expect(input).toBeInTheDocument() + expect(copyButton).toBeInTheDocument() + + fireEvent.click(copyButton) + expect(copyToClipboard).toHaveBeenCalledWith('') + }) + + it('maintains focus on input after copy', async () => { + const mockOnChange = jest.fn() + render() + + const input = screen.getByDisplayValue('test value') + const copyButton = screen.getByRole('button') + + input.focus() + expect(input).toHaveFocus() + + fireEvent.click(copyButton) + + // Input should maintain focus after copy + expect(input).toHaveFocus() + }) +}) diff --git a/web/app/components/base/input-with-copy/index.tsx b/web/app/components/base/input-with-copy/index.tsx new file mode 100644 index 0000000000..87b7de5005 --- /dev/null +++ b/web/app/components/base/input-with-copy/index.tsx @@ -0,0 +1,104 @@ +'use client' +import React, { useEffect, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { RiClipboardFill, RiClipboardLine } from '@remixicon/react' +import { debounce } from 'lodash-es' +import copy from 'copy-to-clipboard' +import type { InputProps } from '../input' +import Tooltip from '../tooltip' +import ActionButton from '../action-button' +import cn from '@/utils/classnames' + +export type InputWithCopyProps = { + showCopyButton?: boolean + copyValue?: string // Value to copy, defaults to input value + onCopy?: (value: string) => void // Callback when copy is triggered +} & Omit // Remove conflicting props + +const prefixEmbedded = 'appOverview.overview.appInfo.embedded' + +const InputWithCopy = React.forwardRef(( + { + showCopyButton = true, + copyValue, + onCopy, + value, + wrapperClassName, + ...inputProps + }, + ref, +) => { + const { t } = useTranslation() + const [isCopied, setIsCopied] = useState(false) + // Determine what value to copy + const valueToString = typeof value === 'string' ? value : String(value || '') + const finalCopyValue = copyValue || valueToString + + const onClickCopy = debounce(() => { + copy(finalCopyValue) + setIsCopied(true) + onCopy?.(finalCopyValue) + }, 100) + + const onMouseLeave = debounce(() => { + setIsCopied(false) + }, 100) + + useEffect(() => { + if (isCopied) { + const timeout = setTimeout(() => { + setIsCopied(false) + }, 2000) + return () => { + clearTimeout(timeout) + } + } + }, [isCopied]) + + return ( +
+ rest)(inputProps)} + /> + {showCopyButton && ( +
+ + + {isCopied ? ( + + ) : ( + + )} + + +
+ )} +
+ ) +}) + +InputWithCopy.displayName = 'InputWithCopy' + +export default InputWithCopy diff --git a/web/app/components/base/input/index.stories.tsx b/web/app/components/base/input/index.stories.tsx index cd857bc180..c877579879 100644 --- a/web/app/components/base/input/index.stories.tsx +++ b/web/app/components/base/input/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import Input from '.' const meta = { - title: 'Base/Input', + title: 'Base/Data Entry/Input', component: Input, parameters: { layout: 'centered', diff --git a/web/app/components/base/input/index.tsx b/web/app/components/base/input/index.tsx index 881aa1d610..688e1dd880 100644 --- a/web/app/components/base/input/index.tsx +++ b/web/app/components/base/input/index.tsx @@ -1,10 +1,11 @@ +import cn from '@/utils/classnames' +import { RiCloseCircleFill, RiErrorWarningLine, RiSearchLine } from '@remixicon/react' +import { type VariantProps, cva } from 'class-variance-authority' +import { noop } from 'lodash-es' import type { CSSProperties, ChangeEventHandler, FocusEventHandler } from 'react' import React from 'react' import { useTranslation } from 'react-i18next' -import { RiCloseCircleFill, RiErrorWarningLine, RiSearchLine } from '@remixicon/react' -import { type VariantProps, cva } from 'class-variance-authority' -import cn from '@/utils/classnames' -import { noop } from 'lodash-es' +import { CopyFeedbackNew } from '../copy-feedback' export const inputVariants = cva( '', @@ -24,6 +25,7 @@ export const inputVariants = cva( export type InputProps = { showLeftIcon?: boolean showClearIcon?: boolean + showCopyIcon?: boolean onClear?: () => void disabled?: boolean destructive?: boolean @@ -41,6 +43,7 @@ const Input = ({ destructive, showLeftIcon, showClearIcon, + showCopyIcon, onClear, wrapperClassName, className, @@ -92,8 +95,8 @@ const Input = ({ showLeftIcon && size === 'large' && 'pl-7', showClearIcon && value && 'pr-[26px]', showClearIcon && value && size === 'large' && 'pr-7', - destructive && 'pr-[26px]', - destructive && size === 'large' && 'pr-7', + (destructive || showCopyIcon) && 'pr-[26px]', + (destructive || showCopyIcon) && size === 'large' && 'pr-7', disabled && 'cursor-not-allowed border-transparent bg-components-input-bg-disabled text-components-input-text-filled-disabled hover:border-transparent hover:bg-components-input-bg-disabled', destructive && 'border-components-input-border-destructive bg-components-input-bg-destructive text-components-input-text-filled hover:border-components-input-border-destructive hover:bg-components-input-bg-destructive focus:border-components-input-border-destructive focus:bg-components-input-bg-destructive', className, @@ -115,6 +118,14 @@ const Input = ({ {destructive && ( )} + {showCopyIcon && ( +
+ +
+ )} { unit && (
diff --git a/web/app/components/base/linked-apps-panel/index.stories.tsx b/web/app/components/base/linked-apps-panel/index.stories.tsx new file mode 100644 index 0000000000..da8abb0677 --- /dev/null +++ b/web/app/components/base/linked-apps-panel/index.stories.tsx @@ -0,0 +1,73 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import LinkedAppsPanel from '.' +import type { RelatedApp } from '@/models/datasets' +import { AppModeEnum } from '@/types/app' + +const mockRelatedApps: RelatedApp[] = [ + { + id: 'app-cx', + name: 'Customer Support Assistant', + mode: AppModeEnum.CHAT, + icon_type: 'emoji', + icon: '\u{1F4AC}', + icon_background: '#EEF2FF', + icon_url: '', + }, + { + id: 'app-ops', + name: 'Ops Workflow Orchestrator', + mode: AppModeEnum.WORKFLOW, + icon_type: 'emoji', + icon: '\u{1F6E0}\u{FE0F}', + icon_background: '#ECFDF3', + icon_url: '', + }, + { + id: 'app-research', + name: 'Research Synthesizer', + mode: AppModeEnum.ADVANCED_CHAT, + icon_type: 'emoji', + icon: '\u{1F9E0}', + icon_background: '#FDF2FA', + icon_url: '', + }, +] + +const meta = { + title: 'Base/Feedback/LinkedAppsPanel', + component: LinkedAppsPanel, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Shows a curated list of related applications, pairing each app icon with quick navigation links.', + }, + }, + }, + args: { + relatedApps: mockRelatedApps, + isMobile: false, + }, + argTypes: { + isMobile: { + control: 'boolean', + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Desktop: Story = {} + +export const Mobile: Story = { + args: { + isMobile: true, + }, + parameters: { + viewport: { + defaultViewport: 'mobile2', + }, + }, +} diff --git a/web/app/components/base/linked-apps-panel/index.tsx b/web/app/components/base/linked-apps-panel/index.tsx index c3c3f5b46c..561bd49c2a 100644 --- a/web/app/components/base/linked-apps-panel/index.tsx +++ b/web/app/components/base/linked-apps-panel/index.tsx @@ -6,6 +6,7 @@ import { RiArrowRightUpLine } from '@remixicon/react' import cn from '@/utils/classnames' import AppIcon from '@/app/components/base/app-icon' import type { RelatedApp } from '@/models/datasets' +import { AppModeEnum } from '@/types/app' type ILikedItemProps = { appStatus?: boolean @@ -14,11 +15,11 @@ type ILikedItemProps = { } const appTypeMap = { - 'chat': 'Chatbot', - 'completion': 'Completion', - 'agent-chat': 'Agent', - 'advanced-chat': 'Chatflow', - 'workflow': 'Workflow', + [AppModeEnum.CHAT]: 'Chatbot', + [AppModeEnum.COMPLETION]: 'Completion', + [AppModeEnum.AGENT_CHAT]: 'Agent', + [AppModeEnum.ADVANCED_CHAT]: 'Chatflow', + [AppModeEnum.WORKFLOW]: 'Workflow', } const LikedItem = ({ diff --git a/web/app/components/base/list-empty/index.stories.tsx b/web/app/components/base/list-empty/index.stories.tsx new file mode 100644 index 0000000000..36c0e3c7a7 --- /dev/null +++ b/web/app/components/base/list-empty/index.stories.tsx @@ -0,0 +1,49 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import ListEmpty from '.' + +const meta = { + title: 'Base/Data Display/ListEmpty', + component: ListEmpty, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Large empty state card used in panels and drawers to hint at the next action for the user.', + }, + }, + }, + args: { + title: 'No items yet', + description: ( +

+ Add your first entry to see it appear here. Empty states help users discover what happens next. +

+ ), + }, + argTypes: { + description: { control: false }, + icon: { control: false }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} + +export const WithCustomIcon: Story = { + args: { + title: 'Connect a data source', + description: ( +

+ Choose a database, knowledge base, or upload documents to get started with retrieval. +

+ ), + icon: ( +
+ {'\u{26A1}\u{FE0F}'} +
+ ), + }, +} diff --git a/web/app/components/base/loading/index.stories.tsx b/web/app/components/base/loading/index.stories.tsx new file mode 100644 index 0000000000..f22f87516c --- /dev/null +++ b/web/app/components/base/loading/index.stories.tsx @@ -0,0 +1,52 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import Loading from '.' + +const meta = { + title: 'Base/Feedback/Loading', + component: Loading, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Spinner used while fetching data (`area`) or bootstrapping the full application shell (`app`).', + }, + }, + }, + argTypes: { + type: { + control: 'radio', + options: ['area', 'app'], + }, + }, + args: { + type: 'area', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const LoadingPreview = ({ type }: { type: 'area' | 'app' }) => { + const containerHeight = type === 'app' ? 'h-48' : 'h-20' + const title = type === 'app' ? 'App loading state' : 'Inline loading state' + + return ( +
+ {title} +
+ +
+
+ ) +} + +export const AreaSpinner: Story = { + render: () => , +} + +export const AppSpinner: Story = { + render: () => , +} diff --git a/web/app/components/base/logo/index.stories.tsx b/web/app/components/base/logo/index.stories.tsx new file mode 100644 index 0000000000..01464b8c13 --- /dev/null +++ b/web/app/components/base/logo/index.stories.tsx @@ -0,0 +1,82 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { ThemeProvider } from 'next-themes' +import type { ReactNode } from 'react' +import DifyLogo from './dify-logo' +import LogoSite from './logo-site' +import LogoEmbeddedChatHeader from './logo-embedded-chat-header' +import LogoEmbeddedChatAvatar from './logo-embedded-chat-avatar' + +const meta = { + title: 'Base/General/Logo', + component: DifyLogo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Brand assets rendered in different contexts. DifyLogo adapts to the active theme while other variants target specific surfaces.', + }, + }, + }, + args: { + size: 'medium', + style: 'default', + }, + argTypes: { + size: { + control: 'radio', + options: ['large', 'medium', 'small'], + }, + style: { + control: 'radio', + options: ['default', 'monochromeWhite'], + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +const ThemePreview = ({ theme, children }: { theme: 'light' | 'dark'; children: ReactNode }) => { + return ( + +
+ {children} +
+
+ ) +} + +export const Playground: Story = { + render: ({ size, style }) => { + return ( + +
+
+ Primary logo +
+ + {`size="${size}" | style="${style}"`} +
+
+
+
+ Site favicon + +
+
+ Embedded header + +
+
+ Embedded avatar + +
+
+
+
+ ) + }, +} diff --git a/web/app/components/base/markdown-blocks/code-block.stories.tsx b/web/app/components/base/markdown-blocks/code-block.stories.tsx new file mode 100644 index 0000000000..98473bdf57 --- /dev/null +++ b/web/app/components/base/markdown-blocks/code-block.stories.tsx @@ -0,0 +1,70 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import CodeBlock from './code-block' + +const SAMPLE_CODE = `const greet = (name: string) => { + return \`Hello, \${name}\` +} + +console.log(greet('Dify'))` + +const CodeBlockDemo = ({ + language = 'typescript', +}: { + language?: string +}) => { + return ( +
+
Code block
+ + {SAMPLE_CODE} + +
+ ) +} + +const meta = { + title: 'Base/Data Display/CodeBlock', + component: CodeBlockDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Syntax highlighted code block with copy button and SVG toggle support.', + }, + }, + }, + argTypes: { + language: { + control: 'radio', + options: ['typescript', 'json', 'mermaid'], + }, + }, + args: { + language: 'typescript', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const Mermaid: Story = { + args: { + language: 'mermaid', + }, + render: ({ language }) => ( +
+ + {`graph TD + Start --> Decision{User message?} + Decision -->|Tool| ToolCall[Call web search] + Decision -->|Respond| Answer[Compose draft] +`} + +
+ ), +} diff --git a/web/app/components/base/markdown-blocks/index.ts b/web/app/components/base/markdown-blocks/index.ts index ba68b4e8b1..ab6be2e9e7 100644 --- a/web/app/components/base/markdown-blocks/index.ts +++ b/web/app/components/base/markdown-blocks/index.ts @@ -5,9 +5,11 @@ export { default as AudioBlock } from './audio-block' export { default as CodeBlock } from './code-block' +export * from './plugin-img' +export * from './plugin-paragraph' export { default as Img } from './img' -export { default as Link } from './link' export { default as Paragraph } from './paragraph' +export { default as Link } from './link' export { default as PreCode } from './pre-code' export { default as ScriptBlock } from './script-block' export { default as VideoBlock } from './video-block' diff --git a/web/app/components/base/markdown-blocks/plugin-img.tsx b/web/app/components/base/markdown-blocks/plugin-img.tsx new file mode 100644 index 0000000000..ed1ee8fa0b --- /dev/null +++ b/web/app/components/base/markdown-blocks/plugin-img.tsx @@ -0,0 +1,48 @@ +/** + * @fileoverview Img component for rendering tags in Markdown. + * Extracted from the main markdown renderer for modularity. + * Uses the ImageGallery component to display images. + */ +import React, { useEffect, useMemo, useState } from 'react' +import ImageGallery from '@/app/components/base/image-gallery' +import { getMarkdownImageURL } from './utils' +import { usePluginReadmeAsset } from '@/service/use-plugins' +import type { SimplePluginInfo } from '../markdown/react-markdown-wrapper' + +type ImgProps = { + src: string + pluginInfo?: SimplePluginInfo +} + +export const PluginImg: React.FC = ({ src, pluginInfo }) => { + const { pluginUniqueIdentifier, pluginId } = pluginInfo || {} + const { data: assetData } = usePluginReadmeAsset({ plugin_unique_identifier: pluginUniqueIdentifier, file_name: src }) + const [blobUrl, setBlobUrl] = useState() + + useEffect(() => { + if (!assetData) { + setBlobUrl(undefined) + return + } + + const objectUrl = URL.createObjectURL(assetData) + setBlobUrl(objectUrl) + + return () => { + URL.revokeObjectURL(objectUrl) + } + }, [assetData]) + + const imageUrl = useMemo(() => { + if (blobUrl) + return blobUrl + + return getMarkdownImageURL(src, pluginId) + }, [blobUrl, pluginId, src]) + + return ( +
+ +
+ ) +} diff --git a/web/app/components/base/markdown-blocks/plugin-paragraph.tsx b/web/app/components/base/markdown-blocks/plugin-paragraph.tsx new file mode 100644 index 0000000000..ae1e2d7101 --- /dev/null +++ b/web/app/components/base/markdown-blocks/plugin-paragraph.tsx @@ -0,0 +1,69 @@ +/** + * @fileoverview Paragraph component for rendering

tags in Markdown. + * Extracted from the main markdown renderer for modularity. + * Handles special rendering for paragraphs that directly contain an image. + */ +import ImageGallery from '@/app/components/base/image-gallery' +import { usePluginReadmeAsset } from '@/service/use-plugins' +import React, { useEffect, useMemo, useState } from 'react' +import type { SimplePluginInfo } from '../markdown/react-markdown-wrapper' +import { getMarkdownImageURL } from './utils' + +type PluginParagraphProps = { + pluginInfo?: SimplePluginInfo + node?: any + children?: React.ReactNode +} + +export const PluginParagraph: React.FC = ({ pluginInfo, node, children }) => { + const { pluginUniqueIdentifier, pluginId } = pluginInfo || {} + const childrenNode = node?.children as Array | undefined + const firstChild = childrenNode?.[0] + const isImageParagraph = firstChild?.tagName === 'img' + const imageSrc = isImageParagraph ? firstChild?.properties?.src : undefined + + const { data: assetData } = usePluginReadmeAsset({ + plugin_unique_identifier: pluginUniqueIdentifier, + file_name: isImageParagraph && imageSrc ? imageSrc : '', + }) + + const [blobUrl, setBlobUrl] = useState() + + useEffect(() => { + if (!assetData) { + setBlobUrl(undefined) + return + } + + const objectUrl = URL.createObjectURL(assetData) + setBlobUrl(objectUrl) + + return () => { + URL.revokeObjectURL(objectUrl) + } + }, [assetData]) + + const imageUrl = useMemo(() => { + if (blobUrl) + return blobUrl + + if (isImageParagraph && imageSrc) + return getMarkdownImageURL(imageSrc, pluginId) + + return '' + }, [blobUrl, imageSrc, isImageParagraph, pluginId]) + + if (isImageParagraph) { + const remainingChildren = Array.isArray(children) && children.length > 1 ? children.slice(1) : undefined + + return ( +

+ + {remainingChildren && ( +
{remainingChildren}
+ )} +
+ ) + } + return

{children}

+} diff --git a/web/app/components/base/markdown-blocks/think-block.stories.tsx b/web/app/components/base/markdown-blocks/think-block.stories.tsx new file mode 100644 index 0000000000..571959259a --- /dev/null +++ b/web/app/components/base/markdown-blocks/think-block.stories.tsx @@ -0,0 +1,78 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import ThinkBlock from './think-block' +import { ChatContextProvider } from '@/app/components/base/chat/chat/context' + +const THOUGHT_TEXT = ` +Gather docs from knowledge base. +Score snippets against query. +[ENDTHINKFLAG] +` + +const ThinkBlockDemo = ({ + responding = false, +}: { + responding?: boolean +}) => { + const [isResponding, setIsResponding] = useState(responding) + + return ( + +
+
+ Think block + +
+ +
+            {THOUGHT_TEXT}
+          
+
+
+
+ ) +} + +const meta = { + title: 'Base/Data Display/ThinkBlock', + component: ThinkBlockDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Expandable chain-of-thought block used in chat responses. Toggles between “thinking” and completed states.', + }, + }, + }, + argTypes: { + responding: { control: 'boolean' }, + }, + args: { + responding: false, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/markdown-blocks/think-block.tsx b/web/app/components/base/markdown-blocks/think-block.tsx index a3b0561677..9c43578e4c 100644 --- a/web/app/components/base/markdown-blocks/think-block.tsx +++ b/web/app/components/base/markdown-blocks/think-block.tsx @@ -1,6 +1,7 @@ import React, { useEffect, useRef, useState } from 'react' import { useTranslation } from 'react-i18next' import { useChatContext } from '../chat/chat/context' +import cn from '@/utils/classnames' const hasEndThink = (children: any): boolean => { if (typeof children === 'string') @@ -40,7 +41,7 @@ const useThinkTimer = (children: any) => { const [startTime] = useState(() => Date.now()) const [elapsedTime, setElapsedTime] = useState(0) const [isComplete, setIsComplete] = useState(false) - const timerRef = useRef() + const timerRef = useRef(null) useEffect(() => { if (isComplete) return @@ -63,16 +64,26 @@ const useThinkTimer = (children: any) => { return { elapsedTime, isComplete } } -const ThinkBlock = ({ children, ...props }: React.ComponentProps<'details'>) => { +type ThinkBlockProps = React.ComponentProps<'details'> & { + 'data-think'?: boolean +} + +const ThinkBlock = ({ children, ...props }: ThinkBlockProps) => { const { elapsedTime, isComplete } = useThinkTimer(children) const displayContent = removeEndThink(children) const { t } = useTranslation() + const { 'data-think': isThink = false, className, open, ...rest } = props - if (!(props['data-think'] ?? false)) + if (!isThink) return (
{children}
) return ( -
+
{ const validPrefixes = ['http:', 'https:', '//', 'mailto:'] if (ALLOW_UNSAFE_DATA_SCHEME) validPrefixes.push('data:') return validPrefixes.some(prefix => url.startsWith(prefix)) } + +export const getMarkdownImageURL = (url: string, pathname?: string) => { + const regex = /(^\.\/_assets|^_assets)/ + if (regex.test(url)) + return `${MARKETPLACE_API_PREFIX}${MARKETPLACE_API_PREFIX.endsWith('/') ? '' : '/'}plugins/${pathname ?? ''}${url.replace(regex, '/_assets')}` + return url +} diff --git a/web/app/components/base/markdown/index.stories.tsx b/web/app/components/base/markdown/index.stories.tsx new file mode 100644 index 0000000000..8c940e01cf --- /dev/null +++ b/web/app/components/base/markdown/index.stories.tsx @@ -0,0 +1,88 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import { Markdown } from '.' + +const SAMPLE_MD = ` +# Product Update + +Our agent now supports **tool-runs** with structured outputs. + +## Highlights +- Faster reasoning with \\(O(n \\log n)\\) planning. +- Inline chain-of-thought: + +
+Thinking aloud + +Check cached metrics first. +If missing, fetch raw warehouse data. +[ENDTHINKFLAG] + +
+ +## Mermaid Diagram +\`\`\`mermaid +graph TD + Start[User Message] --> Parse{Detect Intent?} + Parse -->|Tool| ToolCall[Call search tool] + Parse -->|Answer| Respond[Stream response] + ToolCall --> Respond +\`\`\` + +## Code Example +\`\`\`typescript +const reply = await client.chat({ + message: 'Summarise weekly metrics.', + tags: ['analytics'], +}) +\`\`\` +` + +const MarkdownDemo = ({ + compact = false, +}: { + compact?: boolean +}) => { + const [content] = useState(SAMPLE_MD.trim()) + + return ( +
+
Markdown renderer
+ +
+ ) +} + +const meta = { + title: 'Base/Data Display/Markdown', + component: MarkdownDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Markdown wrapper with GitHub-flavored markdown, Mermaid diagrams, math, and custom blocks (details, audio, etc.).', + }, + }, + }, + argTypes: { + compact: { control: 'boolean' }, + }, + args: { + compact: false, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const Compact: Story = { + args: { + compact: true, + }, +} diff --git a/web/app/components/base/markdown/index.tsx b/web/app/components/base/markdown/index.tsx index 19f39d8aaa..bb49fe1b14 100644 --- a/web/app/components/base/markdown/index.tsx +++ b/web/app/components/base/markdown/index.tsx @@ -3,7 +3,7 @@ import 'katex/dist/katex.min.css' import { flow } from 'lodash-es' import cn from '@/utils/classnames' import { preprocessLaTeX, preprocessThinkTag } from './markdown-utils' -import type { ReactMarkdownWrapperProps } from './react-markdown-wrapper' +import type { ReactMarkdownWrapperProps, SimplePluginInfo } from './react-markdown-wrapper' const ReactMarkdown = dynamic(() => import('./react-markdown-wrapper').then(mod => mod.ReactMarkdownWrapper), { ssr: false }) @@ -17,10 +17,11 @@ const ReactMarkdown = dynamic(() => import('./react-markdown-wrapper').then(mod export type MarkdownProps = { content: string className?: string + pluginInfo?: SimplePluginInfo } & Pick export const Markdown = (props: MarkdownProps) => { - const { customComponents = {} } = props + const { customComponents = {}, pluginInfo } = props const latexContent = flow([ preprocessThinkTag, preprocessLaTeX, @@ -28,7 +29,7 @@ export const Markdown = (props: MarkdownProps) => { return (
- +
) } diff --git a/web/app/components/base/markdown/markdown-utils.ts b/web/app/components/base/markdown/markdown-utils.ts index 0089bef0ac..d02f98b51b 100644 --- a/web/app/components/base/markdown/markdown-utils.ts +++ b/web/app/components/base/markdown/markdown-utils.ts @@ -32,8 +32,8 @@ export const preprocessLaTeX = (content: string) => { } export const preprocessThinkTag = (content: string) => { - const thinkOpenTagRegex = /(\n)+/g - const thinkCloseTagRegex = /\n<\/think>/g + const thinkOpenTagRegex = /(\s*)+/g + const thinkCloseTagRegex = /(\s*<\/think>)+/g return flow([ (str: string) => str.replace(thinkOpenTagRegex, '
\n'), (str: string) => str.replace(thinkCloseTagRegex, '\n[ENDTHINKFLAG]
'), diff --git a/web/app/components/base/markdown/react-markdown-wrapper.tsx b/web/app/components/base/markdown/react-markdown-wrapper.tsx index 054b5f66cb..22964ec04f 100644 --- a/web/app/components/base/markdown/react-markdown-wrapper.tsx +++ b/web/app/components/base/markdown/react-markdown-wrapper.tsx @@ -1,46 +1,43 @@ +import { AudioBlock, Img, Link, MarkdownButton, MarkdownForm, Paragraph, PluginImg, PluginParagraph, ScriptBlock, ThinkBlock, VideoBlock } from '@/app/components/base/markdown-blocks' +import { ENABLE_SINGLE_DOLLAR_LATEX } from '@/config' +import dynamic from 'next/dynamic' +import type { FC } from 'react' import ReactMarkdown from 'react-markdown' -import RemarkMath from 'remark-math' -import RemarkBreaks from 'remark-breaks' import RehypeKatex from 'rehype-katex' -import RemarkGfm from 'remark-gfm' import RehypeRaw from 'rehype-raw' -import AudioBlock from '@/app/components/base/markdown-blocks/audio-block' -import Img from '@/app/components/base/markdown-blocks/img' -import Link from '@/app/components/base/markdown-blocks/link' -import MarkdownButton from '@/app/components/base/markdown-blocks/button' -import MarkdownForm from '@/app/components/base/markdown-blocks/form' -import Paragraph from '@/app/components/base/markdown-blocks/paragraph' -import ScriptBlock from '@/app/components/base/markdown-blocks/script-block' -import ThinkBlock from '@/app/components/base/markdown-blocks/think-block' -import VideoBlock from '@/app/components/base/markdown-blocks/video-block' +import RemarkBreaks from 'remark-breaks' +import RemarkGfm from 'remark-gfm' +import RemarkMath from 'remark-math' import { customUrlTransform } from './markdown-utils' -import type { FC } from 'react' - -import dynamic from 'next/dynamic' - const CodeBlock = dynamic(() => import('@/app/components/base/markdown-blocks/code-block'), { ssr: false }) +export type SimplePluginInfo = { + pluginUniqueIdentifier: string + pluginId: string +} + export type ReactMarkdownWrapperProps = { latexContent: any customDisallowedElements?: string[] customComponents?: Record> + pluginInfo?: SimplePluginInfo } export const ReactMarkdownWrapper: FC = (props) => { - const { customComponents, latexContent } = props + const { customComponents, latexContent, pluginInfo } = props return ( { return (tree: any) => { const iterate = (node: any) => { @@ -63,11 +60,11 @@ export const ReactMarkdownWrapper: FC = (props) => { disallowedElements={['iframe', 'head', 'html', 'meta', 'link', 'style', 'body', ...(props.customDisallowedElements || [])]} components={{ code: CodeBlock, - img: Img, + img: (props: any) => pluginInfo ? : , video: VideoBlock, audio: AudioBlock, a: Link, - p: Paragraph, + p: (props: any) => pluginInfo ? : , button: MarkdownButton, form: MarkdownForm, script: ScriptBlock as any, diff --git a/web/app/components/base/mermaid/index.stories.tsx b/web/app/components/base/mermaid/index.stories.tsx new file mode 100644 index 0000000000..73030d7905 --- /dev/null +++ b/web/app/components/base/mermaid/index.stories.tsx @@ -0,0 +1,64 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import Flowchart from '.' + +const SAMPLE = ` +flowchart LR + A[User Message] --> B{Agent decides} + B -->|Needs tool| C[Search Tool] + C --> D[Combine result] + B -->|Direct answer| D + D --> E[Send response] +` + +const MermaidDemo = ({ + theme = 'light', +}: { + theme?: 'light' | 'dark' +}) => { + const [currentTheme, setCurrentTheme] = useState<'light' | 'dark'>(theme) + + return ( +
+
+ Mermaid diagram + +
+ +
+ ) +} + +const meta = { + title: 'Base/Data Display/Mermaid', + component: MermaidDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Mermaid renderer with custom theme toggle and caching. Useful for visualizing agent flows.', + }, + }, + }, + argTypes: { + theme: { + control: 'inline-radio', + options: ['light', 'dark'], + }, + }, + args: { + theme: 'light', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/message-log-modal/index.stories.tsx b/web/app/components/base/message-log-modal/index.stories.tsx new file mode 100644 index 0000000000..4173a85ebc --- /dev/null +++ b/web/app/components/base/message-log-modal/index.stories.tsx @@ -0,0 +1,188 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect } from 'react' +import MessageLogModal from '.' +import type { IChatItem } from '@/app/components/base/chat/chat/type' +import { useStore } from '@/app/components/app/store' +import type { WorkflowRunDetailResponse } from '@/models/log' +import type { NodeTracing, NodeTracingListResponse } from '@/types/workflow' +import { BlockEnum } from '@/app/components/workflow/types' +import { WorkflowContextProvider } from '@/app/components/workflow/context' + +const SAMPLE_APP_DETAIL = { + id: 'app-demo-1', + name: 'Support Assistant', + mode: 'chat', +} as any + +const mockRunDetail: WorkflowRunDetailResponse = { + id: 'run-demo-1', + version: 'v1.0.0', + graph: { + nodes: [], + edges: [], + }, + inputs: JSON.stringify({ question: 'How do I reset my password?' }, null, 2), + inputs_truncated: false, + status: 'succeeded', + outputs: JSON.stringify({ answer: 'Follow the reset link we just emailed you.' }, null, 2), + outputs_truncated: false, + total_steps: 3, + created_by_role: 'account', + created_by_account: { + id: 'account-1', + name: 'Demo Admin', + email: 'demo@example.com', + }, + created_at: 1700000000, + finished_at: 1700000006, + elapsed_time: 5.2, + total_tokens: 864, +} + +const buildNode = (override: Partial): NodeTracing => ({ + id: 'node-start', + index: 0, + predecessor_node_id: '', + node_id: 'node-start', + node_type: BlockEnum.Start, + title: 'Start', + inputs: {}, + inputs_truncated: false, + process_data: {}, + process_data_truncated: false, + outputs: {}, + outputs_truncated: false, + status: 'succeeded', + metadata: { + iterator_length: 1, + iterator_index: 0, + loop_length: 1, + loop_index: 0, + }, + created_at: 1700000000, + created_by: { + id: 'account-1', + name: 'Demo Admin', + email: 'demo@example.com', + }, + finished_at: 1700000001, + elapsed_time: 1.1, + extras: {}, + ...override, +}) + +const mockTracingList: NodeTracingListResponse = { + data: [ + buildNode({}), + buildNode({ + id: 'node-answer', + node_id: 'node-answer', + node_type: BlockEnum.Answer, + title: 'Answer', + inputs: { prompt: 'How do I reset my password?' }, + outputs: { output: 'Follow the reset link we just emailed you.' }, + finished_at: 1700000005, + elapsed_time: 2.6, + }), + ], +} + +const mockCurrentLogItem: IChatItem = { + id: 'message-1', + content: 'Follow the reset link we just emailed you.', + isAnswer: true, + workflow_run_id: 'run-demo-1', +} + +const useMessageLogMocks = () => { + useEffect(() => { + const store = useStore.getState() + store.setAppDetail(SAMPLE_APP_DETAIL) + + const originalFetch = globalThis.fetch?.bind(globalThis) ?? null + + const handle = async (input: RequestInfo | URL, init?: RequestInit) => { + const url = typeof input === 'string' + ? input + : input instanceof URL + ? input.toString() + : input.url + + if (url.includes('/workflow-runs/run-demo-1/') && url.endsWith('/node-executions')) { + return new Response( + JSON.stringify(mockTracingList), + { headers: { 'Content-Type': 'application/json' }, status: 200 }, + ) + } + + if (url.endsWith('/workflow-runs/run-demo-1')) { + return new Response( + JSON.stringify(mockRunDetail), + { headers: { 'Content-Type': 'application/json' }, status: 200 }, + ) + } + + if (originalFetch) + return originalFetch(input, init) + + throw new Error(`Unmocked fetch call for ${url}`) + } + + globalThis.fetch = handle as typeof globalThis.fetch + + return () => { + globalThis.fetch = originalFetch || globalThis.fetch + useStore.getState().setAppDetail(undefined) + } + }, []) +} + +type MessageLogModalProps = React.ComponentProps + +const MessageLogPreview = (props: MessageLogModalProps) => { + useMessageLogMocks() + + return ( +
+ + + +
+ ) +} + +const meta = { + title: 'Base/Feedback/MessageLogModal', + component: MessageLogPreview, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Workflow run inspector presented alongside chat transcripts. This Storybook mock provides canned run details and tracing metadata.', + }, + }, + }, + args: { + defaultTab: 'DETAIL', + width: 960, + fixedWidth: true, + onCancel: () => { + console.log('Modal closed') + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const FixedPanel: Story = {} + +export const FloatingPanel: Story = { + args: { + fixedWidth: false, + }, +} diff --git a/web/app/components/base/modal-like-wrap/index.stories.tsx b/web/app/components/base/modal-like-wrap/index.stories.tsx new file mode 100644 index 0000000000..c7d66b8e6a --- /dev/null +++ b/web/app/components/base/modal-like-wrap/index.stories.tsx @@ -0,0 +1,131 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import ModalLikeWrap from '.' + +const meta = { + title: 'Base/Feedback/ModalLikeWrap', + component: ModalLikeWrap, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Compact “modal-like” card used in wizards. Provides header actions, optional back slot, and confirm/cancel buttons.', + }, + }, + }, + tags: ['autodocs'], + argTypes: { + title: { + control: 'text', + description: 'Header title text.', + }, + className: { + control: 'text', + description: 'Additional classes on the wrapper.', + }, + beforeHeader: { + control: false, + description: 'Slot rendered before the header (commonly a back link).', + }, + hideCloseBtn: { + control: 'boolean', + description: 'Hides the top-right close icon when true.', + }, + children: { + control: false, + }, + onClose: { + control: false, + }, + onConfirm: { + control: false, + }, + }, + args: { + title: 'Create dataset field', + hideCloseBtn: false, + onClose: () => console.log('close'), + onConfirm: () => console.log('confirm'), + children: null, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const BaseContent = () => ( +
+

+ Describe the new field your dataset should collect. Provide a clear label and optional helper text. +

+
+ Form inputs would be placed here in the real flow. +
+
+) + +export const Default: Story = { + render: args => ( + + + + ), + args: { + children: null, + }, +} + +export const WithBackLink: Story = { + render: args => ( + console.log('back')} + > + {'<'} + Back + + )} + > + + + ), + args: { + title: 'Select metadata type', + children: null, + }, + parameters: { + docs: { + description: { + story: 'Demonstrates feeding content into `beforeHeader` while hiding the close button.', + }, + }, + }, +} + +export const CustomWidth: Story = { + render: args => ( + + +
+ Tip: metadata keys may only include letters, numbers, and underscores. +
+
+ ), + args: { + title: 'Advanced configuration', + children: null, + }, + parameters: { + docs: { + description: { + story: 'Applies extra width and helper messaging to emulate configuration panels.', + }, + }, + }, +} diff --git a/web/app/components/base/modal/index.stories.tsx b/web/app/components/base/modal/index.stories.tsx new file mode 100644 index 0000000000..c0ea31eb42 --- /dev/null +++ b/web/app/components/base/modal/index.stories.tsx @@ -0,0 +1,133 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect, useState } from 'react' +import Modal from '.' + +const meta = { + title: 'Base/Feedback/Modal', + component: Modal, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Lightweight modal wrapper with optional header/description, close icon, and high-priority stacking for dropdown overlays.', + }, + }, + }, + tags: ['autodocs'], + argTypes: { + className: { + control: 'text', + description: 'Extra classes applied to the modal panel.', + }, + wrapperClassName: { + control: 'text', + description: 'Additional wrapper classes for the dialog.', + }, + isShow: { + control: 'boolean', + description: 'Controls whether the modal is visible.', + }, + title: { + control: 'text', + description: 'Heading displayed at the top of the modal.', + }, + description: { + control: 'text', + description: 'Secondary text beneath the title.', + }, + closable: { + control: 'boolean', + description: 'Whether the close icon should be shown.', + }, + overflowVisible: { + control: 'boolean', + description: 'Allows content to overflow the modal panel.', + }, + highPriority: { + control: 'boolean', + description: 'Lifts the modal above other high z-index elements like dropdowns.', + }, + onClose: { + control: false, + description: 'Callback invoked when the modal requests to close.', + }, + }, + args: { + isShow: false, + title: 'Create new API key', + description: 'Generate a scoped key for this workspace. You can revoke it at any time.', + closable: true, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +const ModalDemo = (props: React.ComponentProps) => { + const [open, setOpen] = useState(props.isShow) + + useEffect(() => { + setOpen(props.isShow) + }, [props.isShow]) + + return ( +
+ + + { + props.onClose?.() + setOpen(false) + }} + > +
+

+ Provide a descriptive name for this key so collaborators know its purpose. Restrict usage with scopes to limit access. +

+
+ Form fields and validation messaging would appear here. This placeholder keeps the story lightweight. +
+
+
+ + +
+
+
+ ) +} + +export const Default: Story = { + render: args => , +} + +export const HighPriorityOverflow: Story = { + render: args => , + args: { + highPriority: true, + overflowVisible: true, + description: 'Demonstrates the modal configured to sit above dropdowns while letting the body content overflow.', + className: 'max-w-[540px]', + }, + parameters: { + docs: { + description: { + story: 'Shows the modal with `highPriority` and `overflowVisible` enabled, useful when nested within complex surfaces.', + }, + }, + }, +} diff --git a/web/app/components/base/modal/index.tsx b/web/app/components/base/modal/index.tsx index 426953261e..f091717191 100644 --- a/web/app/components/base/modal/index.tsx +++ b/web/app/components/base/modal/index.tsx @@ -8,6 +8,7 @@ import { noop } from 'lodash-es' type IModal = { className?: string wrapperClassName?: string + containerClassName?: string isShow: boolean onClose?: () => void title?: React.ReactNode @@ -16,11 +17,14 @@ type IModal = { closable?: boolean overflowVisible?: boolean highPriority?: boolean // For modals that need to appear above dropdowns + overlayOpacity?: boolean // For semi-transparent overlay instead of default + clickOutsideNotClose?: boolean // Prevent closing when clicking outside modal } export default function Modal({ className, wrapperClassName, + containerClassName, isShow, onClose = noop, title, @@ -29,19 +33,21 @@ export default function Modal({ closable = false, overflowVisible = false, highPriority = false, + overlayOpacity = false, + clickOutsideNotClose = false, }: IModal) { return ( - +
-
{ @@ -49,7 +55,7 @@ export default function Modal({ e.stopPropagation() }} > -
+
console.log('Modal closed'), + onConfirm: () => console.log('Confirm pressed'), + onCancel: () => console.log('Cancel pressed'), + onExtraButtonClick: () => console.log('Extra button pressed'), + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +type ModalProps = React.ComponentProps + +const ModalDemo = (props: ModalProps) => { + const [open, setOpen] = useState(false) + + useEffect(() => { + if (props.disabled && open) + setOpen(false) + }, [props.disabled, open]) + + const { + onClose, + onConfirm, + onCancel, + onExtraButtonClick, + children, + ...rest + } = props + + const handleClose = () => { + onClose?.() + setOpen(false) + } + + const handleConfirm = () => { + onConfirm?.() + setOpen(false) + } + + const handleCancel = () => { + onCancel?.() + setOpen(false) + } + + const handleExtra = () => { + onExtraButtonClick?.() + } + + return ( +
+ + + {open && ( + +

+ Removing integrations immediately stops workflow automations related to this connection. + Make sure no scheduled jobs depend on this integration before proceeding. +

+
    +
  • All API credentials issued by this integration will be revoked.
  • +
  • Historical logs remain accessible for auditing.
  • +
  • You can re-enable the integration later with fresh credentials.
  • +
+
+ )} + /> + )} +
+ ) +} + +export const Default: Story = { + render: args => , +} + +export const WithExtraAction: Story = { + render: args => , + args: { + showExtraButton: true, + extraButtonVariant: 'secondary', + extraButtonText: 'Disable only', + footerSlot: ( + Last synced 5 minutes ago + ), + }, + parameters: { + docs: { + description: { + story: 'Illustrates the optional extra button and footer slot for advanced workflows.', + }, + }, + }, +} + +export const MediumSized: Story = { + render: args => , + args: { + size: 'md', + subTitle: 'Use the larger width to surface forms with more fields or supporting descriptions.', + bottomSlot: ( +
+ Need finer control? Configure automation rules in the integration settings page. +
+ ), + }, + parameters: { + docs: { + description: { + story: 'Shows the medium sized panel and a populated `bottomSlot` for supplemental messaging.', + }, + }, + }, +} diff --git a/web/app/components/base/modal/modal.tsx b/web/app/components/base/modal/modal.tsx index e6e9fb8804..49be4f2f54 100644 --- a/web/app/components/base/modal/modal.tsx +++ b/web/app/components/base/modal/modal.tsx @@ -1,13 +1,14 @@ -import { memo } from 'react' -import { useTranslation } from 'react-i18next' -import { RiCloseLine } from '@remixicon/react' +import type { ButtonProps } from '@/app/components/base/button' +import Button from '@/app/components/base/button' import { PortalToFollowElem, PortalToFollowElemContent, } from '@/app/components/base/portal-to-follow-elem' -import Button from '@/app/components/base/button' -import type { ButtonProps } from '@/app/components/base/button' import cn from '@/utils/classnames' +import { RiCloseLine } from '@remixicon/react' +import { noop } from 'lodash-es' +import { memo } from 'react' +import { useTranslation } from 'react-i18next' type ModalProps = { onClose?: () => void @@ -26,6 +27,9 @@ type ModalProps = { footerSlot?: React.ReactNode bottomSlot?: React.ReactNode disabled?: boolean + containerClassName?: string + wrapperClassName?: string + clickOutsideNotClose?: boolean } const Modal = ({ onClose, @@ -44,24 +48,28 @@ const Modal = ({ footerSlot, bottomSlot, disabled, + containerClassName, + wrapperClassName, + clickOutsideNotClose = false, }: ModalProps) => { const { t } = useTranslation() return (
e.stopPropagation()} > -
+
{title} { subTitle && ( @@ -79,10 +87,10 @@ const Modal = ({
{ children && ( -
{children}
+
{children}
) } -
+
{footerSlot}
@@ -117,7 +125,11 @@ const Modal = ({
- {bottomSlot} + {bottomSlot && ( +
+ {bottomSlot} +
+ )}
diff --git a/web/app/components/base/new-audio-button/index.stories.tsx b/web/app/components/base/new-audio-button/index.stories.tsx new file mode 100644 index 0000000000..c672392562 --- /dev/null +++ b/web/app/components/base/new-audio-button/index.stories.tsx @@ -0,0 +1,67 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect } from 'react' +import type { ComponentProps } from 'react' +import AudioBtn from '.' +import { ensureMockAudioManager } from '../../../../.storybook/utils/audio-player-manager.mock' + +ensureMockAudioManager() + +const StoryWrapper = (props: ComponentProps) => { + useEffect(() => { + ensureMockAudioManager() + }, []) + + return ( +
+ + Audio toggle using ActionButton styling +
+ ) +} + +const meta = { + title: 'Base/General/NewAudioButton', + component: AudioBtn, + tags: ['autodocs'], + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Updated audio playback trigger styled with `ActionButton`. Behaves like the legacy audio button but adopts the new button design system.', + }, + }, + nextjs: { + appDirectory: true, + navigation: { + pathname: '/apps/demo-app/text-to-audio', + params: { appId: 'demo-app' }, + }, + }, + }, + argTypes: { + id: { + control: 'text', + description: 'Message identifier used by the audio request.', + }, + value: { + control: 'text', + description: 'Prompt or response text that will be converted to speech.', + }, + voice: { + control: 'text', + description: 'Voice profile for the generated speech.', + }, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = { + render: args => , + args: { + id: 'message-1', + value: 'Listen to the latest assistant message.', + voice: 'alloy', + }, +} diff --git a/web/app/components/base/node-status/index.tsx b/web/app/components/base/node-status/index.tsx new file mode 100644 index 0000000000..a09737809d --- /dev/null +++ b/web/app/components/base/node-status/index.tsx @@ -0,0 +1,74 @@ +'use client' +import AlertTriangle from '@/app/components/base/icons/src/vender/solid/alertsAndFeedback/AlertTriangle' +import classNames from '@/utils/classnames' +import { RiErrorWarningFill } from '@remixicon/react' +import { type VariantProps, cva } from 'class-variance-authority' +import type { CSSProperties } from 'react' +import React from 'react' + +export enum NodeStatusEnum { + warning = 'warning', + error = 'error', +} + +const nodeStatusVariants = cva( + 'flex items-center gap-1 rounded-md px-2 py-1 system-xs-medium', + { + variants: { + status: { + [NodeStatusEnum.warning]: 'bg-state-warning-hover text-text-warning', + [NodeStatusEnum.error]: 'bg-state-destructive-hover text-text-destructive', + }, + }, + defaultVariants: { + status: NodeStatusEnum.warning, + }, + }, +) + +const StatusIconMap: Record = { + [NodeStatusEnum.warning]: { IconComponent: AlertTriangle, message: 'Warning' }, + [NodeStatusEnum.error]: { IconComponent: RiErrorWarningFill, message: 'Error' }, +} + +export type NodeStatusProps = { + message?: string + styleCss?: CSSProperties + iconClassName?: string +} & React.HTMLAttributes & VariantProps + +const NodeStatus = ({ + className, + status, + message, + styleCss, + iconClassName, + children, + ...props +}: NodeStatusProps) => { + const Icon = StatusIconMap[status ?? NodeStatusEnum.warning].IconComponent + const defaultMessage = StatusIconMap[status ?? NodeStatusEnum.warning].message + + return ( +
+ + {message ?? defaultMessage} + {children} +
+ ) +} + +NodeStatus.displayName = 'NodeStatus' + +export default React.memo(NodeStatus) diff --git a/web/app/components/base/notion-connector/index.stories.tsx b/web/app/components/base/notion-connector/index.stories.tsx new file mode 100644 index 0000000000..eb8b17df3f --- /dev/null +++ b/web/app/components/base/notion-connector/index.stories.tsx @@ -0,0 +1,26 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import NotionConnector from '.' + +const meta = { + title: 'Base/Other/NotionConnector', + component: NotionConnector, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Call-to-action card inviting users to connect a Notion workspace. Shows the product icon, copy, and primary button.', + }, + }, + }, + args: { + onSetting: () => { + console.log('Open Notion settings') + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/notion-icon/index.stories.tsx b/web/app/components/base/notion-icon/index.stories.tsx new file mode 100644 index 0000000000..5389a6f935 --- /dev/null +++ b/web/app/components/base/notion-icon/index.stories.tsx @@ -0,0 +1,129 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import NotionIcon from '.' + +const meta = { + title: 'Base/General/NotionIcon', + component: NotionIcon, + parameters: { + docs: { + description: { + component: 'Renders workspace and page icons returned from Notion APIs, falling back to text initials or the default document glyph.', + }, + }, + }, + tags: ['autodocs'], + args: { + type: 'workspace', + name: 'Knowledge Base', + src: 'https://cloud.dify.ai/logo/logo.svg', + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const WorkspaceIcon: Story = { + render: args => ( +
+ + Workspace icon pulled from a remote URL. +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +` + .trim(), + }, + }, + }, +} + +export const WorkspaceInitials: Story = { + render: args => ( +
+ + Fallback initial rendered when no icon URL is available. +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +` + .trim(), + }, + }, + }, +} + +export const PageEmoji: Story = { + render: args => ( +
+ + Page-level emoji icon returned by the API. +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +` + .trim(), + }, + }, + }, +} + +export const PageImage: Story = { + render: args => ( +
+ + Page icon resolved from an image URL. +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +` + .trim(), + }, + }, + }, +} + +export const DefaultIcon: Story = { + render: args => ( +
+ + When neither emoji nor URL is provided, the generic document icon is shown. +
+ ), + parameters: { + docs: { + source: { + language: 'tsx', + code: ` +` + .trim(), + }, + }, + }, +} diff --git a/web/app/components/base/notion-page-selector/base.tsx b/web/app/components/base/notion-page-selector/base.tsx index adf044c406..1f9ddeaebd 100644 --- a/web/app/components/base/notion-page-selector/base.tsx +++ b/web/app/components/base/notion-page-selector/base.tsx @@ -10,6 +10,7 @@ import { useInvalidPreImportNotionPages, usePreImportNotionPages } from '@/servi import Header from '../../datasets/create/website/base/header' import type { DataSourceCredential } from '../../header/account-setting/data-source-page-new/types' import Loading from '../loading' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' type NotionPageSelectorProps = { value?: string[] @@ -124,7 +125,7 @@ const NotionPageSelector = ({ }, [pagesMapAndSelectedPagesId, onPreview]) const handleConfigureNotion = useCallback(() => { - setShowAccountSettingModal({ payload: 'data-source' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.DATA_SOURCE }) }, [setShowAccountSettingModal]) if (isFetchingNotionPagesError) { diff --git a/web/app/components/base/notion-page-selector/index.stories.tsx b/web/app/components/base/notion-page-selector/index.stories.tsx new file mode 100644 index 0000000000..6fdee03adb --- /dev/null +++ b/web/app/components/base/notion-page-selector/index.stories.tsx @@ -0,0 +1,200 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect, useMemo, useState } from 'react' +import { CredentialTypeEnum } from '@/app/components/plugins/plugin-auth/types' +import { NotionPageSelector } from '.' +import type { DataSourceCredential } from '@/app/components/header/account-setting/data-source-page-new/types' +import type { NotionPage } from '@/models/common' + +const DATASET_ID = 'dataset-demo' +const CREDENTIALS: DataSourceCredential[] = [ + { + id: 'cred-1', + name: 'Marketing Workspace', + type: CredentialTypeEnum.OAUTH2, + is_default: true, + avatar_url: '', + credential: { + workspace_name: 'Marketing Workspace', + workspace_icon: null, + workspace_id: 'workspace-1', + }, + }, + { + id: 'cred-2', + name: 'Product Workspace', + type: CredentialTypeEnum.OAUTH2, + is_default: false, + avatar_url: '', + credential: { + workspace_name: 'Product Workspace', + workspace_icon: null, + workspace_id: 'workspace-2', + }, + }, +] + +const marketingPages = { + notion_info: [ + { + workspace_name: 'Marketing Workspace', + workspace_id: 'workspace-1', + workspace_icon: null, + pages: [ + { + page_icon: { type: 'emoji', emoji: '\u{1F4CB}', url: null }, + page_id: 'briefs', + page_name: 'Campaign Briefs', + parent_id: 'root', + type: 'page', + is_bound: false, + }, + { + page_icon: { type: 'emoji', emoji: '\u{1F4DD}', url: null }, + page_id: 'notes', + page_name: 'Meeting Notes', + parent_id: 'root', + type: 'page', + is_bound: true, + }, + { + page_icon: { type: 'emoji', emoji: '\u{1F30D}', url: null }, + page_id: 'localizations', + page_name: 'Localization Pipeline', + parent_id: 'briefs', + type: 'page', + is_bound: false, + }, + ], + }, + ], +} + +const productPages = { + notion_info: [ + { + workspace_name: 'Product Workspace', + workspace_id: 'workspace-2', + workspace_icon: null, + pages: [ + { + page_icon: { type: 'emoji', emoji: '\u{1F4A1}', url: null }, + page_id: 'ideas', + page_name: 'Idea Backlog', + parent_id: 'root', + type: 'page', + is_bound: false, + }, + { + page_icon: { type: 'emoji', emoji: '\u{1F9EA}', url: null }, + page_id: 'experiments', + page_name: 'Experiments', + parent_id: 'ideas', + type: 'page', + is_bound: false, + }, + ], + }, + ], +} + +type NotionApiResponse = typeof marketingPages +const emptyNotionResponse: NotionApiResponse = { notion_info: [] } + +const useMockNotionApi = () => { + const responseMap = useMemo(() => ({ + [`${DATASET_ID}:cred-1`]: marketingPages, + [`${DATASET_ID}:cred-2`]: productPages, + }) satisfies Record<`${typeof DATASET_ID}:${typeof CREDENTIALS[number]['id']}`, NotionApiResponse>, []) + + useEffect(() => { + const originalFetch = globalThis.fetch?.bind(globalThis) + + const handler = async (input: RequestInfo | URL, init?: RequestInit) => { + const url = typeof input === 'string' + ? input + : input instanceof URL + ? input.toString() + : input.url + + if (url.includes('/notion/pre-import/pages')) { + const parsed = new URL(url, globalThis.location.origin) + const datasetId = parsed.searchParams.get('dataset_id') || '' + const credentialId = parsed.searchParams.get('credential_id') || '' + let payload: NotionApiResponse = emptyNotionResponse + + if (datasetId === DATASET_ID) { + const credential = CREDENTIALS.find(item => item.id === credentialId) + if (credential) { + const mapKey = `${DATASET_ID}:${credential.id}` as keyof typeof responseMap + payload = responseMap[mapKey] + } + } + + return new Response( + JSON.stringify(payload), + { headers: { 'Content-Type': 'application/json' }, status: 200 }, + ) + } + + if (originalFetch) + return originalFetch(input, init) + + throw new Error(`Unmocked fetch call for ${url}`) + } + + globalThis.fetch = handler as typeof globalThis.fetch + + return () => { + if (originalFetch) + globalThis.fetch = originalFetch + } + }, [responseMap]) +} + +const NotionSelectorPreview = () => { + const [selectedPages, setSelectedPages] = useState([]) + const [credentialId, setCredentialId] = useState() + + useMockNotionApi() + + return ( +
+ page.page_id)} + onSelect={setSelectedPages} + onSelectCredential={setCredentialId} + canPreview + /> +
+
+ Debug state +
+

Active credential: {credentialId || 'None'}

+
+          {JSON.stringify(selectedPages, null, 2)}
+        
+
+
+ ) +} + +const meta = { + title: 'Base/Other/NotionPageSelector', + component: NotionSelectorPreview, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Credential-aware selector that fetches Notion pages and lets users choose which ones to sync.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/pagination/index.stories.tsx b/web/app/components/base/pagination/index.stories.tsx new file mode 100644 index 0000000000..4ad5488b96 --- /dev/null +++ b/web/app/components/base/pagination/index.stories.tsx @@ -0,0 +1,81 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useMemo, useState } from 'react' +import Pagination from '.' + +const TOTAL_ITEMS = 120 + +const PaginationDemo = ({ + initialPage = 0, + initialLimit = 10, +}: { + initialPage?: number + initialLimit?: number +}) => { + const [current, setCurrent] = useState(initialPage) + const [limit, setLimit] = useState(initialLimit) + + const pageSummary = useMemo(() => { + const start = current * limit + 1 + const end = Math.min((current + 1) * limit, TOTAL_ITEMS) + return `${start}-${end} of ${TOTAL_ITEMS}` + }, [current, limit]) + + return ( +
+
+ Log pagination + + {pageSummary} + +
+ { + setCurrent(0) + setLimit(nextLimit) + }} + /> +
+ ) +} + +const meta = { + title: 'Base/Navigation/Pagination', + component: PaginationDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Paginate long lists with optional per-page selector. Demonstrates the inline page jump input and quick limit toggles.', + }, + }, + }, + args: { + initialPage: 0, + initialLimit: 10, + }, + argTypes: { + initialPage: { + control: { type: 'number', min: 0, max: 9, step: 1 }, + }, + initialLimit: { + control: { type: 'radio' }, + options: [10, 25, 50], + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const StartAtMiddle: Story = { + args: { + initialPage: 4, + }, +} diff --git a/web/app/components/base/param-item/index.stories.tsx b/web/app/components/base/param-item/index.stories.tsx new file mode 100644 index 0000000000..a256b56dbf --- /dev/null +++ b/web/app/components/base/param-item/index.stories.tsx @@ -0,0 +1,121 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import ParamItem from '.' + +type ParamConfig = { + id: string + name: string + tip: string + value: number + min: number + max: number + step: number + allowToggle?: boolean +} + +const PARAMS: ParamConfig[] = [ + { + id: 'temperature', + name: 'Temperature', + tip: 'Controls randomness. Lower values make the model more deterministic, higher values encourage creativity.', + value: 0.7, + min: 0, + max: 2, + step: 0.1, + allowToggle: true, + }, + { + id: 'top_p', + name: 'Top P', + tip: 'Nucleus sampling keeps only the most probable tokens whose cumulative probability exceeds this threshold.', + value: 0.9, + min: 0, + max: 1, + step: 0.05, + }, + { + id: 'frequency_penalty', + name: 'Frequency Penalty', + tip: 'Discourages repeating tokens. Increase to reduce repetition.', + value: 0.2, + min: 0, + max: 1, + step: 0.05, + }, +] + +const ParamItemPlayground = () => { + const [state, setState] = useState>(() => { + return PARAMS.reduce((acc, item) => { + acc[item.id] = { value: item.value, enabled: true } + return acc + }, {} as Record) + }) + + const handleChange = (id: string, value: number) => { + setState(prev => ({ + ...prev, + [id]: { + ...prev[id], + value: Number.parseFloat(value.toFixed(3)), + }, + })) + } + + const handleToggle = (id: string, enabled: boolean) => { + setState(prev => ({ + ...prev, + [id]: { + ...prev[id], + enabled, + }, + })) + } + + return ( +
+
+ Generation parameters + + {JSON.stringify(state, null, 0)} + +
+ {PARAMS.map(param => ( + + ))} +
+ ) +} + +const meta = { + title: 'Base/Data Entry/ParamItem', + component: ParamItemPlayground, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Slider + numeric input pairing used for model parameter tuning. Supports optional enable toggles per parameter.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/popover/index.stories.tsx b/web/app/components/base/popover/index.stories.tsx new file mode 100644 index 0000000000..1977c89116 --- /dev/null +++ b/web/app/components/base/popover/index.stories.tsx @@ -0,0 +1,120 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import CustomPopover from '.' + +type PopoverContentProps = { + open?: boolean + onClose?: () => void + onClick?: () => void + title: string + description: string +} + +const PopoverContent = ({ title, description, onClose }: PopoverContentProps) => { + return ( +
+
+ {title} +
+

{description}

+ +
+ ) +} + +const Template = ({ + trigger = 'hover', + position = 'bottom', + manualClose, + disabled, +}: { + trigger?: 'click' | 'hover' + position?: 'bottom' | 'bl' | 'br' + manualClose?: boolean + disabled?: boolean +}) => { + const [hoverHint] = useState( + trigger === 'hover' + ? 'Hover over the badge to reveal quick tips.' + : 'Click the badge to open the contextual menu.', + ) + + return ( +
+

{hoverHint}

+
+ Popover trigger} + htmlContent={ + + } + /> +
+
+ ) +} + +const meta = { + title: 'Base/Feedback/Popover', + component: Template, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Headless UI popover wrapper supporting hover and click triggers. These examples highlight alignment controls and manual closing.', + }, + }, + }, + argTypes: { + trigger: { + control: 'radio', + options: ['hover', 'click'], + }, + position: { + control: 'radio', + options: ['bottom', 'bl', 'br'], + }, + manualClose: { control: 'boolean' }, + disabled: { control: 'boolean' }, + }, + args: { + trigger: 'hover', + position: 'bottom', + manualClose: false, + disabled: false, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const HoverPopover: Story = {} + +export const ClickPopover: Story = { + args: { + trigger: 'click', + position: 'br', + }, +} + +export const DisabledState: Story = { + args: { + disabled: true, + }, +} diff --git a/web/app/components/base/popover/index.tsx b/web/app/components/base/popover/index.tsx index 41df06f43a..2387737d02 100644 --- a/web/app/components/base/popover/index.tsx +++ b/web/app/components/base/popover/index.tsx @@ -1,5 +1,5 @@ import { Popover, PopoverButton, PopoverPanel, Transition } from '@headlessui/react' -import { Fragment, cloneElement, useRef } from 'react' +import { Fragment, cloneElement, isValidElement, useRef } from 'react' import cn from '@/utils/classnames' export type HtmlContentProps = { @@ -103,15 +103,17 @@ export default function CustomPopover({ }) } > - {cloneElement(htmlContent as React.ReactElement, { - open, - onClose: close, - ...(manualClose - ? { - onClick: close, - } - : {}), - })} + {isValidElement(htmlContent) + ? cloneElement(htmlContent as React.ReactElement, { + open, + onClose: close, + ...(manualClose + ? { + onClick: close, + } + : {}), + }) + : htmlContent}
)} diff --git a/web/app/components/base/portal-to-follow-elem/index.stories.tsx b/web/app/components/base/portal-to-follow-elem/index.stories.tsx new file mode 100644 index 0000000000..44c8e964ce --- /dev/null +++ b/web/app/components/base/portal-to-follow-elem/index.stories.tsx @@ -0,0 +1,103 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import { + PortalToFollowElem, + PortalToFollowElemContent, + PortalToFollowElemTrigger, +} from '.' + +const TooltipCard = ({ title, description }: { title: string; description: string }) => ( +
+
+ {title} +
+

{description}

+
+) + +const PortalDemo = ({ + placement = 'bottom', + triggerPopupSameWidth = false, +}: { + placement?: Parameters[0]['placement'] + triggerPopupSameWidth?: boolean +}) => { + const [controlledOpen, setControlledOpen] = useState(false) + + return ( +
+
+ + + Hover me + + + + + + + + + + + + + + +
+
+ ) +} + +const meta = { + title: 'Base/Feedback/PortalToFollowElem', + component: PortalDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Floating UI based portal that tracks trigger positioning. Demonstrates both hover-driven and controlled usage.', + }, + }, + }, + argTypes: { + placement: { + control: 'select', + options: ['top', 'top-start', 'top-end', 'bottom', 'bottom-start', 'bottom-end'], + }, + triggerPopupSameWidth: { control: 'boolean' }, + }, + args: { + placement: 'bottom', + triggerPopupSameWidth: false, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const SameWidthPanel: Story = { + args: { + triggerPopupSameWidth: true, + }, +} diff --git a/web/app/components/base/portal-to-follow-elem/index.tsx b/web/app/components/base/portal-to-follow-elem/index.tsx index 71ee251edd..e1192fe73b 100644 --- a/web/app/components/base/portal-to-follow-elem/index.tsx +++ b/web/app/components/base/portal-to-follow-elem/index.tsx @@ -125,7 +125,7 @@ export const PortalToFollowElemTrigger = ( children, asChild = false, ...props - }: React.HTMLProps & { ref?: React.RefObject, asChild?: boolean }, + }: React.HTMLProps & { ref?: React.RefObject, asChild?: boolean }, ) => { const context = usePortalToFollowElemContext() const childrenRef = (children as any).props?.ref @@ -133,12 +133,13 @@ export const PortalToFollowElemTrigger = ( // `asChild` allows the user to pass any element as the anchor if (asChild && React.isValidElement(children)) { + const childProps = (children.props ?? {}) as Record return React.cloneElement( children, context.getReferenceProps({ ref, ...props, - ...children.props, + ...childProps, 'data-state': context.open ? 'open' : 'closed', } as React.HTMLProps), ) @@ -164,7 +165,7 @@ export const PortalToFollowElemContent = ( style, ...props }: React.HTMLProps & { - ref?: React.RefObject; + ref?: React.RefObject; }, ) => { const context = usePortalToFollowElemContext() diff --git a/web/app/components/base/premium-badge/index.stories.tsx b/web/app/components/base/premium-badge/index.stories.tsx new file mode 100644 index 0000000000..c1f6ede869 --- /dev/null +++ b/web/app/components/base/premium-badge/index.stories.tsx @@ -0,0 +1,64 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import PremiumBadge from '.' + +const colors: Array['color']>> = ['blue', 'indigo', 'gray', 'orange'] + +const PremiumBadgeGallery = ({ + size = 'm', + allowHover = false, +}: { + size?: 's' | 'm' + allowHover?: boolean +}) => { + return ( +
+

Brand badge variants

+
+ {colors.map(color => ( +
+ + Premium + + {color} +
+ ))} +
+
+ ) +} + +const meta = { + title: 'Base/General/PremiumBadge', + component: PremiumBadgeGallery, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Gradient badge used for premium features and upsell prompts. Hover animations can be toggled per instance.', + }, + }, + }, + argTypes: { + size: { + control: 'radio', + options: ['s', 'm'], + }, + allowHover: { control: 'boolean' }, + }, + args: { + size: 'm', + allowHover: false, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const HoverEnabled: Story = { + args: { + allowHover: true, + }, +} diff --git a/web/app/components/base/progress-bar/progress-circle.stories.tsx b/web/app/components/base/progress-bar/progress-circle.stories.tsx new file mode 100644 index 0000000000..a6a21d2695 --- /dev/null +++ b/web/app/components/base/progress-bar/progress-circle.stories.tsx @@ -0,0 +1,89 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import ProgressCircle from './progress-circle' + +const ProgressCircleDemo = ({ + initialPercentage = 42, + size = 24, +}: { + initialPercentage?: number + size?: number +}) => { + const [percentage, setPercentage] = useState(initialPercentage) + + return ( +
+
+ Upload progress + + {percentage}% + +
+
+ + setPercentage(Number.parseInt(event.target.value, 10))} + className="h-2 w-full cursor-pointer appearance-none rounded-full bg-divider-subtle accent-primary-600" + /> +
+
+ +
+
+ ProgressCircle renders a deterministic SVG slice. Advance the slider to preview how the arc grows for upload indicators. +
+
+ ) +} + +const meta = { + title: 'Base/Feedback/ProgressCircle', + component: ProgressCircleDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Compact radial progress indicator wired to upload flows. The story provides a slider to scrub through percentages.', + }, + }, + }, + argTypes: { + initialPercentage: { + control: { type: 'range', min: 0, max: 100, step: 1 }, + }, + size: { + control: { type: 'number', min: 12, max: 48, step: 2 }, + }, + }, + args: { + initialPercentage: 42, + size: 24, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const NearComplete: Story = { + args: { + initialPercentage: 92, + }, +} diff --git a/web/app/components/base/prompt-editor/hooks.ts b/web/app/components/base/prompt-editor/hooks.ts index 87119f8b49..b3d2b22236 100644 --- a/web/app/components/base/prompt-editor/hooks.ts +++ b/web/app/components/base/prompt-editor/hooks.ts @@ -35,7 +35,7 @@ import { DELETE_QUERY_BLOCK_COMMAND } from './plugins/query-block' import type { CustomTextNode } from './plugins/custom-text/node' import { registerLexicalTextEntity } from './utils' -export type UseSelectOrDeleteHandler = (nodeKey: string, command?: LexicalCommand) => [RefObject, boolean] +export type UseSelectOrDeleteHandler = (nodeKey: string, command?: LexicalCommand) => [RefObject, boolean] export const useSelectOrDelete: UseSelectOrDeleteHandler = (nodeKey: string, command?: LexicalCommand) => { const ref = useRef(null) const [editor] = useLexicalComposerContext() @@ -110,7 +110,7 @@ export const useSelectOrDelete: UseSelectOrDeleteHandler = (nodeKey: string, com return [ref, isSelected] } -export type UseTriggerHandler = () => [RefObject, boolean, Dispatch>] +export type UseTriggerHandler = () => [RefObject, boolean, Dispatch>] export const useTrigger: UseTriggerHandler = () => { const triggerRef = useRef(null) const [open, setOpen] = useState(false) diff --git a/web/app/components/base/prompt-editor/index.stories.tsx b/web/app/components/base/prompt-editor/index.stories.tsx index 17b04e4af0..35058ac37d 100644 --- a/web/app/components/base/prompt-editor/index.stories.tsx +++ b/web/app/components/base/prompt-editor/index.stories.tsx @@ -25,7 +25,7 @@ const PromptEditorMock = ({ value, onChange, placeholder, editable, compact, cla } const meta = { - title: 'Base/PromptEditor', + title: 'Base/Data Entry/PromptEditor', component: PromptEditorMock, parameters: { layout: 'centered', diff --git a/web/app/components/base/prompt-editor/plugins/placeholder.tsx b/web/app/components/base/prompt-editor/plugins/placeholder.tsx index c2c2623992..187b574cea 100644 --- a/web/app/components/base/prompt-editor/plugins/placeholder.tsx +++ b/web/app/components/base/prompt-editor/plugins/placeholder.tsx @@ -1,4 +1,5 @@ import { memo } from 'react' +import type { ReactNode } from 'react' import { useTranslation } from 'react-i18next' import cn from '@/utils/classnames' @@ -8,7 +9,7 @@ const Placeholder = ({ className, }: { compact?: boolean - value?: string | JSX.Element + value?: ReactNode className?: string }) => { const { t } = useTranslation() diff --git a/web/app/components/base/prompt-editor/plugins/workflow-variable-block/component.tsx b/web/app/components/base/prompt-editor/plugins/workflow-variable-block/component.tsx index e165b93a66..2bd67d0ced 100644 --- a/web/app/components/base/prompt-editor/plugins/workflow-variable-block/component.tsx +++ b/web/app/components/base/prompt-editor/plugins/workflow-variable-block/component.tsx @@ -2,6 +2,7 @@ import { memo, useCallback, useEffect, + useMemo, useState, } from 'react' import { useTranslation } from 'react-i18next' @@ -18,7 +19,7 @@ import { DELETE_WORKFLOW_VARIABLE_BLOCK_COMMAND, UPDATE_WORKFLOW_NODES_MAP, } from './index' -import { isConversationVar, isENV, isRagVariableVar, isSystemVar } from '@/app/components/workflow/nodes/_base/components/variable/utils' +import { isConversationVar, isENV, isGlobalVar, isRagVariableVar, isSystemVar } from '@/app/components/workflow/nodes/_base/components/variable/utils' import Tooltip from '@/app/components/base/tooltip' import { isExceptionVariable } from '@/app/components/workflow/utils' import VarFullPathPanel from '@/app/components/workflow/nodes/_base/components/variable/var-full-path-panel' @@ -65,25 +66,33 @@ const WorkflowVariableBlockComponent = ({ )() const [localWorkflowNodesMap, setLocalWorkflowNodesMap] = useState(workflowNodesMap) const node = localWorkflowNodesMap![variables[isRagVar ? 1 : 0]] - const isEnv = isENV(variables) - const isChatVar = isConversationVar(variables) + const isException = isExceptionVariable(varName, node?.type) - let variableValid = true - if (isEnv) { - if (environmentVariables) - variableValid = environmentVariables.some(v => v.variable === `${variables?.[0] ?? ''}.${variables?.[1] ?? ''}`) - } - else if (isChatVar) { - if (conversationVariables) - variableValid = conversationVariables.some(v => v.variable === `${variables?.[0] ?? ''}.${variables?.[1] ?? ''}`) - } - else if (isRagVar) { - if (ragVariables) - variableValid = ragVariables.some(v => v.variable === `${variables?.[0] ?? ''}.${variables?.[1] ?? ''}.${variables?.[2] ?? ''}`) - } - else { - variableValid = !!node - } + const variableValid = useMemo(() => { + let variableValid = true + const isEnv = isENV(variables) + const isChatVar = isConversationVar(variables) + const isGlobal = isGlobalVar(variables) + if (isGlobal) + return true + + if (isEnv) { + if (environmentVariables) + variableValid = environmentVariables.some(v => v.variable === `${variables?.[0] ?? ''}.${variables?.[1] ?? ''}`) + } + else if (isChatVar) { + if (conversationVariables) + variableValid = conversationVariables.some(v => v.variable === `${variables?.[0] ?? ''}.${variables?.[1] ?? ''}`) + } + else if (isRagVar) { + if (ragVariables) + variableValid = ragVariables.some(v => v.variable === `${variables?.[0] ?? ''}.${variables?.[1] ?? ''}.${variables?.[2] ?? ''}`) + } + else { + variableValid = !!node + } + return variableValid + }, [variables, node, environmentVariables, conversationVariables, isRagVar, ragVariables]) const reactflow = useReactFlow() const store = useStoreApi() diff --git a/web/app/components/base/prompt-log-modal/index.stories.tsx b/web/app/components/base/prompt-log-modal/index.stories.tsx new file mode 100644 index 0000000000..55389874cd --- /dev/null +++ b/web/app/components/base/prompt-log-modal/index.stories.tsx @@ -0,0 +1,74 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect } from 'react' +import PromptLogModal from '.' +import { useStore } from '@/app/components/app/store' +import type { IChatItem } from '@/app/components/base/chat/chat/type' + +type PromptLogModalProps = React.ComponentProps + +const mockLogItem: IChatItem = { + id: 'message-1', + isAnswer: true, + content: 'Summarize our meeting notes about launch blockers.', + log: [ + { + role: 'system', + text: 'You are an assistant that extracts key launch blockers from the dialogue.', + }, + { + role: 'user', + text: 'Team discussed QA, marketing assets, and infra readiness. Highlight risks.', + }, + { + role: 'assistant', + text: 'Blocking items:\n1. QA needs staging data by Friday.\n2. Marketing awaiting final visuals.\n3. Infra rollout still missing approval.', + }, + ], +} + +const usePromptLogMocks = () => { + useEffect(() => { + useStore.getState().setCurrentLogItem(mockLogItem) + return () => { + useStore.getState().setCurrentLogItem(undefined) + } + }, []) +} + +const PromptLogPreview = (props: PromptLogModalProps) => { + usePromptLogMocks() + + return ( +
+ +
+ ) +} + +const meta = { + title: 'Base/Feedback/PromptLogModal', + component: PromptLogPreview, + parameters: { + layout: 'fullscreen', + docs: { + description: { + component: 'Shows the prompt and message transcript used for a chat completion, with copy-to-clipboard support for single prompts.', + }, + }, + }, + args: { + width: 960, + onCancel: () => { + console.log('Prompt log closed') + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/qrcode/index.stories.tsx b/web/app/components/base/qrcode/index.stories.tsx new file mode 100644 index 0000000000..312dc6a5a8 --- /dev/null +++ b/web/app/components/base/qrcode/index.stories.tsx @@ -0,0 +1,52 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import ShareQRCode from '.' + +const QRDemo = ({ + content = 'https://dify.ai', +}: { + content?: string +}) => { + return ( +
+

Share QR

+
+ Generated URL: + {content} +
+ +
+ ) +} + +const meta = { + title: 'Base/Data Display/QRCode', + component: QRDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Toggleable QR code generator for sharing app URLs. Clicking the trigger reveals the code with a download CTA.', + }, + }, + }, + argTypes: { + content: { + control: 'text', + }, + }, + args: { + content: 'https://dify.ai', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const DemoLink: Story = { + args: { + content: 'https://dify.ai/docs', + }, +} diff --git a/web/app/components/base/radio-card/index.stories.tsx b/web/app/components/base/radio-card/index.stories.tsx index e129cd7033..63dd1ad1ec 100644 --- a/web/app/components/base/radio-card/index.stories.tsx +++ b/web/app/components/base/radio-card/index.stories.tsx @@ -4,7 +4,7 @@ import { RiCloudLine, RiCpuLine, RiDatabase2Line, RiLightbulbLine, RiRocketLine, import RadioCard from '.' const meta = { - title: 'Base/RadioCard', + title: 'Base/Data Entry/RadioCard', component: RadioCard, parameters: { layout: 'centered', diff --git a/web/app/components/base/radio/index.stories.tsx b/web/app/components/base/radio/index.stories.tsx index 01f7263d74..699372097f 100644 --- a/web/app/components/base/radio/index.stories.tsx +++ b/web/app/components/base/radio/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import Radio from '.' const meta = { - title: 'Base/Radio', + title: 'Base/Data Entry/Radio', component: Radio, parameters: { layout: 'centered', diff --git a/web/app/components/base/search-input/index.stories.tsx b/web/app/components/base/search-input/index.stories.tsx index 99d60d52ff..6b2326322b 100644 --- a/web/app/components/base/search-input/index.stories.tsx +++ b/web/app/components/base/search-input/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import SearchInput from '.' const meta = { - title: 'Base/SearchInput', + title: 'Base/Data Entry/SearchInput', component: SearchInput, parameters: { layout: 'centered', diff --git a/web/app/components/base/search-input/index.tsx b/web/app/components/base/search-input/index.tsx index 3330b55330..abf1817e88 100644 --- a/web/app/components/base/search-input/index.tsx +++ b/web/app/components/base/search-input/index.tsx @@ -22,7 +22,7 @@ const SearchInput: FC = ({ const { t } = useTranslation() const [focus, setFocus] = useState(false) const isComposing = useRef(false) - const [internalValue, setInternalValue] = useState(value) + const [compositionValue, setCompositionValue] = useState('') return (
= ({ white && '!bg-white placeholder:!text-gray-400 hover:!bg-white group-hover:!bg-white', )} placeholder={placeholder || t('common.operation.search')!} - value={internalValue} + value={isComposing.current ? compositionValue : value} onChange={(e) => { - setInternalValue(e.target.value) - if (!isComposing.current) - onChange(e.target.value) + const newValue = e.target.value + if (isComposing.current) + setCompositionValue(newValue) + else + onChange(newValue) }} onCompositionStart={() => { isComposing.current = true + setCompositionValue(value) }} onCompositionEnd={(e) => { isComposing.current = false + setCompositionValue('') onChange(e.currentTarget.value) }} onFocus={() => setFocus(true)} @@ -64,7 +68,6 @@ const SearchInput: FC = ({ className='group/clear flex h-4 w-4 shrink-0 cursor-pointer items-center justify-center' onClick={() => { onChange('') - setInternalValue('') }} > diff --git a/web/app/components/base/segmented-control/index.stories.tsx b/web/app/components/base/segmented-control/index.stories.tsx new file mode 100644 index 0000000000..c83112bd54 --- /dev/null +++ b/web/app/components/base/segmented-control/index.stories.tsx @@ -0,0 +1,92 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { RiLineChartLine, RiListCheck2, RiRobot2Line } from '@remixicon/react' +import { useState } from 'react' +import { SegmentedControl } from '.' + +const SEGMENTS = [ + { value: 'overview', text: 'Overview', Icon: RiLineChartLine }, + { value: 'tasks', text: 'Tasks', Icon: RiListCheck2, count: 8 }, + { value: 'agents', text: 'Agents', Icon: RiRobot2Line }, +] + +const SegmentedControlDemo = ({ + initialValue = 'overview', + size = 'regular', + padding = 'with', + activeState = 'default', +}: { + initialValue?: string + size?: 'regular' | 'small' | 'large' + padding?: 'none' | 'with' + activeState?: 'default' | 'accent' | 'accentLight' +}) => { + const [value, setValue] = useState(initialValue) + + return ( +
+
+ Segmented control + + value="{value}" + +
+ +
+ ) +} + +const meta = { + title: 'Base/Data Entry/SegmentedControl', + component: SegmentedControlDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Multi-tab segmented control with optional icons and badge counts. Adjust sizing and accent states via controls.', + }, + }, + }, + argTypes: { + initialValue: { + control: 'radio', + options: SEGMENTS.map(segment => segment.value), + }, + size: { + control: 'inline-radio', + options: ['small', 'regular', 'large'], + }, + padding: { + control: 'inline-radio', + options: ['none', 'with'], + }, + activeState: { + control: 'inline-radio', + options: ['default', 'accent', 'accentLight'], + }, + }, + args: { + initialValue: 'overview', + size: 'regular', + padding: 'with', + activeState: 'default', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const AccentState: Story = { + args: { + activeState: 'accent', + }, +} diff --git a/web/app/components/base/select/custom.tsx b/web/app/components/base/select/custom.tsx index 444c975f7e..f9032658c3 100644 --- a/web/app/components/base/select/custom.tsx +++ b/web/app/components/base/select/custom.tsx @@ -58,6 +58,7 @@ const CustomSelect = ({ onOpenChange, placement, offset, + triggerPopupSameWidth = true, } = containerProps || {} const { className: triggerClassName, @@ -85,6 +86,7 @@ const CustomSelect = ({ offset={offset || 4} open={mergedOpen} onOpenChange={handleOpenChange} + triggerPopupSameWidth={triggerPopupSameWidth} > handleOpenChange(!mergedOpen)} diff --git a/web/app/components/base/select/index.stories.tsx b/web/app/components/base/select/index.stories.tsx index 2a107155a5..f1b46f2d55 100644 --- a/web/app/components/base/select/index.stories.tsx +++ b/web/app/components/base/select/index.stories.tsx @@ -4,7 +4,7 @@ import Select, { PortalSelect, SimpleSelect } from '.' import type { Item } from '.' const meta = { - title: 'Base/Select', + title: 'Base/Data Entry/Select', component: SimpleSelect, parameters: { layout: 'centered', diff --git a/web/app/components/base/select/index.tsx b/web/app/components/base/select/index.tsx index a1e8ac2724..1a096d7f93 100644 --- a/web/app/components/base/select/index.tsx +++ b/web/app/components/base/select/index.tsx @@ -26,12 +26,15 @@ const defaultItems = [ export type Item = { value: number | string name: string + isGroup?: boolean + disabled?: boolean + extra?: React.ReactNode } & Record export type ISelectProps = { className?: string wrapperClassName?: string - renderTrigger?: (value: Item | null) => React.JSX.Element | null + renderTrigger?: (value: Item | null, isOpen: boolean) => React.JSX.Element | null items?: Item[] defaultValue?: number | string disabled?: boolean @@ -70,14 +73,13 @@ const Select: FC = ({ const [open, setOpen] = useState(false) const [selectedItem, setSelectedItem] = useState(null) + // Ensure selectedItem is properly set when defaultValue or items change useEffect(() => { let defaultSelect = null - const existed = items.find((item: Item) => item.value === defaultValue) - if (existed) - defaultSelect = existed - + // Handle cases where defaultValue might be undefined, null, or empty string + defaultSelect = (defaultValue && items.find((item: Item) => item.value === defaultValue)) || null setSelectedItem(defaultSelect) - }, [defaultValue]) + }, [defaultValue, items]) const filteredItems: Item[] = query === '' @@ -193,14 +195,18 @@ const SimpleSelect: FC = ({ const [selectedItem, setSelectedItem] = useState(null) + // Enhanced: Preserve user selection, only reset when necessary useEffect(() => { - let defaultSelect = null - const existed = items.find((item: Item) => item.value === defaultValue) - if (existed) - defaultSelect = existed + // Only reset if no current selection or current selection is invalid + const isCurrentSelectionValid = selectedItem && items.some(item => item.value === selectedItem.value) - setSelectedItem(defaultSelect) - }, [defaultValue]) + if (!isCurrentSelectionValid) { + let defaultSelect = null + // Handle cases where defaultValue might be undefined, null, or empty string + defaultSelect = items.find((item: Item) => item.value === defaultValue) ?? null + setSelectedItem(defaultSelect) + } + }, [defaultValue, items, selectedItem]) const listboxRef = useRef(null) @@ -216,7 +222,7 @@ const SimpleSelect: FC = ({ > {({ open }) => (
- {renderTrigger && {renderTrigger(selectedItem)}} + {renderTrigger && {renderTrigger(selectedItem, open)}} {!renderTrigger && ( { onOpenChange?.(open) @@ -255,38 +261,47 @@ const SimpleSelect: FC = ({ {(!disabled) && ( - {items.map((item: Item) => ( - - {({ /* active, */ selected }) => ( - <> - {renderOption - ? renderOption({ item, selected }) - : (<> - {item.name} - {selected && !hideChecked && ( - - - )} - )} - - )} - - ))} + {items.map((item: Item) => + item.isGroup ? ( +
+ {item.name} +
+ ) : ( + + {({ /* active, */ selected }) => ( + <> + {renderOption + ? renderOption({ item, selected }) + : (<> + {item.name} + {selected && !hideChecked && ( + + + )} + )} + + )} + + ), + )}
)}
@@ -334,6 +349,7 @@ const PortalSelect: FC = ({ onOpenChange={setOpen} placement='bottom-start' offset={4} + triggerPopupSameWidth={true} > !readonly && setOpen(v => !v)} className='w-full'> {renderTrigger @@ -361,7 +377,7 @@ const PortalSelect: FC = ({
{items.map((item: Item) => (
= ({ {!hideChecked && item.value === value && ( )} + {item.extra}
))}
diff --git a/web/app/components/base/select/pure.tsx b/web/app/components/base/select/pure.tsx index cede31d2ba..3de8245025 100644 --- a/web/app/components/base/select/pure.tsx +++ b/web/app/components/base/select/pure.tsx @@ -1,5 +1,6 @@ import { useCallback, + useMemo, useState, } from 'react' import { useTranslation } from 'react-i18next' @@ -22,10 +23,8 @@ export type Option = { value: string } -export type PureSelectProps = { +type SharedPureSelectProps = { options: Option[] - value?: string - onChange?: (value: string) => void containerProps?: PortalToFollowElemOptions & { open?: boolean onOpenChange?: (open: boolean) => void @@ -38,22 +37,39 @@ export type PureSelectProps = { className?: string itemClassName?: string title?: string + titleClassName?: string }, placeholder?: string disabled?: boolean triggerPopupSameWidth?: boolean } -const PureSelect = ({ - options, - value, - onChange, - containerProps, - triggerProps, - popupProps, - placeholder, - disabled, - triggerPopupSameWidth, -}: PureSelectProps) => { + +type SingleSelectProps = { + multiple?: false + value?: string + onChange?: (value: string) => void +} + +type MultiSelectProps = { + multiple: true + value?: string[] + onChange?: (value: string[]) => void +} + +export type PureSelectProps = SharedPureSelectProps & (SingleSelectProps | MultiSelectProps) +const PureSelect = (props: PureSelectProps) => { + const { + options, + containerProps, + triggerProps, + popupProps, + placeholder, + disabled, + triggerPopupSameWidth, + multiple, + value, + onChange, + } = props const { t } = useTranslation() const { open, @@ -69,6 +85,7 @@ const PureSelect = ({ className: popupClassName, itemClassName: popupItemClassName, title: popupTitle, + titleClassName: popupTitleClassName, } = popupProps || {} const [localOpen, setLocalOpen] = useState(false) @@ -79,8 +96,13 @@ const PureSelect = ({ setLocalOpen(openValue) }, [onOpenChange]) - const selectedOption = options.find(option => option.value === value) - const triggerText = selectedOption?.label || placeholder || t('common.placeholder.select') + const triggerText = useMemo(() => { + const placeholderText = placeholder || t('common.placeholder.select') + if (multiple) + return value?.length ? t('common.dynamicSelect.selected', { count: value.length }) : placeholderText + + return options.find(option => option.value === value)?.label || placeholderText + }, [multiple, value, options, placeholder]) return (
{ popupTitle && ( -
+
{popupTitle}
) @@ -144,6 +169,14 @@ const PureSelect = ({ title={option.label} onClick={() => { if (disabled) return + if (multiple) { + const currentValues = value ?? [] + const nextValues = currentValues.includes(option.value) + ? currentValues.filter(valueItem => valueItem !== option.value) + : [...currentValues, option.value] + onChange?.(nextValues) + return + } onChange?.(option.value) handleOpenChange(false) }} @@ -152,7 +185,11 @@ const PureSelect = ({ {option.label}
{ - value === option.value && + ( + multiple + ? (value ?? []).includes(option.value) + : value === option.value + ) && }
)) diff --git a/web/app/components/base/simple-pie-chart/index.stories.tsx b/web/app/components/base/simple-pie-chart/index.stories.tsx new file mode 100644 index 0000000000..d08c8fa0ce --- /dev/null +++ b/web/app/components/base/simple-pie-chart/index.stories.tsx @@ -0,0 +1,89 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useMemo, useState } from 'react' +import SimplePieChart from '.' + +const PieChartPlayground = ({ + initialPercentage = 65, + fill = '#fdb022', + stroke = '#f79009', +}: { + initialPercentage?: number + fill?: string + stroke?: string +}) => { + const [percentage, setPercentage] = useState(initialPercentage) + + const label = useMemo(() => `${percentage}%`, [percentage]) + + return ( +
+
+ Conversion snapshot + + {label} + +
+
+ +
+ + setPercentage(Number.parseInt(event.target.value, 10))} + className="h-2 w-full cursor-pointer appearance-none rounded-full bg-divider-subtle accent-primary-600" + /> +
+
+
+ ) +} + +const meta = { + title: 'Base/Data Display/SimplePieChart', + component: PieChartPlayground, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Thin radial indicator built with ECharts. Use it for quick percentage snapshots inside cards.', + }, + }, + }, + argTypes: { + initialPercentage: { + control: { type: 'range', min: 0, max: 100, step: 1 }, + }, + fill: { control: 'color' }, + stroke: { control: 'color' }, + }, + args: { + initialPercentage: 65, + fill: '#fdb022', + stroke: '#f79009', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const BrandAccent: Story = { + args: { + fill: '#155EEF', + stroke: '#0040C1', + initialPercentage: 82, + }, +} diff --git a/web/app/components/base/skeleton/index.stories.tsx b/web/app/components/base/skeleton/index.stories.tsx new file mode 100644 index 0000000000..b5ea649b34 --- /dev/null +++ b/web/app/components/base/skeleton/index.stories.tsx @@ -0,0 +1,59 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { + SkeletonContainer, + SkeletonPoint, + SkeletonRectangle, + SkeletonRow, +} from '.' + +const SkeletonDemo = () => { + return ( +
+
Loading skeletons
+
+ + + + + + + + + + + + + +
+
+ + + + + + + + +
+
+ ) +} + +const meta = { + title: 'Base/Feedback/Skeleton', + component: SkeletonDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Composable skeleton primitives (container, row, rectangle, point) to sketch loading states for panels and lists.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/slider/index.stories.tsx b/web/app/components/base/slider/index.stories.tsx index d350877d18..4d06381d16 100644 --- a/web/app/components/base/slider/index.stories.tsx +++ b/web/app/components/base/slider/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import Slider from '.' const meta = { - title: 'Base/Slider', + title: 'Base/Data Entry/Slider', component: Slider, parameters: { layout: 'centered', diff --git a/web/app/components/base/sort/index.stories.tsx b/web/app/components/base/sort/index.stories.tsx new file mode 100644 index 0000000000..fea21e8edc --- /dev/null +++ b/web/app/components/base/sort/index.stories.tsx @@ -0,0 +1,59 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useMemo, useState } from 'react' +import Sort from '.' + +const SORT_ITEMS = [ + { value: 'created_at', name: 'Created time' }, + { value: 'updated_at', name: 'Updated time' }, + { value: 'latency', name: 'Latency' }, +] + +const SortPlayground = () => { + const [sortBy, setSortBy] = useState('-created_at') + + const { order, value } = useMemo(() => { + const isDesc = sortBy.startsWith('-') + return { + order: isDesc ? '-' : '', + value: sortBy.replace('-', '') || 'created_at', + } + }, [sortBy]) + + return ( +
+
+ Sort control + + sort_by="{sortBy}" + +
+ { + setSortBy(next as string) + }} + /> +
+ ) +} + +const meta = { + title: 'Base/Data Display/Sort', + component: SortPlayground, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Sorting trigger used in log tables. Includes dropdown selection and quick toggle between ascending and descending.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/spinner/index.stories.tsx b/web/app/components/base/spinner/index.stories.tsx new file mode 100644 index 0000000000..9792b9b2fc --- /dev/null +++ b/web/app/components/base/spinner/index.stories.tsx @@ -0,0 +1,50 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import Spinner from '.' + +const SpinnerPlayground = ({ + loading = true, +}: { + loading?: boolean +}) => { + const [isLoading, setIsLoading] = useState(loading) + + return ( +
+

Spinner

+ + +
+ ) +} + +const meta = { + title: 'Base/Feedback/Spinner', + component: SpinnerPlayground, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Minimal spinner powered by Tailwind utilities. Toggle the state to inspect motion-reduced behaviour.', + }, + }, + }, + argTypes: { + loading: { control: 'boolean' }, + }, + args: { + loading: true, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/svg-gallery/index.stories.tsx b/web/app/components/base/svg-gallery/index.stories.tsx new file mode 100644 index 0000000000..65da97d243 --- /dev/null +++ b/web/app/components/base/svg-gallery/index.stories.tsx @@ -0,0 +1,51 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import SVGRenderer from '.' + +const SAMPLE_SVG = ` + + + + + + + + + + SVG Preview + Click to open high-resolution preview + + + + + Inline SVG asset + +`.trim() + +const meta = { + title: 'Base/Data Display/SVGRenderer', + component: SVGRenderer, + parameters: { + docs: { + description: { + component: 'Renders sanitized SVG markup with zoom-to-preview capability.', + }, + source: { + language: 'tsx', + code: ` +... +\`} /> + `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + content: SAMPLE_SVG, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} diff --git a/web/app/components/base/svg/index.stories.tsx b/web/app/components/base/svg/index.stories.tsx new file mode 100644 index 0000000000..0b7d8d23c9 --- /dev/null +++ b/web/app/components/base/svg/index.stories.tsx @@ -0,0 +1,36 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import SVGBtn from '.' + +const SvgToggleDemo = () => { + const [isSVG, setIsSVG] = useState(false) + + return ( +
+

SVG toggle

+ + + Mode: {isSVG ? 'SVG' : 'PNG'} + +
+ ) +} + +const meta = { + title: 'Base/General/SVGBtn', + component: SvgToggleDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Small toggle used in icon pickers to switch between SVG and bitmap assets.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/switch/index.stories.tsx b/web/app/components/base/switch/index.stories.tsx index aaeab4c41f..5b2b6e59c4 100644 --- a/web/app/components/base/switch/index.stories.tsx +++ b/web/app/components/base/switch/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import Switch from '.' const meta = { - title: 'Base/Switch', + title: 'Base/Data Entry/Switch', component: Switch, parameters: { layout: 'centered', diff --git a/web/app/components/base/tab-header/index.stories.tsx b/web/app/components/base/tab-header/index.stories.tsx new file mode 100644 index 0000000000..cb383947d9 --- /dev/null +++ b/web/app/components/base/tab-header/index.stories.tsx @@ -0,0 +1,64 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import TabHeader from '.' +import type { ITabHeaderProps } from '.' + +const items: ITabHeaderProps['items'] = [ + { id: 'overview', name: 'Overview' }, + { id: 'playground', name: 'Playground' }, + { id: 'changelog', name: 'Changelog', extra: New }, + { id: 'docs', name: 'Docs', isRight: true }, + { id: 'settings', name: 'Settings', isRight: true, disabled: true }, +] + +const TabHeaderDemo = ({ + initialTab = 'overview', +}: { + initialTab?: string +}) => { + const [activeTab, setActiveTab] = useState(initialTab) + + return ( +
+
+ Tabs + + active="{activeTab}" + +
+ +
+ ) +} + +const meta = { + title: 'Base/Navigation/TabHeader', + component: TabHeaderDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Two-sided header tabs with optional right-aligned actions. Disabled items illustrate read-only states.', + }, + }, + }, + argTypes: { + initialTab: { + control: 'radio', + options: items.map(item => item.id), + }, + }, + args: { + initialTab: 'overview', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/tab-slider-new/index.stories.tsx b/web/app/components/base/tab-slider-new/index.stories.tsx new file mode 100644 index 0000000000..669ec9eed9 --- /dev/null +++ b/web/app/components/base/tab-slider-new/index.stories.tsx @@ -0,0 +1,52 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import { RiSparklingFill, RiTerminalBoxLine } from '@remixicon/react' +import TabSliderNew from '.' + +const OPTIONS = [ + { value: 'visual', text: 'Visual builder', icon: }, + { value: 'code', text: 'Code', icon: }, +] + +const TabSliderNewDemo = ({ + initialValue = 'visual', +}: { + initialValue?: string +}) => { + const [value, setValue] = useState(initialValue) + + return ( +
+
Pill tabs
+ +
+ ) +} + +const meta = { + title: 'Base/Navigation/TabSliderNew', + component: TabSliderNewDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Rounded pill tabs suited for switching between editors. Icons illustrate mixed text/icon options.', + }, + }, + }, + argTypes: { + initialValue: { + control: 'radio', + options: OPTIONS.map(option => option.value), + }, + }, + args: { + initialValue: 'visual', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/tab-slider-plain/index.stories.tsx b/web/app/components/base/tab-slider-plain/index.stories.tsx new file mode 100644 index 0000000000..dd8c7e0d30 --- /dev/null +++ b/web/app/components/base/tab-slider-plain/index.stories.tsx @@ -0,0 +1,56 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useState } from 'react' +import TabSliderPlain from '.' + +const OPTIONS = [ + { value: 'analytics', text: 'Analytics' }, + { value: 'activity', text: 'Recent activity' }, + { value: 'alerts', text: 'Alerts' }, +] + +const TabSliderPlainDemo = ({ + initialValue = 'analytics', +}: { + initialValue?: string +}) => { + const [value, setValue] = useState(initialValue) + + return ( +
+
Underline tabs
+ +
+ ) +} + +const meta = { + title: 'Base/Navigation/TabSliderPlain', + component: TabSliderPlainDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Underline-style navigation commonly used in dashboards. Toggle between three sections.', + }, + }, + }, + argTypes: { + initialValue: { + control: 'radio', + options: OPTIONS.map(option => option.value), + }, + }, + args: { + initialValue: 'analytics', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/tab-slider/index.stories.tsx b/web/app/components/base/tab-slider/index.stories.tsx new file mode 100644 index 0000000000..703116fe19 --- /dev/null +++ b/web/app/components/base/tab-slider/index.stories.tsx @@ -0,0 +1,93 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect, useState } from 'react' +import TabSlider from '.' + +const OPTIONS = [ + { value: 'models', text: 'Models' }, + { value: 'datasets', text: 'Datasets' }, + { value: 'plugins', text: 'Plugins' }, +] + +const TabSliderDemo = ({ + initialValue = 'models', +}: { + initialValue?: string +}) => { + const [value, setValue] = useState(initialValue) + + useEffect(() => { + const originalFetch = globalThis.fetch?.bind(globalThis) + + const handler = async (input: RequestInfo | URL, init?: RequestInit) => { + const url = typeof input === 'string' + ? input + : input instanceof URL + ? input.toString() + : input.url + + if (url.includes('/workspaces/current/plugin/list')) { + return new Response( + JSON.stringify({ + total: 6, + plugins: [], + }), + { + status: 200, + headers: { 'Content-Type': 'application/json' }, + }, + ) + } + + if (originalFetch) + return originalFetch(input, init) + + throw new Error(`Unhandled request for ${url}`) + } + + globalThis.fetch = handler as typeof globalThis.fetch + + return () => { + if (originalFetch) + globalThis.fetch = originalFetch + } + }, []) + + return ( +
+
Segmented tabs
+ +
+ ) +} + +const meta = { + title: 'Base/Navigation/TabSlider', + component: TabSliderDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Animated segmented control with sliding highlight. A badge appears when plugins are installed (mocked in Storybook).', + }, + }, + }, + argTypes: { + initialValue: { + control: 'radio', + options: OPTIONS.map(option => option.value), + }, + }, + args: { + initialValue: 'models', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/tab-slider/index.tsx b/web/app/components/base/tab-slider/index.tsx index 55c44d5ea8..7c9364baf9 100644 --- a/web/app/components/base/tab-slider/index.tsx +++ b/web/app/components/base/tab-slider/index.tsx @@ -11,12 +11,14 @@ type Option = { type TabSliderProps = { className?: string value: string + itemClassName?: string | ((active: boolean) => string) onChange: (v: string) => void options: Option[] } const TabSlider: FC = ({ className, + itemClassName, value, onChange, options, @@ -58,6 +60,7 @@ const TabSlider: FC = ({ index === activeIndex ? 'text-text-primary' : 'text-text-tertiary', + typeof itemClassName === 'function' ? itemClassName(index === activeIndex) : itemClassName, )} onClick={() => { if (index !== activeIndex) { diff --git a/web/app/components/base/tag-input/index.stories.tsx b/web/app/components/base/tag-input/index.stories.tsx index dacb222c8c..7aae9f2773 100644 --- a/web/app/components/base/tag-input/index.stories.tsx +++ b/web/app/components/base/tag-input/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import TagInput from '.' const meta = { - title: 'Base/TagInput', + title: 'Base/Data Entry/TagInput', component: TagInput, parameters: { layout: 'centered', diff --git a/web/app/components/base/tag-management/index.stories.tsx b/web/app/components/base/tag-management/index.stories.tsx new file mode 100644 index 0000000000..51f4233461 --- /dev/null +++ b/web/app/components/base/tag-management/index.stories.tsx @@ -0,0 +1,131 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useEffect, useRef } from 'react' +import TagManagementModal from '.' +import { ToastProvider } from '@/app/components/base/toast' +import { useStore as useTagStore } from './store' +import type { Tag } from './constant' + +const INITIAL_TAGS: Tag[] = [ + { id: 'tag-product', name: 'Product', type: 'app', binding_count: 12 }, + { id: 'tag-growth', name: 'Growth', type: 'app', binding_count: 4 }, + { id: 'tag-beta', name: 'Beta User', type: 'app', binding_count: 2 }, + { id: 'tag-rag', name: 'RAG', type: 'knowledge', binding_count: 3 }, + { id: 'tag-updates', name: 'Release Notes', type: 'knowledge', binding_count: 6 }, +] + +const TagManagementPlayground = ({ + type = 'app', +}: { + type?: 'app' | 'knowledge' +}) => { + const originalFetchRef = useRef(null) + const tagsRef = useRef(INITIAL_TAGS) + const setTagList = useTagStore(s => s.setTagList) + const showModal = useTagStore(s => s.showTagManagementModal) + const setShowModal = useTagStore(s => s.setShowTagManagementModal) + + useEffect(() => { + setTagList(tagsRef.current) + setShowModal(true) + }, [setTagList, setShowModal]) + + useEffect(() => { + originalFetchRef.current = globalThis.fetch?.bind(globalThis) + + const handler = async (input: RequestInfo | URL, init?: RequestInit) => { + const request = input instanceof Request ? input : new Request(input, init) + const url = request.url + const method = request.method.toUpperCase() + const parsedUrl = new URL(url, window.location.origin) + + if (parsedUrl.pathname.endsWith('/tags')) { + if (method === 'GET') { + const tagType = parsedUrl.searchParams.get('type') || 'app' + const payload = tagsRef.current.filter(tag => tag.type === tagType) + return new Response(JSON.stringify(payload), { + status: 200, + headers: { 'Content-Type': 'application/json' }, + }) + } + if (method === 'POST') { + const body = await request.clone().json() as { name: string; type: string } + const newTag: Tag = { + id: `tag-${Date.now()}`, + name: body.name, + type: body.type, + binding_count: 0, + } + tagsRef.current = [newTag, ...tagsRef.current] + setTagList(tagsRef.current) + return new Response(JSON.stringify(newTag), { + status: 200, + headers: { 'Content-Type': 'application/json' }, + }) + } + } + + if (parsedUrl.pathname.endsWith('/tag-bindings/create') || parsedUrl.pathname.endsWith('/tag-bindings/remove')) { + return new Response(JSON.stringify({ ok: true }), { + status: 200, + headers: { 'Content-Type': 'application/json' }, + }) + } + + if (originalFetchRef.current) + return originalFetchRef.current(request) + + throw new Error(`Unhandled request in mock fetch: ${url}`) + } + + globalThis.fetch = handler as typeof globalThis.fetch + + return () => { + if (originalFetchRef.current) + globalThis.fetch = originalFetchRef.current + } + }, [setTagList]) + + return ( + +
+ +

Mocked tag management flows with create and bind actions.

+
+ +
+ ) +} + +const meta = { + title: 'Base/Data Display/TagManagementModal', + component: TagManagementPlayground, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Complete tag management modal with mocked service calls for browsing and creating tags.', + }, + }, + }, + argTypes: { + type: { + control: 'radio', + options: ['app', 'knowledge'], + }, + }, + args: { + type: 'app', + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/tag/index.stories.tsx b/web/app/components/base/tag/index.stories.tsx new file mode 100644 index 0000000000..8ca15c0c8b --- /dev/null +++ b/web/app/components/base/tag/index.stories.tsx @@ -0,0 +1,62 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import Tag from '.' + +const COLORS: Array['color']>> = ['green', 'yellow', 'red', 'gray'] + +const TagGallery = ({ + bordered = false, + hideBg = false, +}: { + bordered?: boolean + hideBg?: boolean +}) => { + return ( +
+
Tag variants
+
+ {COLORS.map(color => ( +
+ + {color.charAt(0).toUpperCase() + color.slice(1)} + + {color} +
+ ))} +
+
+ ) +} + +const meta = { + title: 'Base/Data Display/Tag', + component: TagGallery, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Color-coded label component. Toggle borders or remove background to fit dark/light surfaces.', + }, + }, + }, + argTypes: { + bordered: { control: 'boolean' }, + hideBg: { control: 'boolean' }, + }, + args: { + bordered: false, + hideBg: false, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} + +export const Outlined: Story = { + args: { + bordered: true, + hideBg: true, + }, +} diff --git a/web/app/components/base/textarea/index.stories.tsx b/web/app/components/base/textarea/index.stories.tsx index d03b3decb7..41d8bda458 100644 --- a/web/app/components/base/textarea/index.stories.tsx +++ b/web/app/components/base/textarea/index.stories.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import Textarea from '.' const meta = { - title: 'Base/Textarea', + title: 'Base/Data Entry/Textarea', component: Textarea, parameters: { layout: 'centered', diff --git a/web/app/components/base/textarea/index.tsx b/web/app/components/base/textarea/index.tsx index 7813eb7209..609f1ad51d 100644 --- a/web/app/components/base/textarea/index.tsx +++ b/web/app/components/base/textarea/index.tsx @@ -20,7 +20,7 @@ const textareaVariants = cva( ) export type TextareaProps = { - value: string + value: string | number disabled?: boolean destructive?: boolean styleCss?: CSSProperties diff --git a/web/app/components/base/timezone-label/__tests__/index.test.tsx b/web/app/components/base/timezone-label/__tests__/index.test.tsx new file mode 100644 index 0000000000..1c36ac929a --- /dev/null +++ b/web/app/components/base/timezone-label/__tests__/index.test.tsx @@ -0,0 +1,145 @@ +import React from 'react' +import { render, screen } from '@testing-library/react' +import TimezoneLabel from '../index' + +// Mock the convertTimezoneToOffsetStr function +jest.mock('@/app/components/base/date-and-time-picker/utils/dayjs', () => ({ + convertTimezoneToOffsetStr: (timezone?: string) => { + if (!timezone) return 'UTC+0' + + // Mock implementation matching the actual timezone conversions + const timezoneOffsets: Record = { + 'Asia/Shanghai': 'UTC+8', + 'America/New_York': 'UTC-5', + 'Europe/London': 'UTC+0', + 'Pacific/Auckland': 'UTC+13', + 'Pacific/Niue': 'UTC-11', + 'UTC': 'UTC+0', + } + + return timezoneOffsets[timezone] || 'UTC+0' + }, +})) + +describe('TimezoneLabel', () => { + describe('Basic Rendering', () => { + it('should render timezone offset correctly', () => { + render() + expect(screen.getByText('UTC+8')).toBeInTheDocument() + }) + + it('should display UTC+0 for invalid timezone', () => { + render() + expect(screen.getByText('UTC+0')).toBeInTheDocument() + }) + + it('should handle UTC timezone', () => { + render() + expect(screen.getByText('UTC+0')).toBeInTheDocument() + }) + }) + + describe('Styling', () => { + it('should apply default tertiary text color', () => { + const { container } = render() + const span = container.querySelector('span') + expect(span).toHaveClass('text-text-tertiary') + expect(span).not.toHaveClass('text-text-quaternary') + }) + + it('should apply quaternary text color in inline mode', () => { + const { container } = render() + const span = container.querySelector('span') + expect(span).toHaveClass('text-text-quaternary') + }) + + it('should apply custom className', () => { + const { container } = render( + , + ) + const span = container.querySelector('span') + expect(span).toHaveClass('custom-class') + }) + + it('should maintain default classes with custom className', () => { + const { container } = render( + , + ) + const span = container.querySelector('span') + expect(span).toHaveClass('system-sm-regular') + expect(span).toHaveClass('text-text-tertiary') + expect(span).toHaveClass('custom-class') + }) + }) + + describe('Tooltip', () => { + it('should include timezone information in title attribute', () => { + const { container } = render() + const span = container.querySelector('span') + expect(span).toHaveAttribute('title', 'Timezone: Asia/Shanghai (UTC+8)') + }) + + it('should update tooltip for different timezones', () => { + const { container } = render() + const span = container.querySelector('span') + expect(span).toHaveAttribute('title', 'Timezone: America/New_York (UTC-5)') + }) + }) + + describe('Edge Cases', () => { + it('should handle positive offset timezones', () => { + render() + expect(screen.getByText('UTC+13')).toBeInTheDocument() + }) + + it('should handle negative offset timezones', () => { + render() + expect(screen.getByText('UTC-11')).toBeInTheDocument() + }) + + it('should handle zero offset timezone', () => { + render() + expect(screen.getByText('UTC+0')).toBeInTheDocument() + }) + }) + + describe('Props Variations', () => { + it('should render with only required timezone prop', () => { + render() + expect(screen.getByText('UTC+8')).toBeInTheDocument() + }) + + it('should render with all props', () => { + const { container } = render( + , + ) + const span = container.querySelector('span') + expect(screen.getByText('UTC-5')).toBeInTheDocument() + expect(span).toHaveClass('text-xs') + expect(span).toHaveClass('text-text-quaternary') + }) + }) + + describe('Memoization', () => { + it('should memoize offset calculation', () => { + const { rerender } = render() + expect(screen.getByText('UTC+8')).toBeInTheDocument() + + // Rerender with same props should not trigger recalculation + rerender() + expect(screen.getByText('UTC+8')).toBeInTheDocument() + }) + + it('should recalculate when timezone changes', () => { + const { rerender } = render() + expect(screen.getByText('UTC+8')).toBeInTheDocument() + + rerender() + expect(screen.getByText('UTC-5')).toBeInTheDocument() + }) + }) +}) diff --git a/web/app/components/base/timezone-label/index.tsx b/web/app/components/base/timezone-label/index.tsx new file mode 100644 index 0000000000..b151ceb9b8 --- /dev/null +++ b/web/app/components/base/timezone-label/index.tsx @@ -0,0 +1,56 @@ +import React, { useMemo } from 'react' +import { convertTimezoneToOffsetStr } from '@/app/components/base/date-and-time-picker/utils/dayjs' +import cn from '@/utils/classnames' + +export type TimezoneLabelProps = { + /** IANA timezone identifier (e.g., 'Asia/Shanghai', 'America/New_York') */ + timezone: string + /** Additional CSS classes to apply */ + className?: string + /** Use inline mode with lighter text color for secondary display */ + inline?: boolean +} + +/** + * TimezoneLabel component displays timezone information in UTC offset format. + * + * @example + * // Standard display + * + * // Output: UTC+8 + * + * @example + * // Inline mode with lighter color + * + * // Output: UTC-5 + * + * @example + * // Custom styling + * + */ +const TimezoneLabel: React.FC = ({ + timezone, + className, + inline = false, +}) => { + // Memoize offset calculation to avoid redundant computations + const offsetStr = useMemo( + () => convertTimezoneToOffsetStr(timezone), + [timezone], + ) + + return ( + + {offsetStr} + + ) +} + +export default React.memo(TimezoneLabel) diff --git a/web/app/components/base/toast/index.stories.tsx b/web/app/components/base/toast/index.stories.tsx new file mode 100644 index 0000000000..6ef65475cb --- /dev/null +++ b/web/app/components/base/toast/index.stories.tsx @@ -0,0 +1,104 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import { useCallback } from 'react' +import Toast, { ToastProvider, useToastContext } from '.' + +const ToastControls = () => { + const { notify } = useToastContext() + + const trigger = useCallback((type: 'success' | 'error' | 'warning' | 'info') => { + notify({ + type, + message: `This is a ${type} toast`, + children: type === 'info' ? 'Additional details can live here.' : undefined, + }) + }, [notify]) + + return ( +
+ + + + +
+ ) +} + +const ToastProviderDemo = () => { + return ( + +
+
Toast provider
+ +
+
+ ) +} + +const StaticToastDemo = () => { + return ( +
+
Static API
+ +
+ ) +} + +const meta = { + title: 'Base/Feedback/Toast', + component: ToastProviderDemo, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'ToastProvider based notifications and the static Toast.notify helper. Buttons showcase each toast variant.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Provider: Story = {} + +export const StaticApi: Story = { + render: () => , +} diff --git a/web/app/components/base/tooltip/index.stories.tsx b/web/app/components/base/tooltip/index.stories.tsx new file mode 100644 index 0000000000..aeca69464f --- /dev/null +++ b/web/app/components/base/tooltip/index.stories.tsx @@ -0,0 +1,60 @@ +import type { Meta, StoryObj } from '@storybook/nextjs' +import Tooltip from '.' + +const TooltipGrid = () => { + return ( +
+
Hover tooltips
+
+ + + + + + Right tooltip + + +
+
Click tooltips
+
+ + + + + + Plain content + + +
+
+ ) +} + +const meta = { + title: 'Base/Feedback/Tooltip', + component: TooltipGrid, + parameters: { + layout: 'centered', + docs: { + description: { + component: 'Portal-based tooltip component supporting hover and click triggers, custom placements, and decorated content.', + }, + }, + }, + tags: ['autodocs'], +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Playground: Story = {} diff --git a/web/app/components/base/tooltip/index.tsx b/web/app/components/base/tooltip/index.tsx index eb7ca56cb0..46680c8f5b 100644 --- a/web/app/components/base/tooltip/index.tsx +++ b/web/app/components/base/tooltip/index.tsx @@ -17,6 +17,7 @@ export type TooltipProps = { popupContent?: React.ReactNode children?: React.ReactNode popupClassName?: string + portalContentClassName?: string noDecoration?: boolean offset?: OffsetOptions needsDelay?: boolean @@ -32,6 +33,7 @@ const Tooltip: FC = ({ popupContent, children, popupClassName, + portalContentClassName, noDecoration, offset, asChild = true, @@ -104,7 +106,7 @@ const Tooltip: FC = ({ {children ||
}
{popupContent && (
+ `.trim(), + }, + }, + }, + tags: ['autodocs'], + args: { + srcs: VIDEO_SOURCES, + }, +} satisfies Meta + +export default meta +type Story = StoryObj + +export const Default: Story = {} diff --git a/web/app/components/base/voice-input/index.stories.tsx b/web/app/components/base/voice-input/index.stories.tsx index 8d92f587c4..de6a675ab3 100644 --- a/web/app/components/base/voice-input/index.stories.tsx +++ b/web/app/components/base/voice-input/index.stories.tsx @@ -29,7 +29,7 @@ const VoiceInputMock = ({ onConverted, onCancel }: any) => {
{/* Waveform visualization placeholder */}
- {new Array(40).fill(0).map((_, i) => ( + {Array.from({ length: 40 }).map((_, i) => (
{ } const meta = { - title: 'Base/VoiceInput', + title: 'Base/Data Entry/VoiceInput', component: VoiceInputMock, parameters: { layout: 'centered', diff --git a/web/app/components/base/voice-input/utils.ts b/web/app/components/base/voice-input/utils.ts index 70133f459f..a8ac9eba03 100644 --- a/web/app/components/base/voice-input/utils.ts +++ b/web/app/components/base/voice-input/utils.ts @@ -14,13 +14,19 @@ export const convertToMp3 = (recorder: any) => { const { channels, sampleRate } = wav const mp3enc = new lamejs.Mp3Encoder(channels, sampleRate, 128) const result = recorder.getChannelData() - const buffer = [] + const buffer: BlobPart[] = [] const leftData = result.left && new Int16Array(result.left.buffer, 0, result.left.byteLength / 2) const rightData = result.right && new Int16Array(result.right.buffer, 0, result.right.byteLength / 2) const remaining = leftData.length + (rightData ? rightData.length : 0) const maxSamples = 1152 + const toArrayBuffer = (bytes: Int8Array) => { + const arrayBuffer = new ArrayBuffer(bytes.length) + new Uint8Array(arrayBuffer).set(bytes) + return arrayBuffer + } + for (let i = 0; i < remaining; i += maxSamples) { const left = leftData.subarray(i, i + maxSamples) let right = null @@ -35,13 +41,13 @@ export const convertToMp3 = (recorder: any) => { } if (mp3buf.length > 0) - buffer.push(mp3buf) + buffer.push(toArrayBuffer(mp3buf)) } const enc = mp3enc.flush() if (enc.length > 0) - buffer.push(enc) + buffer.push(toArrayBuffer(enc)) return new Blob(buffer, { type: 'audio/mp3' }) } diff --git a/web/app/components/base/with-input-validation/index.stories.tsx b/web/app/components/base/with-input-validation/index.stories.tsx index 98d2d0bafb..26fa9747d8 100644 --- a/web/app/components/base/with-input-validation/index.stories.tsx +++ b/web/app/components/base/with-input-validation/index.stories.tsx @@ -63,7 +63,7 @@ const ValidatedUserCard = withValidation(UserCard, userSchema) const ValidatedProductCard = withValidation(ProductCard, productSchema) const meta = { - title: 'Base/WithInputValidation', + title: 'Base/Data Entry/WithInputValidation', parameters: { layout: 'centered', docs: { diff --git a/web/app/components/billing/config.ts b/web/app/components/billing/config.ts index 1d5fbc7491..c0a21c1ebf 100644 --- a/web/app/components/billing/config.ts +++ b/web/app/components/billing/config.ts @@ -26,6 +26,7 @@ export const ALL_PLANS: Record = { apiRateLimit: 5000, documentProcessingPriority: Priority.standard, messageRequest: 200, + triggerEvents: 3000, annotatedResponse: 10, logHistory: 30, }, @@ -43,6 +44,7 @@ export const ALL_PLANS: Record = { apiRateLimit: NUM_INFINITE, documentProcessingPriority: Priority.priority, messageRequest: 5000, + triggerEvents: 20000, annotatedResponse: 2000, logHistory: NUM_INFINITE, }, @@ -60,6 +62,7 @@ export const ALL_PLANS: Record = { apiRateLimit: NUM_INFINITE, documentProcessingPriority: Priority.topPriority, messageRequest: 10000, + triggerEvents: NUM_INFINITE, annotatedResponse: 5000, logHistory: NUM_INFINITE, }, @@ -74,6 +77,8 @@ export const defaultPlan = { teamMembers: 1, annotatedResponse: 1, documentsUploadQuota: 0, + apiRateLimit: 0, + triggerEvents: 0, }, total: { documents: 50, @@ -82,5 +87,7 @@ export const defaultPlan = { teamMembers: 1, annotatedResponse: 10, documentsUploadQuota: 0, + apiRateLimit: ALL_PLANS.sandbox.apiRateLimit, + triggerEvents: ALL_PLANS.sandbox.triggerEvents, }, } diff --git a/web/app/components/billing/plan/index.tsx b/web/app/components/billing/plan/index.tsx index dd3908635b..4b68fcfb15 100644 --- a/web/app/components/billing/plan/index.tsx +++ b/web/app/components/billing/plan/index.tsx @@ -6,8 +6,10 @@ import { useRouter } from 'next/navigation' import { RiBook2Line, RiFileEditLine, + RiFlashlightLine, RiGraduationCapLine, RiGroupLine, + RiSpeedLine, } from '@remixicon/react' import { Plan, SelfHostedPlan } from '../type' import VectorSpaceInfo from '../usage-info/vector-space-info' @@ -43,6 +45,8 @@ const PlanComp: FC = ({ usage, total, } = plan + const perMonthUnit = ` ${t('billing.usagePage.perMonth')}` + const triggerEventUnit = plan.type === Plan.sandbox ? undefined : perMonthUnit const [showModal, setShowModal] = React.useState(false) const { mutateAsync } = useEducationVerify() @@ -119,6 +123,20 @@ const PlanComp: FC = ({ usage={usage.annotatedResponse} total={total.annotatedResponse} /> + +
= ({ const [planRange, setPlanRange] = React.useState(PlanRange.monthly) const [currentCategory, setCurrentCategory] = useState(CategoryEnum.CLOUD) const canPay = isCurrentWorkspaceManager - useKeyPress(['esc'], onCancel) const pricingPageLanguage = useGetPricingPageLanguage() diff --git a/web/app/components/billing/pricing/plans/cloud-plan-item/list/index.tsx b/web/app/components/billing/pricing/plans/cloud-plan-item/list/index.tsx index 0420bfc7a3..0b35ee7e97 100644 --- a/web/app/components/billing/pricing/plans/cloud-plan-item/list/index.tsx +++ b/web/app/components/billing/pricing/plans/cloud-plan-item/list/index.tsx @@ -56,6 +56,31 @@ const List = ({ + + + { + const currentPlanType: BasicPlan = plan.type === Plan.enterprise ? Plan.team : plan.type return (
@@ -28,21 +29,21 @@ const Plans = ({ currentPlan === 'cloud' && ( <> & { vectorSpace: number } +export type UsagePlanInfo = Pick & { vectorSpace: number } export enum DocumentProcessingPriority { standard = 'standard', @@ -87,6 +88,14 @@ export type CurrentPlanInfoBackend = { size: number limit: number // total. 0 means unlimited } + api_rate_limit?: { + size: number + limit: number // total. 0 means unlimited + } + trigger_events?: { + size: number + limit: number // total. 0 means unlimited + } docs_processing: DocumentProcessingPriority can_replace_logo: boolean model_load_balancing_enabled: boolean diff --git a/web/app/components/billing/usage-info/index.tsx b/web/app/components/billing/usage-info/index.tsx index 30b4bca776..0ed8775772 100644 --- a/web/app/components/billing/usage-info/index.tsx +++ b/web/app/components/billing/usage-info/index.tsx @@ -15,6 +15,7 @@ type Props = { usage: number total: number unit?: string + unitPosition?: 'inline' | 'suffix' } const LOW = 50 @@ -27,7 +28,8 @@ const UsageInfo: FC = ({ tooltip, usage, total, - unit = '', + unit, + unitPosition = 'suffix', }) => { const { t } = useTranslation() @@ -41,6 +43,12 @@ const UsageInfo: FC = ({ return 'bg-components-progress-error-progress' })() + const isUnlimited = total === NUM_INFINITE + let totalDisplay: string | number = isUnlimited ? t('billing.plansCommon.unlimited') : total + if (!isUnlimited && unit && unitPosition === 'inline') + totalDisplay = `${total}${unit}` + const showUnit = !!unit && !isUnlimited && unitPosition === 'suffix' + return (
@@ -56,10 +64,17 @@ const UsageInfo: FC = ({ /> )}
-
- {usage} -
/
-
{total === NUM_INFINITE ? t('billing.plansCommon.unlimited') : `${total}${unit}`}
+
+
+ {usage} +
/
+
{totalDisplay}
+
+ {showUnit && ( +
+ {unit} +
+ )}
= ({ usage={usage.vectorSpace} total={total.vectorSpace} unit='MB' + unitPosition='inline' /> ) } diff --git a/web/app/components/billing/utils/index.ts b/web/app/components/billing/utils/index.ts index 111f02e3cf..00ab7913b5 100644 --- a/web/app/components/billing/utils/index.ts +++ b/web/app/components/billing/utils/index.ts @@ -1,5 +1,5 @@ import type { CurrentPlanInfoBackend } from '../type' -import { NUM_INFINITE } from '@/app/components/billing/config' +import { ALL_PLANS, NUM_INFINITE } from '@/app/components/billing/config' const parseLimit = (limit: number) => { if (limit === 0) @@ -9,14 +9,23 @@ const parseLimit = (limit: number) => { } export const parseCurrentPlan = (data: CurrentPlanInfoBackend) => { + const planType = data.billing.subscription.plan + const planPreset = ALL_PLANS[planType] + const resolveLimit = (limit?: number, fallback?: number) => { + const value = limit ?? fallback ?? 0 + return parseLimit(value) + } + return { - type: data.billing.subscription.plan, + type: planType, usage: { vectorSpace: data.vector_space.size, buildApps: data.apps?.size || 0, teamMembers: data.members.size, annotatedResponse: data.annotation_quota_limit.size, documentsUploadQuota: data.documents_upload_quota.size, + apiRateLimit: data.api_rate_limit?.size ?? 0, + triggerEvents: data.trigger_events?.size ?? 0, }, total: { vectorSpace: parseLimit(data.vector_space.limit), @@ -24,6 +33,8 @@ export const parseCurrentPlan = (data: CurrentPlanInfoBackend) => { teamMembers: parseLimit(data.members.limit), annotatedResponse: parseLimit(data.annotation_quota_limit.limit), documentsUploadQuota: parseLimit(data.documents_upload_quota.limit), + apiRateLimit: resolveLimit(data.api_rate_limit?.limit, planPreset?.apiRateLimit ?? NUM_INFINITE), + triggerEvents: resolveLimit(data.trigger_events?.limit, planPreset?.triggerEvents), }, } } diff --git a/web/app/components/custom/custom-web-app-brand/index.tsx b/web/app/components/custom/custom-web-app-brand/index.tsx index eb06265042..fee0bf75f7 100644 --- a/web/app/components/custom/custom-web-app-brand/index.tsx +++ b/web/app/components/custom/custom-web-app-brand/index.tsx @@ -16,7 +16,7 @@ import Button from '@/app/components/base/button' import Divider from '@/app/components/base/divider' import { useProviderContext } from '@/context/provider-context' import { Plan } from '@/app/components/billing/type' -import { imageUpload } from '@/app/components/base/image-uploader/utils' +import { getImageUploadErrorMessage, imageUpload } from '@/app/components/base/image-uploader/utils' import { useToastContext } from '@/app/components/base/toast' import { BubbleTextMod } from '@/app/components/base/icons/src/vender/solid/communication' import { @@ -67,8 +67,9 @@ const CustomWebAppBrand = () => { setUploadProgress(100) setFileId(res.id) }, - onErrorCallback: () => { - notify({ type: 'error', message: t('common.imageUploader.uploadFromComputerUploadError') }) + onErrorCallback: (error?: any) => { + const errorMessage = getImageUploadErrorMessage(error, t('common.imageUploader.uploadFromComputerUploadError'), t) + notify({ type: 'error', message: errorMessage }) setUploadProgress(-1) }, }, false, '/workspaces/custom-config/webapp-logo/upload') diff --git a/web/app/components/datasets/create-from-pipeline/list/built-in-pipeline-list.tsx b/web/app/components/datasets/create-from-pipeline/list/built-in-pipeline-list.tsx index 6d22f2115a..74e565a494 100644 --- a/web/app/components/datasets/create-from-pipeline/list/built-in-pipeline-list.tsx +++ b/web/app/components/datasets/create-from-pipeline/list/built-in-pipeline-list.tsx @@ -4,6 +4,7 @@ import CreateCard from './create-card' import { useI18N } from '@/context/i18n' import { useMemo } from 'react' import { LanguagesSupported } from '@/i18n-config/language' +import { useGlobalPublicStore } from '@/context/global-public-context' const BuiltInPipelineList = () => { const { locale } = useI18N() @@ -12,7 +13,8 @@ const BuiltInPipelineList = () => { return locale return LanguagesSupported[0] }, [locale]) - const { data: pipelineList, isLoading } = usePipelineTemplateList({ type: 'built-in', language }) + const enableMarketplace = useGlobalPublicStore(s => s.systemFeatures.enable_marketplace) + const { data: pipelineList, isLoading } = usePipelineTemplateList({ type: 'built-in', language }, enableMarketplace) const list = pipelineList?.pipeline_templates || [] return ( diff --git a/web/app/components/datasets/create/embedding-process/index.tsx b/web/app/components/datasets/create/embedding-process/index.tsx index df64f26ab9..7b2eda1dcd 100644 --- a/web/app/components/datasets/create/embedding-process/index.tsx +++ b/web/app/components/datasets/create/embedding-process/index.tsx @@ -18,7 +18,13 @@ import DocumentFileIcon from '../../common/document-file-icon' import cn from '@/utils/classnames' import { FieldInfo } from '@/app/components/datasets/documents/detail/metadata' import Button from '@/app/components/base/button' -import type { FullDocumentDetail, IndexingStatusResponse, ProcessRuleResponse } from '@/models/datasets' +import type { + DataSourceInfo, + FullDocumentDetail, + IndexingStatusResponse, + LegacyDataSourceInfo, + ProcessRuleResponse, +} from '@/models/datasets' import { fetchIndexingStatusBatch as doFetchIndexingStatus, fetchProcessRule } from '@/service/datasets' import { DataSourceType, ProcessMode } from '@/models/datasets' import NotionIcon from '@/app/components/base/notion-icon' @@ -241,10 +247,16 @@ const EmbeddingProcess: FC = ({ datasetId, batchId, documents = [], index return doc?.data_source_type as DataSourceType } + const isLegacyDataSourceInfo = (info: DataSourceInfo): info is LegacyDataSourceInfo => { + return info != null && typeof (info as LegacyDataSourceInfo).upload_file === 'object' + } + const getIcon = (id: string) => { const doc = documents.find(document => document.id === id) - - return doc?.data_source_info.notion_page_icon + const info = doc?.data_source_info + if (info && isLegacyDataSourceInfo(info)) + return info.notion_page_icon + return undefined } const isSourceEmbedding = (detail: IndexingStatusResponse) => ['indexing', 'splitting', 'parsing', 'cleaning', 'waiting'].includes(detail.indexing_status || '') diff --git a/web/app/components/datasets/create/file-uploader/index.tsx b/web/app/components/datasets/create/file-uploader/index.tsx index e2bbad2776..4aec0d4082 100644 --- a/web/app/components/datasets/create/file-uploader/index.tsx +++ b/web/app/components/datasets/create/file-uploader/index.tsx @@ -18,8 +18,7 @@ import { LanguagesSupported } from '@/i18n-config/language' import { IS_CE_EDITION } from '@/config' import { Theme } from '@/types/app' import useTheme from '@/hooks/use-theme' - -const FILES_NUMBER_LIMIT = 20 +import { getFileUploadErrorMessage } from '@/app/components/base/file-uploader/utils' type IFileUploaderProps = { fileList: FileItem[] @@ -72,6 +71,7 @@ const FileUploader = ({ const fileUploadConfig = useMemo(() => fileUploadConfigResponse ?? { file_size_limit: 15, batch_count_limit: 5, + file_upload_limit: 5, }, [fileUploadConfigResponse]) const fileListRef = useRef([]) @@ -121,10 +121,10 @@ const FileUploader = ({ data: formData, onprogress: onProgress, }, false, undefined, '?source=datasets') - .then((res: File) => { + .then((res) => { const completeFile = { fileID: fileItem.fileID, - file: res, + file: res as unknown as File, progress: -1, } const index = fileListRef.current.findIndex(item => item.fileID === fileItem.fileID) @@ -133,7 +133,8 @@ const FileUploader = ({ return Promise.resolve({ ...completeFile }) }) .catch((e) => { - notify({ type: 'error', message: e?.response?.code === 'forbidden' ? e?.response?.message : t('datasetCreation.stepOne.uploader.failed') }) + const errorMessage = getFileUploadErrorMessage(e, t('datasetCreation.stepOne.uploader.failed'), t) + notify({ type: 'error', message: errorMessage }) onFileUpdate(fileItem, -2, fileListRef.current) return Promise.resolve({ ...fileItem }) }) @@ -163,11 +164,12 @@ const FileUploader = ({ }, [fileUploadConfig, uploadBatchFiles]) const initialUpload = useCallback((files: File[]) => { + const filesCountLimit = fileUploadConfig.file_upload_limit if (!files.length) return false - if (files.length + fileList.length > FILES_NUMBER_LIMIT && !IS_CE_EDITION) { - notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.validation.filesNumber', { filesNumber: FILES_NUMBER_LIMIT }) }) + if (files.length + fileList.length > filesCountLimit && !IS_CE_EDITION) { + notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.validation.filesNumber', { filesNumber: filesCountLimit }) }) return false } @@ -180,7 +182,7 @@ const FileUploader = ({ prepareFileList(newFiles) fileListRef.current = newFiles uploadMultipleFiles(preparedFiles) - }, [prepareFileList, uploadMultipleFiles, notify, t, fileList]) + }, [prepareFileList, uploadMultipleFiles, notify, t, fileList, fileUploadConfig]) const handleDragEnter = (e: DragEvent) => { e.preventDefault() @@ -255,10 +257,11 @@ const FileUploader = ({ ) let files = nested.flat() if (notSupportBatchUpload) files = files.slice(0, 1) + files = files.slice(0, fileUploadConfig.batch_count_limit) const valid = files.filter(isValid) initialUpload(valid) }, - [initialUpload, isValid, notSupportBatchUpload, traverseFileEntry], + [initialUpload, isValid, notSupportBatchUpload, traverseFileEntry, fileUploadConfig], ) const selectHandle = () => { if (fileUploader.current) @@ -273,9 +276,10 @@ const FileUploader = ({ onFileListUpdate?.([...fileListRef.current]) } const fileChangeHandle = useCallback((e: React.ChangeEvent) => { - const files = [...(e.target.files ?? [])] as File[] + let files = [...(e.target.files ?? [])] as File[] + files = files.slice(0, fileUploadConfig.batch_count_limit) initialUpload(files.filter(isValid)) - }, [isValid, initialUpload]) + }, [isValid, initialUpload, fileUploadConfig]) const { theme } = useTheme() const chartColor = useMemo(() => theme === Theme.dark ? '#5289ff' : '#296dff', [theme]) @@ -324,7 +328,8 @@ const FileUploader = ({
{t('datasetCreation.stepOne.uploader.tip', { size: fileUploadConfig.file_size_limit, supportTypes: supportTypesShowNames, - batchCount: fileUploadConfig.batch_count_limit, + batchCount: notSupportBatchUpload ? 1 : fileUploadConfig.batch_count_limit, + totalCount: fileUploadConfig.file_upload_limit, })}
{dragging &&
}
diff --git a/web/app/components/datasets/create/index.tsx b/web/app/components/datasets/create/index.tsx index 11def1a8bc..b04bd85530 100644 --- a/web/app/components/datasets/create/index.tsx +++ b/web/app/components/datasets/create/index.tsx @@ -16,6 +16,7 @@ import { useGetDefaultDataSourceListAuth } from '@/service/use-datasource' import { produce } from 'immer' import { useDatasetDetailContextWithSelector } from '@/context/dataset-detail' import Loading from '@/app/components/base/loading' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' type DatasetUpdateFormProps = { datasetId?: string @@ -117,7 +118,7 @@ const DatasetUpdateForm = ({ datasetId }: DatasetUpdateFormProps) => { {step === 1 && ( setShowAccountSettingModal({ payload: 'data-source' })} + onSetting={() => setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.DATA_SOURCE })} datasetId={datasetId} dataSourceType={dataSourceType} dataSourceTypeDisable={!!datasetDetail?.data_source_type} @@ -141,7 +142,7 @@ const DatasetUpdateForm = ({ datasetId }: DatasetUpdateFormProps) => { {(step === 2 && (!datasetId || (datasetId && !!datasetDetail))) && ( setShowAccountSettingModal({ payload: 'provider' })} + onSetting={() => setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.PROVIDER })} indexingType={datasetDetail?.indexing_technique} datasetId={datasetId} dataSourceType={dataSourceType} diff --git a/web/app/components/datasets/create/website/firecrawl/index.tsx b/web/app/components/datasets/create/website/firecrawl/index.tsx index 8d207a0386..51c2c7d505 100644 --- a/web/app/components/datasets/create/website/firecrawl/index.tsx +++ b/web/app/components/datasets/create/website/firecrawl/index.tsx @@ -14,6 +14,7 @@ import Toast from '@/app/components/base/toast' import { checkFirecrawlTaskStatus, createFirecrawlTask } from '@/service/datasets' import { sleep } from '@/utils' import Header from '../base/header' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const ERROR_I18N_PREFIX = 'common.errorMsg' const I18N_PREFIX = 'datasetCreation.stepOne.website' @@ -51,7 +52,7 @@ const FireCrawl: FC = ({ const setShowAccountSettingModal = useModalContextSelector(s => s.setShowAccountSettingModal) const handleSetting = useCallback(() => { setShowAccountSettingModal({ - payload: 'data-source', + payload: ACCOUNT_SETTING_TAB.DATA_SOURCE, }) }, [setShowAccountSettingModal]) diff --git a/web/app/components/datasets/create/website/index.tsx b/web/app/components/datasets/create/website/index.tsx index 7190ca3228..ee7ace6815 100644 --- a/web/app/components/datasets/create/website/index.tsx +++ b/web/app/components/datasets/create/website/index.tsx @@ -13,6 +13,7 @@ import type { CrawlOptions, CrawlResultItem } from '@/models/datasets' import { DataSourceProvider } from '@/models/common' import { ENABLE_WEBSITE_FIRECRAWL, ENABLE_WEBSITE_JINAREADER, ENABLE_WEBSITE_WATERCRAWL } from '@/config' import type { DataSourceAuth } from '@/app/components/header/account-setting/data-source-page-new/types' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' type Props = { onPreview: (payload: CrawlResultItem) => void @@ -48,7 +49,7 @@ const Website: FC = ({ const handleOnConfig = useCallback(() => { setShowAccountSettingModal({ - payload: 'data-source', + payload: ACCOUNT_SETTING_TAB.DATA_SOURCE, }) }, [setShowAccountSettingModal]) diff --git a/web/app/components/datasets/create/website/jina-reader/index.tsx b/web/app/components/datasets/create/website/jina-reader/index.tsx index 460c169fb4..b6e6177af2 100644 --- a/web/app/components/datasets/create/website/jina-reader/index.tsx +++ b/web/app/components/datasets/create/website/jina-reader/index.tsx @@ -14,6 +14,7 @@ import { checkJinaReaderTaskStatus, createJinaReaderTask } from '@/service/datas import { sleep } from '@/utils' import type { CrawlOptions, CrawlResultItem } from '@/models/datasets' import Header from '../base/header' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const ERROR_I18N_PREFIX = 'common.errorMsg' const I18N_PREFIX = 'datasetCreation.stepOne.website' @@ -51,7 +52,7 @@ const JinaReader: FC = ({ const { setShowAccountSettingModal } = useModalContext() const handleSetting = useCallback(() => { setShowAccountSettingModal({ - payload: 'data-source', + payload: ACCOUNT_SETTING_TAB.DATA_SOURCE, }) }, [setShowAccountSettingModal]) diff --git a/web/app/components/datasets/create/website/watercrawl/index.tsx b/web/app/components/datasets/create/website/watercrawl/index.tsx index 640b1c2063..67a3e53feb 100644 --- a/web/app/components/datasets/create/website/watercrawl/index.tsx +++ b/web/app/components/datasets/create/website/watercrawl/index.tsx @@ -14,6 +14,7 @@ import Toast from '@/app/components/base/toast' import { checkWatercrawlTaskStatus, createWatercrawlTask } from '@/service/datasets' import { sleep } from '@/utils' import Header from '../base/header' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const ERROR_I18N_PREFIX = 'common.errorMsg' const I18N_PREFIX = 'datasetCreation.stepOne.website' @@ -51,7 +52,7 @@ const WaterCrawl: FC = ({ const { setShowAccountSettingModal } = useModalContext() const handleSetting = useCallback(() => { setShowAccountSettingModal({ - payload: 'data-source', + payload: ACCOUNT_SETTING_TAB.DATA_SOURCE, }) }, [setShowAccountSettingModal]) diff --git a/web/app/components/datasets/documents/create-from-pipeline/data-source/local-file/index.tsx b/web/app/components/datasets/documents/create-from-pipeline/data-source/local-file/index.tsx index da47a4664c..868621e1a3 100644 --- a/web/app/components/datasets/documents/create-from-pipeline/data-source/local-file/index.tsx +++ b/web/app/components/datasets/documents/create-from-pipeline/data-source/local-file/index.tsx @@ -8,6 +8,7 @@ import cn from '@/utils/classnames' import type { CustomFile as File, FileItem } from '@/models/datasets' import { ToastContext } from '@/app/components/base/toast' import { upload } from '@/service/base' +import { getFileUploadErrorMessage } from '@/app/components/base/file-uploader/utils' import I18n from '@/context/i18n' import { LanguagesSupported } from '@/i18n-config/language' import { IS_CE_EDITION } from '@/config' @@ -121,6 +122,8 @@ const LocalFile = ({ return isValidType && isValidSize }, [fileUploadConfig, notify, t, ACCEPTS]) + type UploadResult = Awaited> + const fileUpload = useCallback(async (fileItem: FileItem): Promise => { const formData = new FormData() formData.append('file', fileItem.file) @@ -136,10 +139,14 @@ const LocalFile = ({ data: formData, onprogress: onProgress, }, false, undefined, '?source=datasets') - .then((res: File) => { - const completeFile = { + .then((res: UploadResult) => { + const updatedFile = Object.assign({}, fileItem.file, { + id: res.id, + ...(res as Partial), + }) as File + const completeFile: FileItem = { fileID: fileItem.fileID, - file: res, + file: updatedFile, progress: -1, } const index = fileListRef.current.findIndex(item => item.fileID === fileItem.fileID) @@ -148,7 +155,8 @@ const LocalFile = ({ return Promise.resolve({ ...completeFile }) }) .catch((e) => { - notify({ type: 'error', message: e?.response?.code === 'forbidden' ? e?.response?.message : t('datasetCreation.stepOne.uploader.failed') }) + const errorMessage = getFileUploadErrorMessage(e, t('datasetCreation.stepOne.uploader.failed'), t) + notify({ type: 'error', message: errorMessage }) updateFile(fileItem, -2, fileListRef.current) return Promise.resolve({ ...fileItem }) }) @@ -287,7 +295,7 @@ const LocalFile = ({ - {t('datasetCreation.stepOne.uploader.button')} + {notSupportBatchUpload ? t('datasetCreation.stepOne.uploader.buttonSingleFile') : t('datasetCreation.stepOne.uploader.button')} {allowedExtensions.length > 0 && ( )} @@ -296,7 +304,7 @@ const LocalFile = ({
{t('datasetCreation.stepOne.uploader.tip', { size: fileUploadConfig.file_size_limit, supportTypes: supportTypesShowNames, - batchCount: fileUploadConfig.batch_count_limit, + batchCount: notSupportBatchUpload ? 1 : fileUploadConfig.batch_count_limit, })}
{dragging &&
}
diff --git a/web/app/components/datasets/documents/create-from-pipeline/data-source/online-documents/index.tsx b/web/app/components/datasets/documents/create-from-pipeline/data-source/online-documents/index.tsx index f5cbac909d..97d6721e00 100644 --- a/web/app/components/datasets/documents/create-from-pipeline/data-source/online-documents/index.tsx +++ b/web/app/components/datasets/documents/create-from-pipeline/data-source/online-documents/index.tsx @@ -16,6 +16,7 @@ import Title from './title' import { useGetDataSourceAuth } from '@/service/use-datasource' import Loading from '@/app/components/base/loading' import { useDocLink } from '@/context/i18n' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' type OnlineDocumentsProps = { isInPipeline?: boolean @@ -120,7 +121,7 @@ const OnlineDocuments = ({ const handleSetting = useCallback(() => { setShowAccountSettingModal({ - payload: 'data-source', + payload: ACCOUNT_SETTING_TAB.DATA_SOURCE, }) }, [setShowAccountSettingModal]) diff --git a/web/app/components/datasets/documents/create-from-pipeline/data-source/online-drive/index.tsx b/web/app/components/datasets/documents/create-from-pipeline/data-source/online-drive/index.tsx index ed2820675c..da8fd5dcc0 100644 --- a/web/app/components/datasets/documents/create-from-pipeline/data-source/online-drive/index.tsx +++ b/web/app/components/datasets/documents/create-from-pipeline/data-source/online-drive/index.tsx @@ -15,6 +15,7 @@ import { useShallow } from 'zustand/react/shallow' import { useModalContextSelector } from '@/context/modal-context' import { useGetDataSourceAuth } from '@/service/use-datasource' import { useDocLink } from '@/context/i18n' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' type OnlineDriveProps = { nodeId: string @@ -180,7 +181,7 @@ const OnlineDrive = ({ const handleSetting = useCallback(() => { setShowAccountSettingModal({ - payload: 'data-source', + payload: ACCOUNT_SETTING_TAB.DATA_SOURCE, }) }, [setShowAccountSettingModal]) diff --git a/web/app/components/datasets/documents/create-from-pipeline/data-source/website-crawl/index.tsx b/web/app/components/datasets/documents/create-from-pipeline/data-source/website-crawl/index.tsx index c46cbdf0f1..648f6a5d93 100644 --- a/web/app/components/datasets/documents/create-from-pipeline/data-source/website-crawl/index.tsx +++ b/web/app/components/datasets/documents/create-from-pipeline/data-source/website-crawl/index.tsx @@ -26,6 +26,7 @@ import { useShallow } from 'zustand/react/shallow' import { useModalContextSelector } from '@/context/modal-context' import { useGetDataSourceAuth } from '@/service/use-datasource' import { useDocLink } from '@/context/i18n' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const I18N_PREFIX = 'datasetCreation.stepOne.website' @@ -139,7 +140,7 @@ const WebsiteCrawl = ({ const handleSetting = useCallback(() => { setShowAccountSettingModal({ - payload: 'data-source', + payload: ACCOUNT_SETTING_TAB.DATA_SOURCE, }) }, [setShowAccountSettingModal]) diff --git a/web/app/components/datasets/documents/detail/batch-modal/csv-uploader.tsx b/web/app/components/datasets/documents/detail/batch-modal/csv-uploader.tsx index 7e8749f0bf..317db84c43 100644 --- a/web/app/components/datasets/documents/detail/batch-modal/csv-uploader.tsx +++ b/web/app/components/datasets/documents/detail/batch-modal/csv-uploader.tsx @@ -12,6 +12,7 @@ import { ToastContext } from '@/app/components/base/toast' import Button from '@/app/components/base/button' import type { FileItem } from '@/models/datasets' import { upload } from '@/service/base' +import { getFileUploadErrorMessage } from '@/app/components/base/file-uploader/utils' import useSWR from 'swr' import { fetchFileUploadConfig } from '@/service/common' import SimplePieChart from '@/app/components/base/simple-pie-chart' @@ -38,6 +39,8 @@ const CSVUploader: FC = ({ file_size_limit: 15, }, [fileUploadConfigResponse]) + type UploadResult = Awaited> + const fileUpload = useCallback(async (fileItem: FileItem): Promise => { fileItem.progress = 0 @@ -58,17 +61,22 @@ const CSVUploader: FC = ({ data: formData, onprogress: onProgress, }, false, undefined, '?source=datasets') - .then((res: File) => { - const completeFile = { + .then((res: UploadResult) => { + const updatedFile = Object.assign({}, fileItem.file, { + id: res.id, + ...(res as Partial), + }) as File + const completeFile: FileItem = { fileID: fileItem.fileID, - file: res, + file: updatedFile, progress: 100, } updateFile(completeFile) return Promise.resolve({ ...completeFile }) }) .catch((e) => { - notify({ type: 'error', message: e?.response?.code === 'forbidden' ? e?.response?.message : t('datasetCreation.stepOne.uploader.failed') }) + const errorMessage = getFileUploadErrorMessage(e, t('datasetCreation.stepOne.uploader.failed'), t) + notify({ type: 'error', message: errorMessage }) const errorFile = { ...fileItem, progress: -2, diff --git a/web/app/components/datasets/documents/detail/completed/common/regeneration-modal.tsx b/web/app/components/datasets/documents/detail/completed/common/regeneration-modal.tsx index 95bb339db9..f90fd7ac60 100644 --- a/web/app/components/datasets/documents/detail/completed/common/regeneration-modal.tsx +++ b/web/app/components/datasets/documents/detail/completed/common/regeneration-modal.tsx @@ -121,7 +121,7 @@ const RegenerationModal: FC = ({ }) return ( - + {!loading && !updateSucceeded && } {loading && !updateSucceeded && } {!loading && updateSucceeded && } diff --git a/web/app/components/datasets/documents/detail/completed/index.tsx b/web/app/components/datasets/documents/detail/completed/index.tsx index 8fa167f976..09c63d54a1 100644 --- a/web/app/components/datasets/documents/detail/completed/index.tsx +++ b/web/app/components/datasets/documents/detail/completed/index.tsx @@ -124,6 +124,7 @@ const Completed: FC = ({ const [limit, setLimit] = useState(DEFAULT_LIMIT) const [fullScreen, setFullScreen] = useState(false) const [showNewChildSegmentModal, setShowNewChildSegmentModal] = useState(false) + const [isRegenerationModalOpen, setIsRegenerationModalOpen] = useState(false) const segmentListRef = useRef(null) const childSegmentListRef = useRef(null) @@ -669,6 +670,7 @@ const Completed: FC = ({ onClose={onCloseSegmentDetail} showOverlay={false} needCheckChunks + modal={isRegenerationModalOpen} > = ({ isEditMode={currSegment.isEditMode} onUpdate={handleUpdateSegment} onCancel={onCloseSegmentDetail} + onModalStateChange={setIsRegenerationModalOpen} /> {/* Create New Segment */} diff --git a/web/app/components/datasets/documents/detail/completed/segment-detail.tsx b/web/app/components/datasets/documents/detail/completed/segment-detail.tsx index bbd9df1adc..5e5ae6b485 100644 --- a/web/app/components/datasets/documents/detail/completed/segment-detail.tsx +++ b/web/app/components/datasets/documents/detail/completed/segment-detail.tsx @@ -27,6 +27,7 @@ type ISegmentDetailProps = { onCancel: () => void isEditMode?: boolean docForm: ChunkingMode + onModalStateChange?: (isOpen: boolean) => void } /** @@ -38,6 +39,7 @@ const SegmentDetail: FC = ({ onCancel, isEditMode, docForm, + onModalStateChange, }) => { const { t } = useTranslation() const [question, setQuestion] = useState(isEditMode ? segInfo?.content || '' : segInfo?.sign_content || '') @@ -68,11 +70,19 @@ const SegmentDetail: FC = ({ const handleRegeneration = useCallback(() => { setShowRegenerationModal(true) - }, []) + onModalStateChange?.(true) + }, [onModalStateChange]) const onCancelRegeneration = useCallback(() => { setShowRegenerationModal(false) - }, []) + onModalStateChange?.(false) + }, [onModalStateChange]) + + const onCloseAfterRegeneration = useCallback(() => { + setShowRegenerationModal(false) + onModalStateChange?.(false) + onCancel() // Close the edit drawer + }, [onCancel, onModalStateChange]) const onConfirmRegeneration = useCallback(() => { onUpdate(segInfo?.id || '', question, answer, keywords, true) @@ -161,7 +171,7 @@ const SegmentDetail: FC = ({ isShow={showRegenerationModal} onConfirm={onConfirmRegeneration} onCancel={onCancelRegeneration} - onClose={onCancelRegeneration} + onClose={onCloseAfterRegeneration} /> ) } diff --git a/web/app/components/datasets/documents/detail/index.tsx b/web/app/components/datasets/documents/detail/index.tsx index b4f47253fb..ddec9b6dbe 100644 --- a/web/app/components/datasets/documents/detail/index.tsx +++ b/web/app/components/datasets/documents/detail/index.tsx @@ -17,7 +17,7 @@ import Divider from '@/app/components/base/divider' import Loading from '@/app/components/base/loading' import Toast from '@/app/components/base/toast' import { ChunkingMode } from '@/models/datasets' -import type { FileItem } from '@/models/datasets' +import type { DataSourceInfo, FileItem, LegacyDataSourceInfo } from '@/models/datasets' import { useDatasetDetailContextWithSelector } from '@/context/dataset-detail' import FloatRightContainer from '@/app/components/base/float-right-container' import useBreakpoints, { MediaType } from '@/hooks/use-breakpoints' @@ -109,6 +109,18 @@ const DocumentDetail: FC = ({ datasetId, documentId }) => { const embedding = ['queuing', 'indexing', 'paused'].includes((documentDetail?.display_status || '').toLowerCase()) + const isLegacyDataSourceInfo = (info?: DataSourceInfo): info is LegacyDataSourceInfo => { + return !!info && 'upload_file' in info + } + + const documentUploadFile = useMemo(() => { + if (!documentDetail?.data_source_info) + return undefined + if (isLegacyDataSourceInfo(documentDetail.data_source_info)) + return documentDetail.data_source_info.upload_file + return undefined + }, [documentDetail?.data_source_info]) + const invalidChunkList = useInvalid(useSegmentListKey) const invalidChildChunkList = useInvalid(useChildSegmentListKey) const invalidDocumentList = useInvalidDocumentList(datasetId) @@ -153,7 +165,7 @@ const DocumentDetail: FC = ({ datasetId, documentId }) => {
void } +type MetadataState = { + documentType?: DocType | '' + metadata: Record +} + const Metadata: FC = ({ docDetail, loading, onUpdate }) => { const { doc_metadata = {} } = docDetail || {} - const doc_type = docDetail?.doc_type || '' + const rawDocType = docDetail?.doc_type ?? '' + const doc_type = rawDocType === 'others' ? '' : rawDocType const { t } = useTranslation() const metadataMap = useMetadataMap() @@ -143,18 +149,16 @@ const Metadata: FC = ({ docDetail, loading, onUpdate }) => { const businessDocCategoryMap = useBusinessDocCategories() const [editStatus, setEditStatus] = useState(!doc_type) // if no documentType, in editing status by default // the initial values are according to the documentType - const [metadataParams, setMetadataParams] = useState<{ - documentType?: DocType | '' - metadata: { [key: string]: string } - }>( + const [metadataParams, setMetadataParams] = useState( doc_type ? { - documentType: doc_type, - metadata: doc_metadata || {}, + documentType: doc_type as DocType, + metadata: (doc_metadata || {}) as Record, } - : { metadata: {} }) + : { metadata: {} }, + ) const [showDocTypes, setShowDocTypes] = useState(!doc_type) // whether show doc types - const [tempDocType, setTempDocType] = useState('') // for remember icon click + const [tempDocType, setTempDocType] = useState('') // for remember icon click const [saveLoading, setSaveLoading] = useState(false) const { notify } = useContext(ToastContext) @@ -165,13 +169,13 @@ const Metadata: FC = ({ docDetail, loading, onUpdate }) => { if (docDetail?.doc_type) { setEditStatus(false) setShowDocTypes(false) - setTempDocType(docDetail?.doc_type) + setTempDocType(doc_type as DocType | '') setMetadataParams({ - documentType: docDetail?.doc_type, - metadata: docDetail?.doc_metadata || {}, + documentType: doc_type as DocType | '', + metadata: (docDetail?.doc_metadata || {}) as Record, }) } - }, [docDetail?.doc_type]) + }, [docDetail?.doc_type, docDetail?.doc_metadata, doc_type]) // confirm doc type const confirmDocType = () => { @@ -179,7 +183,7 @@ const Metadata: FC = ({ docDetail, loading, onUpdate }) => { return setMetadataParams({ documentType: tempDocType, - metadata: tempDocType === metadataParams.documentType ? metadataParams.metadata : {}, // change doc type, clear metadata + metadata: tempDocType === metadataParams.documentType ? metadataParams.metadata : {} as Record, // change doc type, clear metadata }) setEditStatus(true) setShowDocTypes(false) @@ -187,7 +191,7 @@ const Metadata: FC = ({ docDetail, loading, onUpdate }) => { // cancel doc type const cancelDocType = () => { - setTempDocType(metadataParams.documentType) + setTempDocType(metadataParams.documentType ?? '') setEditStatus(true) setShowDocTypes(false) } @@ -209,7 +213,7 @@ const Metadata: FC = ({ docDetail, loading, onUpdate }) => { {t('datasetDocuments.metadata.docTypeChangeTitle')} {t('datasetDocuments.metadata.docTypeSelectWarning')} } - + {CUSTOMIZABLE_DOC_TYPES.map((type, index) => { const currValue = tempDocType ?? documentType return diff --git a/web/app/components/datasets/documents/detail/settings/document-settings.tsx b/web/app/components/datasets/documents/detail/settings/document-settings.tsx index 048645c9cf..3bcb8ef3aa 100644 --- a/web/app/components/datasets/documents/detail/settings/document-settings.tsx +++ b/web/app/components/datasets/documents/detail/settings/document-settings.tsx @@ -4,7 +4,17 @@ import { useBoolean } from 'ahooks' import { useContext } from 'use-context-selector' import { useRouter } from 'next/navigation' import DatasetDetailContext from '@/context/dataset-detail' -import type { CrawlOptions, CustomFile, DataSourceType } from '@/models/datasets' +import type { + CrawlOptions, + CustomFile, + DataSourceInfo, + DataSourceType, + LegacyDataSourceInfo, + LocalFileInfo, + OnlineDocumentInfo, + WebsiteCrawlInfo, +} from '@/models/datasets' +import type { DataSourceProvider } from '@/models/common' import Loading from '@/app/components/base/loading' import StepTwo from '@/app/components/datasets/create/step-two' import AccountSetting from '@/app/components/header/account-setting' @@ -42,15 +52,78 @@ const DocumentSettings = ({ datasetId, documentId }: DocumentSettingsProps) => { params: { metadata: 'without' }, }) + const dataSourceInfo = documentDetail?.data_source_info + + const isLegacyDataSourceInfo = (info: DataSourceInfo | undefined): info is LegacyDataSourceInfo => { + return !!info && 'upload_file' in info + } + const isWebsiteCrawlInfo = (info: DataSourceInfo | undefined): info is WebsiteCrawlInfo => { + return !!info && 'source_url' in info && 'title' in info + } + const isOnlineDocumentInfo = (info: DataSourceInfo | undefined): info is OnlineDocumentInfo => { + return !!info && 'page' in info + } + const isLocalFileInfo = (info: DataSourceInfo | undefined): info is LocalFileInfo => { + return !!info && 'related_id' in info && 'transfer_method' in info + } + const legacyInfo = isLegacyDataSourceInfo(dataSourceInfo) ? dataSourceInfo : undefined + const websiteInfo = isWebsiteCrawlInfo(dataSourceInfo) ? dataSourceInfo : undefined + const onlineDocumentInfo = isOnlineDocumentInfo(dataSourceInfo) ? dataSourceInfo : undefined + const localFileInfo = isLocalFileInfo(dataSourceInfo) ? dataSourceInfo : undefined + const currentPage = useMemo(() => { - return { - workspace_id: documentDetail?.data_source_info.notion_workspace_id, - page_id: documentDetail?.data_source_info.notion_page_id, - page_name: documentDetail?.name, - page_icon: documentDetail?.data_source_info.notion_page_icon, - type: documentDetail?.data_source_type, + if (legacyInfo) { + return { + workspace_id: legacyInfo.notion_workspace_id ?? '', + page_id: legacyInfo.notion_page_id ?? '', + page_name: documentDetail?.name, + page_icon: legacyInfo.notion_page_icon, + type: documentDetail?.data_source_type, + } } - }, [documentDetail]) + if (onlineDocumentInfo) { + return { + workspace_id: onlineDocumentInfo.workspace_id, + page_id: onlineDocumentInfo.page.page_id, + page_name: onlineDocumentInfo.page.page_name, + page_icon: onlineDocumentInfo.page.page_icon, + type: onlineDocumentInfo.page.type, + } + } + return undefined + }, [documentDetail?.data_source_type, documentDetail?.name, legacyInfo, onlineDocumentInfo]) + + const files = useMemo(() => { + if (legacyInfo?.upload_file) + return [legacyInfo.upload_file as CustomFile] + if (localFileInfo) { + const { related_id, name, extension } = localFileInfo + return [{ + id: related_id, + name, + extension, + } as unknown as CustomFile] + } + return [] + }, [legacyInfo?.upload_file, localFileInfo]) + + const websitePages = useMemo(() => { + if (!websiteInfo) + return [] + return [{ + title: websiteInfo.title, + source_url: websiteInfo.source_url, + content: websiteInfo.content, + description: websiteInfo.description, + }] + }, [websiteInfo]) + + const crawlOptions = (dataSourceInfo && typeof dataSourceInfo === 'object' && 'includes' in dataSourceInfo && 'excludes' in dataSourceInfo) + ? dataSourceInfo as unknown as CrawlOptions + : undefined + + const websiteCrawlProvider = (websiteInfo?.provider ?? legacyInfo?.provider) as DataSourceProvider | undefined + const websiteCrawlJobId = websiteInfo?.job_id ?? legacyInfo?.job_id if (error) return @@ -65,22 +138,16 @@ const DocumentSettings = ({ datasetId, documentId }: DocumentSettingsProps) => { onSetting={showSetAPIKey} datasetId={datasetId} dataSourceType={documentDetail.data_source_type as DataSourceType} - notionPages={[currentPage as unknown as NotionPage]} - websitePages={[ - { - title: documentDetail.name, - source_url: documentDetail.data_source_info?.url, - content: '', - description: '', - }, - ]} - websiteCrawlProvider={documentDetail.data_source_info?.provider} - websiteCrawlJobId={documentDetail.data_source_info?.job_id} - crawlOptions={documentDetail.data_source_info as unknown as CrawlOptions} + notionPages={currentPage ? [currentPage as unknown as NotionPage] : []} + notionCredentialId={legacyInfo?.credential_id || onlineDocumentInfo?.credential_id || ''} + websitePages={websitePages} + websiteCrawlProvider={websiteCrawlProvider} + websiteCrawlJobId={websiteCrawlJobId || ''} + crawlOptions={crawlOptions} indexingType={indexingTechnique} isSetting documentDetail={documentDetail} - files={[documentDetail.data_source_info.upload_file as CustomFile]} + files={files} onSave={saveHandler} onCancel={cancelHandler} /> diff --git a/web/app/components/datasets/settings/form/index.tsx b/web/app/components/datasets/settings/form/index.tsx index c6a5da9699..cd7a60b817 100644 --- a/web/app/components/datasets/settings/form/index.tsx +++ b/web/app/components/datasets/settings/form/index.tsx @@ -146,8 +146,8 @@ const Form = () => { return } if (retrievalConfig.weights) { - retrievalConfig.weights.vector_setting.embedding_provider_name = currentDataset?.embedding_model_provider || '' - retrievalConfig.weights.vector_setting.embedding_model_name = currentDataset?.embedding_model || '' + retrievalConfig.weights.vector_setting.embedding_provider_name = embeddingModel.provider || '' + retrievalConfig.weights.vector_setting.embedding_model_name = embeddingModel.model || '' } try { setLoading(true) diff --git a/web/app/components/develop/doc.tsx b/web/app/components/develop/doc.tsx index 82b6b00e44..28a3219535 100644 --- a/web/app/components/develop/doc.tsx +++ b/web/app/components/develop/doc.tsx @@ -18,7 +18,7 @@ import TemplateChatJa from './template/template_chat.ja.mdx' import I18n from '@/context/i18n' import { LanguagesSupported } from '@/i18n-config/language' import useTheme from '@/hooks/use-theme' -import { Theme } from '@/types/app' +import { AppModeEnum, Theme } from '@/types/app' import cn from '@/utils/classnames' type IDocProps = { @@ -115,7 +115,7 @@ const Doc = ({ appDetail }: IDocProps) => { } const Template = useMemo(() => { - if (appDetail?.mode === 'chat' || appDetail?.mode === 'agent-chat') { + if (appDetail?.mode === AppModeEnum.CHAT || appDetail?.mode === AppModeEnum.AGENT_CHAT) { switch (locale) { case LanguagesSupported[1]: return @@ -125,7 +125,7 @@ const Doc = ({ appDetail }: IDocProps) => { return } } - if (appDetail?.mode === 'advanced-chat') { + if (appDetail?.mode === AppModeEnum.ADVANCED_CHAT) { switch (locale) { case LanguagesSupported[1]: return @@ -135,7 +135,7 @@ const Doc = ({ appDetail }: IDocProps) => { return } } - if (appDetail?.mode === 'workflow') { + if (appDetail?.mode === AppModeEnum.WORKFLOW) { switch (locale) { case LanguagesSupported[1]: return @@ -145,7 +145,7 @@ const Doc = ({ appDetail }: IDocProps) => { return } } - if (appDetail?.mode === 'completion') { + if (appDetail?.mode === AppModeEnum.COMPLETION) { switch (locale) { case LanguagesSupported[1]: return diff --git a/web/app/components/develop/template/template_chat.en.mdx b/web/app/components/develop/template/template_chat.en.mdx index df31177127..1e4e767d6d 100644 --- a/web/app/components/develop/template/template_chat.en.mdx +++ b/web/app/components/develop/template/template_chat.en.mdx @@ -74,7 +74,8 @@ Chat applications support session persistence, allowing previous chat history to If set to `false`, can achieve async title generation by calling the conversation rename API and setting `auto_generate` to `true`. - (Optional) Workflow ID to specify a specific version, if not provided, uses the default published version. + (Optional) Workflow ID to specify a specific version, if not provided, uses the default published version.
+ How to obtain: In the version history interface, click the copy icon on the right side of each version entry to copy the complete workflow ID.
(Optional) Trace ID. Used for integration with existing business trace components to achieve end-to-end distributed tracing. If not provided, the system will automatically generate a trace_id. Supports the following three ways to pass, in order of priority:
diff --git a/web/app/components/develop/template/template_chat.ja.mdx b/web/app/components/develop/template/template_chat.ja.mdx index eafa653cad..6ba80d8890 100644 --- a/web/app/components/develop/template/template_chat.ja.mdx +++ b/web/app/components/develop/template/template_chat.ja.mdx @@ -74,7 +74,8 @@ import { Row, Col, Properties, Property, Heading, SubProperty, Paragraph } from `false`に設定すると、会話のリネームAPIを呼び出し、`auto_generate`を`true`に設定することで非同期タイトル生成を実現できます。
- (オプション)ワークフローID、特定のバージョンを指定するために使用、提供されない場合はデフォルトの公開バージョンを使用。 + (オプション)ワークフローID、特定のバージョンを指定するために使用、提供されない場合はデフォルトの公開バージョンを使用。
+ 取得方法:バージョン履歴インターフェースで、各バージョンエントリの右側にあるコピーアイコンをクリックすると、完全なワークフローIDをコピーできます。
(オプション)トレースID。既存の業務システムのトレースコンポーネントと連携し、エンドツーエンドの分散トレーシングを実現するために使用します。指定がない場合、システムが自動的に trace_id を生成します。以下の3つの方法で渡すことができ、優先順位は次のとおりです:
diff --git a/web/app/components/develop/template/template_chat.zh.mdx b/web/app/components/develop/template/template_chat.zh.mdx index fc3fd6d0d2..bf69be2c28 100644 --- a/web/app/components/develop/template/template_chat.zh.mdx +++ b/web/app/components/develop/template/template_chat.zh.mdx @@ -72,7 +72,8 @@ import { Row, Col, Properties, Property, Heading, SubProperty } from '../md.tsx' (选填)自动生成标题,默认 `true`。 若设置为 `false`,则可通过调用会话重命名接口并设置 `auto_generate` 为 `true` 实现异步生成标题。
- (选填)工作流ID,用于指定特定版本,如果不提供则使用默认的已发布版本。 + (选填)工作流ID,用于指定特定版本,如果不提供则使用默认的已发布版本。
+ 获取方式:在版本历史界面,点击每个版本条目右侧的复制图标即可复制完整的工作流 ID。
(选填)链路追踪ID。适用于与业务系统已有的trace组件打通,实现端到端分布式追踪等场景。如果未指定,系统会自动生成trace_id。支持以下三种方式传递,具体优先级依次为:
diff --git a/web/app/components/develop/template/template_workflow.ja.mdx b/web/app/components/develop/template/template_workflow.ja.mdx index ff809b8a9c..688aaae9be 100644 --- a/web/app/components/develop/template/template_workflow.ja.mdx +++ b/web/app/components/develop/template/template_workflow.ja.mdx @@ -344,7 +344,7 @@ import { Row, Col, Properties, Property, Heading, SubProperty, Paragraph } from ### パス - `workflow_id` (string) 必須 特定バージョンのワークフローを指定するためのワークフローID - 取得方法:バージョン履歴で特定バージョンのワークフローIDを照会できます。 + 取得方法:バージョン履歴インターフェースで、各バージョンエントリの右側にあるコピーアイコンをクリックすると、完全なワークフローIDをコピーできます。 ### リクエストボディ - `inputs` (object) 必須 diff --git a/web/app/components/develop/template/template_workflow.zh.mdx b/web/app/components/develop/template/template_workflow.zh.mdx index 0e2b19df83..32ad342c71 100644 --- a/web/app/components/develop/template/template_workflow.zh.mdx +++ b/web/app/components/develop/template/template_workflow.zh.mdx @@ -334,7 +334,7 @@ Workflow 应用无会话支持,适合用于翻译/文章写作/总结 AI 等 ### Path - `workflow_id` (string) Required 工作流ID,用于指定特定版本的工作流 - 获取方式:可以在版本历史中查询特定版本的工作流ID。 + 获取方式:在版本历史界面,点击每个版本条目右侧的复制图标即可复制完整的工作流 ID。 ### Request Body - `inputs` (object) Required diff --git a/web/app/components/explore/app-card/index.tsx b/web/app/components/explore/app-card/index.tsx index 5408fd2eb8..6447abf824 100644 --- a/web/app/components/explore/app-card/index.tsx +++ b/web/app/components/explore/app-card/index.tsx @@ -11,6 +11,7 @@ import { RiInformation2Line } from '@remixicon/react' import { useCallback } from 'react' import ExploreContext from '@/context/explore-context' import { useContextSelector } from 'use-context-selector' +import { AppModeEnum } from '@/types/app' export type AppCardProps = { app: App @@ -55,11 +56,11 @@ const AppCard = ({
{appBasicInfo.name}
- {appBasicInfo.mode === 'advanced-chat' &&
{t('app.types.advanced').toUpperCase()}
} - {appBasicInfo.mode === 'chat' &&
{t('app.types.chatbot').toUpperCase()}
} - {appBasicInfo.mode === 'agent-chat' &&
{t('app.types.agent').toUpperCase()}
} - {appBasicInfo.mode === 'workflow' &&
{t('app.types.workflow').toUpperCase()}
} - {appBasicInfo.mode === 'completion' &&
{t('app.types.completion').toUpperCase()}
} + {appBasicInfo.mode === AppModeEnum.ADVANCED_CHAT &&
{t('app.types.advanced').toUpperCase()}
} + {appBasicInfo.mode === AppModeEnum.CHAT &&
{t('app.types.chatbot').toUpperCase()}
} + {appBasicInfo.mode === AppModeEnum.AGENT_CHAT &&
{t('app.types.agent').toUpperCase()}
} + {appBasicInfo.mode === AppModeEnum.WORKFLOW &&
{t('app.types.workflow').toUpperCase()}
} + {appBasicInfo.mode === AppModeEnum.COMPLETION &&
{t('app.types.completion').toUpperCase()}
}
diff --git a/web/app/components/explore/create-app-modal/index.tsx b/web/app/components/explore/create-app-modal/index.tsx index e94999db04..84621858be 100644 --- a/web/app/components/explore/create-app-modal/index.tsx +++ b/web/app/components/explore/create-app-modal/index.tsx @@ -13,7 +13,7 @@ import Toast from '@/app/components/base/toast' import AppIcon from '@/app/components/base/app-icon' import { useProviderContext } from '@/context/provider-context' import AppsFull from '@/app/components/billing/apps-full-in-dialog' -import type { AppIconType } from '@/types/app' +import { type AppIconType, AppModeEnum } from '@/types/app' import { noop } from 'lodash-es' export type CreateAppModalProps = { @@ -158,7 +158,7 @@ const CreateAppModal = ({ />
{/* answer icon */} - {isEditModal && (appMode === 'chat' || appMode === 'advanced-chat' || appMode === 'agent-chat') && ( + {isEditModal && (appMode === AppModeEnum.CHAT || appMode === AppModeEnum.ADVANCED_CHAT || appMode === AppModeEnum.AGENT_CHAT) && (
{t('app.answerIcon.title')}
diff --git a/web/app/components/explore/installed-app/index.tsx b/web/app/components/explore/installed-app/index.tsx index b4321d6336..3b03fd6265 100644 --- a/web/app/components/explore/installed-app/index.tsx +++ b/web/app/components/explore/installed-app/index.tsx @@ -13,6 +13,7 @@ import { useGetUserCanAccessApp } from '@/service/access-control' import { useGetInstalledAppAccessModeByAppId, useGetInstalledAppMeta, useGetInstalledAppParams } from '@/service/use-explore' import type { AppData } from '@/models/share' import type { AccessMode } from '@/models/access-control' +import { AppModeEnum } from '@/types/app' export type IInstalledAppProps = { id: string @@ -103,13 +104,13 @@ const InstalledApp: FC = ({ } return (
- {installedApp?.app.mode !== 'completion' && installedApp?.app.mode !== 'workflow' && ( + {installedApp?.app.mode !== AppModeEnum.COMPLETION && installedApp?.app.mode !== AppModeEnum.WORKFLOW && ( )} - {installedApp?.app.mode === 'completion' && ( + {installedApp?.app.mode === AppModeEnum.COMPLETION && ( )} - {installedApp?.app.mode === 'workflow' && ( + {installedApp?.app.mode === AppModeEnum.WORKFLOW && ( )}
diff --git a/web/app/components/goto-anything/actions/commands/feedback.tsx b/web/app/components/goto-anything/actions/commands/forum.tsx similarity index 54% rename from web/app/components/goto-anything/actions/commands/feedback.tsx rename to web/app/components/goto-anything/actions/commands/forum.tsx index cce0aeb5f4..66237cb348 100644 --- a/web/app/components/goto-anything/actions/commands/feedback.tsx +++ b/web/app/components/goto-anything/actions/commands/forum.tsx @@ -4,27 +4,27 @@ import { RiFeedbackLine } from '@remixicon/react' import i18n from '@/i18n-config/i18next-config' import { registerCommands, unregisterCommands } from './command-bus' -// Feedback command dependency types -type FeedbackDeps = Record +// Forum command dependency types +type ForumDeps = Record /** - * Feedback command - Opens GitHub feedback discussions + * Forum command - Opens Dify community forum */ -export const feedbackCommand: SlashCommandHandler = { - name: 'feedback', - description: 'Open feedback discussions', +export const forumCommand: SlashCommandHandler = { + name: 'forum', + description: 'Open Dify community forum', mode: 'direct', // Direct execution function execute: () => { - const url = 'https://github.com/langgenius/dify/discussions/categories/feedbacks' + const url = 'https://forum.dify.ai' window.open(url, '_blank', 'noopener,noreferrer') }, async search(args: string, locale: string = 'en') { return [{ - id: 'feedback', - title: i18n.t('common.userProfile.communityFeedback', { lng: locale }), + id: 'forum', + title: i18n.t('common.userProfile.forum', { lng: locale }), description: i18n.t('app.gotoAnything.actions.feedbackDesc', { lng: locale }) || 'Open community feedback discussions', type: 'command' as const, icon: ( @@ -32,20 +32,20 @@ export const feedbackCommand: SlashCommandHandler = {
), - data: { command: 'navigation.feedback', args: { url: 'https://github.com/langgenius/dify/discussions/categories/feedbacks' } }, + data: { command: 'navigation.forum', args: { url: 'https://forum.dify.ai' } }, }] }, - register(_deps: FeedbackDeps) { + register(_deps: ForumDeps) { registerCommands({ - 'navigation.feedback': async (args) => { - const url = args?.url || 'https://github.com/langgenius/dify/discussions/categories/feedbacks' + 'navigation.forum': async (args) => { + const url = args?.url || 'https://forum.dify.ai' window.open(url, '_blank', 'noopener,noreferrer') }, }) }, unregister() { - unregisterCommands(['navigation.feedback']) + unregisterCommands(['navigation.forum']) }, } diff --git a/web/app/components/goto-anything/actions/commands/slash.tsx b/web/app/components/goto-anything/actions/commands/slash.tsx index e0d03d5019..b99215255f 100644 --- a/web/app/components/goto-anything/actions/commands/slash.tsx +++ b/web/app/components/goto-anything/actions/commands/slash.tsx @@ -7,7 +7,7 @@ import { useTheme } from 'next-themes' import { setLocaleOnClient } from '@/i18n-config' import { themeCommand } from './theme' import { languageCommand } from './language' -import { feedbackCommand } from './feedback' +import { forumCommand } from './forum' import { docsCommand } from './docs' import { communityCommand } from './community' import { accountCommand } from './account' @@ -34,7 +34,7 @@ export const registerSlashCommands = (deps: Record) => { // Register command handlers to the registry system with their respective dependencies slashCommandRegistry.register(themeCommand, { setTheme: deps.setTheme }) slashCommandRegistry.register(languageCommand, { setLocale: deps.setLocale }) - slashCommandRegistry.register(feedbackCommand, {}) + slashCommandRegistry.register(forumCommand, {}) slashCommandRegistry.register(docsCommand, {}) slashCommandRegistry.register(communityCommand, {}) slashCommandRegistry.register(accountCommand, {}) @@ -44,7 +44,7 @@ export const unregisterSlashCommands = () => { // Remove command handlers from registry system (automatically calls each command's unregister method) slashCommandRegistry.unregister('theme') slashCommandRegistry.unregister('language') - slashCommandRegistry.unregister('feedback') + slashCommandRegistry.unregister('forum') slashCommandRegistry.unregister('docs') slashCommandRegistry.unregister('community') slashCommandRegistry.unregister('account') diff --git a/web/app/components/goto-anything/context.tsx b/web/app/components/goto-anything/context.tsx index fee4b72c91..25fe2ddf96 100644 --- a/web/app/components/goto-anything/context.tsx +++ b/web/app/components/goto-anything/context.tsx @@ -3,6 +3,7 @@ import type { ReactNode } from 'react' import React, { createContext, useContext, useEffect, useState } from 'react' import { usePathname } from 'next/navigation' +import { isInWorkflowPage } from '../workflow/constants' /** * Interface for the GotoAnything context @@ -50,7 +51,7 @@ export const GotoAnythingProvider: React.FC = ({ chil } // Workflow pages: /app/[appId]/workflow or /workflow/[token] (shared) - const isWorkflow = /^\/app\/[^/]+\/workflow$/.test(pathname) || /^\/workflow\/[^/]+$/.test(pathname) + const isWorkflow = isInWorkflowPage() // RAG Pipeline pages: /datasets/[datasetId]/pipeline const isRagPipeline = /^\/datasets\/[^/]+\/pipeline$/.test(pathname) diff --git a/web/app/components/header/account-dropdown/compliance.tsx b/web/app/components/header/account-dropdown/compliance.tsx index b5849682e9..8dc4aeec32 100644 --- a/web/app/components/header/account-dropdown/compliance.tsx +++ b/web/app/components/header/account-dropdown/compliance.tsx @@ -16,6 +16,7 @@ import cn from '@/utils/classnames' import { useProviderContext } from '@/context/provider-context' import { Plan } from '@/app/components/billing/type' import { useModalContext } from '@/context/modal-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' import { getDocDownloadUrl } from '@/service/common' enum DocName { @@ -38,7 +39,7 @@ const UpgradeOrDownload: FC = ({ doc_name }) => { if (isFreePlan) setShowPricingModal() else - setShowAccountSettingModal({ payload: 'billing' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.BILLING }) }, [isFreePlan, setShowAccountSettingModal, setShowPricingModal]) const { isPending, mutate: downloadCompliance } = useMutation({ diff --git a/web/app/components/header/account-dropdown/index.tsx b/web/app/components/header/account-dropdown/index.tsx index 30b2bfdf6f..d00cddc693 100644 --- a/web/app/components/header/account-dropdown/index.tsx +++ b/web/app/components/header/account-dropdown/index.tsx @@ -33,6 +33,7 @@ import cn from '@/utils/classnames' import { useGlobalPublicStore } from '@/context/global-public-context' import { useDocLink } from '@/context/i18n' import { useLogout } from '@/service/use-common' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' export default function AppSelector() { const itemClassName = ` @@ -122,7 +123,7 @@ export default function AppSelector() {
setShowAccountSettingModal({ payload: 'members' })}> + )} onClick={() => setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.MEMBERS })}>
{t('common.userProfile.settings')}
diff --git a/web/app/components/header/account-dropdown/support.tsx b/web/app/components/header/account-dropdown/support.tsx index b165c5fcca..f354cc4ab0 100644 --- a/web/app/components/header/account-dropdown/support.tsx +++ b/web/app/components/header/account-dropdown/support.tsx @@ -1,5 +1,5 @@ import { Menu, MenuButton, MenuItem, MenuItems, Transition } from '@headlessui/react' -import { RiArrowRightSLine, RiArrowRightUpLine, RiChatSmile2Line, RiDiscordLine, RiFeedbackLine, RiMailSendLine, RiQuestionLine } from '@remixicon/react' +import { RiArrowRightSLine, RiArrowRightUpLine, RiChatSmile2Line, RiDiscordLine, RiDiscussLine, RiMailSendLine, RiQuestionLine } from '@remixicon/react' import { Fragment } from 'react' import Link from 'next/link' import { useTranslation } from 'react-i18next' @@ -86,10 +86,10 @@ export default function Support({ closeAccountDropdown }: SupportProps) { className={cn(itemClassName, 'group justify-between', 'data-[active]:bg-state-base-hover', )} - href='https://github.com/langgenius/dify/discussions/categories/feedbacks' + href='https://forum.dify.ai/' target='_blank' rel='noopener noreferrer'> - -
{t('common.userProfile.communityFeedback')}
+ +
{t('common.userProfile.forum')}
diff --git a/web/app/components/header/account-setting/api-based-extension-page/selector.tsx b/web/app/components/header/account-setting/api-based-extension-page/selector.tsx index ce218540ee..549b5e7910 100644 --- a/web/app/components/header/account-setting/api-based-extension-page/selector.tsx +++ b/web/app/components/header/account-setting/api-based-extension-page/selector.tsx @@ -16,6 +16,7 @@ import { } from '@/app/components/base/icons/src/vender/line/arrows' import { useModalContext } from '@/context/modal-context' import { fetchApiBasedExtensionList } from '@/service/common' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' type ApiBasedExtensionSelectorProps = { value: string @@ -83,7 +84,7 @@ const ApiBasedExtensionSelector: FC = ({ className='flex cursor-pointer items-center text-xs text-text-accent' onClick={() => { setOpen(false) - setShowAccountSettingModal({ payload: 'api-based-extension' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.API_BASED_EXTENSION }) }} > {t('common.apiBasedExtension.selector.manage')} diff --git a/web/app/components/header/account-setting/constants.ts b/web/app/components/header/account-setting/constants.ts new file mode 100644 index 0000000000..2bf2f2eff5 --- /dev/null +++ b/web/app/components/header/account-setting/constants.ts @@ -0,0 +1,21 @@ +export const ACCOUNT_SETTING_MODAL_ACTION = 'showSettings' + +export const ACCOUNT_SETTING_TAB = { + PROVIDER: 'provider', + MEMBERS: 'members', + BILLING: 'billing', + DATA_SOURCE: 'data-source', + API_BASED_EXTENSION: 'api-based-extension', + CUSTOM: 'custom', + LANGUAGE: 'language', +} as const + +export type AccountSettingTab = typeof ACCOUNT_SETTING_TAB[keyof typeof ACCOUNT_SETTING_TAB] + +export const DEFAULT_ACCOUNT_SETTING_TAB = ACCOUNT_SETTING_TAB.MEMBERS + +export const isValidAccountSettingTab = (tab: string | null): tab is AccountSettingTab => { + if (!tab) + return false + return Object.values(ACCOUNT_SETTING_TAB).includes(tab as AccountSettingTab) +} diff --git a/web/app/components/header/account-setting/data-source-page-new/card.tsx b/web/app/components/header/account-setting/data-source-page-new/card.tsx index 7a8790e76d..1e2e60bb7a 100644 --- a/web/app/components/header/account-setting/data-source-page-new/card.tsx +++ b/web/app/components/header/account-setting/data-source-page-new/card.tsx @@ -20,6 +20,7 @@ import { useDataSourceAuthUpdate } from './hooks' import Confirm from '@/app/components/base/confirm' import { useGetDataSourceOAuthUrl } from '@/service/use-datasource' import { openOAuthPopup } from '@/hooks/use-oauth' +import { CollectionType } from '@/app/components/tools/types' type CardProps = { item: DataSourceAuth @@ -42,6 +43,7 @@ const Card = ({ const pluginPayload = { category: AuthCategory.datasource, provider: `${item.plugin_id}/${item.name}`, + providerType: CollectionType.datasource, } const { handleAuthUpdate } = useDataSourceAuthUpdate({ pluginId: item.plugin_id, diff --git a/web/app/components/header/account-setting/data-source-page-new/hooks/use-marketplace-all-plugins.ts b/web/app/components/header/account-setting/data-source-page-new/hooks/use-marketplace-all-plugins.ts index e4d9ba8950..01790d7002 100644 --- a/web/app/components/header/account-setting/data-source-page-new/hooks/use-marketplace-all-plugins.ts +++ b/web/app/components/header/account-setting/data-source-page-new/hooks/use-marketplace-all-plugins.ts @@ -8,7 +8,7 @@ import { useMarketplacePlugins, } from '@/app/components/plugins/marketplace/hooks' import type { Plugin } from '@/app/components/plugins/types' -import { PluginType } from '@/app/components/plugins/types' +import { PluginCategoryEnum } from '@/app/components/plugins/types' import { getMarketplacePluginsByCollectionId } from '@/app/components/plugins/marketplace/utils' export const useMarketplaceAllPlugins = (providers: any[], searchText: string) => { @@ -38,7 +38,7 @@ export const useMarketplaceAllPlugins = (providers: any[], searchText: string) = if (searchText) { queryPluginsWithDebounced({ query: searchText, - category: PluginType.datasource, + category: PluginCategoryEnum.datasource, exclude, type: 'plugin', sortBy: 'install_count', @@ -48,7 +48,7 @@ export const useMarketplaceAllPlugins = (providers: any[], searchText: string) = else { queryPlugins({ query: '', - category: PluginType.datasource, + category: PluginCategoryEnum.datasource, type: 'plugin', pageSize: 1000, exclude, diff --git a/web/app/components/header/account-setting/index.tsx b/web/app/components/header/account-setting/index.tsx index 8e71597e9c..49f6f62a08 100644 --- a/web/app/components/header/account-setting/index.tsx +++ b/web/app/components/header/account-setting/index.tsx @@ -31,6 +31,10 @@ import { useProviderContext } from '@/context/provider-context' import { useAppContext } from '@/context/app-context' import MenuDialog from '@/app/components/header/account-setting/menu-dialog' import Input from '@/app/components/base/input' +import { + ACCOUNT_SETTING_TAB, + type AccountSettingTab, +} from '@/app/components/header/account-setting/constants' const iconClassName = ` w-5 h-5 mr-2 @@ -38,11 +42,12 @@ const iconClassName = ` type IAccountSettingProps = { onCancel: () => void - activeTab?: string + activeTab?: AccountSettingTab + onTabChange?: (tab: AccountSettingTab) => void } type GroupItem = { - key: string + key: AccountSettingTab name: string description?: string icon: React.JSX.Element @@ -51,56 +56,71 @@ type GroupItem = { export default function AccountSetting({ onCancel, - activeTab = 'members', + activeTab = ACCOUNT_SETTING_TAB.MEMBERS, + onTabChange, }: IAccountSettingProps) { - const [activeMenu, setActiveMenu] = useState(activeTab) + const [activeMenu, setActiveMenu] = useState(activeTab) + useEffect(() => { + setActiveMenu(activeTab) + }, [activeTab]) const { t } = useTranslation() const { enableBilling, enableReplaceWebAppLogo } = useProviderContext() const { isCurrentWorkspaceDatasetOperator } = useAppContext() - const workplaceGroupItems = (() => { + const workplaceGroupItems: GroupItem[] = (() => { if (isCurrentWorkspaceDatasetOperator) return [] - return [ + + const items: GroupItem[] = [ { - key: 'provider', + key: ACCOUNT_SETTING_TAB.PROVIDER, name: t('common.settings.provider'), icon: , activeIcon: , }, { - key: 'members', + key: ACCOUNT_SETTING_TAB.MEMBERS, name: t('common.settings.members'), icon: , activeIcon: , }, - { - // Use key false to hide this item - key: enableBilling ? 'billing' : false, + ] + + if (enableBilling) { + items.push({ + key: ACCOUNT_SETTING_TAB.BILLING, name: t('common.settings.billing'), description: t('billing.plansCommon.receiptInfo'), icon: , activeIcon: , - }, + }) + } + + items.push( { - key: 'data-source', + key: ACCOUNT_SETTING_TAB.DATA_SOURCE, name: t('common.settings.dataSource'), icon: , activeIcon: , }, { - key: 'api-based-extension', + key: ACCOUNT_SETTING_TAB.API_BASED_EXTENSION, name: t('common.settings.apiBasedExtension'), icon: , activeIcon: , }, - { - key: (enableReplaceWebAppLogo || enableBilling) ? 'custom' : false, + ) + + if (enableReplaceWebAppLogo || enableBilling) { + items.push({ + key: ACCOUNT_SETTING_TAB.CUSTOM, name: t('custom.custom'), icon: , activeIcon: , - }, - ].filter(item => !!item.key) as GroupItem[] + }) + } + + return items })() const media = useBreakpoints() @@ -117,7 +137,7 @@ export default function AccountSetting({ name: t('common.settings.generalGroup'), items: [ { - key: 'language', + key: ACCOUNT_SETTING_TAB.LANGUAGE, name: t('common.settings.language'), icon: , activeIcon: , @@ -167,7 +187,10 @@ export default function AccountSetting({ 'mb-0.5 flex h-[37px] cursor-pointer items-center rounded-lg p-1 pl-3 text-sm', activeMenu === item.key ? 'system-sm-semibold bg-state-base-active text-components-menu-item-text-active' : 'system-sm-medium text-components-menu-item-text')} title={item.name} - onClick={() => setActiveMenu(item.key)} + onClick={() => { + setActiveMenu(item.key) + onTabChange?.(item.key) + }} > {activeMenu === item.key ? item.activeIcon : item.icon} {!isMobile &&
{item.name}
} diff --git a/web/app/components/header/account-setting/members-page/invite-modal/index.tsx b/web/app/components/header/account-setting/members-page/invite-modal/index.tsx index 264665805d..a432b8a4f0 100644 --- a/web/app/components/header/account-setting/members-page/invite-modal/index.tsx +++ b/web/app/components/header/account-setting/members-page/invite-modal/index.tsx @@ -17,8 +17,9 @@ import type { InvitationResult } from '@/models/common' import I18n from '@/context/i18n' import 'react-multi-email/dist/style.css' import { noop } from 'lodash-es' - import { useProviderContextSelector } from '@/context/provider-context' +import { useBoolean } from 'ahooks' + type IInviteModalProps = { isEmailSetup: boolean onCancel: () => void @@ -49,9 +50,15 @@ const InviteModal = ({ const { locale } = useContext(I18n) const [role, setRole] = useState('normal') + const [isSubmitting, { + setTrue: setIsSubmitting, + setFalse: setIsSubmitted, + }] = useBoolean(false) + const handleSend = useCallback(async () => { - if (isLimitExceeded) + if (isLimitExceeded || isSubmitting) return + setIsSubmitting() if (emails.map((email: string) => emailRegex.test(email)).every(Boolean)) { try { const { result, invitation_results } = await inviteMember({ @@ -70,7 +77,8 @@ const InviteModal = ({ else { notify({ type: 'error', message: t('common.members.emailInvalid') }) } - }, [isLimitExceeded, emails, role, locale, onCancel, onSend, notify, t]) + setIsSubmitted() + }, [isLimitExceeded, emails, role, locale, onCancel, onSend, notify, t, isSubmitting]) return (
@@ -133,7 +141,7 @@ const InviteModal = ({ tabIndex={0} className='w-full' onClick={handleSend} - disabled={!emails.length || isLimitExceeded} + disabled={!emails.length || isLimitExceeded || isSubmitting} variant='primary' > {t('common.members.sendInvite')} diff --git a/web/app/components/header/account-setting/model-provider-page/declarations.ts b/web/app/components/header/account-setting/model-provider-page/declarations.ts index 62cb1a96e9..134df7b3e8 100644 --- a/web/app/components/header/account-setting/model-provider-page/declarations.ts +++ b/web/app/components/header/account-setting/model-provider-page/declarations.ts @@ -14,7 +14,8 @@ export enum FormTypeEnum { secretInput = 'secret-input', select = 'select', radio = 'radio', - boolean = 'checkbox', + checkbox = 'checkbox', + boolean = 'boolean', files = 'files', file = 'file', modelSelector = 'model-selector', diff --git a/web/app/components/header/account-setting/model-provider-page/hooks.ts b/web/app/components/header/account-setting/model-provider-page/hooks.ts index 48dc609795..8cfd144681 100644 --- a/web/app/components/header/account-setting/model-provider-page/hooks.ts +++ b/web/app/components/header/account-setting/model-provider-page/hooks.ts @@ -35,7 +35,7 @@ import { useMarketplacePlugins, } from '@/app/components/plugins/marketplace/hooks' import type { Plugin } from '@/app/components/plugins/types' -import { PluginType } from '@/app/components/plugins/types' +import { PluginCategoryEnum } from '@/app/components/plugins/types' import { getMarketplacePluginsByCollectionId } from '@/app/components/plugins/marketplace/utils' import { useModalContextSelector } from '@/context/modal-context' import { useEventEmitterContextContext } from '@/context/event-emitter' @@ -278,7 +278,7 @@ export const useMarketplaceAllPlugins = (providers: ModelProvider[], searchText: if (searchText) { queryPluginsWithDebounced({ query: searchText, - category: PluginType.model, + category: PluginCategoryEnum.model, exclude, type: 'plugin', sortBy: 'install_count', @@ -288,7 +288,7 @@ export const useMarketplaceAllPlugins = (providers: ModelProvider[], searchText: else { queryPlugins({ query: '', - category: PluginType.model, + category: PluginCategoryEnum.model, type: 'plugin', pageSize: 1000, exclude, diff --git a/web/app/components/header/account-setting/model-provider-page/index.tsx b/web/app/components/header/account-setting/model-provider-page/index.tsx index 35de29185f..239c462ffe 100644 --- a/web/app/components/header/account-setting/model-provider-page/index.tsx +++ b/web/app/components/header/account-setting/model-provider-page/index.tsx @@ -93,7 +93,7 @@ const ModelProviderPage = ({ searchText }: Props) => { {defaultModelNotConfigured && (
- {t('common.modelProvider.notConfigured')} + {t('common.modelProvider.notConfigured')}
)} { + const text = hasCredential ? t('common.operation.config') : t('common.operation.setup') const Item = ( ) if (notAllowCustomCredential && !hasCredential) { diff --git a/web/app/components/header/account-setting/model-provider-page/model-modal/Form.tsx b/web/app/components/header/account-setting/model-provider-page/model-modal/Form.tsx index 164aeb5bc3..3c51762c52 100644 --- a/web/app/components/header/account-setting/model-provider-page/model-modal/Form.tsx +++ b/web/app/components/header/account-setting/model-provider-page/model-modal/Form.tsx @@ -264,7 +264,7 @@ function Form< ) } - if (formSchema.type === FormTypeEnum.boolean) { + if (formSchema.type === FormTypeEnum.checkbox) { const { variable, label, show_on, required, } = formSchema as CredentialFormSchemaRadio diff --git a/web/app/components/header/account-setting/model-provider-page/model-parameter-modal/index.tsx b/web/app/components/header/account-setting/model-provider-page/model-parameter-modal/index.tsx index 82ba072b94..e56def4113 100644 --- a/web/app/components/header/account-setting/model-provider-page/model-parameter-modal/index.tsx +++ b/web/app/components/header/account-setting/model-provider-page/model-parameter-modal/index.tsx @@ -36,7 +36,6 @@ export type ModelParameterModalProps = { popupClassName?: string portalToFollowElemContentClassName?: string isAdvancedMode: boolean - mode: string modelId: string provider: string setModel: (model: { modelId: string; provider: string; mode?: string; features?: string[] }) => void diff --git a/web/app/components/header/account-setting/model-provider-page/model-selector/index.tsx b/web/app/components/header/account-setting/model-provider-page/model-selector/index.tsx index d28959a509..58e96fde69 100644 --- a/web/app/components/header/account-setting/model-provider-page/model-selector/index.tsx +++ b/web/app/components/header/account-setting/model-provider-page/model-selector/index.tsx @@ -5,6 +5,7 @@ import type { Model, ModelItem, } from '../declarations' +import type { ModelFeatureEnum } from '../declarations' import { useCurrentProviderAndModel } from '../hooks' import ModelTrigger from './model-trigger' import EmptyTrigger from './empty-trigger' @@ -24,7 +25,7 @@ type ModelSelectorProps = { popupClassName?: string onSelect?: (model: DefaultModel) => void readonly?: boolean - scopeFeatures?: string[] + scopeFeatures?: ModelFeatureEnum[] deprecatedClassName?: string showDeprecatedWarnIcon?: boolean } diff --git a/web/app/components/header/account-setting/model-provider-page/model-selector/popup.tsx b/web/app/components/header/account-setting/model-provider-page/model-selector/popup.tsx index ff32b438ed..ae7d863d91 100644 --- a/web/app/components/header/account-setting/model-provider-page/model-selector/popup.tsx +++ b/web/app/components/header/account-setting/model-provider-page/model-selector/popup.tsx @@ -15,6 +15,7 @@ import { useLanguage } from '../hooks' import PopupItem from './popup-item' import { XCircle } from '@/app/components/base/icons/src/vender/solid/general' import { useModalContext } from '@/context/modal-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' import { supportFunctionCall } from '@/utils/tool-call' import { tooltipManager } from '@/app/components/base/tooltip/TooltipManager' @@ -22,7 +23,7 @@ type PopupProps = { defaultModel?: DefaultModel modelList: Model[] onSelect: (provider: string, model: ModelItem) => void - scopeFeatures?: string[] + scopeFeatures?: ModelFeatureEnum[] onHide: () => void } const Popup: FC = ({ @@ -129,7 +130,7 @@ const Popup: FC = ({
{ onHide() - setShowAccountSettingModal({ payload: 'provider' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.PROVIDER }) }}> {t('common.model.settingsLink')} diff --git a/web/app/components/header/account-setting/model-provider-page/provider-added-card/model-list.tsx b/web/app/components/header/account-setting/model-provider-page/provider-added-card/model-list.tsx index 9e26d233c9..2e008a0b35 100644 --- a/web/app/components/header/account-setting/model-provider-page/provider-added-card/model-list.tsx +++ b/web/app/components/header/account-setting/model-provider-page/provider-added-card/model-list.tsx @@ -86,7 +86,7 @@ const ModelList: FC = ({ { models.map(model => ( { return `/app/${app.id}/overview` } else { - if (app.mode === 'workflow' || app.mode === 'advanced-chat') + if (app.mode === AppModeEnum.WORKFLOW || app.mode === AppModeEnum.ADVANCED_CHAT) return `/app/${app.id}/workflow` else return `/app/${app.id}/configuration` diff --git a/web/app/components/header/index.tsx b/web/app/components/header/index.tsx index fc511d2954..ef24d471e0 100644 --- a/web/app/components/header/index.tsx +++ b/web/app/components/header/index.tsx @@ -19,6 +19,7 @@ import PlanBadge from './plan-badge' import LicenseNav from './license-env' import { Plan } from '../billing/type' import { useGlobalPublicStore } from '@/context/global-public-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const navClassName = ` flex items-center relative px-3 h-8 rounded-xl @@ -38,7 +39,7 @@ const Header = () => { if (isFreePlan) setShowPricingModal() else - setShowAccountSettingModal({ payload: 'billing' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.BILLING }) }, [isFreePlan, setShowAccountSettingModal, setShowPricingModal]) if (isMobile) { diff --git a/web/app/components/header/nav/nav-selector/index.tsx b/web/app/components/header/nav/nav-selector/index.tsx index 6c9db287e7..4a13bc8a3c 100644 --- a/web/app/components/header/nav/nav-selector/index.tsx +++ b/web/app/components/header/nav/nav-selector/index.tsx @@ -15,7 +15,7 @@ import { AppTypeIcon } from '@/app/components/app/type-selector' import { useAppContext } from '@/context/app-context' import { useStore as useAppStore } from '@/app/components/app/store' import { FileArrow01, FilePlus01, FilePlus02 } from '@/app/components/base/icons/src/vender/line/files' -import type { AppIconType, AppMode } from '@/types/app' +import type { AppIconType, AppModeEnum } from '@/types/app' export type NavItem = { id: string @@ -25,7 +25,7 @@ export type NavItem = { icon: string icon_background: string | null icon_url: string | null - mode?: AppMode + mode?: AppModeEnum } export type INavSelectorProps = { navigationItems: NavItem[] diff --git a/web/app/components/plugins/card/index.tsx b/web/app/components/plugins/card/index.tsx index b3c09b5bfd..e20aef6220 100644 --- a/web/app/components/plugins/card/index.tsx +++ b/web/app/components/plugins/card/index.tsx @@ -1,21 +1,21 @@ 'use client' -import React from 'react' -import type { Plugin } from '../types' -import Icon from '../card/base/card-icon' -import CornerMark from './base/corner-mark' -import Title from './base/title' -import OrgInfo from './base/org-info' -import Description from './base/description' -import Placeholder from './base/placeholder' -import cn from '@/utils/classnames' -import { useGetLanguage } from '@/context/i18n' -import { getLanguage } from '@/i18n-config/language' -import { useSingleCategories } from '../hooks' -import { renderI18nObject } from '@/i18n-config' import { useMixedTranslation } from '@/app/components/plugins/marketplace/hooks' +import { useGetLanguage } from '@/context/i18n' +import { renderI18nObject } from '@/i18n-config' +import { getLanguage } from '@/i18n-config/language' +import cn from '@/utils/classnames' +import { RiAlertFill } from '@remixicon/react' +import React from 'react' import Partner from '../base/badges/partner' import Verified from '../base/badges/verified' -import { RiAlertFill } from '@remixicon/react' +import Icon from '../card/base/card-icon' +import { useCategories } from '../hooks' +import type { Plugin } from '../types' +import CornerMark from './base/corner-mark' +import Description from './base/description' +import OrgInfo from './base/org-info' +import Placeholder from './base/placeholder' +import Title from './base/title' export type Props = { className?: string @@ -49,10 +49,8 @@ const Card = ({ const defaultLocale = useGetLanguage() const locale = localeFromProps ? getLanguage(localeFromProps) : defaultLocale const { t } = useMixedTranslation(localeFromProps) - const { categoriesMap } = useSingleCategories(t) + const { categoriesMap } = useCategories(t, true) const { category, type, name, org, label, brief, icon, verified, badges = [] } = payload - const isBundle = !['plugin', 'model', 'tool', 'datasource', 'extension', 'agent-strategy'].includes(type) - const cornerMark = isBundle ? categoriesMap.bundle?.label : categoriesMap[category]?.label const getLocalizedText = (obj: Record | undefined) => obj ? renderI18nObject(obj, locale) : '' const isPartner = badges.includes('partner') @@ -70,7 +68,7 @@ const Card = ({ return (
- {!hideCornerMark && } + {!hideCornerMark && } {/* Header */}
diff --git a/web/app/components/plugins/constants.ts b/web/app/components/plugins/constants.ts index 7436611c79..d9203fd4ea 100644 --- a/web/app/components/plugins/constants.ts +++ b/web/app/components/plugins/constants.ts @@ -1,3 +1,5 @@ +import { PluginCategoryEnum } from './types' + export const tagKeys = [ 'agent', 'rag', @@ -20,10 +22,11 @@ export const tagKeys = [ ] export const categoryKeys = [ - 'model', - 'tool', - 'datasource', - 'agent-strategy', - 'extension', + PluginCategoryEnum.model, + PluginCategoryEnum.tool, + PluginCategoryEnum.datasource, + PluginCategoryEnum.agent, + PluginCategoryEnum.extension, 'bundle', + PluginCategoryEnum.trigger, ] diff --git a/web/app/components/plugins/hooks.ts b/web/app/components/plugins/hooks.ts index f22b2c4d69..8303a4cc46 100644 --- a/web/app/components/plugins/hooks.ts +++ b/web/app/components/plugins/hooks.ts @@ -1,10 +1,11 @@ +import type { TFunction } from 'i18next' import { useMemo } from 'react' import { useTranslation } from 'react-i18next' -import type { TFunction } from 'i18next' import { categoryKeys, tagKeys, } from './constants' +import { PluginCategoryEnum } from './types' export type Tag = { name: string @@ -51,56 +52,24 @@ type Category = { label: string } -export const useCategories = (translateFromOut?: TFunction) => { +export const useCategories = (translateFromOut?: TFunction, isSingle?: boolean) => { const { t: translation } = useTranslation() const t = translateFromOut || translation const categories = useMemo(() => { return categoryKeys.map((category) => { - if (category === 'agent-strategy') { + if (category === PluginCategoryEnum.agent) { return { - name: 'agent-strategy', - label: t('plugin.category.agents'), + name: PluginCategoryEnum.agent, + label: isSingle ? t('plugin.categorySingle.agent') : t('plugin.category.agents'), } } return { name: category, - label: t(`plugin.category.${category}s`), + label: isSingle ? t(`plugin.categorySingle.${category}`) : t(`plugin.category.${category}s`), } }) - }, [t]) - - const categoriesMap = useMemo(() => { - return categories.reduce((acc, category) => { - acc[category.name] = category - return acc - }, {} as Record) - }, [categories]) - - return { - categories, - categoriesMap, - } -} - -export const useSingleCategories = (translateFromOut?: TFunction) => { - const { t: translation } = useTranslation() - const t = translateFromOut || translation - - const categories = useMemo(() => { - return categoryKeys.map((category) => { - if (category === 'agent-strategy') { - return { - name: 'agent-strategy', - label: t('plugin.categorySingle.agent'), - } - } - return { - name: category, - label: t(`plugin.categorySingle.${category}`), - } - }) - }, [t]) + }, [t, isSingle]) const categoriesMap = useMemo(() => { return categories.reduce((acc, category) => { diff --git a/web/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list.tsx b/web/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list.tsx index 024444cd6a..264c4782cd 100644 --- a/web/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list.tsx +++ b/web/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list.tsx @@ -2,12 +2,13 @@ import { useModelList } from '@/app/components/header/account-setting/model-prov import { ModelTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' import { useProviderContext } from '@/context/provider-context' import { useInvalidateInstalledPluginList } from '@/service/use-plugins' -import { useInvalidateAllBuiltInTools, useInvalidateAllToolProviders } from '@/service/use-tools' +import { useInvalidateAllBuiltInTools, useInvalidateAllToolProviders, useInvalidateRAGRecommendedPlugins } from '@/service/use-tools' import { useInvalidateStrategyProviders } from '@/service/use-strategy' import type { Plugin, PluginDeclaration, PluginManifestInMarket } from '../../types' -import { PluginType } from '../../types' +import { PluginCategoryEnum } from '../../types' import { useInvalidDataSourceList } from '@/service/use-pipeline' import { useInvalidDataSourceListAuth } from '@/service/use-datasource' +import { useInvalidateAllTriggerPlugins } from '@/service/use-triggers' const useRefreshPluginList = () => { const invalidateInstalledPluginList = useInvalidateInstalledPluginList() @@ -23,25 +24,33 @@ const useRefreshPluginList = () => { const invalidateDataSourceListAuth = useInvalidDataSourceListAuth() const invalidateStrategyProviders = useInvalidateStrategyProviders() + + const invalidateAllTriggerPlugins = useInvalidateAllTriggerPlugins() + + const invalidateRAGRecommendedPlugins = useInvalidateRAGRecommendedPlugins() return { refreshPluginList: (manifest?: PluginManifestInMarket | Plugin | PluginDeclaration | null, refreshAllType?: boolean) => { // installed list invalidateInstalledPluginList() // tool page, tool select - if ((manifest && PluginType.tool.includes(manifest.category)) || refreshAllType) { + if ((manifest && PluginCategoryEnum.tool.includes(manifest.category)) || refreshAllType) { invalidateAllToolProviders() invalidateAllBuiltInTools() + invalidateRAGRecommendedPlugins() // TODO: update suggested tools. It's a function in hook useMarketplacePlugins,handleUpdatePlugins } - if ((manifest && PluginType.datasource.includes(manifest.category)) || refreshAllType) { + if ((manifest && PluginCategoryEnum.trigger.includes(manifest.category)) || refreshAllType) + invalidateAllTriggerPlugins() + + if ((manifest && PluginCategoryEnum.datasource.includes(manifest.category)) || refreshAllType) { invalidateAllDataSources() invalidateDataSourceListAuth() } // model select - if ((manifest && PluginType.model.includes(manifest.category)) || refreshAllType) { + if ((manifest && PluginCategoryEnum.model.includes(manifest.category)) || refreshAllType) { refreshModelProviders() refetchLLMModelList() refetchEmbeddingModelList() @@ -49,7 +58,7 @@ const useRefreshPluginList = () => { } // agent select - if ((manifest && PluginType.agent.includes(manifest.category)) || refreshAllType) + if ((manifest && PluginCategoryEnum.agent.includes(manifest.category)) || refreshAllType) invalidateStrategyProviders() }, } diff --git a/web/app/components/plugins/install-plugin/install-bundle/ready-to-install.tsx b/web/app/components/plugins/install-plugin/install-bundle/ready-to-install.tsx index 63c0b5b07e..b2b0aefb9b 100644 --- a/web/app/components/plugins/install-plugin/install-bundle/ready-to-install.tsx +++ b/web/app/components/plugins/install-plugin/install-bundle/ready-to-install.tsx @@ -4,7 +4,7 @@ import React, { useCallback, useState } from 'react' import { InstallStep } from '../../types' import Install from './steps/install' import Installed from './steps/installed' -import type { Dependency, InstallStatusResponse, Plugin } from '../../types' +import type { Dependency, InstallStatus, Plugin } from '../../types' type Props = { step: InstallStep @@ -26,8 +26,8 @@ const ReadyToInstall: FC = ({ isFromMarketPlace, }) => { const [installedPlugins, setInstalledPlugins] = useState([]) - const [installStatus, setInstallStatus] = useState([]) - const handleInstalled = useCallback((plugins: Plugin[], installStatus: InstallStatusResponse[]) => { + const [installStatus, setInstallStatus] = useState([]) + const handleInstalled = useCallback((plugins: Plugin[], installStatus: InstallStatus[]) => { setInstallStatus(installStatus) setInstalledPlugins(plugins) onStepChange(InstallStep.installed) diff --git a/web/app/components/plugins/install-plugin/install-bundle/steps/install.tsx b/web/app/components/plugins/install-plugin/install-bundle/steps/install.tsx index 758daafca0..a717e0a24a 100644 --- a/web/app/components/plugins/install-plugin/install-bundle/steps/install.tsx +++ b/web/app/components/plugins/install-plugin/install-bundle/steps/install.tsx @@ -2,23 +2,31 @@ import type { FC } from 'react' import { useRef } from 'react' import React, { useCallback, useState } from 'react' -import type { Dependency, InstallStatusResponse, Plugin, VersionInfo } from '../../../types' +import { + type Dependency, + type InstallStatus, + type InstallStatusResponse, + type Plugin, + TaskStatus, + type VersionInfo, +} from '../../../types' import Button from '@/app/components/base/button' import { RiLoader2Line } from '@remixicon/react' import { useTranslation } from 'react-i18next' import type { ExposeRefs } from './install-multi' import InstallMulti from './install-multi' -import { useInstallOrUpdate } from '@/service/use-plugins' +import { useInstallOrUpdate, usePluginTaskList } from '@/service/use-plugins' import useRefreshPluginList from '../../hooks/use-refresh-plugin-list' import { useCanInstallPluginFromMarketplace } from '@/app/components/plugins/plugin-page/use-reference-setting' import { useMittContextSelector } from '@/context/mitt-context' import Checkbox from '@/app/components/base/checkbox' +import checkTaskStatus from '../../base/check-task-status' const i18nPrefix = 'plugin.installModal' type Props = { allPlugins: Dependency[] onStartToInstall?: () => void - onInstalled: (plugins: Plugin[], installStatus: InstallStatusResponse[]) => void + onInstalled: (plugins: Plugin[], installStatus: InstallStatus[]) => void onCancel: () => void isFromMarketPlace?: boolean isHideButton?: boolean @@ -55,18 +63,60 @@ const Install: FC = ({ setCanInstall(true) }, []) + const { + check, + stop, + } = checkTaskStatus() + + const handleCancel = useCallback(() => { + stop() + onCancel() + }, [onCancel, stop]) + + const { handleRefetch } = usePluginTaskList() + // Install from marketplace and github const { mutate: installOrUpdate, isPending: isInstalling } = useInstallOrUpdate({ - onSuccess: (res: InstallStatusResponse[]) => { - onInstalled(selectedPlugins, res.map((r, i) => { - return ({ - ...r, - isFromMarketPlace: allPlugins[selectedIndexes[i]].type === 'marketplace', + onSuccess: async (res: InstallStatusResponse[]) => { + const isAllSettled = res.every(r => r.status === TaskStatus.success || r.status === TaskStatus.failed) + // if all settled, return the install status + if (isAllSettled) { + onInstalled(selectedPlugins, res.map((r, i) => { + return ({ + success: r.status === TaskStatus.success, + isFromMarketPlace: allPlugins[selectedIndexes[i]].type === 'marketplace', + }) + })) + const hasInstallSuccess = res.some(r => r.status === TaskStatus.success) + if (hasInstallSuccess) { + refreshPluginList(undefined, true) + emit('plugin:install:success', selectedPlugins.map((p) => { + return `${p.plugin_id}/${p.name}` + })) + } + return + } + // if not all settled, keep checking the status of the plugins + handleRefetch() + const installStatus = await Promise.all(res.map(async (item, index) => { + if (item.status !== TaskStatus.running) { + return { + success: item.status === TaskStatus.success, + isFromMarketPlace: allPlugins[selectedIndexes[index]].type === 'marketplace', + } + } + const { status } = await check({ + taskId: item.taskId, + pluginUniqueIdentifier: item.uniqueIdentifier, }) + return { + success: status === TaskStatus.success, + isFromMarketPlace: allPlugins[selectedIndexes[index]].type === 'marketplace', + } })) - const hasInstallSuccess = res.some(r => r.success) + onInstalled(selectedPlugins, installStatus) + const hasInstallSuccess = installStatus.some(r => r.success) if (hasInstallSuccess) { - refreshPluginList(undefined, true) emit('plugin:install:success', selectedPlugins.map((p) => { return `${p.plugin_id}/${p.name}` })) @@ -150,7 +200,7 @@ const Install: FC = ({
{!canInstall && ( - )} diff --git a/web/app/components/plugins/install-plugin/install-bundle/steps/installed.tsx b/web/app/components/plugins/install-plugin/install-bundle/steps/installed.tsx index 4e16d200e7..f787882211 100644 --- a/web/app/components/plugins/install-plugin/install-bundle/steps/installed.tsx +++ b/web/app/components/plugins/install-plugin/install-bundle/steps/installed.tsx @@ -1,7 +1,7 @@ 'use client' import type { FC } from 'react' import React from 'react' -import type { InstallStatusResponse, Plugin } from '../../../types' +import type { InstallStatus, Plugin } from '../../../types' import Card from '@/app/components/plugins/card' import Button from '@/app/components/base/button' import { useTranslation } from 'react-i18next' @@ -11,7 +11,7 @@ import { MARKETPLACE_API_PREFIX } from '@/config' type Props = { list: Plugin[] - installStatus: InstallStatusResponse[] + installStatus: InstallStatus[] onCancel: () => void isHideButton?: boolean } diff --git a/web/app/components/plugins/install-plugin/utils.ts b/web/app/components/plugins/install-plugin/utils.ts index f19a7fd287..79c6d7b031 100644 --- a/web/app/components/plugins/install-plugin/utils.ts +++ b/web/app/components/plugins/install-plugin/utils.ts @@ -5,15 +5,17 @@ import { isEmpty } from 'lodash-es' export const pluginManifestToCardPluginProps = (pluginManifest: PluginDeclaration): Plugin => { return { plugin_id: pluginManifest.plugin_unique_identifier, - type: pluginManifest.category, + type: pluginManifest.category as Plugin['type'], category: pluginManifest.category, name: pluginManifest.name, version: pluginManifest.version, latest_version: '', latest_package_identifier: '', org: pluginManifest.author, + author: pluginManifest.author, label: pluginManifest.label, brief: pluginManifest.description, + description: pluginManifest.description, icon: pluginManifest.icon, verified: pluginManifest.verified, introduction: '', @@ -22,14 +24,17 @@ export const pluginManifestToCardPluginProps = (pluginManifest: PluginDeclaratio endpoint: { settings: [], }, - tags: [], + tags: pluginManifest.tags.map(tag => ({ name: tag })), + badges: [], + verification: { authorized_category: 'langgenius' }, + from: 'package', } } export const pluginManifestInMarketToPluginProps = (pluginManifest: PluginManifestInMarket): Plugin => { return { plugin_id: pluginManifest.plugin_unique_identifier, - type: pluginManifest.category, + type: pluginManifest.category as Plugin['type'], category: pluginManifest.category, name: pluginManifest.name, version: pluginManifest.latest_version, @@ -38,6 +43,7 @@ export const pluginManifestInMarketToPluginProps = (pluginManifest: PluginManife org: pluginManifest.org, label: pluginManifest.label, brief: pluginManifest.brief, + description: pluginManifest.brief, icon: pluginManifest.icon, verified: true, introduction: pluginManifest.introduction, @@ -49,6 +55,7 @@ export const pluginManifestInMarketToPluginProps = (pluginManifest: PluginManife tags: [], badges: pluginManifest.badges, verification: isEmpty(pluginManifest.verification) ? { authorized_category: 'langgenius' } : pluginManifest.verification, + from: pluginManifest.from, } } diff --git a/web/app/components/plugins/marketplace/hooks.ts b/web/app/components/plugins/marketplace/hooks.ts index 10aead17c4..5bc9263aaa 100644 --- a/web/app/components/plugins/marketplace/hooks.ts +++ b/web/app/components/plugins/marketplace/hooks.ts @@ -65,10 +65,12 @@ export const useMarketplacePlugins = () => { } = useMutationPluginsFromMarketplace() const [prevPlugins, setPrevPlugins] = useState() + const resetPlugins = useCallback(() => { reset() setPrevPlugins(undefined) }, [reset]) + const handleUpdatePlugins = useCallback((pluginsSearchParams: PluginsSearchParams) => { mutateAsync(pluginsSearchParams).then((res) => { const currentPage = pluginsSearchParams.page || 1 @@ -85,9 +87,6 @@ export const useMarketplacePlugins = () => { } }) }, [mutateAsync]) - const queryPlugins = useCallback((pluginsSearchParams: PluginsSearchParams) => { - handleUpdatePlugins(pluginsSearchParams) - }, [handleUpdatePlugins]) const { run: queryPluginsWithDebounced, cancel: cancelQueryPluginsWithDebounced } = useDebounceFn((pluginsSearchParams: PluginsSearchParams) => { handleUpdatePlugins(pluginsSearchParams) @@ -99,7 +98,7 @@ export const useMarketplacePlugins = () => { plugins: prevPlugins, total: data?.data?.total, resetPlugins, - queryPlugins, + queryPlugins: handleUpdatePlugins, queryPluginsWithDebounced, cancelQueryPluginsWithDebounced, isLoading: isPending, diff --git a/web/app/components/plugins/marketplace/plugin-type-switch.tsx b/web/app/components/plugins/marketplace/plugin-type-switch.tsx index 4a40eb0e06..249be1ef83 100644 --- a/web/app/components/plugins/marketplace/plugin-type-switch.tsx +++ b/web/app/components/plugins/marketplace/plugin-type-switch.tsx @@ -1,4 +1,6 @@ 'use client' +import { Trigger as TriggerIcon } from '@/app/components/base/icons/src/vender/plugin' +import cn from '@/utils/classnames' import { RiArchive2Line, RiBrain2Line, @@ -7,22 +9,22 @@ import { RiPuzzle2Line, RiSpeakAiLine, } from '@remixicon/react' -import { PluginType } from '../types' +import { useCallback, useEffect } from 'react' +import { PluginCategoryEnum } from '../types' import { useMarketplaceContext } from './context' import { useMixedTranslation, useSearchBoxAutoAnimate, } from './hooks' -import cn from '@/utils/classnames' -import { useCallback, useEffect } from 'react' export const PLUGIN_TYPE_SEARCH_MAP = { all: 'all', - model: PluginType.model, - tool: PluginType.tool, - agent: PluginType.agent, - extension: PluginType.extension, - datasource: PluginType.datasource, + model: PluginCategoryEnum.model, + tool: PluginCategoryEnum.tool, + agent: PluginCategoryEnum.agent, + extension: PluginCategoryEnum.extension, + datasource: PluginCategoryEnum.datasource, + trigger: PluginCategoryEnum.trigger, bundle: 'bundle', } type PluginTypeSwitchProps = { @@ -63,6 +65,11 @@ const PluginTypeSwitch = ({ text: t('plugin.category.datasources'), icon: , }, + { + value: PLUGIN_TYPE_SEARCH_MAP.trigger, + text: t('plugin.category.triggers'), + icon: , + }, { value: PLUGIN_TYPE_SEARCH_MAP.agent, text: t('plugin.category.agents'), diff --git a/web/app/components/plugins/marketplace/search-box/index.tsx b/web/app/components/plugins/marketplace/search-box/index.tsx index 0bc214ae1a..c398964b4e 100644 --- a/web/app/components/plugins/marketplace/search-box/index.tsx +++ b/web/app/components/plugins/marketplace/search-box/index.tsx @@ -19,6 +19,7 @@ type SearchBoxProps = { usedInMarketplace?: boolean onShowAddCustomCollectionModal?: () => void onAddedCustomTool?: () => void + autoFocus?: boolean } const SearchBox = ({ search, @@ -32,6 +33,7 @@ const SearchBox = ({ usedInMarketplace = false, supportAddCustomTool, onShowAddCustomCollectionModal, + autoFocus = false, }: SearchBoxProps) => { return (
@@ -82,11 +84,12 @@ const SearchBox = ({ { !usedInMarketplace && ( <> -
+
{ - if (pluginType === PluginType.tool) - return 'category=tool' + if ([PluginCategoryEnum.tool, PluginCategoryEnum.agent, PluginCategoryEnum.model, PluginCategoryEnum.datasource, PluginCategoryEnum.trigger].includes(pluginType as PluginCategoryEnum)) + return `category=${pluginType}` - if (pluginType === PluginType.agent) - return 'category=agent-strategy' - - if (pluginType === PluginType.model) - return 'category=model' - - if (pluginType === PluginType.extension) + if (pluginType === PluginCategoryEnum.extension) return 'category=endpoint' - if (pluginType === PluginType.datasource) - return 'category=datasource' - if (pluginType === 'bundle') return 'type=bundle' diff --git a/web/app/components/plugins/plugin-auth/authorize/api-key-modal.tsx b/web/app/components/plugins/plugin-auth/authorize/api-key-modal.tsx index 95676c656e..cb90b075b0 100644 --- a/web/app/components/plugins/plugin-auth/authorize/api-key-modal.tsx +++ b/web/app/components/plugins/plugin-auth/authorize/api-key-modal.tsx @@ -6,7 +6,6 @@ import { useState, } from 'react' import { useTranslation } from 'react-i18next' -import { Lock01 } from '@/app/components/base/icons/src/vender/solid/security' import Modal from '@/app/components/base/modal/modal' import { CredentialTypeEnum } from '../types' import AuthForm from '@/app/components/base/form/form-scenarios/auth' @@ -23,6 +22,9 @@ import { useGetPluginCredentialSchemaHook, useUpdatePluginCredentialHook, } from '../hooks/use-credential' +import { ReadmeEntrance } from '../../readme-panel/entrance' +import { ReadmeShowType } from '../../readme-panel/store' +import { EncryptedBottom } from '@/app/components/base/encrypted-bottom' export type ApiKeyModalProps = { pluginPayload: PluginPayload @@ -134,25 +136,17 @@ const ApiKeyModal = ({ footerSlot={ (
) } - bottomSlot={ -
- - {t('common.modelProvider.encrypted.front')} - - PKCS1_OAEP - - {t('common.modelProvider.encrypted.back')} -
- } + bottomSlot={} onConfirm={handleConfirm} showExtraButton={!!editValues} onExtraButtonClick={onRemove} disabled={disabled || isLoading || doingAction} + clickOutsideNotClose={true} + wrapperClassName='!z-[101]' > + {pluginPayload.detail && ( + + )} { isLoading && (
diff --git a/web/app/components/plugins/plugin-auth/authorize/oauth-client-settings.tsx b/web/app/components/plugins/plugin-auth/authorize/oauth-client-settings.tsx index c10b06166b..256f6d0f4b 100644 --- a/web/app/components/plugins/plugin-auth/authorize/oauth-client-settings.tsx +++ b/web/app/components/plugins/plugin-auth/authorize/oauth-client-settings.tsx @@ -23,6 +23,8 @@ import type { } from '@/app/components/base/form/types' import { useToastContext } from '@/app/components/base/toast' import Button from '@/app/components/base/button' +import { ReadmeEntrance } from '../../readme-panel/entrance' +import { ReadmeShowType } from '../../readme-panel/store' type OAuthClientSettingsProps = { pluginPayload: PluginPayload @@ -154,16 +156,20 @@ const OAuthClientSettings = ({
) } + containerClassName='pt-0' + wrapperClassName='!z-[101]' + clickOutsideNotClose={true} > - <> - - + {pluginPayload.detail && ( + + )} + ) } diff --git a/web/app/components/plugins/plugin-auth/hooks/use-credential.ts b/web/app/components/plugins/plugin-auth/hooks/use-credential.ts index 5a7a497ad9..9c342f2ced 100644 --- a/web/app/components/plugins/plugin-auth/hooks/use-credential.ts +++ b/web/app/components/plugins/plugin-auth/hooks/use-credential.ts @@ -15,6 +15,7 @@ import { import { useGetApi } from './use-get-api' import type { PluginPayload } from '../types' import type { CredentialTypeEnum } from '../types' +import { useInvalidToolsByType } from '@/service/use-tools' export const useGetPluginCredentialInfoHook = (pluginPayload: PluginPayload, enable?: boolean) => { const apiMap = useGetApi(pluginPayload) @@ -29,8 +30,14 @@ export const useDeletePluginCredentialHook = (pluginPayload: PluginPayload) => { export const useInvalidPluginCredentialInfoHook = (pluginPayload: PluginPayload) => { const apiMap = useGetApi(pluginPayload) + const invalidPluginCredentialInfo = useInvalidPluginCredentialInfo(apiMap.getCredentialInfo) + const providerType = pluginPayload.providerType + const invalidToolsByType = useInvalidToolsByType(providerType) - return useInvalidPluginCredentialInfo(apiMap.getCredentialInfo) + return () => { + invalidPluginCredentialInfo() + invalidToolsByType() + } } export const useSetPluginDefaultCredentialHook = (pluginPayload: PluginPayload) => { diff --git a/web/app/components/plugins/plugin-auth/types.ts b/web/app/components/plugins/plugin-auth/types.ts index 6366c80de3..9974586302 100644 --- a/web/app/components/plugins/plugin-auth/types.ts +++ b/web/app/components/plugins/plugin-auth/types.ts @@ -1,3 +1,6 @@ +import type { CollectionType } from '../../tools/types' +import type { PluginDetail } from '../types' + export type { AddApiKeyButtonProps } from './authorize/add-api-key-button' export type { AddOAuthButtonProps } from './authorize/add-oauth-button' @@ -5,11 +8,14 @@ export enum AuthCategory { tool = 'tool', datasource = 'datasource', model = 'model', + trigger = 'trigger', } export type PluginPayload = { category: AuthCategory provider: string + providerType?: CollectionType | string + detail?: PluginDetail } export enum CredentialTypeEnum { diff --git a/web/app/components/plugins/plugin-detail-panel/app-selector/app-inputs-panel.tsx b/web/app/components/plugins/plugin-detail-panel/app-selector/app-inputs-panel.tsx index 12cd74e10a..edf15a4419 100644 --- a/web/app/components/plugins/plugin-detail-panel/app-selector/app-inputs-panel.tsx +++ b/web/app/components/plugins/plugin-detail-panel/app-selector/app-inputs-panel.tsx @@ -6,7 +6,7 @@ import AppInputsForm from '@/app/components/plugins/plugin-detail-panel/app-sele import { useAppDetail } from '@/service/use-apps' import { useAppWorkflow } from '@/service/use-workflow' import { useFileUploadConfig } from '@/service/use-common' -import { Resolution } from '@/types/app' +import { AppModeEnum, Resolution } from '@/types/app' import { FILE_EXTS } from '@/app/components/base/prompt-editor/constants' import type { App } from '@/types/app' import type { FileUpload } from '@/app/components/base/features/types' @@ -30,7 +30,7 @@ const AppInputsPanel = ({ }: Props) => { const { t } = useTranslation() const inputsRef = useRef(value?.inputs || {}) - const isBasicApp = appDetail.mode !== 'advanced-chat' && appDetail.mode !== 'workflow' + const isBasicApp = appDetail.mode !== AppModeEnum.ADVANCED_CHAT && appDetail.mode !== AppModeEnum.WORKFLOW const { data: fileUploadConfig } = useFileUploadConfig() const { data: currentApp, isFetching: isAppLoading } = useAppDetail(appDetail.id) const { data: currentWorkflow, isFetching: isWorkflowLoading } = useAppWorkflow(isBasicApp ? '' : appDetail.id) @@ -77,7 +77,7 @@ const AppInputsPanel = ({ required: false, } } - if(item.checkbox) { + if (item.checkbox) { return { ...item.checkbox, type: 'checkbox', @@ -148,7 +148,7 @@ const AppInputsPanel = ({ } }) || [] } - if ((currentApp.mode === 'completion' || currentApp.mode === 'workflow') && basicAppFileConfig.enabled) { + if ((currentApp.mode === AppModeEnum.COMPLETION || currentApp.mode === AppModeEnum.WORKFLOW) && basicAppFileConfig.enabled) { inputFormSchema.push({ label: 'Image Upload', variable: '#image#', diff --git a/web/app/components/plugins/plugin-detail-panel/app-selector/app-picker.tsx b/web/app/components/plugins/plugin-detail-panel/app-selector/app-picker.tsx index 10c28507f7..43fb4b30e0 100644 --- a/web/app/components/plugins/plugin-detail-panel/app-selector/app-picker.tsx +++ b/web/app/components/plugins/plugin-detail-panel/app-selector/app-picker.tsx @@ -12,7 +12,7 @@ import type { } from '@floating-ui/react' import Input from '@/app/components/base/input' import AppIcon from '@/app/components/base/app-icon' -import type { App } from '@/types/app' +import { type App, AppModeEnum } from '@/types/app' import { useTranslation } from 'react-i18next' type Props = { @@ -118,15 +118,15 @@ const AppPicker: FC = ({ const getAppType = (app: App) => { switch (app.mode) { - case 'advanced-chat': + case AppModeEnum.ADVANCED_CHAT: return 'chatflow' - case 'agent-chat': + case AppModeEnum.AGENT_CHAT: return 'agent' - case 'chat': + case AppModeEnum.CHAT: return 'chat' - case 'completion': + case AppModeEnum.COMPLETION: return 'completion' - case 'workflow': + case AppModeEnum.WORKFLOW: return 'workflow' } } diff --git a/web/app/components/plugins/plugin-detail-panel/detail-header.tsx b/web/app/components/plugins/plugin-detail-panel/detail-header.tsx index 318d1112bb..44ddb8360e 100644 --- a/web/app/components/plugins/plugin-detail-panel/detail-header.tsx +++ b/web/app/components/plugins/plugin-detail-panel/detail-header.tsx @@ -1,7 +1,26 @@ -import React, { useCallback, useMemo, useState } from 'react' -import { useTheme } from 'next-themes' -import { useTranslation } from 'react-i18next' -import { useBoolean } from 'ahooks' +import ActionButton from '@/app/components/base/action-button' +import Badge from '@/app/components/base/badge' +import Button from '@/app/components/base/button' +import Confirm from '@/app/components/base/confirm' +import { Github } from '@/app/components/base/icons/src/public/common' +import { BoxSparkleFill } from '@/app/components/base/icons/src/vender/plugin' +import Toast from '@/app/components/base/toast' +import Tooltip from '@/app/components/base/tooltip' +import { AuthCategory, PluginAuth } from '@/app/components/plugins/plugin-auth' +import OperationDropdown from '@/app/components/plugins/plugin-detail-panel/operation-dropdown' +import PluginInfo from '@/app/components/plugins/plugin-page/plugin-info' +import UpdateFromMarketplace from '@/app/components/plugins/update-plugin/from-market-place' +import PluginVersionPicker from '@/app/components/plugins/update-plugin/plugin-version-picker' +import { API_PREFIX } from '@/config' +import { useAppContext } from '@/context/app-context' +import { useGlobalPublicStore } from '@/context/global-public-context' +import { useGetLanguage, useI18N } from '@/context/i18n' +import { useModalContext } from '@/context/modal-context' +import { useProviderContext } from '@/context/provider-context' +import { uninstallPlugin } from '@/service/plugins' +import { useAllToolProviders, useInvalidateAllToolProviders } from '@/service/use-tools' +import cn from '@/utils/classnames' +import { getMarketplaceUrl } from '@/utils/var' import { RiArrowLeftRightLine, RiBugLine, @@ -9,54 +28,35 @@ import { RiHardDrive3Line, RiVerifiedBadgeLine, } from '@remixicon/react' -import type { PluginDetail } from '../types' -import { PluginSource, PluginType } from '../types' -import Description from '../card/base/description' -import Icon from '../card/base/card-icon' -import Title from '../card/base/title' -import OrgInfo from '../card/base/org-info' -import { useGitHubReleases } from '../install-plugin/hooks' -import PluginVersionPicker from '@/app/components/plugins/update-plugin/plugin-version-picker' -import UpdateFromMarketplace from '@/app/components/plugins/update-plugin/from-market-place' -import OperationDropdown from '@/app/components/plugins/plugin-detail-panel/operation-dropdown' -import PluginInfo from '@/app/components/plugins/plugin-page/plugin-info' -import ActionButton from '@/app/components/base/action-button' -import Button from '@/app/components/base/button' -import Badge from '@/app/components/base/badge' -import Confirm from '@/app/components/base/confirm' -import Tooltip from '@/app/components/base/tooltip' -import Toast from '@/app/components/base/toast' -import { BoxSparkleFill } from '@/app/components/base/icons/src/vender/plugin' -import { Github } from '@/app/components/base/icons/src/public/common' -import { uninstallPlugin } from '@/service/plugins' -import { useGetLanguage, useI18N } from '@/context/i18n' -import { useModalContext } from '@/context/modal-context' -import { useProviderContext } from '@/context/provider-context' -import { useInvalidateAllToolProviders } from '@/service/use-tools' -import { API_PREFIX } from '@/config' -import cn from '@/utils/classnames' -import { getMarketplaceUrl } from '@/utils/var' -import { PluginAuth } from '@/app/components/plugins/plugin-auth' -import { AuthCategory } from '@/app/components/plugins/plugin-auth' -import { useAllToolProviders } from '@/service/use-tools' -import DeprecationNotice from '../base/deprecation-notice' +import { useBoolean } from 'ahooks' +import { useTheme } from 'next-themes' +import React, { useCallback, useMemo, useState } from 'react' +import { useTranslation } from 'react-i18next' import { AutoUpdateLine } from '../../base/icons/src/vender/system' -import { convertUTCDaySecondsToLocalSeconds, timeOfDayToDayjs } from '../reference-setting-modal/auto-update-setting/utils' +import DeprecationNotice from '../base/deprecation-notice' +import Icon from '../card/base/card-icon' +import Description from '../card/base/description' +import OrgInfo from '../card/base/org-info' +import Title from '../card/base/title' +import { useGitHubReleases } from '../install-plugin/hooks' import useReferenceSetting from '../plugin-page/use-reference-setting' import { AUTO_UPDATE_MODE } from '../reference-setting-modal/auto-update-setting/types' -import { useAppContext } from '@/context/app-context' -import { useGlobalPublicStore } from '@/context/global-public-context' +import { convertUTCDaySecondsToLocalSeconds, timeOfDayToDayjs } from '../reference-setting-modal/auto-update-setting/utils' +import type { PluginDetail } from '../types' +import { PluginCategoryEnum, PluginSource } from '../types' const i18nPrefix = 'plugin.action' type Props = { detail: PluginDetail - onHide: () => void - onUpdate: (isDelete?: boolean) => void + isReadmeView?: boolean + onHide?: () => void + onUpdate?: (isDelete?: boolean) => void } const DetailHeader = ({ detail, + isReadmeView = false, onHide, onUpdate, }: Props) => { @@ -73,7 +73,7 @@ const DetailHeader = ({ const { enable_marketplace } = useGlobalPublicStore(s => s.systemFeatures) const { - installation_id, + id, source, tenant_id, version, @@ -85,8 +85,9 @@ const DetailHeader = ({ deprecated_reason, alternative_plugin_id, } = detail - const { author, category, name, label, description, icon, verified, tool } = detail.declaration - const isTool = category === PluginType.tool + + const { author, category, name, label, description, icon, verified, tool } = detail.declaration || detail + const isTool = category === PluginCategoryEnum.tool const providerBriefInfo = tool?.identity const providerKey = `${plugin_id}/${providerBriefInfo?.name}` const { data: collectionList = [] } = useAllToolProviders(isTool) @@ -128,13 +129,13 @@ const DetailHeader = ({ return false if (!autoUpgradeInfo || !isFromMarketplace) return false - if(autoUpgradeInfo.strategy_setting === 'disabled') + if (autoUpgradeInfo.strategy_setting === 'disabled') return false - if(autoUpgradeInfo.upgrade_mode === AUTO_UPDATE_MODE.update_all) + if (autoUpgradeInfo.upgrade_mode === AUTO_UPDATE_MODE.update_all) return true - if(autoUpgradeInfo.upgrade_mode === AUTO_UPDATE_MODE.partial && autoUpgradeInfo.include_plugins.includes(plugin_id)) + if (autoUpgradeInfo.upgrade_mode === AUTO_UPDATE_MODE.partial && autoUpgradeInfo.include_plugins.includes(plugin_id)) return true - if(autoUpgradeInfo.upgrade_mode === AUTO_UPDATE_MODE.exclude && !autoUpgradeInfo.exclude_plugins.includes(plugin_id)) + if (autoUpgradeInfo.upgrade_mode === AUTO_UPDATE_MODE.exclude && !autoUpgradeInfo.exclude_plugins.includes(plugin_id)) return true return false }, [autoUpgradeInfo, plugin_id, isFromMarketplace]) @@ -156,7 +157,7 @@ const DetailHeader = ({ if (needUpdate) { setShowUpdatePluginModal({ onSaveCallback: () => { - onUpdate() + onUpdate?.() }, payload: { type: PluginSource.github, @@ -176,7 +177,7 @@ const DetailHeader = ({ } const handleUpdatedFromMarketplace = () => { - onUpdate() + onUpdate?.() hideUpdateModal() } @@ -197,30 +198,30 @@ const DetailHeader = ({ const handleDelete = useCallback(async () => { showDeleting() - const res = await uninstallPlugin(installation_id) + const res = await uninstallPlugin(id) hideDeleting() if (res.success) { hideDeleteConfirm() - onUpdate(true) - if (PluginType.model.includes(category)) + onUpdate?.(true) + if (PluginCategoryEnum.model.includes(category)) refreshModelProviders() - if (PluginType.tool.includes(category)) + if (PluginCategoryEnum.tool.includes(category)) invalidateAllToolProviders() } - }, [showDeleting, installation_id, hideDeleting, hideDeleteConfirm, onUpdate, category, refreshModelProviders, invalidateAllToolProviders]) + }, [showDeleting, id, hideDeleting, hideDeleteConfirm, onUpdate, category, refreshModelProviders, invalidateAllToolProviders]) return ( -
+
-
- +
+
- {verified && <RiVerifiedBadgeLine className="ml-0.5 h-4 w-4 shrink-0 text-text-accent" />} - <PluginVersionPicker - disabled={!isFromMarketplace} + {verified && !isReadmeView && <RiVerifiedBadgeLine className="ml-0.5 h-4 w-4 shrink-0 text-text-accent" />} + {version && <PluginVersionPicker + disabled={!isFromMarketplace || isReadmeView} isShow={isShow} onShowChange={setIsShow} pluginID={plugin_id} @@ -240,15 +241,15 @@ const DetailHeader = ({ text={ <> <div>{isFromGitHub ? meta!.version : version}</div> - {isFromMarketplace && <RiArrowLeftRightLine className='ml-1 h-3 w-3 text-text-tertiary' />} + {isFromMarketplace && !isReadmeView && <RiArrowLeftRightLine className='ml-1 h-3 w-3 text-text-tertiary' />} </> } hasRedCornerMark={hasNewVersion} /> } - /> + />} {/* Auto update info */} - {isAutoUpgradeEnabled && ( + {isAutoUpgradeEnabled && !isReadmeView && ( <Tooltip popupContent={t('plugin.autoUpdate.nextUpdateTime', { time: timeOfDayToDayjs(convertUTCDaySecondsToLocalSeconds(autoUpgradeInfo?.upgrade_time_of_day || 0, timezone!)).format('hh:mm A') })}> {/* add a a div to fix tooltip hover not show problem */} <div> @@ -276,44 +277,47 @@ const DetailHeader = ({ <OrgInfo packageNameClassName='w-auto' orgName={author} - packageName={name} + packageName={name?.includes('/') ? (name.split('/').pop() || '') : name} /> - <div className='system-xs-regular ml-1 mr-0.5 text-text-quaternary'>·</div> - {detail.source === PluginSource.marketplace && ( - <Tooltip popupContent={t('plugin.detailPanel.categoryTip.marketplace')} > - <div><BoxSparkleFill className='h-3.5 w-3.5 text-text-tertiary hover:text-text-accent' /></div> - </Tooltip> - )} - {detail.source === PluginSource.github && ( - <Tooltip popupContent={t('plugin.detailPanel.categoryTip.github')} > - <div><Github className='h-3.5 w-3.5 text-text-secondary hover:text-text-primary' /></div> - </Tooltip> - )} - {detail.source === PluginSource.local && ( - <Tooltip popupContent={t('plugin.detailPanel.categoryTip.local')} > - <div><RiHardDrive3Line className='h-3.5 w-3.5 text-text-tertiary' /></div> - </Tooltip> - )} - {detail.source === PluginSource.debugging && ( - <Tooltip popupContent={t('plugin.detailPanel.categoryTip.debugging')} > - <div><RiBugLine className='h-3.5 w-3.5 text-text-tertiary hover:text-text-warning' /></div> - </Tooltip> - )} + {source && <> + <div className='system-xs-regular ml-1 mr-0.5 text-text-quaternary'>·</div> + {source === PluginSource.marketplace && ( + <Tooltip popupContent={t('plugin.detailPanel.categoryTip.marketplace')} > + <div><BoxSparkleFill className='h-3.5 w-3.5 text-text-tertiary hover:text-text-accent' /></div> + </Tooltip> + )} + {source === PluginSource.github && ( + <Tooltip popupContent={t('plugin.detailPanel.categoryTip.github')} > + <div><Github className='h-3.5 w-3.5 text-text-secondary hover:text-text-primary' /></div> + </Tooltip> + )} + {source === PluginSource.local && ( + <Tooltip popupContent={t('plugin.detailPanel.categoryTip.local')} > + <div><RiHardDrive3Line className='h-3.5 w-3.5 text-text-tertiary' /></div> + </Tooltip> + )} + {source === PluginSource.debugging && ( + <Tooltip popupContent={t('plugin.detailPanel.categoryTip.debugging')} > + <div><RiBugLine className='h-3.5 w-3.5 text-text-tertiary hover:text-text-warning' /></div> + </Tooltip> + )} + </>} </div> </div> </div> - <div className='flex gap-1'> - <OperationDropdown - source={detail.source} - onInfo={showPluginInfo} - onCheckVersion={handleUpdate} - onRemove={showDeleteConfirm} - detailUrl={detailUrl} - /> - <ActionButton onClick={onHide}> - <RiCloseLine className='h-4 w-4' /> - </ActionButton> - </div> + {!isReadmeView && ( + <div className='flex gap-1'> + <OperationDropdown + source={source} + onInfo={showPluginInfo} + onCheckVersion={handleUpdate} + onRemove={showDeleteConfirm} + detailUrl={detailUrl} + /> + <ActionButton onClick={onHide}> + <RiCloseLine className='h-4 w-4' /> + </ActionButton> + </div>)} </div> {isFromMarketplace && ( <DeprecationNotice @@ -324,13 +328,15 @@ const DetailHeader = ({ className='mt-3' /> )} - <Description className='mb-2 mt-3 h-auto' text={description[locale]} descriptionLineRows={2}></Description> + {!isReadmeView && <Description className='mb-2 mt-3 h-auto' text={description[locale]} descriptionLineRows={2}></Description>} { - category === PluginType.tool && ( + category === PluginCategoryEnum.tool && !isReadmeView && ( <PluginAuth pluginPayload={{ provider: provider?.name || '', category: AuthCategory.tool, + providerType: provider?.type || '', + detail, }} /> ) @@ -350,7 +356,6 @@ const DetailHeader = ({ content={ <div> {t(`${i18nPrefix}.deleteContentLeft`)}<span className='system-md-semibold'>{label[locale]}</span>{t(`${i18nPrefix}.deleteContentRight`)}<br /> - {/* {usedInApps > 0 && t(`${i18nPrefix}.usedInApps`, { num: usedInApps })} */} </div> } onCancel={hideDeleteConfirm} diff --git a/web/app/components/plugins/plugin-detail-panel/endpoint-card.tsx b/web/app/components/plugins/plugin-detail-panel/endpoint-card.tsx index 00cd1b88ae..9c3765def3 100644 --- a/web/app/components/plugins/plugin-detail-panel/endpoint-card.tsx +++ b/web/app/components/plugins/plugin-detail-panel/endpoint-card.tsx @@ -3,7 +3,7 @@ import { useTranslation } from 'react-i18next' import { useBoolean } from 'ahooks' import copy from 'copy-to-clipboard' import { RiClipboardLine, RiDeleteBinLine, RiEditLine, RiLoginCircleLine } from '@remixicon/react' -import type { EndpointListItem } from '../types' +import type { EndpointListItem, PluginDetail } from '../types' import EndpointModal from './endpoint-modal' import { NAME_FIELD } from './utils' import { addDefaultValue, toolCredentialToFormSchemas } from '@/app/components/tools/utils/to-form-schema' @@ -22,11 +22,13 @@ import { } from '@/service/use-endpoints' type Props = { + pluginDetail: PluginDetail data: EndpointListItem handleChange: () => void } const EndpointCard = ({ + pluginDetail, data, handleChange, }: Props) => { @@ -206,10 +208,11 @@ const EndpointCard = ({ )} {isShowEndpointModal && ( <EndpointModal - formSchemas={formSchemas} + formSchemas={formSchemas as any} defaultValues={formValue} onCancel={hideEndpointModalConfirm} onSaved={handleUpdate} + pluginDetail={pluginDetail} /> )} </div> diff --git a/web/app/components/plugins/plugin-detail-panel/endpoint-list.tsx b/web/app/components/plugins/plugin-detail-panel/endpoint-list.tsx index 5735022c5d..fff6775495 100644 --- a/web/app/components/plugins/plugin-detail-panel/endpoint-list.tsx +++ b/web/app/components/plugins/plugin-detail-panel/endpoint-list.tsx @@ -102,14 +102,16 @@ const EndpointList = ({ detail }: Props) => { key={index} data={item} handleChange={() => invalidateEndpointList(detail.plugin_id)} + pluginDetail={detail} /> ))} </div> {isShowEndpointModal && ( <EndpointModal - formSchemas={formSchemas} + formSchemas={formSchemas as any} onCancel={hideEndpointModal} onSaved={handleCreate} + pluginDetail={detail} /> )} </div> diff --git a/web/app/components/plugins/plugin-detail-panel/endpoint-modal.tsx b/web/app/components/plugins/plugin-detail-panel/endpoint-modal.tsx index 3041f13f2f..48aeecf1b2 100644 --- a/web/app/components/plugins/plugin-detail-panel/endpoint-modal.tsx +++ b/web/app/components/plugins/plugin-detail-panel/endpoint-modal.tsx @@ -10,12 +10,16 @@ import Form from '@/app/components/header/account-setting/model-provider-page/mo import Toast from '@/app/components/base/toast' import { useRenderI18nObject } from '@/hooks/use-i18n' import cn from '@/utils/classnames' +import { ReadmeEntrance } from '../readme-panel/entrance' +import type { PluginDetail } from '../types' +import type { FormSchema } from '../../base/form/types' type Props = { - formSchemas: any + formSchemas: FormSchema[] defaultValues?: any onCancel: () => void onSaved: (value: Record<string, any>) => void + pluginDetail: PluginDetail } const extractDefaultValues = (schemas: any[]) => { @@ -32,6 +36,7 @@ const EndpointModal: FC<Props> = ({ defaultValues = {}, onCancel, onSaved, + pluginDetail, }) => { const getValueFromI18nObject = useRenderI18nObject() const { t } = useTranslation() @@ -43,14 +48,14 @@ const EndpointModal: FC<Props> = ({ const handleSave = () => { for (const field of formSchemas) { if (field.required && !tempCredential[field.name]) { - Toast.notify({ type: 'error', message: t('common.errorMsg.fieldRequired', { field: getValueFromI18nObject(field.label) }) }) + Toast.notify({ type: 'error', message: t('common.errorMsg.fieldRequired', { field: typeof field.label === 'string' ? field.label : getValueFromI18nObject(field.label as Record<string, string>) }) }) return } } // Fix: Process boolean fields to ensure they are sent as proper boolean values const processedCredential = { ...tempCredential } - formSchemas.forEach((field) => { + formSchemas.forEach((field: any) => { if (field.type === 'boolean' && processedCredential[field.name] !== undefined) { const value = processedCredential[field.name] if (typeof value === 'string') @@ -84,6 +89,7 @@ const EndpointModal: FC<Props> = ({ </ActionButton> </div> <div className='system-xs-regular mt-0.5 text-text-tertiary'>{t('plugin.detailPanel.endpointModalDesc')}</div> + <ReadmeEntrance pluginDetail={pluginDetail} className='px-0 pt-3' /> </div> <div className='grow overflow-y-auto'> <div className='px-4 py-2'> @@ -92,7 +98,7 @@ const EndpointModal: FC<Props> = ({ onChange={(v) => { setTempCredential(v) }} - formSchemas={formSchemas} + formSchemas={formSchemas as any} isEditMode={true} showOnVariableMap={{}} validating={false} diff --git a/web/app/components/plugins/plugin-detail-panel/index.tsx b/web/app/components/plugins/plugin-detail-panel/index.tsx index de248390f4..380d2329f6 100644 --- a/web/app/components/plugins/plugin-detail-panel/index.tsx +++ b/web/app/components/plugins/plugin-detail-panel/index.tsx @@ -1,15 +1,19 @@ 'use client' -import React from 'react' +import Drawer from '@/app/components/base/drawer' +import { PluginCategoryEnum, type PluginDetail } from '@/app/components/plugins/types' +import cn from '@/utils/classnames' import type { FC } from 'react' +import { useCallback, useEffect } from 'react' +import ActionList from './action-list' +import AgentStrategyList from './agent-strategy-list' +import DatasourceActionList from './datasource-action-list' import DetailHeader from './detail-header' import EndpointList from './endpoint-list' -import ActionList from './action-list' -import DatasourceActionList from './datasource-action-list' import ModelList from './model-list' -import AgentStrategyList from './agent-strategy-list' -import Drawer from '@/app/components/base/drawer' -import type { PluginDetail } from '@/app/components/plugins/types' -import cn from '@/utils/classnames' +import { SubscriptionList } from './subscription-list' +import { usePluginStore } from './store' +import { TriggerEventsList } from './trigger/event-list' +import { ReadmeEntrance } from '../readme-panel/entrance' type Props = { detail?: PluginDetail @@ -22,11 +26,24 @@ const PluginDetailPanel: FC<Props> = ({ onUpdate, onHide, }) => { - const handleUpdate = (isDelete = false) => { + const handleUpdate = useCallback((isDelete = false) => { if (isDelete) onHide() onUpdate() - } + }, [onHide, onUpdate]) + + const { setDetail } = usePluginStore() + + useEffect(() => { + setDetail(!detail ? undefined : { + plugin_id: detail.plugin_id, + provider: `${detail.plugin_id}/${detail.declaration.name}`, + plugin_unique_identifier: detail.plugin_unique_identifier || '', + declaration: detail.declaration, + name: detail.name, + id: detail.id, + }) + }, [detail]) if (!detail) return null @@ -43,17 +60,24 @@ const PluginDetailPanel: FC<Props> = ({ > {detail && ( <> - <DetailHeader - detail={detail} - onHide={onHide} - onUpdate={handleUpdate} - /> + <DetailHeader detail={detail} onUpdate={handleUpdate} onHide={onHide} /> <div className='grow overflow-y-auto'> - {!!detail.declaration.tool && <ActionList detail={detail} />} - {!!detail.declaration.agent_strategy && <AgentStrategyList detail={detail} />} - {!!detail.declaration.endpoint && <EndpointList detail={detail} />} - {!!detail.declaration.model && <ModelList detail={detail} />} - {!!detail.declaration.datasource && <DatasourceActionList detail={detail} />} + <div className='flex min-h-full flex-col'> + <div className='flex-1'> + {detail.declaration.category === PluginCategoryEnum.trigger && ( + <> + <SubscriptionList /> + <TriggerEventsList /> + </> + )} + {!!detail.declaration.tool && <ActionList detail={detail} />} + {!!detail.declaration.agent_strategy && <AgentStrategyList detail={detail} />} + {!!detail.declaration.endpoint && <EndpointList detail={detail} />} + {!!detail.declaration.model && <ModelList detail={detail} />} + {!!detail.declaration.datasource && <DatasourceActionList detail={detail} />} + </div> + <ReadmeEntrance pluginDetail={detail} className='mt-auto' /> + </div> </div> </> )} diff --git a/web/app/components/plugins/plugin-detail-panel/model-selector/index.tsx b/web/app/components/plugins/plugin-detail-panel/model-selector/index.tsx index 873f187e8f..1393a1844f 100644 --- a/web/app/components/plugins/plugin-detail-panel/model-selector/index.tsx +++ b/web/app/components/plugins/plugin-detail-panel/model-selector/index.tsx @@ -7,6 +7,7 @@ import { useTranslation } from 'react-i18next' import type { DefaultModel, FormValue, + ModelFeatureEnum, } from '@/app/components/header/account-setting/model-provider-page/declarations' import { ModelStatusEnum, ModelTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' import ModelSelector from '@/app/components/header/account-setting/model-provider-page/model-selector' @@ -57,7 +58,7 @@ const ModelParameterModal: FC<ModelParameterModalProps> = ({ const { isAPIKeySet } = useProviderContext() const [open, setOpen] = useState(false) const scopeArray = scope.split('&') - const scopeFeatures = useMemo(() => { + const scopeFeatures = useMemo((): ModelFeatureEnum[] => { if (scopeArray.includes('all')) return [] return scopeArray.filter(item => ![ @@ -67,7 +68,7 @@ const ModelParameterModal: FC<ModelParameterModalProps> = ({ ModelTypeEnum.moderation, ModelTypeEnum.speech2text, ModelTypeEnum.tts, - ].includes(item as ModelTypeEnum)) + ].includes(item as ModelTypeEnum)).map(item => item as ModelFeatureEnum) }, [scopeArray]) const { data: textGenerationList } = useModelList(ModelTypeEnum.textGeneration) diff --git a/web/app/components/plugins/plugin-detail-panel/store.ts b/web/app/components/plugins/plugin-detail-panel/store.ts new file mode 100644 index 0000000000..931b08215d --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/store.ts @@ -0,0 +1,29 @@ +import { create } from 'zustand' +import type { + ParametersSchema, + PluginDeclaration, + PluginDetail, + PluginTriggerSubscriptionConstructor, +} from '../types' + +type TriggerDeclarationSummary = { + subscription_schema?: ParametersSchema[] + subscription_constructor?: PluginTriggerSubscriptionConstructor | null +} + +export type SimpleDetail = Pick<PluginDetail, 'plugin_id' | 'name' | 'plugin_unique_identifier' | 'id'> & { + provider: string + declaration: Partial<Omit<PluginDeclaration, 'trigger'>> & { + trigger?: TriggerDeclarationSummary + } +} + +type Shape = { + detail: SimpleDetail | undefined + setDetail: (detail?: SimpleDetail) => void +} + +export const usePluginStore = create<Shape>(set => ({ + detail: undefined, + setDetail: (detail?: SimpleDetail) => set({ detail }), +})) diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/create/common-modal.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/create/common-modal.tsx new file mode 100644 index 0000000000..17a46febdf --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/create/common-modal.tsx @@ -0,0 +1,449 @@ +'use client' +// import { CopyFeedbackNew } from '@/app/components/base/copy-feedback' +import { EncryptedBottom } from '@/app/components/base/encrypted-bottom' +import { BaseForm } from '@/app/components/base/form/components/base' +import type { FormRefObject } from '@/app/components/base/form/types' +import { FormTypeEnum } from '@/app/components/base/form/types' +import Modal from '@/app/components/base/modal/modal' +import Toast from '@/app/components/base/toast' +import { SupportedCreationMethods } from '@/app/components/plugins/types' +import type { TriggerSubscriptionBuilder } from '@/app/components/workflow/block-selector/types' +import { TriggerCredentialTypeEnum } from '@/app/components/workflow/block-selector/types' +import type { BuildTriggerSubscriptionPayload } from '@/service/use-triggers' +import { + useBuildTriggerSubscription, + useCreateTriggerSubscriptionBuilder, + useTriggerSubscriptionBuilderLogs, + useUpdateTriggerSubscriptionBuilder, + useVerifyTriggerSubscriptionBuilder, +} from '@/service/use-triggers' +import { parsePluginErrorMessage } from '@/utils/error-parser' +import { isPrivateOrLocalAddress } from '@/utils/urlValidation' +import { RiLoader2Line } from '@remixicon/react' +import { debounce } from 'lodash-es' +import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react' +import { useTranslation } from 'react-i18next' +import LogViewer from '../log-viewer' +import { usePluginSubscriptionStore } from '../store' +import { usePluginStore } from '../../store' + +type Props = { + onClose: () => void + createType: SupportedCreationMethods + builder?: TriggerSubscriptionBuilder +} + +const CREDENTIAL_TYPE_MAP: Record<SupportedCreationMethods, TriggerCredentialTypeEnum> = { + [SupportedCreationMethods.APIKEY]: TriggerCredentialTypeEnum.ApiKey, + [SupportedCreationMethods.OAUTH]: TriggerCredentialTypeEnum.Oauth2, + [SupportedCreationMethods.MANUAL]: TriggerCredentialTypeEnum.Unauthorized, +} + +enum ApiKeyStep { + Verify = 'verify', + Configuration = 'configuration', +} + +const defaultFormValues = { values: {}, isCheckValidated: false } + +const normalizeFormType = (type: FormTypeEnum | string): FormTypeEnum => { + if (Object.values(FormTypeEnum).includes(type as FormTypeEnum)) + return type as FormTypeEnum + + switch (type) { + case 'string': + case 'text': + return FormTypeEnum.textInput + case 'password': + case 'secret': + return FormTypeEnum.secretInput + case 'number': + case 'integer': + return FormTypeEnum.textNumber + case 'boolean': + return FormTypeEnum.boolean + default: + return FormTypeEnum.textInput + } +} + +const StatusStep = ({ isActive, text }: { isActive: boolean, text: string }) => { + return <div className={`system-2xs-semibold-uppercase flex items-center gap-1 ${isActive + ? 'text-state-accent-solid' + : 'text-text-tertiary'}`}> + {/* Active indicator dot */} + {isActive && ( + <div className='h-1 w-1 rounded-full bg-state-accent-solid'></div> + )} + {text} + </div> +} + +const MultiSteps = ({ currentStep }: { currentStep: ApiKeyStep }) => { + const { t } = useTranslation() + return <div className='mb-6 flex w-1/3 items-center gap-2'> + <StatusStep isActive={currentStep === ApiKeyStep.Verify} text={t('pluginTrigger.modal.steps.verify')} /> + <div className='h-px w-3 shrink-0 bg-divider-deep'></div> + <StatusStep isActive={currentStep === ApiKeyStep.Configuration} text={t('pluginTrigger.modal.steps.configuration')} /> + </div> +} + +export const CommonCreateModal = ({ onClose, createType, builder }: Props) => { + const { t } = useTranslation() + const detail = usePluginStore(state => state.detail) + const { refresh } = usePluginSubscriptionStore() + + const [currentStep, setCurrentStep] = useState<ApiKeyStep>(createType === SupportedCreationMethods.APIKEY ? ApiKeyStep.Verify : ApiKeyStep.Configuration) + + const [subscriptionBuilder, setSubscriptionBuilder] = useState<TriggerSubscriptionBuilder | undefined>(builder) + const isInitializedRef = useRef(false) + + const { mutate: verifyCredentials, isPending: isVerifyingCredentials } = useVerifyTriggerSubscriptionBuilder() + const { mutateAsync: createBuilder /* isPending: isCreatingBuilder */ } = useCreateTriggerSubscriptionBuilder() + const { mutate: buildSubscription, isPending: isBuilding } = useBuildTriggerSubscription() + const { mutate: updateBuilder } = useUpdateTriggerSubscriptionBuilder() + + const manualPropertiesSchema = detail?.declaration?.trigger?.subscription_schema || [] // manual + const manualPropertiesFormRef = React.useRef<FormRefObject>(null) + + const subscriptionFormRef = React.useRef<FormRefObject>(null) + + const autoCommonParametersSchema = detail?.declaration.trigger?.subscription_constructor?.parameters || [] // apikey and oauth + const autoCommonParametersFormRef = React.useRef<FormRefObject>(null) + + const rawApiKeyCredentialsSchema = detail?.declaration.trigger?.subscription_constructor?.credentials_schema || [] + const apiKeyCredentialsSchema = useMemo(() => { + return rawApiKeyCredentialsSchema.map(schema => ({ + ...schema, + tooltip: schema.help, + })) + }, [rawApiKeyCredentialsSchema]) + const apiKeyCredentialsFormRef = React.useRef<FormRefObject>(null) + + const { data: logData } = useTriggerSubscriptionBuilderLogs( + detail?.provider || '', + subscriptionBuilder?.id || '', + { + enabled: createType === SupportedCreationMethods.MANUAL, + refetchInterval: 3000, + }, + ) + + useEffect(() => { + const initializeBuilder = async () => { + isInitializedRef.current = true + try { + const response = await createBuilder({ + provider: detail?.provider || '', + credential_type: CREDENTIAL_TYPE_MAP[createType], + }) + setSubscriptionBuilder(response.subscription_builder) + } + catch (error) { + console.error('createBuilder error:', error) + Toast.notify({ + type: 'error', + message: t('pluginTrigger.modal.errors.createFailed'), + }) + } + } + if (!isInitializedRef.current && !subscriptionBuilder && detail?.provider) + initializeBuilder() + }, [subscriptionBuilder, detail?.provider, createType, createBuilder, t]) + + useEffect(() => { + if (subscriptionBuilder?.endpoint && subscriptionFormRef.current && currentStep === ApiKeyStep.Configuration) { + const form = subscriptionFormRef.current.getForm() + if (form) + form.setFieldValue('callback_url', subscriptionBuilder.endpoint) + if (isPrivateOrLocalAddress(subscriptionBuilder.endpoint)) { + console.log('isPrivateOrLocalAddress', isPrivateOrLocalAddress(subscriptionBuilder.endpoint)) + subscriptionFormRef.current?.setFields([{ + name: 'callback_url', + warnings: [t('pluginTrigger.modal.form.callbackUrl.privateAddressWarning')], + }]) + } + else { + subscriptionFormRef.current?.setFields([{ + name: 'callback_url', + warnings: [], + }]) + } + } + }, [subscriptionBuilder?.endpoint, currentStep, t]) + + const debouncedUpdate = useMemo( + () => debounce((provider: string, builderId: string, properties: Record<string, any>) => { + updateBuilder( + { + provider, + subscriptionBuilderId: builderId, + properties, + }, + { + onError: (error: any) => { + console.error('Failed to update subscription builder:', error) + Toast.notify({ + type: 'error', + message: error?.message || t('pluginTrigger.modal.errors.updateFailed'), + }) + }, + }, + ) + }, 500), + [updateBuilder, t], + ) + + const handleManualPropertiesChange = useCallback(() => { + if (!subscriptionBuilder || !detail?.provider) + return + + const formValues = manualPropertiesFormRef.current?.getFormValues({ needCheckValidatedValues: false }) || { values: {}, isCheckValidated: true } + + debouncedUpdate(detail.provider, subscriptionBuilder.id, formValues.values) + }, [subscriptionBuilder, detail?.provider, debouncedUpdate]) + + useEffect(() => { + return () => { + debouncedUpdate.cancel() + } + }, [debouncedUpdate]) + + const handleVerify = () => { + const apiKeyCredentialsFormValues = apiKeyCredentialsFormRef.current?.getFormValues({}) || defaultFormValues + const credentials = apiKeyCredentialsFormValues.values + + if (!Object.keys(credentials).length) { + Toast.notify({ + type: 'error', + message: 'Please fill in all required credentials', + }) + return + } + + apiKeyCredentialsFormRef.current?.setFields([{ + name: Object.keys(credentials)[0], + errors: [], + }]) + + verifyCredentials( + { + provider: detail?.provider || '', + subscriptionBuilderId: subscriptionBuilder?.id || '', + credentials, + }, + { + onSuccess: () => { + Toast.notify({ + type: 'success', + message: t('pluginTrigger.modal.apiKey.verify.success'), + }) + setCurrentStep(ApiKeyStep.Configuration) + }, + onError: async (error: any) => { + const errorMessage = await parsePluginErrorMessage(error) || t('pluginTrigger.modal.apiKey.verify.error') + apiKeyCredentialsFormRef.current?.setFields([{ + name: Object.keys(credentials)[0], + errors: [errorMessage], + }]) + }, + }, + ) + } + + const handleCreate = () => { + if (!subscriptionBuilder) { + Toast.notify({ + type: 'error', + message: 'Subscription builder not found', + }) + return + } + + const subscriptionFormValues = subscriptionFormRef.current?.getFormValues({}) + if (!subscriptionFormValues?.isCheckValidated) + return + + const subscriptionNameValue = subscriptionFormValues?.values?.subscription_name as string + + const params: BuildTriggerSubscriptionPayload = { + provider: detail?.provider || '', + subscriptionBuilderId: subscriptionBuilder.id, + name: subscriptionNameValue, + } + + if (createType !== SupportedCreationMethods.MANUAL) { + if (autoCommonParametersSchema.length > 0) { + const autoCommonParametersFormValues = autoCommonParametersFormRef.current?.getFormValues({}) || defaultFormValues + if (!autoCommonParametersFormValues?.isCheckValidated) + return + params.parameters = autoCommonParametersFormValues.values + } + } + else if (manualPropertiesSchema.length > 0) { + const manualFormValues = manualPropertiesFormRef.current?.getFormValues({}) || defaultFormValues + if (!manualFormValues?.isCheckValidated) + return + } + + buildSubscription( + params, + { + onSuccess: () => { + Toast.notify({ + type: 'success', + message: t('pluginTrigger.subscription.createSuccess'), + }) + onClose() + refresh?.() + }, + onError: async (error: any) => { + const errorMessage = await parsePluginErrorMessage(error) || t('pluginTrigger.subscription.createFailed') + Toast.notify({ + type: 'error', + message: errorMessage, + }) + }, + }, + ) + } + + const handleConfirm = () => { + if (currentStep === ApiKeyStep.Verify) + handleVerify() + else + handleCreate() + } + + const handleApiKeyCredentialsChange = () => { + apiKeyCredentialsFormRef.current?.setFields([{ + name: apiKeyCredentialsSchema[0].name, + errors: [], + }]) + } + + return ( + <Modal + title={t(`pluginTrigger.modal.${createType === SupportedCreationMethods.APIKEY ? 'apiKey' : createType.toLowerCase()}.title`)} + confirmButtonText={ + currentStep === ApiKeyStep.Verify + ? isVerifyingCredentials ? t('pluginTrigger.modal.common.verifying') : t('pluginTrigger.modal.common.verify') + : isBuilding ? t('pluginTrigger.modal.common.creating') : t('pluginTrigger.modal.common.create') + } + onClose={onClose} + onCancel={onClose} + onConfirm={handleConfirm} + disabled={isVerifyingCredentials || isBuilding} + bottomSlot={currentStep === ApiKeyStep.Verify ? <EncryptedBottom /> : null} + size={createType === SupportedCreationMethods.MANUAL ? 'md' : 'sm'} + containerClassName='min-h-[360px]' + clickOutsideNotClose + > + {createType === SupportedCreationMethods.APIKEY && <MultiSteps currentStep={currentStep} />} + {currentStep === ApiKeyStep.Verify && ( + <> + {apiKeyCredentialsSchema.length > 0 && ( + <div className='mb-4'> + <BaseForm + formSchemas={apiKeyCredentialsSchema} + ref={apiKeyCredentialsFormRef} + labelClassName='system-sm-medium mb-2 flex items-center gap-1 text-text-primary' + preventDefaultSubmit={true} + formClassName='space-y-4' + onChange={handleApiKeyCredentialsChange} + /> + </div> + )} + </> + )} + {currentStep === ApiKeyStep.Configuration && <div className='max-h-[70vh]'> + <BaseForm + formSchemas={[ + { + name: 'subscription_name', + label: t('pluginTrigger.modal.form.subscriptionName.label'), + placeholder: t('pluginTrigger.modal.form.subscriptionName.placeholder'), + type: FormTypeEnum.textInput, + required: true, + }, + { + name: 'callback_url', + label: t('pluginTrigger.modal.form.callbackUrl.label'), + placeholder: t('pluginTrigger.modal.form.callbackUrl.placeholder'), + type: FormTypeEnum.textInput, + required: false, + default: subscriptionBuilder?.endpoint || '', + disabled: true, + tooltip: t('pluginTrigger.modal.form.callbackUrl.tooltip'), + showCopy: true, + }, + ]} + ref={subscriptionFormRef} + labelClassName='system-sm-medium mb-2 flex items-center gap-1 text-text-primary' + formClassName='space-y-4 mb-4' + /> + {/* <div className='system-xs-regular mb-6 mt-[-1rem] text-text-tertiary'> + {t('pluginTrigger.modal.form.callbackUrl.description')} + </div> */} + {createType !== SupportedCreationMethods.MANUAL && autoCommonParametersSchema.length > 0 && ( + <BaseForm + formSchemas={autoCommonParametersSchema.map((schema) => { + const normalizedType = normalizeFormType(schema.type as FormTypeEnum | string) + return { + ...schema, + tooltip: schema.description, + type: normalizedType, + dynamicSelectParams: normalizedType === FormTypeEnum.dynamicSelect ? { + plugin_id: detail?.plugin_id || '', + provider: detail?.provider || '', + action: 'provider', + parameter: schema.name, + credential_id: subscriptionBuilder?.id || '', + } : undefined, + fieldClassName: schema.type === FormTypeEnum.boolean ? 'flex items-center justify-between' : undefined, + labelClassName: schema.type === FormTypeEnum.boolean ? 'mb-0' : undefined, + } + })} + ref={autoCommonParametersFormRef} + labelClassName='system-sm-medium mb-2 flex items-center gap-1 text-text-primary' + formClassName='space-y-4' + /> + )} + {createType === SupportedCreationMethods.MANUAL && <> + {manualPropertiesSchema.length > 0 && ( + <div className='mb-6'> + <BaseForm + formSchemas={manualPropertiesSchema.map(schema => ({ + ...schema, + tooltip: schema.description, + }))} + ref={manualPropertiesFormRef} + labelClassName='system-sm-medium mb-2 flex items-center gap-1 text-text-primary' + formClassName='space-y-4' + onChange={handleManualPropertiesChange} + /> + </div> + )} + <div className='mb-6'> + <div className='mb-3 flex items-center gap-2'> + <div className='system-xs-medium-uppercase text-text-tertiary'> + {t('pluginTrigger.modal.manual.logs.title')} + </div> + <div className='h-px flex-1 bg-gradient-to-r from-divider-regular to-transparent' /> + </div> + + <div className='mb-1 flex items-center justify-center gap-1 rounded-lg bg-background-section p-3'> + <div className='h-3.5 w-3.5'> + <RiLoader2Line className='h-full w-full animate-spin' /> + </div> + <div className='system-xs-regular text-text-tertiary'> + {t('pluginTrigger.modal.manual.logs.loading', { pluginName: detail?.name || '' })} + </div> + </div> + <LogViewer logs={logData?.logs || []} /> + </div> + </>} + </div>} + </Modal> + ) +} diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/create/index.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/create/index.tsx new file mode 100644 index 0000000000..7515ba4b4a --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/create/index.tsx @@ -0,0 +1,242 @@ +import { ActionButton, ActionButtonState } from '@/app/components/base/action-button' +import Badge from '@/app/components/base/badge' +import { Button } from '@/app/components/base/button' +import type { Option } from '@/app/components/base/select/custom' +import CustomSelect from '@/app/components/base/select/custom' +import Toast from '@/app/components/base/toast' +import Tooltip from '@/app/components/base/tooltip' +import type { TriggerSubscriptionBuilder } from '@/app/components/workflow/block-selector/types' +import { openOAuthPopup } from '@/hooks/use-oauth' +import { useInitiateTriggerOAuth, useTriggerOAuthConfig, useTriggerProviderInfo } from '@/service/use-triggers' +import cn from '@/utils/classnames' +import { RiAddLine, RiEqualizer2Line } from '@remixicon/react' +import { useBoolean } from 'ahooks' +import { useMemo, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { SupportedCreationMethods } from '../../../types' +import { usePluginStore } from '../../store' +import { useSubscriptionList } from '../use-subscription-list' +import { CommonCreateModal } from './common-modal' +import { OAuthClientSettingsModal } from './oauth-client' + +export enum CreateButtonType { + FULL_BUTTON = 'full-button', + ICON_BUTTON = 'icon-button', +} + +type Props = { + className?: string + buttonType?: CreateButtonType + shape?: 'square' | 'circle' +} + +const MAX_COUNT = 10 + +export const DEFAULT_METHOD = 'default' + +export const CreateSubscriptionButton = ({ buttonType = CreateButtonType.FULL_BUTTON, shape = 'square' }: Props) => { + const { t } = useTranslation() + const { subscriptions } = useSubscriptionList() + const subscriptionCount = subscriptions?.length || 0 + const [selectedCreateInfo, setSelectedCreateInfo] = useState<{ type: SupportedCreationMethods, builder?: TriggerSubscriptionBuilder } | null>(null) + + const detail = usePluginStore(state => state.detail) + + const { data: providerInfo } = useTriggerProviderInfo(detail?.provider || '') + const supportedMethods = providerInfo?.supported_creation_methods || [] + const { data: oauthConfig, refetch: refetchOAuthConfig } = useTriggerOAuthConfig(detail?.provider || '', supportedMethods.includes(SupportedCreationMethods.OAUTH)) + const { mutate: initiateOAuth } = useInitiateTriggerOAuth() + + const methodType = supportedMethods.length === 1 ? supportedMethods[0] : DEFAULT_METHOD + + const [isShowClientSettingsModal, { + setTrue: showClientSettingsModal, + setFalse: hideClientSettingsModal, + }] = useBoolean(false) + + const buttonTextMap = useMemo(() => { + return { + [SupportedCreationMethods.OAUTH]: t('pluginTrigger.subscription.createButton.oauth'), + [SupportedCreationMethods.APIKEY]: t('pluginTrigger.subscription.createButton.apiKey'), + [SupportedCreationMethods.MANUAL]: t('pluginTrigger.subscription.createButton.manual'), + [DEFAULT_METHOD]: t('pluginTrigger.subscription.empty.button'), + } + }, [t]) + + const onClickClientSettings = (e: React.MouseEvent<HTMLDivElement | HTMLButtonElement>) => { + e.stopPropagation() + e.preventDefault() + showClientSettingsModal() + } + + const allOptions = useMemo(() => { + const showCustomBadge = oauthConfig?.custom_enabled && oauthConfig?.custom_configured + + return [ + { + value: SupportedCreationMethods.OAUTH, + label: t('pluginTrigger.subscription.addType.options.oauth.title'), + tag: !showCustomBadge ? null : <Badge className='ml-1 mr-0.5'> + {t('plugin.auth.custom')} + </Badge>, + extra: <Tooltip popupContent={t('pluginTrigger.subscription.addType.options.oauth.clientSettings')}> + <ActionButton onClick={onClickClientSettings}> + <RiEqualizer2Line className='h-4 w-4 text-text-tertiary' /> + </ActionButton> + </Tooltip>, + show: supportedMethods.includes(SupportedCreationMethods.OAUTH), + }, + { + value: SupportedCreationMethods.APIKEY, + label: t('pluginTrigger.subscription.addType.options.apikey.title'), + show: supportedMethods.includes(SupportedCreationMethods.APIKEY), + }, + { + value: SupportedCreationMethods.MANUAL, + label: t('pluginTrigger.subscription.addType.options.manual.description'), + extra: <Tooltip popupContent={t('pluginTrigger.subscription.addType.options.manual.tip')} />, + show: supportedMethods.includes(SupportedCreationMethods.MANUAL), + }, + ] + }, [t, oauthConfig, supportedMethods, methodType]) + + const onChooseCreateType = async (type: SupportedCreationMethods) => { + if (type === SupportedCreationMethods.OAUTH) { + if (oauthConfig?.configured) { + initiateOAuth(detail?.provider || '', { + onSuccess: (response) => { + openOAuthPopup(response.authorization_url, (callbackData) => { + if (callbackData) { + Toast.notify({ + type: 'success', + message: t('pluginTrigger.modal.oauth.authorization.authSuccess'), + }) + setSelectedCreateInfo({ type: SupportedCreationMethods.OAUTH, builder: response.subscription_builder }) + } + }) + }, + onError: () => { + Toast.notify({ + type: 'error', + message: t('pluginTrigger.modal.oauth.authorization.authFailed'), + }) + }, + }) + } + else { + showClientSettingsModal() + } + } + else { + setSelectedCreateInfo({ type }) + } + } + + const onClickCreate = (e: React.MouseEvent<HTMLButtonElement>) => { + if (subscriptionCount >= MAX_COUNT) { + e.stopPropagation() + return + } + + if (methodType === DEFAULT_METHOD || (methodType === SupportedCreationMethods.OAUTH && supportedMethods.length === 1)) + return + + e.stopPropagation() + e.preventDefault() + onChooseCreateType(methodType) + } + + if (!supportedMethods.length) + return null + + return <> + <CustomSelect<Option & { show: boolean; extra?: React.ReactNode; tag?: React.ReactNode }> + options={allOptions.filter(option => option.show)} + value={methodType} + onChange={value => onChooseCreateType(value as any)} + containerProps={{ + open: (methodType === DEFAULT_METHOD || (methodType === SupportedCreationMethods.OAUTH && supportedMethods.length === 1)) ? undefined : false, + placement: 'bottom-start', + offset: 4, + triggerPopupSameWidth: buttonType === CreateButtonType.FULL_BUTTON, + }} + triggerProps={{ + className: cn('h-8 bg-transparent px-0 hover:bg-transparent', methodType !== DEFAULT_METHOD && supportedMethods.length > 1 && 'pointer-events-none', buttonType === CreateButtonType.FULL_BUTTON && 'grow'), + }} + popupProps={{ + wrapperClassName: 'z-[1000]', + }} + CustomTrigger={() => { + return buttonType === CreateButtonType.FULL_BUTTON ? ( + <Button + variant='primary' + size='medium' + className='flex w-full items-center justify-between px-0' + onClick={onClickCreate} + > + <div className='flex flex-1 items-center justify-center'> + <RiAddLine className='mr-2 size-4' /> + {buttonTextMap[methodType]} + {methodType === SupportedCreationMethods.OAUTH && oauthConfig?.custom_enabled && oauthConfig?.custom_configured && <Badge + className='ml-1 mr-0.5 border-text-primary-on-surface bg-components-badge-bg-dimm text-text-primary-on-surface' + > + {t('plugin.auth.custom')} + </Badge>} + </div> + {methodType === SupportedCreationMethods.OAUTH + && <div className='ml-auto flex items-center'> + <div className="h-4 w-px bg-text-primary-on-surface opacity-15" /> + <Tooltip popupContent={t('pluginTrigger.subscription.addType.options.oauth.clientSettings')}> + <div onClick={onClickClientSettings} className='p-2'> + <RiEqualizer2Line className='size-4 text-components-button-primary-text' /> + </div> + </Tooltip> + </div> + } + </Button> + ) : ( + <Tooltip + popupContent={subscriptionCount >= MAX_COUNT ? t('pluginTrigger.subscription.maxCount', { num: MAX_COUNT }) : t(`pluginTrigger.subscription.addType.options.${methodType.toLowerCase()}.description`)} + disabled={!(supportedMethods?.length === 1 || subscriptionCount >= MAX_COUNT)}> + <ActionButton + onClick={onClickCreate} + className={cn( + 'float-right', + shape === 'circle' && '!rounded-full border-[0.5px] border-components-button-secondary-border-hover bg-components-button-secondary-bg-hover text-components-button-secondary-accent-text shadow-xs hover:border-components-button-secondary-border-disabled hover:bg-components-button-secondary-bg-disabled hover:text-components-button-secondary-accent-text-disabled', + )} + state={subscriptionCount >= MAX_COUNT ? ActionButtonState.Disabled : ActionButtonState.Default} + > + <RiAddLine className='size-4' /> + </ActionButton> + </Tooltip> + ) + }} + CustomOption={option => ( + <> + <div className='mr-8 flex grow items-center gap-1 truncate px-1'> + {option.label} + {option.tag} + </div> + {option.extra} + </> + )} + /> + {selectedCreateInfo && ( + <CommonCreateModal + createType={selectedCreateInfo.type} + builder={selectedCreateInfo.builder} + onClose={() => setSelectedCreateInfo(null)} + /> + )} + {isShowClientSettingsModal && ( + <OAuthClientSettingsModal + oauthConfig={oauthConfig} + onClose={() => { + hideClientSettingsModal() + refetchOAuthConfig() + }} + showOAuthCreateModal={builder => setSelectedCreateInfo({ type: SupportedCreationMethods.OAUTH, builder })} + /> + )} + </> +} diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/create/oauth-client.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/create/oauth-client.tsx new file mode 100644 index 0000000000..ef182a70aa --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/create/oauth-client.tsx @@ -0,0 +1,257 @@ +'use client' +import Button from '@/app/components/base/button' +import { BaseForm } from '@/app/components/base/form/components/base' +import type { FormRefObject } from '@/app/components/base/form/types' +import Modal from '@/app/components/base/modal/modal' +import Toast from '@/app/components/base/toast' +import type { TriggerOAuthClientParams, TriggerOAuthConfig, TriggerSubscriptionBuilder } from '@/app/components/workflow/block-selector/types' +import OptionCard from '@/app/components/workflow/nodes/_base/components/option-card' +import { openOAuthPopup } from '@/hooks/use-oauth' +import type { ConfigureTriggerOAuthPayload } from '@/service/use-triggers' +import { + useConfigureTriggerOAuth, + useDeleteTriggerOAuth, + useInitiateTriggerOAuth, + useVerifyTriggerSubscriptionBuilder, +} from '@/service/use-triggers' +import { + RiClipboardLine, + RiInformation2Fill, +} from '@remixicon/react' +import React, { useEffect, useMemo, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { usePluginStore } from '../../store' + +type Props = { + oauthConfig?: TriggerOAuthConfig + onClose: () => void + showOAuthCreateModal: (builder: TriggerSubscriptionBuilder) => void +} + +enum AuthorizationStatusEnum { + Pending = 'pending', + Success = 'success', + Failed = 'failed', +} + +enum ClientTypeEnum { + Default = 'default', + Custom = 'custom', +} + +export const OAuthClientSettingsModal = ({ oauthConfig, onClose, showOAuthCreateModal }: Props) => { + const { t } = useTranslation() + const detail = usePluginStore(state => state.detail) + const { system_configured, params, oauth_client_schema } = oauthConfig || {} + const [subscriptionBuilder, setSubscriptionBuilder] = useState<TriggerSubscriptionBuilder | undefined>() + const [authorizationStatus, setAuthorizationStatus] = useState<AuthorizationStatusEnum>() + + const [clientType, setClientType] = useState<ClientTypeEnum>(system_configured ? ClientTypeEnum.Default : ClientTypeEnum.Custom) + + const clientFormRef = React.useRef<FormRefObject>(null) + + const oauthClientSchema = useMemo(() => { + if (oauth_client_schema && oauth_client_schema.length > 0 && params) { + const oauthConfigPramaKeys = Object.keys(params || {}) + for (const schema of oauth_client_schema) { + if (oauthConfigPramaKeys.includes(schema.name)) + schema.default = params?.[schema.name] + } + return oauth_client_schema + } + return [] + }, [oauth_client_schema, params]) + + const providerName = detail?.provider || '' + const { mutate: initiateOAuth } = useInitiateTriggerOAuth() + const { mutate: verifyBuilder } = useVerifyTriggerSubscriptionBuilder() + const { mutate: configureOAuth } = useConfigureTriggerOAuth() + const { mutate: deleteOAuth } = useDeleteTriggerOAuth() + + const handleAuthorization = () => { + setAuthorizationStatus(AuthorizationStatusEnum.Pending) + initiateOAuth(providerName, { + onSuccess: (response) => { + setSubscriptionBuilder(response.subscription_builder) + openOAuthPopup(response.authorization_url, (callbackData) => { + if (callbackData) { + Toast.notify({ + type: 'success', + message: t('pluginTrigger.modal.oauth.authorization.authSuccess'), + }) + onClose() + showOAuthCreateModal(response.subscription_builder) + } + }) + }, + onError: () => { + setAuthorizationStatus(AuthorizationStatusEnum.Failed) + Toast.notify({ + type: 'error', + message: t('pluginTrigger.modal.oauth.authorization.authFailed'), + }) + }, + }) + } + + useEffect(() => { + if (providerName && subscriptionBuilder && authorizationStatus === AuthorizationStatusEnum.Pending) { + const pollInterval = setInterval(() => { + verifyBuilder( + { + provider: providerName, + subscriptionBuilderId: subscriptionBuilder.id, + }, + { + onSuccess: (response) => { + if (response.verified) { + setAuthorizationStatus(AuthorizationStatusEnum.Success) + clearInterval(pollInterval) + } + }, + onError: () => { + // Continue polling - auth might still be in progress + }, + }, + ) + }, 3000) + + return () => clearInterval(pollInterval) + } + }, [subscriptionBuilder, authorizationStatus, verifyBuilder, providerName, t]) + + const handleRemove = () => { + deleteOAuth(providerName, { + onSuccess: () => { + onClose() + Toast.notify({ + type: 'success', + message: t('pluginTrigger.modal.oauth.remove.success'), + }) + }, + onError: (error: any) => { + Toast.notify({ + type: 'error', + message: error?.message || t('pluginTrigger.modal.oauth.remove.failed'), + }) + }, + }) + } + + const handleSave = (needAuth: boolean) => { + const isCustom = clientType === ClientTypeEnum.Custom + const params: ConfigureTriggerOAuthPayload = { + provider: providerName, + enabled: isCustom, + } + + if (isCustom) { + const clientFormValues = clientFormRef.current?.getFormValues({}) as { values: TriggerOAuthClientParams, isCheckValidated: boolean } + if (!clientFormValues.isCheckValidated) + return + const clientParams = clientFormValues.values + if (clientParams.client_id === oauthConfig?.params.client_id) + clientParams.client_id = '[__HIDDEN__]' + + if (clientParams.client_secret === oauthConfig?.params.client_secret) + clientParams.client_secret = '[__HIDDEN__]' + + params.client_params = clientParams + } + + configureOAuth(params, { + onSuccess: () => { + if (needAuth) { + handleAuthorization() + } + else { + onClose() + Toast.notify({ + type: 'success', + message: t('pluginTrigger.modal.oauth.save.success'), + }) + } + }, + }) + } + + return ( + <Modal + title={t('pluginTrigger.modal.oauth.title')} + confirmButtonText={authorizationStatus === AuthorizationStatusEnum.Pending ? t('pluginTrigger.modal.common.authorizing') + : authorizationStatus === AuthorizationStatusEnum.Success ? t('pluginTrigger.modal.oauth.authorization.waitingJump') : t('plugin.auth.saveAndAuth')} + cancelButtonText={t('plugin.auth.saveOnly')} + extraButtonText={t('common.operation.cancel')} + showExtraButton + clickOutsideNotClose + extraButtonVariant='secondary' + onExtraButtonClick={onClose} + onClose={onClose} + onCancel={() => handleSave(false)} + onConfirm={() => handleSave(true)} + footerSlot={ + oauthConfig?.custom_enabled && oauthConfig?.params && clientType === ClientTypeEnum.Custom && ( + <div className='grow'> + <Button + variant='secondary' + className='text-components-button-destructive-secondary-text' + // disabled={disabled || doingAction || !editValues} + onClick={handleRemove} + > + {t('common.operation.remove')} + </Button> + </div> + ) + } + > + <div className='system-sm-medium mb-2 text-text-secondary'>{t('pluginTrigger.subscription.addType.options.oauth.clientTitle')}</div> + {oauthConfig?.system_configured && <div className='mb-4 flex w-full items-start justify-between gap-2'> + {[ClientTypeEnum.Default, ClientTypeEnum.Custom].map(option => ( + <OptionCard + key={option} + title={t(`pluginTrigger.subscription.addType.options.oauth.${option}`)} + onSelect={() => setClientType(option)} + selected={clientType === option} + className="flex-1" + /> + ))} + </div>} + {clientType === ClientTypeEnum.Custom && oauthConfig?.redirect_uri && ( + <div className='mb-4 flex items-start gap-3 rounded-xl bg-background-section-burn p-4'> + <div className='rounded-lg border-[0.5px] border-components-card-border bg-components-card-bg p-2 shadow-xs shadow-shadow-shadow-3'> + <RiInformation2Fill className='h-5 w-5 shrink-0 text-text-accent' /> + </div> + <div className='flex-1 text-text-secondary'> + <div className='system-sm-regular whitespace-pre-wrap leading-4'> + {t('pluginTrigger.modal.oauthRedirectInfo')} + </div> + <div className='system-sm-medium my-1.5 break-all leading-4'> + {oauthConfig.redirect_uri} + </div> + <Button + variant='secondary' + size='small' + onClick={() => { + navigator.clipboard.writeText(oauthConfig.redirect_uri) + Toast.notify({ + type: 'success', + message: t('common.actionMsg.copySuccessfully'), + }) + }}> + <RiClipboardLine className='mr-1 h-[14px] w-[14px]' /> + {t('common.operation.copy')} + </Button> + </div> + </div> + )} + {clientType === ClientTypeEnum.Custom && oauthClientSchema.length > 0 && ( + <BaseForm + formSchemas={oauthClientSchema} + ref={clientFormRef} + labelClassName='system-sm-medium mb-2 block text-text-secondary' + formClassName='space-y-4' + /> + )} + </Modal > + ) +} diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/delete-confirm.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/delete-confirm.tsx new file mode 100644 index 0000000000..178983c6b1 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/delete-confirm.tsx @@ -0,0 +1,75 @@ +import Confirm from '@/app/components/base/confirm' +import Input from '@/app/components/base/input' +import Toast from '@/app/components/base/toast' +import { useDeleteTriggerSubscription } from '@/service/use-triggers' +import { useState } from 'react' +import { useTranslation } from 'react-i18next' +import { usePluginSubscriptionStore } from './store' + +type Props = { + onClose: (deleted: boolean) => void + isShow: boolean + currentId: string + currentName: string + workflowsInUse: number +} + +const tPrefix = 'pluginTrigger.subscription.list.item.actions.deleteConfirm' + +export const DeleteConfirm = (props: Props) => { + const { onClose, isShow, currentId, currentName, workflowsInUse } = props + const { refresh } = usePluginSubscriptionStore() + const { mutate: deleteSubscription, isPending: isDeleting } = useDeleteTriggerSubscription() + const { t } = useTranslation() + const [inputName, setInputName] = useState('') + + const onConfirm = () => { + if (workflowsInUse > 0 && inputName !== currentName) { + Toast.notify({ + type: 'error', + message: t(`${tPrefix}.confirmInputWarning`), + // temporarily + className: 'z-[10000001]', + }) + return + } + deleteSubscription(currentId, { + onSuccess: () => { + Toast.notify({ + type: 'success', + message: t(`${tPrefix}.success`, { name: currentName }), + className: 'z-[10000001]', + }) + refresh?.() + onClose(true) + }, + onError: (error: any) => { + Toast.notify({ + type: 'error', + message: error?.message || t(`${tPrefix}.error`, { name: currentName }), + className: 'z-[10000001]', + }) + }, + }) + } + return <Confirm + title={t(`${tPrefix}.title`, { name: currentName })} + confirmText={t(`${tPrefix}.confirm`)} + content={workflowsInUse > 0 ? <> + {t(`${tPrefix}.contentWithApps`, { count: workflowsInUse })} + <div className='system-sm-medium mb-2 mt-6 text-text-secondary'>{t(`${tPrefix}.confirmInputTip`, { name: currentName })}</div> + <Input + value={inputName} + onChange={e => setInputName(e.target.value)} + placeholder={t(`${tPrefix}.confirmInputPlaceholder`, { name: currentName })} + /> + </> + : t(`${tPrefix}.content`)} + isShow={isShow} + isLoading={isDeleting} + isDisabled={isDeleting} + onConfirm={onConfirm} + onCancel={() => onClose(false)} + maskClosable={false} + /> +} diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/index.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/index.tsx new file mode 100644 index 0000000000..8acb8f40df --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/index.tsx @@ -0,0 +1,51 @@ +import { withErrorBoundary } from '@/app/components/base/error-boundary' +import Loading from '@/app/components/base/loading' +import { SubscriptionListView } from './list-view' +import { SubscriptionSelectorView } from './selector-view' +import { useSubscriptionList } from './use-subscription-list' + +export enum SubscriptionListMode { + PANEL = 'panel', + SELECTOR = 'selector', +} + +export type SimpleSubscription = { + id: string, + name: string +} + +type SubscriptionListProps = { + mode?: SubscriptionListMode + selectedId?: string + onSelect?: (v: SimpleSubscription, callback?: () => void) => void +} + +export { SubscriptionSelectorEntry } from './selector-entry' + +export const SubscriptionList = withErrorBoundary(({ + mode = SubscriptionListMode.PANEL, + selectedId, + onSelect, +}: SubscriptionListProps) => { + const { isLoading, refetch } = useSubscriptionList() + if (isLoading) { + return ( + <div className='flex items-center justify-center py-4'> + <Loading /> + </div> + ) + } + + if (mode === SubscriptionListMode.SELECTOR) { + return ( + <SubscriptionSelectorView + selectedId={selectedId} + onSelect={(v) => { + onSelect?.(v, refetch) + }} + /> + ) + } + + return <SubscriptionListView /> +}) diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/list-view.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/list-view.tsx new file mode 100644 index 0000000000..a64d2f4070 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/list-view.tsx @@ -0,0 +1,50 @@ +'use client' +import Tooltip from '@/app/components/base/tooltip' +import cn from '@/utils/classnames' +import React from 'react' +import { useTranslation } from 'react-i18next' +import { CreateButtonType, CreateSubscriptionButton } from './create' +import SubscriptionCard from './subscription-card' +import { useSubscriptionList } from './use-subscription-list' + +type SubscriptionListViewProps = { + showTopBorder?: boolean +} + +export const SubscriptionListView: React.FC<SubscriptionListViewProps> = ({ + showTopBorder = false, +}) => { + const { t } = useTranslation() + const { subscriptions } = useSubscriptionList() + + const subscriptionCount = subscriptions?.length || 0 + + return ( + <div className={cn('border-divider-subtle px-4 py-2', showTopBorder && 'border-t')}> + <div className='relative flex items-center justify-between'> + {subscriptionCount > 0 && ( + <div className='flex h-8 shrink-0 items-center gap-1'> + <span className='system-sm-semibold-uppercase text-text-secondary'> + {t('pluginTrigger.subscription.listNum', { num: subscriptionCount })} + </span> + <Tooltip popupContent={t('pluginTrigger.subscription.list.tip')} /> + </div> + )} + <CreateSubscriptionButton + buttonType={subscriptionCount > 0 ? CreateButtonType.ICON_BUTTON : CreateButtonType.FULL_BUTTON} + /> + </div> + + {subscriptionCount > 0 && ( + <div className='flex flex-col gap-1'> + {subscriptions?.map(subscription => ( + <SubscriptionCard + key={subscription.id} + data={subscription} + /> + ))} + </div> + )} + </div> + ) +} diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/log-viewer.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/log-viewer.tsx new file mode 100644 index 0000000000..8b16d2c60a --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/log-viewer.tsx @@ -0,0 +1,193 @@ +'use client' +import React, { useState } from 'react' +import { useTranslation } from 'react-i18next' +import { + RiArrowDownSLine, + RiArrowRightSLine, + RiCheckboxCircleFill, + RiErrorWarningFill, + RiFileCopyLine, +} from '@remixicon/react' +import cn from '@/utils/classnames' +import Toast from '@/app/components/base/toast' +import CodeEditor from '@/app/components/workflow/nodes/_base/components/editor/code-editor' +import { CodeLanguage } from '@/app/components/workflow/nodes/code/types' +import type { TriggerLogEntity } from '@/app/components/workflow/block-selector/types' +import dayjs from 'dayjs' + +type Props = { + logs: TriggerLogEntity[] + className?: string +} + +enum LogTypeEnum { + REQUEST = 'request', + RESPONSE = 'response', +} + +const LogViewer = ({ logs, className }: Props) => { + const { t } = useTranslation() + const [expandedLogs, setExpandedLogs] = useState<Set<string>>(new Set()) + + const toggleLogExpansion = (logId: string) => { + const newExpanded = new Set(expandedLogs) + if (newExpanded.has(logId)) + newExpanded.delete(logId) + else + newExpanded.add(logId) + + setExpandedLogs(newExpanded) + } + + const parseRequestData = (data: any) => { + if (typeof data === 'string' && data.startsWith('payload=')) { + try { + const urlDecoded = decodeURIComponent(data.substring(8)) // Remove 'payload=' + return JSON.parse(urlDecoded) + } + catch { + return data + } + } + + if (typeof data === 'object') + return data + + try { + return JSON.parse(data) + } + catch { + return data + } + } + + const renderJsonContent = (originalData: any, title: LogTypeEnum) => { + const parsedData = title === LogTypeEnum.REQUEST ? { headers: originalData.headers, data: parseRequestData(originalData.data) } : originalData + const isJsonObject = typeof parsedData === 'object' + + if (isJsonObject) { + return ( + <CodeEditor + readOnly + title={<div className="system-xs-semibold-uppercase text-text-secondary">{title}</div>} + language={CodeLanguage.json} + value={parsedData} + isJSONStringifyBeauty + nodeId="" + /> + ) + } + + return ( + <div className='rounded-md bg-components-input-bg-normal'> + <div className='flex items-center justify-between px-2 py-1'> + <div className='system-xs-semibold-uppercase text-text-secondary'> + {title} + </div> + <button + onClick={(e) => { + e.stopPropagation() + navigator.clipboard.writeText(String(parsedData)) + Toast.notify({ + type: 'success', + message: t('common.actionMsg.copySuccessfully'), + }) + }} + className='rounded-md p-0.5 hover:bg-components-panel-border' + > + <RiFileCopyLine className='h-4 w-4 text-text-tertiary' /> + </button> + </div> + <div className='px-2 pb-2 pt-1'> + <pre className='code-xs-regular whitespace-pre-wrap break-all text-text-secondary'> + {String(parsedData)} + </pre> + </div> + </div> + ) + } + + if (!logs || logs.length === 0) + return null + + return ( + <div className={cn('flex flex-col gap-1', className)}> + {logs.map((log, index) => { + const logId = log.id || index.toString() + const isExpanded = expandedLogs.has(logId) + const isSuccess = log.response.status_code === 200 + const isError = log.response.status_code >= 400 + + return ( + <div + key={logId} + className={cn( + 'relative overflow-hidden rounded-lg border bg-components-panel-on-panel-item-bg shadow-sm hover:bg-components-panel-on-panel-item-bg-hover', + isError && 'border-state-destructive-border', + !isError && isExpanded && 'border-components-panel-border', + !isError && !isExpanded && 'border-components-panel-border-subtle', + )} + > + {isError && ( + <div className='pointer-events-none absolute left-0 top-0 h-7 w-[179px]'> + <svg xmlns="http://www.w3.org/2000/svg" width="179" height="28" viewBox="0 0 179 28" fill="none" className='h-full w-full'> + <g filter="url(#filter0_f_error_glow)"> + <circle cx="27" cy="14" r="32" fill="#F04438" fillOpacity="0.25" /> + </g> + <defs> + <filter id="filter0_f_error_glow" x="-125" y="-138" width="304" height="304" filterUnits="userSpaceOnUse" colorInterpolationFilters="sRGB"> + <feFlood floodOpacity="0" result="BackgroundImageFix" /> + <feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape" /> + <feGaussianBlur stdDeviation="60" result="effect1_foregroundBlur" /> + </filter> + </defs> + </svg> + </div> + )} + + <button + onClick={() => toggleLogExpansion(logId)} + className={cn( + 'flex w-full items-center justify-between px-2 py-1.5 text-left', + isExpanded ? 'pb-1 pt-2' : 'min-h-7', + )} + > + <div className='flex items-center gap-0'> + {isExpanded ? ( + <RiArrowDownSLine className='h-4 w-4 text-text-tertiary' /> + ) : ( + <RiArrowRightSLine className='h-4 w-4 text-text-tertiary' /> + )} + <div className='system-xs-semibold-uppercase text-text-secondary'> + {t(`pluginTrigger.modal.manual.logs.${LogTypeEnum.REQUEST}`)} #{index + 1} + </div> + </div> + + <div className='flex items-center gap-1'> + <div className='system-xs-regular text-text-tertiary'> + {dayjs(log.created_at).format('HH:mm:ss')} + </div> + <div className='h-3.5 w-3.5'> + {isSuccess ? ( + <RiCheckboxCircleFill className='h-full w-full text-text-success' /> + ) : ( + <RiErrorWarningFill className='h-full w-full text-text-destructive' /> + )} + </div> + </div> + </button> + + {isExpanded && ( + <div className='flex flex-col gap-1 px-1 pb-1'> + {renderJsonContent(log.request, LogTypeEnum.REQUEST)} + {renderJsonContent(log.response, LogTypeEnum.RESPONSE)} + </div> + )} + </div> + ) + })} + </div> + ) +} + +export default LogViewer diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/selector-entry.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/selector-entry.tsx new file mode 100644 index 0000000000..c23e022ac5 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/selector-entry.tsx @@ -0,0 +1,126 @@ +'use client' +import { + PortalToFollowElem, + PortalToFollowElemContent, + PortalToFollowElemTrigger, +} from '@/app/components/base/portal-to-follow-elem' +import type { SimpleSubscription } from '@/app/components/plugins/plugin-detail-panel/subscription-list' +import { SubscriptionList, SubscriptionListMode } from '@/app/components/plugins/plugin-detail-panel/subscription-list' +import cn from '@/utils/classnames' +import { RiArrowDownSLine, RiWebhookLine } from '@remixicon/react' +import { useMemo, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { useSubscriptionList } from './use-subscription-list' + +type SubscriptionTriggerButtonProps = { + selectedId?: string + onClick?: () => void + isOpen?: boolean + className?: string +} + +const SubscriptionTriggerButton: React.FC<SubscriptionTriggerButtonProps> = ({ + selectedId, + onClick, + isOpen = false, + className, +}) => { + const { t } = useTranslation() + const { subscriptions } = useSubscriptionList() + + const statusConfig = useMemo(() => { + if (!selectedId) { + if (isOpen) { + return { + label: t('pluginTrigger.subscription.selectPlaceholder'), + color: 'yellow' as const, + } + } + return { + label: t('pluginTrigger.subscription.noSubscriptionSelected'), + color: 'red' as const, + } + } + + if (subscriptions && subscriptions.length > 0) { + const selectedSubscription = subscriptions?.find(sub => sub.id === selectedId) + + if (!selectedSubscription) { + return { + label: t('pluginTrigger.subscription.subscriptionRemoved'), + color: 'red' as const, + } + } + + return { + label: selectedSubscription.name, + color: 'green' as const, + } + } + + return { + label: t('pluginTrigger.subscription.noSubscriptionSelected'), + color: 'red' as const, + } + }, [selectedId, subscriptions, t, isOpen]) + + return ( + <button + className={cn( + 'flex h-8 items-center gap-1 rounded-lg px-2 transition-colors', + 'hover:bg-state-base-hover-alt', + isOpen && 'bg-state-base-hover-alt', + className, + )} + onClick={onClick} + > + <RiWebhookLine className={cn('h-3.5 w-3.5 shrink-0 text-text-secondary', statusConfig.color === 'red' && 'text-components-button-destructive-secondary-text')} /> + <span className={cn('system-xs-medium truncate text-components-button-ghost-text', statusConfig.color === 'red' && 'text-components-button-destructive-secondary-text')}> + {statusConfig.label} + </span> + <RiArrowDownSLine + className={cn( + 'ml-auto h-4 w-4 shrink-0 text-text-quaternary transition-transform', + isOpen && 'rotate-180', + statusConfig.color === 'red' && 'text-components-button-destructive-secondary-text', + )} + /> + </button> + ) +} + +export const SubscriptionSelectorEntry = ({ selectedId, onSelect }: { + selectedId?: string, + onSelect: (v: SimpleSubscription, callback?: () => void) => void +}) => { + const [isOpen, setIsOpen] = useState(false) + + return <PortalToFollowElem + placement='bottom-start' + offset={4} + open={isOpen} + onOpenChange={setIsOpen} + > + <PortalToFollowElemTrigger asChild> + <div> + <SubscriptionTriggerButton + selectedId={selectedId} + onClick={() => setIsOpen(!isOpen)} + isOpen={isOpen} + /> + </div> + </PortalToFollowElemTrigger> + <PortalToFollowElemContent className='z-[11]'> + <div className='rounded-xl border border-components-panel-border bg-components-panel-bg shadow-lg'> + <SubscriptionList + mode={SubscriptionListMode.SELECTOR} + selectedId={selectedId} + onSelect={(...args) => { + onSelect(...args) + setIsOpen(false) + }} + /> + </div> + </PortalToFollowElemContent> + </PortalToFollowElem> +} diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/selector-view.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/selector-view.tsx new file mode 100644 index 0000000000..04b078e347 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/selector-view.tsx @@ -0,0 +1,90 @@ +'use client' +import ActionButton from '@/app/components/base/action-button' +import Tooltip from '@/app/components/base/tooltip' +import type { TriggerSubscription } from '@/app/components/workflow/block-selector/types' +import cn from '@/utils/classnames' +import { RiCheckLine, RiDeleteBinLine, RiWebhookLine } from '@remixicon/react' +import React, { useState } from 'react' +import { useTranslation } from 'react-i18next' +import { CreateButtonType, CreateSubscriptionButton } from './create' +import { DeleteConfirm } from './delete-confirm' +import { useSubscriptionList } from './use-subscription-list' + +type SubscriptionSelectorProps = { + selectedId?: string + onSelect?: ({ id, name }: { id: string, name: string }) => void +} + +export const SubscriptionSelectorView: React.FC<SubscriptionSelectorProps> = ({ + selectedId, + onSelect, +}) => { + const { t } = useTranslation() + const { subscriptions } = useSubscriptionList() + const [deletedSubscription, setDeletedSubscription] = useState<TriggerSubscription | null>(null) + const subscriptionCount = subscriptions?.length || 0 + + return ( + <div className='w-[320px] p-1'> + {subscriptionCount > 0 && <div className='ml-7 mr-1.5 flex h-8 items-center justify-between'> + <div className='flex shrink-0 items-center gap-1'> + <span className='system-sm-semibold-uppercase text-text-secondary'> + {t('pluginTrigger.subscription.listNum', { num: subscriptionCount })} + </span> + <Tooltip popupContent={t('pluginTrigger.subscription.list.tip')} /> + </div> + <CreateSubscriptionButton + buttonType={CreateButtonType.ICON_BUTTON} + shape='circle' + /> + </div>} + <div className='max-h-[320px] overflow-y-auto'> + {subscriptions?.map(subscription => ( + <div + key={subscription.id} + className={cn( + 'group flex w-full items-center justify-between rounded-lg p-1 text-left transition-colors', + 'hover:bg-state-base-hover has-[.subscription-delete-btn:hover]:!bg-state-destructive-hover', + selectedId === subscription.id && 'bg-state-base-hover', + )} + > + <button + type='button' + className='flex flex-1 items-center text-left' + onClick={() => onSelect?.(subscription)} + > + <div className='flex items-center'> + {selectedId === subscription.id && ( + <RiCheckLine className='mr-2 h-4 w-4 shrink-0 text-text-accent' /> + )} + <RiWebhookLine className={cn('mr-1.5 h-3.5 w-3.5 text-text-secondary', selectedId !== subscription.id && 'ml-6')} /> + <span className='system-md-regular leading-6 text-text-secondary'> + {subscription.name} + </span> + </div> + </button> + <ActionButton onClick={(e) => { + e.stopPropagation() + setDeletedSubscription(subscription) + }} className='subscription-delete-btn hidden shrink-0 text-text-tertiary hover:bg-state-destructive-hover hover:text-text-destructive group-hover:flex'> + <RiDeleteBinLine className='size-4' /> + </ActionButton> + </div> + ))} + </div> + {deletedSubscription && ( + <DeleteConfirm + onClose={(deleted) => { + if (deleted) + onSelect?.({ id: '', name: '' }) + setDeletedSubscription(null) + }} + isShow={!!deletedSubscription} + currentId={deletedSubscription.id} + currentName={deletedSubscription.name} + workflowsInUse={deletedSubscription.workflows_in_use} + /> + )} + </div> + ) +} diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/store.ts b/web/app/components/plugins/plugin-detail-panel/subscription-list/store.ts new file mode 100644 index 0000000000..24840e9971 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/store.ts @@ -0,0 +1,11 @@ +import { create } from 'zustand' + +type ShapeSubscription = { + refresh?: () => void + setRefresh: (refresh: () => void) => void +} + +export const usePluginSubscriptionStore = create<ShapeSubscription>(set => ({ + refresh: undefined, + setRefresh: (refresh: () => void) => set({ refresh }), +})) diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/subscription-card.tsx b/web/app/components/plugins/plugin-detail-panel/subscription-list/subscription-card.tsx new file mode 100644 index 0000000000..b2a86b5c76 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/subscription-card.tsx @@ -0,0 +1,85 @@ +'use client' +import ActionButton from '@/app/components/base/action-button' +import Tooltip from '@/app/components/base/tooltip' +import type { TriggerSubscription } from '@/app/components/workflow/block-selector/types' +import cn from '@/utils/classnames' +import { + RiDeleteBinLine, + RiWebhookLine, +} from '@remixicon/react' +import { useBoolean } from 'ahooks' +import { useTranslation } from 'react-i18next' +import { DeleteConfirm } from './delete-confirm' + +type Props = { + data: TriggerSubscription +} + +const SubscriptionCard = ({ data }: Props) => { + const { t } = useTranslation() + const [isShowDeleteModal, { + setTrue: showDeleteModal, + setFalse: hideDeleteModal, + }] = useBoolean(false) + + return ( + <> + <div + className={cn( + 'group relative cursor-pointer rounded-lg border-[0.5px] px-4 py-3 shadow-xs transition-all', + 'border-components-panel-border-subtle bg-components-panel-on-panel-item-bg', + 'hover:bg-components-panel-on-panel-item-bg-hover', + 'has-[.subscription-delete-btn:hover]:!border-state-destructive-border has-[.subscription-delete-btn:hover]:!bg-state-destructive-hover', + )} + > + <div className='flex items-center justify-between'> + <div className='flex h-6 items-center gap-1'> + <RiWebhookLine className='h-4 w-4 text-text-secondary' /> + <span className='system-md-semibold text-text-secondary'> + {data.name} + </span> + </div> + + <ActionButton + onClick={showDeleteModal} + className='subscription-delete-btn hidden transition-colors hover:bg-state-destructive-hover hover:text-text-destructive group-hover:block' + > + <RiDeleteBinLine className='h-4 w-4' /> + </ActionButton> + </div> + + <div className='mt-1 flex items-center justify-between'> + <Tooltip + disabled={!data.endpoint} + popupContent={data.endpoint && ( + <div className='max-w-[320px] break-all'> + {data.endpoint} + </div> + )} + position='left' + > + <div className='system-xs-regular flex-1 truncate text-text-tertiary'> + {data.endpoint} + </div> + </Tooltip> + <div className="mx-2 text-xs text-text-tertiary opacity-30">·</div> + <div className='system-xs-regular shrink-0 text-text-tertiary'> + {data.workflows_in_use > 0 ? t('pluginTrigger.subscription.list.item.usedByNum', { num: data.workflows_in_use }) : t('pluginTrigger.subscription.list.item.noUsed')} + </div> + </div> + </div> + + {isShowDeleteModal && ( + <DeleteConfirm + onClose={hideDeleteModal} + isShow={isShowDeleteModal} + currentId={data.id} + currentName={data.name} + workflowsInUse={data.workflows_in_use} + /> + )} + </> + ) +} + +export default SubscriptionCard diff --git a/web/app/components/plugins/plugin-detail-panel/subscription-list/use-subscription-list.ts b/web/app/components/plugins/plugin-detail-panel/subscription-list/use-subscription-list.ts new file mode 100644 index 0000000000..ff3e903a31 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/subscription-list/use-subscription-list.ts @@ -0,0 +1,23 @@ +import { useEffect } from 'react' +import { useTriggerSubscriptions } from '@/service/use-triggers' +import { usePluginStore } from '../store' +import { usePluginSubscriptionStore } from './store' + +export const useSubscriptionList = () => { + const detail = usePluginStore(state => state.detail) + const { setRefresh } = usePluginSubscriptionStore() + + const { data: subscriptions, isLoading, refetch } = useTriggerSubscriptions(detail?.provider || '') + + useEffect(() => { + if (refetch) + setRefresh(refetch) + }, [refetch, setRefresh]) + + return { + detail, + subscriptions, + isLoading, + refetch, + } +} diff --git a/web/app/components/plugins/plugin-detail-panel/tool-selector/index.tsx b/web/app/components/plugins/plugin-detail-panel/tool-selector/index.tsx index d2797b99f4..ea7892be32 100644 --- a/web/app/components/plugins/plugin-detail-panel/tool-selector/index.tsx +++ b/web/app/components/plugins/plugin-detail-panel/tool-selector/index.tsx @@ -40,6 +40,7 @@ import { AuthCategory, PluginAuthInAgent, } from '@/app/components/plugins/plugin-auth' +import { ReadmeEntrance } from '../../readme-panel/entrance' type Props = { disabled?: boolean @@ -272,7 +273,10 @@ const ToolSelector: FC<Props> = ({ {/* base form */} <div className='flex flex-col gap-3 px-4 py-2'> <div className='flex flex-col gap-1'> - <div className='system-sm-semibold flex h-6 items-center text-text-secondary'>{t('plugin.detailPanel.toolSelector.toolLabel')}</div> + <div className='system-sm-semibold flex h-6 items-center justify-between text-text-secondary'> + {t('plugin.detailPanel.toolSelector.toolLabel')} + <ReadmeEntrance pluginDetail={currentProvider as any} showShortTip className='pb-0' /> + </div> <ToolPicker placement='bottom' offset={offset} @@ -314,6 +318,8 @@ const ToolSelector: FC<Props> = ({ pluginPayload={{ provider: currentProvider.name, category: AuthCategory.tool, + providerType: currentProvider.type, + detail: currentProvider as any, }} credentialId={value?.credential_id} onAuthorizationItemClick={handleAuthorizationItemClick} diff --git a/web/app/components/plugins/plugin-detail-panel/tool-selector/reasoning-config-form.tsx b/web/app/components/plugins/plugin-detail-panel/tool-selector/reasoning-config-form.tsx index b79ee78664..88bf7f0dfd 100644 --- a/web/app/components/plugins/plugin-detail-panel/tool-selector/reasoning-config-form.tsx +++ b/web/app/components/plugins/plugin-detail-panel/tool-selector/reasoning-config-form.tsx @@ -54,7 +54,7 @@ const ReasoningConfigForm: React.FC<Props> = ({ const getVarKindType = (type: FormTypeEnum) => { if (type === FormTypeEnum.file || type === FormTypeEnum.files) return VarKindType.variable - if (type === FormTypeEnum.select || type === FormTypeEnum.boolean || type === FormTypeEnum.textNumber || type === FormTypeEnum.array || type === FormTypeEnum.object) + if (type === FormTypeEnum.select || type === FormTypeEnum.checkbox || type === FormTypeEnum.textNumber || type === FormTypeEnum.array || type === FormTypeEnum.object) return VarKindType.constant if (type === FormTypeEnum.textInput || type === FormTypeEnum.secretInput) return VarKindType.mixed @@ -164,7 +164,7 @@ const ReasoningConfigForm: React.FC<Props> = ({ const isArray = type === FormTypeEnum.array const isShowJSONEditor = isObject || isArray const isFile = type === FormTypeEnum.file || type === FormTypeEnum.files - const isBoolean = type === FormTypeEnum.boolean + const isBoolean = type === FormTypeEnum.checkbox const isSelect = type === FormTypeEnum.select const isAppSelector = type === FormTypeEnum.appSelector const isModelSelector = type === FormTypeEnum.modelSelector diff --git a/web/app/components/plugins/plugin-detail-panel/trigger/event-detail-drawer.tsx b/web/app/components/plugins/plugin-detail-panel/trigger/event-detail-drawer.tsx new file mode 100644 index 0000000000..2083f34263 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/trigger/event-detail-drawer.tsx @@ -0,0 +1,157 @@ +'use client' +import ActionButton from '@/app/components/base/action-button' +import Divider from '@/app/components/base/divider' +import Drawer from '@/app/components/base/drawer' +import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' +import Icon from '@/app/components/plugins/card/base/card-icon' +import Description from '@/app/components/plugins/card/base/description' +import OrgInfo from '@/app/components/plugins/card/base/org-info' +import { triggerEventParametersToFormSchemas } from '@/app/components/tools/utils/to-form-schema' +import type { TriggerProviderApiEntity } from '@/app/components/workflow/block-selector/types' +import Field from '@/app/components/workflow/nodes/_base/components/variable/object-child-tree-panel/show/field' +import cn from '@/utils/classnames' +import { + RiArrowLeftLine, + RiCloseLine, +} from '@remixicon/react' +import type { TFunction } from 'i18next' +import type { FC } from 'react' +import { useTranslation } from 'react-i18next' +import type { TriggerEvent } from '@/app/components/plugins/types' + +type EventDetailDrawerProps = { + eventInfo: TriggerEvent + providerInfo: TriggerProviderApiEntity + onClose: () => void +} + +const getType = (type: string, t: TFunction) => { + if (type === 'number-input') + return t('tools.setBuiltInTools.number') + if (type === 'text-input') + return t('tools.setBuiltInTools.string') + if (type === 'checkbox') + return 'boolean' + if (type === 'file') + return t('tools.setBuiltInTools.file') + return type +} + +// Convert JSON Schema to StructuredOutput format +const convertSchemaToField = (schema: any): any => { + const field: any = { + type: Array.isArray(schema.type) ? schema.type[0] : schema.type || 'string', + } + + if (schema.description) + field.description = schema.description + + if (schema.properties) { + field.properties = Object.entries(schema.properties).reduce((acc, [key, value]: [string, any]) => ({ + ...acc, + [key]: convertSchemaToField(value), + }), {}) + } + + if (schema.required) + field.required = schema.required + + if (schema.items) + field.items = convertSchemaToField(schema.items) + + if (schema.enum) + field.enum = schema.enum + + return field +} + +export const EventDetailDrawer: FC<EventDetailDrawerProps> = (props) => { + const { eventInfo, providerInfo, onClose } = props + const language = useLanguage() + const { t } = useTranslation() + const parametersSchemas = triggerEventParametersToFormSchemas(eventInfo.parameters) + + // Convert output_schema properties to array for direct rendering + const outputFields = eventInfo.output_schema?.properties + ? Object.entries(eventInfo.output_schema.properties).map(([name, schema]: [string, any]) => ({ + name, + field: convertSchemaToField(schema), + required: eventInfo.output_schema.required?.includes(name) || false, + })) + : [] + + return ( + <Drawer + isOpen + clickOutsideNotOpen={false} + onClose={onClose} + footer={null} + mask={false} + positionCenter={false} + panelClassName={cn('mb-2 mr-2 mt-[64px] !w-[420px] !max-w-[420px] justify-start rounded-2xl border-[0.5px] border-components-panel-border !bg-components-panel-bg !p-0 shadow-xl')} + > + <div className='relative border-b border-divider-subtle p-4 pb-3'> + <div className='absolute right-3 top-3'> + <ActionButton onClick={onClose}> + <RiCloseLine className='h-4 w-4' /> + </ActionButton> + </div> + <div + className='system-xs-semibold-uppercase mb-2 flex cursor-pointer items-center gap-1 text-text-accent-secondary' + onClick={onClose} + > + <RiArrowLeftLine className='h-4 w-4' /> + {t('plugin.detailPanel.operation.back')} + </div> + <div className='flex items-center gap-1'> + <Icon size='tiny' className='h-6 w-6' src={providerInfo.icon!} /> + <OrgInfo + packageNameClassName='w-auto' + orgName={providerInfo.author} + packageName={providerInfo.name.split('/').pop() || ''} + /> + </div> + <div className='system-md-semibold mt-1 text-text-primary'>{eventInfo?.identity?.label[language]}</div> + <Description className='mb-2 mt-3 h-auto' text={eventInfo.description[language]} descriptionLineRows={2}></Description> + </div> + <div className='flex h-full flex-col gap-2 overflow-y-auto px-4 pb-2 pt-4'> + <div className='system-sm-semibold-uppercase text-text-secondary'>{t('tools.setBuiltInTools.parameters')}</div> + {parametersSchemas.length > 0 ? ( + parametersSchemas.map((item, index) => ( + <div key={index} className='py-1'> + <div className='flex items-center gap-2'> + <div className='code-sm-semibold text-text-secondary'>{item.label[language]}</div> + <div className='system-xs-regular text-text-tertiary'> + {getType(item.type, t)} + </div> + {item.required && ( + <div className='system-xs-medium text-text-warning-secondary'>{t('tools.setBuiltInTools.required')}</div> + )} + </div> + {item.description && ( + <div className='system-xs-regular mt-0.5 text-text-tertiary'> + {item.description?.[language]} + </div> + )} + </div> + )) + ) : <div className='system-xs-regular text-text-tertiary'>{t('pluginTrigger.events.item.noParameters')}</div>} + <Divider className='mb-2 mt-1 h-px' /> + <div className='flex flex-col gap-2'> + <div className='system-sm-semibold-uppercase text-text-secondary'>{t('pluginTrigger.events.output')}</div> + <div className='relative left-[-7px]'> + {outputFields.map(item => ( + <Field + key={item.name} + name={item.name} + payload={item.field} + required={item.required} + rootClassName='code-sm-semibold text-text-secondary' + /> + ))} + </div> + </div> + </div> + </Drawer> + ) +} diff --git a/web/app/components/plugins/plugin-detail-panel/trigger/event-list.tsx b/web/app/components/plugins/plugin-detail-panel/trigger/event-list.tsx new file mode 100644 index 0000000000..93f2fcc9c7 --- /dev/null +++ b/web/app/components/plugins/plugin-detail-panel/trigger/event-list.tsx @@ -0,0 +1,71 @@ +import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' +import type { TriggerEvent } from '@/app/components/plugins/types' +import type { TriggerProviderApiEntity } from '@/app/components/workflow/block-selector/types' +import { useTriggerProviderInfo } from '@/service/use-triggers' +import cn from '@/utils/classnames' +import { useState } from 'react' +import { useTranslation } from 'react-i18next' +import { usePluginStore } from '../store' +import { EventDetailDrawer } from './event-detail-drawer' + +type TriggerEventCardProps = { + eventInfo: TriggerEvent + providerInfo: TriggerProviderApiEntity +} + +const TriggerEventCard = ({ eventInfo, providerInfo }: TriggerEventCardProps) => { + const { identity, description } = eventInfo + const language = useLanguage() + const [showDetail, setShowDetail] = useState(false) + const title = identity.label?.[language] ?? identity.label?.en_US ?? '' + const descriptionText = description?.[language] ?? description?.en_US ?? '' + return ( + <> + <div + className={cn('bg-components-panel-item-bg cursor-pointer rounded-xl border-[0.5px] border-components-panel-border-subtle px-4 py-3 shadow-xs hover:bg-components-panel-on-panel-item-bg-hover')} + onClick={() => setShowDetail(true)} + > + <div className='system-md-semibold pb-0.5 text-text-secondary'>{title}</div> + <div className='system-xs-regular line-clamp-2 text-text-tertiary' title={descriptionText}>{descriptionText}</div> + </div> + {showDetail && ( + <EventDetailDrawer + eventInfo={eventInfo} + providerInfo={providerInfo} + onClose={() => setShowDetail(false)} + /> + )} + </> + ) +} + +export const TriggerEventsList = () => { + const { t } = useTranslation() + const detail = usePluginStore(state => state.detail) + + const { data: providerInfo } = useTriggerProviderInfo(detail?.provider || '') + const triggerEvents = providerInfo?.events || [] + + if (!providerInfo || !triggerEvents.length) + return null + + return ( + <div className='px-4 pb-4 pt-2'> + <div className='mb-1 py-1'> + <div className='system-sm-semibold-uppercase mb-1 flex h-6 items-center justify-between text-text-secondary'> + {t('pluginTrigger.events.actionNum', { num: triggerEvents.length, event: t(`pluginTrigger.events.${triggerEvents.length > 1 ? 'events' : 'event'}`) })} + </div> + </div> + <div className='flex flex-col gap-2'> + { + triggerEvents.map((triggerEvent: TriggerEvent) => ( + <TriggerEventCard + key={`${detail?.plugin_id}${triggerEvent.identity?.name || ''}`} + eventInfo={triggerEvent} + providerInfo={providerInfo} + />)) + } + </div> + </div> + ) +} diff --git a/web/app/components/plugins/plugin-item/action.tsx b/web/app/components/plugins/plugin-item/action.tsx index 6d7ff388ea..80dfd78f12 100644 --- a/web/app/components/plugins/plugin-item/action.tsx +++ b/web/app/components/plugins/plugin-item/action.tsx @@ -14,7 +14,7 @@ import { useGitHubReleases } from '../install-plugin/hooks' import Toast from '@/app/components/base/toast' import { useModalContext } from '@/context/modal-context' import { useInvalidateInstalledPluginList } from '@/service/use-plugins' -import type { PluginType } from '@/app/components/plugins/types' +import type { PluginCategoryEnum } from '@/app/components/plugins/types' const i18nPrefix = 'plugin.action' @@ -23,7 +23,7 @@ type Props = { installationId: string pluginUniqueIdentifier: string pluginName: string - category: PluginType + category: PluginCategoryEnum usedInApps: number isShowFetchNewVersion: boolean isShowInfo: boolean @@ -92,11 +92,18 @@ const Action: FC<Props> = ({ const handleDelete = useCallback(async () => { showDeleting() - const res = await uninstallPlugin(installationId) - hideDeleting() - if (res.success) { - hideDeleteConfirm() - onDelete() + try{ + const res = await uninstallPlugin(installationId) + if (res.success) { + hideDeleteConfirm() + onDelete() + } + } + catch (error) { + console.error('uninstallPlugin error', error) + } + finally { + hideDeleting() } }, [installationId, onDelete]) return ( diff --git a/web/app/components/plugins/plugin-item/index.tsx b/web/app/components/plugins/plugin-item/index.tsx index ed7cf47bb7..9352df23c8 100644 --- a/web/app/components/plugins/plugin-item/index.tsx +++ b/web/app/components/plugins/plugin-item/index.tsx @@ -1,7 +1,12 @@ 'use client' -import type { FC } from 'react' -import React, { useCallback, useMemo } from 'react' -import { useTheme } from 'next-themes' +import Tooltip from '@/app/components/base/tooltip' +import useRefreshPluginList from '@/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list' +import { API_PREFIX } from '@/config' +import { useAppContext } from '@/context/app-context' +import { useGlobalPublicStore } from '@/context/global-public-context' +import { useRenderI18nObject } from '@/hooks/use-i18n' +import cn from '@/utils/classnames' +import { getMarketplaceUrl } from '@/utils/var' import { RiArrowRightUpLine, RiBugLine, @@ -10,26 +15,21 @@ import { RiLoginCircleLine, RiVerifiedBadgeLine, } from '@remixicon/react' +import { useTheme } from 'next-themes' +import type { FC } from 'react' +import React, { useCallback, useMemo } from 'react' import { useTranslation } from 'react-i18next' -import { usePluginPageContext } from '../plugin-page/context' -import { Github } from '../../base/icons/src/public/common' +import { gte } from 'semver' import Badge from '../../base/badge' -import { type PluginDetail, PluginSource, PluginType } from '../types' +import { Github } from '../../base/icons/src/public/common' import CornerMark from '../card/base/corner-mark' import Description from '../card/base/description' import OrgInfo from '../card/base/org-info' import Title from '../card/base/title' +import { useCategories } from '../hooks' +import { usePluginPageContext } from '../plugin-page/context' +import { PluginCategoryEnum, type PluginDetail, PluginSource } from '../types' import Action from './action' -import cn from '@/utils/classnames' -import { API_PREFIX } from '@/config' -import { useSingleCategories } from '../hooks' -import { useRenderI18nObject } from '@/hooks/use-i18n' -import useRefreshPluginList from '@/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list' -import { useAppContext } from '@/context/app-context' -import { gte } from 'semver' -import Tooltip from '@/app/components/base/tooltip' -import { getMarketplaceUrl } from '@/utils/var' -import { useGlobalPublicStore } from '@/context/global-public-context' type Props = { className?: string @@ -42,7 +42,7 @@ const PluginItem: FC<Props> = ({ }) => { const { t } = useTranslation() const { theme } = useTheme() - const { categoriesMap } = useSingleCategories() + const { categoriesMap } = useCategories(t, true) const currentPluginID = usePluginPageContext(v => v.currentPluginID) const setCurrentPluginID = usePluginPageContext(v => v.setCurrentPluginID) const { refreshPluginList } = useRefreshPluginList() @@ -150,7 +150,7 @@ const PluginItem: FC<Props> = ({ packageName={name} packageNameClassName='w-auto max-w-[150px]' /> - {category === PluginType.extension && ( + {category === PluginCategoryEnum.extension && ( <> <div className='system-xs-regular mx-2 text-text-quaternary'>·</div> <div className='system-xs-regular flex items-center gap-x-1 overflow-hidden text-text-tertiary'> diff --git a/web/app/components/plugins/plugin-page/index.tsx b/web/app/components/plugins/plugin-page/index.tsx index d326fdf6e4..4b8444ab34 100644 --- a/web/app/components/plugins/plugin-page/index.tsx +++ b/web/app/components/plugins/plugin-page/index.tsx @@ -72,6 +72,8 @@ const PluginPage = ({ } }, [searchParams]) + const [uniqueIdentifier, setUniqueIdentifier] = useState<string | null>(null) + const [dependencies, setDependencies] = useState<Dependency[]>([]) const bundleInfo = useMemo(() => { const info = searchParams.get(BUNDLE_INFO_KEY) @@ -99,6 +101,7 @@ const PluginPage = ({ useEffect(() => { (async () => { + setUniqueIdentifier(null) await sleep(100) if (packageId) { const { data } = await fetchManifestFromMarketPlace(encodeURIComponent(packageId)) @@ -108,6 +111,7 @@ const PluginPage = ({ version: version.version, icon: `${MARKETPLACE_API_PREFIX}/plugins/${plugin.org}/${plugin.name}/icon`, }) + setUniqueIdentifier(packageId) showInstallFromMarketplace() return } @@ -283,10 +287,10 @@ const PluginPage = ({ )} { - isShowInstallFromMarketplace && ( + isShowInstallFromMarketplace && uniqueIdentifier && ( <InstallFromMarketplace manifest={manifest! as PluginManifestInMarket} - uniqueIdentifier={packageId} + uniqueIdentifier={uniqueIdentifier} isBundle={!!bundleInfo} dependencies={dependencies} onClose={hideInstallFromMarketplace} diff --git a/web/app/components/plugins/plugin-page/plugin-tasks/index.tsx b/web/app/components/plugins/plugin-page/plugin-tasks/index.tsx index 3ddc2be549..c0bf5824e7 100644 --- a/web/app/components/plugins/plugin-page/plugin-tasks/index.tsx +++ b/web/app/components/plugins/plugin-page/plugin-tasks/index.tsx @@ -63,7 +63,7 @@ const PluginTasks = () => { return ( <div - className='flex items-center' + className={cn('flex items-center', opacity < 0 && 'hidden')} style={{ opacity }} > <PortalToFollowElem diff --git a/web/app/components/plugins/readme-panel/constants.ts b/web/app/components/plugins/readme-panel/constants.ts new file mode 100644 index 0000000000..7d6782e665 --- /dev/null +++ b/web/app/components/plugins/readme-panel/constants.ts @@ -0,0 +1,6 @@ +export const BUILTIN_TOOLS_ARRAY = [ + 'code', + 'audio', + 'time', + 'webscraper', +] diff --git a/web/app/components/plugins/readme-panel/entrance.tsx b/web/app/components/plugins/readme-panel/entrance.tsx new file mode 100644 index 0000000000..f3b4c98412 --- /dev/null +++ b/web/app/components/plugins/readme-panel/entrance.tsx @@ -0,0 +1,49 @@ +import React from 'react' +import { useTranslation } from 'react-i18next' +import { RiBookReadLine } from '@remixicon/react' +import cn from '@/utils/classnames' +import { ReadmeShowType, useReadmePanelStore } from './store' +import { BUILTIN_TOOLS_ARRAY } from './constants' +import type { PluginDetail } from '../types' + +export const ReadmeEntrance = ({ + pluginDetail, + showType = ReadmeShowType.drawer, + className, + showShortTip = false, +}: { + pluginDetail: PluginDetail + showType?: ReadmeShowType + className?: string + showShortTip?: boolean +}) => { + const { t } = useTranslation() + const { setCurrentPluginDetail } = useReadmePanelStore() + + const handleReadmeClick = () => { + if (pluginDetail) + setCurrentPluginDetail(pluginDetail, showType) + } + if (!pluginDetail || BUILTIN_TOOLS_ARRAY.includes(pluginDetail.id)) + return null + + return ( + <div className={cn('flex flex-col items-start justify-center gap-2 pb-4 pt-0', showType === ReadmeShowType.drawer && 'px-4', className)}> + {!showShortTip && <div className="relative h-1 w-8 shrink-0"> + <div className="h-px w-full bg-divider-regular"></div> + </div>} + + <button + onClick={handleReadmeClick} + className="flex w-full items-center justify-start gap-1 text-text-tertiary transition-opacity hover:text-text-accent-light-mode-only" + > + <div className="relative flex h-3 w-3 items-center justify-center overflow-hidden"> + <RiBookReadLine className="h-3 w-3" /> + </div> + <span className="text-xs font-normal leading-4"> + {!showShortTip ? t('plugin.readmeInfo.needHelpCheckReadme') : t('plugin.readmeInfo.title')} + </span> + </button> + </div> + ) +} diff --git a/web/app/components/plugins/readme-panel/index.tsx b/web/app/components/plugins/readme-panel/index.tsx new file mode 100644 index 0000000000..70d1e0db2c --- /dev/null +++ b/web/app/components/plugins/readme-panel/index.tsx @@ -0,0 +1,120 @@ +'use client' +import ActionButton from '@/app/components/base/action-button' +import Loading from '@/app/components/base/loading' +import { Markdown } from '@/app/components/base/markdown' +import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' +import { usePluginReadme } from '@/service/use-plugins' +import cn from '@/utils/classnames' +import { RiBookReadLine, RiCloseLine } from '@remixicon/react' +import type { FC } from 'react' +import { createPortal } from 'react-dom' +import { useTranslation } from 'react-i18next' +import DetailHeader from '../plugin-detail-panel/detail-header' +import { ReadmeShowType, useReadmePanelStore } from './store' + +const ReadmePanel: FC = () => { + const { currentPluginDetail, setCurrentPluginDetail } = useReadmePanelStore() + const { detail, showType } = currentPluginDetail || {} + const { t } = useTranslation() + const language = useLanguage() + + const pluginUniqueIdentifier = detail?.plugin_unique_identifier || '' + + const { data: readmeData, isLoading, error } = usePluginReadme( + { plugin_unique_identifier: pluginUniqueIdentifier, language: language === 'zh-Hans' ? undefined : language }, + ) + + const onClose = () => { + setCurrentPluginDetail() + } + + if (!detail) return null + + const children = ( + <div className="flex h-full w-full flex-col overflow-hidden"> + <div className="rounded-t-xl bg-background-body px-4 py-4"> + <div className="mb-3 flex items-center justify-between"> + <div className="flex items-center gap-1"> + <RiBookReadLine className="h-3 w-3 text-text-tertiary" /> + <span className="text-xs font-medium uppercase text-text-tertiary"> + {t('plugin.readmeInfo.title')} + </span> + </div> + <ActionButton onClick={onClose}> + <RiCloseLine className='h-4 w-4' /> + </ActionButton> + </div> + <DetailHeader detail={detail} isReadmeView={true} /> + </div> + + <div className="flex-1 overflow-y-auto px-4 py-3"> + {(() => { + if (isLoading) { + return ( + <div className="flex h-40 items-center justify-center"> + <Loading type="area" /> + </div> + ) + } + + if (error) { + return ( + <div className="py-8 text-center text-text-tertiary"> + <p>{t('plugin.readmeInfo.failedToFetch')}</p> + </div> + ) + } + + if (readmeData?.readme) { + return ( + <Markdown + content={readmeData.readme} + pluginInfo={{ pluginUniqueIdentifier, pluginId: detail.plugin_id }} + /> + ) + } + + return ( + <div className="py-8 text-center text-text-tertiary"> + <p>{t('plugin.readmeInfo.noReadmeAvailable')}</p> + </div> + ) + })()} + </div> + </div> + ) + + const portalContent = showType === ReadmeShowType.drawer + ? ( + <div className='pointer-events-none fixed inset-0 z-[9997] flex justify-start'> + <div + className={cn( + 'pointer-events-auto mb-2 ml-2 mr-2 mt-16 w-[600px] max-w-[600px] justify-start rounded-2xl border-[0.5px] border-components-panel-border bg-components-panel-bg p-0 shadow-xl', + )} + > + {children} + </div> + </div> + ) + : ( + <div className='pointer-events-none fixed inset-0 z-[9997] flex items-center justify-center p-2'> + <div + className={cn( + 'pointer-events-auto relative h-[calc(100vh-16px)] w-full max-w-[800px] rounded-2xl bg-components-panel-bg p-0 shadow-xl', + )} + onClick={(event) => { + event.stopPropagation() + }} + > + {children} + </div> + </div> + ) + + return createPortal( + portalContent, + document.body, + ) +} + +export default ReadmePanel diff --git a/web/app/components/plugins/readme-panel/store.ts b/web/app/components/plugins/readme-panel/store.ts new file mode 100644 index 0000000000..29c989db10 --- /dev/null +++ b/web/app/components/plugins/readme-panel/store.ts @@ -0,0 +1,25 @@ +import { create } from 'zustand' +import type { PluginDetail } from '@/app/components/plugins/types' + +export enum ReadmeShowType { + drawer = 'drawer', + modal = 'modal', +} + +type Shape = { + currentPluginDetail?: { + detail: PluginDetail + showType: ReadmeShowType + } + setCurrentPluginDetail: (detail?: PluginDetail, showType?: ReadmeShowType) => void +} + +export const useReadmePanelStore = create<Shape>(set => ({ + currentPluginDetail: undefined, + setCurrentPluginDetail: (detail?: PluginDetail, showType?: ReadmeShowType) => set({ + currentPluginDetail: !detail ? undefined : { + detail, + showType: showType ?? ReadmeShowType.drawer, + }, + }), +})) diff --git a/web/app/components/plugins/reference-setting-modal/auto-update-setting/index.tsx b/web/app/components/plugins/reference-setting-modal/auto-update-setting/index.tsx index 2d00788142..dfbeaad9cb 100644 --- a/web/app/components/plugins/reference-setting-modal/auto-update-setting/index.tsx +++ b/web/app/components/plugins/reference-setting-modal/auto-update-setting/index.tsx @@ -15,6 +15,7 @@ import { RiTimeLine } from '@remixicon/react' import cn from '@/utils/classnames' import { convertTimezoneToOffsetStr } from '@/app/components/base/date-and-time-picker/utils/dayjs' import { useModalContextSelector } from '@/context/modal-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' const i18nPrefix = 'plugin.autoUpdate' @@ -30,7 +31,7 @@ const SettingTimeZone: FC<{ }) => { const setShowAccountSettingModal = useModalContextSelector(s => s.setShowAccountSettingModal) return ( - <span className='body-xs-regular cursor-pointer text-text-accent' onClick={() => setShowAccountSettingModal({ payload: 'language' })} >{children}</span> + <span className='body-xs-regular cursor-pointer text-text-accent' onClick={() => setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.LANGUAGE })} >{children}</span> ) } const AutoUpdateSetting: FC<Props> = ({ @@ -143,6 +144,7 @@ const AutoUpdateSetting: FC<Props> = ({ title={t(`${i18nPrefix}.updateTime`)} minuteFilter={minuteFilter} renderTrigger={renderTimePickerTrigger} + placement='bottom-end' /> <div className='body-xs-regular mt-1 text-right text-text-tertiary'> <Trans diff --git a/web/app/components/plugins/types.ts b/web/app/components/plugins/types.ts index 0f312476d5..d9659df3ad 100644 --- a/web/app/components/plugins/types.ts +++ b/web/app/components/plugins/types.ts @@ -3,12 +3,16 @@ import type { ToolCredential } from '@/app/components/tools/types' import type { Locale } from '@/i18n-config' import type { AgentFeature } from '@/app/components/workflow/nodes/agent/types' import type { AutoUpdateConfig } from './reference-setting-modal/auto-update-setting/types' -export enum PluginType { +import type { FormTypeEnum } from '../base/form/types' +import type { TypeWithI18N } from '@/app/components/base/form/types' + +export enum PluginCategoryEnum { tool = 'tool', model = 'model', extension = 'extension', agent = 'agent-strategy', datasource = 'datasource', + trigger = 'trigger', } export enum PluginSource { @@ -68,7 +72,7 @@ export type PluginDeclaration = { author: string icon: string name: string - category: PluginType + category: PluginCategoryEnum label: Record<Locale, string> description: Record<Locale, string> created_at: string @@ -82,6 +86,111 @@ export type PluginDeclaration = { tags: string[] agent_strategy: any meta: PluginDeclarationMeta + trigger: PluginTriggerDefinition +} + +export type PluginTriggerSubscriptionConstructor = { + credentials_schema: CredentialsSchema[] + oauth_schema: OauthSchema + parameters: ParametersSchema[] +} + +export type PluginTriggerDefinition = { + events: TriggerEvent[] + identity: Identity + subscription_constructor: PluginTriggerSubscriptionConstructor + subscription_schema: ParametersSchema[] +} + +export type CredentialsSchema = { + name: string + label: Record<Locale, string> + description: Record<Locale, string> + type: FormTypeEnum + scope: any + required: boolean + default: any + options: any + help: Record<Locale, string> + url: string + placeholder: Record<Locale, string> +} + +export type OauthSchema = { + client_schema: CredentialsSchema[] + credentials_schema: CredentialsSchema[] +} + +export type ParametersSchema = { + name: string + label: Record<Locale, string> + type: FormTypeEnum + auto_generate: any + template: any + scope: any + required: boolean + multiple: boolean + default?: string[] + min: any + max: any + precision: any + options?: Array<{ + value: string + label: Record<Locale, string> + icon?: string + }> + description: Record<Locale, string> +} + +export type PropertiesSchema = { + type: FormTypeEnum + name: string + scope: any + required: boolean + default: any + options: Array<{ + value: string + label: Record<Locale, string> + icon?: string + }> + label: Record<Locale, string> + help: Record<Locale, string> + url: any + placeholder: any +} + +export type TriggerEventParameter = { + name: string + label: TypeWithI18N + type: string + auto_generate: any + template: any + scope: any + required: boolean + multiple: boolean + default: any + min: any + max: any + precision: any + options?: Array<{ + value: string + label: TypeWithI18N + icon?: string + }> + description?: TypeWithI18N +} + +export type TriggerEvent = { + name: string + identity: { + author: string + name: string + label: TypeWithI18N + provider?: string + } + description: TypeWithI18N + parameters: TriggerEventParameter[] + output_schema: Record<string, any> } export type PluginManifestInMarket = { @@ -90,7 +199,7 @@ export type PluginManifestInMarket = { org: string icon: string label: Record<Locale, string> - category: PluginType + category: PluginCategoryEnum version: string // combine the other place to it latest_version: string brief: Record<Locale, string> @@ -104,6 +213,12 @@ export type PluginManifestInMarket = { from: Dependency['type'] } +export enum SupportedCreationMethods { + OAUTH = 'OAUTH', + APIKEY = 'APIKEY', + MANUAL = 'MANUAL', +} + export type PluginDetail = { id: string created_at: string @@ -127,7 +242,7 @@ export type PluginDetail = { } export type PluginInfoFromMarketPlace = { - category: PluginType + category: PluginCategoryEnum latest_package_identifier: string latest_version: string } @@ -149,7 +264,7 @@ export type Plugin = { // Repo readme.md content introduction: string repository: string - category: PluginType + category: PluginCategoryEnum install_count: number endpoint: { settings: CredentialFormSchemaBase[] @@ -179,7 +294,7 @@ export type ReferenceSetting = { } export type UpdateFromMarketPlacePayload = { - category: PluginType + category: PluginCategoryEnum originalPackageInfo: { id: string payload: PluginDeclaration @@ -202,7 +317,7 @@ export type UpdateFromGitHubPayload = { export type UpdatePluginPayload = { type: PluginSource - category: PluginType + category: PluginCategoryEnum marketPlace?: UpdateFromMarketPlacePayload github?: UpdateFromGitHubPayload } @@ -280,6 +395,12 @@ export type InstallPackageResponse = { } export type InstallStatusResponse = { + status: TaskStatus, + taskId: string, + uniqueIdentifier: string, +} + +export type InstallStatus = { success: boolean, isFromMarketPlace?: boolean } @@ -463,15 +584,18 @@ export type StrategyDetail = { features: AgentFeature[] } +export type Identity = { + author: string + name: string + label: Record<Locale, string> + description: Record<Locale, string> + icon: string + icon_dark?: string + tags: string[] +} + export type StrategyDeclaration = { - identity: { - author: string - name: string - description: Record<Locale, string> - icon: string - label: Record<Locale, string> - tags: string[] - }, + identity: Identity, plugin_id: string strategies: StrategyDetail[] } diff --git a/web/app/components/rag-pipeline/components/panel/index.tsx b/web/app/components/rag-pipeline/components/panel/index.tsx index e2fd958405..793248e3eb 100644 --- a/web/app/components/rag-pipeline/components/panel/index.tsx +++ b/web/app/components/rag-pipeline/components/panel/index.tsx @@ -22,14 +22,19 @@ const InputFieldEditorPanel = dynamic(() => import('./input-field/editor'), { const PreviewPanel = dynamic(() => import('./input-field/preview'), { ssr: false, }) - +const GlobalVariablePanel = dynamic(() => import('@/app/components/workflow/panel/global-variable-panel'), { + ssr: false, +}) const RagPipelinePanelOnRight = () => { const historyWorkflowData = useStore(s => s.historyWorkflowData) const showDebugAndPreviewPanel = useStore(s => s.showDebugAndPreviewPanel) + const showGlobalVariablePanel = useStore(s => s.showGlobalVariablePanel) + return ( <> {historyWorkflowData && <Record />} {showDebugAndPreviewPanel && <TestRunPanel />} + {showGlobalVariablePanel && <GlobalVariablePanel />} </> ) } diff --git a/web/app/components/rag-pipeline/hooks/use-nodes-sync-draft.ts b/web/app/components/rag-pipeline/hooks/use-nodes-sync-draft.ts index d9de69716e..b70a2e6a34 100644 --- a/web/app/components/rag-pipeline/hooks/use-nodes-sync-draft.ts +++ b/web/app/components/rag-pipeline/hooks/use-nodes-sync-draft.ts @@ -7,6 +7,7 @@ import { import { useNodesReadOnly, } from '@/app/components/workflow/hooks/use-workflow' +import { useSerialAsyncCallback } from '@/app/components/workflow/hooks/use-serial-async-callback' import { API_PREFIX } from '@/config' import { syncWorkflowDraft } from '@/service/workflow' import { usePipelineRefreshDraft } from '.' @@ -83,7 +84,7 @@ export const useNodesSyncDraft = () => { } }, [getPostParams, getNodesReadOnly]) - const doSyncWorkflowDraft = useCallback(async ( + const performSync = useCallback(async ( notRefreshWhenSyncError?: boolean, callback?: { onSuccess?: () => void @@ -121,6 +122,8 @@ export const useNodesSyncDraft = () => { } }, [getPostParams, getNodesReadOnly, workflowStore, handleRefreshWorkflowDraft]) + const doSyncWorkflowDraft = useSerialAsyncCallback(performSync, getNodesReadOnly) + return { doSyncWorkflowDraft, syncWorkflowDraftWhenPageClose, diff --git a/web/app/components/rag-pipeline/hooks/use-pipeline-init.ts b/web/app/components/rag-pipeline/hooks/use-pipeline-init.ts index c70bce8523..6af72bee05 100644 --- a/web/app/components/rag-pipeline/hooks/use-pipeline-init.ts +++ b/web/app/components/rag-pipeline/hooks/use-pipeline-init.ts @@ -60,7 +60,10 @@ export const usePipelineInit = () => { if (error && error.json && !error.bodyUsed && datasetId) { error.json().then((err: any) => { if (err.code === 'draft_workflow_not_exist') { - workflowStore.setState({ notInitialWorkflow: true }) + workflowStore.setState({ + notInitialWorkflow: true, + shouldAutoOpenStartNodeSelector: true, + }) syncWorkflowDraft({ url: `/rag/pipelines/${datasetId}/workflows/draft`, params: { diff --git a/web/app/components/share/text-generation/run-once/index.tsx b/web/app/components/share/text-generation/run-once/index.tsx index d24428f32a..112f08a1d7 100644 --- a/web/app/components/share/text-generation/run-once/index.tsx +++ b/web/app/components/share/text-generation/run-once/index.tsx @@ -100,7 +100,10 @@ const RunOnce: FC<IRunOnceProps> = ({ : promptConfig.prompt_variables.map(item => ( <div className='mt-4 w-full' key={item.key}> {item.type !== 'checkbox' && ( - <label className='system-md-semibold flex h-6 items-center text-text-secondary'>{item.name}</label> + <div className='system-md-semibold flex h-6 items-center gap-1 text-text-secondary'> + <div className='truncate'>{item.name}</div> + {!item.required && <span className='system-xs-regular text-text-tertiary'>{t('workflow.panel.optional')}</span>} + </div> )} <div className='mt-1'> {item.type === 'select' && ( @@ -115,7 +118,7 @@ const RunOnce: FC<IRunOnceProps> = ({ {item.type === 'string' && ( <Input type="text" - placeholder={`${item.name}${!item.required ? `(${t('appDebug.variableTable.optional')})` : ''}`} + placeholder={item.name} value={inputs[item.key]} onChange={(e: ChangeEvent<HTMLInputElement>) => { handleInputsChange({ ...inputsRef.current, [item.key]: e.target.value }) }} maxLength={item.max_length || DEFAULT_VALUE_MAX_LEN} @@ -124,7 +127,7 @@ const RunOnce: FC<IRunOnceProps> = ({ {item.type === 'paragraph' && ( <Textarea className='h-[104px] sm:text-xs' - placeholder={`${item.name}${!item.required ? `(${t('appDebug.variableTable.optional')})` : ''}`} + placeholder={item.name} value={inputs[item.key]} onChange={(e: ChangeEvent<HTMLTextAreaElement>) => { handleInputsChange({ ...inputsRef.current, [item.key]: e.target.value }) }} /> @@ -132,7 +135,7 @@ const RunOnce: FC<IRunOnceProps> = ({ {item.type === 'number' && ( <Input type="number" - placeholder={`${item.name}${!item.required ? `(${t('appDebug.variableTable.optional')})` : ''}`} + placeholder={item.name} value={inputs[item.key]} onChange={(e: ChangeEvent<HTMLInputElement>) => { handleInputsChange({ ...inputsRef.current, [item.key]: e.target.value }) }} /> diff --git a/web/app/components/splash.tsx b/web/app/components/splash.tsx new file mode 100644 index 0000000000..22e3217f01 --- /dev/null +++ b/web/app/components/splash.tsx @@ -0,0 +1,21 @@ +'use client' +import type { FC, PropsWithChildren } from 'react' +import React from 'react' +import { useIsLogin } from '@/service/use-common' +import Loading from './base/loading' + +const Splash: FC<PropsWithChildren> = () => { + // would auto redirect to signin page if not logged in + const { isLoading, data: loginData } = useIsLogin() + const isLoggedIn = loginData?.logged_in + + if (isLoading || !isLoggedIn) { + return ( + <div className='fixed inset-0 z-[9999999] flex h-full items-center justify-center bg-background-body'> + <Loading /> + </div> + ) + } + return null +} +export default React.memo(Splash) diff --git a/web/app/components/swr-initializer.tsx b/web/app/components/swr-initializer.tsx index 1ab1567659..b7cd767c7a 100644 --- a/web/app/components/swr-initializer.tsx +++ b/web/app/components/swr-initializer.tsx @@ -56,10 +56,10 @@ const SwrInitializer = ({ } const redirectUrl = resolvePostLoginRedirect(searchParams) - if (redirectUrl) + if (redirectUrl) { location.replace(redirectUrl) - else - router.replace(pathname) + return + } setInit(true) } diff --git a/web/app/components/tools/add-tool-modal/D.png b/web/app/components/tools/add-tool-modal/D.png deleted file mode 100644 index 70b829c821..0000000000 Binary files a/web/app/components/tools/add-tool-modal/D.png and /dev/null differ diff --git a/web/app/components/tools/add-tool-modal/category.tsx b/web/app/components/tools/add-tool-modal/category.tsx deleted file mode 100644 index 270b4fc2bf..0000000000 --- a/web/app/components/tools/add-tool-modal/category.tsx +++ /dev/null @@ -1,69 +0,0 @@ -'use client' -import { useRef } from 'react' -import { useTranslation } from 'react-i18next' -import { useContext } from 'use-context-selector' -import { useMount } from 'ahooks' -import cn from '@/utils/classnames' -import { Apps02 } from '@/app/components/base/icons/src/vender/line/others' -import I18n from '@/context/i18n' -import { getLanguage } from '@/i18n-config/language' -import { useStore as useLabelStore } from '@/app/components/tools/labels/store' -import { fetchLabelList } from '@/service/tools' - -type Props = { - value: string - onSelect: (type: string) => void -} - -const Icon = ({ svgString, active }: { svgString: string; active: boolean }) => { - const svgRef = useRef<SVGSVGElement | null>(null) - const SVGParser = (svg: string) => { - if (!svg) - return null - const parser = new DOMParser() - const doc = parser.parseFromString(svg, 'image/svg+xml') - return doc.documentElement - } - useMount(() => { - const svgElement = SVGParser(svgString) - if (svgRef.current && svgElement) - svgRef.current.appendChild(svgElement) - }) - return <svg className={cn('h-4 w-4 text-gray-700', active && '!text-primary-600')} ref={svgRef} /> -} - -const Category = ({ - value, - onSelect, -}: Props) => { - const { t } = useTranslation() - const { locale } = useContext(I18n) - const language = getLanguage(locale) - const labelList = useLabelStore(s => s.labelList) - const setLabelList = useLabelStore(s => s.setLabelList) - - useMount(() => { - fetchLabelList().then((res) => { - setLabelList(res) - }) - }) - - return ( - <div className='mb-3'> - <div className='px-3 py-0.5 text-xs font-medium leading-[18px] text-gray-500'>{t('tools.addToolModal.category').toLocaleUpperCase()}</div> - <div className={cn('mb-0.5 flex cursor-pointer items-center rounded-lg p-1 pl-3 text-sm leading-5 text-gray-700 hover:bg-white', value === '' && '!bg-white font-medium !text-primary-600')} onClick={() => onSelect('')}> - <Apps02 className='mr-2 h-4 w-4 shrink-0' /> - {t('tools.type.all')} - </div> - {labelList.map(label => ( - <div key={label.name} title={label.label[language]} className={cn('mb-0.5 flex cursor-pointer items-center overflow-hidden truncate rounded-lg p-1 pl-3 text-sm leading-5 text-gray-700 hover:bg-white', value === label.name && '!bg-white font-medium !text-primary-600')} onClick={() => onSelect(label.name)}> - <div className='mr-2 h-4 w-4 shrink-0'> - <Icon active={value === label.name} svgString={label.icon} /> - </div> - {label.label[language]} - </div> - ))} - </div> - ) -} -export default Category diff --git a/web/app/components/tools/add-tool-modal/empty.png b/web/app/components/tools/add-tool-modal/empty.png deleted file mode 100644 index da4007e45a..0000000000 Binary files a/web/app/components/tools/add-tool-modal/empty.png and /dev/null differ diff --git a/web/app/components/tools/add-tool-modal/index.tsx b/web/app/components/tools/add-tool-modal/index.tsx deleted file mode 100644 index e12ba3e334..0000000000 --- a/web/app/components/tools/add-tool-modal/index.tsx +++ /dev/null @@ -1,257 +0,0 @@ -'use client' -import type { FC } from 'react' -import React, { useMemo, useState } from 'react' -import { useTranslation } from 'react-i18next' -import { useContext } from 'use-context-selector' -import { produce } from 'immer' -import { - RiAddLine, - RiCloseLine, -} from '@remixicon/react' -import { useMount } from 'ahooks' -import type { Collection, CustomCollectionBackend, Tool } from '../types' -import Type from './type' -import Category from './category' -import Tools from './tools' -import cn from '@/utils/classnames' -import { basePath } from '@/utils/var' -import I18n from '@/context/i18n' -import Drawer from '@/app/components/base/drawer' -import Button from '@/app/components/base/button' -import Loading from '@/app/components/base/loading' -import Input from '@/app/components/base/input' -import EditCustomToolModal from '@/app/components/tools/edit-custom-collection-modal' -import ConfigCredential from '@/app/components/tools/setting/build-in/config-credentials' -import { - createCustomCollection, - fetchAllBuiltInTools, - fetchAllCustomTools, - fetchAllWorkflowTools, - removeBuiltInToolCredential, - updateBuiltInToolCredential, -} from '@/service/tools' -import type { ToolWithProvider } from '@/app/components/workflow/types' -import Toast from '@/app/components/base/toast' -import ConfigContext from '@/context/debug-configuration' -import type { ModelConfig } from '@/models/debug' - -type Props = { - onHide: () => void -} -// Add and Edit -const AddToolModal: FC<Props> = ({ - onHide, -}) => { - const { t } = useTranslation() - const { locale } = useContext(I18n) - const [currentType, setCurrentType] = useState('builtin') - const [currentCategory, setCurrentCategory] = useState('') - const [keywords, setKeywords] = useState<string>('') - const handleKeywordsChange = (value: string) => { - setKeywords(value) - } - const isMatchingKeywords = (text: string, keywords: string) => { - return text.toLowerCase().includes(keywords.toLowerCase()) - } - const [toolList, setToolList] = useState<ToolWithProvider[]>([]) - const [listLoading, setListLoading] = useState(true) - const getAllTools = async () => { - setListLoading(true) - const buildInTools = await fetchAllBuiltInTools() - if (basePath) { - buildInTools.forEach((item) => { - if (typeof item.icon == 'string' && !item.icon.includes(basePath)) - item.icon = `${basePath}${item.icon}` - }) - } - const customTools = await fetchAllCustomTools() - const workflowTools = await fetchAllWorkflowTools() - const mergedToolList = [ - ...buildInTools, - ...customTools, - ...workflowTools.filter((toolWithProvider) => { - return !toolWithProvider.tools.some((tool) => { - return !!tool.parameters.find(item => item.name === '__image') - }) - }), - ] - setToolList(mergedToolList) - setListLoading(false) - } - const filteredList = useMemo(() => { - return toolList.filter((toolWithProvider) => { - if (currentType === 'all') - return true - else - return toolWithProvider.type === currentType - }).filter((toolWithProvider) => { - if (!currentCategory) - return true - else - return toolWithProvider.labels.includes(currentCategory) - }).filter((toolWithProvider) => { - return ( - isMatchingKeywords(toolWithProvider.name, keywords) - || toolWithProvider.tools.some((tool) => { - return Object.values(tool.label).some((label) => { - return isMatchingKeywords(label, keywords) - }) - }) - ) - }) - }, [currentType, currentCategory, toolList, keywords]) - - const { - modelConfig, - setModelConfig, - } = useContext(ConfigContext) - - const [isShowEditCollectionToolModal, setIsShowEditCustomCollectionModal] = useState(false) - const doCreateCustomToolCollection = async (data: CustomCollectionBackend) => { - await createCustomCollection(data) - Toast.notify({ - type: 'success', - message: t('common.api.actionSuccess'), - }) - setIsShowEditCustomCollectionModal(false) - getAllTools() - } - const [showSettingAuth, setShowSettingAuth] = useState(false) - const [collection, setCollection] = useState<Collection>() - const toolSelectHandle = (collection: Collection, tool: Tool) => { - const parameters: Record<string, string> = {} - if (tool.parameters) { - tool.parameters.forEach((item) => { - parameters[item.name] = '' - }) - } - - const nexModelConfig = produce(modelConfig, (draft: ModelConfig) => { - draft.agentConfig.tools.push({ - provider_id: collection.id || collection.name, - provider_type: collection.type, - provider_name: collection.name, - tool_name: tool.name, - tool_label: tool.label[locale] || tool.label[locale.replaceAll('-', '_')], - tool_parameters: parameters, - enabled: true, - }) - }) - setModelConfig(nexModelConfig) - } - const authSelectHandle = (provider: Collection) => { - setCollection(provider) - setShowSettingAuth(true) - } - const updateBuiltinAuth = async (value: Record<string, any>) => { - if (!collection) - return - await updateBuiltInToolCredential(collection.name, value) - Toast.notify({ - type: 'success', - message: t('common.api.actionSuccess'), - }) - await getAllTools() - setShowSettingAuth(false) - } - const removeBuiltinAuth = async () => { - if (!collection) - return - await removeBuiltInToolCredential(collection.name) - Toast.notify({ - type: 'success', - message: t('common.api.actionSuccess'), - }) - await getAllTools() - setShowSettingAuth(false) - } - - useMount(() => { - getAllTools() - }) - - return ( - <> - <Drawer - isOpen - mask - clickOutsideNotOpen - onClose={onHide} - footer={null} - panelClassName={cn('mx-2 mb-3 mt-16 rounded-xl !p-0 sm:mr-2', 'mt-2 !w-[640px]', '!max-w-[640px]')} - > - <div - className='flex w-full rounded-xl border-[0.5px] border-gray-200 bg-white shadow-xl' - style={{ - height: 'calc(100vh - 16px)', - }} - > - <div className='relative w-[200px] shrink-0 overflow-y-auto rounded-l-xl border-r-[0.5px] border-black/2 bg-gray-100 pb-3'> - <div className='sticky left-0 right-0 top-0'> - <div className='text-md sticky left-0 right-0 top-0 px-5 py-3 font-semibold text-gray-900'>{t('tools.addTool')}</div> - <div className='px-3 pb-4 pt-2'> - <Button variant='primary' className='w-[176px]' onClick={() => setIsShowEditCustomCollectionModal(true)}> - <RiAddLine className='mr-1 h-4 w-4' /> - {t('tools.createCustomTool')} - </Button> - </div> - </div> - <div className='px-2 py-1'> - <Type value={currentType} onSelect={setCurrentType} /> - <Category value={currentCategory} onSelect={setCurrentCategory} /> - </div> - </div> - <div className='relative grow overflow-y-auto rounded-r-xl bg-white'> - <div className='sticky left-0 right-0 top-0 z-10 flex items-center gap-1 bg-white p-2'> - <div className='grow'> - <Input - showLeftIcon - showClearIcon - value={keywords} - onChange={e => handleKeywordsChange(e.target.value)} - onClear={() => handleKeywordsChange('')} - /> - </div> - <div className='ml-2 mr-1 h-4 w-[1px] bg-gray-200'></div> - <div className='cursor-pointer p-2' onClick={onHide}> - <RiCloseLine className='h-4 w-4 text-gray-500' /> - </div> - </div> - {listLoading && ( - <div className='flex h-[200px] items-center justify-center bg-white'> - <Loading /> - </div> - )} - {!listLoading && ( - <Tools - showWorkflowEmpty={currentType === 'workflow'} - tools={filteredList} - addedTools={(modelConfig?.agentConfig?.tools as any) || []} - onSelect={toolSelectHandle} - onAuthSetup={authSelectHandle} - /> - )} - </div> - </div> - </Drawer> - {isShowEditCollectionToolModal && ( - <EditCustomToolModal - positionLeft - payload={null} - onHide={() => setIsShowEditCustomCollectionModal(false)} - onAdd={doCreateCustomToolCollection} - /> - )} - {showSettingAuth && collection && ( - <ConfigCredential - collection={collection} - onCancel={() => setShowSettingAuth(false)} - onSaved={updateBuiltinAuth} - onRemove={removeBuiltinAuth} - /> - )} - </> - - ) -} -export default React.memo(AddToolModal) diff --git a/web/app/components/tools/add-tool-modal/tools.tsx b/web/app/components/tools/add-tool-modal/tools.tsx deleted file mode 100644 index 17a3df8357..0000000000 --- a/web/app/components/tools/add-tool-modal/tools.tsx +++ /dev/null @@ -1,150 +0,0 @@ -import { - memo, - useCallback, -} from 'react' -import { basePath } from '@/utils/var' -import { useTranslation } from 'react-i18next' -import { - RiAddLine, -} from '@remixicon/react' -import cn from '@/utils/classnames' -import { ArrowUpRight } from '@/app/components/base/icons/src/vender/line/arrows' -import { Check } from '@/app/components/base/icons/src/vender/line/general' -import { Tag01 } from '@/app/components/base/icons/src/vender/line/financeAndECommerce' -import type { ToolWithProvider } from '@/app/components/workflow/types' -import { BlockEnum } from '@/app/components/workflow/types' -import BlockIcon from '@/app/components/workflow/block-icon' -import Tooltip from '@/app/components/base/tooltip' -import Button from '@/app/components/base/button' -import { useGetLanguage } from '@/context/i18n' -import { useStore as useLabelStore } from '@/app/components/tools/labels/store' -import Empty from '@/app/components/tools/add-tool-modal/empty' -import type { Tool } from '@/app/components/tools/types' -import { CollectionType } from '@/app/components/tools/types' -import type { AgentTool } from '@/types/app' -import { MAX_TOOLS_NUM } from '@/config' - -type ToolsProps = { - showWorkflowEmpty: boolean - tools: ToolWithProvider[] - addedTools: AgentTool[] - onSelect: (provider: ToolWithProvider, tool: Tool) => void - onAuthSetup: (provider: ToolWithProvider) => void -} -const Blocks = ({ - showWorkflowEmpty, - tools, - addedTools, - onSelect, - onAuthSetup, -}: ToolsProps) => { - const { t } = useTranslation() - const language = useGetLanguage() - const labelList = useLabelStore(s => s.labelList) - const addable = addedTools.length < MAX_TOOLS_NUM - - const renderGroup = useCallback((toolWithProvider: ToolWithProvider) => { - const list = toolWithProvider.tools - const needAuth = toolWithProvider.allow_delete && !toolWithProvider.is_team_authorization && toolWithProvider.type === CollectionType.builtIn - - return ( - <div - key={toolWithProvider.id} - className='group mb-1 last-of-type:mb-0' - > - <div className='flex h-[22px] w-full items-center justify-between pl-3 pr-1 text-xs font-medium text-gray-500'> - {toolWithProvider.label[language]} - <a className='hidden cursor-pointer items-center group-hover:flex' href={`${basePath}/tools?category=${toolWithProvider.type}`} target='_blank'>{t('tools.addToolModal.manageInTools')}<ArrowUpRight className='ml-0.5 h-3 w-3' /></a> - </div> - {list.map((tool) => { - const labelContent = (() => { - if (!tool.labels) - return '' - return tool.labels.map((name) => { - const label = labelList.find(item => item.name === name) - return label?.label[language] - }).filter(Boolean).join(', ') - })() - const added = !!addedTools?.find(v => v.provider_id === toolWithProvider.id && v.provider_type === toolWithProvider.type && v.tool_name === tool.name) - return ( - <Tooltip - key={tool.name} - position='bottom' - popupClassName='!p-0 !px-3 !py-2.5 !w-[210px] !leading-[18px] !text-xs !text-gray-700 !border-[0.5px] !border-black/5 !bg-transparent !rounded-xl !shadow-lg translate-x-[108px]' - popupContent={( - <div> - <BlockIcon - size='md' - className='mb-2' - type={BlockEnum.Tool} - toolIcon={toolWithProvider.icon} - /> - <div className='mb-1 text-sm leading-5 text-gray-900'>{tool.label[language]}</div> - <div className='text-xs leading-[18px] text-gray-700'>{tool.description[language]}</div> - {tool.labels?.length > 0 && ( - <div className='mt-1 flex shrink-0 items-center'> - <div className='relative flex w-full items-center gap-1 rounded-md py-1 text-gray-500' title={labelContent}> - <Tag01 className='h-3 w-3 shrink-0 text-gray-500' /> - <div className='grow truncate text-start text-xs font-normal leading-[18px]'>{labelContent}</div> - </div> - </div> - )} - </div> - )} - > - <div className='group/item flex h-8 w-full cursor-pointer items-center rounded-lg pl-3 pr-1 hover:bg-gray-50'> - <BlockIcon - className={cn('mr-2 shrink-0', needAuth && 'opacity-30')} - type={BlockEnum.Tool} - toolIcon={toolWithProvider.icon} - /> - <div className={cn('grow truncate text-sm text-gray-900', needAuth && 'opacity-30')}>{tool.label[language]}</div> - {!needAuth && added && ( - <div className='flex items-center gap-1 rounded-[6px] border border-gray-100 bg-white px-2 py-[3px] text-xs font-medium leading-[18px] text-gray-300'> - <Check className='h-3 w-3' /> - {t('tools.addToolModal.added').toLocaleUpperCase()} - </div> - )} - {!needAuth && !added && addable && ( - <Button - variant='secondary-accent' - size='small' - className={cn('hidden shrink-0 items-center group-hover/item:flex')} - onClick={() => onSelect(toolWithProvider, tool)} - > - <RiAddLine className='h-3 w-3' /> - {t('tools.addToolModal.add').toLocaleUpperCase()} - </Button> - )} - {needAuth && ( - <Button - variant='secondary-accent' - size='small' - className={cn('hidden shrink-0 group-hover/item:flex')} - onClick={() => onAuthSetup(toolWithProvider)} - >{t('tools.auth.setup')}</Button> - )} - </div> - </Tooltip> - ) - })} - </div> - ) - }, [addable, language, t, labelList, addedTools, onAuthSetup, onSelect]) - - return ( - <div className='max-w-[440px] p-1 pb-6'> - {!tools.length && !showWorkflowEmpty && ( - <div className='flex h-[22px] items-center px-3 text-xs font-medium text-gray-500'>{t('workflow.tabs.noResult')}</div> - )} - {!tools.length && showWorkflowEmpty && ( - <div className='pt-[280px]'> - <Empty /> - </div> - )} - {!!tools.length && tools.map(renderGroup)} - </div> - ) -} - -export default memo(Blocks) diff --git a/web/app/components/tools/add-tool-modal/type.tsx b/web/app/components/tools/add-tool-modal/type.tsx deleted file mode 100644 index 26e78a7525..0000000000 --- a/web/app/components/tools/add-tool-modal/type.tsx +++ /dev/null @@ -1,34 +0,0 @@ -'use client' -import { useTranslation } from 'react-i18next' -import cn from '@/utils/classnames' -import { Exchange02, FileCode } from '@/app/components/base/icons/src/vender/line/others' - -type Props = { - value: string - onSelect: (type: string) => void -} - -const Types = ({ - value, - onSelect, -}: Props) => { - const { t } = useTranslation() - - return ( - <div className='mb-3'> - <div className={cn('mb-0.5 flex cursor-pointer items-center rounded-lg p-1 pl-3 text-sm leading-5 hover:bg-white', value === 'builtin' && '!bg-white font-medium')} onClick={() => onSelect('builtin')}> - <div className="mr-2 h-4 w-4 shrink-0 bg-[url('~@/app/components/tools/add-tool-modal/D.png')] bg-cover bg-no-repeat" /> - <span className={cn('text-gray-700', value === 'builtin' && '!text-primary-600')}>{t('tools.type.builtIn')}</span> - </div> - <div className={cn('mb-0.5 flex cursor-pointer items-center rounded-lg p-1 pl-3 text-sm leading-5 text-gray-700 hover:bg-white', value === 'api' && '!bg-white font-medium !text-primary-600')} onClick={() => onSelect('api')}> - <FileCode className='mr-2 h-4 w-4 shrink-0' /> - {t('tools.type.custom')} - </div> - <div className={cn('mb-0.5 flex cursor-pointer items-center rounded-lg p-1 pl-3 text-sm leading-5 text-gray-700 hover:bg-white', value === 'workflow' && '!bg-white font-medium !text-primary-600')} onClick={() => onSelect('workflow')}> - <Exchange02 className='mr-2 h-4 w-4 shrink-0' /> - {t('tools.type.workflow')} - </div> - </div> - ) -} -export default Types diff --git a/web/app/components/tools/edit-custom-collection-modal/index.tsx b/web/app/components/tools/edit-custom-collection-modal/index.tsx index 95a204c1ec..48801b018f 100644 --- a/web/app/components/tools/edit-custom-collection-modal/index.tsx +++ b/web/app/components/tools/edit-custom-collection-modal/index.tsx @@ -24,6 +24,7 @@ import Toast from '@/app/components/base/toast' type Props = { positionLeft?: boolean + dialogClassName?: string payload: any onHide: () => void onAdd?: (payload: CustomCollectionBackend) => void @@ -33,6 +34,7 @@ type Props = { // Add and Edit const EditCustomCollectionModal: FC<Props> = ({ positionLeft, + dialogClassName = '', payload, onHide, onAdd, @@ -186,6 +188,7 @@ const EditCustomCollectionModal: FC<Props> = ({ positionCenter={isAdd && !positionLeft} onHide={onHide} title={t(`tools.createTool.${isAdd ? 'title' : 'editTitle'}`)!} + dialogClassName={dialogClassName} panelClassName='mt-2 !w-[640px]' maxWidthClassName='!max-w-[640px]' height='calc(100vh - 16px)' diff --git a/web/app/components/tools/edit-custom-collection-modal/modal.tsx b/web/app/components/tools/edit-custom-collection-modal/modal.tsx deleted file mode 100644 index 3e278f7b53..0000000000 --- a/web/app/components/tools/edit-custom-collection-modal/modal.tsx +++ /dev/null @@ -1,361 +0,0 @@ -'use client' -import type { FC } from 'react' -import React, { useEffect, useState } from 'react' -import { useTranslation } from 'react-i18next' -import { useDebounce, useGetState } from 'ahooks' -import { produce } from 'immer' -import { LinkExternal02, Settings01 } from '../../base/icons/src/vender/line/general' -import type { Credential, CustomCollectionBackend, CustomParamSchema, Emoji } from '../types' -import { AuthHeaderPrefix, AuthType } from '../types' -import GetSchema from './get-schema' -import ConfigCredentials from './config-credentials' -import TestApi from './test-api' -import cn from '@/utils/classnames' -import Input from '@/app/components/base/input' -import Textarea from '@/app/components/base/textarea' -import EmojiPicker from '@/app/components/base/emoji-picker' -import AppIcon from '@/app/components/base/app-icon' -import { parseParamsSchema } from '@/service/tools' -import LabelSelector from '@/app/components/tools/labels/selector' -import Toast from '@/app/components/base/toast' -import Modal from '../../base/modal' -import Button from '@/app/components/base/button' - -type Props = { - positionLeft?: boolean - payload: any - onHide: () => void - onAdd?: (payload: CustomCollectionBackend) => void - onRemove?: () => void - onEdit?: (payload: CustomCollectionBackend) => void -} -// Add and Edit -const EditCustomCollectionModal: FC<Props> = ({ - payload, - onHide, - onAdd, - onEdit, - onRemove, -}) => { - const { t } = useTranslation() - const isAdd = !payload - const isEdit = !!payload - - const [editFirst, setEditFirst] = useState(!isAdd) - const [paramsSchemas, setParamsSchemas] = useState<CustomParamSchema[]>(payload?.tools || []) - const [customCollection, setCustomCollection, getCustomCollection] = useGetState<CustomCollectionBackend>(isAdd - ? { - provider: '', - credentials: { - auth_type: AuthType.none, - api_key_header: 'Authorization', - api_key_header_prefix: AuthHeaderPrefix.basic, - }, - icon: { - content: '🕵️', - background: '#FEF7C3', - }, - schema_type: '', - schema: '', - } - : payload) - - const originalProvider = isEdit ? payload.provider : '' - - const [showEmojiPicker, setShowEmojiPicker] = useState(false) - const emoji = customCollection.icon - const setEmoji = (emoji: Emoji) => { - const newCollection = produce(customCollection, (draft) => { - draft.icon = emoji - }) - setCustomCollection(newCollection) - } - const schema = customCollection.schema - const debouncedSchema = useDebounce(schema, { wait: 500 }) - const setSchema = (schema: any) => { - const newCollection = produce(customCollection, (draft) => { - draft.schema = schema - }) - setCustomCollection(newCollection) - } - - useEffect(() => { - if (!debouncedSchema) - return - if (isEdit && editFirst) { - setEditFirst(false) - return - } - (async () => { - try { - const { parameters_schema, schema_type } = await parseParamsSchema(debouncedSchema) - const customCollection = getCustomCollection() - const newCollection = produce(customCollection, (draft) => { - draft.schema_type = schema_type - }) - setCustomCollection(newCollection) - setParamsSchemas(parameters_schema) - } - catch { - const customCollection = getCustomCollection() - const newCollection = produce(customCollection, (draft) => { - draft.schema_type = '' - }) - setCustomCollection(newCollection) - setParamsSchemas([]) - } - })() - }, [debouncedSchema]) - - const [credentialsModalShow, setCredentialsModalShow] = useState(false) - const credential = customCollection.credentials - const setCredential = (credential: Credential) => { - const newCollection = produce(customCollection, (draft) => { - draft.credentials = credential - }) - setCustomCollection(newCollection) - } - - const [currTool, setCurrTool] = useState<CustomParamSchema | null>(null) - const [isShowTestApi, setIsShowTestApi] = useState(false) - - const [labels, setLabels] = useState<string[]>(payload?.labels || []) - const handleLabelSelect = (value: string[]) => { - setLabels(value) - } - - const handleSave = () => { - // const postData = clone(customCollection) - const postData = produce(customCollection, (draft) => { - delete draft.tools - - if (draft.credentials.auth_type === AuthType.none) { - delete draft.credentials.api_key_header - delete draft.credentials.api_key_header_prefix - delete draft.credentials.api_key_value - } - - draft.labels = labels - }) - - let errorMessage = '' - if (!postData.provider) - errorMessage = t('common.errorMsg.fieldRequired', { field: t('tools.createTool.name') }) - - if (!postData.schema) - errorMessage = t('common.errorMsg.fieldRequired', { field: t('tools.createTool.schema') }) - - if (errorMessage) { - Toast.notify({ - type: 'error', - message: errorMessage, - }) - return - } - - if (isAdd) { - onAdd?.(postData) - return - } - - onEdit?.({ - ...postData, - original_provider: originalProvider, - }) - } - - const getPath = (url: string) => { - if (!url) - return '' - - try { - const path = decodeURI(new URL(url).pathname) - return path || '' - } - catch { - return url - } - } - - return ( - <> - <Modal - isShow - onClose={onHide} - closable - className='!h-[calc(100vh-16px)] !max-w-[630px] !p-0' - wrapperClassName='z-[1000]' - > - <div className='flex h-full flex-col'> - <div className='ml-6 mt-6 text-base font-semibold text-text-primary'> - {t('tools.createTool.title')} - </div> - <div className='h-0 grow space-y-4 overflow-y-auto px-6 py-3'> - <div> - <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.name')} <span className='ml-1 text-red-500'>*</span></div> - <div className='flex items-center justify-between gap-3'> - <AppIcon size='large' onClick={() => { setShowEmojiPicker(true) }} className='cursor-pointer' icon={emoji.content} background={emoji.background} /> - <Input - className='h-10 grow' placeholder={t('tools.createTool.toolNamePlaceHolder')!} - value={customCollection.provider} - onChange={(e) => { - const newCollection = produce(customCollection, (draft) => { - draft.provider = e.target.value - }) - setCustomCollection(newCollection) - }} - /> - </div> - </div> - - {/* Schema */} - <div className='select-none'> - <div className='flex items-center justify-between'> - <div className='flex items-center'> - <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.schema')}<span className='ml-1 text-red-500'>*</span></div> - <div className='mx-2 h-3 w-px bg-divider-regular'></div> - <a - href="https://swagger.io/specification/" - target='_blank' rel='noopener noreferrer' - className='flex h-[18px] items-center space-x-1 text-text-accent' - > - <div className='text-xs font-normal'>{t('tools.createTool.viewSchemaSpec')}</div> - <LinkExternal02 className='h-3 w-3' /> - </a> - </div> - <GetSchema onChange={setSchema} /> - - </div> - <Textarea - className='h-[240px] resize-none' - value={schema} - onChange={e => setSchema(e.target.value)} - placeholder={t('tools.createTool.schemaPlaceHolder')!} - /> - </div> - - {/* Available Tools */} - <div> - <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.availableTools.title')}</div> - <div className='w-full overflow-x-auto rounded-lg border border-divider-regular'> - <table className='system-xs-regular w-full text-text-secondary'> - <thead className='uppercase text-text-tertiary'> - <tr className={cn(paramsSchemas.length > 0 && 'border-b', 'border-divider-regular')}> - <th className="p-2 pl-3 font-medium">{t('tools.createTool.availableTools.name')}</th> - <th className="w-[236px] p-2 pl-3 font-medium">{t('tools.createTool.availableTools.description')}</th> - <th className="p-2 pl-3 font-medium">{t('tools.createTool.availableTools.method')}</th> - <th className="p-2 pl-3 font-medium">{t('tools.createTool.availableTools.path')}</th> - <th className="w-[54px] p-2 pl-3 font-medium">{t('tools.createTool.availableTools.action')}</th> - </tr> - </thead> - <tbody> - {paramsSchemas.map((item, index) => ( - <tr key={index} className='border-b border-divider-regular last:border-0'> - <td className="p-2 pl-3">{item.operation_id}</td> - <td className="w-[236px] p-2 pl-3">{item.summary}</td> - <td className="p-2 pl-3">{item.method}</td> - <td className="p-2 pl-3">{getPath(item.server_url)}</td> - <td className="w-[62px] p-2 pl-3"> - <Button - size='small' - onClick={() => { - setCurrTool(item) - setIsShowTestApi(true) - }} - > - {t('tools.createTool.availableTools.test')} - </Button> - </td> - </tr> - ))} - </tbody> - </table> - </div> - </div> - - {/* Authorization method */} - <div> - <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.authMethod.title')}</div> - <div className='flex h-9 cursor-pointer items-center justify-between rounded-lg bg-components-input-bg-normal px-2.5' onClick={() => setCredentialsModalShow(true)}> - <div className='system-xs-regular text-text-primary'>{t(`tools.createTool.authMethod.types.${credential.auth_type}`)}</div> - <Settings01 className='h-4 w-4 text-text-secondary' /> - </div> - </div> - - {/* Labels */} - <div> - <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.toolInput.label')}</div> - <LabelSelector value={labels} onChange={handleLabelSelect} /> - </div> - - {/* Privacy Policy */} - <div> - <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.privacyPolicy')}</div> - <Input - value={customCollection.privacy_policy} - onChange={(e) => { - const newCollection = produce(customCollection, (draft) => { - draft.privacy_policy = e.target.value - }) - setCustomCollection(newCollection) - }} - className='h-10 grow' placeholder={t('tools.createTool.privacyPolicyPlaceholder') || ''} /> - </div> - - <div> - <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.customDisclaimer')}</div> - <Input - value={customCollection.custom_disclaimer} - onChange={(e) => { - const newCollection = produce(customCollection, (draft) => { - draft.custom_disclaimer = e.target.value - }) - setCustomCollection(newCollection) - }} - className='h-10 grow' placeholder={t('tools.createTool.customDisclaimerPlaceholder') || ''} /> - </div> - - </div> - <div className={cn(isEdit ? 'justify-between' : 'justify-end', 'mt-2 flex shrink-0 rounded-b-[10px] border-t border-divider-regular bg-background-section-burn px-6 py-4')} > - { - isEdit && ( - <Button variant='warning' onClick={onRemove}>{t('common.operation.delete')}</Button> - ) - } - <div className='flex space-x-2 '> - <Button onClick={onHide}>{t('common.operation.cancel')}</Button> - <Button variant='primary' onClick={handleSave}>{t('common.operation.save')}</Button> - </div> - </div> - {showEmojiPicker && <EmojiPicker - onSelect={(icon, icon_background) => { - setEmoji({ content: icon, background: icon_background }) - setShowEmojiPicker(false) - }} - onClose={() => { - setShowEmojiPicker(false) - }} - />} - {credentialsModalShow && ( - <ConfigCredentials - positionCenter={isAdd} - credential={credential} - onChange={setCredential} - onHide={() => setCredentialsModalShow(false)} - />) - } - {isShowTestApi && ( - <TestApi - positionCenter={isAdd} - tool={currTool as CustomParamSchema} - customCollection={customCollection} - onHide={() => setIsShowTestApi(false)} - /> - )} - </div> - </Modal> - </> - - ) -} -export default React.memo(EditCustomCollectionModal) diff --git a/web/app/components/tools/marketplace/hooks.ts b/web/app/components/tools/marketplace/hooks.ts index 0790d52721..e3fad24710 100644 --- a/web/app/components/tools/marketplace/hooks.ts +++ b/web/app/components/tools/marketplace/hooks.ts @@ -9,7 +9,7 @@ import { useMarketplaceCollectionsAndPlugins, useMarketplacePlugins, } from '@/app/components/plugins/marketplace/hooks' -import { PluginType } from '@/app/components/plugins/types' +import { PluginCategoryEnum } from '@/app/components/plugins/types' import { getMarketplaceListCondition } from '@/app/components/plugins/marketplace/utils' import { useAllToolProviders } from '@/service/use-tools' @@ -49,7 +49,7 @@ export const useMarketplace = (searchPluginText: string, filterPluginTags: strin if (searchPluginText) { queryPluginsWithDebounced({ - category: PluginType.tool, + category: PluginCategoryEnum.tool, query: searchPluginText, tags: filterPluginTags, exclude, @@ -59,7 +59,7 @@ export const useMarketplace = (searchPluginText: string, filterPluginTags: strin return } queryPlugins({ - category: PluginType.tool, + category: PluginCategoryEnum.tool, query: searchPluginText, tags: filterPluginTags, exclude, @@ -70,8 +70,8 @@ export const useMarketplace = (searchPluginText: string, filterPluginTags: strin else { if (isSuccess) { queryMarketplaceCollectionsAndPlugins({ - category: PluginType.tool, - condition: getMarketplaceListCondition(PluginType.tool), + category: PluginCategoryEnum.tool, + condition: getMarketplaceListCondition(PluginCategoryEnum.tool), exclude, type: 'plugin', }) @@ -95,7 +95,7 @@ export const useMarketplace = (searchPluginText: string, filterPluginTags: strin pageRef.current++ queryPlugins({ - category: PluginType.tool, + category: PluginCategoryEnum.tool, query: searchPluginText, tags: filterPluginTags, exclude, diff --git a/web/app/components/tools/mcp/headers-input.tsx b/web/app/components/tools/mcp/headers-input.tsx index 81d62993c9..ede5b6cffe 100644 --- a/web/app/components/tools/mcp/headers-input.tsx +++ b/web/app/components/tools/mcp/headers-input.tsx @@ -1,6 +1,7 @@ 'use client' -import React, { useCallback } from 'react' +import React from 'react' import { useTranslation } from 'react-i18next' +import { v4 as uuid } from 'uuid' import { RiAddLine, RiDeleteBinLine } from '@remixicon/react' import Input from '@/app/components/base/input' import Button from '@/app/components/base/button' @@ -8,57 +9,46 @@ import ActionButton from '@/app/components/base/action-button' import cn from '@/utils/classnames' export type HeaderItem = { + id: string key: string value: string } type Props = { - headers: Record<string, string> - onChange: (headers: Record<string, string>) => void + headersItems: HeaderItem[] + onChange: (headerItems: HeaderItem[]) => void readonly?: boolean isMasked?: boolean } const HeadersInput = ({ - headers, + headersItems, onChange, readonly = false, isMasked = false, }: Props) => { const { t } = useTranslation() - const headerItems = Object.entries(headers).map(([key, value]) => ({ key, value })) - - const handleItemChange = useCallback((index: number, field: 'key' | 'value', value: string) => { - const newItems = [...headerItems] + const handleItemChange = (index: number, field: 'key' | 'value', value: string) => { + const newItems = [...headersItems] newItems[index] = { ...newItems[index], [field]: value } - const newHeaders = newItems.reduce((acc, item) => { - if (item.key.trim()) - acc[item.key.trim()] = item.value - return acc - }, {} as Record<string, string>) + onChange(newItems) + } - onChange(newHeaders) - }, [headerItems, onChange]) + const handleRemoveItem = (index: number) => { + const newItems = headersItems.filter((_, i) => i !== index) - const handleRemoveItem = useCallback((index: number) => { - const newItems = headerItems.filter((_, i) => i !== index) - const newHeaders = newItems.reduce((acc, item) => { - if (item.key.trim()) - acc[item.key.trim()] = item.value + onChange(newItems) + } - return acc - }, {} as Record<string, string>) - onChange(newHeaders) - }, [headerItems, onChange]) + const handleAddItem = () => { + const newItems = [...headersItems, { id: uuid(), key: '', value: '' }] - const handleAddItem = useCallback(() => { - const newHeaders = { ...headers, '': '' } - onChange(newHeaders) - }, [headers, onChange]) + onChange(newItems) + } - if (headerItems.length === 0) { + if (headersItems.length === 0) { return ( <div className='space-y-2'> <div className='body-xs-regular text-text-tertiary'> @@ -91,10 +81,10 @@ const HeadersInput = ({ <div className='h-full w-1/2 border-r border-divider-regular pl-3'>{t('tools.mcp.modal.headerKey')}</div> <div className='h-full w-1/2 pl-3 pr-1'>{t('tools.mcp.modal.headerValue')}</div> </div> - {headerItems.map((item, index) => ( - <div key={index} className={cn( + {headersItems.map((item, index) => ( + <div key={item.id} className={cn( 'flex items-center border-divider-regular', - index < headerItems.length - 1 && 'border-b', + index < headersItems.length - 1 && 'border-b', )}> <div className='w-1/2 border-r border-divider-regular'> <Input @@ -113,7 +103,7 @@ const HeadersInput = ({ className='flex-1 rounded-none border-0' readOnly={readonly} /> - {!readonly && headerItems.length > 1 && ( + {!readonly && !!headersItems.length && ( <ActionButton onClick={() => handleRemoveItem(index)} className='mr-2' diff --git a/web/app/components/tools/mcp/mcp-service-card.tsx b/web/app/components/tools/mcp/mcp-service-card.tsx index 6c86932b32..1f40b1e4b3 100644 --- a/web/app/components/tools/mcp/mcp-service-card.tsx +++ b/web/app/components/tools/mcp/mcp-service-card.tsx @@ -13,7 +13,7 @@ import CopyFeedback from '@/app/components/base/copy-feedback' import Confirm from '@/app/components/base/confirm' import type { AppDetailResponse } from '@/models/app' import { useAppContext } from '@/context/app-context' -import type { AppSSO } from '@/types/app' +import { AppModeEnum, type AppSSO } from '@/types/app' import Indicator from '@/app/components/header/indicator' import MCPServerModal from '@/app/components/tools/mcp/mcp-server-modal' import { useAppWorkflow } from '@/service/use-workflow' @@ -26,6 +26,7 @@ import { import { BlockEnum } from '@/app/components/workflow/types' import cn from '@/utils/classnames' import { fetchAppDetail } from '@/service/apps' +import { useDocLink } from '@/context/i18n' export type IAppCardProps = { appInfo: AppDetailResponse & Partial<AppSSO> @@ -35,6 +36,7 @@ function MCPServiceCard({ appInfo, }: IAppCardProps) { const { t } = useTranslation() + const docLink = useDocLink() const appId = appInfo.id const { mutateAsync: updateMCPServer } = useUpdateMCPServer() const { mutateAsync: refreshMCPServerCode, isPending: genLoading } = useRefreshMCPServerCode() @@ -43,7 +45,7 @@ function MCPServiceCard({ const [showConfirmDelete, setShowConfirmDelete] = useState(false) const [showMCPServerModal, setShowMCPServerModal] = useState(false) - const isAdvancedApp = appInfo?.mode === 'advanced-chat' || appInfo?.mode === 'workflow' + const isAdvancedApp = appInfo?.mode === AppModeEnum.ADVANCED_CHAT || appInfo?.mode === AppModeEnum.WORKFLOW const isBasicApp = !isAdvancedApp const { data: currentWorkflow } = useAppWorkflow(isAdvancedApp ? appId : '') const [basicAppConfig, setBasicAppConfig] = useState<any>({}) @@ -69,11 +71,16 @@ function MCPServiceCard({ const { data: detail } = useMCPServerDetail(appId) const { id, status, server_code } = detail ?? {} + const isWorkflowApp = appInfo.mode === AppModeEnum.WORKFLOW const appUnpublished = isAdvancedApp ? !currentWorkflow?.graph : !basicAppConfig.updated_at const serverPublished = !!id const serverActivated = status === 'active' const serverURL = serverPublished ? `${appInfo.api_base_url.replace('/v1', '')}/mcp/server/${server_code}/mcp` : '***********' - const toggleDisabled = !isCurrentWorkspaceEditor || appUnpublished + const hasStartNode = currentWorkflow?.graph?.nodes?.some(node => node.data.type === BlockEnum.Start) + const missingStartNode = isWorkflowApp && !hasStartNode + const hasInsufficientPermissions = !isCurrentWorkspaceEditor + const toggleDisabled = hasInsufficientPermissions || appUnpublished || missingStartNode + const isMinimalState = appUnpublished || missingStartNode const [activated, setActivated] = useState(serverActivated) @@ -136,12 +143,12 @@ function MCPServiceCard({ return ( <> - <div className={cn('w-full max-w-full rounded-xl border-l-[0.5px] border-t border-effects-highlight')}> + <div className={cn('w-full max-w-full rounded-xl border-l-[0.5px] border-t border-effects-highlight', isMinimalState && 'h-12')}> <div className='rounded-xl bg-background-default'> - <div className='flex w-full flex-col items-start justify-center gap-3 self-stretch border-b-[0.5px] border-divider-subtle p-3'> + <div className={cn('flex w-full flex-col items-start justify-center gap-3 self-stretch p-3', isMinimalState ? 'border-0' : 'border-b-[0.5px] border-divider-subtle')}> <div className='flex w-full items-center gap-3 self-stretch'> <div className='flex grow items-center'> - <div className='mr-3 shrink-0 rounded-lg border-[0.5px] border-divider-subtle bg-util-colors-indigo-indigo-500 p-1 shadow-md'> + <div className='mr-2 shrink-0 rounded-lg border-[0.5px] border-divider-subtle bg-util-colors-blue-brand-blue-brand-500 p-1 shadow-md'> <Mcp className='h-4 w-4 text-text-primary-on-surface' /> </div> <div className="group w-full"> @@ -159,61 +166,86 @@ function MCPServiceCard({ </div> </div> <Tooltip - popupContent={appUnpublished ? t('tools.mcp.server.publishTip') : ''} + popupContent={ + toggleDisabled ? ( + appUnpublished ? ( + t('tools.mcp.server.publishTip') + ) : missingStartNode ? ( + <> + <div className="mb-1 text-xs font-normal text-text-secondary"> + {t('appOverview.overview.appInfo.enableTooltip.description')} + </div> + <div + className="cursor-pointer text-xs font-normal text-text-accent hover:underline" + onClick={() => window.open(docLink('/guides/workflow/node/user-input'), '_blank')} + > + {t('appOverview.overview.appInfo.enableTooltip.learnMore')} + </div> + </> + ) : '' + ) : '' + } + position="right" + popupClassName="w-58 max-w-60 rounded-xl bg-components-panel-bg px-3.5 py-3 shadow-lg" + offset={24} > <div> <Switch defaultValue={activated} onChange={onChangeStatus} disabled={toggleDisabled} /> </div> </Tooltip> </div> - <div className='flex flex-col items-start justify-center self-stretch'> - <div className="system-xs-medium pb-1 text-text-tertiary"> - {t('tools.mcp.server.url')} - </div> - <div className="inline-flex h-9 w-full items-center gap-0.5 rounded-lg bg-components-input-bg-normal p-1 pl-2"> - <div className="flex h-4 min-w-0 flex-1 items-start justify-start gap-2 px-1"> - <div className="overflow-hidden text-ellipsis whitespace-nowrap text-xs font-medium text-text-secondary"> - {serverURL} - </div> + {!isMinimalState && ( + <div className='flex flex-col items-start justify-center self-stretch'> + <div className="system-xs-medium pb-1 text-text-tertiary"> + {t('tools.mcp.server.url')} </div> - {serverPublished && ( - <> - <CopyFeedback - content={serverURL} - className={'!size-6'} - /> - <Divider type="vertical" className="!mx-0.5 !h-3.5 shrink-0" /> - {isCurrentWorkspaceManager && ( - <Tooltip - popupContent={t('appOverview.overview.appInfo.regenerate') || ''} - > - <div - className="cursor-pointer rounded-md p-1 hover:bg-state-base-hover" - onClick={() => setShowConfirmDelete(true)} + <div className="inline-flex h-9 w-full items-center gap-0.5 rounded-lg bg-components-input-bg-normal p-1 pl-2"> + <div className="flex h-4 min-w-0 flex-1 items-start justify-start gap-2 px-1"> + <div className="overflow-hidden text-ellipsis whitespace-nowrap text-xs font-medium text-text-secondary"> + {serverURL} + </div> + </div> + {serverPublished && ( + <> + <CopyFeedback + content={serverURL} + className={'!size-6'} + /> + <Divider type="vertical" className="!mx-0.5 !h-3.5 shrink-0" /> + {isCurrentWorkspaceManager && ( + <Tooltip + popupContent={t('appOverview.overview.appInfo.regenerate') || ''} > - <RiLoopLeftLine className={cn('h-4 w-4 text-text-tertiary hover:text-text-secondary', genLoading && 'animate-spin')}/> - </div> - </Tooltip> - )} - </> - )} + <div + className="cursor-pointer rounded-md p-1 hover:bg-state-base-hover" + onClick={() => setShowConfirmDelete(true)} + > + <RiLoopLeftLine className={cn('h-4 w-4 text-text-tertiary hover:text-text-secondary', genLoading && 'animate-spin')}/> + </div> + </Tooltip> + )} + </> + )} + </div> </div> - </div> + )} </div> - <div className='flex items-center gap-1 self-stretch p-3'> - <Button - disabled={toggleDisabled} - size='small' - variant='ghost' - onClick={() => setShowMCPServerModal(true)} - > + {!isMinimalState && ( + <div className='flex items-center gap-1 self-stretch p-3'> + <Button + disabled={toggleDisabled} + size='small' + variant='ghost' + onClick={() => setShowMCPServerModal(true)} + > - <div className="flex items-center justify-center gap-[1px]"> - <RiEditLine className="h-3.5 w-3.5" /> - <div className="system-xs-medium px-[3px] text-text-tertiary">{serverPublished ? t('tools.mcp.server.edit') : t('tools.mcp.server.addDescription')}</div> - </div> - </Button> - </div> + <div className="flex items-center justify-center gap-[1px]"> + <RiEditLine className="h-3.5 w-3.5" /> + <div className="system-xs-medium px-[3px] text-text-tertiary">{serverPublished ? t('tools.mcp.server.edit') : t('tools.mcp.server.addDescription')}</div> + </div> + </Button> + </div> + )} </div> </div> {showMCPServerModal && ( diff --git a/web/app/components/tools/mcp/modal.tsx b/web/app/components/tools/mcp/modal.tsx index 1d888c57e8..ad528e9fb9 100644 --- a/web/app/components/tools/mcp/modal.tsx +++ b/web/app/components/tools/mcp/modal.tsx @@ -1,6 +1,7 @@ 'use client' -import React, { useRef, useState } from 'react' +import React, { useCallback, useRef, useState } from 'react' import { useTranslation } from 'react-i18next' +import { v4 as uuid } from 'uuid' import { getDomain } from 'tldts' import { RiCloseLine, RiEditLine } from '@remixicon/react' import { Mcp } from '@/app/components/base/icons/src/vender/other' @@ -11,6 +12,7 @@ import Modal from '@/app/components/base/modal' import Button from '@/app/components/base/button' import Input from '@/app/components/base/input' import HeadersInput from './headers-input' +import type { HeaderItem } from './headers-input' import type { AppIconType } from '@/types/app' import type { ToolWithProvider } from '@/app/components/workflow/types' import { noop } from 'lodash-es' @@ -19,6 +21,9 @@ import { uploadRemoteFileInfo } from '@/service/common' import cn from '@/utils/classnames' import { useHover } from 'ahooks' import { shouldUseMcpIconForAppIcon } from '@/utils/mcp' +import TabSlider from '@/app/components/base/tab-slider' +import { MCPAuthMethod } from '@/app/components/tools/types' +import Switch from '@/app/components/base/switch' export type DuplicateAppModalProps = { data?: ToolWithProvider @@ -30,9 +35,17 @@ export type DuplicateAppModalProps = { icon: string icon_background?: string | null server_identifier: string - timeout: number - sse_read_timeout: number headers?: Record<string, string> + is_dynamic_registration?: boolean + authentication?: { + client_id?: string + client_secret?: string + grant_type?: string + } + configuration: { + timeout: number + sse_read_timeout: number + } }) => void onHide: () => void } @@ -63,6 +76,20 @@ const MCPModal = ({ const { t } = useTranslation() const isCreate = !data + const authMethods = [ + { + text: t('tools.mcp.modal.authentication'), + value: MCPAuthMethod.authentication, + }, + { + text: t('tools.mcp.modal.headers'), + value: MCPAuthMethod.headers, + }, + { + text: t('tools.mcp.modal.configurations'), + value: MCPAuthMethod.configurations, + }, + ] const originalServerUrl = data?.server_url const originalServerID = data?.server_identifier const [url, setUrl] = React.useState(data?.server_url || '') @@ -72,12 +99,16 @@ const MCPModal = ({ const [serverIdentifier, setServerIdentifier] = React.useState(data?.server_identifier || '') const [timeout, setMcpTimeout] = React.useState(data?.timeout || 30) const [sseReadTimeout, setSseReadTimeout] = React.useState(data?.sse_read_timeout || 300) - const [headers, setHeaders] = React.useState<Record<string, string>>( - data?.masked_headers || {}, + const [headers, setHeaders] = React.useState<HeaderItem[]>( + Object.entries(data?.masked_headers || {}).map(([key, value]) => ({ id: uuid(), key, value })), ) const [isFetchingIcon, setIsFetchingIcon] = useState(false) const appIconRef = useRef<HTMLDivElement>(null) const isHovering = useHover(appIconRef) + const [authMethod, setAuthMethod] = useState(MCPAuthMethod.authentication) + const [isDynamicRegistration, setIsDynamicRegistration] = useState(isCreate ? true : data?.is_dynamic_registration) + const [clientID, setClientID] = useState(data?.authentication?.client_id || '') + const [credentials, setCredentials] = useState(data?.authentication?.client_secret || '') // Update states when data changes (for edit mode) React.useEffect(() => { @@ -87,8 +118,11 @@ const MCPModal = ({ setServerIdentifier(data.server_identifier || '') setMcpTimeout(data.timeout || 30) setSseReadTimeout(data.sse_read_timeout || 300) - setHeaders(data.masked_headers || {}) + setHeaders(Object.entries(data.masked_headers || {}).map(([key, value]) => ({ id: uuid(), key, value }))) setAppIcon(getIcon(data)) + setIsDynamicRegistration(data.is_dynamic_registration) + setClientID(data.authentication?.client_id || '') + setCredentials(data.authentication?.client_secret || '') } else { // Reset for create mode @@ -97,15 +131,18 @@ const MCPModal = ({ setServerIdentifier('') setMcpTimeout(30) setSseReadTimeout(300) - setHeaders({}) + setHeaders([]) setAppIcon(DEFAULT_ICON as AppIconSelection) + setIsDynamicRegistration(true) + setClientID('') + setCredentials('') } }, [data]) const isValidUrl = (string: string) => { try { - const urlPattern = /^(https?:\/\/)((([a-z\d]([a-z\d-]*[a-z\d])*)\.)+[a-z]{2,}|((\d{1,3}\.){3}\d{1,3})|localhost)(:\d+)?(\/[-a-z\d%_.~+]*)*(\?[;&a-z\d%_.~+=-]*)?/i - return urlPattern.test(string) + const url = new URL(string) + return url.protocol === 'http:' || url.protocol === 'https:' } catch { return false @@ -150,6 +187,11 @@ const MCPModal = ({ Toast.notify({ type: 'error', message: 'invalid server identifier' }) return } + const formattedHeaders = headers.reduce((acc, item) => { + if (item.key.trim()) + acc[item.key.trim()] = item.value + return acc + }, {} as Record<string, string>) await onConfirm({ server_url: originalServerUrl === url ? '[__HIDDEN__]' : url.trim(), name, @@ -157,14 +199,25 @@ const MCPModal = ({ icon: appIcon.type === 'emoji' ? appIcon.icon : appIcon.fileId, icon_background: appIcon.type === 'emoji' ? appIcon.background : undefined, server_identifier: serverIdentifier.trim(), - timeout: timeout || 30, - sse_read_timeout: sseReadTimeout || 300, - headers: Object.keys(headers).length > 0 ? headers : undefined, + headers: Object.keys(formattedHeaders).length > 0 ? formattedHeaders : undefined, + is_dynamic_registration: isDynamicRegistration, + authentication: { + client_id: clientID, + client_secret: credentials, + }, + configuration: { + timeout: timeout || 30, + sse_read_timeout: sseReadTimeout || 300, + }, }) if(isCreate) onHide() } + const handleAuthMethodChange = useCallback((value: string) => { + setAuthMethod(value as MCPAuthMethod) + }, []) + return ( <> <Modal @@ -239,42 +292,101 @@ const MCPModal = ({ </div> )} </div> - <div> - <div className='mb-1 flex h-6 items-center'> - <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.timeout')}</span> - </div> - <Input - type='number' - value={timeout} - onChange={e => setMcpTimeout(Number(e.target.value))} - onBlur={e => handleBlur(e.target.value.trim())} - placeholder={t('tools.mcp.modal.timeoutPlaceholder')} - /> - </div> - <div> - <div className='mb-1 flex h-6 items-center'> - <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.sseReadTimeout')}</span> - </div> - <Input - type='number' - value={sseReadTimeout} - onChange={e => setSseReadTimeout(Number(e.target.value))} - onBlur={e => handleBlur(e.target.value.trim())} - placeholder={t('tools.mcp.modal.timeoutPlaceholder')} - /> - </div> - <div> - <div className='mb-1 flex h-6 items-center'> - <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.headers')}</span> - </div> - <div className='body-xs-regular mb-2 text-text-tertiary'>{t('tools.mcp.modal.headersTip')}</div> - <HeadersInput - headers={headers} - onChange={setHeaders} - readonly={false} - isMasked={!isCreate && Object.keys(headers).length > 0} - /> - </div> + <TabSlider + className='w-full' + itemClassName={(isActive) => { + return `flex-1 ${isActive && 'text-text-accent-light-mode-only'}` + }} + value={authMethod} + onChange={handleAuthMethodChange} + options={authMethods} + /> + { + authMethod === MCPAuthMethod.authentication && ( + <> + <div> + <div className='mb-1 flex h-6 items-center'> + <Switch + className='mr-2' + defaultValue={isDynamicRegistration} + onChange={setIsDynamicRegistration} + /> + <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.useDynamicClientRegistration')}</span> + </div> + </div> + <div> + <div className={cn('mb-1 flex h-6 items-center', isDynamicRegistration && 'opacity-50')}> + <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.clientID')}</span> + </div> + <Input + value={clientID} + onChange={e => setClientID(e.target.value)} + onBlur={e => handleBlur(e.target.value.trim())} + placeholder={t('tools.mcp.modal.clientID')} + disabled={isDynamicRegistration} + /> + </div> + <div> + <div className={cn('mb-1 flex h-6 items-center', isDynamicRegistration && 'opacity-50')}> + <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.clientSecret')}</span> + </div> + <Input + value={credentials} + onChange={e => setCredentials(e.target.value)} + onBlur={e => handleBlur(e.target.value.trim())} + placeholder={t('tools.mcp.modal.clientSecretPlaceholder')} + disabled={isDynamicRegistration} + /> + </div> + </> + ) + } + { + authMethod === MCPAuthMethod.headers && ( + <div> + <div className='mb-1 flex h-6 items-center'> + <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.headers')}</span> + </div> + <div className='body-xs-regular mb-2 text-text-tertiary'>{t('tools.mcp.modal.headersTip')}</div> + <HeadersInput + headersItems={headers} + onChange={setHeaders} + readonly={false} + isMasked={!isCreate && headers.filter(item => item.key.trim()).length > 0} + /> + </div> + ) + } + { + authMethod === MCPAuthMethod.configurations && ( + <> + <div> + <div className='mb-1 flex h-6 items-center'> + <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.timeout')}</span> + </div> + <Input + type='number' + value={timeout} + onChange={e => setMcpTimeout(Number(e.target.value))} + onBlur={e => handleBlur(e.target.value.trim())} + placeholder={t('tools.mcp.modal.timeoutPlaceholder')} + /> + </div> + <div> + <div className='mb-1 flex h-6 items-center'> + <span className='system-sm-medium text-text-secondary'>{t('tools.mcp.modal.sseReadTimeout')}</span> + </div> + <Input + type='number' + value={sseReadTimeout} + onChange={e => setSseReadTimeout(Number(e.target.value))} + onBlur={e => handleBlur(e.target.value.trim())} + placeholder={t('tools.mcp.modal.timeoutPlaceholder')} + /> + </div> + </> + ) + } </div> <div className='flex flex-row-reverse pt-5'> <Button disabled={!name || !url || !serverIdentifier || isFetchingIcon} className='ml-2' variant='primary' onClick={submit}>{data ? t('tools.mcp.modal.save') : t('tools.mcp.modal.confirm')}</Button> diff --git a/web/app/components/tools/provider-list.tsx b/web/app/components/tools/provider-list.tsx index 1679b4469b..01f9f09127 100644 --- a/web/app/components/tools/provider-list.tsx +++ b/web/app/components/tools/provider-list.tsx @@ -11,7 +11,7 @@ import Input from '@/app/components/base/input' import ProviderDetail from '@/app/components/tools/provider/detail' import Empty from '@/app/components/plugins/marketplace/empty' import CustomCreateCard from '@/app/components/tools/provider/custom-create-card' -import WorkflowToolEmpty from '@/app/components/tools/add-tool-modal/empty' +import WorkflowToolEmpty from '@/app/components/tools/provider/empty' import Card from '@/app/components/plugins/card' import CardMoreInfo from '@/app/components/plugins/card/card-more-info' import PluginDetailPanel from '@/app/components/plugins/plugin-detail-panel' diff --git a/web/app/components/tools/add-tool-modal/empty.tsx b/web/app/components/tools/provider/empty.tsx similarity index 96% rename from web/app/components/tools/add-tool-modal/empty.tsx rename to web/app/components/tools/provider/empty.tsx index 5759589c8e..4d69dc1076 100644 --- a/web/app/components/tools/add-tool-modal/empty.tsx +++ b/web/app/components/tools/provider/empty.tsx @@ -35,7 +35,7 @@ const Empty = ({ const hasTitle = t(`tools.addToolModal.${renderType}.title`) !== `tools.addToolModal.${renderType}.title` return ( - <div className='flex h-[336px] flex-col items-center justify-center'> + <div className='flex flex-col items-center justify-center'> <NoToolPlaceholder className={theme === 'dark' ? 'invert' : ''} /> <div className='mb-1 mt-2 text-[13px] font-medium leading-[18px] text-text-primary'> {hasTitle ? t(`tools.addToolModal.${renderType}.title`) : 'No tools available'} diff --git a/web/app/components/tools/types.ts b/web/app/components/tools/types.ts index 623a7b6d8a..1b76afc5c7 100644 --- a/web/app/components/tools/types.ts +++ b/web/app/components/tools/types.ts @@ -34,6 +34,7 @@ export enum CollectionType { workflow = 'workflow', mcp = 'mcp', datasource = 'datasource', + trigger = 'trigger', } export type Emoji = { @@ -65,6 +66,16 @@ export type Collection = { masked_headers?: Record<string, string> is_authorized?: boolean provider?: string + credential_id?: string + is_dynamic_registration?: boolean + authentication?: { + client_id?: string + client_secret?: string + } + configuration?: { + timeout?: number + sse_read_timeout?: number + } } export type ToolParameter = { @@ -75,6 +86,7 @@ export type ToolParameter = { form: string llm_description: string required: boolean + multiple: boolean default: string options?: { label: TypeWithI18N @@ -84,7 +96,33 @@ export type ToolParameter = { max?: number } +export type TriggerParameter = { + name: string + label: TypeWithI18N + human_description: TypeWithI18N + type: string + form: string + llm_description: string + required: boolean + multiple: boolean + default: string + options?: { + label: TypeWithI18N + value: string + }[] +} + // Action +export type Event = { + name: string + author: string + label: TypeWithI18N + description: TypeWithI18N + parameters: TriggerParameter[] + labels: string[] + output_schema: Record<string, any> +} + export type Tool = { name: string author: string @@ -192,3 +230,9 @@ export type MCPServerDetail = { parameters?: Record<string, string> headers?: Record<string, string> } + +export enum MCPAuthMethod { + authentication = 'authentication', + headers = 'headers', + configurations = 'configurations', +} diff --git a/web/app/components/tools/utils/to-form-schema.ts b/web/app/components/tools/utils/to-form-schema.ts index 8e85a5f9b0..69f5dd5f2f 100644 --- a/web/app/components/tools/utils/to-form-schema.ts +++ b/web/app/components/tools/utils/to-form-schema.ts @@ -1,6 +1,7 @@ -import type { ToolCredential, ToolParameter } from '../types' import { FormTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' import { VarType as VarKindType } from '@/app/components/workflow/nodes/tool/types' +import type { TriggerEventParameter } from '../../plugins/types' +import type { ToolCredential, ToolParameter } from '../types' export const toType = (type: string) => { switch (type) { @@ -14,6 +15,21 @@ export const toType = (type: string) => { return type } } + +export const triggerEventParametersToFormSchemas = (parameters: TriggerEventParameter[]) => { + if (!parameters?.length) + return [] + + return parameters.map((parameter) => { + return { + ...parameter, + type: toType(parameter.type), + _type: parameter.type, + tooltip: parameter.description, + } + }) +} + export const toolParametersToFormSchemas = (parameters: ToolParameter[]) => { if (!parameters) return [] @@ -165,7 +181,7 @@ export const getConfiguredValue = (value: Record<string, any>, formSchemas: { va const getVarKindType = (type: FormTypeEnum) => { if (type === FormTypeEnum.file || type === FormTypeEnum.files) return VarKindType.variable - if (type === FormTypeEnum.select || type === FormTypeEnum.boolean || type === FormTypeEnum.textNumber) + if (type === FormTypeEnum.select || type === FormTypeEnum.checkbox || type === FormTypeEnum.textNumber) return VarKindType.constant if (type === FormTypeEnum.textInput || type === FormTypeEnum.secretInput) return VarKindType.mixed diff --git a/web/app/components/tools/workflow-tool/configure-button.tsx b/web/app/components/tools/workflow-tool/configure-button.tsx index 095ed369b2..bf0d789ff9 100644 --- a/web/app/components/tools/workflow-tool/configure-button.tsx +++ b/web/app/components/tools/workflow-tool/configure-button.tsx @@ -28,6 +28,7 @@ type Props = { inputs?: InputVar[] handlePublish: (params?: PublishWorkflowParams) => Promise<void> onRefreshData?: () => void + disabledReason?: string } const WorkflowToolConfigureButton = ({ @@ -41,6 +42,7 @@ const WorkflowToolConfigureButton = ({ inputs, handlePublish, onRefreshData, + disabledReason, }: Props) => { const { t } = useTranslation() const router = useRouter() @@ -200,7 +202,8 @@ const WorkflowToolConfigureButton = ({ {t('workflow.common.configureRequired')} </span> )} - </div>) + </div> + ) : ( <div className='flex items-center justify-start gap-2 p-2 pl-2.5' @@ -214,6 +217,11 @@ const WorkflowToolConfigureButton = ({ </div> </div> )} + {disabledReason && ( + <div className='mt-1 px-2.5 pb-2 text-xs leading-[18px] text-text-tertiary'> + {disabledReason} + </div> + )} {published && ( <div className='border-t-[0.5px] border-divider-regular px-2.5 py-2'> <div className='flex justify-between gap-x-2'> @@ -221,7 +229,7 @@ const WorkflowToolConfigureButton = ({ size='small' className='w-[140px]' onClick={() => setShowModal(true)} - disabled={!isCurrentWorkspaceManager} + disabled={!isCurrentWorkspaceManager || disabled} > {t('workflow.common.configure')} {outdated && <Indicator className='ml-1' color={'yellow'} />} @@ -230,14 +238,17 @@ const WorkflowToolConfigureButton = ({ size='small' className='w-[140px]' onClick={() => router.push('/tools?category=workflow')} + disabled={disabled} > {t('workflow.common.manageInTools')} <RiArrowRightUpLine className='ml-1 h-4 w-4' /> </Button> </div> - {outdated && <div className='mt-1 text-xs leading-[18px] text-text-warning'> - {t('workflow.common.workflowAsToolTip')} - </div>} + {outdated && ( + <div className='mt-1 text-xs leading-[18px] text-text-warning'> + {t('workflow.common.workflowAsToolTip')} + </div> + )} </div> )} </div> diff --git a/web/app/components/workflow-app/components/workflow-children.tsx b/web/app/components/workflow-app/components/workflow-children.tsx index af61e8a849..1c8ed0cdf9 100644 --- a/web/app/components/workflow-app/components/workflow-children.tsx +++ b/web/app/components/workflow-app/components/workflow-children.tsx @@ -1,19 +1,32 @@ import { memo, + useCallback, useState, } from 'react' import type { EnvironmentVariable } from '@/app/components/workflow/types' import { DSL_EXPORT_CHECK } from '@/app/components/workflow/constants' +import { START_INITIAL_POSITION } from '@/app/components/workflow/constants' +import { generateNewNode } from '@/app/components/workflow/utils' import { useStore } from '@/app/components/workflow/store' +import { useStoreApi } from 'reactflow' import PluginDependency from '../../workflow/plugin-dependency' import { + useAutoGenerateWebhookUrl, useDSL, usePanelInteractions, } from '@/app/components/workflow/hooks' +import { useNodesSyncDraft } from '@/app/components/workflow/hooks/use-nodes-sync-draft' import { useEventEmitterContextContext } from '@/context/event-emitter' import WorkflowHeader from './workflow-header' import WorkflowPanel from './workflow-panel' import dynamic from 'next/dynamic' +import { BlockEnum } from '@/app/components/workflow/types' +import type { + PluginDefaultValue, + TriggerDefaultValue, +} from '@/app/components/workflow/block-selector/types' +import { useAutoOnboarding } from '../hooks/use-auto-onboarding' +import { useAvailableNodesMetaData } from '../hooks' const Features = dynamic(() => import('@/app/components/workflow/features'), { ssr: false, @@ -24,6 +37,34 @@ const UpdateDSLModal = dynamic(() => import('@/app/components/workflow/update-ds const DSLExportConfirmModal = dynamic(() => import('@/app/components/workflow/dsl-export-confirm-modal'), { ssr: false, }) +const WorkflowOnboardingModal = dynamic(() => import('./workflow-onboarding-modal'), { + ssr: false, +}) + +const getTriggerPluginNodeData = ( + triggerConfig: TriggerDefaultValue, + fallbackTitle?: string, + fallbackDesc?: string, +) => { + return { + plugin_id: triggerConfig.plugin_id, + provider_id: triggerConfig.provider_name, + provider_type: triggerConfig.provider_type, + provider_name: triggerConfig.provider_name, + event_name: triggerConfig.event_name, + event_label: triggerConfig.event_label, + event_description: triggerConfig.event_description, + title: triggerConfig.event_label || triggerConfig.title || fallbackTitle, + desc: triggerConfig.event_description || fallbackDesc, + output_schema: { ...triggerConfig.output_schema }, + parameters_schema: triggerConfig.paramSchemas ? [...triggerConfig.paramSchemas] : [], + config: { ...triggerConfig.params }, + subscription_id: triggerConfig.subscription_id, + plugin_unique_identifier: triggerConfig.plugin_unique_identifier, + is_team_authorization: triggerConfig.is_team_authorization, + meta: triggerConfig.meta ? { ...triggerConfig.meta } : undefined, + } +} const WorkflowChildren = () => { const { eventEmitter } = useEventEmitterContextContext() @@ -31,6 +72,14 @@ const WorkflowChildren = () => { const showFeaturesPanel = useStore(s => s.showFeaturesPanel) const showImportDSLModal = useStore(s => s.showImportDSLModal) const setShowImportDSLModal = useStore(s => s.setShowImportDSLModal) + const showOnboarding = useStore(s => s.showOnboarding) + const setShowOnboarding = useStore(s => s.setShowOnboarding) + const setHasSelectedStartNode = useStore(s => s.setHasSelectedStartNode) + const setShouldAutoOpenStartNodeSelector = useStore(s => s.setShouldAutoOpenStartNodeSelector) + const reactFlowStore = useStoreApi() + const availableNodesMetaData = useAvailableNodesMetaData() + const { handleSyncWorkflowDraft } = useNodesSyncDraft() + const { handleOnboardingClose } = useAutoOnboarding() const { handlePaneContextmenuCancel, } = usePanelInteractions() @@ -44,12 +93,84 @@ const WorkflowChildren = () => { setSecretEnvList(v.payload.data as EnvironmentVariable[]) }) + const autoGenerateWebhookUrl = useAutoGenerateWebhookUrl() + + const handleCloseOnboarding = useCallback(() => { + handleOnboardingClose() + }, [handleOnboardingClose]) + + const handleSelectStartNode = useCallback((nodeType: BlockEnum, toolConfig?: PluginDefaultValue) => { + const nodeDefault = availableNodesMetaData.nodesMap?.[nodeType] + if (!nodeDefault?.defaultValue) + return + + const baseNodeData = { ...nodeDefault.defaultValue } + + const mergedNodeData = (() => { + if (nodeType !== BlockEnum.TriggerPlugin || !toolConfig) { + return { + ...baseNodeData, + ...toolConfig, + } + } + + const triggerNodeData = getTriggerPluginNodeData( + toolConfig as TriggerDefaultValue, + baseNodeData.title, + baseNodeData.desc, + ) + + return { + ...baseNodeData, + ...triggerNodeData, + config: { + ...(baseNodeData as { config?: Record<string, any> }).config, + ...triggerNodeData.config, + }, + } + })() + + const { newNode } = generateNewNode({ + data: { + ...mergedNodeData, + } as any, + position: START_INITIAL_POSITION, + }) + + const { setNodes, setEdges } = reactFlowStore.getState() + setNodes([newNode]) + setEdges([]) + + setShowOnboarding?.(false) + setHasSelectedStartNode?.(true) + setShouldAutoOpenStartNodeSelector?.(true) + + handleSyncWorkflowDraft(true, false, { + onSuccess: () => { + autoGenerateWebhookUrl(newNode.id) + console.log('Node successfully saved to draft') + }, + onError: () => { + console.error('Failed to save node to draft') + }, + }) + }, [availableNodesMetaData, setShowOnboarding, setHasSelectedStartNode, reactFlowStore, handleSyncWorkflowDraft]) + return ( <> <PluginDependency /> { showFeaturesPanel && <Features /> } + { + showOnboarding && ( + <WorkflowOnboardingModal + isShow={showOnboarding} + onClose={handleCloseOnboarding} + onSelectStartNode={handleSelectStartNode} + /> + ) + } { showImportDSLModal && ( <UpdateDSLModal diff --git a/web/app/components/workflow-app/components/workflow-header/features-trigger.tsx b/web/app/components/workflow-app/components/workflow-header/features-trigger.tsx index 05b37c1469..d229006177 100644 --- a/web/app/components/workflow-app/components/workflow-header/features-trigger.tsx +++ b/web/app/components/workflow-app/components/workflow-header/features-trigger.tsx @@ -3,7 +3,7 @@ import { useCallback, useMemo, } from 'react' -import { useEdges, useNodes, useStore as useReactflowStore } from 'reactflow' +import { useEdges, useNodes } from 'reactflow' import { RiApps2AddLine } from '@remixicon/react' import { useTranslation } from 'react-i18next' import { @@ -15,6 +15,7 @@ import { useChecklistBeforePublish, useNodesReadOnly, useNodesSyncDraft, + // useWorkflowRunValidation, } from '@/app/components/workflow/hooks' import Button from '@/app/components/base/button' import AppPublisher from '@/app/components/app/app-publisher' @@ -22,36 +23,44 @@ import { useFeatures } from '@/app/components/base/features/hooks' import type { CommonEdgeType, CommonNodeType, + Node, } from '@/app/components/workflow/types' import { BlockEnum, InputVarType, + isTriggerNode, } from '@/app/components/workflow/types' import { useToastContext } from '@/app/components/base/toast' import { useInvalidateAppWorkflow, usePublishWorkflow, useResetWorkflowVersionHistory } from '@/service/use-workflow' +import { useInvalidateAppTriggers } from '@/service/use-tools' import type { PublishWorkflowParams } from '@/types/workflow' import { fetchAppDetail } from '@/service/apps' import { useStore as useAppStore } from '@/app/components/app/store' import useTheme from '@/hooks/use-theme' import cn from '@/utils/classnames' +import { useIsChatMode } from '@/app/components/workflow/hooks' +import type { StartNodeType } from '@/app/components/workflow/nodes/start/types' const FeaturesTrigger = () => { const { t } = useTranslation() const { theme } = useTheme() + const isChatMode = useIsChatMode() const workflowStore = useWorkflowStore() const appDetail = useAppStore(s => s.appDetail) const appID = appDetail?.id const setAppDetail = useAppStore(s => s.setAppDetail) - const { - nodesReadOnly, - getNodesReadOnly, - } = useNodesReadOnly() + const { nodesReadOnly, getNodesReadOnly } = useNodesReadOnly() const publishedAt = useStore(s => s.publishedAt) const draftUpdatedAt = useStore(s => s.draftUpdatedAt) const toolPublished = useStore(s => s.toolPublished) - const startVariables = useReactflowStore( - s => s.getNodes().find(node => node.data.type === BlockEnum.Start)?.data.variables, - ) + const lastPublishedHasUserInput = useStore(s => s.lastPublishedHasUserInput) + + const nodes = useNodes<CommonNodeType>() + const hasWorkflowNodes = nodes.length > 0 + const startNode = nodes.find(node => node.data.type === BlockEnum.Start) + const startVariables = (startNode as Node<StartNodeType>)?.data?.variables + const edges = useEdges<CommonEdgeType>() + const fileSettings = useFeatures(s => s.features.file) const variables = useMemo(() => { const data = startVariables || [] @@ -73,6 +82,22 @@ const FeaturesTrigger = () => { const { handleCheckBeforePublish } = useChecklistBeforePublish() const { handleSyncWorkflowDraft } = useNodesSyncDraft() const { notify } = useToastContext() + const startNodeIds = useMemo( + () => nodes.filter(node => node.data.type === BlockEnum.Start).map(node => node.id), + [nodes], + ) + const hasUserInputNode = useMemo(() => { + if (!startNodeIds.length) + return false + return edges.some(edge => startNodeIds.includes(edge.source)) + }, [edges, startNodeIds]) + // Track trigger presence so the publisher can adjust UI (e.g. hide missing start section). + const hasTriggerNode = useMemo(() => ( + nodes.some(node => isTriggerNode(node.data.type as BlockEnum)) + ), [nodes]) + + const resetWorkflowVersionHistory = useResetWorkflowVersionHistory() + const invalidateAppTriggers = useInvalidateAppTriggers() const handleShowFeatures = useCallback(() => { const { @@ -85,8 +110,6 @@ const FeaturesTrigger = () => { setShowFeaturesPanel(!showFeaturesPanel) }, [workflowStore, getNodesReadOnly]) - const resetWorkflowVersionHistory = useResetWorkflowVersionHistory() - const updateAppDetail = useCallback(async () => { try { const res = await fetchAppDetail({ url: '/apps', id: appID! }) @@ -96,14 +119,17 @@ const FeaturesTrigger = () => { console.error(error) } }, [appID, setAppDetail]) + const { mutateAsync: publishWorkflow } = usePublishWorkflow() - const nodes = useNodes<CommonNodeType>() - const edges = useEdges<CommonEdgeType>() + // const { validateBeforeRun } = useWorkflowRunValidation() const needWarningNodes = useChecklist(nodes, edges) const updatePublishedWorkflow = useInvalidateAppWorkflow() const onPublish = useCallback(async (params?: PublishWorkflowParams) => { // First check if there are any items in the checklist + // if (!validateBeforeRun()) + // throw new Error('Checklist has unresolved items') + if (needWarningNodes.length > 0) { notify({ type: 'error', message: t('workflow.panel.checklistTip') }) throw new Error('Checklist has unresolved items') @@ -121,14 +147,16 @@ const FeaturesTrigger = () => { notify({ type: 'success', message: t('common.api.actionSuccess') }) updatePublishedWorkflow(appID!) updateAppDetail() + invalidateAppTriggers(appID!) workflowStore.getState().setPublishedAt(res.created_at) + workflowStore.getState().setLastPublishedHasUserInput(hasUserInputNode) resetWorkflowVersionHistory() } } else { throw new Error('Checklist failed') } - }, [needWarningNodes, handleCheckBeforePublish, publishWorkflow, notify, appID, t, updatePublishedWorkflow, updateAppDetail, workflowStore, resetWorkflowVersionHistory]) + }, [needWarningNodes, handleCheckBeforePublish, publishWorkflow, notify, appID, t, updatePublishedWorkflow, updateAppDetail, workflowStore, resetWorkflowVersionHistory, invalidateAppTriggers]) const onPublisherToggle = useCallback((state: boolean) => { if (state) @@ -141,27 +169,34 @@ const FeaturesTrigger = () => { return ( <> - <Button - className={cn( - 'text-components-button-secondary-text', - theme === 'dark' && 'rounded-lg border border-black/5 bg-white/10 backdrop-blur-sm', - )} - onClick={handleShowFeatures} - > - <RiApps2AddLine className='mr-1 h-4 w-4 text-components-button-secondary-text' /> - {t('workflow.common.features')} - </Button> + {/* Feature button is only visible in chatflow mode (advanced-chat) */} + {isChatMode && ( + <Button + className={cn( + 'text-components-button-secondary-text', + theme === 'dark' && 'rounded-lg border border-black/5 bg-white/10 backdrop-blur-sm', + )} + onClick={handleShowFeatures} + > + <RiApps2AddLine className='mr-1 h-4 w-4 text-components-button-secondary-text' /> + {t('workflow.common.features')} + </Button> + )} <AppPublisher {...{ publishedAt, draftUpdatedAt, - disabled: nodesReadOnly, + disabled: nodesReadOnly || !hasWorkflowNodes, toolPublished, inputs: variables, onRefreshData: handleToolConfigureUpdate, onPublish, onToggle: onPublisherToggle, + workflowToolAvailable: lastPublishedHasUserInput, crossAxisOffset: 4, + missingStartNode: !startNode, + hasTriggerNode, + publishDisabled: !hasWorkflowNodes, }} /> </> diff --git a/web/app/components/workflow-app/components/workflow-header/index.tsx b/web/app/components/workflow-app/components/workflow-header/index.tsx index 53a050146e..c0b8a37b87 100644 --- a/web/app/components/workflow-app/components/workflow-header/index.tsx +++ b/web/app/components/workflow-app/components/workflow-header/index.tsx @@ -41,8 +41,8 @@ const WorkflowHeader = () => { return { normal: { components: { - left: <ChatVariableTrigger />, middle: <FeaturesTrigger />, + chatVariableTrigger: <ChatVariableTrigger />, }, runAndHistoryProps: { showRunButton: !isChatMode, diff --git a/web/app/components/workflow-app/components/workflow-main.tsx b/web/app/components/workflow-app/components/workflow-main.tsx index f979a12f26..e90b2904c9 100644 --- a/web/app/components/workflow-app/components/workflow-main.tsx +++ b/web/app/components/workflow-app/components/workflow-main.tsx @@ -66,6 +66,10 @@ const WorkflowMain = ({ handleStartWorkflowRun, handleWorkflowStartRunInChatflow, handleWorkflowStartRunInWorkflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, } = useWorkflowStartRun() const availableNodesMetaData = useAvailableNodesMetaData() const { getWorkflowRunAndTraceUrl } = useGetRunAndTraceUrl() @@ -108,6 +112,10 @@ const WorkflowMain = ({ handleStartWorkflowRun, handleWorkflowStartRunInChatflow, handleWorkflowStartRunInWorkflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, availableNodesMetaData, getWorkflowRunAndTraceUrl, exportCheck, @@ -141,6 +149,10 @@ const WorkflowMain = ({ handleStartWorkflowRun, handleWorkflowStartRunInChatflow, handleWorkflowStartRunInWorkflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, availableNodesMetaData, getWorkflowRunAndTraceUrl, exportCheck, diff --git a/web/app/components/workflow-app/components/workflow-onboarding-modal/index.tsx b/web/app/components/workflow-app/components/workflow-onboarding-modal/index.tsx new file mode 100644 index 0000000000..747a232ca7 --- /dev/null +++ b/web/app/components/workflow-app/components/workflow-onboarding-modal/index.tsx @@ -0,0 +1,99 @@ +'use client' +import type { FC } from 'react' +import { + useCallback, + useEffect, +} from 'react' +import { useTranslation } from 'react-i18next' +import { BlockEnum } from '@/app/components/workflow/types' +import type { PluginDefaultValue } from '@/app/components/workflow/block-selector/types' +import Modal from '@/app/components/base/modal' +import StartNodeSelectionPanel from './start-node-selection-panel' +import { useDocLink } from '@/context/i18n' + +type WorkflowOnboardingModalProps = { + isShow: boolean + onClose: () => void + onSelectStartNode: (nodeType: BlockEnum, toolConfig?: PluginDefaultValue) => void +} + +const WorkflowOnboardingModal: FC<WorkflowOnboardingModalProps> = ({ + isShow, + onClose, + onSelectStartNode, +}) => { + const { t } = useTranslation() + const docLink = useDocLink() + + const handleSelectUserInput = useCallback(() => { + onSelectStartNode(BlockEnum.Start) + onClose() // Close modal after selection + }, [onSelectStartNode, onClose]) + + const handleTriggerSelect = useCallback((nodeType: BlockEnum, toolConfig?: PluginDefaultValue) => { + onSelectStartNode(nodeType, toolConfig) + onClose() // Close modal after selection + }, [onSelectStartNode, onClose]) + + useEffect(() => { + const handleEsc = (e: KeyboardEvent) => { + if (e.key === 'Escape' && isShow) + onClose() + } + document.addEventListener('keydown', handleEsc) + return () => document.removeEventListener('keydown', handleEsc) + }, [isShow, onClose]) + + return ( + <> + <Modal + isShow={isShow} + onClose={onClose} + className="w-[618px] max-w-[618px] rounded-2xl border border-effects-highlight bg-background-default-subtle shadow-lg" + overlayOpacity + closable + clickOutsideNotClose + > + <div className="pb-4"> + {/* Header */} + <div className="mb-6"> + <h3 className="title-2xl-semi-bold mb-2 text-text-primary"> + {t('workflow.onboarding.title')} + </h3> + <div className="body-xs-regular leading-4 text-text-tertiary"> + {t('workflow.onboarding.description')}{' '} + <a + href={docLink('/guides/workflow/node/start')} + target="_blank" + rel="noopener noreferrer" + className="hover:text-text-accent-hover cursor-pointer text-text-accent underline" + > + {t('workflow.onboarding.learnMore')} + </a>{' '} + {t('workflow.onboarding.aboutStartNode')} + </div> + </div> + + {/* Content */} + <StartNodeSelectionPanel + onSelectUserInput={handleSelectUserInput} + onSelectTrigger={handleTriggerSelect} + /> + </div> + </Modal> + + {/* ESC tip below modal */} + {isShow && ( + <div className="body-xs-regular pointer-events-none fixed left-1/2 top-1/2 z-[70] flex -translate-x-1/2 translate-y-[165px] items-center gap-1 text-text-quaternary"> + <span>{t('workflow.onboarding.escTip.press')}</span> + <kbd className="system-kbd inline-flex h-4 min-w-4 items-center justify-center rounded bg-components-kbd-bg-gray px-1 text-text-tertiary"> + {t('workflow.onboarding.escTip.key')} + </kbd> + <span>{t('workflow.onboarding.escTip.toDismiss')}</span> + </div> + )} + </> + ) +} + +export default WorkflowOnboardingModal diff --git a/web/app/components/workflow-app/components/workflow-onboarding-modal/start-node-option.tsx b/web/app/components/workflow-app/components/workflow-onboarding-modal/start-node-option.tsx new file mode 100644 index 0000000000..e28de39fdd --- /dev/null +++ b/web/app/components/workflow-app/components/workflow-onboarding-modal/start-node-option.tsx @@ -0,0 +1,53 @@ +'use client' +import type { FC, ReactNode } from 'react' +import cn from '@/utils/classnames' + +type StartNodeOptionProps = { + icon: ReactNode + title: string + subtitle?: string + description: string + onClick: () => void +} + +const StartNodeOption: FC<StartNodeOptionProps> = ({ + icon, + title, + subtitle, + description, + onClick, +}) => { + return ( + <div + onClick={onClick} + className={cn( + 'hover:border-components-panel-border-active flex h-40 w-[280px] cursor-pointer flex-col gap-2 rounded-xl border-[0.5px] border-components-option-card-option-border bg-components-panel-on-panel-item-bg p-4 shadow-sm transition-all hover:shadow-md', + )} + > + {/* Icon */} + <div className="shrink-0"> + {icon} + </div> + + {/* Text content */} + <div className="flex h-[74px] flex-col gap-1 py-0.5"> + <div className="h-5 leading-5"> + <h3 className="system-md-semi-bold text-text-primary"> + {title} + {subtitle && ( + <span className="system-md-regular text-text-quaternary"> {subtitle}</span> + )} + </h3> + </div> + + <div className="h-12 leading-4"> + <p className="system-xs-regular text-text-tertiary"> + {description} + </p> + </div> + </div> + </div> + ) +} + +export default StartNodeOption diff --git a/web/app/components/workflow-app/components/workflow-onboarding-modal/start-node-selection-panel.tsx b/web/app/components/workflow-app/components/workflow-onboarding-modal/start-node-selection-panel.tsx new file mode 100644 index 0000000000..de934a13b2 --- /dev/null +++ b/web/app/components/workflow-app/components/workflow-onboarding-modal/start-node-selection-panel.tsx @@ -0,0 +1,80 @@ +'use client' +import type { FC } from 'react' +import { useCallback, useState } from 'react' +import { useTranslation } from 'react-i18next' +import StartNodeOption from './start-node-option' +import NodeSelector from '@/app/components/workflow/block-selector' +import { Home } from '@/app/components/base/icons/src/vender/workflow' +import { TriggerAll } from '@/app/components/base/icons/src/vender/workflow' +import { BlockEnum } from '@/app/components/workflow/types' +import type { PluginDefaultValue } from '@/app/components/workflow/block-selector/types' +import { TabsEnum } from '@/app/components/workflow/block-selector/types' + +type StartNodeSelectionPanelProps = { + onSelectUserInput: () => void + onSelectTrigger: (nodeType: BlockEnum, toolConfig?: PluginDefaultValue) => void +} + +const StartNodeSelectionPanel: FC<StartNodeSelectionPanelProps> = ({ + onSelectUserInput, + onSelectTrigger, +}) => { + const { t } = useTranslation() + const [showTriggerSelector, setShowTriggerSelector] = useState(false) + + const handleTriggerClick = useCallback(() => { + setShowTriggerSelector(true) + }, []) + + const handleTriggerSelect = useCallback((nodeType: BlockEnum, toolConfig?: PluginDefaultValue) => { + setShowTriggerSelector(false) + onSelectTrigger(nodeType, toolConfig) + }, [onSelectTrigger]) + + return ( + <div className="grid grid-cols-2 gap-4"> + <StartNodeOption + icon={ + <div className="flex h-9 w-9 items-center justify-center rounded-[10px] border-[0.5px] border-transparent bg-util-colors-blue-brand-blue-brand-500 p-2"> + <Home className="h-5 w-5 text-white" /> + </div> + } + title={t('workflow.onboarding.userInputFull')} + description={t('workflow.onboarding.userInputDescription')} + onClick={onSelectUserInput} + /> + + <NodeSelector + open={showTriggerSelector} + onOpenChange={setShowTriggerSelector} + onSelect={handleTriggerSelect} + placement="right" + offset={-200} + noBlocks={true} + showStartTab={true} + defaultActiveTab={TabsEnum.Start} + forceShowStartContent={true} + availableBlocksTypes={[ + BlockEnum.TriggerSchedule, + BlockEnum.TriggerWebhook, + BlockEnum.TriggerPlugin, + ]} + trigger={() => ( + <StartNodeOption + icon={ + <div className="flex h-9 w-9 items-center justify-center rounded-[10px] border-[0.5px] border-transparent bg-util-colors-blue-brand-blue-brand-500 p-2"> + <TriggerAll className="h-5 w-5 text-white" /> + </div> + } + title={t('workflow.onboarding.trigger')} + description={t('workflow.onboarding.triggerDescription')} + onClick={handleTriggerClick} + /> + )} + popupClassName="z-[1200]" + /> + </div> + ) +} + +export default StartNodeSelectionPanel diff --git a/web/app/components/workflow-app/hooks/use-auto-onboarding.ts b/web/app/components/workflow-app/hooks/use-auto-onboarding.ts new file mode 100644 index 0000000000..e4f5774adf --- /dev/null +++ b/web/app/components/workflow-app/hooks/use-auto-onboarding.ts @@ -0,0 +1,68 @@ +import { useCallback, useEffect } from 'react' +import { useStoreApi } from 'reactflow' +import { useWorkflowStore } from '@/app/components/workflow/store' + +export const useAutoOnboarding = () => { + const store = useStoreApi() + const workflowStore = useWorkflowStore() + + const checkAndShowOnboarding = useCallback(() => { + const { getNodes } = store.getState() + const { + showOnboarding, + hasShownOnboarding, + notInitialWorkflow, + setShowOnboarding, + setHasShownOnboarding, + setShouldAutoOpenStartNodeSelector, + } = workflowStore.getState() + + // Skip if already showing onboarding or it's the initial workflow creation + if (showOnboarding || notInitialWorkflow) + return + + const nodes = getNodes() + + // Check if canvas is completely empty (no nodes at all) + // Only trigger onboarding when canvas is completely blank to avoid data loss + const isCompletelyEmpty = nodes.length === 0 + + // Show onboarding only if canvas is completely empty and we haven't shown it before in this session + if (isCompletelyEmpty && !hasShownOnboarding) { + setShowOnboarding?.(true) + setHasShownOnboarding?.(true) + setShouldAutoOpenStartNodeSelector?.(true) + } + }, [store, workflowStore]) + + const handleOnboardingClose = useCallback(() => { + const { + setShowOnboarding, + setHasShownOnboarding, + setShouldAutoOpenStartNodeSelector, + hasSelectedStartNode, + setHasSelectedStartNode, + } = workflowStore.getState() + setShowOnboarding?.(false) + setHasShownOnboarding?.(true) + if (hasSelectedStartNode) + setHasSelectedStartNode?.(false) + else + setShouldAutoOpenStartNodeSelector?.(false) + }, [workflowStore]) + + // Check on mount and when nodes change + useEffect(() => { + // Small delay to ensure the workflow data is loaded + const timer = setTimeout(() => { + checkAndShowOnboarding() + }, 500) + + return () => clearTimeout(timer) + }, [checkAndShowOnboarding]) + + return { + checkAndShowOnboarding, + handleOnboardingClose, + } +} diff --git a/web/app/components/workflow-app/hooks/use-available-nodes-meta-data.ts b/web/app/components/workflow-app/hooks/use-available-nodes-meta-data.ts index ba51b06401..b95d4d47f7 100644 --- a/web/app/components/workflow-app/hooks/use-available-nodes-meta-data.ts +++ b/web/app/components/workflow-app/hooks/use-available-nodes-meta-data.ts @@ -1,7 +1,10 @@ import { useMemo } from 'react' import { useTranslation } from 'react-i18next' -import { useGetLanguage } from '@/context/i18n' +import { useDocLink } from '@/context/i18n' import StartDefault from '@/app/components/workflow/nodes/start/default' +import TriggerWebhookDefault from '@/app/components/workflow/nodes/trigger-webhook/default' +import TriggerScheduleDefault from '@/app/components/workflow/nodes/trigger-schedule/default' +import TriggerPluginDefault from '@/app/components/workflow/nodes/trigger-plugin/default' import EndDefault from '@/app/components/workflow/nodes/end/default' import AnswerDefault from '@/app/components/workflow/nodes/answer/default' import { WORKFLOW_COMMON_NODES } from '@/app/components/workflow/constants/node' @@ -12,36 +15,43 @@ import { BlockEnum } from '@/app/components/workflow/types' export const useAvailableNodesMetaData = () => { const { t } = useTranslation() const isChatMode = useIsChatMode() - const language = useGetLanguage() + const docLink = useDocLink() + + const startNodeMetaData = useMemo(() => ({ + ...StartDefault, + metaData: { + ...StartDefault.metaData, + isUndeletable: isChatMode, // start node is undeletable in chat mode, @use-nodes-interactions: handleNodeDelete function + }, + }), [isChatMode]) const mergedNodesMetaData = useMemo(() => [ ...WORKFLOW_COMMON_NODES, - StartDefault, + startNodeMetaData, ...( isChatMode ? [AnswerDefault] - : [EndDefault] + : [ + EndDefault, + TriggerWebhookDefault, + TriggerScheduleDefault, + TriggerPluginDefault, + ] ), - ], [isChatMode]) - - const prefixLink = useMemo(() => { - if (language === 'zh_Hans') - return 'https://docs.dify.ai/zh-hans/guides/workflow/node/' - - return 'https://docs.dify.ai/guides/workflow/node/' - }, [language]) + ], [isChatMode, startNodeMetaData]) const availableNodesMetaData = useMemo(() => mergedNodesMetaData.map((node) => { const { metaData } = node const title = t(`workflow.blocks.${metaData.type}`) const description = t(`workflow.blocksAbout.${metaData.type}`) + const helpLinkPath = `guides/workflow/node/${metaData.helpLinkUri}` return { ...node, metaData: { ...metaData, title, description, - helpLinkUri: `${prefixLink}${metaData.helpLinkUri}`, + helpLinkUri: docLink(helpLinkPath), }, defaultValue: { ...node.defaultValue, @@ -49,7 +59,7 @@ export const useAvailableNodesMetaData = () => { title, }, } - }), [mergedNodesMetaData, t, prefixLink]) + }), [mergedNodesMetaData, t, docLink]) const availableNodesMetaDataMap = useMemo(() => availableNodesMetaData.reduce((acc, node) => { acc![node.metaData.type] = node diff --git a/web/app/components/workflow-app/hooks/use-is-chat-mode.ts b/web/app/components/workflow-app/hooks/use-is-chat-mode.ts index 3cdfc77b2a..d286c1a540 100644 --- a/web/app/components/workflow-app/hooks/use-is-chat-mode.ts +++ b/web/app/components/workflow-app/hooks/use-is-chat-mode.ts @@ -1,7 +1,8 @@ import { useStore as useAppStore } from '@/app/components/app/store' +import { AppModeEnum } from '@/types/app' export const useIsChatMode = () => { const appDetail = useAppStore(s => s.appDetail) - return appDetail?.mode === 'advanced-chat' + return appDetail?.mode === AppModeEnum.ADVANCED_CHAT } diff --git a/web/app/components/workflow-app/hooks/use-nodes-sync-draft.ts b/web/app/components/workflow-app/hooks/use-nodes-sync-draft.ts index c1f40c9d8c..56d9021feb 100644 --- a/web/app/components/workflow-app/hooks/use-nodes-sync-draft.ts +++ b/web/app/components/workflow-app/hooks/use-nodes-sync-draft.ts @@ -1,14 +1,9 @@ import { useCallback } from 'react' import { produce } from 'immer' import { useStoreApi } from 'reactflow' -import { useParams } from 'next/navigation' -import { - useWorkflowStore, -} from '@/app/components/workflow/store' -import { BlockEnum } from '@/app/components/workflow/types' -import { - useNodesReadOnly, -} from '@/app/components/workflow/hooks/use-workflow' +import { useWorkflowStore } from '@/app/components/workflow/store' +import { useNodesReadOnly } from '@/app/components/workflow/hooks/use-workflow' +import { useSerialAsyncCallback } from '@/app/components/workflow/hooks/use-serial-async-callback' import { syncWorkflowDraft } from '@/service/workflow' import { useFeaturesStore } from '@/app/components/base/features/hooks' import { API_PREFIX } from '@/config' @@ -20,7 +15,6 @@ export const useNodesSyncDraft = () => { const featuresStore = useFeaturesStore() const { getNodesReadOnly } = useNodesReadOnly() const { handleRefreshWorkflowDraft } = useWorkflowRefreshDraft() - const params = useParams() const getPostParams = useCallback(() => { const { @@ -28,65 +22,60 @@ export const useNodesSyncDraft = () => { edges, transform, } = store.getState() - const nodes = getNodes() + const nodes = getNodes().filter(node => !node.data?._isTempNode) const [x, y, zoom] = transform const { appId, conversationVariables, environmentVariables, syncWorkflowDraftHash, + isWorkflowDataLoaded, } = workflowStore.getState() - if (appId && !!nodes.length) { - const hasStartNode = nodes.find(node => node.data.type === BlockEnum.Start) + if (!appId || !isWorkflowDataLoaded) + return null - if (!hasStartNode) - return - - const features = featuresStore!.getState().features - const producedNodes = produce(nodes, (draft) => { - draft.forEach((node) => { - Object.keys(node.data).forEach((key) => { - if (key.startsWith('_')) - delete node.data[key] - }) + const features = featuresStore!.getState().features + const producedNodes = produce(nodes, (draft) => { + draft.forEach((node) => { + Object.keys(node.data).forEach((key) => { + if (key.startsWith('_')) + delete node.data[key] }) }) - const producedEdges = produce(edges.filter(edge => !edge.data?._isTemp), (draft) => { - draft.forEach((edge) => { - Object.keys(edge.data).forEach((key) => { - if (key.startsWith('_')) - delete edge.data[key] - }) + }) + const producedEdges = produce(edges.filter(edge => !edge.data?._isTemp), (draft) => { + draft.forEach((edge) => { + Object.keys(edge.data).forEach((key) => { + if (key.startsWith('_')) + delete edge.data[key] }) }) - return { - url: `/apps/${appId}/workflows/draft`, - params: { - graph: { - nodes: producedNodes, - edges: producedEdges, - viewport: { - x, - y, - zoom, - }, - }, - features: { - opening_statement: features.opening?.enabled ? (features.opening?.opening_statement || '') : '', - suggested_questions: features.opening?.enabled ? (features.opening?.suggested_questions || []) : [], - suggested_questions_after_answer: features.suggested, - text_to_speech: features.text2speech, - speech_to_text: features.speech2text, - retriever_resource: features.citation, - sensitive_word_avoidance: features.moderation, - file_upload: features.file, - }, - environment_variables: environmentVariables, - conversation_variables: conversationVariables, - hash: syncWorkflowDraftHash, + }) + const viewport = { x, y, zoom } + + return { + url: `/apps/${appId}/workflows/draft`, + params: { + graph: { + nodes: producedNodes, + edges: producedEdges, + viewport, }, - } + features: { + opening_statement: features.opening?.enabled ? (features.opening?.opening_statement || '') : '', + suggested_questions: features.opening?.enabled ? (features.opening?.suggested_questions || []) : [], + suggested_questions_after_answer: features.suggested, + text_to_speech: features.text2speech, + speech_to_text: features.speech2text, + retriever_resource: features.citation, + sensitive_word_avoidance: features.moderation, + file_upload: features.file, + }, + environment_variables: environmentVariables, + conversation_variables: conversationVariables, + hash: syncWorkflowDraftHash, + }, } }, [store, featuresStore, workflowStore]) @@ -95,15 +84,11 @@ export const useNodesSyncDraft = () => { return const postParams = getPostParams() - if (postParams) { - navigator.sendBeacon( - `${API_PREFIX}/apps/${params.appId}/workflows/draft`, - JSON.stringify(postParams.params), - ) - } - }, [getPostParams, params.appId, getNodesReadOnly]) + if (postParams) + navigator.sendBeacon(`${API_PREFIX}${postParams.url}`, JSON.stringify(postParams.params)) + }, [getPostParams, getNodesReadOnly]) - const doSyncWorkflowDraft = useCallback(async ( + const performSync = useCallback(async ( notRefreshWhenSyncError?: boolean, callback?: { onSuccess?: () => void @@ -141,6 +126,8 @@ export const useNodesSyncDraft = () => { } }, [workflowStore, getPostParams, getNodesReadOnly, handleRefreshWorkflowDraft]) + const doSyncWorkflowDraft = useSerialAsyncCallback(performSync, getNodesReadOnly) + return { doSyncWorkflowDraft, syncWorkflowDraftWhenPageClose, diff --git a/web/app/components/workflow-app/hooks/use-workflow-init.ts b/web/app/components/workflow-app/hooks/use-workflow-init.ts index fadd2007bc..a0a6cc22a1 100644 --- a/web/app/components/workflow-app/hooks/use-workflow-init.ts +++ b/web/app/components/workflow-app/hooks/use-workflow-init.ts @@ -18,7 +18,20 @@ import { import type { FetchWorkflowDraftResponse } from '@/types/workflow' import { useWorkflowConfig } from '@/service/use-workflow' import type { FileUploadConfigResponse } from '@/models/common' +import type { Edge, Node } from '@/app/components/workflow/types' +import { BlockEnum } from '@/app/components/workflow/types' +import { AppModeEnum } from '@/types/app' +const hasConnectedUserInput = (nodes: Node[] = [], edges: Edge[] = []): boolean => { + const startNodeIds = nodes + .filter(node => node?.data?.type === BlockEnum.Start) + .map(node => node.id) + + if (!startNodeIds.length) + return false + + return edges.some(edge => startNodeIds.includes(edge.source)) +} export const useWorkflowInit = () => { const workflowStore = useWorkflowStore() const { @@ -53,6 +66,7 @@ export const useWorkflowInit = () => { }, {} as Record<string, string>), environmentVariables: res.environment_variables?.map(env => env.value_type === 'secret' ? { ...env, value: '[__HIDDEN__]' } : env) || [], conversationVariables: res.conversation_variables || [], + isWorkflowDataLoaded: true, }) setSyncWorkflowDraftHash(res.hash) setIsLoading(false) @@ -61,13 +75,22 @@ export const useWorkflowInit = () => { if (error && error.json && !error.bodyUsed && appDetail) { error.json().then((err: any) => { if (err.code === 'draft_workflow_not_exist') { - workflowStore.setState({ notInitialWorkflow: true }) + const isAdvancedChat = appDetail.mode === AppModeEnum.ADVANCED_CHAT + workflowStore.setState({ + notInitialWorkflow: true, + showOnboarding: !isAdvancedChat, + shouldAutoOpenStartNodeSelector: !isAdvancedChat, + hasShownOnboarding: false, + }) + const nodesData = isAdvancedChat ? nodesTemplate : [] + const edgesData = isAdvancedChat ? edgesTemplate : [] + syncWorkflowDraft({ url: `/apps/${appDetail.id}/workflows/draft`, params: { graph: { - nodes: nodesTemplate, - edges: edgesTemplate, + nodes: nodesData, + edges: edgesData, }, features: { retriever_resource: { enabled: true }, @@ -101,9 +124,14 @@ export const useWorkflowInit = () => { }, {} as Record<string, any>), }) workflowStore.getState().setPublishedAt(publishedWorkflow?.created_at) + const graph = publishedWorkflow?.graph + workflowStore.getState().setLastPublishedHasUserInput( + hasConnectedUserInput(graph?.nodes, graph?.edges), + ) } catch (e) { console.error(e) + workflowStore.getState().setLastPublishedHasUserInput(false) } }, [workflowStore, appDetail]) diff --git a/web/app/components/workflow-app/hooks/use-workflow-refresh-draft.ts b/web/app/components/workflow-app/hooks/use-workflow-refresh-draft.ts index c944e10c4c..910ddd4a8d 100644 --- a/web/app/components/workflow-app/hooks/use-workflow-refresh-draft.ts +++ b/web/app/components/workflow-app/hooks/use-workflow-refresh-draft.ts @@ -16,18 +16,43 @@ export const useWorkflowRefreshDraft = () => { setEnvironmentVariables, setEnvSecrets, setConversationVariables, + setIsWorkflowDataLoaded, + isWorkflowDataLoaded, + debouncedSyncWorkflowDraft, } = workflowStore.getState() + + if (debouncedSyncWorkflowDraft && typeof (debouncedSyncWorkflowDraft as any).cancel === 'function') + (debouncedSyncWorkflowDraft as any).cancel() + + const wasLoaded = isWorkflowDataLoaded + if (wasLoaded) + setIsWorkflowDataLoaded(false) setIsSyncingWorkflowDraft(true) - fetchWorkflowDraft(`/apps/${appId}/workflows/draft`).then((response) => { - handleUpdateWorkflowCanvas(response.graph as WorkflowDataUpdater) - setSyncWorkflowDraftHash(response.hash) - setEnvSecrets((response.environment_variables || []).filter(env => env.value_type === 'secret').reduce((acc, env) => { - acc[env.id] = env.value - return acc - }, {} as Record<string, string>)) - setEnvironmentVariables(response.environment_variables?.map(env => env.value_type === 'secret' ? { ...env, value: '[__HIDDEN__]' } : env) || []) - setConversationVariables(response.conversation_variables || []) - }).finally(() => setIsSyncingWorkflowDraft(false)) + fetchWorkflowDraft(`/apps/${appId}/workflows/draft`) + .then((response) => { + // Ensure we have a valid workflow structure with viewport + const workflowData: WorkflowDataUpdater = { + nodes: response.graph?.nodes || [], + edges: response.graph?.edges || [], + viewport: response.graph?.viewport || { x: 0, y: 0, zoom: 1 }, + } + handleUpdateWorkflowCanvas(workflowData) + setSyncWorkflowDraftHash(response.hash) + setEnvSecrets((response.environment_variables || []).filter(env => env.value_type === 'secret').reduce((acc, env) => { + acc[env.id] = env.value + return acc + }, {} as Record<string, string>)) + setEnvironmentVariables(response.environment_variables?.map(env => env.value_type === 'secret' ? { ...env, value: '[__HIDDEN__]' } : env) || []) + setConversationVariables(response.conversation_variables || []) + setIsWorkflowDataLoaded(true) + }) + .catch(() => { + if (wasLoaded) + setIsWorkflowDataLoaded(true) + }) + .finally(() => { + setIsSyncingWorkflowDraft(false) + }) }, [handleUpdateWorkflowCanvas, workflowStore]) return { diff --git a/web/app/components/workflow-app/hooks/use-workflow-run.ts b/web/app/components/workflow-app/hooks/use-workflow-run.ts index 0cfcd6099b..3ab1c522e7 100644 --- a/web/app/components/workflow-app/hooks/use-workflow-run.ts +++ b/web/app/components/workflow-app/hooks/use-workflow-run.ts @@ -1,4 +1,4 @@ -import { useCallback } from 'react' +import { useCallback, useRef } from 'react' import { useReactFlow, useStoreApi, @@ -12,7 +12,8 @@ import { useWorkflowUpdate } from '@/app/components/workflow/hooks/use-workflow- import { useWorkflowRunEvent } from '@/app/components/workflow/hooks/use-workflow-run-event/use-workflow-run-event' import { useStore as useAppStore } from '@/app/components/app/store' import type { IOtherOptions } from '@/service/base' -import { ssePost } from '@/service/base' +import Toast from '@/app/components/base/toast' +import { handleStream, ssePost } from '@/service/base' import { stopWorkflowRun } from '@/service/workflow' import { useFeaturesStore } from '@/app/components/base/features/hooks' import { AudioPlayerManager } from '@/app/components/base/audio-btn/audio.player.manager' @@ -22,6 +23,35 @@ import { useNodesSyncDraft } from './use-nodes-sync-draft' import { useInvalidAllLastRun } from '@/service/use-workflow' import { useSetWorkflowVarsWithValue } from '../../workflow/hooks/use-fetch-workflow-inspect-vars' import { useConfigsMap } from './use-configs-map' +import { post } from '@/service/base' +import { ContentType } from '@/service/fetch' +import { TriggerType } from '@/app/components/workflow/header/test-run-menu' +import { AppModeEnum } from '@/types/app' + +type HandleRunMode = TriggerType +type HandleRunOptions = { + mode?: HandleRunMode + scheduleNodeId?: string + webhookNodeId?: string + pluginNodeId?: string + allNodeIds?: string[] +} + +type DebuggableTriggerType = Exclude<TriggerType, TriggerType.UserInput> + +const controllerKeyMap: Record<DebuggableTriggerType, string> = { + [TriggerType.Webhook]: '__webhookDebugAbortController', + [TriggerType.Plugin]: '__pluginDebugAbortController', + [TriggerType.All]: '__allTriggersDebugAbortController', + [TriggerType.Schedule]: '__scheduleDebugAbortController', +} + +const debugLabelMap: Record<DebuggableTriggerType, string> = { + [TriggerType.Webhook]: 'Webhook', + [TriggerType.Plugin]: 'Plugin', + [TriggerType.All]: 'All', + [TriggerType.Schedule]: 'Schedule', +} export const useWorkflowRun = () => { const store = useStoreApi() @@ -39,6 +69,8 @@ export const useWorkflowRun = () => { ...configsMap, }) + const abortControllerRef = useRef<AbortController | null>(null) + const { handleWorkflowStarted, handleWorkflowFinished, @@ -111,7 +143,10 @@ export const useWorkflowRun = () => { const handleRun = useCallback(async ( params: any, callback?: IOtherOptions, + options?: HandleRunOptions, ) => { + const runMode: HandleRunMode = options?.mode ?? TriggerType.UserInput + const resolvedParams = params ?? {} const { getNodes, setNodes, @@ -139,6 +174,7 @@ export const useWorkflowRun = () => { onNodeRetry, onAgentLog, onError, + onCompleted, ...restCallback } = callback || {} workflowStore.setState({ historyWorkflowData: undefined }) @@ -150,175 +186,531 @@ export const useWorkflowRun = () => { clientHeight, } = workflowContainer! - const isInWorkflowDebug = appDetail?.mode === 'workflow' + const isInWorkflowDebug = appDetail?.mode === AppModeEnum.WORKFLOW let url = '' - if (appDetail?.mode === 'advanced-chat') + if (runMode === TriggerType.Plugin || runMode === TriggerType.Webhook || runMode === TriggerType.Schedule) { + if (!appDetail?.id) { + console.error('handleRun: missing app id for trigger plugin run') + return + } + url = `/apps/${appDetail.id}/workflows/draft/trigger/run` + } + else if (runMode === TriggerType.All) { + if (!appDetail?.id) { + console.error('handleRun: missing app id for trigger run all') + return + } + url = `/apps/${appDetail.id}/workflows/draft/trigger/run-all` + } + else if (appDetail?.mode === AppModeEnum.ADVANCED_CHAT) { url = `/apps/${appDetail.id}/advanced-chat/workflows/draft/run` - - if (isInWorkflowDebug) + } + else if (isInWorkflowDebug && appDetail?.id) { url = `/apps/${appDetail.id}/workflows/draft/run` + } + + let requestBody = {} + + if (runMode === TriggerType.Schedule) + requestBody = { node_id: options?.scheduleNodeId } + + else if (runMode === TriggerType.Webhook) + requestBody = { node_id: options?.webhookNodeId } + + else if (runMode === TriggerType.Plugin) + requestBody = { node_id: options?.pluginNodeId } + + else if (runMode === TriggerType.All) + requestBody = { node_ids: options?.allNodeIds } + + else + requestBody = resolvedParams + + if (!url) + return + + if (runMode === TriggerType.Schedule && !options?.scheduleNodeId) { + console.error('handleRun: schedule trigger run requires node id') + return + } + + if (runMode === TriggerType.Webhook && !options?.webhookNodeId) { + console.error('handleRun: webhook trigger run requires node id') + return + } + + if (runMode === TriggerType.Plugin && !options?.pluginNodeId) { + console.error('handleRun: plugin trigger run requires node id') + return + } + + if (runMode === TriggerType.All && !options?.allNodeIds && options?.allNodeIds?.length === 0) { + console.error('handleRun: all trigger run requires node ids') + return + } + + abortControllerRef.current?.abort() + abortControllerRef.current = null const { setWorkflowRunningData, + setIsListening, + setShowVariableInspectPanel, + setListeningTriggerType, + setListeningTriggerNodeIds, + setListeningTriggerIsAll, + setListeningTriggerNodeId, } = workflowStore.getState() - setWorkflowRunningData({ - result: { - inputs_truncated: false, - process_data_truncated: false, - outputs_truncated: false, - status: WorkflowRunningStatus.Running, - }, - tracing: [], - resultText: '', - }) + + if ( + runMode === TriggerType.Webhook + || runMode === TriggerType.Plugin + || runMode === TriggerType.All + || runMode === TriggerType.Schedule + ) { + setIsListening(true) + setShowVariableInspectPanel(true) + setListeningTriggerIsAll(runMode === TriggerType.All) + if (runMode === TriggerType.All) + setListeningTriggerNodeIds(options?.allNodeIds ?? []) + else if (runMode === TriggerType.Webhook && options?.webhookNodeId) + setListeningTriggerNodeIds([options.webhookNodeId]) + else if (runMode === TriggerType.Schedule && options?.scheduleNodeId) + setListeningTriggerNodeIds([options.scheduleNodeId]) + else if (runMode === TriggerType.Plugin && options?.pluginNodeId) + setListeningTriggerNodeIds([options.pluginNodeId]) + else + setListeningTriggerNodeIds([]) + setWorkflowRunningData({ + result: { + status: WorkflowRunningStatus.Running, + inputs_truncated: false, + process_data_truncated: false, + outputs_truncated: false, + }, + tracing: [], + resultText: '', + }) + } + else { + setIsListening(false) + setListeningTriggerType(null) + setListeningTriggerNodeId(null) + setListeningTriggerNodeIds([]) + setListeningTriggerIsAll(false) + setWorkflowRunningData({ + result: { + status: WorkflowRunningStatus.Running, + inputs_truncated: false, + process_data_truncated: false, + outputs_truncated: false, + }, + tracing: [], + resultText: '', + }) + } let ttsUrl = '' let ttsIsPublic = false - if (params.token) { + if (resolvedParams.token) { ttsUrl = '/text-to-audio' ttsIsPublic = true } - else if (params.appId) { + else if (resolvedParams.appId) { if (pathname.search('explore/installed') > -1) - ttsUrl = `/installed-apps/${params.appId}/text-to-audio` + ttsUrl = `/installed-apps/${resolvedParams.appId}/text-to-audio` else - ttsUrl = `/apps/${params.appId}/text-to-audio` + ttsUrl = `/apps/${resolvedParams.appId}/text-to-audio` } const player = AudioPlayerManager.getInstance().getAudioPlayer(ttsUrl, ttsIsPublic, uuidV4(), 'none', 'none', noop) + const clearAbortController = () => { + abortControllerRef.current = null + delete (window as any).__webhookDebugAbortController + delete (window as any).__pluginDebugAbortController + delete (window as any).__scheduleDebugAbortController + delete (window as any).__allTriggersDebugAbortController + } + + const clearListeningState = () => { + const state = workflowStore.getState() + state.setIsListening(false) + state.setListeningTriggerType(null) + state.setListeningTriggerNodeId(null) + state.setListeningTriggerNodeIds([]) + state.setListeningTriggerIsAll(false) + } + + const wrappedOnError = (params: any) => { + clearAbortController() + handleWorkflowFailed() + clearListeningState() + + if (onError) + onError(params) + } + + const wrappedOnCompleted: IOtherOptions['onCompleted'] = async (hasError?: boolean, errorMessage?: string) => { + clearAbortController() + clearListeningState() + if (onCompleted) + onCompleted(hasError, errorMessage) + } + + const baseSseOptions: IOtherOptions = { + ...restCallback, + onWorkflowStarted: (params) => { + const state = workflowStore.getState() + if (state.workflowRunningData) { + state.setWorkflowRunningData(produce(state.workflowRunningData, (draft) => { + draft.resultText = '' + })) + } + handleWorkflowStarted(params) + + if (onWorkflowStarted) + onWorkflowStarted(params) + }, + onWorkflowFinished: (params) => { + clearListeningState() + handleWorkflowFinished(params) + + if (onWorkflowFinished) + onWorkflowFinished(params) + if (isInWorkflowDebug) { + fetchInspectVars({}) + invalidAllLastRun() + } + }, + onNodeStarted: (params) => { + handleWorkflowNodeStarted( + params, + { + clientWidth, + clientHeight, + }, + ) + + if (onNodeStarted) + onNodeStarted(params) + }, + onNodeFinished: (params) => { + handleWorkflowNodeFinished(params) + + if (onNodeFinished) + onNodeFinished(params) + }, + onIterationStart: (params) => { + handleWorkflowNodeIterationStarted( + params, + { + clientWidth, + clientHeight, + }, + ) + + if (onIterationStart) + onIterationStart(params) + }, + onIterationNext: (params) => { + handleWorkflowNodeIterationNext(params) + + if (onIterationNext) + onIterationNext(params) + }, + onIterationFinish: (params) => { + handleWorkflowNodeIterationFinished(params) + + if (onIterationFinish) + onIterationFinish(params) + }, + onLoopStart: (params) => { + handleWorkflowNodeLoopStarted( + params, + { + clientWidth, + clientHeight, + }, + ) + + if (onLoopStart) + onLoopStart(params) + }, + onLoopNext: (params) => { + handleWorkflowNodeLoopNext(params) + + if (onLoopNext) + onLoopNext(params) + }, + onLoopFinish: (params) => { + handleWorkflowNodeLoopFinished(params) + + if (onLoopFinish) + onLoopFinish(params) + }, + onNodeRetry: (params) => { + handleWorkflowNodeRetry(params) + + if (onNodeRetry) + onNodeRetry(params) + }, + onAgentLog: (params) => { + handleWorkflowAgentLog(params) + + if (onAgentLog) + onAgentLog(params) + }, + onTextChunk: (params) => { + handleWorkflowTextChunk(params) + }, + onTextReplace: (params) => { + handleWorkflowTextReplace(params) + }, + onTTSChunk: (messageId: string, audio: string) => { + if (!audio || audio === '') + return + player.playAudioWithAudio(audio, true) + AudioPlayerManager.getInstance().resetMsgId(messageId) + }, + onTTSEnd: (messageId: string, audio: string) => { + player.playAudioWithAudio(audio, false) + }, + onError: wrappedOnError, + onCompleted: wrappedOnCompleted, + } + + const waitWithAbort = (signal: AbortSignal, delay: number) => new Promise<void>((resolve) => { + const timer = window.setTimeout(resolve, delay) + signal.addEventListener('abort', () => { + clearTimeout(timer) + resolve() + }, { once: true }) + }) + + const runTriggerDebug = async (debugType: DebuggableTriggerType) => { + const controller = new AbortController() + abortControllerRef.current = controller + + const controllerKey = controllerKeyMap[debugType] + + ; (window as any)[controllerKey] = controller + + const debugLabel = debugLabelMap[debugType] + + const poll = async (): Promise<void> => { + try { + const response = await post<Response>(url, { + body: requestBody, + signal: controller.signal, + }, { + needAllResponseContent: true, + }) + + if (controller.signal.aborted) + return + + if (!response) { + const message = `${debugLabel} debug request failed` + Toast.notify({ type: 'error', message }) + clearAbortController() + return + } + + const contentType = response.headers.get('content-type') || '' + + if (contentType.includes(ContentType.json)) { + let data: any = null + try { + data = await response.json() + } + catch (jsonError) { + console.error(`handleRun: ${debugLabel.toLowerCase()} debug response parse error`, jsonError) + Toast.notify({ type: 'error', message: `${debugLabel} debug request failed` }) + clearAbortController() + clearListeningState() + return + } + + if (controller.signal.aborted) + return + + if (data?.status === 'waiting') { + const delay = Number(data.retry_in) || 2000 + await waitWithAbort(controller.signal, delay) + if (controller.signal.aborted) + return + await poll() + return + } + + const errorMessage = data?.message || `${debugLabel} debug failed` + Toast.notify({ type: 'error', message: errorMessage }) + clearAbortController() + setWorkflowRunningData({ + result: { + status: WorkflowRunningStatus.Failed, + error: errorMessage, + inputs_truncated: false, + process_data_truncated: false, + outputs_truncated: false, + }, + tracing: [], + }) + clearListeningState() + return + } + + clearListeningState() + handleStream( + response, + baseSseOptions.onData ?? noop, + baseSseOptions.onCompleted, + baseSseOptions.onThought, + baseSseOptions.onMessageEnd, + baseSseOptions.onMessageReplace, + baseSseOptions.onFile, + baseSseOptions.onWorkflowStarted, + baseSseOptions.onWorkflowFinished, + baseSseOptions.onNodeStarted, + baseSseOptions.onNodeFinished, + baseSseOptions.onIterationStart, + baseSseOptions.onIterationNext, + baseSseOptions.onIterationFinish, + baseSseOptions.onLoopStart, + baseSseOptions.onLoopNext, + baseSseOptions.onLoopFinish, + baseSseOptions.onNodeRetry, + baseSseOptions.onParallelBranchStarted, + baseSseOptions.onParallelBranchFinished, + baseSseOptions.onTextChunk, + baseSseOptions.onTTSChunk, + baseSseOptions.onTTSEnd, + baseSseOptions.onTextReplace, + baseSseOptions.onAgentLog, + baseSseOptions.onDataSourceNodeProcessing, + baseSseOptions.onDataSourceNodeCompleted, + baseSseOptions.onDataSourceNodeError, + ) + } + catch (error) { + if (controller.signal.aborted) + return + if (error instanceof Response) { + const data = await error.clone().json() as Record<string, any> + const { error: respError } = data || {} + Toast.notify({ type: 'error', message: respError }) + clearAbortController() + setWorkflowRunningData({ + result: { + status: WorkflowRunningStatus.Failed, + error: respError, + inputs_truncated: false, + process_data_truncated: false, + outputs_truncated: false, + }, + tracing: [], + }) + } + clearListeningState() + } + } + + await poll() + } + + if (runMode === TriggerType.Schedule) { + await runTriggerDebug(TriggerType.Schedule) + return + } + + if (runMode === TriggerType.Webhook) { + await runTriggerDebug(TriggerType.Webhook) + return + } + + if (runMode === TriggerType.Plugin) { + await runTriggerDebug(TriggerType.Plugin) + return + } + + if (runMode === TriggerType.All) { + await runTriggerDebug(TriggerType.All) + return + } + ssePost( url, { - body: params, + body: requestBody, }, { - onWorkflowStarted: (params) => { - handleWorkflowStarted(params) - - if (onWorkflowStarted) - onWorkflowStarted(params) + ...baseSseOptions, + getAbortController: (controller: AbortController) => { + abortControllerRef.current = controller }, - onWorkflowFinished: (params) => { - handleWorkflowFinished(params) - - if (onWorkflowFinished) - onWorkflowFinished(params) - if (isInWorkflowDebug) { - fetchInspectVars({}) - invalidAllLastRun() - } - }, - onError: (params) => { - handleWorkflowFailed() - - if (onError) - onError(params) - }, - onNodeStarted: (params) => { - handleWorkflowNodeStarted( - params, - { - clientWidth, - clientHeight, - }, - ) - - if (onNodeStarted) - onNodeStarted(params) - }, - onNodeFinished: (params) => { - handleWorkflowNodeFinished(params) - - if (onNodeFinished) - onNodeFinished(params) - }, - onIterationStart: (params) => { - handleWorkflowNodeIterationStarted( - params, - { - clientWidth, - clientHeight, - }, - ) - - if (onIterationStart) - onIterationStart(params) - }, - onIterationNext: (params) => { - handleWorkflowNodeIterationNext(params) - - if (onIterationNext) - onIterationNext(params) - }, - onIterationFinish: (params) => { - handleWorkflowNodeIterationFinished(params) - - if (onIterationFinish) - onIterationFinish(params) - }, - onLoopStart: (params) => { - handleWorkflowNodeLoopStarted( - params, - { - clientWidth, - clientHeight, - }, - ) - - if (onLoopStart) - onLoopStart(params) - }, - onLoopNext: (params) => { - handleWorkflowNodeLoopNext(params) - - if (onLoopNext) - onLoopNext(params) - }, - onLoopFinish: (params) => { - handleWorkflowNodeLoopFinished(params) - - if (onLoopFinish) - onLoopFinish(params) - }, - onNodeRetry: (params) => { - handleWorkflowNodeRetry(params) - - if (onNodeRetry) - onNodeRetry(params) - }, - onAgentLog: (params) => { - handleWorkflowAgentLog(params) - - if (onAgentLog) - onAgentLog(params) - }, - onTextChunk: (params) => { - handleWorkflowTextChunk(params) - }, - onTextReplace: (params) => { - handleWorkflowTextReplace(params) - }, - onTTSChunk: (messageId: string, audio: string) => { - if (!audio || audio === '') - return - player.playAudioWithAudio(audio, true) - AudioPlayerManager.getInstance().resetMsgId(messageId) - }, - onTTSEnd: (messageId: string, audio: string) => { - player.playAudioWithAudio(audio, false) - }, - ...restCallback, }, ) }, [store, doSyncWorkflowDraft, workflowStore, pathname, handleWorkflowStarted, handleWorkflowFinished, fetchInspectVars, invalidAllLastRun, handleWorkflowFailed, handleWorkflowNodeStarted, handleWorkflowNodeFinished, handleWorkflowNodeIterationStarted, handleWorkflowNodeIterationNext, handleWorkflowNodeIterationFinished, handleWorkflowNodeLoopStarted, handleWorkflowNodeLoopNext, handleWorkflowNodeLoopFinished, handleWorkflowNodeRetry, handleWorkflowAgentLog, handleWorkflowTextChunk, handleWorkflowTextReplace], ) const handleStopRun = useCallback((taskId: string) => { - const appId = useAppStore.getState().appDetail?.id + const setStoppedState = () => { + const { + setWorkflowRunningData, + setIsListening, + setShowVariableInspectPanel, + setListeningTriggerType, + setListeningTriggerNodeId, + } = workflowStore.getState() - stopWorkflowRun(`/apps/${appId}/workflow-runs/tasks/${taskId}/stop`) - }, []) + setWorkflowRunningData({ + result: { + status: WorkflowRunningStatus.Stopped, + inputs_truncated: false, + process_data_truncated: false, + outputs_truncated: false, + }, + tracing: [], + resultText: '', + }) + setIsListening(false) + setListeningTriggerType(null) + setListeningTriggerNodeId(null) + setShowVariableInspectPanel(true) + } + + if (taskId) { + const appId = useAppStore.getState().appDetail?.id + stopWorkflowRun(`/apps/${appId}/workflow-runs/tasks/${taskId}/stop`) + setStoppedState() + return + } + + // Try webhook debug controller from global variable first + const webhookController = (window as any).__webhookDebugAbortController + if (webhookController) + webhookController.abort() + + const pluginController = (window as any).__pluginDebugAbortController + if (pluginController) + pluginController.abort() + + const scheduleController = (window as any).__scheduleDebugAbortController + if (scheduleController) + scheduleController.abort() + + const allTriggerController = (window as any).__allTriggersDebugAbortController + if (allTriggerController) + allTriggerController.abort() + + // Also try the ref + if (abortControllerRef.current) + abortControllerRef.current.abort() + + abortControllerRef.current = null + setStoppedState() + }, [workflowStore]) const handleRestoreFromPublishedWorkflow = useCallback((publishedWorkflow: VersionHistory) => { const nodes = publishedWorkflow.graph.nodes.map(node => ({ ...node, selected: false, data: { ...node.data, selected: false } })) diff --git a/web/app/components/workflow-app/hooks/use-workflow-start-run.tsx b/web/app/components/workflow-app/hooks/use-workflow-start-run.tsx index 3f5ea1c1df..d2e3b3e3c9 100644 --- a/web/app/components/workflow-app/hooks/use-workflow-start-run.tsx +++ b/web/app/components/workflow-app/hooks/use-workflow-start-run.tsx @@ -12,6 +12,7 @@ import { useNodesSyncDraft, useWorkflowRun, } from '.' +import { TriggerType } from '@/app/components/workflow/header/test-run-menu' export const useWorkflowStartRun = () => { const store = useStoreApi() @@ -40,9 +41,11 @@ export const useWorkflowStartRun = () => { setShowDebugAndPreviewPanel, setShowInputsPanel, setShowEnvPanel, + setShowGlobalVariablePanel, } = workflowStore.getState() setShowEnvPanel(false) + setShowGlobalVariablePanel(false) if (showDebugAndPreviewPanel) { handleCancelDebugAndPreviewPanel() @@ -61,6 +64,203 @@ export const useWorkflowStartRun = () => { } }, [store, workflowStore, featuresStore, handleCancelDebugAndPreviewPanel, handleRun, doSyncWorkflowDraft]) + const handleWorkflowTriggerScheduleRunInWorkflow = useCallback(async (nodeId?: string) => { + if (!nodeId) + return + + const { + workflowRunningData, + showDebugAndPreviewPanel, + setShowDebugAndPreviewPanel, + setShowInputsPanel, + setShowEnvPanel, + setShowGlobalVariablePanel, + setListeningTriggerType, + setListeningTriggerNodeId, + setListeningTriggerNodeIds, + setListeningTriggerIsAll, + } = workflowStore.getState() + + if (workflowRunningData?.result.status === WorkflowRunningStatus.Running) + return + + const { getNodes } = store.getState() + const nodes = getNodes() + const scheduleNode = nodes.find(node => node.id === nodeId && node.data.type === BlockEnum.TriggerSchedule) + + if (!scheduleNode) { + console.warn('handleWorkflowTriggerScheduleRunInWorkflow: schedule node not found', nodeId) + return + } + + setShowEnvPanel(false) + setShowGlobalVariablePanel(false) + + if (showDebugAndPreviewPanel) { + handleCancelDebugAndPreviewPanel() + return + } + + setListeningTriggerType(BlockEnum.TriggerSchedule) + setListeningTriggerNodeId(nodeId) + setListeningTriggerNodeIds([nodeId]) + setListeningTriggerIsAll(false) + + await doSyncWorkflowDraft() + handleRun( + {}, + undefined, + { + mode: TriggerType.Schedule, + scheduleNodeId: nodeId, + }, + ) + setShowDebugAndPreviewPanel(true) + setShowInputsPanel(false) + }, [store, workflowStore, handleCancelDebugAndPreviewPanel, handleRun, doSyncWorkflowDraft]) + + const handleWorkflowTriggerWebhookRunInWorkflow = useCallback(async ({ nodeId }: { nodeId: string }) => { + if (!nodeId) + return + + const { + workflowRunningData, + showDebugAndPreviewPanel, + setShowDebugAndPreviewPanel, + setShowInputsPanel, + setShowEnvPanel, + setShowGlobalVariablePanel, + setListeningTriggerType, + setListeningTriggerNodeId, + setListeningTriggerNodeIds, + setListeningTriggerIsAll, + } = workflowStore.getState() + + if (workflowRunningData?.result.status === WorkflowRunningStatus.Running) + return + + const { getNodes } = store.getState() + const nodes = getNodes() + const webhookNode = nodes.find(node => node.id === nodeId && node.data.type === BlockEnum.TriggerWebhook) + + if (!webhookNode) { + console.warn('handleWorkflowTriggerWebhookRunInWorkflow: webhook node not found', nodeId) + return + } + + setShowEnvPanel(false) + setShowGlobalVariablePanel(false) + + if (!showDebugAndPreviewPanel) + setShowDebugAndPreviewPanel(true) + + setShowInputsPanel(false) + setListeningTriggerType(BlockEnum.TriggerWebhook) + setListeningTriggerNodeId(nodeId) + setListeningTriggerNodeIds([nodeId]) + setListeningTriggerIsAll(false) + + await doSyncWorkflowDraft() + handleRun( + { node_id: nodeId }, + undefined, + { + mode: TriggerType.Webhook, + webhookNodeId: nodeId, + }, + ) + }, [store, workflowStore, handleRun, doSyncWorkflowDraft]) + + const handleWorkflowTriggerPluginRunInWorkflow = useCallback(async (nodeId?: string) => { + if (!nodeId) + return + const { + workflowRunningData, + showDebugAndPreviewPanel, + setShowDebugAndPreviewPanel, + setShowInputsPanel, + setShowEnvPanel, + setShowGlobalVariablePanel, + setListeningTriggerType, + setListeningTriggerNodeId, + setListeningTriggerNodeIds, + setListeningTriggerIsAll, + } = workflowStore.getState() + + if (workflowRunningData?.result.status === WorkflowRunningStatus.Running) + return + + const { getNodes } = store.getState() + const nodes = getNodes() + const pluginNode = nodes.find(node => node.id === nodeId && node.data.type === BlockEnum.TriggerPlugin) + + if (!pluginNode) { + console.warn('handleWorkflowTriggerPluginRunInWorkflow: plugin node not found', nodeId) + return + } + + setShowEnvPanel(false) + setShowGlobalVariablePanel(false) + + if (!showDebugAndPreviewPanel) + setShowDebugAndPreviewPanel(true) + + setShowInputsPanel(false) + setListeningTriggerType(BlockEnum.TriggerPlugin) + setListeningTriggerNodeId(nodeId) + setListeningTriggerNodeIds([nodeId]) + setListeningTriggerIsAll(false) + + await doSyncWorkflowDraft() + handleRun( + { node_id: nodeId }, + undefined, + { + mode: TriggerType.Plugin, + pluginNodeId: nodeId, + }, + ) + }, [store, workflowStore, handleRun, doSyncWorkflowDraft]) + + const handleWorkflowRunAllTriggersInWorkflow = useCallback(async (nodeIds: string[]) => { + if (!nodeIds.length) + return + const { + workflowRunningData, + showDebugAndPreviewPanel, + setShowDebugAndPreviewPanel, + setShowInputsPanel, + setShowEnvPanel, + setShowGlobalVariablePanel, + setListeningTriggerIsAll, + setListeningTriggerNodeIds, + setListeningTriggerNodeId, + } = workflowStore.getState() + + if (workflowRunningData?.result.status === WorkflowRunningStatus.Running) + return + + setShowEnvPanel(false) + setShowGlobalVariablePanel(false) + setShowInputsPanel(false) + setListeningTriggerIsAll(true) + setListeningTriggerNodeIds(nodeIds) + setListeningTriggerNodeId(null) + + if (!showDebugAndPreviewPanel) + setShowDebugAndPreviewPanel(true) + + await doSyncWorkflowDraft() + handleRun( + { node_ids: nodeIds }, + undefined, + { + mode: TriggerType.All, + allNodeIds: nodeIds, + }, + ) + }, [store, workflowStore, handleRun, doSyncWorkflowDraft]) + const handleWorkflowStartRunInChatflow = useCallback(async () => { const { showDebugAndPreviewPanel, @@ -68,10 +268,12 @@ export const useWorkflowStartRun = () => { setHistoryWorkflowData, setShowEnvPanel, setShowChatVariablePanel, + setShowGlobalVariablePanel, } = workflowStore.getState() setShowEnvPanel(false) setShowChatVariablePanel(false) + setShowGlobalVariablePanel(false) if (showDebugAndPreviewPanel) handleCancelDebugAndPreviewPanel() @@ -92,5 +294,9 @@ export const useWorkflowStartRun = () => { handleStartWorkflowRun, handleWorkflowStartRunInWorkflow, handleWorkflowStartRunInChatflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, } } diff --git a/web/app/components/workflow-app/index.tsx b/web/app/components/workflow-app/index.tsx index df83b3ca26..fcd247ef22 100644 --- a/web/app/components/workflow-app/index.tsx +++ b/web/app/components/workflow-app/index.tsx @@ -10,6 +10,10 @@ import { import { useWorkflowInit, } from './hooks/use-workflow-init' +import { useAppTriggers } from '@/service/use-tools' +import { useTriggerStatusStore } from '@/app/components/workflow/store/trigger-status' +import { useStore as useAppStore } from '@/app/components/app/store' +import { useWorkflowStore } from '@/app/components/workflow/store' import { initialEdges, initialNodes, @@ -24,13 +28,13 @@ import { WorkflowContextProvider, } from '@/app/components/workflow/context' import type { InjectWorkflowStoreSliceFn } from '@/app/components/workflow/store' -import { useWorkflowStore } from '@/app/components/workflow/store' import { createWorkflowSlice } from './store/workflow/workflow-slice' import WorkflowAppMain from './components/workflow-main' import { useSearchParams } from 'next/navigation' import { fetchRunDetail } from '@/service/log' import { useGetRunAndTraceUrl } from './hooks/use-get-run-and-trace-url' +import { AppModeEnum } from '@/types/app' const WorkflowAppWithAdditionalContext = () => { const { @@ -38,8 +42,46 @@ const WorkflowAppWithAdditionalContext = () => { isLoading, fileUploadConfigResponse, } = useWorkflowInit() + const workflowStore = useWorkflowStore() const { isLoadingCurrentWorkspace, currentWorkspace } = useAppContext() + // Initialize trigger status at application level + const { setTriggerStatuses } = useTriggerStatusStore() + const appDetail = useAppStore(s => s.appDetail) + const appId = appDetail?.id + const isWorkflowMode = appDetail?.mode === AppModeEnum.WORKFLOW + const { data: triggersResponse } = useAppTriggers(isWorkflowMode ? appId : undefined, { + staleTime: 5 * 60 * 1000, // 5 minutes cache + refetchOnWindowFocus: false, + }) + + // Sync trigger statuses to store when data loads + useEffect(() => { + if (triggersResponse?.data) { + // Map API status to EntryNodeStatus: 'enabled' stays 'enabled', all others become 'disabled' + const statusMap = triggersResponse.data.reduce((acc, trigger) => { + acc[trigger.node_id] = trigger.status === 'enabled' ? 'enabled' : 'disabled' + return acc + }, {} as Record<string, 'enabled' | 'disabled'>) + + setTriggerStatuses(statusMap) + } + }, [triggersResponse?.data, setTriggerStatuses]) + + // Cleanup on unmount + useEffect(() => { + return () => { + // Reset the loaded flag when component unmounts + workflowStore.setState({ isWorkflowDataLoaded: false }) + + // Cancel any pending debounced sync operations + const { debouncedSyncWorkflowDraft } = workflowStore.getState() + // The debounced function from lodash has a cancel method + if (debouncedSyncWorkflowDraft && 'cancel' in debouncedSyncWorkflowDraft) + (debouncedSyncWorkflowDraft as any).cancel() + } + }, [workflowStore]) + const nodesData = useMemo(() => { if (data) return initialNodes(data.graph.nodes, data.graph.edges) @@ -54,7 +96,6 @@ const WorkflowAppWithAdditionalContext = () => { }, [data]) const searchParams = useSearchParams() - const workflowStore = useWorkflowStore() const { getWorkflowRunAndTraceUrl } = useGetRunAndTraceUrl() const replayRunId = searchParams.get('replayRunId') diff --git a/web/app/components/workflow-app/store/workflow/workflow-slice.ts b/web/app/components/workflow-app/store/workflow/workflow-slice.ts index f26d9b509b..72230629f0 100644 --- a/web/app/components/workflow-app/store/workflow/workflow-slice.ts +++ b/web/app/components/workflow-app/store/workflow/workflow-slice.ts @@ -5,8 +5,16 @@ export type WorkflowSliceShape = { appName: string notInitialWorkflow: boolean setNotInitialWorkflow: (notInitialWorkflow: boolean) => void + shouldAutoOpenStartNodeSelector: boolean + setShouldAutoOpenStartNodeSelector: (shouldAutoOpen: boolean) => void nodesDefaultConfigs: Record<string, any> setNodesDefaultConfigs: (nodesDefaultConfigs: Record<string, any>) => void + showOnboarding: boolean + setShowOnboarding: (showOnboarding: boolean) => void + hasSelectedStartNode: boolean + setHasSelectedStartNode: (hasSelectedStartNode: boolean) => void + hasShownOnboarding: boolean + setHasShownOnboarding: (hasShownOnboarding: boolean) => void } export type CreateWorkflowSlice = StateCreator<WorkflowSliceShape> @@ -15,6 +23,14 @@ export const createWorkflowSlice: StateCreator<WorkflowSliceShape> = set => ({ appName: '', notInitialWorkflow: false, setNotInitialWorkflow: notInitialWorkflow => set(() => ({ notInitialWorkflow })), + shouldAutoOpenStartNodeSelector: false, + setShouldAutoOpenStartNodeSelector: shouldAutoOpenStartNodeSelector => set(() => ({ shouldAutoOpenStartNodeSelector })), nodesDefaultConfigs: {}, setNodesDefaultConfigs: nodesDefaultConfigs => set(() => ({ nodesDefaultConfigs })), + showOnboarding: false, + setShowOnboarding: showOnboarding => set(() => ({ showOnboarding })), + hasSelectedStartNode: false, + setHasSelectedStartNode: hasSelectedStartNode => set(() => ({ hasSelectedStartNode })), + hasShownOnboarding: false, + setHasShownOnboarding: hasShownOnboarding => set(() => ({ hasShownOnboarding })), }) diff --git a/web/app/components/workflow/__tests__/trigger-status-sync.test.tsx b/web/app/components/workflow/__tests__/trigger-status-sync.test.tsx new file mode 100644 index 0000000000..dc208047cb --- /dev/null +++ b/web/app/components/workflow/__tests__/trigger-status-sync.test.tsx @@ -0,0 +1,339 @@ +import React, { useCallback } from 'react' +import { act, render } from '@testing-library/react' +import { useTriggerStatusStore } from '../store/trigger-status' +import { isTriggerNode } from '../types' +import type { EntryNodeStatus } from '../store/trigger-status' + +// Mock the isTriggerNode function +jest.mock('../types', () => ({ + isTriggerNode: jest.fn(), +})) + +const mockIsTriggerNode = isTriggerNode as jest.MockedFunction<typeof isTriggerNode> + +// Test component that mimics BaseNode's usage pattern +const TestTriggerNode: React.FC<{ + nodeId: string + nodeType: string +}> = ({ nodeId, nodeType }) => { + const triggerStatus = useTriggerStatusStore(state => + mockIsTriggerNode(nodeType) ? (state.triggerStatuses[nodeId] || 'disabled') : 'enabled', + ) + + return ( + <div data-testid={`node-${nodeId}`} data-status={triggerStatus}> + Status: {triggerStatus} + </div> + ) +} + +// Test component that mimics TriggerCard's usage pattern +const TestTriggerController: React.FC = () => { + const { setTriggerStatus, setTriggerStatuses } = useTriggerStatusStore() + + const handleToggle = (nodeId: string, enabled: boolean) => { + const newStatus = enabled ? 'enabled' : 'disabled' + setTriggerStatus(nodeId, newStatus) + } + + const handleBatchUpdate = (statuses: Record<string, EntryNodeStatus>) => { + setTriggerStatuses(statuses) + } + + return ( + <div> + <button + data-testid="toggle-node-1" + onClick={() => handleToggle('node-1', true)} + > + Enable Node 1 + </button> + <button + data-testid="toggle-node-2" + onClick={() => handleToggle('node-2', false)} + > + Disable Node 2 + </button> + <button + data-testid="batch-update" + onClick={() => handleBatchUpdate({ + 'node-1': 'disabled', + 'node-2': 'enabled', + 'node-3': 'enabled', + })} + > + Batch Update + </button> + </div> + ) +} + +describe('Trigger Status Synchronization Integration', () => { + beforeEach(() => { + // Clear store state + act(() => { + const store = useTriggerStatusStore.getState() + store.clearTriggerStatuses() + }) + + // Reset mocks + jest.clearAllMocks() + }) + + describe('Real-time Status Synchronization', () => { + it('should sync status changes between trigger controller and nodes', () => { + mockIsTriggerNode.mockReturnValue(true) + + const { getByTestId } = render( + <> + <TestTriggerController /> + <TestTriggerNode nodeId="node-1" nodeType="trigger-webhook" /> + <TestTriggerNode nodeId="node-2" nodeType="trigger-schedule" /> + </>, + ) + + // Initial state - should be 'disabled' by default + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'disabled') + expect(getByTestId('node-node-2')).toHaveAttribute('data-status', 'disabled') + + // Enable node-1 + act(() => { + getByTestId('toggle-node-1').click() + }) + + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'enabled') + expect(getByTestId('node-node-2')).toHaveAttribute('data-status', 'disabled') + + // Disable node-2 (should remain disabled) + act(() => { + getByTestId('toggle-node-2').click() + }) + + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'enabled') + expect(getByTestId('node-node-2')).toHaveAttribute('data-status', 'disabled') + }) + + it('should handle batch status updates correctly', () => { + mockIsTriggerNode.mockReturnValue(true) + + const { getByTestId } = render( + <> + <TestTriggerController /> + <TestTriggerNode nodeId="node-1" nodeType="trigger-webhook" /> + <TestTriggerNode nodeId="node-2" nodeType="trigger-schedule" /> + <TestTriggerNode nodeId="node-3" nodeType="trigger-plugin" /> + </>, + ) + + // Initial state + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'disabled') + expect(getByTestId('node-node-2')).toHaveAttribute('data-status', 'disabled') + expect(getByTestId('node-node-3')).toHaveAttribute('data-status', 'disabled') + + // Batch update + act(() => { + getByTestId('batch-update').click() + }) + + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'disabled') + expect(getByTestId('node-node-2')).toHaveAttribute('data-status', 'enabled') + expect(getByTestId('node-node-3')).toHaveAttribute('data-status', 'enabled') + }) + + it('should handle mixed node types (trigger vs non-trigger)', () => { + // Mock different node types + mockIsTriggerNode.mockImplementation((nodeType: string) => { + return nodeType.startsWith('trigger-') + }) + + const { getByTestId } = render( + <> + <TestTriggerController /> + <TestTriggerNode nodeId="node-1" nodeType="trigger-webhook" /> + <TestTriggerNode nodeId="node-2" nodeType="start" /> + <TestTriggerNode nodeId="node-3" nodeType="llm" /> + </>, + ) + + // Trigger node should use store status, non-trigger nodes should be 'enabled' + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'disabled') // trigger node + expect(getByTestId('node-node-2')).toHaveAttribute('data-status', 'enabled') // start node + expect(getByTestId('node-node-3')).toHaveAttribute('data-status', 'enabled') // llm node + + // Update trigger node status + act(() => { + getByTestId('toggle-node-1').click() + }) + + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'enabled') // updated + expect(getByTestId('node-node-2')).toHaveAttribute('data-status', 'enabled') // unchanged + expect(getByTestId('node-node-3')).toHaveAttribute('data-status', 'enabled') // unchanged + }) + }) + + describe('Store State Management', () => { + it('should maintain state consistency across multiple components', () => { + mockIsTriggerNode.mockReturnValue(true) + + // Render multiple instances of the same node + const { getByTestId, rerender } = render( + <> + <TestTriggerController /> + <TestTriggerNode nodeId="shared-node" nodeType="trigger-webhook" /> + </>, + ) + + // Update status + act(() => { + getByTestId('toggle-node-1').click() // This updates node-1, not shared-node + }) + + // Add another component with the same nodeId + rerender( + <> + <TestTriggerController /> + <TestTriggerNode nodeId="shared-node" nodeType="trigger-webhook" /> + <TestTriggerNode nodeId="shared-node" nodeType="trigger-webhook" /> + </>, + ) + + // Both components should show the same status + const nodes = document.querySelectorAll('[data-testid="node-shared-node"]') + expect(nodes).toHaveLength(2) + nodes.forEach((node) => { + expect(node).toHaveAttribute('data-status', 'disabled') + }) + }) + + it('should handle rapid status changes correctly', () => { + mockIsTriggerNode.mockReturnValue(true) + + const { getByTestId } = render( + <> + <TestTriggerController /> + <TestTriggerNode nodeId="node-1" nodeType="trigger-webhook" /> + </>, + ) + + // Rapid consecutive updates + act(() => { + // Multiple rapid clicks + getByTestId('toggle-node-1').click() // enable + getByTestId('toggle-node-2').click() // disable (different node) + getByTestId('toggle-node-1').click() // enable again + }) + + // Should reflect the final state + expect(getByTestId('node-node-1')).toHaveAttribute('data-status', 'enabled') + }) + }) + + describe('Error Scenarios', () => { + it('should handle non-existent node IDs gracefully', () => { + mockIsTriggerNode.mockReturnValue(true) + + const { getByTestId } = render( + <TestTriggerNode nodeId="non-existent-node" nodeType="trigger-webhook" />, + ) + + // Should default to 'disabled' for non-existent nodes + expect(getByTestId('node-non-existent-node')).toHaveAttribute('data-status', 'disabled') + }) + + it('should handle component unmounting gracefully', () => { + mockIsTriggerNode.mockReturnValue(true) + + const { getByTestId, unmount } = render( + <> + <TestTriggerController /> + <TestTriggerNode nodeId="node-1" nodeType="trigger-webhook" /> + </>, + ) + + // Update status + act(() => { + getByTestId('toggle-node-1').click() + }) + + // Unmount components + expect(() => unmount()).not.toThrow() + + // Store should still maintain the state + const store = useTriggerStatusStore.getState() + expect(store.triggerStatuses['node-1']).toBe('enabled') + }) + }) + + describe('Performance Optimization', () => { + // Component that uses optimized selector with useCallback + const OptimizedTriggerNode: React.FC<{ + nodeId: string + nodeType: string + }> = ({ nodeId, nodeType }) => { + const triggerStatusSelector = useCallback((state: any) => + mockIsTriggerNode(nodeType) ? (state.triggerStatuses[nodeId] || 'disabled') : 'enabled', + [nodeId, nodeType], + ) + const triggerStatus = useTriggerStatusStore(triggerStatusSelector) + + return ( + <div data-testid={`optimized-node-${nodeId}`} data-status={triggerStatus}> + Status: {triggerStatus} + </div> + ) + } + + it('should work correctly with optimized selector using useCallback', () => { + mockIsTriggerNode.mockImplementation(nodeType => nodeType === 'trigger-webhook') + + const { getByTestId } = render( + <> + <OptimizedTriggerNode nodeId="node-1" nodeType="trigger-webhook" /> + <OptimizedTriggerNode nodeId="node-2" nodeType="start" /> + <TestTriggerController /> + </>, + ) + + // Initial state + expect(getByTestId('optimized-node-node-1')).toHaveAttribute('data-status', 'disabled') + expect(getByTestId('optimized-node-node-2')).toHaveAttribute('data-status', 'enabled') + + // Update status via controller + act(() => { + getByTestId('toggle-node-1').click() + }) + + // Verify optimized component updates correctly + expect(getByTestId('optimized-node-node-1')).toHaveAttribute('data-status', 'enabled') + expect(getByTestId('optimized-node-node-2')).toHaveAttribute('data-status', 'enabled') + }) + + it('should handle selector dependency changes correctly', () => { + mockIsTriggerNode.mockImplementation(nodeType => nodeType === 'trigger-webhook') + + const TestComponent: React.FC<{ nodeType: string }> = ({ nodeType }) => { + const triggerStatusSelector = useCallback((state: any) => + mockIsTriggerNode(nodeType) ? (state.triggerStatuses['test-node'] || 'disabled') : 'enabled', + ['test-node', nodeType], // Dependencies should match implementation + ) + const status = useTriggerStatusStore(triggerStatusSelector) + return <div data-testid="test-component" data-status={status} /> + } + + const { getByTestId, rerender } = render(<TestComponent nodeType="trigger-webhook" />) + + // Initial trigger node + expect(getByTestId('test-component')).toHaveAttribute('data-status', 'disabled') + + // Set status for the node + act(() => { + useTriggerStatusStore.getState().setTriggerStatus('test-node', 'enabled') + }) + expect(getByTestId('test-component')).toHaveAttribute('data-status', 'enabled') + + // Change node type to non-trigger - should return 'enabled' regardless of store + rerender(<TestComponent nodeType="start" />) + expect(getByTestId('test-component')).toHaveAttribute('data-status', 'enabled') + }) + }) +}) diff --git a/web/app/components/workflow/block-icon.tsx b/web/app/components/workflow/block-icon.tsx index 60fa813cd9..a4f53f2a64 100644 --- a/web/app/components/workflow/block-icon.tsx +++ b/web/app/components/workflow/block-icon.tsx @@ -21,8 +21,10 @@ import { LoopEnd, ParameterExtractor, QuestionClassifier, + Schedule, TemplatingTransform, VariableX, + WebhookLine, } from '@/app/components/base/icons/src/vender/workflow' import AppIcon from '@/app/components/base/app-icon' import cn from '@/utils/classnames' @@ -38,35 +40,45 @@ const ICON_CONTAINER_CLASSNAME_SIZE_MAP: Record<string, string> = { sm: 'w-5 h-5 rounded-md shadow-xs', md: 'w-6 h-6 rounded-lg shadow-md', } + +const DEFAULT_ICON_MAP: Record<BlockEnum, React.ComponentType<{ className: string }>> = { + [BlockEnum.Start]: Home, + [BlockEnum.LLM]: Llm, + [BlockEnum.Code]: Code, + [BlockEnum.End]: End, + [BlockEnum.IfElse]: IfElse, + [BlockEnum.HttpRequest]: Http, + [BlockEnum.Answer]: Answer, + [BlockEnum.KnowledgeRetrieval]: KnowledgeRetrieval, + [BlockEnum.QuestionClassifier]: QuestionClassifier, + [BlockEnum.TemplateTransform]: TemplatingTransform, + [BlockEnum.VariableAssigner]: VariableX, + [BlockEnum.VariableAggregator]: VariableX, + [BlockEnum.Assigner]: Assigner, + [BlockEnum.Tool]: VariableX, + [BlockEnum.IterationStart]: VariableX, + [BlockEnum.Iteration]: Iteration, + [BlockEnum.LoopStart]: VariableX, + [BlockEnum.Loop]: Loop, + [BlockEnum.LoopEnd]: LoopEnd, + [BlockEnum.ParameterExtractor]: ParameterExtractor, + [BlockEnum.DocExtractor]: DocsExtractor, + [BlockEnum.ListFilter]: ListFilter, + [BlockEnum.Agent]: Agent, + [BlockEnum.KnowledgeBase]: KnowledgeBase, + [BlockEnum.DataSource]: Datasource, + [BlockEnum.DataSourceEmpty]: () => null, + [BlockEnum.TriggerSchedule]: Schedule, + [BlockEnum.TriggerWebhook]: WebhookLine, + [BlockEnum.TriggerPlugin]: VariableX, +} + const getIcon = (type: BlockEnum, className: string) => { - return { - [BlockEnum.Start]: <Home className={className} />, - [BlockEnum.LLM]: <Llm className={className} />, - [BlockEnum.Code]: <Code className={className} />, - [BlockEnum.End]: <End className={className} />, - [BlockEnum.IfElse]: <IfElse className={className} />, - [BlockEnum.HttpRequest]: <Http className={className} />, - [BlockEnum.Answer]: <Answer className={className} />, - [BlockEnum.KnowledgeRetrieval]: <KnowledgeRetrieval className={className} />, - [BlockEnum.QuestionClassifier]: <QuestionClassifier className={className} />, - [BlockEnum.TemplateTransform]: <TemplatingTransform className={className} />, - [BlockEnum.VariableAssigner]: <VariableX className={className} />, - [BlockEnum.VariableAggregator]: <VariableX className={className} />, - [BlockEnum.Assigner]: <Assigner className={className} />, - [BlockEnum.Tool]: <VariableX className={className} />, - [BlockEnum.IterationStart]: <VariableX className={className} />, - [BlockEnum.Iteration]: <Iteration className={className} />, - [BlockEnum.LoopStart]: <VariableX className={className} />, - [BlockEnum.Loop]: <Loop className={className} />, - [BlockEnum.LoopEnd]: <LoopEnd className={className} />, - [BlockEnum.ParameterExtractor]: <ParameterExtractor className={className} />, - [BlockEnum.DocExtractor]: <DocsExtractor className={className} />, - [BlockEnum.ListFilter]: <ListFilter className={className} />, - [BlockEnum.Agent]: <Agent className={className} />, - [BlockEnum.KnowledgeBase]: <KnowledgeBase className={className} />, - [BlockEnum.DataSource]: <Datasource className={className} />, - [BlockEnum.DataSourceEmpty]: <></>, - }[type] + const DefaultIcon = DEFAULT_ICON_MAP[type] + if (!DefaultIcon) + return null + + return <DefaultIcon className={className} /> } const ICON_CONTAINER_BG_COLOR_MAP: Record<string, string> = { [BlockEnum.Start]: 'bg-util-colors-blue-brand-blue-brand-500', @@ -92,6 +104,9 @@ const ICON_CONTAINER_BG_COLOR_MAP: Record<string, string> = { [BlockEnum.Agent]: 'bg-util-colors-indigo-indigo-500', [BlockEnum.KnowledgeBase]: 'bg-util-colors-warning-warning-500', [BlockEnum.DataSource]: 'bg-components-icon-bg-midnight-solid', + [BlockEnum.TriggerSchedule]: 'bg-util-colors-violet-violet-500', + [BlockEnum.TriggerWebhook]: 'bg-util-colors-blue-blue-500', + [BlockEnum.TriggerPlugin]: 'bg-util-colors-blue-blue-500', } const BlockIcon: FC<BlockIconProps> = ({ type, @@ -99,8 +114,8 @@ const BlockIcon: FC<BlockIconProps> = ({ className, toolIcon, }) => { - const isToolOrDataSource = type === BlockEnum.Tool || type === BlockEnum.DataSource - const showDefaultIcon = !isToolOrDataSource || !toolIcon + const isToolOrDataSourceOrTriggerPlugin = type === BlockEnum.Tool || type === BlockEnum.DataSource || type === BlockEnum.TriggerPlugin + const showDefaultIcon = !isToolOrDataSourceOrTriggerPlugin || !toolIcon return ( <div className={ @@ -114,11 +129,15 @@ const BlockIcon: FC<BlockIconProps> = ({ > { showDefaultIcon && ( - getIcon(type, size === 'xs' ? 'w-3 h-3' : 'w-3.5 h-3.5') + getIcon(type, + (type === BlockEnum.TriggerSchedule || type === BlockEnum.TriggerWebhook) + ? (size === 'xs' ? 'w-4 h-4' : 'w-4.5 h-4.5') + : (size === 'xs' ? 'w-3 h-3' : 'w-3.5 h-3.5'), + ) ) } { - isToolOrDataSource && toolIcon && ( + !showDefaultIcon && ( <> { typeof toolIcon === 'string' diff --git a/web/app/components/workflow/block-selector/all-start-blocks.tsx b/web/app/components/workflow/block-selector/all-start-blocks.tsx new file mode 100644 index 0000000000..a089978bdd --- /dev/null +++ b/web/app/components/workflow/block-selector/all-start-blocks.tsx @@ -0,0 +1,179 @@ +'use client' +import { + useCallback, + useEffect, + useMemo, + useState, +} from 'react' +import { useTranslation } from 'react-i18next' +import type { BlockEnum, OnSelectBlock } from '../types' +import type { TriggerDefaultValue, TriggerWithProvider } from './types' +import StartBlocks from './start-blocks' +import TriggerPluginList from './trigger-plugin/list' +import { ENTRY_NODE_TYPES } from './constants' +import cn from '@/utils/classnames' +import Link from 'next/link' +import { RiArrowRightUpLine } from '@remixicon/react' +import { getMarketplaceUrl } from '@/utils/var' +import Button from '@/app/components/base/button' +import { SearchMenu } from '@/app/components/base/icons/src/vender/line/general' +import { BlockEnum as BlockEnumValue } from '../types' +import FeaturedTriggers from './featured-triggers' +import Divider from '@/app/components/base/divider' +import { useGlobalPublicStore } from '@/context/global-public-context' +import { useAllTriggerPlugins, useInvalidateAllTriggerPlugins } from '@/service/use-triggers' +import { useFeaturedTriggersRecommendations } from '@/service/use-plugins' + +const marketplaceFooterClassName = 'system-sm-medium z-10 flex h-8 flex-none cursor-pointer items-center rounded-b-lg border-[0.5px] border-t border-components-panel-border bg-components-panel-bg-blur px-4 py-1 text-text-accent-light-mode-only shadow-lg' + +type AllStartBlocksProps = { + className?: string + searchText: string + onSelect: (type: BlockEnum, trigger?: TriggerDefaultValue) => void + availableBlocksTypes?: BlockEnum[] + tags?: string[] + allowUserInputSelection?: boolean // Allow user input option even when trigger node already exists (e.g. when no Start node yet or changing node type). +} + +const AllStartBlocks = ({ + className, + searchText, + onSelect, + availableBlocksTypes, + tags = [], + allowUserInputSelection = false, +}: AllStartBlocksProps) => { + const { t } = useTranslation() + const [hasStartBlocksContent, setHasStartBlocksContent] = useState(false) + const [hasPluginContent, setHasPluginContent] = useState(false) + const { enable_marketplace } = useGlobalPublicStore(s => s.systemFeatures) + + const entryNodeTypes = availableBlocksTypes?.length + ? availableBlocksTypes + : ENTRY_NODE_TYPES + const enableTriggerPlugin = entryNodeTypes.includes(BlockEnumValue.TriggerPlugin) + const { data: triggerProviders = [] } = useAllTriggerPlugins(enableTriggerPlugin) + const providerMap = useMemo(() => { + const map = new Map<string, TriggerWithProvider>() + triggerProviders.forEach((provider) => { + const keys = [ + provider.plugin_id, + provider.plugin_unique_identifier, + provider.id, + ].filter(Boolean) as string[] + keys.forEach((key) => { + if (!map.has(key)) + map.set(key, provider) + }) + }) + return map + }, [triggerProviders]) + const invalidateTriggers = useInvalidateAllTriggerPlugins() + const trimmedSearchText = searchText.trim() + const hasSearchText = trimmedSearchText.length > 0 + const { + plugins: featuredPlugins = [], + isLoading: featuredLoading, + } = useFeaturedTriggersRecommendations(enableTriggerPlugin && enable_marketplace && !hasSearchText) + + const shouldShowFeatured = enableTriggerPlugin + && enable_marketplace + && !hasSearchText + + const handleStartBlocksContentChange = useCallback((hasContent: boolean) => { + setHasStartBlocksContent(hasContent) + }, []) + + const handlePluginContentChange = useCallback((hasContent: boolean) => { + setHasPluginContent(hasContent) + }, []) + + const hasAnyContent = hasStartBlocksContent || hasPluginContent || shouldShowFeatured + const shouldShowEmptyState = hasSearchText && !hasAnyContent + + useEffect(() => { + if (!enableTriggerPlugin && hasPluginContent) + setHasPluginContent(false) + }, [enableTriggerPlugin, hasPluginContent]) + + return ( + <div className={cn('min-w-[400px] max-w-[500px]', className)}> + <div className='flex max-h-[640px] flex-col'> + <div className='flex-1 overflow-y-auto'> + <div className={cn(shouldShowEmptyState && 'hidden')}> + {shouldShowFeatured && ( + <> + <FeaturedTriggers + plugins={featuredPlugins} + providerMap={providerMap} + onSelect={onSelect} + isLoading={featuredLoading} + onInstallSuccess={async () => { + invalidateTriggers() + }} + /> + <div className='px-3'> + <Divider className='!h-px' /> + </div> + </> + )} + <div className='px-3 pb-1 pt-2'> + <span className='system-xs-medium text-text-primary'>{t('workflow.tabs.allTriggers')}</span> + </div> + <StartBlocks + searchText={trimmedSearchText} + onSelect={onSelect as OnSelectBlock} + availableBlocksTypes={entryNodeTypes as unknown as BlockEnum[]} + hideUserInput={!allowUserInputSelection} + onContentStateChange={handleStartBlocksContentChange} + /> + + {enableTriggerPlugin && ( + <TriggerPluginList + onSelect={onSelect} + searchText={trimmedSearchText} + onContentStateChange={handlePluginContentChange} + tags={tags} + /> + )} + </div> + + {shouldShowEmptyState && ( + <div className='flex h-full flex-col items-center justify-center gap-3 py-12 text-center'> + <SearchMenu className='h-8 w-8 text-text-quaternary' /> + <div className='text-sm font-medium text-text-secondary'> + {t('workflow.tabs.noPluginsFound')} + </div> + <Link + href='https://github.com/langgenius/dify-plugins/issues/new?template=plugin_request.yaml' + target='_blank' + > + <Button + size='small' + variant='secondary-accent' + className='h-6 cursor-pointer px-3 text-xs' + > + {t('workflow.tabs.requestToCommunity')} + </Button> + </Link> + </div> + )} + </div> + + {!shouldShowEmptyState && ( + // Footer - Same as Tools tab marketplace footer + <Link + className={marketplaceFooterClassName} + href={getMarketplaceUrl('')} + target='_blank' + > + <span>{t('plugin.findMoreInMarketplace')}</span> + <RiArrowRightUpLine className='ml-0.5 h-3 w-3' /> + </Link> + )} + </div> + </div> + ) +} + +export default AllStartBlocks diff --git a/web/app/components/workflow/block-selector/all-tools.tsx b/web/app/components/workflow/block-selector/all-tools.tsx index 6a2e07a411..d330eb182b 100644 --- a/web/app/components/workflow/block-selector/all-tools.tsx +++ b/web/app/components/workflow/block-selector/all-tools.tsx @@ -1,16 +1,12 @@ import type { Dispatch, + RefObject, SetStateAction, } from 'react' -import { - useEffect, - useMemo, - useRef, - useState, -} from 'react' +import { useCallback, useEffect, useMemo, useRef, useState } from 'react' +import { useTranslation } from 'react-i18next' import type { BlockEnum, - OnSelectBlock, ToolWithProvider, } from '../types' import type { ToolDefaultValue, ToolValue } from './types' @@ -19,13 +15,24 @@ import Tools from './tools' import { useToolTabs } from './hooks' import ViewTypeSelect, { ViewType } from './view-type-select' import cn from '@/utils/classnames' -import { useGetLanguage } from '@/context/i18n' +import Button from '@/app/components/base/button' +import { SearchMenu } from '@/app/components/base/icons/src/vender/line/general' import type { ListRef } from '@/app/components/workflow/block-selector/market-place-plugin/list' import PluginList, { type ListProps } from '@/app/components/workflow/block-selector/market-place-plugin/list' -import { PluginType } from '../../plugins/types' +import type { Plugin } from '../../plugins/types' +import { PluginCategoryEnum } from '../../plugins/types' import { useMarketplacePlugins } from '../../plugins/marketplace/hooks' import { useGlobalPublicStore } from '@/context/global-public-context' -import RAGToolSuggestions from './rag-tool-suggestions' +import RAGToolRecommendations from './rag-tool-recommendations' +import FeaturedTools from './featured-tools' +import Link from 'next/link' +import Divider from '@/app/components/base/divider' +import { RiArrowRightUpLine } from '@remixicon/react' +import { getMarketplaceUrl } from '@/utils/var' +import { useGetLanguage } from '@/context/i18n' +import type { OnSelectBlock } from '@/app/components/workflow/types' + +const marketplaceFooterClassName = 'system-sm-medium z-10 flex h-8 flex-none cursor-pointer items-center rounded-b-lg border-[0.5px] border-t border-components-panel-border bg-components-panel-bg-blur px-4 py-1 text-text-accent-light-mode-only shadow-lg' type AllToolsProps = { className?: string @@ -36,13 +43,17 @@ type AllToolsProps = { customTools: ToolWithProvider[] workflowTools: ToolWithProvider[] mcpTools: ToolWithProvider[] - onSelect: OnSelectBlock + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void canNotSelectMultiple?: boolean onSelectMultiple?: (type: BlockEnum, tools: ToolDefaultValue[]) => void selectedTools?: ToolValue[] canChooseMCPTool?: boolean - onTagsChange: Dispatch<SetStateAction<string[]>> + onTagsChange?: Dispatch<SetStateAction<string[]>> isInRAGPipeline?: boolean + featuredPlugins?: Plugin[] + featuredLoading?: boolean + showFeatured?: boolean + onFeaturedInstallSuccess?: () => Promise<void> | void } const DEFAULT_TAGS: AllToolsProps['tags'] = [] @@ -63,15 +74,33 @@ const AllTools = ({ canChooseMCPTool, onTagsChange, isInRAGPipeline = false, + featuredPlugins = [], + featuredLoading = false, + showFeatured = false, + onFeaturedInstallSuccess, }: AllToolsProps) => { + const { t } = useTranslation() const language = useGetLanguage() const tabs = useToolTabs() const [activeTab, setActiveTab] = useState(ToolTypeEnum.All) const [activeView, setActiveView] = useState<ViewType>(ViewType.flat) - const hasFilter = searchText || tags.length > 0 + const trimmedSearchText = searchText.trim() + const hasSearchText = trimmedSearchText.length > 0 + const hasTags = tags.length > 0 + const hasFilter = hasSearchText || hasTags const isMatchingKeywords = (text: string, keywords: string) => { return text.toLowerCase().includes(keywords.toLowerCase()) } + const allProviders = useMemo(() => [...buildInTools, ...customTools, ...workflowTools, ...mcpTools], [buildInTools, customTools, workflowTools, mcpTools]) + const providerMap = useMemo(() => { + const map = new Map<string, ToolWithProvider>() + allProviders.forEach((provider) => { + const key = provider.plugin_id || provider.id + if (key) + map.set(key, provider) + }) + return map + }, [allProviders]) const tools = useMemo(() => { let mergedTools: ToolWithProvider[] = [] if (activeTab === ToolTypeEnum.All) @@ -85,15 +114,55 @@ const AllTools = ({ if (activeTab === ToolTypeEnum.MCP) mergedTools = mcpTools - if (!hasFilter) + const normalizedSearch = trimmedSearchText.toLowerCase() + const getLocalizedText = (text?: Record<string, string> | null) => { + if (!text) + return '' + + if (text[language]) + return text[language] + + if (text['en-US']) + return text['en-US'] + + const firstValue = Object.values(text).find(Boolean) + return firstValue || '' + } + + if (!hasFilter || !normalizedSearch) return mergedTools.filter(toolWithProvider => toolWithProvider.tools.length > 0) - return mergedTools.filter((toolWithProvider) => { - return isMatchingKeywords(toolWithProvider.name, searchText) || toolWithProvider.tools.some((tool) => { - return tool.label[language].toLowerCase().includes(searchText.toLowerCase()) || tool.name.toLowerCase().includes(searchText.toLowerCase()) + return mergedTools.reduce<ToolWithProvider[]>((acc, toolWithProvider) => { + const providerLabel = getLocalizedText(toolWithProvider.label) + const providerMatches = [ + toolWithProvider.name, + providerLabel, + ].some(text => isMatchingKeywords(text || '', normalizedSearch)) + + if (providerMatches) { + if (toolWithProvider.tools.length > 0) + acc.push(toolWithProvider) + return acc + } + + const matchedTools = toolWithProvider.tools.filter((tool) => { + const toolLabel = getLocalizedText(tool.label) + return [ + tool.name, + toolLabel, + ].some(text => isMatchingKeywords(text || '', normalizedSearch)) }) - }) - }, [activeTab, buildInTools, customTools, workflowTools, mcpTools, searchText, language, hasFilter]) + + if (matchedTools.length > 0) { + acc.push({ + ...toolWithProvider, + tools: matchedTools, + }) + } + + return acc + }, []) + }, [activeTab, buildInTools, customTools, workflowTools, mcpTools, trimmedSearchText, hasFilter, language]) const { queryPluginsWithDebounced: fetchPlugins, @@ -101,22 +170,38 @@ const AllTools = ({ } = useMarketplacePlugins() const { enable_marketplace } = useGlobalPublicStore(s => s.systemFeatures) + useEffect(() => { if (!enable_marketplace) return - if (searchText || tags.length > 0) { + if (hasFilter) { fetchPlugins({ query: searchText, tags, - category: PluginType.tool, + category: PluginCategoryEnum.tool, }) } - }, [searchText, tags, enable_marketplace]) + }, [searchText, tags, enable_marketplace, hasFilter, fetchPlugins]) const pluginRef = useRef<ListRef>(null) const wrapElemRef = useRef<HTMLDivElement>(null) const isSupportGroupView = [ToolTypeEnum.All, ToolTypeEnum.BuiltIn].includes(activeTab) const isShowRAGRecommendations = isInRAGPipeline && activeTab === ToolTypeEnum.All && !hasFilter + const hasToolsListContent = tools.length > 0 || isShowRAGRecommendations + const hasPluginContent = enable_marketplace && notInstalledPlugins.length > 0 + const shouldShowEmptyState = hasFilter && !hasToolsListContent && !hasPluginContent + const shouldShowFeatured = showFeatured + && enable_marketplace + && !isInRAGPipeline + && activeTab === ToolTypeEnum.All + && !hasFilter + const shouldShowMarketplaceFooter = enable_marketplace && !hasFilter + + const handleRAGSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { + if (!pluginDefaultValue) + return + onSelect(type, pluginDefaultValue as ToolDefaultValue) + }, [onSelect]) return ( <div className={cn('min-w-[400px] max-w-[500px]', className)}> @@ -142,41 +227,100 @@ const AllTools = ({ <ViewTypeSelect viewType={activeView} onChange={setActiveView} /> )} </div> - <div - ref={wrapElemRef} - className='max-h-[464px] overflow-y-auto' - onScroll={pluginRef.current?.handleScroll} - > - {isShowRAGRecommendations && ( - <RAGToolSuggestions - viewType={isSupportGroupView ? activeView : ViewType.flat} - onSelect={onSelect} - onTagsChange={onTagsChange} - /> - )} - <Tools - className={toolContentClassName} - tools={tools} - onSelect={onSelect} - canNotSelectMultiple={canNotSelectMultiple} - onSelectMultiple={onSelectMultiple} - toolType={activeTab} - viewType={isSupportGroupView ? activeView : ViewType.flat} - hasSearchText={!!searchText} - selectedTools={selectedTools} - canChooseMCPTool={canChooseMCPTool} - isShowRAGRecommendations={isShowRAGRecommendations} - /> - {/* Plugins from marketplace */} - {enable_marketplace && ( - <PluginList - ref={pluginRef} - wrapElemRef={wrapElemRef} - list={notInstalledPlugins} - searchText={searchText} - toolContentClassName={toolContentClassName} - tags={tags} - /> + <div className='flex max-h-[464px] flex-col'> + <div + ref={wrapElemRef} + className='flex-1 overflow-y-auto' + onScroll={pluginRef.current?.handleScroll} + > + <div className={cn(shouldShowEmptyState && 'hidden')}> + {isShowRAGRecommendations && onTagsChange && ( + <RAGToolRecommendations + viewType={isSupportGroupView ? activeView : ViewType.flat} + onSelect={handleRAGSelect} + onTagsChange={onTagsChange} + /> + )} + {shouldShowFeatured && ( + <> + <FeaturedTools + plugins={featuredPlugins} + providerMap={providerMap} + onSelect={onSelect} + selectedTools={selectedTools} + canChooseMCPTool={canChooseMCPTool} + isLoading={featuredLoading} + onInstallSuccess={async () => { + await onFeaturedInstallSuccess?.() + }} + /> + <div className='px-3'> + <Divider className='!h-px' /> + </div> + </> + )} + {hasToolsListContent && ( + <> + <div className='px-3 pb-1 pt-2'> + <span className='system-xs-medium text-text-primary'>{t('tools.allTools')}</span> + </div> + <Tools + className={toolContentClassName} + tools={tools} + onSelect={onSelect} + canNotSelectMultiple={canNotSelectMultiple} + onSelectMultiple={onSelectMultiple} + toolType={activeTab} + viewType={isSupportGroupView ? activeView : ViewType.flat} + hasSearchText={hasSearchText} + selectedTools={selectedTools} + canChooseMCPTool={canChooseMCPTool} + /> + </> + )} + {enable_marketplace && ( + <PluginList + ref={pluginRef} + wrapElemRef={wrapElemRef as RefObject<HTMLElement>} + list={notInstalledPlugins} + searchText={searchText} + toolContentClassName={toolContentClassName} + tags={tags} + hideFindMoreFooter + /> + )} + </div> + + {shouldShowEmptyState && ( + <div className='flex h-full flex-col items-center justify-center gap-3 py-12 text-center'> + <SearchMenu className='h-8 w-8 text-text-quaternary' /> + <div className='text-sm font-medium text-text-secondary'> + {t('workflow.tabs.noPluginsFound')} + </div> + <Link + href='https://github.com/langgenius/dify-plugins/issues/new?template=plugin_request.yaml' + target='_blank' + > + <Button + size='small' + variant='secondary-accent' + className='h-6 cursor-pointer px-3 text-xs' + > + {t('workflow.tabs.requestToCommunity')} + </Button> + </Link> + </div> + )} + </div> + {shouldShowMarketplaceFooter && ( + <Link + className={marketplaceFooterClassName} + href={getMarketplaceUrl('')} + target='_blank' + > + <span>{t('plugin.findMoreInMarketplace')}</span> + <RiArrowRightUpLine className='ml-0.5 h-3 w-3' /> + </Link> )} </div> </div> diff --git a/web/app/components/workflow/block-selector/blocks.tsx b/web/app/components/workflow/block-selector/blocks.tsx index 18bf55f3f9..cae1ec32a5 100644 --- a/web/app/components/workflow/block-selector/blocks.tsx +++ b/web/app/components/workflow/block-selector/blocks.tsx @@ -10,28 +10,50 @@ import BlockIcon from '../block-icon' import { BlockEnum } from '../types' import type { NodeDefault } from '../types' import { BLOCK_CLASSIFICATIONS } from './constants' -import type { ToolDefaultValue } from './types' +import { useBlocks } from './hooks' import Tooltip from '@/app/components/base/tooltip' import Badge from '@/app/components/base/badge' type BlocksProps = { searchText: string - onSelect: (type: BlockEnum, tool?: ToolDefaultValue) => void + onSelect: (type: BlockEnum) => void availableBlocksTypes?: BlockEnum[] - blocks: NodeDefault[] + blocks?: NodeDefault[] } const Blocks = ({ searchText, onSelect, availableBlocksTypes = [], - blocks, + blocks: blocksFromProps, }: BlocksProps) => { const { t } = useTranslation() const store = useStoreApi() + const blocksFromHooks = useBlocks() + + // Use external blocks if provided, otherwise fallback to hook-based blocks + const blocks = blocksFromProps || blocksFromHooks.map(block => ({ + metaData: { + classification: block.classification, + sort: 0, // Default sort order + type: block.type, + title: block.title, + author: 'Dify', + description: block.description, + }, + defaultValue: {}, + checkValid: () => ({ isValid: true }), + }) as NodeDefault) const groups = useMemo(() => { return BLOCK_CLASSIFICATIONS.reduce((acc, classification) => { - const list = groupBy(blocks, 'metaData.classification')[classification].filter((block) => { + const grouped = groupBy(blocks, 'metaData.classification') + const list = (grouped[classification] || []).filter((block) => { + // Filter out trigger types from Blocks tab + if (block.metaData.type === BlockEnum.TriggerWebhook + || block.metaData.type === BlockEnum.TriggerSchedule + || block.metaData.type === BlockEnum.TriggerPlugin) + return false + return block.metaData.title.toLowerCase().includes(searchText.toLowerCase()) && availableBlocksTypes.includes(block.metaData.type) }) @@ -44,7 +66,7 @@ const Blocks = ({ const isEmpty = Object.values(groups).every(list => !list.length) const renderGroup = useCallback((classification: string) => { - const list = groups[classification].sort((a, b) => a.metaData.sort - b.metaData.sort) + const list = groups[classification].sort((a, b) => (a.metaData.sort || 0) - (b.metaData.sort || 0)) const { getNodes } = store.getState() const nodes = getNodes() const hasKnowledgeBaseNode = nodes.some(node => node.data.type === BlockEnum.KnowledgeBase) @@ -71,7 +93,7 @@ const Blocks = ({ <Tooltip key={block.metaData.type} position='right' - popupClassName='w-[200px]' + popupClassName='w-[200px] rounded-xl' needsDelay={false} popupContent={( <div> @@ -112,7 +134,7 @@ const Blocks = ({ }, [groups, onSelect, t, store]) return ( - <div className='max-h-[480px] overflow-y-auto p-1'> + <div className='max-h-[480px] min-w-[400px] max-w-[500px] overflow-y-auto p-1'> { isEmpty && ( <div className='flex h-[22px] items-center px-3 text-xs font-medium text-text-tertiary'>{t('workflow.tabs.noResult')}</div> diff --git a/web/app/components/workflow/block-selector/constants.tsx b/web/app/components/workflow/block-selector/constants.tsx index ab0c9586dc..ec05985453 100644 --- a/web/app/components/workflow/block-selector/constants.tsx +++ b/web/app/components/workflow/block-selector/constants.tsx @@ -1,3 +1,5 @@ +import type { Block } from '../types' +import { BlockEnum } from '../types' import { BlockClassificationEnum } from './types' export const BLOCK_CLASSIFICATIONS: string[] = [ @@ -29,3 +31,125 @@ export const DEFAULT_FILE_EXTENSIONS_IN_LOCAL_FILE_DATA_SOURCE = [ 'ppt', 'md', ] + +export const START_BLOCKS: Block[] = [ + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.Start, + title: 'User Input', + description: 'Traditional start node for user input', + }, + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.TriggerSchedule, + title: 'Schedule Trigger', + description: 'Time-based workflow trigger', + }, + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.TriggerWebhook, + title: 'Webhook Trigger', + description: 'HTTP callback trigger', + }, +] + +export const ENTRY_NODE_TYPES = [ + BlockEnum.Start, + BlockEnum.TriggerSchedule, + BlockEnum.TriggerWebhook, + BlockEnum.TriggerPlugin, +] as const + +export const BLOCKS: Block[] = [ + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.LLM, + title: 'LLM', + }, + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.KnowledgeRetrieval, + title: 'Knowledge Retrieval', + }, + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.End, + title: 'End', + }, + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.Answer, + title: 'Direct Answer', + }, + { + classification: BlockClassificationEnum.QuestionUnderstand, + type: BlockEnum.QuestionClassifier, + title: 'Question Classifier', + }, + { + classification: BlockClassificationEnum.Logic, + type: BlockEnum.IfElse, + title: 'IF/ELSE', + }, + { + classification: BlockClassificationEnum.Logic, + type: BlockEnum.LoopEnd, + title: 'Exit Loop', + description: '', + }, + { + classification: BlockClassificationEnum.Logic, + type: BlockEnum.Iteration, + title: 'Iteration', + }, + { + classification: BlockClassificationEnum.Logic, + type: BlockEnum.Loop, + title: 'Loop', + }, + { + classification: BlockClassificationEnum.Transform, + type: BlockEnum.Code, + title: 'Code', + }, + { + classification: BlockClassificationEnum.Transform, + type: BlockEnum.TemplateTransform, + title: 'Templating Transform', + }, + { + classification: BlockClassificationEnum.Transform, + type: BlockEnum.VariableAggregator, + title: 'Variable Aggregator', + }, + { + classification: BlockClassificationEnum.Transform, + type: BlockEnum.DocExtractor, + title: 'Doc Extractor', + }, + { + classification: BlockClassificationEnum.Transform, + type: BlockEnum.Assigner, + title: 'Variable Assigner', + }, + { + classification: BlockClassificationEnum.Transform, + type: BlockEnum.ParameterExtractor, + title: 'Parameter Extractor', + }, + { + classification: BlockClassificationEnum.Utilities, + type: BlockEnum.HttpRequest, + title: 'HTTP Request', + }, + { + classification: BlockClassificationEnum.Utilities, + type: BlockEnum.ListFilter, + title: 'List Filter', + }, + { + classification: BlockClassificationEnum.Default, + type: BlockEnum.Agent, + title: 'Agent', + }, +] diff --git a/web/app/components/workflow/block-selector/data-sources.tsx b/web/app/components/workflow/block-selector/data-sources.tsx index 441ede2334..b98a52dcff 100644 --- a/web/app/components/workflow/block-selector/data-sources.tsx +++ b/web/app/components/workflow/block-selector/data-sources.tsx @@ -17,7 +17,7 @@ import PluginList, { type ListRef } from '@/app/components/workflow/block-select import { useGlobalPublicStore } from '@/context/global-public-context' import { DEFAULT_FILE_EXTENSIONS_IN_LOCAL_FILE_DATA_SOURCE } from './constants' import { useMarketplacePlugins } from '../../plugins/marketplace/hooks' -import { PluginType } from '../../plugins/types' +import { PluginCategoryEnum } from '../../plugins/types' import { useGetLanguage } from '@/context/i18n' type AllToolsProps = { @@ -55,7 +55,7 @@ const DataSources = ({ }) }, [searchText, dataSources, language]) - const handleSelect = useCallback((_: any, toolDefaultValue: ToolDefaultValue) => { + const handleSelect = useCallback((_: BlockEnum, toolDefaultValue: ToolDefaultValue) => { let defaultValue: DataSourceDefaultValue = { plugin_id: toolDefaultValue?.provider_id, provider_type: toolDefaultValue?.provider_type, @@ -63,6 +63,7 @@ const DataSources = ({ datasource_name: toolDefaultValue?.tool_name, datasource_label: toolDefaultValue?.tool_label, title: toolDefaultValue?.title, + plugin_unique_identifier: toolDefaultValue?.plugin_unique_identifier, } // Update defaultValue with fileExtensions if this is the local file data source if (toolDefaultValue?.provider_id === 'langgenius/file' && toolDefaultValue?.provider_name === 'file') { @@ -86,16 +87,16 @@ const DataSources = ({ if (searchText) { fetchPlugins({ query: searchText, - category: PluginType.datasource, + category: PluginCategoryEnum.datasource, }) } }, [searchText, enable_marketplace]) return ( - <div className={cn(className)}> + <div className={cn('w-[400px] min-w-0 max-w-full', className)}> <div ref={wrapElemRef} - className='max-h-[464px] overflow-y-auto' + className='max-h-[464px] overflow-y-auto overflow-x-hidden' onScroll={pluginRef.current?.handleScroll} > <Tools diff --git a/web/app/components/workflow/block-selector/featured-tools.tsx b/web/app/components/workflow/block-selector/featured-tools.tsx new file mode 100644 index 0000000000..fe5c561362 --- /dev/null +++ b/web/app/components/workflow/block-selector/featured-tools.tsx @@ -0,0 +1,333 @@ +'use client' +import { useEffect, useMemo, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { BlockEnum, type ToolWithProvider } from '../types' +import type { ToolDefaultValue, ToolValue } from './types' +import type { Plugin } from '@/app/components/plugins/types' +import { useGetLanguage } from '@/context/i18n' +import BlockIcon from '../block-icon' +import Tooltip from '@/app/components/base/tooltip' +import { RiMoreLine } from '@remixicon/react' +import Loading from '@/app/components/base/loading' +import Link from 'next/link' +import { getMarketplaceUrl } from '@/utils/var' +import { ToolTypeEnum } from './types' +import { ViewType } from './view-type-select' +import Tools from './tools' +import { formatNumber } from '@/utils/format' +import Action from '@/app/components/workflow/block-selector/market-place-plugin/action' +import { ArrowDownDoubleLine, ArrowDownRoundFill, ArrowUpDoubleLine } from '@/app/components/base/icons/src/vender/solid/arrows' +import InstallFromMarketplace from '@/app/components/plugins/install-plugin/install-from-marketplace' + +const MAX_RECOMMENDED_COUNT = 15 +const INITIAL_VISIBLE_COUNT = 5 + +type FeaturedToolsProps = { + plugins: Plugin[] + providerMap: Map<string, ToolWithProvider> + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void + selectedTools?: ToolValue[] + canChooseMCPTool?: boolean + isLoading?: boolean + onInstallSuccess?: () => void +} + +const STORAGE_KEY = 'workflow_tools_featured_collapsed' + +const FeaturedTools = ({ + plugins, + providerMap, + onSelect, + selectedTools, + canChooseMCPTool, + isLoading = false, + onInstallSuccess, +}: FeaturedToolsProps) => { + const { t } = useTranslation() + const language = useGetLanguage() + const [visibleCount, setVisibleCount] = useState(INITIAL_VISIBLE_COUNT) + const [isCollapsed, setIsCollapsed] = useState<boolean>(() => { + if (typeof window === 'undefined') + return false + const stored = window.localStorage.getItem(STORAGE_KEY) + return stored === 'true' + }) + + useEffect(() => { + if (typeof window === 'undefined') + return + const stored = window.localStorage.getItem(STORAGE_KEY) + if (stored !== null) + setIsCollapsed(stored === 'true') + }, []) + + useEffect(() => { + if (typeof window === 'undefined') + return + window.localStorage.setItem(STORAGE_KEY, String(isCollapsed)) + }, [isCollapsed]) + + useEffect(() => { + setVisibleCount(INITIAL_VISIBLE_COUNT) + }, [plugins]) + + const limitedPlugins = useMemo( + () => plugins.slice(0, MAX_RECOMMENDED_COUNT), + [plugins], + ) + + const { + installedProviders, + uninstalledPlugins, + } = useMemo(() => { + const installed: ToolWithProvider[] = [] + const uninstalled: Plugin[] = [] + const visitedProviderIds = new Set<string>() + + limitedPlugins.forEach((plugin) => { + const provider = providerMap.get(plugin.plugin_id) + if (provider) { + if (!visitedProviderIds.has(provider.id)) { + installed.push(provider) + visitedProviderIds.add(provider.id) + } + } + else { + uninstalled.push(plugin) + } + }) + + return { + installedProviders: installed, + uninstalledPlugins: uninstalled, + } + }, [limitedPlugins, providerMap]) + + const totalQuota = Math.min(visibleCount, MAX_RECOMMENDED_COUNT) + + const visibleInstalledProviders = useMemo( + () => installedProviders.slice(0, totalQuota), + [installedProviders, totalQuota], + ) + + const remainingSlots = Math.max(totalQuota - visibleInstalledProviders.length, 0) + + const visibleUninstalledPlugins = useMemo( + () => (remainingSlots > 0 ? uninstalledPlugins.slice(0, remainingSlots) : []), + [uninstalledPlugins, remainingSlots], + ) + + const totalVisible = visibleInstalledProviders.length + visibleUninstalledPlugins.length + const maxAvailable = Math.min(MAX_RECOMMENDED_COUNT, installedProviders.length + uninstalledPlugins.length) + const hasMoreToShow = totalVisible < maxAvailable + const canToggleVisibility = maxAvailable > INITIAL_VISIBLE_COUNT + const isExpanded = canToggleVisibility && !hasMoreToShow + const showEmptyState = !isLoading && totalVisible === 0 + + return ( + <div className='px-3 pb-3 pt-2'> + <button + type='button' + className='flex w-full items-center rounded-md px-0 py-1 text-left text-text-primary' + onClick={() => setIsCollapsed(prev => !prev)} + > + <span className='system-xs-medium text-text-primary'>{t('workflow.tabs.featuredTools')}</span> + <ArrowDownRoundFill className={`ml-0.5 h-4 w-4 text-text-tertiary transition-transform ${isCollapsed ? '-rotate-90' : 'rotate-0'}`} /> + </button> + + {!isCollapsed && ( + <> + {isLoading && ( + <div className='py-3'> + <Loading type='app' /> + </div> + )} + + {showEmptyState && ( + <p className='system-xs-regular py-2 text-text-tertiary'> + <Link className='text-text-accent' href={getMarketplaceUrl('', { category: 'tool' })} target='_blank' rel='noopener noreferrer'> + {t('workflow.tabs.noFeaturedPlugins')} + </Link> + </p> + )} + + {!showEmptyState && !isLoading && ( + <> + {visibleInstalledProviders.length > 0 && ( + <Tools + className='p-0' + tools={visibleInstalledProviders} + onSelect={onSelect} + canNotSelectMultiple + toolType={ToolTypeEnum.All} + viewType={ViewType.flat} + hasSearchText={false} + selectedTools={selectedTools} + canChooseMCPTool={canChooseMCPTool} + /> + )} + + {visibleUninstalledPlugins.length > 0 && ( + <div className='mt-1 flex flex-col gap-1'> + {visibleUninstalledPlugins.map(plugin => ( + <FeaturedToolUninstalledItem + key={plugin.plugin_id} + plugin={plugin} + language={language} + onInstallSuccess={async () => { + await onInstallSuccess?.() + }} + t={t} + /> + ))} + </div> + )} + </> + )} + + {!isLoading && totalVisible > 0 && canToggleVisibility && ( + <div + className='group mt-1 flex cursor-pointer items-center gap-x-2 rounded-lg py-1 pl-3 pr-2 text-text-tertiary transition-colors hover:bg-state-base-hover hover:text-text-secondary' + onClick={() => { + setVisibleCount((count) => { + if (count >= maxAvailable) + return INITIAL_VISIBLE_COUNT + + return Math.min(count + INITIAL_VISIBLE_COUNT, maxAvailable) + }) + }} + > + <div className='flex items-center px-1 text-text-tertiary transition-colors group-hover:text-text-secondary'> + <RiMoreLine className='size-4 group-hover:hidden' /> + {isExpanded ? ( + <ArrowUpDoubleLine className='hidden size-4 group-hover:block' /> + ) : ( + <ArrowDownDoubleLine className='hidden size-4 group-hover:block' /> + )} + </div> + <div className='system-xs-regular'> + {t(isExpanded ? 'workflow.tabs.showLessFeatured' : 'workflow.tabs.showMoreFeatured')} + </div> + </div> + )} + </> + )} + </div> + ) +} + +type FeaturedToolUninstalledItemProps = { + plugin: Plugin + language: string + onInstallSuccess?: () => Promise<void> | void + t: (key: string, options?: Record<string, any>) => string +} + +function FeaturedToolUninstalledItem({ + plugin, + language, + onInstallSuccess, + t, +}: FeaturedToolUninstalledItemProps) { + const label = plugin.label?.[language] || plugin.name + const description = typeof plugin.brief === 'object' ? plugin.brief[language] : plugin.brief + const installCountLabel = t('plugin.install', { num: formatNumber(plugin.install_count || 0) }) + const [actionOpen, setActionOpen] = useState(false) + const [isActionHovered, setIsActionHovered] = useState(false) + const [isInstallModalOpen, setIsInstallModalOpen] = useState(false) + + useEffect(() => { + if (!actionOpen) + return + + const handleScroll = () => { + setActionOpen(false) + setIsActionHovered(false) + } + + window.addEventListener('scroll', handleScroll, true) + + return () => { + window.removeEventListener('scroll', handleScroll, true) + } + }, [actionOpen]) + + return ( + <> + <Tooltip + position='right' + needsDelay={false} + popupClassName='!p-0 !px-3 !py-2.5 !w-[224px] !leading-[18px] !text-xs !text-gray-700 !border-[0.5px] !border-black/5 !rounded-xl !shadow-lg' + popupContent={( + <div> + <BlockIcon size='md' className='mb-2' type={BlockEnum.Tool} toolIcon={plugin.icon} /> + <div className='mb-1 text-sm leading-5 text-text-primary'>{label}</div> + <div className='text-xs leading-[18px] text-text-secondary'>{description}</div> + </div> + )} + disabled={!description || isActionHovered || actionOpen || isInstallModalOpen} + > + <div + className='group flex h-8 w-full items-center rounded-lg pl-3 pr-1 hover:bg-state-base-hover' + > + <div className='flex h-full min-w-0 items-center'> + <BlockIcon type={BlockEnum.Tool} toolIcon={plugin.icon} /> + <div className='ml-2 min-w-0'> + <div className='system-sm-medium truncate text-text-secondary'>{label}</div> + </div> + </div> + <div className='ml-auto flex h-full items-center gap-1 pl-1'> + <span className={`system-xs-regular text-text-tertiary ${actionOpen ? 'hidden' : 'group-hover:hidden'}`}>{installCountLabel}</span> + <div + className={`system-xs-medium flex h-full items-center gap-1 text-components-button-secondary-accent-text [&_.action-btn]:h-6 [&_.action-btn]:min-h-0 [&_.action-btn]:w-6 [&_.action-btn]:rounded-lg [&_.action-btn]:p-0 ${actionOpen ? 'flex' : 'hidden group-hover:flex'}`} + onMouseEnter={() => setIsActionHovered(true)} + onMouseLeave={() => { + if (!actionOpen) + setIsActionHovered(false) + }} + > + <button + type='button' + className='cursor-pointer rounded-md px-1.5 py-0.5 hover:bg-state-base-hover' + onClick={() => { + setActionOpen(false) + setIsInstallModalOpen(true) + setIsActionHovered(true) + }} + > + {t('plugin.installAction')} + </button> + <Action + open={actionOpen} + onOpenChange={(value) => { + setActionOpen(value) + setIsActionHovered(value) + }} + author={plugin.org} + name={plugin.name} + version={plugin.latest_version} + /> + </div> + </div> + </div> + </Tooltip> + {isInstallModalOpen && ( + <InstallFromMarketplace + uniqueIdentifier={plugin.latest_package_identifier} + manifest={plugin} + onSuccess={async () => { + setIsInstallModalOpen(false) + setIsActionHovered(false) + await onInstallSuccess?.() + }} + onClose={() => { + setIsInstallModalOpen(false) + setIsActionHovered(false) + }} + /> + )} + </> + ) +} + +export default FeaturedTools diff --git a/web/app/components/workflow/block-selector/featured-triggers.tsx b/web/app/components/workflow/block-selector/featured-triggers.tsx new file mode 100644 index 0000000000..561ebc1784 --- /dev/null +++ b/web/app/components/workflow/block-selector/featured-triggers.tsx @@ -0,0 +1,326 @@ +'use client' +import { useEffect, useMemo, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { BlockEnum } from '../types' +import type { TriggerDefaultValue, TriggerWithProvider } from './types' +import type { Plugin } from '@/app/components/plugins/types' +import { useGetLanguage } from '@/context/i18n' +import BlockIcon from '../block-icon' +import Tooltip from '@/app/components/base/tooltip' +import { RiMoreLine } from '@remixicon/react' +import Loading from '@/app/components/base/loading' +import Link from 'next/link' +import { getMarketplaceUrl } from '@/utils/var' +import TriggerPluginItem from './trigger-plugin/item' +import { formatNumber } from '@/utils/format' +import Action from '@/app/components/workflow/block-selector/market-place-plugin/action' +import { ArrowDownDoubleLine, ArrowDownRoundFill, ArrowUpDoubleLine } from '@/app/components/base/icons/src/vender/solid/arrows' +import InstallFromMarketplace from '@/app/components/plugins/install-plugin/install-from-marketplace' + +const MAX_RECOMMENDED_COUNT = 15 +const INITIAL_VISIBLE_COUNT = 5 + +type FeaturedTriggersProps = { + plugins: Plugin[] + providerMap: Map<string, TriggerWithProvider> + onSelect: (type: BlockEnum, trigger?: TriggerDefaultValue) => void + isLoading?: boolean + onInstallSuccess?: () => void | Promise<void> +} + +const STORAGE_KEY = 'workflow_triggers_featured_collapsed' + +const FeaturedTriggers = ({ + plugins, + providerMap, + onSelect, + isLoading = false, + onInstallSuccess, +}: FeaturedTriggersProps) => { + const { t } = useTranslation() + const language = useGetLanguage() + const [visibleCount, setVisibleCount] = useState(INITIAL_VISIBLE_COUNT) + const [isCollapsed, setIsCollapsed] = useState<boolean>(() => { + if (typeof window === 'undefined') + return false + const stored = window.localStorage.getItem(STORAGE_KEY) + return stored === 'true' + }) + + useEffect(() => { + if (typeof window === 'undefined') + return + const stored = window.localStorage.getItem(STORAGE_KEY) + if (stored !== null) + setIsCollapsed(stored === 'true') + }, []) + + useEffect(() => { + if (typeof window === 'undefined') + return + window.localStorage.setItem(STORAGE_KEY, String(isCollapsed)) + }, [isCollapsed]) + + useEffect(() => { + setVisibleCount(INITIAL_VISIBLE_COUNT) + }, [plugins]) + + const limitedPlugins = useMemo( + () => plugins.slice(0, MAX_RECOMMENDED_COUNT), + [plugins], + ) + + const { + installedProviders, + uninstalledPlugins, + } = useMemo(() => { + const installed: TriggerWithProvider[] = [] + const uninstalled: Plugin[] = [] + const visitedProviderIds = new Set<string>() + + limitedPlugins.forEach((plugin) => { + const provider = providerMap.get(plugin.plugin_id) || providerMap.get(plugin.latest_package_identifier) + if (provider) { + if (!visitedProviderIds.has(provider.id)) { + installed.push(provider) + visitedProviderIds.add(provider.id) + } + } + else { + uninstalled.push(plugin) + } + }) + + return { + installedProviders: installed, + uninstalledPlugins: uninstalled, + } + }, [limitedPlugins, providerMap]) + + const totalQuota = Math.min(visibleCount, MAX_RECOMMENDED_COUNT) + + const visibleInstalledProviders = useMemo( + () => installedProviders.slice(0, totalQuota), + [installedProviders, totalQuota], + ) + + const remainingSlots = Math.max(totalQuota - visibleInstalledProviders.length, 0) + + const visibleUninstalledPlugins = useMemo( + () => (remainingSlots > 0 ? uninstalledPlugins.slice(0, remainingSlots) : []), + [uninstalledPlugins, remainingSlots], + ) + + const totalVisible = visibleInstalledProviders.length + visibleUninstalledPlugins.length + const maxAvailable = Math.min(MAX_RECOMMENDED_COUNT, installedProviders.length + uninstalledPlugins.length) + const hasMoreToShow = totalVisible < maxAvailable + const canToggleVisibility = maxAvailable > INITIAL_VISIBLE_COUNT + const isExpanded = canToggleVisibility && !hasMoreToShow + const showEmptyState = !isLoading && totalVisible === 0 + + return ( + <div className='px-3 pb-3 pt-2'> + <button + type='button' + className='flex w-full items-center rounded-md px-0 py-1 text-left text-text-primary' + onClick={() => setIsCollapsed(prev => !prev)} + > + <span className='system-xs-medium text-text-primary'>{t('workflow.tabs.featuredTools')}</span> + <ArrowDownRoundFill className={`ml-0.5 h-4 w-4 text-text-tertiary transition-transform ${isCollapsed ? '-rotate-90' : 'rotate-0'}`} /> + </button> + + {!isCollapsed && ( + <> + {isLoading && ( + <div className='py-3'> + <Loading type='app' /> + </div> + )} + + {showEmptyState && ( + <p className='system-xs-regular py-2 text-text-tertiary'> + <Link className='text-text-accent' href={getMarketplaceUrl('', { category: 'trigger' })} target='_blank' rel='noopener noreferrer'> + {t('workflow.tabs.noFeaturedTriggers')} + </Link> + </p> + )} + + {!showEmptyState && !isLoading && ( + <> + {visibleInstalledProviders.length > 0 && ( + <div className='mt-1'> + {visibleInstalledProviders.map(provider => ( + <TriggerPluginItem + key={provider.id} + payload={provider} + hasSearchText={false} + onSelect={onSelect} + /> + ))} + </div> + )} + + {visibleUninstalledPlugins.length > 0 && ( + <div className='mt-1 flex flex-col gap-1'> + {visibleUninstalledPlugins.map(plugin => ( + <FeaturedTriggerUninstalledItem + key={plugin.plugin_id} + plugin={plugin} + language={language} + onInstallSuccess={async () => { + await onInstallSuccess?.() + }} + t={t} + /> + ))} + </div> + )} + </> + )} + + {!isLoading && totalVisible > 0 && canToggleVisibility && ( + <div + className='group mt-1 flex cursor-pointer items-center gap-x-2 rounded-lg py-1 pl-3 pr-2 text-text-tertiary transition-colors hover:bg-state-base-hover hover:text-text-secondary' + onClick={() => { + setVisibleCount((count) => { + if (count >= maxAvailable) + return INITIAL_VISIBLE_COUNT + + return Math.min(count + INITIAL_VISIBLE_COUNT, maxAvailable) + }) + }} + > + <div className='flex items-center px-1 text-text-tertiary transition-colors group-hover:text-text-secondary'> + <RiMoreLine className='size-4 group-hover:hidden' /> + {isExpanded ? ( + <ArrowUpDoubleLine className='hidden size-4 group-hover:block' /> + ) : ( + <ArrowDownDoubleLine className='hidden size-4 group-hover:block' /> + )} + </div> + <div className='system-xs-regular'> + {t(isExpanded ? 'workflow.tabs.showLessFeatured' : 'workflow.tabs.showMoreFeatured')} + </div> + </div> + )} + </> + )} + </div> + ) +} + +type FeaturedTriggerUninstalledItemProps = { + plugin: Plugin + language: string + onInstallSuccess?: () => Promise<void> | void + t: (key: string, options?: Record<string, any>) => string +} + +function FeaturedTriggerUninstalledItem({ + plugin, + language, + onInstallSuccess, + t, +}: FeaturedTriggerUninstalledItemProps) { + const label = plugin.label?.[language] || plugin.name + const description = typeof plugin.brief === 'object' ? plugin.brief[language] : plugin.brief + const installCountLabel = t('plugin.install', { num: formatNumber(plugin.install_count || 0) }) + const [actionOpen, setActionOpen] = useState(false) + const [isActionHovered, setIsActionHovered] = useState(false) + const [isInstallModalOpen, setIsInstallModalOpen] = useState(false) + + useEffect(() => { + if (!actionOpen) + return + + const handleScroll = () => { + setActionOpen(false) + setIsActionHovered(false) + } + + window.addEventListener('scroll', handleScroll, true) + + return () => { + window.removeEventListener('scroll', handleScroll, true) + } + }, [actionOpen]) + + return ( + <> + <Tooltip + position='right' + needsDelay={false} + popupClassName='!p-0 !px-3 !py-2.5 !w-[224px] !leading-[18px] !text-xs !text-gray-700 !border-[0.5px] !border-black/5 !rounded-xl !shadow-lg' + popupContent={( + <div> + <BlockIcon size='md' className='mb-2' type={BlockEnum.TriggerPlugin} toolIcon={plugin.icon} /> + <div className='mb-1 text-sm leading-5 text-text-primary'>{label}</div> + <div className='text-xs leading-[18px] text-text-secondary'>{description}</div> + </div> + )} + disabled={!description || isActionHovered || actionOpen || isInstallModalOpen} + > + <div + className='group flex h-8 w-full items-center rounded-lg pl-3 pr-1 hover:bg-state-base-hover' + > + <div className='flex h-full min-w-0 items-center'> + <BlockIcon type={BlockEnum.TriggerPlugin} toolIcon={plugin.icon} /> + <div className='ml-2 min-w-0'> + <div className='system-sm-medium truncate text-text-secondary'>{label}</div> + </div> + </div> + <div className='ml-auto flex h-full items-center gap-1 pl-1'> + <span className={`system-xs-regular text-text-tertiary ${actionOpen ? 'hidden' : 'group-hover:hidden'}`}>{installCountLabel}</span> + <div + className={`system-xs-medium flex h-full items-center gap-1 text-components-button-secondary-accent-text [&_.action-btn]:h-6 [&_.action-btn]:min-h-0 [&_.action-btn]:w-6 [&_.action-btn]:rounded-lg [&_.action-btn]:p-0 ${actionOpen ? 'flex' : 'hidden group-hover:flex'}`} + onMouseEnter={() => setIsActionHovered(true)} + onMouseLeave={() => { + if (!actionOpen) + setIsActionHovered(false) + }} + > + <button + type='button' + className='cursor-pointer rounded-md px-1.5 py-0.5 hover:bg-state-base-hover' + onClick={() => { + setActionOpen(false) + setIsInstallModalOpen(true) + setIsActionHovered(true) + }} + > + {t('plugin.installAction')} + </button> + <Action + open={actionOpen} + onOpenChange={(value) => { + setActionOpen(value) + setIsActionHovered(value) + }} + author={plugin.org} + name={plugin.name} + version={plugin.latest_version} + /> + </div> + </div> + </div> + </Tooltip> + {isInstallModalOpen && ( + <InstallFromMarketplace + uniqueIdentifier={plugin.latest_package_identifier} + manifest={plugin} + onSuccess={async () => { + setIsInstallModalOpen(false) + setIsActionHovered(false) + await onInstallSuccess?.() + }} + onClose={() => { + setIsInstallModalOpen(false) + setIsActionHovered(false) + }} + /> + )} + </> + ) +} + +export default FeaturedTriggers diff --git a/web/app/components/workflow/block-selector/hooks.ts b/web/app/components/workflow/block-selector/hooks.ts index b974922e6b..e2dd14e16c 100644 --- a/web/app/components/workflow/block-selector/hooks.ts +++ b/web/app/components/workflow/block-selector/hooks.ts @@ -1,60 +1,123 @@ import { + useCallback, + useEffect, useMemo, useState, } from 'react' import { useTranslation } from 'react-i18next' +import { BLOCKS, START_BLOCKS } from './constants' import { TabsEnum, ToolTypeEnum, } from './types' -export const useTabs = (noBlocks?: boolean, noSources?: boolean, noTools?: boolean) => { +export const useBlocks = () => { const { t } = useTranslation() + + return BLOCKS.map((block) => { + return { + ...block, + title: t(`workflow.blocks.${block.type}`), + } + }) +} + +export const useStartBlocks = () => { + const { t } = useTranslation() + + return START_BLOCKS.map((block) => { + return { + ...block, + title: t(`workflow.blocks.${block.type}`), + } + }) +} + +export const useTabs = ({ + noBlocks, + noSources, + noTools, + noStart = true, + defaultActiveTab, + hasUserInputNode = false, + forceEnableStartTab = false, // When true, Start tab remains enabled even if trigger/user input nodes already exist. +}: { + noBlocks?: boolean + noSources?: boolean + noTools?: boolean + noStart?: boolean + defaultActiveTab?: TabsEnum + hasUserInputNode?: boolean + forceEnableStartTab?: boolean +}) => { + const { t } = useTranslation() + const shouldShowStartTab = !noStart + const shouldDisableStartTab = !forceEnableStartTab && hasUserInputNode const tabs = useMemo(() => { - return [ - ...( - noBlocks - ? [] - : [ - { - key: TabsEnum.Blocks, - name: t('workflow.tabs.blocks'), - }, - ] - ), - ...( - noSources - ? [] - : [ - { - key: TabsEnum.Sources, - name: t('workflow.tabs.sources'), - }, - ] - ), - ...( - noTools - ? [] - : [ - { - key: TabsEnum.Tools, - name: t('workflow.tabs.tools'), - }, - ] - ), - ] - }, [t, noBlocks, noSources, noTools]) + const tabConfigs = [{ + key: TabsEnum.Blocks, + name: t('workflow.tabs.blocks'), + show: !noBlocks, + }, { + key: TabsEnum.Sources, + name: t('workflow.tabs.sources'), + show: !noSources, + }, { + key: TabsEnum.Tools, + name: t('workflow.tabs.tools'), + show: !noTools, + }, + { + key: TabsEnum.Start, + name: t('workflow.tabs.start'), + show: shouldShowStartTab, + disabled: shouldDisableStartTab, + }] + + return tabConfigs.filter(tab => tab.show) + }, [t, noBlocks, noSources, noTools, shouldShowStartTab, shouldDisableStartTab]) + + const getValidTabKey = useCallback((targetKey?: TabsEnum) => { + if (!targetKey) + return undefined + const tab = tabs.find(tabItem => tabItem.key === targetKey) + if (!tab || tab.disabled) + return undefined + return tab.key + }, [tabs]) + const initialTab = useMemo(() => { - if (noBlocks) - return noTools ? TabsEnum.Sources : TabsEnum.Tools + const fallbackTab = tabs.find(tab => !tab.disabled)?.key ?? TabsEnum.Blocks + const preferredDefault = getValidTabKey(defaultActiveTab) + if (preferredDefault) + return preferredDefault - if (noTools) - return noBlocks ? TabsEnum.Sources : TabsEnum.Blocks + const preferredOrder: TabsEnum[] = [] + if (!noBlocks) + preferredOrder.push(TabsEnum.Blocks) + if (!noTools) + preferredOrder.push(TabsEnum.Tools) + if (!noSources) + preferredOrder.push(TabsEnum.Sources) + if (!noStart) + preferredOrder.push(TabsEnum.Start) - return TabsEnum.Blocks - }, [noBlocks, noSources, noTools]) + for (const tabKey of preferredOrder) { + const validKey = getValidTabKey(tabKey) + if (validKey) + return validKey + } + + return fallbackTab + }, [defaultActiveTab, noBlocks, noSources, noTools, noStart, tabs, getValidTabKey]) const [activeTab, setActiveTab] = useState(initialTab) + useEffect(() => { + const currentTab = tabs.find(tab => tab.key === activeTab) + if (!currentTab || currentTab.disabled) + setActiveTab(initialTab) + }, [tabs, activeTab, initialTab]) + return { tabs, activeTab, diff --git a/web/app/components/workflow/block-selector/index.tsx b/web/app/components/workflow/block-selector/index.tsx index 54e8078e7b..9f7989265a 100644 --- a/web/app/components/workflow/block-selector/index.tsx +++ b/web/app/components/workflow/block-selector/index.tsx @@ -40,8 +40,8 @@ const NodeSelectorWrapper = (props: NodeSelectorProps) => { return ( <NodeSelector {...props} - blocks={blocks} - dataSources={dataSourceList || []} + blocks={props.blocks || blocks} + dataSources={props.dataSources || dataSourceList || []} /> ) } diff --git a/web/app/components/workflow/block-selector/main.tsx b/web/app/components/workflow/block-selector/main.tsx index 631b85cd8c..3e13384785 100644 --- a/web/app/components/workflow/block-selector/main.tsx +++ b/web/app/components/workflow/block-selector/main.tsx @@ -9,16 +9,18 @@ import { useState, } from 'react' import { useTranslation } from 'react-i18next' +import { useNodes } from 'reactflow' import type { OffsetOptions, Placement, } from '@floating-ui/react' import type { - BlockEnum, + CommonNodeType, NodeDefault, OnSelectBlock, ToolWithProvider, } from '../types' +import { BlockEnum, isTriggerNode } from '../types' import Tabs from './tabs' import { TabsEnum } from './types' import { useTabs } from './hooks' @@ -51,6 +53,12 @@ export type NodeSelectorProps = { dataSources?: ToolWithProvider[] noBlocks?: boolean noTools?: boolean + showStartTab?: boolean + defaultActiveTab?: TabsEnum + forceShowStartContent?: boolean + ignoreNodeIds?: string[] + forceEnableStartTab?: boolean // Force enabling Start tab regardless of existing trigger/user input nodes (e.g., when changing Start node type). + allowUserInputSelection?: boolean // Override user-input availability; default logic blocks it when triggers exist. } const NodeSelector: FC<NodeSelectorProps> = ({ open: openFromProps, @@ -70,11 +78,47 @@ const NodeSelector: FC<NodeSelectorProps> = ({ dataSources = [], noBlocks = false, noTools = false, + showStartTab = false, + defaultActiveTab, + forceShowStartContent = false, + ignoreNodeIds = [], + forceEnableStartTab = false, + allowUserInputSelection, }) => { const { t } = useTranslation() + const nodes = useNodes() const [searchText, setSearchText] = useState('') const [tags, setTags] = useState<string[]>([]) const [localOpen, setLocalOpen] = useState(false) + // Exclude nodes explicitly ignored (such as the node currently being edited) when checking canvas state. + const filteredNodes = useMemo(() => { + if (!ignoreNodeIds.length) + return nodes + const ignoreSet = new Set(ignoreNodeIds) + return nodes.filter(node => !ignoreSet.has(node.id)) + }, [nodes, ignoreNodeIds]) + + const { hasTriggerNode, hasUserInputNode } = useMemo(() => { + const result = { + hasTriggerNode: false, + hasUserInputNode: false, + } + for (const node of filteredNodes) { + const nodeType = (node.data as CommonNodeType | undefined)?.type + if (!nodeType) + continue + if (nodeType === BlockEnum.Start) + result.hasUserInputNode = true + if (isTriggerNode(nodeType)) + result.hasTriggerNode = true + if (result.hasTriggerNode && result.hasUserInputNode) + break + } + return result + }, [filteredNodes]) + // Default rule: user input option is only available when no Start node nor Trigger node exists on canvas. + const defaultAllowUserInputSelection = !hasUserInputNode && !hasTriggerNode + const canSelectUserInput = allowUserInputSelection ?? defaultAllowUserInputSelection const open = openFromProps === undefined ? localOpen : openFromProps const handleOpenChange = useCallback((newOpen: boolean) => { setLocalOpen(newOpen) @@ -91,22 +135,34 @@ const NodeSelector: FC<NodeSelectorProps> = ({ e.stopPropagation() handleOpenChange(!open) }, [handleOpenChange, open, disabled]) - const handleSelect = useCallback<OnSelectBlock>((type, toolDefaultValue) => { + + const handleSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { handleOpenChange(false) - onSelect(type, toolDefaultValue) + onSelect(type, pluginDefaultValue) }, [handleOpenChange, onSelect]) const { activeTab, setActiveTab, tabs, - } = useTabs(noBlocks, !dataSources.length, noTools) + } = useTabs({ + noBlocks, + noSources: !dataSources.length, + noTools, + noStart: !showStartTab, + defaultActiveTab, + hasUserInputNode, + forceEnableStartTab, + }) const handleActiveTabChange = useCallback((newActiveTab: TabsEnum) => { setActiveTab(newActiveTab) }, [setActiveTab]) const searchPlaceholder = useMemo(() => { + if (activeTab === TabsEnum.Start) + return t('workflow.tabs.searchTrigger') + if (activeTab === TabsEnum.Blocks) return t('workflow.tabs.searchBlock') @@ -136,7 +192,7 @@ const NodeSelector: FC<NodeSelectorProps> = ({ : ( <div className={` - z-10 flex h-4 + z-10 flex h-4 w-4 cursor-pointer items-center justify-center rounded-full bg-components-button-primary-bg text-text-primary-on-surface hover:bg-components-button-primary-bg-hover ${triggerClassName?.(open)} `} @@ -153,9 +209,21 @@ const NodeSelector: FC<NodeSelectorProps> = ({ tabs={tabs} activeTab={activeTab} blocks={blocks} + allowStartNodeSelection={canSelectUserInput} onActiveTabChange={handleActiveTabChange} filterElem={ <div className='relative m-2' onClick={e => e.stopPropagation()}> + {activeTab === TabsEnum.Start && ( + <SearchBox + autoFocus + search={searchText} + onSearchChange={setSearchText} + tags={tags} + onTagsChange={setTags} + placeholder={searchPlaceholder} + inputClassName='grow' + /> + )} {activeTab === TabsEnum.Blocks && ( <Input showLeftIcon @@ -180,6 +248,7 @@ const NodeSelector: FC<NodeSelectorProps> = ({ )} {activeTab === TabsEnum.Tools && ( <SearchBox + autoFocus search={searchText} onSearchChange={setSearchText} tags={tags} @@ -198,6 +267,7 @@ const NodeSelector: FC<NodeSelectorProps> = ({ dataSources={dataSources} noTools={noTools} onTagsChange={setTags} + forceShowStartContent={forceShowStartContent} /> </div> </PortalToFollowElemContent> diff --git a/web/app/components/workflow/block-selector/market-place-plugin/action.tsx b/web/app/components/workflow/block-selector/market-place-plugin/action.tsx index 56ee420cff..034ecbad45 100644 --- a/web/app/components/workflow/block-selector/market-place-plugin/action.tsx +++ b/web/app/components/workflow/block-selector/market-place-plugin/action.tsx @@ -1,6 +1,6 @@ 'use client' import type { FC } from 'react' -import React, { useCallback, useEffect, useRef, useState } from 'react' +import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react' import { useTheme } from 'next-themes' import { useTranslation } from 'react-i18next' import { RiMoreFill } from '@remixicon/react' @@ -15,6 +15,7 @@ import cn from '@/utils/classnames' import { useDownloadPlugin } from '@/service/use-plugins' import { downloadFile } from '@/utils/format' import { getMarketplaceUrl } from '@/utils/var' +import { useQueryClient } from '@tanstack/react-query' type Props = { open: boolean @@ -33,6 +34,7 @@ const OperationDropdown: FC<Props> = ({ }) => { const { t } = useTranslation() const { theme } = useTheme() + const queryClient = useQueryClient() const openRef = useRef(open) const setOpen = useCallback((v: boolean) => { onOpenChange(v) @@ -44,23 +46,32 @@ const OperationDropdown: FC<Props> = ({ }, [setOpen]) const [needDownload, setNeedDownload] = useState(false) - const { data: blob, isLoading } = useDownloadPlugin({ + const downloadInfo = useMemo(() => ({ organization: author, pluginName: name, version, - }, needDownload) + }), [author, name, version]) + const { data: blob, isLoading } = useDownloadPlugin(downloadInfo, needDownload) const handleDownload = useCallback(() => { if (isLoading) return + queryClient.removeQueries({ + queryKey: ['plugins', 'downloadPlugin', downloadInfo], + exact: true, + }) setNeedDownload(true) - }, [isLoading]) + }, [downloadInfo, isLoading, queryClient]) useEffect(() => { - if (blob) { - const fileName = `${author}-${name}_${version}.zip` - downloadFile({ data: blob, fileName }) - setNeedDownload(false) - } - }, [blob]) + if (!needDownload || !blob) + return + const fileName = `${author}-${name}_${version}.zip` + downloadFile({ data: blob, fileName }) + setNeedDownload(false) + queryClient.removeQueries({ + queryKey: ['plugins', 'downloadPlugin', downloadInfo], + exact: true, + }) + }, [author, blob, downloadInfo, name, needDownload, queryClient, version]) return ( <PortalToFollowElem open={open} @@ -77,7 +88,7 @@ const OperationDropdown: FC<Props> = ({ </ActionButton> </PortalToFollowElemTrigger> <PortalToFollowElemContent className='z-[9999]'> - <div className='w-[112px] rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur p-1 shadow-lg'> + <div className='min-w-[176px] rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur p-1 shadow-lg'> <div onClick={handleDownload} className='system-md-regular cursor-pointer rounded-lg px-3 py-1.5 text-text-secondary hover:bg-state-base-hover'>{t('common.operation.download')}</div> <a href={getMarketplaceUrl(`/plugins/${author}/${name}`, { theme })} target='_blank' className='system-md-regular block cursor-pointer rounded-lg px-3 py-1.5 text-text-secondary hover:bg-state-base-hover'>{t('common.operation.viewDetails')}</a> </div> diff --git a/web/app/components/workflow/block-selector/market-place-plugin/item.tsx b/web/app/components/workflow/block-selector/market-place-plugin/item.tsx index 4826108c5c..3c9c9b9f59 100644 --- a/web/app/components/workflow/block-selector/market-place-plugin/item.tsx +++ b/web/app/components/workflow/block-selector/market-place-plugin/item.tsx @@ -52,8 +52,13 @@ const Item: FC<Props> = ({ </div> </div> {/* Action */} - <div className={cn(!open ? 'hidden' : 'flex', 'system-xs-medium h-4 items-center space-x-1 text-components-button-secondary-accent-text group-hover/plugin:flex')}> - <div className='cursor-pointer px-1.5' onClick={showInstallModal}>{t('plugin.installAction')}</div> + <div className={cn(!open ? 'hidden' : 'flex', 'system-xs-medium h-4 items-center space-x-1 text-components-button-secondary-accent-text group-hover/plugin:flex')}> + <div + className='cursor-pointer rounded-md px-1.5 py-0.5 hover:bg-state-base-hover' + onClick={showInstallModal} + > + {t('plugin.installAction')} + </div> <Action open={open} onOpenChange={setOpen} diff --git a/web/app/components/workflow/block-selector/market-place-plugin/list.tsx b/web/app/components/workflow/block-selector/market-place-plugin/list.tsx index 07ed17f86c..8c050b60d6 100644 --- a/web/app/components/workflow/block-selector/market-place-plugin/list.tsx +++ b/web/app/components/workflow/block-selector/market-place-plugin/list.tsx @@ -1,9 +1,10 @@ 'use client' -import React, { useEffect, useImperativeHandle, useMemo, useRef } from 'react' +import { useEffect, useImperativeHandle, useMemo, useRef } from 'react' +import type { RefObject } from 'react' import { useTranslation } from 'react-i18next' import useStickyScroll, { ScrollPosition } from '../use-sticky-scroll' import Item from './item' -import type { Plugin } from '@/app/components/plugins/types.ts' +import type { Plugin } from '@/app/components/plugins/types' import cn from '@/utils/classnames' import Link from 'next/link' import { RiArrowRightUpLine, RiSearchLine } from '@remixicon/react' @@ -17,6 +18,7 @@ export type ListProps = { tags: string[] toolContentClassName?: string disableMaxWidth?: boolean + hideFindMoreFooter?: boolean ref?: React.Ref<ListRef> } @@ -29,6 +31,7 @@ const List = ({ list, toolContentClassName, disableMaxWidth = false, + hideFindMoreFooter = false, ref, }: ListProps) => { const { t } = useTranslation() @@ -39,7 +42,7 @@ const List = ({ const { handleScroll, scrollPosition } = useStickyScroll({ wrapElemRef, - nextToStickyELemRef, + nextToStickyELemRef: nextToStickyELemRef as RefObject<HTMLElement>, }) const stickyClassName = useMemo(() => { switch (scrollPosition) { @@ -69,6 +72,9 @@ const List = ({ } if (noFilter) { + if (hideFindMoreFooter) + return null + return ( <Link className='system-sm-medium sticky bottom-0 z-10 flex h-8 cursor-pointer items-center rounded-b-lg border-[0.5px] border-t border-components-panel-border bg-components-panel-bg-blur px-4 py-1 text-text-accent-light-mode-only shadow-lg' diff --git a/web/app/components/workflow/block-selector/rag-tool-recommendations/index.tsx b/web/app/components/workflow/block-selector/rag-tool-recommendations/index.tsx new file mode 100644 index 0000000000..240c0814a1 --- /dev/null +++ b/web/app/components/workflow/block-selector/rag-tool-recommendations/index.tsx @@ -0,0 +1,139 @@ +'use client' +import type { Dispatch, SetStateAction } from 'react' +import React, { useCallback, useEffect, useMemo, useState } from 'react' +import { Trans, useTranslation } from 'react-i18next' +import type { OnSelectBlock } from '@/app/components/workflow/types' +import type { ViewType } from '@/app/components/workflow/block-selector/view-type-select' +import { RiMoreLine } from '@remixicon/react' +import Loading from '@/app/components/base/loading' +import Link from 'next/link' +import { getMarketplaceUrl } from '@/utils/var' +import { useRAGRecommendedPlugins } from '@/service/use-tools' +import List from './list' +import { getFormattedPlugin } from '@/app/components/plugins/marketplace/utils' +import { ArrowDownRoundFill } from '@/app/components/base/icons/src/vender/solid/arrows' + +type RAGToolRecommendationsProps = { + viewType: ViewType + onSelect: OnSelectBlock + onTagsChange: Dispatch<SetStateAction<string[]>> +} + +const STORAGE_KEY = 'workflow_rag_recommendations_collapsed' + +const RAGToolRecommendations = ({ + viewType, + onSelect, + onTagsChange, +}: RAGToolRecommendationsProps) => { + const { t } = useTranslation() + const [isCollapsed, setIsCollapsed] = useState<boolean>(() => { + if (typeof window === 'undefined') + return false + const stored = window.localStorage.getItem(STORAGE_KEY) + return stored === 'true' + }) + + useEffect(() => { + if (typeof window === 'undefined') + return + const stored = window.localStorage.getItem(STORAGE_KEY) + if (stored !== null) + setIsCollapsed(stored === 'true') + }, []) + + useEffect(() => { + if (typeof window === 'undefined') + return + window.localStorage.setItem(STORAGE_KEY, String(isCollapsed)) + }, [isCollapsed]) + + const { + data: ragRecommendedPlugins, + isLoading: isLoadingRAGRecommendedPlugins, + isFetching: isFetchingRAGRecommendedPlugins, + } = useRAGRecommendedPlugins() + + const recommendedPlugins = useMemo(() => { + if (ragRecommendedPlugins) + return ragRecommendedPlugins.installed_recommended_plugins + return [] + }, [ragRecommendedPlugins]) + + const unInstalledPlugins = useMemo(() => { + if (ragRecommendedPlugins) + return (ragRecommendedPlugins.uninstalled_recommended_plugins).map(getFormattedPlugin) + return [] + }, [ragRecommendedPlugins]) + + const loadMore = useCallback(() => { + onTagsChange((prev) => { + if (prev.includes('rag')) + return prev + return [...prev, 'rag'] + }) + }, [onTagsChange]) + + return ( + <div className='flex flex-col p-1'> + <button + type='button' + className='flex w-full items-center rounded-md px-3 pb-0.5 pt-1 text-left text-text-tertiary' + onClick={() => setIsCollapsed(prev => !prev)} + > + <span className='system-xs-medium text-text-tertiary'>{t('pipeline.ragToolSuggestions.title')}</span> + <ArrowDownRoundFill className={`ml-1 h-4 w-4 text-text-tertiary transition-transform ${isCollapsed ? '-rotate-90' : 'rotate-0'}`} /> + </button> + {!isCollapsed && ( + <> + {/* For first time loading, show loading */} + {isLoadingRAGRecommendedPlugins && ( + <div className='py-2'> + <Loading type='app' /> + </div> + )} + {!isFetchingRAGRecommendedPlugins && recommendedPlugins.length === 0 && unInstalledPlugins.length === 0 && ( + <p className='system-xs-regular px-3 py-1 text-text-tertiary'> + <Trans + i18nKey='pipeline.ragToolSuggestions.noRecommendationPlugins' + components={{ + CustomLink: ( + <Link + className='text-text-accent' + target='_blank' + rel='noopener noreferrer' + href={getMarketplaceUrl('', { tags: 'rag' })} + /> + ), + }} + /> + </p> + )} + {(recommendedPlugins.length > 0 || unInstalledPlugins.length > 0) && ( + <> + <List + tools={recommendedPlugins} + unInstalledPlugins={unInstalledPlugins} + onSelect={onSelect} + viewType={viewType} + /> + <div + className='flex cursor-pointer items-center gap-x-2 py-1 pl-3 pr-2' + onClick={loadMore} + > + <div className='px-1'> + <RiMoreLine className='size-4 text-text-tertiary' /> + </div> + <div className='system-xs-regular text-text-tertiary'> + {t('common.operation.more')} + </div> + </div> + </> + )} + </> + )} + </div> + ) +} + +export default React.memo(RAGToolRecommendations) diff --git a/web/app/components/workflow/block-selector/rag-tool-recommendations/list.tsx b/web/app/components/workflow/block-selector/rag-tool-recommendations/list.tsx new file mode 100644 index 0000000000..8c98fa9d7c --- /dev/null +++ b/web/app/components/workflow/block-selector/rag-tool-recommendations/list.tsx @@ -0,0 +1,104 @@ +import { useCallback, useMemo, useRef } from 'react' +import type { BlockEnum, ToolWithProvider } from '../../types' +import type { ToolDefaultValue } from '../types' +import { ViewType } from '../view-type-select' +import { useGetLanguage } from '@/context/i18n' +import { groupItems } from '../index-bar' +import cn from '@/utils/classnames' +import ToolListTreeView from '../tool/tool-list-tree-view/list' +import ToolListFlatView from '../tool/tool-list-flat-view/list' +import UninstalledItem from './uninstalled-item' +import type { Plugin } from '@/app/components/plugins/types' +import type { OnSelectBlock } from '@/app/components/workflow/types' + +type ListProps = { + onSelect: OnSelectBlock + tools: ToolWithProvider[] + viewType: ViewType + unInstalledPlugins: Plugin[] + className?: string +} + +const List = ({ + onSelect, + tools, + viewType, + unInstalledPlugins, + className, +}: ListProps) => { + const language = useGetLanguage() + const isFlatView = viewType === ViewType.flat + + const { letters, groups: withLetterAndGroupViewToolsData } = groupItems(tools, tool => tool.label[language][0]) + const treeViewToolsData = useMemo(() => { + const result: Record<string, ToolWithProvider[]> = {} + Object.keys(withLetterAndGroupViewToolsData).forEach((letter) => { + Object.keys(withLetterAndGroupViewToolsData[letter]).forEach((groupName) => { + if (!result[groupName]) + result[groupName] = [] + result[groupName].push(...withLetterAndGroupViewToolsData[letter][groupName]) + }) + }) + return result + }, [withLetterAndGroupViewToolsData]) + + const listViewToolData = useMemo(() => { + const result: ToolWithProvider[] = [] + letters.forEach((letter) => { + Object.keys(withLetterAndGroupViewToolsData[letter]).forEach((groupName) => { + result.push(...withLetterAndGroupViewToolsData[letter][groupName].map((item) => { + return { + ...item, + letter, + } + })) + }) + }) + + return result + }, [withLetterAndGroupViewToolsData, letters]) + + const toolRefs = useRef({}) + + const handleSelect = useCallback((type: BlockEnum, tool: ToolDefaultValue) => { + onSelect(type, tool) + }, [onSelect]) + + return ( + <div className={cn('max-w-[100%] p-1', className)}> + {!!tools.length && ( + isFlatView ? ( + <ToolListFlatView + toolRefs={toolRefs} + letters={letters} + payload={listViewToolData} + isShowLetterIndex={false} + hasSearchText={false} + onSelect={handleSelect} + canNotSelectMultiple + indexBar={null} + /> + ) : ( + <ToolListTreeView + payload={treeViewToolsData} + hasSearchText={false} + onSelect={handleSelect} + canNotSelectMultiple + /> + ) + )} + { + unInstalledPlugins.map((item) => { + return ( + <UninstalledItem + key={item.plugin_id} + payload={item} + /> + ) + }) + } + </div> + ) +} + +export default List diff --git a/web/app/components/workflow/block-selector/rag-tool-recommendations/uninstalled-item.tsx b/web/app/components/workflow/block-selector/rag-tool-recommendations/uninstalled-item.tsx new file mode 100644 index 0000000000..98395ec25a --- /dev/null +++ b/web/app/components/workflow/block-selector/rag-tool-recommendations/uninstalled-item.tsx @@ -0,0 +1,63 @@ +'use client' +import React from 'react' +import { useContext } from 'use-context-selector' +import { useTranslation } from 'react-i18next' +import type { Plugin } from '@/app/components/plugins/types' +import InstallFromMarketplace from '@/app/components/plugins/install-plugin/install-from-marketplace' +import I18n from '@/context/i18n' +import { useBoolean } from 'ahooks' +import { BlockEnum } from '../../types' +import BlockIcon from '../../block-icon' + +type UninstalledItemProps = { + payload: Plugin +} + +const UninstalledItem = ({ + payload, +}: UninstalledItemProps) => { + const { t } = useTranslation() + const { locale } = useContext(I18n) + + const getLocalizedText = (obj: Record<string, string> | undefined) => + obj?.[locale] || obj?.['en-US'] || obj?.en_US || '' + const [isShowInstallModal, { + setTrue: showInstallModal, + setFalse: hideInstallModal, + }] = useBoolean(false) + + return ( + <div className='flex h-8 items-center rounded-lg pl-3 pr-2 hover:bg-state-base-hover'> + <BlockIcon + className='shrink-0' + type={BlockEnum.Tool} + toolIcon={payload.icon} + /> + <div className='ml-2 flex w-0 grow items-center'> + <div className='flex w-0 grow items-center gap-x-2'> + <span className='system-sm-regular truncate text-text-primary'> + {getLocalizedText(payload.label)} + </span> + <span className='system-xs-regular text-text-quaternary'> + {payload.org} + </span> + </div> + <div + className='system-xs-medium cursor-pointer pl-1.5 text-components-button-secondary-accent-text' + onClick={showInstallModal} + > + {t('plugin.installAction')} + </div> + {isShowInstallModal && ( + <InstallFromMarketplace + uniqueIdentifier={payload.latest_package_identifier} + manifest={payload} + onSuccess={hideInstallModal} + onClose={hideInstallModal} + /> + )} + </div> + </div> + ) +} +export default React.memo(UninstalledItem) diff --git a/web/app/components/workflow/block-selector/rag-tool-suggestions.tsx b/web/app/components/workflow/block-selector/rag-tool-suggestions.tsx deleted file mode 100644 index 075d2c4e3b..0000000000 --- a/web/app/components/workflow/block-selector/rag-tool-suggestions.tsx +++ /dev/null @@ -1,101 +0,0 @@ -import type { Dispatch, SetStateAction } from 'react' -import React, { useCallback, useMemo } from 'react' -import { Trans, useTranslation } from 'react-i18next' -import type { OnSelectBlock } from '../types' -import Tools from './tools' -import { ToolTypeEnum } from './types' -import type { ViewType } from './view-type-select' -import { RiMoreLine } from '@remixicon/react' -import Loading from '@/app/components/base/loading' -import Link from 'next/link' -import { getMarketplaceUrl } from '@/utils/var' -import { useRAGRecommendedPlugins } from '@/service/use-tools' - -type RAGToolSuggestionsProps = { - viewType: ViewType - onSelect: OnSelectBlock - onTagsChange: Dispatch<SetStateAction<string[]>> -} - -const RAGToolSuggestions: React.FC<RAGToolSuggestionsProps> = ({ - viewType, - onSelect, - onTagsChange, -}) => { - const { t } = useTranslation() - - const { - data: ragRecommendedPlugins, - isFetching: isFetchingRAGRecommendedPlugins, - } = useRAGRecommendedPlugins() - - const recommendedPlugins = useMemo(() => { - if (ragRecommendedPlugins) - return [...ragRecommendedPlugins.installed_recommended_plugins] - return [] - }, [ragRecommendedPlugins]) - - const loadMore = useCallback(() => { - onTagsChange((prev) => { - if (prev.includes('rag')) - return prev - return [...prev, 'rag'] - }) - }, [onTagsChange]) - - return ( - <div className='flex flex-col p-1'> - <div className='system-xs-medium px-3 pb-0.5 pt-1 text-text-tertiary'> - {t('pipeline.ragToolSuggestions.title')} - </div> - {isFetchingRAGRecommendedPlugins && ( - <div className='py-2'> - <Loading type='app' /> - </div> - )} - {!isFetchingRAGRecommendedPlugins && recommendedPlugins.length === 0 && ( - <p className='system-xs-regular px-3 py-1 text-text-tertiary'> - <Trans - i18nKey='pipeline.ragToolSuggestions.noRecommendationPluginsInstalled' - components={{ - CustomLink: ( - <Link - className='text-text-accent' - target='_blank' - rel='noopener noreferrer' - href={getMarketplaceUrl('', { tags: 'rag' })} - /> - ), - }} - /> - </p> - )} - {!isFetchingRAGRecommendedPlugins && recommendedPlugins.length > 0 && ( - <> - <Tools - className='p-0' - tools={recommendedPlugins} - onSelect={onSelect} - canNotSelectMultiple - toolType={ToolTypeEnum.All} - viewType={viewType} - hasSearchText={false} - /> - <div - className='flex cursor-pointer items-center gap-x-2 py-1 pl-3 pr-2' - onClick={loadMore} - > - <div className='px-1'> - <RiMoreLine className='size-4 text-text-tertiary' /> - </div> - <div className='system-xs-regular text-text-tertiary'> - {t('common.operation.more')} - </div> - </div> - </> - )} - </div> - ) -} - -export default React.memo(RAGToolSuggestions) diff --git a/web/app/components/workflow/block-selector/start-blocks.tsx b/web/app/components/workflow/block-selector/start-blocks.tsx new file mode 100644 index 0000000000..31b6abce6c --- /dev/null +++ b/web/app/components/workflow/block-selector/start-blocks.tsx @@ -0,0 +1,139 @@ +import { + memo, + useCallback, + useEffect, + useMemo, +} from 'react' +import { useNodes } from 'reactflow' +import { useTranslation } from 'react-i18next' +import BlockIcon from '../block-icon' +import type { BlockEnum, CommonNodeType } from '../types' +import { BlockEnum as BlockEnumValues } from '../types' +// import { useNodeMetaData } from '../hooks' +import { START_BLOCKS } from './constants' +import type { TriggerDefaultValue } from './types' +import Tooltip from '@/app/components/base/tooltip' +import { useAvailableNodesMetaData } from '../../workflow-app/hooks' + +type StartBlocksProps = { + searchText: string + onSelect: (type: BlockEnum, triggerDefaultValue?: TriggerDefaultValue) => void + availableBlocksTypes?: BlockEnum[] + onContentStateChange?: (hasContent: boolean) => void + hideUserInput?: boolean +} + +const StartBlocks = ({ + searchText, + onSelect, + availableBlocksTypes = [], + onContentStateChange, + hideUserInput = false, // Allow parent to explicitly hide Start node option (e.g. when one already exists). +}: StartBlocksProps) => { + const { t } = useTranslation() + const nodes = useNodes() + // const nodeMetaData = useNodeMetaData() + const availableNodesMetaData = useAvailableNodesMetaData() + + const filteredBlocks = useMemo(() => { + // Check if Start node already exists in workflow + const hasStartNode = nodes.some(node => (node.data as CommonNodeType)?.type === BlockEnumValues.Start) + const normalizedSearch = searchText.toLowerCase() + const getDisplayName = (blockType: BlockEnum) => { + if (blockType === BlockEnumValues.TriggerWebhook) + return t('workflow.customWebhook') + + return t(`workflow.blocks.${blockType}`) + } + + return START_BLOCKS.filter((block) => { + // Hide User Input (Start) if it already exists in workflow or if hideUserInput is true + if (block.type === BlockEnumValues.Start && (hasStartNode || hideUserInput)) + return false + + // Filter by search text + const displayName = getDisplayName(block.type).toLowerCase() + if (!displayName.includes(normalizedSearch) && !block.title.toLowerCase().includes(normalizedSearch)) + return false + + // availableBlocksTypes now contains properly filtered entry node types from parent + return availableBlocksTypes.includes(block.type) + }) + }, [searchText, availableBlocksTypes, nodes, t, hideUserInput]) + + const isEmpty = filteredBlocks.length === 0 + + useEffect(() => { + onContentStateChange?.(!isEmpty) + }, [isEmpty, onContentStateChange]) + + const renderBlock = useCallback((block: { type: BlockEnum; title: string; description?: string }) => ( + <Tooltip + key={block.type} + position='right' + popupClassName='w-[224px] rounded-xl' + needsDelay={false} + popupContent={( + <div> + <BlockIcon + size='md' + className='mb-2' + type={block.type} + /> + <div className='system-md-medium mb-1 text-text-primary'> + {block.type === BlockEnumValues.TriggerWebhook + ? t('workflow.customWebhook') + : t(`workflow.blocks.${block.type}`) + } + </div> + <div className='system-xs-regular text-text-secondary'> + {t(`workflow.blocksAbout.${block.type}`)} + </div> + {(block.type === BlockEnumValues.TriggerWebhook || block.type === BlockEnumValues.TriggerSchedule) && ( + <div className='system-xs-regular mb-1 mt-1 text-text-tertiary'> + {t('tools.author')} {t('workflow.difyTeam')} + </div> + )} + </div> + )} + > + <div + className='flex h-8 w-full cursor-pointer items-center rounded-lg px-3 hover:bg-state-base-hover' + onClick={() => onSelect(block.type)} + > + <BlockIcon + className='mr-2 shrink-0' + type={block.type} + /> + <div className='flex w-0 grow items-center justify-between text-sm text-text-secondary'> + <span className='truncate'>{t(`workflow.blocks.${block.type}`)}</span> + {block.type === BlockEnumValues.Start && ( + <span className='system-xs-regular ml-2 shrink-0 text-text-quaternary'>{t('workflow.blocks.originalStartNode')}</span> + )} + </div> + </div> + </Tooltip> + ), [availableNodesMetaData, onSelect, t]) + + if (isEmpty) + return null + + return ( + <div className='p-1'> + <div className='mb-1'> + {filteredBlocks.map((block, index) => ( + <div key={block.type}> + {renderBlock(block)} + {block.type === BlockEnumValues.Start && index < filteredBlocks.length - 1 && ( + <div className='my-1 px-3'> + <div className='border-t border-divider-subtle' /> + </div> + )} + </div> + ))} + </div> + </div> + ) +} + +export default memo(StartBlocks) diff --git a/web/app/components/workflow/block-selector/tabs.tsx b/web/app/components/workflow/block-selector/tabs.tsx index 91d5ac3af6..ecdb8797c0 100644 --- a/web/app/components/workflow/block-selector/tabs.tsx +++ b/web/app/components/workflow/block-selector/tabs.tsx @@ -1,6 +1,7 @@ import type { Dispatch, FC, SetStateAction } from 'react' -import { memo } from 'react' -import { useAllBuiltInTools, useAllCustomTools, useAllMCPTools, useAllWorkflowTools } from '@/service/use-tools' +import { memo, useEffect, useMemo } from 'react' +import { useTranslation } from 'react-i18next' +import { useAllBuiltInTools, useAllCustomTools, useAllMCPTools, useAllWorkflowTools, useInvalidateAllBuiltInTools } from '@/service/use-tools' import type { BlockEnum, NodeDefault, @@ -9,9 +10,15 @@ import type { } from '../types' import { TabsEnum } from './types' import Blocks from './blocks' +import AllStartBlocks from './all-start-blocks' import AllTools from './all-tools' import DataSources from './data-sources' import cn from '@/utils/classnames' +import { useFeaturedToolsRecommendations } from '@/service/use-plugins' +import { useGlobalPublicStore } from '@/context/global-public-context' +import { useWorkflowStore } from '../store' +import { basePath } from '@/utils/var' +import Tooltip from '@/app/components/base/tooltip' export type TabsProps = { activeTab: TabsEnum @@ -26,10 +33,13 @@ export type TabsProps = { tabs: Array<{ key: TabsEnum name: string + disabled?: boolean }> filterElem: React.ReactNode noBlocks?: boolean noTools?: boolean + forceShowStartContent?: boolean // Force show Start content even when noBlocks=true + allowStartNodeSelection?: boolean // Allow user input option even when trigger node already exists (e.g. change-node flow or when no Start node yet). } const Tabs: FC<TabsProps> = ({ activeTab, @@ -45,11 +55,75 @@ const Tabs: FC<TabsProps> = ({ filterElem, noBlocks, noTools, + forceShowStartContent = false, + allowStartNodeSelection = false, }) => { + const { t } = useTranslation() const { data: buildInTools } = useAllBuiltInTools() const { data: customTools } = useAllCustomTools() const { data: workflowTools } = useAllWorkflowTools() const { data: mcpTools } = useAllMCPTools() + const invalidateBuiltInTools = useInvalidateAllBuiltInTools() + const { enable_marketplace } = useGlobalPublicStore(s => s.systemFeatures) + const workflowStore = useWorkflowStore() + const inRAGPipeline = dataSources.length > 0 + const { + plugins: featuredPlugins = [], + isLoading: isFeaturedLoading, + } = useFeaturedToolsRecommendations(enable_marketplace && !inRAGPipeline) + + const normalizeToolList = useMemo(() => { + return (list?: ToolWithProvider[]) => { + if (!list) + return list + if (!basePath) + return list + let changed = false + const normalized = list.map((provider) => { + if (typeof provider.icon === 'string') { + const icon = provider.icon + const shouldPrefix = Boolean(basePath) + && icon.startsWith('/') + && !icon.startsWith(`${basePath}/`) + + if (shouldPrefix) { + changed = true + return { + ...provider, + icon: `${basePath}${icon}`, + } + } + } + return provider + }) + return changed ? normalized : list + } + }, [basePath]) + + useEffect(() => { + workflowStore.setState((state) => { + const updates: Partial<typeof state> = {} + const normalizedBuiltIn = normalizeToolList(buildInTools) + const normalizedCustom = normalizeToolList(customTools) + const normalizedWorkflow = normalizeToolList(workflowTools) + const normalizedMCP = normalizeToolList(mcpTools) + + if (normalizedBuiltIn !== undefined && state.buildInTools !== normalizedBuiltIn) + updates.buildInTools = normalizedBuiltIn + if (normalizedCustom !== undefined && state.customTools !== normalizedCustom) + updates.customTools = normalizedCustom + if (normalizedWorkflow !== undefined && state.workflowTools !== normalizedWorkflow) + updates.workflowTools = normalizedWorkflow + if (normalizedMCP !== undefined && state.mcpTools !== normalizedMCP) + updates.mcpTools = normalizedMCP + if (!Object.keys(updates).length) + return state + return { + ...state, + ...updates, + } + }) + }, [workflowStore, normalizeToolList, buildInTools, customTools, workflowTools, mcpTools]) return ( <div onClick={e => e.stopPropagation()}> @@ -57,25 +131,64 @@ const Tabs: FC<TabsProps> = ({ !noBlocks && ( <div className='relative flex bg-background-section-burn pl-1 pt-1'> { - tabs.map(tab => ( - <div - key={tab.key} - className={cn( - 'system-sm-medium relative mr-0.5 flex h-8 cursor-pointer items-center rounded-t-lg px-3 ', - activeTab === tab.key - ? 'sm-no-bottom cursor-default bg-components-panel-bg text-text-accent' - : 'text-text-tertiary', - )} - onClick={() => onActiveTabChange(tab.key)} - > - {tab.name} - </div> - )) + tabs.map((tab) => { + const commonProps = { + 'className': cn( + 'system-sm-medium relative mr-0.5 flex h-8 items-center rounded-t-lg px-3', + tab.disabled + ? 'cursor-not-allowed text-text-disabled opacity-60' + : activeTab === tab.key + ? 'sm-no-bottom cursor-default bg-components-panel-bg text-text-accent' + : 'cursor-pointer text-text-tertiary', + ), + 'aria-disabled': tab.disabled, + 'onClick': () => { + if (tab.disabled || activeTab === tab.key) + return + onActiveTabChange(tab.key) + }, + } as const + if (tab.disabled) { + return ( + <Tooltip + key={tab.key} + position='top' + popupClassName='max-w-[200px]' + popupContent={t('workflow.tabs.startDisabledTip')} + > + <div {...commonProps}> + {tab.name} + </div> + </Tooltip> + ) + } + return ( + <div + key={tab.key} + {...commonProps} + > + {tab.name} + </div> + ) + }) } </div> ) } {filterElem} + { + activeTab === TabsEnum.Start && (!noBlocks || forceShowStartContent) && ( + <div className='border-t border-divider-subtle'> + <AllStartBlocks + allowUserInputSelection={allowStartNodeSelection} + searchText={searchText} + onSelect={onSelect} + availableBlocksTypes={availableBlocksTypes} + tags={tags} + /> + </div> + ) + } { activeTab === TabsEnum.Blocks && !noBlocks && ( <div className='border-t border-divider-subtle'> @@ -112,7 +225,13 @@ const Tabs: FC<TabsProps> = ({ mcpTools={mcpTools || []} canChooseMCPTool onTagsChange={onTagsChange} - isInRAGPipeline={dataSources.length > 0} + isInRAGPipeline={inRAGPipeline} + featuredPlugins={featuredPlugins} + featuredLoading={isFeaturedLoading} + showFeatured={enable_marketplace && !inRAGPipeline} + onFeaturedInstallSuccess={async () => { + invalidateBuiltInTools() + }} /> ) } diff --git a/web/app/components/workflow/block-selector/tool-picker.tsx b/web/app/components/workflow/block-selector/tool-picker.tsx index ae4b0d4f02..09f386d657 100644 --- a/web/app/components/workflow/block-selector/tool-picker.tsx +++ b/web/app/components/workflow/block-selector/tool-picker.tsx @@ -17,13 +17,24 @@ import type { BlockEnum, OnSelectBlock } from '@/app/components/workflow/types' import SearchBox from '@/app/components/plugins/marketplace/search-box' import { useTranslation } from 'react-i18next' import { useBoolean } from 'ahooks' -import EditCustomToolModal from '@/app/components/tools/edit-custom-collection-modal/modal' +import EditCustomToolModal from '@/app/components/tools/edit-custom-collection-modal' import { createCustomCollection, } from '@/service/tools' import type { CustomCollectionBackend } from '@/app/components/tools/types' import Toast from '@/app/components/base/toast' -import { useAllBuiltInTools, useAllCustomTools, useAllMCPTools, useAllWorkflowTools, useInvalidateAllCustomTools } from '@/service/use-tools' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, + useInvalidateAllBuiltInTools, + useInvalidateAllCustomTools, + useInvalidateAllMCPTools, + useInvalidateAllWorkflowTools, +} from '@/service/use-tools' +import { useFeaturedToolsRecommendations } from '@/service/use-plugins' +import { useGlobalPublicStore } from '@/context/global-public-context' import cn from '@/utils/classnames' type Props = { @@ -61,11 +72,20 @@ const ToolPicker: FC<Props> = ({ const [searchText, setSearchText] = useState('') const [tags, setTags] = useState<string[]>([]) + const { enable_marketplace } = useGlobalPublicStore(s => s.systemFeatures) const { data: buildInTools } = useAllBuiltInTools() const { data: customTools } = useAllCustomTools() const invalidateCustomTools = useInvalidateAllCustomTools() const { data: workflowTools } = useAllWorkflowTools() const { data: mcpTools } = useAllMCPTools() + const invalidateBuiltInTools = useInvalidateAllBuiltInTools() + const invalidateWorkflowTools = useInvalidateAllWorkflowTools() + const invalidateMcpTools = useInvalidateAllMCPTools() + + const { + plugins: featuredPlugins = [], + isLoading: isFeaturedLoading, + } = useFeaturedToolsRecommendations(enable_marketplace) const { builtinToolList, customToolList, workflowToolList } = useMemo(() => { if (scope === 'plugins') { @@ -129,7 +149,7 @@ const ToolPicker: FC<Props> = ({ if (isShowEditCollectionToolModal) { return ( <EditCustomToolModal - positionLeft + dialogClassName='bg-background-overlay' payload={null} onHide={hideEditCustomCollectionModal} onAdd={doCreateCustomToolCollection} @@ -179,6 +199,15 @@ const ToolPicker: FC<Props> = ({ selectedTools={selectedTools} canChooseMCPTool={canChooseMCPTool} onTagsChange={setTags} + featuredPlugins={featuredPlugins} + featuredLoading={isFeaturedLoading} + showFeatured={scope === 'all' && enable_marketplace} + onFeaturedInstallSuccess={async () => { + invalidateBuiltInTools() + invalidateCustomTools() + invalidateWorkflowTools() + invalidateMcpTools() + }} /> </div> </PortalToFollowElemContent> diff --git a/web/app/components/workflow/block-selector/tool/action-item.tsx b/web/app/components/workflow/block-selector/tool/action-item.tsx index 1005758d43..01c319327a 100644 --- a/web/app/components/workflow/block-selector/tool/action-item.tsx +++ b/web/app/components/workflow/block-selector/tool/action-item.tsx @@ -10,13 +10,20 @@ import { useGetLanguage } from '@/context/i18n' import BlockIcon from '../../block-icon' import cn from '@/utils/classnames' import { useTranslation } from 'react-i18next' +import { basePath } from '@/utils/var' + +const normalizeProviderIcon = (icon: ToolWithProvider['icon']) => { + if (typeof icon === 'string' && basePath && icon.startsWith('/') && !icon.startsWith(`${basePath}/`)) + return `${basePath}${icon}` + return icon +} type Props = { provider: ToolWithProvider payload: Tool disabled?: boolean isAdded?: boolean - onSelect: (type: BlockEnum, tool?: ToolDefaultValue) => void + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void } const ToolItem: FC<Props> = ({ @@ -64,6 +71,9 @@ const ToolItem: FC<Props> = ({ provider_id: provider.id, provider_type: provider.type, provider_name: provider.name, + plugin_id: provider.plugin_id, + plugin_unique_identifier: provider.plugin_unique_identifier, + provider_icon: normalizeProviderIcon(provider.icon), tool_name: payload.name, tool_label: payload.label[language], tool_description: payload.description[language], diff --git a/web/app/components/workflow/block-selector/tool/tool-list-flat-view/list.tsx b/web/app/components/workflow/block-selector/tool/tool-list-flat-view/list.tsx index ca462c082e..510d6f2f4b 100644 --- a/web/app/components/workflow/block-selector/tool/tool-list-flat-view/list.tsx +++ b/web/app/components/workflow/block-selector/tool/tool-list-flat-view/list.tsx @@ -13,7 +13,7 @@ type Props = { isShowLetterIndex: boolean indexBar: React.ReactNode hasSearchText: boolean - onSelect: (type: BlockEnum, tool?: ToolDefaultValue) => void + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void canNotSelectMultiple?: boolean onSelectMultiple?: (type: BlockEnum, tools: ToolDefaultValue[]) => void letters: string[] diff --git a/web/app/components/workflow/block-selector/tool/tool-list-tree-view/item.tsx b/web/app/components/workflow/block-selector/tool/tool-list-tree-view/item.tsx index ac0955da0b..a2833646f3 100644 --- a/web/app/components/workflow/block-selector/tool/tool-list-tree-view/item.tsx +++ b/web/app/components/workflow/block-selector/tool/tool-list-tree-view/item.tsx @@ -11,7 +11,7 @@ type Props = { groupName: string toolList: ToolWithProvider[] hasSearchText: boolean - onSelect: (type: BlockEnum, tool?: ToolDefaultValue) => void + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void canNotSelectMultiple?: boolean onSelectMultiple?: (type: BlockEnum, tools: ToolDefaultValue[]) => void selectedTools?: ToolValue[] diff --git a/web/app/components/workflow/block-selector/tool/tool-list-tree-view/list.tsx b/web/app/components/workflow/block-selector/tool/tool-list-tree-view/list.tsx index d85d1ea682..162b816069 100644 --- a/web/app/components/workflow/block-selector/tool/tool-list-tree-view/list.tsx +++ b/web/app/components/workflow/block-selector/tool/tool-list-tree-view/list.tsx @@ -11,7 +11,7 @@ import { AGENT_GROUP_NAME, CUSTOM_GROUP_NAME, WORKFLOW_GROUP_NAME } from '../../ type Props = { payload: Record<string, ToolWithProvider[]> hasSearchText: boolean - onSelect: (type: BlockEnum, tool?: ToolDefaultValue) => void + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void canNotSelectMultiple?: boolean onSelectMultiple?: (type: BlockEnum, tools: ToolDefaultValue[]) => void selectedTools?: ToolValue[] diff --git a/web/app/components/workflow/block-selector/tool/tool.tsx b/web/app/components/workflow/block-selector/tool/tool.tsx index 30d3e218d2..38be8d19d6 100644 --- a/web/app/components/workflow/block-selector/tool/tool.tsx +++ b/web/app/components/workflow/block-selector/tool/tool.tsx @@ -16,17 +16,25 @@ import { useTranslation } from 'react-i18next' import { useHover } from 'ahooks' import McpToolNotSupportTooltip from '../../nodes/_base/components/mcp-tool-not-support-tooltip' import { Mcp } from '@/app/components/base/icons/src/vender/other' +import { basePath } from '@/utils/var' + +const normalizeProviderIcon = (icon: ToolWithProvider['icon']) => { + if (typeof icon === 'string' && basePath && icon.startsWith('/') && !icon.startsWith(`${basePath}/`)) + return `${basePath}${icon}` + return icon +} type Props = { className?: string payload: ToolWithProvider viewType: ViewType hasSearchText: boolean - onSelect: (type: BlockEnum, tool?: ToolDefaultValue) => void + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void canNotSelectMultiple?: boolean onSelectMultiple?: (type: BlockEnum, tools: ToolDefaultValue[]) => void selectedTools?: ToolValue[] canChooseMCPTool?: boolean + isShowLetterIndex?: boolean } const Tool: FC<Props> = ({ @@ -85,6 +93,9 @@ const Tool: FC<Props> = ({ provider_id: payload.id, provider_type: payload.type, provider_name: payload.name, + plugin_id: payload.plugin_id, + plugin_unique_identifier: payload.plugin_unique_identifier, + provider_icon: normalizeProviderIcon(payload.icon), tool_name: tool.name, tool_label: tool.label[language], tool_description: tool.description[language], @@ -164,6 +175,9 @@ const Tool: FC<Props> = ({ provider_id: payload.id, provider_type: payload.type, provider_name: payload.name, + plugin_id: payload.plugin_id, + plugin_unique_identifier: payload.plugin_unique_identifier, + provider_icon: normalizeProviderIcon(payload.icon), tool_name: tool.name, tool_label: tool.label[language], tool_description: tool.description[language], diff --git a/web/app/components/workflow/block-selector/tools.tsx b/web/app/components/workflow/block-selector/tools.tsx index feb34d2651..66d880d994 100644 --- a/web/app/components/workflow/block-selector/tools.tsx +++ b/web/app/components/workflow/block-selector/tools.tsx @@ -1,22 +1,17 @@ -import { - memo, - useMemo, - useRef, -} from 'react' -import { useTranslation } from 'react-i18next' +import { memo, useMemo, useRef } from 'react' import type { BlockEnum, ToolWithProvider } from '../types' import IndexBar, { groupItems } from './index-bar' import type { ToolDefaultValue, ToolValue } from './types' import type { ToolTypeEnum } from './types' import { ViewType } from './view-type-select' -import Empty from '@/app/components/tools/add-tool-modal/empty' +import Empty from '@/app/components/tools/provider/empty' import { useGetLanguage } from '@/context/i18n' import ToolListTreeView from './tool/tool-list-tree-view/list' import ToolListFlatView from './tool/tool-list-flat-view/list' import classNames from '@/utils/classnames' type ToolsProps = { - onSelect: (type: BlockEnum, tool?: ToolDefaultValue) => void + onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void canNotSelectMultiple?: boolean onSelectMultiple?: (type: BlockEnum, tools: ToolDefaultValue[]) => void tools: ToolWithProvider[] @@ -28,9 +23,8 @@ type ToolsProps = { indexBarClassName?: string selectedTools?: ToolValue[] canChooseMCPTool?: boolean - isShowRAGRecommendations?: boolean } -const Blocks = ({ +const Tools = ({ onSelect, canNotSelectMultiple, onSelectMultiple, @@ -43,10 +37,8 @@ const Blocks = ({ indexBarClassName, selectedTools, canChooseMCPTool, - isShowRAGRecommendations = false, }: ToolsProps) => { // const tools: any = [] - const { t } = useTranslation() const language = useGetLanguage() const isFlatView = viewType === ViewType.flat const isShowLetterIndex = isFlatView && tools.length > 10 @@ -100,21 +92,11 @@ const Blocks = ({ return ( <div className={classNames('max-w-[100%] p-1', className)}> - { - !tools.length && hasSearchText && ( - <div className='mt-2 flex h-[22px] items-center px-3 text-xs font-medium text-text-secondary'>{t('workflow.tabs.noResult')}</div> - ) - } {!tools.length && !hasSearchText && ( <div className='py-10'> <Empty type={toolType!} isAgent={isAgent} /> </div> )} - {!!tools.length && isShowRAGRecommendations && ( - <div className='system-xs-medium px-3 pb-0.5 pt-1 text-text-tertiary'> - {t('tools.allTools')} - </div> - )} {!!tools.length && ( isFlatView ? ( <ToolListFlatView @@ -146,4 +128,4 @@ const Blocks = ({ ) } -export default memo(Blocks) +export default memo(Tools) diff --git a/web/app/components/workflow/block-selector/trigger-plugin/action-item.tsx b/web/app/components/workflow/block-selector/trigger-plugin/action-item.tsx new file mode 100644 index 0000000000..d2bdda8a82 --- /dev/null +++ b/web/app/components/workflow/block-selector/trigger-plugin/action-item.tsx @@ -0,0 +1,90 @@ +'use client' +import type { FC } from 'react' +import React from 'react' +import type { TriggerWithProvider } from '../types' +import type { Event } from '@/app/components/tools/types' +import { BlockEnum } from '../../types' +import type { TriggerDefaultValue } from '../types' +import Tooltip from '@/app/components/base/tooltip' +import { useGetLanguage } from '@/context/i18n' +import BlockIcon from '../../block-icon' +import cn from '@/utils/classnames' +import { useTranslation } from 'react-i18next' + +type Props = { + provider: TriggerWithProvider + payload: Event + disabled?: boolean + isAdded?: boolean + onSelect: (type: BlockEnum, trigger?: TriggerDefaultValue) => void +} + +const TriggerPluginActionItem: FC<Props> = ({ + provider, + payload, + onSelect, + disabled, + isAdded, +}) => { + const { t } = useTranslation() + const language = useGetLanguage() + + return ( + <Tooltip + key={payload.name} + position='right' + needsDelay={false} + popupClassName='!p-0 !px-3 !py-2.5 !w-[224px] !leading-[18px] !text-xs !text-gray-700 !border-[0.5px] !border-black/5 !rounded-xl !shadow-lg' + popupContent={( + <div> + <BlockIcon + size='md' + className='mb-2' + type={BlockEnum.TriggerPlugin} + toolIcon={provider.icon} + /> + <div className='mb-1 text-sm leading-5 text-text-primary'>{payload.label[language]}</div> + <div className='text-xs leading-[18px] text-text-secondary'>{payload.description[language]}</div> + </div> + )} + > + <div + key={payload.name} + className='flex cursor-pointer items-center justify-between rounded-lg pl-[21px] pr-1 hover:bg-state-base-hover' + onClick={() => { + if (disabled) return + const params: Record<string, string> = {} + if (payload.parameters) { + payload.parameters.forEach((item: any) => { + params[item.name] = '' + }) + } + onSelect(BlockEnum.TriggerPlugin, { + plugin_id: provider.plugin_id, + provider_id: provider.name, + provider_type: provider.type as string, + provider_name: provider.name, + event_name: payload.name, + event_label: payload.label[language], + event_description: payload.description[language], + plugin_unique_identifier: provider.plugin_unique_identifier, + title: payload.label[language], + is_team_authorization: provider.is_team_authorization, + output_schema: payload.output_schema || {}, + paramSchemas: payload.parameters, + params, + meta: provider.meta, + }) + }} + > + <div className={cn('system-sm-medium h-8 truncate border-l-2 border-divider-subtle pl-4 leading-8 text-text-secondary')}> + <span className={cn(disabled && 'opacity-30')}>{payload.label[language]}</span> + </div> + {isAdded && ( + <div className='system-xs-regular mr-4 text-text-tertiary'>{t('tools.addToolModal.added')}</div> + )} + </div> + </Tooltip > + ) +} +export default React.memo(TriggerPluginActionItem) diff --git a/web/app/components/workflow/block-selector/trigger-plugin/item.tsx b/web/app/components/workflow/block-selector/trigger-plugin/item.tsx new file mode 100644 index 0000000000..702d3603fb --- /dev/null +++ b/web/app/components/workflow/block-selector/trigger-plugin/item.tsx @@ -0,0 +1,133 @@ +'use client' +import { useGetLanguage } from '@/context/i18n' +import cn from '@/utils/classnames' +import { RiArrowDownSLine, RiArrowRightSLine } from '@remixicon/react' +import type { FC } from 'react' +import React, { useEffect, useMemo, useRef } from 'react' +import { useTranslation } from 'react-i18next' +import { CollectionType } from '@/app/components/tools/types' +import BlockIcon from '@/app/components/workflow/block-icon' +import { BlockEnum } from '@/app/components/workflow/types' +import type { TriggerDefaultValue, TriggerWithProvider } from '@/app/components/workflow/block-selector/types' +import TriggerPluginActionItem from './action-item' + +type Props = { + className?: string + payload: TriggerWithProvider + hasSearchText: boolean + onSelect: (type: BlockEnum, trigger?: TriggerDefaultValue) => void +} + +const TriggerPluginItem: FC<Props> = ({ + className, + payload, + hasSearchText, + onSelect, +}) => { + const { t } = useTranslation() + const language = useGetLanguage() + const notShowProvider = payload.type === CollectionType.workflow + const actions = payload.events + const hasAction = !notShowProvider + const [isFold, setFold] = React.useState<boolean>(true) + const ref = useRef(null) + + useEffect(() => { + if (hasSearchText && isFold) { + setFold(false) + return + } + if (!hasSearchText && !isFold) + setFold(true) + }, [hasSearchText]) + + const FoldIcon = isFold ? RiArrowRightSLine : RiArrowDownSLine + + const groupName = useMemo(() => { + if (payload.type === CollectionType.builtIn) + return payload.author + + if (payload.type === CollectionType.custom) + return t('workflow.tabs.customTool') + + if (payload.type === CollectionType.workflow) + return t('workflow.tabs.workflowTool') + + return payload.author || '' + }, [payload.author, payload.type, t]) + + return ( + <div + key={payload.id} + className={cn('mb-1 last-of-type:mb-0')} + ref={ref} + > + <div className={cn(className)}> + <div + className='group/item flex w-full cursor-pointer select-none items-center justify-between rounded-lg pl-3 pr-1 hover:bg-state-base-hover' + onClick={() => { + if (hasAction) { + setFold(!isFold) + return + } + + const event = actions[0] + const params: Record<string, string> = {} + if (event.parameters) { + event.parameters.forEach((item: any) => { + params[item.name] = '' + }) + } + onSelect(BlockEnum.TriggerPlugin, { + plugin_id: payload.plugin_id, + provider_id: payload.name, + provider_type: payload.type, + provider_name: payload.name, + event_name: event.name, + event_label: event.label[language], + event_description: event.description[language], + title: event.label[language], + plugin_unique_identifier: payload.plugin_unique_identifier, + is_team_authorization: payload.is_team_authorization, + output_schema: event.output_schema || {}, + paramSchemas: event.parameters, + params, + }) + }} + > + <div className='flex h-8 grow items-center'> + <BlockIcon + className='shrink-0' + type={BlockEnum.TriggerPlugin} + toolIcon={payload.icon} + /> + <div className='ml-2 flex min-w-0 flex-1 items-center text-sm text-text-primary'> + <span className='max-w-[200px] truncate'>{notShowProvider ? actions[0]?.label[language] : payload.label[language]}</span> + <span className='system-xs-regular ml-2 truncate text-text-quaternary'>{groupName}</span> + </div> + </div> + + <div className='ml-2 flex items-center'> + {hasAction && ( + <FoldIcon className={cn('h-4 w-4 shrink-0 text-text-tertiary group-hover/item:text-text-tertiary', isFold && 'text-text-quaternary')} /> + )} + </div> + </div> + + {!notShowProvider && hasAction && !isFold && ( + actions.map(action => ( + <TriggerPluginActionItem + key={action.name} + provider={payload} + payload={action} + onSelect={onSelect} + disabled={false} + isAdded={false} + /> + )) + )} + </div> + </div> + ) +} +export default React.memo(TriggerPluginItem) diff --git a/web/app/components/workflow/block-selector/trigger-plugin/list.tsx b/web/app/components/workflow/block-selector/trigger-plugin/list.tsx new file mode 100644 index 0000000000..3caf1149dd --- /dev/null +++ b/web/app/components/workflow/block-selector/trigger-plugin/list.tsx @@ -0,0 +1,105 @@ +'use client' +import { memo, useEffect, useMemo } from 'react' +import { useAllTriggerPlugins } from '@/service/use-triggers' +import TriggerPluginItem from './item' +import type { BlockEnum } from '../../types' +import type { TriggerDefaultValue, TriggerWithProvider } from '../types' +import { useGetLanguage } from '@/context/i18n' + +type TriggerPluginListProps = { + onSelect: (type: BlockEnum, trigger?: TriggerDefaultValue) => void + searchText: string + onContentStateChange?: (hasContent: boolean) => void + tags?: string[] +} + +const TriggerPluginList = ({ + onSelect, + searchText, + onContentStateChange, +}: TriggerPluginListProps) => { + const { data: triggerPluginsData } = useAllTriggerPlugins() + const language = useGetLanguage() + + const normalizedSearch = searchText.trim().toLowerCase() + const triggerPlugins = useMemo(() => { + const plugins = triggerPluginsData || [] + const getLocalizedText = (text?: Record<string, string> | null) => { + if (!text) + return '' + + if (text[language]) + return text[language] + + if (text['en-US']) + return text['en-US'] + + const firstValue = Object.values(text).find(Boolean) + return (typeof firstValue === 'string') ? firstValue : '' + } + const getSearchableTexts = (name: string, label?: Record<string, string> | null) => { + const localized = getLocalizedText(label) + const values = [localized, name].filter(Boolean) + return values.length > 0 ? values : [''] + } + const isMatchingKeywords = (value: string) => value.toLowerCase().includes(normalizedSearch) + + if (!normalizedSearch) + return plugins.filter(triggerWithProvider => triggerWithProvider.events.length > 0) + + return plugins.reduce<TriggerWithProvider[]>((acc, triggerWithProvider) => { + if (triggerWithProvider.events.length === 0) + return acc + + const providerMatches = getSearchableTexts( + triggerWithProvider.name, + triggerWithProvider.label, + ).some(text => isMatchingKeywords(text)) + + if (providerMatches) { + acc.push(triggerWithProvider) + return acc + } + + const matchedEvents = triggerWithProvider.events.filter((event) => { + return getSearchableTexts( + event.name, + event.label, + ).some(text => isMatchingKeywords(text)) + }) + + if (matchedEvents.length > 0) { + acc.push({ + ...triggerWithProvider, + events: matchedEvents, + }) + } + + return acc + }, []) + }, [triggerPluginsData, normalizedSearch, language]) + + const hasContent = triggerPlugins.length > 0 + + useEffect(() => { + onContentStateChange?.(hasContent) + }, [hasContent, onContentStateChange]) + + if (!hasContent) + return null + + return ( + <div className="p-1"> + {triggerPlugins.map(plugin => ( + <TriggerPluginItem + key={plugin.id} + payload={plugin} + onSelect={onSelect} + hasSearchText={!!searchText} + /> + ))} + </div> + ) +} + +export default memo(TriggerPluginList) diff --git a/web/app/components/workflow/block-selector/types.ts b/web/app/components/workflow/block-selector/types.ts index 48fbf6a500..b69453e937 100644 --- a/web/app/components/workflow/block-selector/types.ts +++ b/web/app/components/workflow/block-selector/types.ts @@ -1,8 +1,9 @@ -import type { PluginMeta } from '../../plugins/types' - import type { TypeWithI18N } from '@/app/components/header/account-setting/model-provider-page/declarations' +import type { ParametersSchema, PluginMeta, PluginTriggerSubscriptionConstructor, SupportedCreationMethods, TriggerEvent } from '../../plugins/types' +import type { Collection, Event } from '../../tools/types' export enum TabsEnum { + Start = 'start', Blocks = 'blocks', Tools = 'tools', Sources = 'sources', @@ -24,10 +25,28 @@ export enum BlockClassificationEnum { Utilities = 'utilities', } -export type ToolDefaultValue = { +type PluginCommonDefaultValue = { provider_id: string provider_type: string provider_name: string +} + +export type TriggerDefaultValue = PluginCommonDefaultValue & { + plugin_id?: string + event_name: string + event_label: string + event_description: string + title: string + plugin_unique_identifier: string + is_team_authorization: boolean + params: Record<string, any> + paramSchemas: Record<string, any>[] + output_schema: Record<string, any> + subscription_id?: string + meta?: PluginMeta +} + +export type ToolDefaultValue = PluginCommonDefaultValue & { tool_name: string tool_label: string tool_description: string @@ -35,12 +54,15 @@ export type ToolDefaultValue = { is_team_authorization: boolean params: Record<string, any> paramSchemas: Record<string, any>[] + output_schema?: Record<string, any> credential_id?: string meta?: PluginMeta - output_schema?: Record<string, any> + plugin_id?: string + provider_icon?: Collection['icon'] + plugin_unique_identifier?: string } -export type DataSourceDefaultValue = { +export type DataSourceDefaultValue = Omit<PluginCommonDefaultValue, 'provider_id'> & { plugin_id: string provider_type: string provider_name: string @@ -48,8 +70,11 @@ export type DataSourceDefaultValue = { datasource_label: string title: string fileExtensions?: string[] + plugin_unique_identifier?: string } +export type PluginDefaultValue = ToolDefaultValue | DataSourceDefaultValue | TriggerDefaultValue + export type ToolValue = { provider_name: string provider_show_name?: string @@ -96,3 +121,218 @@ export type DataSourceItem = { } is_authorized: boolean } + +// Backend API types - exact match with Python definitions +export type TriggerParameter = { + multiple: boolean + name: string + label: TypeWithI18N + description?: TypeWithI18N + type: 'string' | 'number' | 'boolean' | 'select' | 'file' | 'files' + | 'model-selector' | 'app-selector' | 'object' | 'array' | 'dynamic-select' + auto_generate?: { + type: string + value?: any + } | null + template?: { + type: string + value?: any + } | null + scope?: string | null + required?: boolean + default?: any + min?: number | null + max?: number | null + precision?: number | null + options?: Array<{ + value: string + label: TypeWithI18N + icon?: string | null + }> | null +} + +export type TriggerCredentialField = { + type: 'secret-input' | 'text-input' | 'select' | 'boolean' + | 'app-selector' | 'model-selector' | 'tools-selector' + name: string + scope?: string | null + required: boolean + default?: string | number | boolean | Array<any> | null + options?: Array<{ + value: string + label: TypeWithI18N + }> | null + label: TypeWithI18N + help?: TypeWithI18N + url?: string | null + placeholder?: TypeWithI18N +} + +export type TriggerSubscriptionSchema = { + parameters_schema: TriggerParameter[] + properties_schema: TriggerCredentialField[] +} + +export type TriggerIdentity = { + author: string + name: string + label: TypeWithI18N + provider: string +} + +export type TriggerDescription = { + human: TypeWithI18N + llm: TypeWithI18N +} + +export type TriggerApiEntity = { + name: string + identity: TriggerIdentity + description: TypeWithI18N + parameters: TriggerParameter[] + output_schema?: Record<string, any> +} + +export type TriggerProviderApiEntity = { + author: string + name: string + label: TypeWithI18N + description: TypeWithI18N + icon?: string + icon_dark?: string + tags: string[] + plugin_id?: string + plugin_unique_identifier: string + supported_creation_methods: SupportedCreationMethods[] + credentials_schema?: TriggerCredentialField[] + subscription_constructor?: PluginTriggerSubscriptionConstructor | null + subscription_schema: ParametersSchema[] + events: TriggerEvent[] +} + +// Frontend types - compatible with ToolWithProvider +export type TriggerWithProvider = Collection & { + events: Event[] + meta: PluginMeta + plugin_unique_identifier: string + credentials_schema?: TriggerCredentialField[] + subscription_constructor?: PluginTriggerSubscriptionConstructor | null + subscription_schema?: ParametersSchema[] + supported_creation_methods: SupportedCreationMethods[] +} + +// ===== API Service Types ===== + +// Trigger subscription instance types + +export enum TriggerCredentialTypeEnum { + ApiKey = 'api-key', + Oauth2 = 'oauth2', + Unauthorized = 'unauthorized', +} + +type TriggerSubscriptionStructure = { + id: string + name: string + provider: string + credential_type: TriggerCredentialTypeEnum + credentials: TriggerSubCredentials + endpoint: string + parameters: TriggerSubParameters + properties: TriggerSubProperties + workflows_in_use: number +} + +export type TriggerSubscription = TriggerSubscriptionStructure + +export type TriggerSubCredentials = { + access_tokens: string +} + +export type TriggerSubParameters = { + repository: string + webhook_secret?: string +} + +export type TriggerSubProperties = { + active: boolean + events: string[] + external_id: string + repository: string + webhook_secret?: string +} + +export type TriggerSubscriptionBuilder = TriggerSubscriptionStructure + +// OAuth configuration types +export type TriggerOAuthConfig = { + configured: boolean + custom_configured: boolean + custom_enabled: boolean + redirect_uri: string + oauth_client_schema: ParametersSchema[] + params: { + client_id: string + client_secret: string + [key: string]: any + } + system_configured: boolean +} + +export type TriggerOAuthClientParams = { + client_id: string + client_secret: string + authorization_url?: string + token_url?: string + scope?: string +} + +export type TriggerOAuthResponse = { + authorization_url: string + subscription_builder: TriggerSubscriptionBuilder +} + +export type TriggerLogEntity = { + id: string + endpoint: string + request: LogRequest + response: LogResponse + created_at: string +} + +export type LogRequest = { + method: string + url: string + headers: LogRequestHeaders + data: string +} + +export type LogRequestHeaders = { + 'Host': string + 'User-Agent': string + 'Content-Length': string + 'Accept': string + 'Content-Type': string + 'X-Forwarded-For': string + 'X-Forwarded-Host': string + 'X-Forwarded-Proto': string + 'X-Github-Delivery': string + 'X-Github-Event': string + 'X-Github-Hook-Id': string + 'X-Github-Hook-Installation-Target-Id': string + 'X-Github-Hook-Installation-Target-Type': string + 'Accept-Encoding': string + [key: string]: string +} + +export type LogResponse = { + status_code: number + headers: LogResponseHeaders + data: string +} + +export type LogResponseHeaders = { + 'Content-Type': string + 'Content-Length': string + [key: string]: string +} diff --git a/web/app/components/workflow/block-selector/utils.ts b/web/app/components/workflow/block-selector/utils.ts index 9b7a5fc076..4272e61644 100644 --- a/web/app/components/workflow/block-selector/utils.ts +++ b/web/app/components/workflow/block-selector/utils.ts @@ -17,6 +17,7 @@ export const transformDataSourceToTool = (dataSourceItem: DataSourceItem) => { is_authorized: dataSourceItem.is_authorized, labels: dataSourceItem.declaration.identity.tags || [], plugin_id: dataSourceItem.plugin_id, + plugin_unique_identifier: dataSourceItem.plugin_unique_identifier, tools: dataSourceItem.declaration.datasources.map((datasource) => { return { name: datasource.identity.name, diff --git a/web/app/components/workflow/candidate-node.tsx b/web/app/components/workflow/candidate-node.tsx index 6f2389aad2..54daf13ebc 100644 --- a/web/app/components/workflow/candidate-node.tsx +++ b/web/app/components/workflow/candidate-node.tsx @@ -12,7 +12,7 @@ import { useStore, useWorkflowStore, } from './store' -import { WorkflowHistoryEvent, useNodesInteractions, useWorkflowHistory } from './hooks' +import { WorkflowHistoryEvent, useAutoGenerateWebhookUrl, useNodesInteractions, useNodesSyncDraft, useWorkflowHistory } from './hooks' import { CUSTOM_NODE } from './constants' import { getIterationStartNode, getLoopStartNode } from './utils' import CustomNode from './nodes' @@ -29,6 +29,8 @@ const CandidateNode = () => { const { zoom } = useViewport() const { handleNodeSelect } = useNodesInteractions() const { saveStateToHistory } = useWorkflowHistory() + const { handleSyncWorkflowDraft } = useNodesSyncDraft() + const autoGenerateWebhookUrl = useAutoGenerateWebhookUrl() useEventListener('click', (e) => { const { candidateNode, mousePosition } = workflowStore.getState() @@ -70,6 +72,12 @@ const CandidateNode = () => { if (candidateNode.type === CUSTOM_NOTE_NODE) handleNodeSelect(candidateNode.id) + + if (candidateNode.data.type === BlockEnum.TriggerWebhook) { + handleSyncWorkflowDraft(true, true, { + onSuccess: () => autoGenerateWebhookUrl(candidateNode.id), + }) + } } }) diff --git a/web/app/components/workflow/constants.ts b/web/app/components/workflow/constants.ts index a8c6a458fc..ad498ff65b 100644 --- a/web/app/components/workflow/constants.ts +++ b/web/app/components/workflow/constants.ts @@ -35,6 +35,54 @@ export const NODE_LAYOUT_HORIZONTAL_PADDING = 60 export const NODE_LAYOUT_VERTICAL_PADDING = 60 export const NODE_LAYOUT_MIN_DISTANCE = 100 +export const isInWorkflowPage = () => { + const pathname = globalThis.location.pathname + return /^\/app\/[^/]+\/workflow$/.test(pathname) || /^\/workflow\/[^/]+$/.test(pathname) +} +export const getGlobalVars = (isChatMode: boolean): Var[] => { + const isInWorkflow = isInWorkflowPage() + const vars: Var[] = [ + ...(isChatMode ? [ + { + variable: 'sys.dialogue_count', + type: VarType.number, + }, + { + variable: 'sys.conversation_id', + type: VarType.string, + }, + ] : []), + { + variable: 'sys.user_id', + type: VarType.string, + }, + { + variable: 'sys.app_id', + type: VarType.string, + }, + { + variable: 'sys.workflow_id', + type: VarType.string, + }, + { + variable: 'sys.workflow_run_id', + type: VarType.string, + }, + ...((isInWorkflow && !isChatMode) ? [ + { + variable: 'sys.timestamp', + type: VarType.number, + }, + ] : []), + ] + return vars +} + +export const VAR_SHOW_NAME_MAP: Record<string, string> = { + 'sys.query': 'query', + 'sys.files': 'files', +} + export const RETRIEVAL_OUTPUT_STRUCT = `{ "content": "", "title": "", @@ -56,7 +104,7 @@ export const RETRIEVAL_OUTPUT_STRUCT = `{ }` export const SUPPORT_OUTPUT_VARS_NODE = [ - BlockEnum.Start, BlockEnum.LLM, BlockEnum.KnowledgeRetrieval, BlockEnum.Code, BlockEnum.TemplateTransform, + BlockEnum.Start, BlockEnum.TriggerWebhook, BlockEnum.TriggerPlugin, BlockEnum.LLM, BlockEnum.KnowledgeRetrieval, BlockEnum.Code, BlockEnum.TemplateTransform, BlockEnum.HttpRequest, BlockEnum.Tool, BlockEnum.VariableAssigner, BlockEnum.VariableAggregator, BlockEnum.QuestionClassifier, BlockEnum.ParameterExtractor, BlockEnum.Iteration, BlockEnum.Loop, BlockEnum.DocExtractor, BlockEnum.ListFilter, diff --git a/web/app/components/workflow/custom-edge.tsx b/web/app/components/workflow/custom-edge.tsx index c38b0ef47d..d4cbc9199d 100644 --- a/web/app/components/workflow/custom-edge.tsx +++ b/web/app/components/workflow/custom-edge.tsx @@ -83,11 +83,11 @@ const CustomEdge = ({ setOpen(v) }, []) - const handleInsert = useCallback<OnSelectBlock>((nodeType, toolDefaultValue) => { + const handleInsert = useCallback<OnSelectBlock>((nodeType, pluginDefaultValue) => { handleNodeAdd( { nodeType, - toolDefaultValue, + pluginDefaultValue, }, { prevNodeId: source, diff --git a/web/app/components/workflow/header/chat-variable-button.tsx b/web/app/components/workflow/header/chat-variable-button.tsx index 36c4a640c4..aa68182c23 100644 --- a/web/app/components/workflow/header/chat-variable-button.tsx +++ b/web/app/components/workflow/header/chat-variable-button.tsx @@ -7,13 +7,16 @@ import cn from '@/utils/classnames' const ChatVariableButton = ({ disabled }: { disabled: boolean }) => { const { theme } = useTheme() + const showChatVariablePanel = useStore(s => s.showChatVariablePanel) const setShowChatVariablePanel = useStore(s => s.setShowChatVariablePanel) const setShowEnvPanel = useStore(s => s.setShowEnvPanel) + const setShowGlobalVariablePanel = useStore(s => s.setShowGlobalVariablePanel) const setShowDebugAndPreviewPanel = useStore(s => s.setShowDebugAndPreviewPanel) const handleClick = () => { setShowChatVariablePanel(true) setShowEnvPanel(false) + setShowGlobalVariablePanel(false) setShowDebugAndPreviewPanel(false) } @@ -21,10 +24,11 @@ const ChatVariableButton = ({ disabled }: { disabled: boolean }) => { <Button className={cn( 'p-2', - theme === 'dark' && 'rounded-lg border border-black/5 bg-white/10 backdrop-blur-sm', + theme === 'dark' && showChatVariablePanel && 'rounded-lg border border-black/5 bg-white/10 backdrop-blur-sm', )} disabled={disabled} onClick={handleClick} + variant='ghost' > <BubbleX className='h-4 w-4 text-components-button-secondary-text' /> </Button> diff --git a/web/app/components/workflow/header/checklist.tsx b/web/app/components/workflow/header/checklist.tsx index 9da16c59c6..794a8997a9 100644 --- a/web/app/components/workflow/header/checklist.tsx +++ b/web/app/components/workflow/header/checklist.tsx @@ -16,6 +16,7 @@ import { useChecklist, useNodesInteractions, } from '../hooks' +import type { ChecklistItem } from '../hooks/use-checklist' import type { CommonEdgeType, CommonNodeType, @@ -29,7 +30,9 @@ import { import { ChecklistSquare, } from '@/app/components/base/icons/src/vender/line/general' -import { AlertTriangle } from '@/app/components/base/icons/src/vender/line/alertsAndFeedback' +import { Warning } from '@/app/components/base/icons/src/vender/line/alertsAndFeedback' +import { IconR } from '@/app/components/base/icons/src/vender/line/arrows' +import type { BlockEnum } from '../types' type WorkflowChecklistProps = { disabled: boolean @@ -44,6 +47,13 @@ const WorkflowChecklist = ({ const needWarningNodes = useChecklist(nodes, edges) const { handleNodeSelect } = useNodesInteractions() + const handleChecklistItemClick = (item: ChecklistItem) => { + if (!item.canNavigate) + return + handleNodeSelect(item.id) + setOpen(false) + } + return ( <PortalToFollowElem placement='bottom-end' @@ -93,38 +103,53 @@ const WorkflowChecklist = ({ <RiCloseLine className='h-4 w-4 text-text-tertiary' /> </div> </div> - <div className='py-2'> + <div className='pb-2'> { !!needWarningNodes.length && ( <> - <div className='px-4 text-xs text-text-tertiary'>{t('workflow.panel.checklistTip')}</div> + <div className='px-4 pt-1 text-xs text-text-tertiary'>{t('workflow.panel.checklistTip')}</div> <div className='px-4 py-2'> { needWarningNodes.map(node => ( <div key={node.id} - className='mb-2 cursor-pointer rounded-lg border-[0.5px] border-components-panel-border bg-components-panel-bg shadow-xs last-of-type:mb-0' - onClick={() => { - handleNodeSelect(node.id) - setOpen(false) - }} + className={cn( + 'group mb-2 rounded-lg border-[0.5px] border-components-panel-border bg-components-panel-bg shadow-xs last-of-type:mb-0', + node.canNavigate ? 'cursor-pointer' : 'cursor-default opacity-80', + )} + onClick={() => handleChecklistItemClick(node)} > <div className='flex h-9 items-center p-2 text-xs font-medium text-text-secondary'> <BlockIcon - type={node.type} + type={node.type as BlockEnum} className='mr-1.5' toolIcon={node.toolIcon} /> <span className='grow truncate'> {node.title} </span> + { + node.canNavigate && ( + <div className='flex h-4 w-[60px] shrink-0 items-center justify-center gap-1 opacity-0 transition-opacity duration-200 group-hover:opacity-100'> + <span className='whitespace-nowrap text-xs font-medium leading-4 text-primary-600'> + {t('workflow.panel.goTo')} + </span> + <IconR className='h-3.5 w-3.5 text-primary-600' /> + </div> + ) + } </div> - <div className='border-t-[0.5px] border-divider-regular'> + <div + className={cn( + 'rounded-b-lg border-t-[0.5px] border-divider-regular', + (node.unConnected || node.errorMessage) && 'bg-gradient-to-r from-components-badge-bg-orange-soft to-transparent', + )} + > { node.unConnected && ( - <div className='px-3 py-2 last:rounded-b-lg'> - <div className='flex text-xs leading-[18px] text-text-tertiary'> - <AlertTriangle className='mr-2 mt-[3px] h-3 w-3 text-[#F79009]' /> + <div className='px-3 py-1 first:pt-1.5 last:pb-1.5'> + <div className='flex text-xs leading-4 text-text-tertiary'> + <Warning className='mr-2 mt-[2px] h-3 w-3 text-[#F79009]' /> {t('workflow.common.needConnectTip')} </div> </div> @@ -132,9 +157,9 @@ const WorkflowChecklist = ({ } { node.errorMessage && ( - <div className='px-3 py-2 last:rounded-b-lg'> - <div className='flex text-xs leading-[18px] text-text-tertiary'> - <AlertTriangle className='mr-2 mt-[3px] h-3 w-3 text-[#F79009]' /> + <div className='px-3 py-1 first:pt-1.5 last:pb-1.5'> + <div className='flex text-xs leading-4 text-text-tertiary'> + <Warning className='mr-2 mt-[2px] h-3 w-3 text-[#F79009]' /> {node.errorMessage} </div> </div> diff --git a/web/app/components/workflow/header/editing-title.tsx b/web/app/components/workflow/header/editing-title.tsx index 32cfd36b4f..81249b05bd 100644 --- a/web/app/components/workflow/header/editing-title.tsx +++ b/web/app/components/workflow/header/editing-title.tsx @@ -11,9 +11,10 @@ const EditingTitle = () => { const draftUpdatedAt = useStore(state => state.draftUpdatedAt) const publishedAt = useStore(state => state.publishedAt) const isSyncingWorkflowDraft = useStore(s => s.isSyncingWorkflowDraft) + const maximizeCanvas = useStore(s => s.maximizeCanvas) return ( - <div className='system-xs-regular flex h-[18px] items-center text-text-tertiary'> + <div className={`system-xs-regular flex h-[18px] min-w-[300px] items-center whitespace-nowrap text-text-tertiary ${maximizeCanvas ? 'ml-2' : ''}`}> { !!draftUpdatedAt && ( <> diff --git a/web/app/components/workflow/header/env-button.tsx b/web/app/components/workflow/header/env-button.tsx index fbb664fbf5..26723305f1 100644 --- a/web/app/components/workflow/header/env-button.tsx +++ b/web/app/components/workflow/header/env-button.tsx @@ -9,13 +9,16 @@ import { useInputFieldPanel } from '@/app/components/rag-pipeline/hooks' const EnvButton = ({ disabled }: { disabled: boolean }) => { const { theme } = useTheme() const setShowChatVariablePanel = useStore(s => s.setShowChatVariablePanel) + const showEnvPanel = useStore(s => s.showEnvPanel) const setShowEnvPanel = useStore(s => s.setShowEnvPanel) + const setShowGlobalVariablePanel = useStore(s => s.setShowGlobalVariablePanel) const setShowDebugAndPreviewPanel = useStore(s => s.setShowDebugAndPreviewPanel) const { closeAllInputFieldPanels } = useInputFieldPanel() const handleClick = () => { setShowEnvPanel(true) setShowChatVariablePanel(false) + setShowGlobalVariablePanel(false) setShowDebugAndPreviewPanel(false) closeAllInputFieldPanels() } @@ -24,8 +27,9 @@ const EnvButton = ({ disabled }: { disabled: boolean }) => { <Button className={cn( 'p-2', - theme === 'dark' && 'rounded-lg border border-black/5 bg-white/10 backdrop-blur-sm', + theme === 'dark' && showEnvPanel && 'rounded-lg border border-black/5 bg-white/10 backdrop-blur-sm', )} + variant='ghost' disabled={disabled} onClick={handleClick} > diff --git a/web/app/components/workflow/header/global-variable-button.tsx b/web/app/components/workflow/header/global-variable-button.tsx index 597c91651e..a133cdeda5 100644 --- a/web/app/components/workflow/header/global-variable-button.tsx +++ b/web/app/components/workflow/header/global-variable-button.tsx @@ -2,16 +2,37 @@ import { memo } from 'react' import Button from '@/app/components/base/button' import { GlobalVariable } from '@/app/components/base/icons/src/vender/line/others' import { useStore } from '@/app/components/workflow/store' +import useTheme from '@/hooks/use-theme' +import cn from '@/utils/classnames' +import { useInputFieldPanel } from '@/app/components/rag-pipeline/hooks' const GlobalVariableButton = ({ disabled }: { disabled: boolean }) => { - const setShowPanel = useStore(s => s.setShowGlobalVariablePanel) + const { theme } = useTheme() + const showGlobalVariablePanel = useStore(s => s.showGlobalVariablePanel) + const setShowGlobalVariablePanel = useStore(s => s.setShowGlobalVariablePanel) + const setShowEnvPanel = useStore(s => s.setShowEnvPanel) + const setShowChatVariablePanel = useStore(s => s.setShowChatVariablePanel) + const setShowDebugAndPreviewPanel = useStore(s => s.setShowDebugAndPreviewPanel) + const { closeAllInputFieldPanels } = useInputFieldPanel() const handleClick = () => { - setShowPanel(true) + setShowGlobalVariablePanel(true) + setShowEnvPanel(false) + setShowChatVariablePanel(false) + setShowDebugAndPreviewPanel(false) + closeAllInputFieldPanels() } return ( - <Button className='p-2' disabled={disabled} onClick={handleClick}> + <Button + className={cn( + 'p-2', + theme === 'dark' && showGlobalVariablePanel && 'rounded-lg border border-black/5 bg-white/10 backdrop-blur-sm', + )} + disabled={disabled} + onClick={handleClick} + variant='ghost' + > <GlobalVariable className='h-4 w-4 text-components-button-secondary-text' /> </Button> ) diff --git a/web/app/components/workflow/header/header-in-normal.tsx b/web/app/components/workflow/header/header-in-normal.tsx index 1c3a442422..20fdafaff5 100644 --- a/web/app/components/workflow/header/header-in-normal.tsx +++ b/web/app/components/workflow/header/header-in-normal.tsx @@ -19,11 +19,14 @@ import EditingTitle from './editing-title' import EnvButton from './env-button' import VersionHistoryButton from './version-history-button' import { useInputFieldPanel } from '@/app/components/rag-pipeline/hooks' +import ScrollToSelectedNodeButton from './scroll-to-selected-node-button' +import GlobalVariableButton from './global-variable-button' export type HeaderInNormalProps = { components?: { left?: React.ReactNode middle?: React.ReactNode + chatVariableTrigger?: React.ReactNode } runAndHistoryProps?: RunAndHistoryProps } @@ -39,6 +42,7 @@ const HeaderInNormal = ({ const setShowDebugAndPreviewPanel = useStore(s => s.setShowDebugAndPreviewPanel) const setShowVariableInspectPanel = useStore(s => s.setShowVariableInspectPanel) const setShowChatVariablePanel = useStore(s => s.setShowChatVariablePanel) + const setShowGlobalVariablePanel = useStore(s => s.setShowGlobalVariablePanel) const nodes = useNodes<StartNodeType>() const selectedNode = nodes.find(node => node.data.selected) const { handleBackupDraft } = useWorkflowRun() @@ -55,23 +59,31 @@ const HeaderInNormal = ({ setShowDebugAndPreviewPanel(false) setShowVariableInspectPanel(false) setShowChatVariablePanel(false) + setShowGlobalVariablePanel(false) closeAllInputFieldPanels() - }, [workflowStore, handleBackupDraft, selectedNode, handleNodeSelect, setShowWorkflowVersionHistoryPanel, setShowEnvPanel, setShowDebugAndPreviewPanel, setShowVariableInspectPanel, setShowChatVariablePanel]) + }, [workflowStore, handleBackupDraft, selectedNode, handleNodeSelect, setShowWorkflowVersionHistoryPanel, setShowEnvPanel, setShowDebugAndPreviewPanel, setShowVariableInspectPanel, setShowChatVariablePanel, setShowGlobalVariablePanel]) return ( - <> + <div className='flex w-full items-center justify-between'> <div> <EditingTitle /> </div> + <div> + <ScrollToSelectedNodeButton /> + </div> <div className='flex items-center gap-2'> {components?.left} - <EnvButton disabled={nodesReadOnly} /> <Divider type='vertical' className='mx-auto h-3.5' /> <RunAndHistory {...runAndHistoryProps} /> + <div className='shrink-0 cursor-pointer rounded-lg border-[0.5px] border-components-button-secondary-border bg-components-button-secondary-bg shadow-xs backdrop-blur-[10px]'> + {components?.chatVariableTrigger} + <EnvButton disabled={nodesReadOnly} /> + <GlobalVariableButton disabled={nodesReadOnly} /> + </div> {components?.middle} <VersionHistoryButton onClick={onStartRestoring} /> </div> - </> + </div> ) } diff --git a/web/app/components/workflow/header/run-mode.tsx b/web/app/components/workflow/header/run-mode.tsx index d1fd3510cc..7a1d444d30 100644 --- a/web/app/components/workflow/header/run-mode.tsx +++ b/web/app/components/workflow/header/run-mode.tsx @@ -1,6 +1,6 @@ -import React, { useCallback } from 'react' +import React, { useCallback, useEffect, useRef } from 'react' import { useTranslation } from 'react-i18next' -import { useWorkflowRun, useWorkflowStartRun } from '@/app/components/workflow/hooks' +import { useWorkflowRun, useWorkflowRunValidation, useWorkflowStartRun } from '@/app/components/workflow/hooks' import { useStore } from '@/app/components/workflow/store' import { WorkflowRunningStatus } from '@/app/components/workflow/types' import { useEventEmitterContextContext } from '@/context/event-emitter' @@ -9,6 +9,9 @@ import { getKeyboardKeyNameBySystem } from '@/app/components/workflow/utils' import cn from '@/utils/classnames' import { RiLoader2Line, RiPlayLargeLine } from '@remixicon/react' import { StopCircle } from '@/app/components/base/icons/src/vender/line/mediaAndDevices' +import { useDynamicTestRunOptions } from '../hooks/use-dynamic-test-run-options' +import TestRunMenu, { type TestRunMenuRef, type TriggerOption, TriggerType } from './test-run-menu' +import { useToastContext } from '@/app/components/base/toast' type RunModeProps = { text?: string @@ -18,16 +21,84 @@ const RunMode = ({ text, }: RunModeProps) => { const { t } = useTranslation() - const { handleWorkflowStartRunInWorkflow } = useWorkflowStartRun() + const { + handleWorkflowStartRunInWorkflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, + } = useWorkflowStartRun() const { handleStopRun } = useWorkflowRun() + const { validateBeforeRun, warningNodes } = useWorkflowRunValidation() const workflowRunningData = useStore(s => s.workflowRunningData) + const isListening = useStore(s => s.isListening) - const isRunning = workflowRunningData?.result.status === WorkflowRunningStatus.Running + const status = workflowRunningData?.result.status + const isRunning = status === WorkflowRunningStatus.Running || isListening + + const dynamicOptions = useDynamicTestRunOptions() + const testRunMenuRef = useRef<TestRunMenuRef>(null) + const { notify } = useToastContext() + + useEffect(() => { + // @ts-expect-error - Dynamic property for backward compatibility with keyboard shortcuts + window._toggleTestRunDropdown = () => { + testRunMenuRef.current?.toggle() + } + return () => { + // @ts-expect-error - Dynamic property cleanup + delete window._toggleTestRunDropdown + } + }, []) const handleStop = useCallback(() => { handleStopRun(workflowRunningData?.task_id || '') }, [handleStopRun, workflowRunningData?.task_id]) + const handleTriggerSelect = useCallback((option: TriggerOption) => { + // Validate checklist before running any workflow + let isValid: boolean = true + warningNodes.forEach((node) => { + if (node.id === option.nodeId) + isValid = false + }) + if (!isValid) { + notify({ type: 'error', message: t('workflow.panel.checklistTip') }) + return + } + + if (option.type === TriggerType.UserInput) { + handleWorkflowStartRunInWorkflow() + } + else if (option.type === TriggerType.Schedule) { + handleWorkflowTriggerScheduleRunInWorkflow(option.nodeId) + } + else if (option.type === TriggerType.Webhook) { + if (option.nodeId) + handleWorkflowTriggerWebhookRunInWorkflow({ nodeId: option.nodeId }) + } + else if (option.type === TriggerType.Plugin) { + if (option.nodeId) + handleWorkflowTriggerPluginRunInWorkflow(option.nodeId) + } + else if (option.type === TriggerType.All) { + const targetNodeIds = option.relatedNodeIds?.filter(Boolean) + if (targetNodeIds && targetNodeIds.length > 0) + handleWorkflowRunAllTriggersInWorkflow(targetNodeIds) + } + else { + // Placeholder for trigger-specific execution logic for schedule, webhook, plugin types + console.log('TODO: Handle trigger execution for type:', option.type, 'nodeId:', option.nodeId) + } + }, [ + validateBeforeRun, + handleWorkflowStartRunInWorkflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, + ]) + const { eventEmitter } = useEventEmitterContextContext() eventEmitter?.useSubscription((v: any) => { if (v.type === EVENT_WORKFLOW_STOP) @@ -36,46 +107,46 @@ const RunMode = ({ return ( <div className='flex items-center gap-x-px'> - <button - type='button' - className={cn( - 'system-xs-medium flex h-7 items-center gap-x-1 px-1.5 text-text-accent hover:bg-state-accent-hover', - isRunning && 'cursor-not-allowed bg-state-accent-hover', - isRunning ? 'rounded-l-md' : 'rounded-md', - )} - onClick={() => { - handleWorkflowStartRunInWorkflow() - }} - disabled={isRunning} - > - { - isRunning - ? ( - <> - <RiLoader2Line className='mr-1 size-4 animate-spin' /> - {t('workflow.common.running')} - </> - ) - : ( - <> + { + isRunning + ? ( + <button + type='button' + className={cn( + 'system-xs-medium flex h-7 cursor-not-allowed items-center gap-x-1 rounded-l-md bg-state-accent-hover px-1.5 text-text-accent', + )} + disabled={true} + > + <RiLoader2Line className='mr-1 size-4 animate-spin' /> + {isListening ? t('workflow.common.listening') : t('workflow.common.running')} + </button> + ) + : ( + <TestRunMenu + ref={testRunMenuRef} + options={dynamicOptions} + onSelect={handleTriggerSelect} + > + <div + className={cn( + 'system-xs-medium flex h-7 cursor-pointer items-center gap-x-1 rounded-md px-1.5 text-text-accent hover:bg-state-accent-hover', + )} + style={{ userSelect: 'none' }} + > <RiPlayLargeLine className='mr-1 size-4' /> {text ?? t('workflow.common.run')} - </> - ) - } - { - !isRunning && ( - <div className='system-kbd flex items-center gap-x-0.5 text-text-tertiary'> - <div className='flex size-4 items-center justify-center rounded-[4px] bg-components-kbd-bg-gray'> - {getKeyboardKeyNameBySystem('alt')} + <div className='system-kbd flex items-center gap-x-0.5 text-text-tertiary'> + <div className='flex size-4 items-center justify-center rounded-[4px] bg-components-kbd-bg-gray'> + {getKeyboardKeyNameBySystem('alt')} + </div> + <div className='flex size-4 items-center justify-center rounded-[4px] bg-components-kbd-bg-gray'> + R + </div> + </div> </div> - <div className='flex size-4 items-center justify-center rounded-[4px] bg-components-kbd-bg-gray'> - R - </div> - </div> + </TestRunMenu> ) - } - </button> + } { isRunning && ( <button diff --git a/web/app/components/workflow/header/scroll-to-selected-node-button.tsx b/web/app/components/workflow/header/scroll-to-selected-node-button.tsx new file mode 100644 index 0000000000..d3e7248d9a --- /dev/null +++ b/web/app/components/workflow/header/scroll-to-selected-node-button.tsx @@ -0,0 +1,34 @@ +import type { FC } from 'react' +import { useCallback } from 'react' +import { useNodes } from 'reactflow' +import { useTranslation } from 'react-i18next' +import type { CommonNodeType } from '../types' +import { scrollToWorkflowNode } from '../utils/node-navigation' +import cn from '@/utils/classnames' + +const ScrollToSelectedNodeButton: FC = () => { + const { t } = useTranslation() + const nodes = useNodes<CommonNodeType>() + const selectedNode = nodes.find(node => node.data.selected) + + const handleScrollToSelectedNode = useCallback(() => { + if (!selectedNode) return + scrollToWorkflowNode(selectedNode.id) + }, [selectedNode]) + + if (!selectedNode) + return null + + return ( + <div + className={cn( + 'system-xs-medium flex h-6 cursor-pointer items-center justify-center whitespace-nowrap rounded-md border-[0.5px] border-effects-highlight bg-components-actionbar-bg px-3 text-text-tertiary shadow-lg backdrop-blur-sm transition-colors duration-200 hover:text-text-accent', + )} + onClick={handleScrollToSelectedNode} + > + {t('workflow.panel.scrollToSelectedNode')} + </div> + ) +} + +export default ScrollToSelectedNodeButton diff --git a/web/app/components/workflow/header/test-run-menu.tsx b/web/app/components/workflow/header/test-run-menu.tsx new file mode 100644 index 0000000000..40aabab6f8 --- /dev/null +++ b/web/app/components/workflow/header/test-run-menu.tsx @@ -0,0 +1,251 @@ +import { + type MouseEvent, + type MouseEventHandler, + type ReactElement, + cloneElement, + forwardRef, + isValidElement, + useCallback, + useEffect, + useImperativeHandle, + useMemo, + useState, +} from 'react' +import { useTranslation } from 'react-i18next' +import { + PortalToFollowElem, + PortalToFollowElemContent, + PortalToFollowElemTrigger, +} from '@/app/components/base/portal-to-follow-elem' +import ShortcutsName from '../shortcuts-name' + +export enum TriggerType { + UserInput = 'user_input', + Schedule = 'schedule', + Webhook = 'webhook', + Plugin = 'plugin', + All = 'all', +} + +export type TriggerOption = { + id: string + type: TriggerType + name: string + icon: React.ReactNode + nodeId?: string + relatedNodeIds?: string[] + enabled: boolean +} + +export type TestRunOptions = { + userInput?: TriggerOption + triggers: TriggerOption[] + runAll?: TriggerOption +} + +type TestRunMenuProps = { + options: TestRunOptions + onSelect: (option: TriggerOption) => void + children: React.ReactNode +} + +export type TestRunMenuRef = { + toggle: () => void +} + +type ShortcutMapping = { + option: TriggerOption + shortcutKey: string +} + +const buildShortcutMappings = (options: TestRunOptions): ShortcutMapping[] => { + const mappings: ShortcutMapping[] = [] + + if (options.userInput && options.userInput.enabled !== false) + mappings.push({ option: options.userInput, shortcutKey: '~' }) + + let numericShortcut = 0 + + if (options.runAll && options.runAll.enabled !== false) + mappings.push({ option: options.runAll, shortcutKey: String(numericShortcut++) }) + + options.triggers.forEach((trigger) => { + if (trigger.enabled !== false) + mappings.push({ option: trigger, shortcutKey: String(numericShortcut++) }) + }) + + return mappings +} + +const TestRunMenu = forwardRef<TestRunMenuRef, TestRunMenuProps>(({ + options, + onSelect, + children, +}, ref) => { + const { t } = useTranslation() + const [open, setOpen] = useState(false) + const shortcutMappings = useMemo(() => buildShortcutMappings(options), [options]) + const shortcutKeyById = useMemo(() => { + const map = new Map<string, string>() + shortcutMappings.forEach(({ option, shortcutKey }) => { + map.set(option.id, shortcutKey) + }) + return map + }, [shortcutMappings]) + + const handleSelect = useCallback((option: TriggerOption) => { + onSelect(option) + setOpen(false) + }, [onSelect]) + + const enabledOptions = useMemo(() => { + const flattened: TriggerOption[] = [] + + if (options.userInput) + flattened.push(options.userInput) + if (options.runAll) + flattened.push(options.runAll) + flattened.push(...options.triggers) + + return flattened.filter(option => option.enabled !== false) + }, [options]) + + const hasSingleEnabledOption = enabledOptions.length === 1 + const soleEnabledOption = hasSingleEnabledOption ? enabledOptions[0] : undefined + + const runSoleOption = useCallback(() => { + if (soleEnabledOption) + handleSelect(soleEnabledOption) + }, [handleSelect, soleEnabledOption]) + + useImperativeHandle(ref, () => ({ + toggle: () => { + if (hasSingleEnabledOption) { + runSoleOption() + return + } + + setOpen(prev => !prev) + }, + }), [hasSingleEnabledOption, runSoleOption]) + + useEffect(() => { + if (!open) + return + + const handleKeyDown = (event: KeyboardEvent) => { + if (event.defaultPrevented || event.repeat || event.altKey || event.ctrlKey || event.metaKey) + return + + const normalizedKey = event.key === '`' ? '~' : event.key + const mapping = shortcutMappings.find(({ shortcutKey }) => shortcutKey === normalizedKey) + + if (mapping) { + event.preventDefault() + handleSelect(mapping.option) + } + } + + window.addEventListener('keydown', handleKeyDown) + return () => { + window.removeEventListener('keydown', handleKeyDown) + } + }, [handleSelect, open, shortcutMappings]) + + const renderOption = (option: TriggerOption) => { + const shortcutKey = shortcutKeyById.get(option.id) + + return ( + <div + key={option.id} + className='system-md-regular flex cursor-pointer items-center rounded-lg px-3 py-1.5 text-text-secondary hover:bg-state-base-hover' + onClick={() => handleSelect(option)} + > + <div className='flex min-w-0 flex-1 items-center'> + <div className='flex h-6 w-6 shrink-0 items-center justify-center'> + {option.icon} + </div> + <span className='ml-2 truncate'>{option.name}</span> + </div> + {shortcutKey && ( + <ShortcutsName keys={[shortcutKey]} className="ml-2" textColor="secondary" /> + )} + </div> + ) + } + + const hasUserInput = !!options.userInput && options.userInput.enabled !== false + const hasTriggers = options.triggers.some(trigger => trigger.enabled !== false) + const hasRunAll = !!options.runAll && options.runAll.enabled !== false + + if (hasSingleEnabledOption && soleEnabledOption) { + const handleRunClick = (event?: MouseEvent<HTMLElement>) => { + if (event?.defaultPrevented) + return + + runSoleOption() + } + + if (isValidElement(children)) { + const childElement = children as ReactElement<{ onClick?: MouseEventHandler<HTMLElement> }> + const originalOnClick = childElement.props?.onClick + + return cloneElement(childElement, { + onClick: (event: MouseEvent<HTMLElement>) => { + if (typeof originalOnClick === 'function') + originalOnClick(event) + + if (event?.defaultPrevented) + return + + runSoleOption() + }, + }) + } + + return ( + <span onClick={handleRunClick}> + {children} + </span> + ) + } + + return ( + <PortalToFollowElem + open={open} + onOpenChange={setOpen} + placement='bottom-start' + offset={{ mainAxis: 8, crossAxis: -4 }} + > + <PortalToFollowElemTrigger asChild onClick={() => setOpen(!open)}> + <div style={{ userSelect: 'none' }}> + {children} + </div> + </PortalToFollowElemTrigger> + <PortalToFollowElemContent className='z-[12]'> + <div className='w-[284px] rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg p-1 shadow-lg'> + <div className='mb-2 px-3 pt-2 text-sm font-medium text-text-primary'> + {t('workflow.common.chooseStartNodeToRun')} + </div> + <div> + {hasUserInput && renderOption(options.userInput!)} + + {(hasTriggers || hasRunAll) && hasUserInput && ( + <div className='mx-3 my-1 h-px bg-divider-subtle' /> + )} + + {hasRunAll && renderOption(options.runAll!)} + + {hasTriggers && options.triggers + .filter(trigger => trigger.enabled !== false) + .map(trigger => renderOption(trigger))} + </div> + </div> + </PortalToFollowElemContent> + </PortalToFollowElem> + ) +}) + +TestRunMenu.displayName = 'TestRunMenu' + +export default TestRunMenu diff --git a/web/app/components/workflow/hooks-store/store.ts b/web/app/components/workflow/hooks-store/store.ts index bd09bc501b..3e205f9521 100644 --- a/web/app/components/workflow/hooks-store/store.ts +++ b/web/app/components/workflow/hooks-store/store.ts @@ -40,11 +40,15 @@ export type CommonHooksFnMap = { handleBackupDraft: () => void handleLoadBackupDraft: () => void handleRestoreFromPublishedWorkflow: (...args: any[]) => void - handleRun: (params: any, callback?: IOtherOptions) => void + handleRun: (params: any, callback?: IOtherOptions, options?: any) => void handleStopRun: (...args: any[]) => void handleStartWorkflowRun: () => void handleWorkflowStartRunInWorkflow: () => void handleWorkflowStartRunInChatflow: () => void + handleWorkflowTriggerScheduleRunInWorkflow: (nodeId?: string) => void + handleWorkflowTriggerWebhookRunInWorkflow: (params: { nodeId: string }) => void + handleWorkflowTriggerPluginRunInWorkflow: (nodeId?: string) => void + handleWorkflowRunAllTriggersInWorkflow: (nodeIds: string[]) => void availableNodesMetaData?: AvailableNodesMetaData getWorkflowRunAndTraceUrl: (runId?: string) => { runUrl: string; traceUrl: string } exportCheck?: () => Promise<void> @@ -87,6 +91,10 @@ export const createHooksStore = ({ handleStartWorkflowRun = noop, handleWorkflowStartRunInWorkflow = noop, handleWorkflowStartRunInChatflow = noop, + handleWorkflowTriggerScheduleRunInWorkflow = noop, + handleWorkflowTriggerWebhookRunInWorkflow = noop, + handleWorkflowTriggerPluginRunInWorkflow = noop, + handleWorkflowRunAllTriggersInWorkflow = noop, availableNodesMetaData = { nodes: [], }, @@ -125,6 +133,10 @@ export const createHooksStore = ({ handleStartWorkflowRun, handleWorkflowStartRunInWorkflow, handleWorkflowStartRunInChatflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, availableNodesMetaData, getWorkflowRunAndTraceUrl, exportCheck, diff --git a/web/app/components/workflow/hooks/index.ts b/web/app/components/workflow/hooks/index.ts index 1dbba6b0e2..1131836b35 100644 --- a/web/app/components/workflow/hooks/index.ts +++ b/web/app/components/workflow/hooks/index.ts @@ -22,3 +22,5 @@ export * from './use-DSL' export * from './use-inspect-vars-crud' export * from './use-set-workflow-vars-with-value' export * from './use-workflow-search' +export * from './use-auto-generate-webhook-url' +export * from './use-serial-async-callback' diff --git a/web/app/components/workflow/hooks/use-auto-generate-webhook-url.ts b/web/app/components/workflow/hooks/use-auto-generate-webhook-url.ts new file mode 100644 index 0000000000..d7d66e31ef --- /dev/null +++ b/web/app/components/workflow/hooks/use-auto-generate-webhook-url.ts @@ -0,0 +1,48 @@ +import { useCallback } from 'react' +import { produce } from 'immer' +import { useStoreApi } from 'reactflow' +import { useStore as useAppStore } from '@/app/components/app/store' +import { BlockEnum } from '@/app/components/workflow/types' +import { fetchWebhookUrl } from '@/service/apps' + +export const useAutoGenerateWebhookUrl = () => { + const reactFlowStore = useStoreApi() + + return useCallback(async (nodeId: string) => { + const appId = useAppStore.getState().appDetail?.id + if (!appId) + return + + const { getNodes } = reactFlowStore.getState() + const node = getNodes().find(n => n.id === nodeId) + if (!node || node.data.type !== BlockEnum.TriggerWebhook) + return + + if (node.data.webhook_url && node.data.webhook_url.length > 0) + return + + try { + const response = await fetchWebhookUrl({ appId, nodeId }) + const { getNodes: getLatestNodes, setNodes } = reactFlowStore.getState() + let hasUpdated = false + const updatedNodes = produce(getLatestNodes(), (draft) => { + const targetNode = draft.find(n => n.id === nodeId) + if (!targetNode || targetNode.data.type !== BlockEnum.TriggerWebhook) + return + + targetNode.data = { + ...targetNode.data, + webhook_url: response.webhook_url, + webhook_debug_url: response.webhook_debug_url, + } + hasUpdated = true + }) + + if (hasUpdated) + setNodes(updatedNodes) + } + catch (error: unknown) { + console.error('Failed to auto-generate webhook URL:', error) + } + }, [reactFlowStore]) +} diff --git a/web/app/components/workflow/hooks/use-available-blocks.ts b/web/app/components/workflow/hooks/use-available-blocks.ts index b4e037d29f..e1a1919afd 100644 --- a/web/app/components/workflow/hooks/use-available-blocks.ts +++ b/web/app/components/workflow/hooks/use-available-blocks.ts @@ -21,7 +21,9 @@ export const useAvailableBlocks = (nodeType?: BlockEnum, inContainer?: boolean) } = useNodesMetaData() const availableNodesType = useMemo(() => availableNodes.map(node => node.metaData.type), [availableNodes]) const availablePrevBlocks = useMemo(() => { - if (!nodeType || nodeType === BlockEnum.Start || nodeType === BlockEnum.DataSource) + if (!nodeType || nodeType === BlockEnum.Start || nodeType === BlockEnum.DataSource + || nodeType === BlockEnum.TriggerPlugin || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerSchedule) return [] return availableNodesType diff --git a/web/app/components/workflow/hooks/use-checklist.ts b/web/app/components/workflow/hooks/use-checklist.ts index 1f474a699a..be7aabbc68 100644 --- a/web/app/components/workflow/hooks/use-checklist.ts +++ b/web/app/components/workflow/hooks/use-checklist.ts @@ -4,8 +4,9 @@ import { useRef, } from 'react' import { useTranslation } from 'react-i18next' -import { useStoreApi } from 'reactflow' +import { useEdges, useNodes, useStoreApi } from 'reactflow' import type { + CommonEdgeType, CommonNodeType, Edge, Node, @@ -21,20 +22,22 @@ import { getToolCheckParams, getValidTreeNodes, } from '../utils' +import { getTriggerCheckParams } from '../utils/trigger' import { CUSTOM_NODE, } from '../constants' import { useGetToolIcon, - useWorkflow, + useNodesMetaData, } from '../hooks' import type { ToolNodeType } from '../nodes/tool/types' import type { DataSourceNodeType } from '../nodes/data-source/types' -import { useNodesMetaData } from './use-nodes-meta-data' +import type { PluginTriggerNodeType } from '../nodes/trigger-plugin/types' import { useToastContext } from '@/app/components/base/toast' import { useGetLanguage } from '@/context/i18n' import type { AgentNodeType } from '../nodes/agent/types' import { useStrategyProviders } from '@/service/use-strategy' +import { useAllTriggerPlugins } from '@/service/use-triggers' import { useDatasetsDetailStore } from '../datasets-detail-store/store' import type { KnowledgeRetrievalNodeType } from '../nodes/knowledge-retrieval/types' import type { DataSet } from '@/models/datasets' @@ -42,22 +45,49 @@ import { fetchDatasets } from '@/service/datasets' import { MAX_TREE_DEPTH } from '@/config' import useNodesAvailableVarList, { useGetNodesAvailableVarList } from './use-nodes-available-var-list' import { getNodeUsedVars, isSpecialVar } from '../nodes/_base/components/variable/utils' +import type { Emoji } from '@/app/components/tools/types' import { useModelList } from '@/app/components/header/account-setting/model-provider-page/hooks' import { ModelTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' import type { KnowledgeBaseNodeType } from '../nodes/knowledge-base/types' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllWorkflowTools, +} from '@/service/use-tools' +import { useStore as useAppStore } from '@/app/components/app/store' +import { AppModeEnum } from '@/types/app' + +export type ChecklistItem = { + id: string + type: BlockEnum | string + title: string + toolIcon?: string | Emoji + unConnected?: boolean + errorMessage?: string + canNavigate: boolean +} + +const START_NODE_TYPES: BlockEnum[] = [ + BlockEnum.Start, + BlockEnum.TriggerSchedule, + BlockEnum.TriggerWebhook, + BlockEnum.TriggerPlugin, +] export const useChecklist = (nodes: Node[], edges: Edge[]) => { const { t } = useTranslation() const language = useGetLanguage() const { nodesMap: nodesExtraData } = useNodesMetaData() - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() const dataSourceList = useStore(s => s.dataSourceList) const { data: strategyProviders } = useStrategyProviders() + const { data: triggerPlugins } = useAllTriggerPlugins() const datasetsDetail = useDatasetsDetailStore(s => s.datasetsDetail) - const { getStartNodes } = useWorkflow() const getToolIcon = useGetToolIcon() + const appMode = useAppStore.getState().appDetail?.mode + const shouldCheckStartNode = appMode === AppModeEnum.WORKFLOW || appMode === AppModeEnum.ADVANCED_CHAT const map = useNodesAvailableVarList(nodes) const { data: embeddingModelList } = useModelList(ModelTypeEnum.textEmbedding) @@ -87,16 +117,10 @@ export const useChecklist = (nodes: Node[], edges: Edge[]) => { return checkData }, [datasetsDetail, embeddingModelList, rerankModelList]) - const needWarningNodes = useMemo(() => { - const list = [] + const needWarningNodes = useMemo<ChecklistItem[]>(() => { + const list: ChecklistItem[] = [] const filteredNodes = nodes.filter(node => node.type === CUSTOM_NODE) - const startNodes = getStartNodes(filteredNodes) - const validNodesFlattened = startNodes.map(startNode => getValidTreeNodes(startNode, filteredNodes, edges)) - const validNodes = validNodesFlattened.reduce((acc, curr) => { - if (curr.validNodes) - acc.push(...curr.validNodes) - return acc - }, [] as Node[]) + const { validNodes } = getValidTreeNodes(filteredNodes, edges) for (let i = 0; i < filteredNodes.length; i++) { const node = filteredNodes[i] @@ -104,11 +128,14 @@ export const useChecklist = (nodes: Node[], edges: Edge[]) => { let usedVars: ValueSelector[] = [] if (node.data.type === BlockEnum.Tool) - moreDataForCheckValid = getToolCheckParams(node.data as ToolNodeType, buildInTools, customTools, workflowTools, language) + moreDataForCheckValid = getToolCheckParams(node.data as ToolNodeType, buildInTools || [], customTools || [], workflowTools || [], language) if (node.data.type === BlockEnum.DataSource) moreDataForCheckValid = getDataSourceCheckParams(node.data as DataSourceNodeType, dataSourceList || [], language) + if (node.data.type === BlockEnum.TriggerPlugin) + moreDataForCheckValid = getTriggerCheckParams(node.data as PluginTriggerNodeType, triggerPlugins, language) + const toolIcon = getToolIcon(node.data) if (node.data.type === BlockEnum.Agent) { const data = node.data as AgentNodeType @@ -128,7 +155,8 @@ export const useChecklist = (nodes: Node[], edges: Edge[]) => { if (node.type === CUSTOM_NODE) { const checkData = getCheckData(node.data) - let { errorMessage } = nodesExtraData![node.data.type].checkValid(checkData, t, moreDataForCheckValid) + const validator = nodesExtraData?.[node.data.type as BlockEnum]?.checkValid + let errorMessage = validator ? validator(checkData, t, moreDataForCheckValid).errorMessage : undefined if (!errorMessage) { const availableVars = map[node.id].availableVars @@ -148,19 +176,43 @@ export const useChecklist = (nodes: Node[], edges: Edge[]) => { } } } - if (errorMessage || !validNodes.find(n => n.id === node.id)) { + + // Start nodes and Trigger nodes should not show unConnected error if they have validation errors + // or if they are valid start nodes (even without incoming connections) + const isStartNodeMeta = nodesExtraData?.[node.data.type as BlockEnum]?.metaData.isStart ?? false + const canSkipConnectionCheck = shouldCheckStartNode ? isStartNodeMeta : true + + const isUnconnected = !validNodes.find(n => n.id === node.id) + const shouldShowError = errorMessage || (isUnconnected && !canSkipConnectionCheck) + + if (shouldShowError) { list.push({ id: node.id, type: node.data.type, title: node.data.title, toolIcon, - unConnected: !validNodes.find(n => n.id === node.id), + unConnected: isUnconnected && !canSkipConnectionCheck, errorMessage, + canNavigate: true, }) } } } + // Check for start nodes (including triggers) + if (shouldCheckStartNode) { + const startNodesFiltered = nodes.filter(node => START_NODE_TYPES.includes(node.data.type as BlockEnum)) + if (startNodesFiltered.length === 0) { + list.push({ + id: 'start-node-required', + type: BlockEnum.Start, + title: t('workflow.panel.startNode'), + errorMessage: t('workflow.common.needStartNode'), + canNavigate: false, + }) + } + } + const isRequiredNodesType = Object.keys(nodesExtraData!).filter((key: any) => (nodesExtraData as any)[key].metaData.isRequired) isRequiredNodesType.forEach((type: string) => { @@ -170,12 +222,13 @@ export const useChecklist = (nodes: Node[], edges: Edge[]) => { type, title: t(`workflow.blocks.${type}`), errorMessage: t('workflow.common.needAdd', { node: t(`workflow.blocks.${type}`) }), + canNavigate: false, }) } }) return list - }, [nodes, getStartNodes, nodesExtraData, edges, buildInTools, customTools, workflowTools, language, dataSourceList, getToolIcon, strategyProviders, getCheckData, t, map]) + }, [nodes, nodesExtraData, edges, buildInTools, customTools, workflowTools, language, dataSourceList, getToolIcon, strategyProviders, getCheckData, t, map, shouldCheckStartNode]) return needWarningNodes } @@ -189,9 +242,15 @@ export const useChecklistBeforePublish = () => { const { data: strategyProviders } = useStrategyProviders() const updateDatasetsDetail = useDatasetsDetailStore(s => s.updateDatasetsDetail) const updateTime = useRef(0) - const { getStartNodes } = useWorkflow() const workflowStore = useWorkflowStore() const { getNodesAvailableVarList } = useGetNodesAvailableVarList() + const { data: embeddingModelList } = useModelList(ModelTypeEnum.textEmbedding) + const { data: rerankModelList } = useModelList(ModelTypeEnum.rerank) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const appMode = useAppStore.getState().appDetail?.mode + const shouldCheckStartNode = appMode === AppModeEnum.WORKFLOW || appMode === AppModeEnum.ADVANCED_CHAT const getCheckData = useCallback((data: CommonNodeType<{}>, datasets: DataSet[]) => { let checkData = data @@ -211,8 +270,15 @@ export const useChecklistBeforePublish = () => { _datasets, } as CommonNodeType<KnowledgeRetrievalNodeType> } + else if (data.type === BlockEnum.KnowledgeBase) { + checkData = { + ...data, + _embeddingModelList: embeddingModelList, + _rerankModelList: rerankModelList, + } as CommonNodeType<KnowledgeBaseNodeType> + } return checkData - }, []) + }, [embeddingModelList, rerankModelList]) const handleCheckBeforePublish = useCallback(async () => { const { @@ -221,26 +287,14 @@ export const useChecklistBeforePublish = () => { } = store.getState() const { dataSourceList, - buildInTools, - customTools, - workflowTools, } = workflowStore.getState() const nodes = getNodes() const filteredNodes = nodes.filter(node => node.type === CUSTOM_NODE) - const startNodes = getStartNodes(filteredNodes) - const validNodesFlattened = startNodes.map(startNode => getValidTreeNodes(startNode, filteredNodes, edges)) - const validNodes = validNodesFlattened.reduce((acc, curr) => { - if (curr.validNodes) - acc.push(...curr.validNodes) - return acc - }, [] as Node[]) - const maxDepthArr = validNodesFlattened.map(item => item.maxDepth) + const { validNodes, maxDepth } = getValidTreeNodes(filteredNodes, edges) - for (let i = 0; i < maxDepthArr.length; i++) { - if (maxDepthArr[i] > MAX_TREE_DEPTH) { - notify({ type: 'error', message: t('workflow.common.maxTreeDepth', { depth: MAX_TREE_DEPTH }) }) - return false - } + if (maxDepth > MAX_TREE_DEPTH) { + notify({ type: 'error', message: t('workflow.common.maxTreeDepth', { depth: MAX_TREE_DEPTH }) }) + return false } // Before publish, we need to fetch datasets detail, in case of the settings of datasets have been changed const knowledgeRetrievalNodes = filteredNodes.filter(node => node.data.type === BlockEnum.KnowledgeRetrieval) @@ -266,7 +320,7 @@ export const useChecklistBeforePublish = () => { let moreDataForCheckValid let usedVars: ValueSelector[] = [] if (node.data.type === BlockEnum.Tool) - moreDataForCheckValid = getToolCheckParams(node.data as ToolNodeType, buildInTools, customTools, workflowTools, language) + moreDataForCheckValid = getToolCheckParams(node.data as ToolNodeType, buildInTools || [], customTools || [], workflowTools || [], language) if (node.data.type === BlockEnum.DataSource) moreDataForCheckValid = getDataSourceCheckParams(node.data as DataSourceNodeType, dataSourceList || [], language) @@ -314,16 +368,29 @@ export const useChecklistBeforePublish = () => { } } - if (!validNodes.find(n => n.id === node.id)) { + const isStartNodeMeta = nodesExtraData?.[node.data.type as BlockEnum]?.metaData.isStart ?? false + const canSkipConnectionCheck = shouldCheckStartNode ? isStartNodeMeta : true + const isUnconnected = !validNodes.find(n => n.id === node.id) + + if (isUnconnected && !canSkipConnectionCheck) { notify({ type: 'error', message: `[${node.data.title}] ${t('workflow.common.needConnectTip')}` }) return false } } + if (shouldCheckStartNode) { + const startNodesFiltered = nodes.filter(node => START_NODE_TYPES.includes(node.data.type as BlockEnum)) + if (startNodesFiltered.length === 0) { + notify({ type: 'error', message: t('workflow.common.needStartNode') }) + return false + } + } + const isRequiredNodesType = Object.keys(nodesExtraData!).filter((key: any) => (nodesExtraData as any)[key].metaData.isRequired) for (let i = 0; i < isRequiredNodesType.length; i++) { const type = isRequiredNodesType[i] + if (!filteredNodes.find(node => node.data.type === type)) { notify({ type: 'error', message: t('workflow.common.needAdd', { node: t(`workflow.blocks.${type}`) }) }) return false @@ -331,9 +398,31 @@ export const useChecklistBeforePublish = () => { } return true - }, [store, notify, t, language, nodesExtraData, strategyProviders, updateDatasetsDetail, getCheckData, getStartNodes, workflowStore]) + }, [store, notify, t, language, nodesExtraData, strategyProviders, updateDatasetsDetail, getCheckData, workflowStore, buildInTools, customTools, workflowTools, shouldCheckStartNode]) return { handleCheckBeforePublish, } } + +export const useWorkflowRunValidation = () => { + const { t } = useTranslation() + const nodes = useNodes<CommonNodeType>() + const edges = useEdges<CommonEdgeType>() + const needWarningNodes = useChecklist(nodes, edges) + const { notify } = useToastContext() + + const validateBeforeRun = useCallback(() => { + if (needWarningNodes.length > 0) { + notify({ type: 'error', message: t('workflow.panel.checklistTip') }) + return false + } + return true + }, [needWarningNodes, notify, t]) + + return { + validateBeforeRun, + hasValidationErrors: needWarningNodes.length > 0, + warningNodes: needWarningNodes, + } +} diff --git a/web/app/components/workflow/hooks/use-dynamic-test-run-options.tsx b/web/app/components/workflow/hooks/use-dynamic-test-run-options.tsx new file mode 100644 index 0000000000..3e35ff0168 --- /dev/null +++ b/web/app/components/workflow/hooks/use-dynamic-test-run-options.tsx @@ -0,0 +1,144 @@ +import { useMemo } from 'react' +import { useNodes } from 'reactflow' +import { useTranslation } from 'react-i18next' +import { BlockEnum, type CommonNodeType } from '../types' +import { getWorkflowEntryNode } from '../utils/workflow-entry' +import { type TestRunOptions, type TriggerOption, TriggerType } from '../header/test-run-menu' +import { TriggerAll } from '@/app/components/base/icons/src/vender/workflow' +import BlockIcon from '../block-icon' +import { useStore } from '../store' +import { useAllTriggerPlugins } from '@/service/use-triggers' + +export const useDynamicTestRunOptions = (): TestRunOptions => { + const { t } = useTranslation() + const nodes = useNodes() + const buildInTools = useStore(s => s.buildInTools) + const customTools = useStore(s => s.customTools) + const workflowTools = useStore(s => s.workflowTools) + const mcpTools = useStore(s => s.mcpTools) + const { data: triggerPlugins } = useAllTriggerPlugins() + + return useMemo(() => { + const allTriggers: TriggerOption[] = [] + let userInput: TriggerOption | undefined + + for (const node of nodes) { + const nodeData = node.data as CommonNodeType + + if (!nodeData?.type) continue + + if (nodeData.type === BlockEnum.Start) { + userInput = { + id: node.id, + type: TriggerType.UserInput, + name: nodeData.title || t('workflow.blocks.start'), + icon: ( + <BlockIcon + type={BlockEnum.Start} + size='md' + /> + ), + nodeId: node.id, + enabled: true, + } + } + else if (nodeData.type === BlockEnum.TriggerSchedule) { + allTriggers.push({ + id: node.id, + type: TriggerType.Schedule, + name: nodeData.title || t('workflow.blocks.trigger-schedule'), + icon: ( + <BlockIcon + type={BlockEnum.TriggerSchedule} + size='md' + /> + ), + nodeId: node.id, + enabled: true, + }) + } + else if (nodeData.type === BlockEnum.TriggerWebhook) { + allTriggers.push({ + id: node.id, + type: TriggerType.Webhook, + name: nodeData.title || t('workflow.blocks.trigger-webhook'), + icon: ( + <BlockIcon + type={BlockEnum.TriggerWebhook} + size='md' + /> + ), + nodeId: node.id, + enabled: true, + }) + } + else if (nodeData.type === BlockEnum.TriggerPlugin) { + let triggerIcon: string | any + + if (nodeData.provider_id) { + const targetTriggers = triggerPlugins || [] + triggerIcon = targetTriggers.find(toolWithProvider => toolWithProvider.name === nodeData.provider_id)?.icon + } + + const icon = ( + <BlockIcon + type={BlockEnum.TriggerPlugin} + size='md' + toolIcon={triggerIcon} + /> + ) + + allTriggers.push({ + id: node.id, + type: TriggerType.Plugin, + name: nodeData.title || (nodeData as any).plugin_name || t('workflow.blocks.trigger-plugin'), + icon, + nodeId: node.id, + enabled: true, + }) + } + } + + if (!userInput) { + const startNode = getWorkflowEntryNode(nodes as any[]) + if (startNode && startNode.data?.type === BlockEnum.Start) { + userInput = { + id: startNode.id, + type: TriggerType.UserInput, + name: (startNode.data as CommonNodeType)?.title || t('workflow.blocks.start'), + icon: ( + <BlockIcon + type={BlockEnum.Start} + size='md' + /> + ), + nodeId: startNode.id, + enabled: true, + } + } + } + + const triggerNodeIds = allTriggers + .map(trigger => trigger.nodeId) + .filter((nodeId): nodeId is string => Boolean(nodeId)) + + const runAll: TriggerOption | undefined = triggerNodeIds.length > 1 ? { + id: 'run-all', + type: TriggerType.All, + name: t('workflow.common.runAllTriggers'), + icon: ( + <div className="flex h-6 w-6 items-center justify-center rounded-lg border-[0.5px] border-white/2 bg-util-colors-purple-purple-500 text-white shadow-md"> + <TriggerAll className="h-4.5 w-4.5" /> + </div> + ), + relatedNodeIds: triggerNodeIds, + enabled: true, + } : undefined + + return { + userInput, + triggers: allTriggers, + runAll, + } + }, [nodes, buildInTools, customTools, workflowTools, mcpTools, triggerPlugins, t]) +} diff --git a/web/app/components/workflow/hooks/use-fetch-workflow-inspect-vars.ts b/web/app/components/workflow/hooks/use-fetch-workflow-inspect-vars.ts index 1527fb82e2..60f839b93d 100644 --- a/web/app/components/workflow/hooks/use-fetch-workflow-inspect-vars.ts +++ b/web/app/components/workflow/hooks/use-fetch-workflow-inspect-vars.ts @@ -11,6 +11,12 @@ import useMatchSchemaType, { getMatchedSchemaType } from '../nodes/_base/compone import { toNodeOutputVars } from '../nodes/_base/components/variable/utils' import type { SchemaTypeDefinition } from '@/service/use-common' import { useCallback } from 'react' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' type Params = { flowType: FlowType @@ -27,17 +33,17 @@ export const useSetWorkflowVarsWithValue = ({ const invalidateSysVarValues = useInvalidateSysVarValues(flowType, flowId) const { handleCancelAllNodeSuccessStatus } = useNodesInteractionsWithoutSync() const { schemaTypeDefinitions } = useMatchSchemaType() - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() const dataSourceList = useStore(s => s.dataSourceList) const allPluginInfoList = { - buildInTools, - customTools, - workflowTools, - mcpTools, - dataSourceList: dataSourceList ?? [], + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], + dataSourceList: dataSourceList || [], } const setInspectVarsToStore = (inspectVars: VarInInspect[], passedInAllPluginInfoList?: Record<string, ToolWithProvider[]>, passedInSchemaTypeDefinitions?: SchemaTypeDefinition[]) => { diff --git a/web/app/components/workflow/hooks/use-helpline.ts b/web/app/components/workflow/hooks/use-helpline.ts index 2eed71a807..55979904fb 100644 --- a/web/app/components/workflow/hooks/use-helpline.ts +++ b/web/app/components/workflow/hooks/use-helpline.ts @@ -1,12 +1,40 @@ import { useCallback } from 'react' import { useStoreApi } from 'reactflow' import type { Node } from '../types' +import { BlockEnum, isTriggerNode } from '../types' import { useWorkflowStore } from '../store' +// Entry node (Start/Trigger) wrapper offsets +// The EntryNodeContainer adds a wrapper with status indicator above the actual node +// These offsets ensure alignment happens on the inner node, not the wrapper +const ENTRY_NODE_WRAPPER_OFFSET = { + x: 0, // No horizontal padding on wrapper (px-0) + y: 21, // Actual measured: pt-0.5 (2px) + status bar height (~19px) +} as const + export const useHelpline = () => { const store = useStoreApi() const workflowStore = useWorkflowStore() + // Check if a node is an entry node (Start or Trigger) + const isEntryNode = useCallback((node: Node): boolean => { + return isTriggerNode(node.data.type as any) || node.data.type === BlockEnum.Start + }, []) + + // Get the actual alignment position of a node (accounting for wrapper offset) + const getNodeAlignPosition = useCallback((node: Node) => { + if (isEntryNode(node)) { + return { + x: node.position.x + ENTRY_NODE_WRAPPER_OFFSET.x, + y: node.position.y + ENTRY_NODE_WRAPPER_OFFSET.y, + } + } + return { + x: node.position.x, + y: node.position.y, + } + }, [isEntryNode]) + const handleSetHelpline = useCallback((node: Node) => { const { getNodes } = store.getState() const nodes = getNodes() @@ -29,6 +57,9 @@ export const useHelpline = () => { } } + // Get the actual alignment position for the dragging node + const nodeAlignPos = getNodeAlignPosition(node) + const showHorizontalHelpLineNodes = nodes.filter((n) => { if (n.id === node.id) return false @@ -39,33 +70,52 @@ export const useHelpline = () => { if (n.data.isInLoop) return false - const nY = Math.ceil(n.position.y) - const nodeY = Math.ceil(node.position.y) + // Get actual alignment position for comparison node + const nAlignPos = getNodeAlignPosition(n) + const nY = Math.ceil(nAlignPos.y) + const nodeY = Math.ceil(nodeAlignPos.y) if (nY - nodeY < 5 && nY - nodeY > -5) return true return false - }).sort((a, b) => a.position.x - b.position.x) + }).sort((a, b) => { + const aPos = getNodeAlignPosition(a) + const bPos = getNodeAlignPosition(b) + return aPos.x - bPos.x + }) const showHorizontalHelpLineNodesLength = showHorizontalHelpLineNodes.length if (showHorizontalHelpLineNodesLength > 0) { const first = showHorizontalHelpLineNodes[0] const last = showHorizontalHelpLineNodes[showHorizontalHelpLineNodesLength - 1] + // Use actual alignment positions for help line rendering + const firstPos = getNodeAlignPosition(first) + const lastPos = getNodeAlignPosition(last) + + // For entry nodes, we need to subtract the offset from width since lastPos already includes it + const lastIsEntryNode = isEntryNode(last) + const lastNodeWidth = lastIsEntryNode ? last.width! - ENTRY_NODE_WRAPPER_OFFSET.x : last.width! + const helpLine = { - top: first.position.y, - left: first.position.x, - width: last.position.x + last.width! - first.position.x, + top: firstPos.y, + left: firstPos.x, + width: lastPos.x + lastNodeWidth - firstPos.x, } - if (node.position.x < first.position.x) { - helpLine.left = node.position.x - helpLine.width = first.position.x + first.width! - node.position.x + if (nodeAlignPos.x < firstPos.x) { + const firstIsEntryNode = isEntryNode(first) + const firstNodeWidth = firstIsEntryNode ? first.width! - ENTRY_NODE_WRAPPER_OFFSET.x : first.width! + helpLine.left = nodeAlignPos.x + helpLine.width = firstPos.x + firstNodeWidth - nodeAlignPos.x } - if (node.position.x > last.position.x) - helpLine.width = node.position.x + node.width! - first.position.x + if (nodeAlignPos.x > lastPos.x) { + const nodeIsEntryNode = isEntryNode(node) + const nodeWidth = nodeIsEntryNode ? node.width! - ENTRY_NODE_WRAPPER_OFFSET.x : node.width! + helpLine.width = nodeAlignPos.x + nodeWidth - firstPos.x + } setHelpLineHorizontal(helpLine) } @@ -81,33 +131,52 @@ export const useHelpline = () => { if (n.data.isInLoop) return false - const nX = Math.ceil(n.position.x) - const nodeX = Math.ceil(node.position.x) + // Get actual alignment position for comparison node + const nAlignPos = getNodeAlignPosition(n) + const nX = Math.ceil(nAlignPos.x) + const nodeX = Math.ceil(nodeAlignPos.x) if (nX - nodeX < 5 && nX - nodeX > -5) return true return false - }).sort((a, b) => a.position.x - b.position.x) + }).sort((a, b) => { + const aPos = getNodeAlignPosition(a) + const bPos = getNodeAlignPosition(b) + return aPos.x - bPos.x + }) const showVerticalHelpLineNodesLength = showVerticalHelpLineNodes.length if (showVerticalHelpLineNodesLength > 0) { const first = showVerticalHelpLineNodes[0] const last = showVerticalHelpLineNodes[showVerticalHelpLineNodesLength - 1] + // Use actual alignment positions for help line rendering + const firstPos = getNodeAlignPosition(first) + const lastPos = getNodeAlignPosition(last) + + // For entry nodes, we need to subtract the offset from height since lastPos already includes it + const lastIsEntryNode = isEntryNode(last) + const lastNodeHeight = lastIsEntryNode ? last.height! - ENTRY_NODE_WRAPPER_OFFSET.y : last.height! + const helpLine = { - top: first.position.y, - left: first.position.x, - height: last.position.y + last.height! - first.position.y, + top: firstPos.y, + left: firstPos.x, + height: lastPos.y + lastNodeHeight - firstPos.y, } - if (node.position.y < first.position.y) { - helpLine.top = node.position.y - helpLine.height = first.position.y + first.height! - node.position.y + if (nodeAlignPos.y < firstPos.y) { + const firstIsEntryNode = isEntryNode(first) + const firstNodeHeight = firstIsEntryNode ? first.height! - ENTRY_NODE_WRAPPER_OFFSET.y : first.height! + helpLine.top = nodeAlignPos.y + helpLine.height = firstPos.y + firstNodeHeight - nodeAlignPos.y } - if (node.position.y > last.position.y) - helpLine.height = node.position.y + node.height! - first.position.y + if (nodeAlignPos.y > lastPos.y) { + const nodeIsEntryNode = isEntryNode(node) + const nodeHeight = nodeIsEntryNode ? node.height! - ENTRY_NODE_WRAPPER_OFFSET.y : node.height! + helpLine.height = nodeAlignPos.y + nodeHeight - firstPos.y + } setHelpLineVertical(helpLine) } @@ -119,7 +188,7 @@ export const useHelpline = () => { showHorizontalHelpLineNodes, showVerticalHelpLineNodes, } - }, [store, workflowStore]) + }, [store, workflowStore, getNodeAlignPosition]) return { handleSetHelpline, diff --git a/web/app/components/workflow/hooks/use-inspect-vars-crud-common.ts b/web/app/components/workflow/hooks/use-inspect-vars-crud-common.ts index f35f0c7dab..6b7acd0a85 100644 --- a/web/app/components/workflow/hooks/use-inspect-vars-crud-common.ts +++ b/web/app/components/workflow/hooks/use-inspect-vars-crud-common.ts @@ -18,6 +18,12 @@ import type { FlowType } from '@/types/common' import useFLow from '@/service/use-flow' import { useStoreApi } from 'reactflow' import type { SchemaTypeDefinition } from '@/service/use-common' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' type Params = { flowId: string @@ -51,6 +57,11 @@ export const useInspectVarsCrudCommon = ({ const { mutateAsync: doEditInspectorVar } = useEditInspectorVar(flowId) const { handleCancelNodeSuccessStatus } = useNodesInteractionsWithoutSync() const { handleEdgeCancelRunningStatus } = useEdgesInteractionsWithoutSync() + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() + const getNodeInspectVars = useCallback((nodeId: string) => { const { nodesWithInspectVars } = workflowStore.getState() const node = nodesWithInspectVars.find(node => node.nodeId === nodeId) @@ -98,10 +109,6 @@ export const useInspectVarsCrudCommon = ({ const fetchInspectVarValue = useCallback(async (selector: ValueSelector, schemaTypeDefinitions: SchemaTypeDefinition[]) => { const { setNodeInspectVars, - buildInTools, - customTools, - workflowTools, - mcpTools, dataSourceList, } = workflowStore.getState() const nodeId = selector[0] @@ -119,11 +126,11 @@ export const useInspectVarsCrudCommon = ({ const nodeArr = getNodes() const currentNode = nodeArr.find(node => node.id === nodeId) const allPluginInfoList = { - buildInTools, - customTools, - workflowTools, - mcpTools, - dataSourceList: dataSourceList ?? [], + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], + dataSourceList: dataSourceList || [], } const currentNodeOutputVars = toNodeOutputVars([currentNode], false, () => true, [], [], [], allPluginInfoList, schemaTypeDefinitions) const vars = await fetchNodeInspectVars(flowType, flowId, nodeId) @@ -135,7 +142,7 @@ export const useInspectVarsCrudCommon = ({ } }) setNodeInspectVars(nodeId, varsWithSchemaType) - }, [workflowStore, flowType, flowId, invalidateSysVarValues, invalidateConversationVarValues]) + }, [workflowStore, flowType, flowId, invalidateSysVarValues, invalidateConversationVarValues, buildInTools, customTools, workflowTools, mcpTools]) // after last run would call this const appendNodeInspectVars = useCallback((nodeId: string, payload: VarInInspect[], allNodes: Node[]) => { diff --git a/web/app/components/workflow/hooks/use-inspect-vars-crud.ts b/web/app/components/workflow/hooks/use-inspect-vars-crud.ts index c922192267..0f58cf8be2 100644 --- a/web/app/components/workflow/hooks/use-inspect-vars-crud.ts +++ b/web/app/components/workflow/hooks/use-inspect-vars-crud.ts @@ -5,13 +5,35 @@ import { useSysVarValues, } from '@/service/use-workflow' import { FlowType } from '@/types/common' +import { produce } from 'immer' +import { BlockEnum } from '../types' +const varsAppendStartNodeKeys = ['query', 'files'] const useInspectVarsCrud = () => { - const nodesWithInspectVars = useStore(s => s.nodesWithInspectVars) + const partOfNodesWithInspectVars = useStore(s => s.nodesWithInspectVars) const configsMap = useHooksStore(s => s.configsMap) const isRagPipeline = configsMap?.flowType === FlowType.ragPipeline const { data: conversationVars } = useConversationVarValues(configsMap?.flowType, !isRagPipeline ? configsMap?.flowId : '') - const { data: systemVars } = useSysVarValues(configsMap?.flowType, !isRagPipeline ? configsMap?.flowId : '') + const { data: allSystemVars } = useSysVarValues(configsMap?.flowType, !isRagPipeline ? configsMap?.flowId : '') + const { varsAppendStartNode, systemVars } = (() => { + if(allSystemVars?.length === 0) + return { varsAppendStartNode: [], systemVars: [] } + const varsAppendStartNode = allSystemVars?.filter(({ name }) => varsAppendStartNodeKeys.includes(name)) || [] + const systemVars = allSystemVars?.filter(({ name }) => !varsAppendStartNodeKeys.includes(name)) || [] + return { varsAppendStartNode, systemVars } + })() + const nodesWithInspectVars = (() => { + if(!partOfNodesWithInspectVars || partOfNodesWithInspectVars.length === 0) + return [] + + const nodesWithInspectVars = produce(partOfNodesWithInspectVars, (draft) => { + draft.forEach((nodeWithVars) => { + if(nodeWithVars.nodeType === BlockEnum.Start) + nodeWithVars.vars = [...nodeWithVars.vars, ...varsAppendStartNode] + }) + }) + return nodesWithInspectVars + })() const hasNodeInspectVars = useHooksStore(s => s.hasNodeInspectVars) const hasSetInspectVar = useHooksStore(s => s.hasSetInspectVar) const fetchInspectVarValue = useHooksStore(s => s.fetchInspectVarValue) diff --git a/web/app/components/workflow/hooks/use-node-data-update.ts b/web/app/components/workflow/hooks/use-node-data-update.ts index edacc31a7c..ac7dca9e4c 100644 --- a/web/app/components/workflow/hooks/use-node-data-update.ts +++ b/web/app/components/workflow/hooks/use-node-data-update.ts @@ -1,6 +1,7 @@ import { useCallback } from 'react' import { produce } from 'immer' import { useStoreApi } from 'reactflow' +import type { SyncCallback } from './use-nodes-sync-draft' import { useNodesSyncDraft } from './use-nodes-sync-draft' import { useNodesReadOnly } from './use-workflow' @@ -28,12 +29,19 @@ export const useNodeDataUpdate = () => { setNodes(newNodes) }, [store]) - const handleNodeDataUpdateWithSyncDraft = useCallback((payload: NodeDataUpdatePayload) => { + const handleNodeDataUpdateWithSyncDraft = useCallback(( + payload: NodeDataUpdatePayload, + options?: { + sync?: boolean + notRefreshWhenSyncError?: boolean + callback?: SyncCallback + }, + ) => { if (getNodesReadOnly()) return handleNodeDataUpdate(payload) - handleSyncWorkflowDraft() + handleSyncWorkflowDraft(options?.sync, options?.notRefreshWhenSyncError, options?.callback) }, [handleSyncWorkflowDraft, handleNodeDataUpdate, getNodesReadOnly]) return { diff --git a/web/app/components/workflow/hooks/use-node-plugin-installation.ts b/web/app/components/workflow/hooks/use-node-plugin-installation.ts new file mode 100644 index 0000000000..96e3919e67 --- /dev/null +++ b/web/app/components/workflow/hooks/use-node-plugin-installation.ts @@ -0,0 +1,218 @@ +import { useCallback, useMemo } from 'react' +import { BlockEnum, type CommonNodeType } from '../types' +import type { ToolNodeType } from '../nodes/tool/types' +import type { PluginTriggerNodeType } from '../nodes/trigger-plugin/types' +import type { DataSourceNodeType } from '../nodes/data-source/types' +import { CollectionType } from '@/app/components/tools/types' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, + useInvalidToolsByType, +} from '@/service/use-tools' +import { + useAllTriggerPlugins, + useInvalidateAllTriggerPlugins, +} from '@/service/use-triggers' +import { useInvalidDataSourceList } from '@/service/use-pipeline' +import { useStore } from '../store' +import { canFindTool } from '@/utils' + +type InstallationState = { + isChecking: boolean + isMissing: boolean + uniqueIdentifier?: string + canInstall: boolean + onInstallSuccess: () => void + shouldDim: boolean +} + +const useToolInstallation = (data: ToolNodeType): InstallationState => { + const builtInQuery = useAllBuiltInTools() + const customQuery = useAllCustomTools() + const workflowQuery = useAllWorkflowTools() + const mcpQuery = useAllMCPTools() + const invalidateTools = useInvalidToolsByType(data.provider_type) + + const collectionInfo = useMemo(() => { + switch (data.provider_type) { + case CollectionType.builtIn: + return { + list: builtInQuery.data, + isLoading: builtInQuery.isLoading, + } + case CollectionType.custom: + return { + list: customQuery.data, + isLoading: customQuery.isLoading, + } + case CollectionType.workflow: + return { + list: workflowQuery.data, + isLoading: workflowQuery.isLoading, + } + case CollectionType.mcp: + return { + list: mcpQuery.data, + isLoading: mcpQuery.isLoading, + } + default: + return undefined + } + }, [ + builtInQuery.data, + builtInQuery.isLoading, + customQuery.data, + customQuery.isLoading, + data.provider_type, + mcpQuery.data, + mcpQuery.isLoading, + workflowQuery.data, + workflowQuery.isLoading, + ]) + + const collection = collectionInfo?.list + const isLoading = collectionInfo?.isLoading ?? false + const isResolved = !!collectionInfo && !isLoading + + const matchedCollection = useMemo(() => { + if (!collection || !collection.length) + return undefined + + return collection.find((toolWithProvider) => { + if (data.plugin_id && toolWithProvider.plugin_id === data.plugin_id) + return true + if (canFindTool(toolWithProvider.id, data.provider_id)) + return true + if (toolWithProvider.name === data.provider_name) + return true + return false + }) + }, [collection, data.plugin_id, data.provider_id, data.provider_name]) + + const uniqueIdentifier = data.plugin_unique_identifier || data.plugin_id || data.provider_id + const canInstall = Boolean(data.plugin_unique_identifier) + + const onInstallSuccess = useCallback(() => { + if (invalidateTools) + invalidateTools() + }, [invalidateTools]) + + const shouldDim = (!!collectionInfo && !isResolved) || (isResolved && !matchedCollection) + + return { + isChecking: !!collectionInfo && !isResolved, + isMissing: isResolved && !matchedCollection, + uniqueIdentifier, + canInstall, + onInstallSuccess, + shouldDim, + } +} + +const useTriggerInstallation = (data: PluginTriggerNodeType): InstallationState => { + const triggerPluginsQuery = useAllTriggerPlugins() + const invalidateTriggers = useInvalidateAllTriggerPlugins() + + const triggerProviders = triggerPluginsQuery.data + const isLoading = triggerPluginsQuery.isLoading + + const matchedProvider = useMemo(() => { + if (!triggerProviders || !triggerProviders.length) + return undefined + + return triggerProviders.find(provider => + provider.name === data.provider_name + || provider.id === data.provider_id + || (data.plugin_id && provider.plugin_id === data.plugin_id), + ) + }, [ + data.plugin_id, + data.provider_id, + data.provider_name, + triggerProviders, + ]) + + const uniqueIdentifier = data.plugin_unique_identifier || data.plugin_id || data.provider_id + const canInstall = Boolean(data.plugin_unique_identifier) + + const onInstallSuccess = useCallback(() => { + invalidateTriggers() + }, [invalidateTriggers]) + + const shouldDim = isLoading || (!isLoading && !!triggerProviders && !matchedProvider) + + return { + isChecking: isLoading, + isMissing: !isLoading && !!triggerProviders && !matchedProvider, + uniqueIdentifier, + canInstall, + onInstallSuccess, + shouldDim, + } +} + +const useDataSourceInstallation = (data: DataSourceNodeType): InstallationState => { + const dataSourceList = useStore(s => s.dataSourceList) + const invalidateDataSourceList = useInvalidDataSourceList() + + const matchedPlugin = useMemo(() => { + if (!dataSourceList || !dataSourceList.length) + return undefined + + return dataSourceList.find((item) => { + if (data.plugin_unique_identifier && item.plugin_unique_identifier === data.plugin_unique_identifier) + return true + if (data.plugin_id && item.plugin_id === data.plugin_id) + return true + if (data.provider_name && item.provider === data.provider_name) + return true + return false + }) + }, [data.plugin_id, data.plugin_unique_identifier, data.provider_name, dataSourceList]) + + const uniqueIdentifier = data.plugin_unique_identifier || data.plugin_id + const canInstall = Boolean(data.plugin_unique_identifier) + + const onInstallSuccess = useCallback(() => { + invalidateDataSourceList() + }, [invalidateDataSourceList]) + + const hasLoadedList = dataSourceList !== undefined + + const shouldDim = !hasLoadedList || (hasLoadedList && !matchedPlugin) + + return { + isChecking: !hasLoadedList, + isMissing: hasLoadedList && !matchedPlugin, + uniqueIdentifier, + canInstall, + onInstallSuccess, + shouldDim, + } +} + +export const useNodePluginInstallation = (data: CommonNodeType): InstallationState => { + const toolInstallation = useToolInstallation(data as ToolNodeType) + const triggerInstallation = useTriggerInstallation(data as PluginTriggerNodeType) + const dataSourceInstallation = useDataSourceInstallation(data as DataSourceNodeType) + + switch (data.type as BlockEnum) { + case BlockEnum.Tool: + return toolInstallation + case BlockEnum.TriggerPlugin: + return triggerInstallation + case BlockEnum.DataSource: + return dataSourceInstallation + default: + return { + isChecking: false, + isMissing: false, + uniqueIdentifier: undefined, + canInstall: false, + onInstallSuccess: () => undefined, + shouldDim: false, + } + } +} diff --git a/web/app/components/workflow/hooks/use-nodes-interactions.ts b/web/app/components/workflow/hooks/use-nodes-interactions.ts index fa61cdeb8c..3cbdf08e43 100644 --- a/web/app/components/workflow/hooks/use-nodes-interactions.ts +++ b/web/app/components/workflow/hooks/use-nodes-interactions.ts @@ -16,9 +16,9 @@ import { useReactFlow, useStoreApi, } from 'reactflow' -import type { DataSourceDefaultValue, ToolDefaultValue } from '../block-selector/types' +import type { PluginDefaultValue } from '../block-selector/types' import type { Edge, Node, OnNodeAdd } from '../types' -import { BlockEnum } from '../types' +import { BlockEnum, isTriggerNode } from '../types' import { useWorkflowStore } from '../store' import { CUSTOM_EDGE, @@ -63,6 +63,15 @@ import type { RAGPipelineVariables } from '@/models/pipeline' import useInspectVarsCrud from './use-inspect-vars-crud' import { getNodeUsedVars } from '../nodes/_base/components/variable/utils' +// Entry node deletion restriction has been removed to allow empty workflows + +// Entry node (Start/Trigger) wrapper offsets for alignment +// Must match the values in use-helpline.ts +const ENTRY_NODE_WRAPPER_OFFSET = { + x: 0, + y: 21, // Adjusted based on visual testing feedback +} as const + export const useNodesInteractions = () => { const { t } = useTranslation() const store = useStoreApi() @@ -138,21 +147,51 @@ export const useNodesInteractions = () => { const newNodes = produce(nodes, (draft) => { const currentNode = draft.find(n => n.id === node.id)! - if (showVerticalHelpLineNodesLength > 0) - currentNode.position.x = showVerticalHelpLineNodes[0].position.x - else if (restrictPosition.x !== undefined) - currentNode.position.x = restrictPosition.x - else if (restrictLoopPosition.x !== undefined) - currentNode.position.x = restrictLoopPosition.x - else currentNode.position.x = node.position.x + // Check if current dragging node is an entry node + const isCurrentEntryNode = isTriggerNode(node.data.type as any) || node.data.type === BlockEnum.Start - if (showHorizontalHelpLineNodesLength > 0) - currentNode.position.y = showHorizontalHelpLineNodes[0].position.y - else if (restrictPosition.y !== undefined) + // X-axis alignment with offset consideration + if (showVerticalHelpLineNodesLength > 0) { + const targetNode = showVerticalHelpLineNodes[0] + const isTargetEntryNode = isTriggerNode(targetNode.data.type as any) || targetNode.data.type === BlockEnum.Start + + // Calculate the wrapper position needed to align the inner nodes + // Target inner position = target.position + target.offset + // Current inner position should equal target inner position + // So: current.position + current.offset = target.position + target.offset + // Therefore: current.position = target.position + target.offset - current.offset + const targetOffset = isTargetEntryNode ? ENTRY_NODE_WRAPPER_OFFSET.x : 0 + const currentOffset = isCurrentEntryNode ? ENTRY_NODE_WRAPPER_OFFSET.x : 0 + currentNode.position.x = targetNode.position.x + targetOffset - currentOffset + } + else if (restrictPosition.x !== undefined) { + currentNode.position.x = restrictPosition.x + } + else if (restrictLoopPosition.x !== undefined) { + currentNode.position.x = restrictLoopPosition.x + } + else { + currentNode.position.x = node.position.x + } + + // Y-axis alignment with offset consideration + if (showHorizontalHelpLineNodesLength > 0) { + const targetNode = showHorizontalHelpLineNodes[0] + const isTargetEntryNode = isTriggerNode(targetNode.data.type as any) || targetNode.data.type === BlockEnum.Start + + const targetOffset = isTargetEntryNode ? ENTRY_NODE_WRAPPER_OFFSET.y : 0 + const currentOffset = isCurrentEntryNode ? ENTRY_NODE_WRAPPER_OFFSET.y : 0 + currentNode.position.y = targetNode.position.y + targetOffset - currentOffset + } + else if (restrictPosition.y !== undefined) { currentNode.position.y = restrictPosition.y - else if (restrictLoopPosition.y !== undefined) + } + else if (restrictLoopPosition.y !== undefined) { currentNode.position.y = restrictLoopPosition.y - else currentNode.position.y = node.position.y + } + else { + currentNode.position.y = node.position.y + } }) setNodes(newNodes) }, @@ -357,6 +396,7 @@ export const useNodesInteractions = () => { if (node.type === CUSTOM_ITERATION_START_NODE) return if (node.type === CUSTOM_LOOP_START_NODE) return if (node.data.type === BlockEnum.DataSourceEmpty) return + if (node.data._pluginInstallLocked) return handleNodeSelect(node.id) }, [handleNodeSelect], @@ -735,7 +775,7 @@ export const useNodesInteractions = () => { nodeType, sourceHandle = 'source', targetHandle = 'target', - toolDefaultValue, + pluginDefaultValue, }, { prevNodeId, prevNodeSourceHandle, nextNodeId, nextNodeTargetHandle }, ) => { @@ -756,7 +796,7 @@ export const useNodesInteractions = () => { nodesWithSameType.length > 0 ? `${defaultValue.title} ${nodesWithSameType.length + 1}` : defaultValue.title, - ...toolDefaultValue, + ...pluginDefaultValue, selected: true, _showAddVariablePopup: (nodeType === BlockEnum.VariableAssigner @@ -1286,7 +1326,7 @@ export const useNodesInteractions = () => { currentNodeId: string, nodeType: BlockEnum, sourceHandle: string, - toolDefaultValue?: ToolDefaultValue | DataSourceDefaultValue, + pluginDefaultValue?: PluginDefaultValue, ) => { if (getNodesReadOnly()) return @@ -1310,7 +1350,7 @@ export const useNodesInteractions = () => { nodesWithSameType.length > 0 ? `${defaultValue.title} ${nodesWithSameType.length + 1}` : defaultValue.title, - ...toolDefaultValue, + ...pluginDefaultValue, _connectedSourceHandleIds: [], _connectedTargetHandleIds: [], selected: currentNode.data.selected, @@ -1445,6 +1485,7 @@ export const useNodesInteractions = () => { // If no nodeId is provided, fall back to the current behavior const bundledNodes = nodes.filter((node) => { if (!node.data._isBundled) return false + if (node.type === CUSTOM_NOTE_NODE) return true const { metaData } = nodesMetaDataMap![node.data.type as BlockEnum] if (metaData.isSingleton) return false return !node.data.isInIteration && !node.data.isInLoop @@ -1457,6 +1498,7 @@ export const useNodesInteractions = () => { const selectedNode = nodes.find((node) => { if (!node.data.selected) return false + if (node.type === CUSTOM_NOTE_NODE) return true const { metaData } = nodesMetaDataMap![node.data.type as BlockEnum] return !metaData.isSingleton }) @@ -1495,7 +1537,7 @@ export const useNodesInteractions = () => { = generateNewNode({ type: nodeToPaste.type, data: { - ...nodesMetaDataMap![nodeType].defaultValue, + ...(nodeToPaste.type !== CUSTOM_NOTE_NODE && nodesMetaDataMap![nodeType].defaultValue), ...nodeToPaste.data, selected: false, _isBundled: false, @@ -1654,7 +1696,7 @@ export const useNodesInteractions = () => { const nodes = getNodes() const bundledNodes = nodes.filter( - node => node.data._isBundled && node.data.type !== BlockEnum.Start, + node => node.data._isBundled, ) if (bundledNodes.length) { @@ -1667,7 +1709,7 @@ export const useNodesInteractions = () => { if (edgeSelected) return const selectedNode = nodes.find( - node => node.data.selected && node.data.type !== BlockEnum.Start, + node => node.data.selected, ) if (selectedNode) handleNodeDelete(selectedNode.id) diff --git a/web/app/components/workflow/hooks/use-nodes-meta-data.ts b/web/app/components/workflow/hooks/use-nodes-meta-data.ts index cfeb41de34..fd63f23590 100644 --- a/web/app/components/workflow/hooks/use-nodes-meta-data.ts +++ b/web/app/components/workflow/hooks/use-nodes-meta-data.ts @@ -7,6 +7,11 @@ import { CollectionType } from '@/app/components/tools/types' import { useStore } from '@/app/components/workflow/store' import { canFindTool } from '@/utils' import { useGetLanguage } from '@/context/i18n' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllWorkflowTools, +} from '@/service/use-tools' export const useNodesMetaData = () => { const availableNodesMetaData = useHooksStore(s => s.availableNodesMetaData) @@ -21,9 +26,9 @@ export const useNodesMetaData = () => { export const useNodeMetaData = (node: Node) => { const language = useGetLanguage() - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() const dataSourceList = useStore(s => s.dataSourceList) const availableNodesMetaData = useNodesMetaData() const { data } = node @@ -34,10 +39,10 @@ export const useNodeMetaData = (node: Node) => { if (data.type === BlockEnum.Tool) { if (data.provider_type === CollectionType.builtIn) - return buildInTools.find(toolWithProvider => canFindTool(toolWithProvider.id, data.provider_id))?.author + return buildInTools?.find(toolWithProvider => canFindTool(toolWithProvider.id, data.provider_id))?.author if (data.provider_type === CollectionType.workflow) - return workflowTools.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.author - return customTools.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.author + return workflowTools?.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.author + return customTools?.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.author } return nodeMetaData?.metaData.author }, [data, buildInTools, customTools, workflowTools, nodeMetaData, dataSourceList]) @@ -47,10 +52,10 @@ export const useNodeMetaData = (node: Node) => { return dataSourceList?.find(dataSource => dataSource.plugin_id === data.plugin_id)?.description[language] if (data.type === BlockEnum.Tool) { if (data.provider_type === CollectionType.builtIn) - return buildInTools.find(toolWithProvider => canFindTool(toolWithProvider.id, data.provider_id))?.description[language] + return buildInTools?.find(toolWithProvider => canFindTool(toolWithProvider.id, data.provider_id))?.description[language] if (data.provider_type === CollectionType.workflow) - return workflowTools.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.description[language] - return customTools.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.description[language] + return workflowTools?.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.description[language] + return customTools?.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.description[language] } return nodeMetaData?.metaData.description }, [data, buildInTools, customTools, workflowTools, nodeMetaData, dataSourceList, language]) diff --git a/web/app/components/workflow/hooks/use-nodes-sync-draft.ts b/web/app/components/workflow/hooks/use-nodes-sync-draft.ts index e6cc3a97e3..a4c9a45542 100644 --- a/web/app/components/workflow/hooks/use-nodes-sync-draft.ts +++ b/web/app/components/workflow/hooks/use-nodes-sync-draft.ts @@ -1,12 +1,14 @@ import { useCallback } from 'react' -import { - useStore, -} from '../store' -import { - useNodesReadOnly, -} from './use-workflow' +import { useStore } from '../store' +import { useNodesReadOnly } from './use-workflow' import { useHooksStore } from '@/app/components/workflow/hooks-store' +export type SyncCallback = { + onSuccess?: () => void + onError?: () => void + onSettled?: () => void +} + export const useNodesSyncDraft = () => { const { getNodesReadOnly } = useNodesReadOnly() const debouncedSyncWorkflowDraft = useStore(s => s.debouncedSyncWorkflowDraft) @@ -16,11 +18,7 @@ export const useNodesSyncDraft = () => { const handleSyncWorkflowDraft = useCallback(( sync?: boolean, notRefreshWhenSyncError?: boolean, - callback?: { - onSuccess?: () => void - onError?: () => void - onSettled?: () => void - }, + callback?: SyncCallback, ) => { if (getNodesReadOnly()) return diff --git a/web/app/components/workflow/hooks/use-serial-async-callback.ts b/web/app/components/workflow/hooks/use-serial-async-callback.ts new file mode 100644 index 0000000000..c36409a776 --- /dev/null +++ b/web/app/components/workflow/hooks/use-serial-async-callback.ts @@ -0,0 +1,22 @@ +import { + useCallback, + useRef, +} from 'react' + +export const useSerialAsyncCallback = <Args extends any[], Result = void>( + fn: (...args: Args) => Promise<Result> | Result, + shouldSkip?: () => boolean, +) => { + const queueRef = useRef<Promise<unknown>>(Promise.resolve()) + + return useCallback((...args: Args) => { + if (shouldSkip?.()) + return Promise.resolve(undefined as Result) + + const lastPromise = queueRef.current.catch(() => undefined) + const nextPromise = lastPromise.then(() => fn(...args)) + queueRef.current = nextPromise + + return nextPromise + }, [fn, shouldSkip]) +} diff --git a/web/app/components/workflow/hooks/use-shortcuts.ts b/web/app/components/workflow/hooks/use-shortcuts.ts index a744fefd50..e8c69ca9b5 100644 --- a/web/app/components/workflow/hooks/use-shortcuts.ts +++ b/web/app/components/workflow/hooks/use-shortcuts.ts @@ -14,7 +14,6 @@ import { useWorkflowCanvasMaximize, useWorkflowMoveMode, useWorkflowOrganize, - useWorkflowStartRun, } from '.' export const useShortcuts = (): void => { @@ -28,7 +27,6 @@ export const useShortcuts = (): void => { dimOtherNodes, undimAllNodes, } = useNodesInteractions() - const { handleStartWorkflowRun } = useWorkflowStartRun() const { shortcutsEnabled: workflowHistoryShortcutsEnabled } = useWorkflowHistoryStore() const { handleSyncWorkflowDraft } = useNodesSyncDraft() const { handleEdgeDelete } = useEdgesInteractions() @@ -46,24 +44,23 @@ export const useShortcuts = (): void => { fitView, } = useReactFlow() - // Zoom out to a minimum of 0.5 for shortcut + // Zoom out to a minimum of 0.25 for shortcut const constrainedZoomOut = () => { const currentZoom = getZoom() - const newZoom = Math.max(currentZoom - 0.1, 0.5) + const newZoom = Math.max(currentZoom - 0.1, 0.25) zoomTo(newZoom) } - // Zoom in to a maximum of 1 for shortcut + // Zoom in to a maximum of 2 for shortcut const constrainedZoomIn = () => { const currentZoom = getZoom() - const newZoom = Math.min(currentZoom + 0.1, 1) + const newZoom = Math.min(currentZoom + 0.1, 2) zoomTo(newZoom) } const shouldHandleShortcut = useCallback((e: KeyboardEvent) => { - const { showFeaturesPanel } = workflowStore.getState() - return !showFeaturesPanel && !isEventTargetInputArea(e.target as HTMLElement) - }, [workflowStore]) + return !isEventTargetInputArea(e.target as HTMLElement) + }, []) useKeyPress(['delete', 'backspace'], (e) => { if (shouldHandleShortcut(e)) { @@ -99,7 +96,11 @@ export const useShortcuts = (): void => { useKeyPress(`${getKeyboardKeyCodeBySystem('alt')}.r`, (e) => { if (shouldHandleShortcut(e)) { e.preventDefault() - handleStartWorkflowRun() + // @ts-expect-error - Dynamic property added by run-and-history component + if (window._toggleTestRunDropdown) { + // @ts-expect-error - Dynamic property added by run-and-history component + window._toggleTestRunDropdown() + } } }, { exactMatch: true, useCapture: true }) diff --git a/web/app/components/workflow/hooks/use-tool-icon.ts b/web/app/components/workflow/hooks/use-tool-icon.ts index 734a7da390..8276989ee3 100644 --- a/web/app/components/workflow/hooks/use-tool-icon.ts +++ b/web/app/components/workflow/hooks/use-tool-icon.ts @@ -1,75 +1,202 @@ -import { - useCallback, - useMemo, -} from 'react' -import type { - Node, -} from '../types' -import { - BlockEnum, -} from '../types' -import { - useStore, - useWorkflowStore, -} from '../store' +import { useCallback, useMemo } from 'react' +import type { Node, ToolWithProvider } from '../types' +import { BlockEnum } from '../types' +import { useStore, useWorkflowStore } from '../store' import { CollectionType } from '@/app/components/tools/types' import { canFindTool } from '@/utils' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' +import { useAllTriggerPlugins } from '@/service/use-triggers' +import type { PluginTriggerNodeType } from '../nodes/trigger-plugin/types' +import type { ToolNodeType } from '../nodes/tool/types' +import type { DataSourceNodeType } from '../nodes/data-source/types' +import type { TriggerWithProvider } from '../block-selector/types' + +const isTriggerPluginNode = (data: Node['data']): data is PluginTriggerNodeType => data.type === BlockEnum.TriggerPlugin + +const isToolNode = (data: Node['data']): data is ToolNodeType => data.type === BlockEnum.Tool + +const isDataSourceNode = (data: Node['data']): data is DataSourceNodeType => data.type === BlockEnum.DataSource + +const findTriggerPluginIcon = ( + identifiers: (string | undefined)[], + triggers: TriggerWithProvider[] | undefined, +) => { + const targetTriggers = triggers || [] + for (const identifier of identifiers) { + if (!identifier) + continue + const matched = targetTriggers.find(trigger => trigger.id === identifier || canFindTool(trigger.id, identifier)) + if (matched?.icon) + return matched.icon + } + return undefined +} export const useToolIcon = (data?: Node['data']) => { - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() const dataSourceList = useStore(s => s.dataSourceList) - // const a = useStore(s => s.data) + const { data: triggerPlugins } = useAllTriggerPlugins() + const toolIcon = useMemo(() => { if (!data) return '' - if (data.type === BlockEnum.Tool) { - // eslint-disable-next-line sonarjs/no-dead-store - let targetTools = buildInTools - if (data.provider_type === CollectionType.builtIn) - targetTools = buildInTools - else if (data.provider_type === CollectionType.custom) - targetTools = customTools - else if (data.provider_type === CollectionType.mcp) - targetTools = mcpTools - else - targetTools = workflowTools - return targetTools.find(toolWithProvider => canFindTool(toolWithProvider.id, data.provider_id))?.icon + + if (isTriggerPluginNode(data)) { + const icon = findTriggerPluginIcon( + [ + data.plugin_id, + data.provider_id, + data.provider_name, + ], + triggerPlugins, + ) + if (icon) + return icon } - if (data.type === BlockEnum.DataSource) - return dataSourceList?.find(toolWithProvider => toolWithProvider.plugin_id === data.plugin_id)?.icon - }, [data, dataSourceList, buildInTools, customTools, mcpTools, workflowTools]) + + if (isToolNode(data)) { + let primaryCollection: ToolWithProvider[] | undefined + switch (data.provider_type) { + case CollectionType.custom: + primaryCollection = customTools + break + case CollectionType.mcp: + primaryCollection = mcpTools + break + case CollectionType.workflow: + primaryCollection = workflowTools + break + case CollectionType.builtIn: + default: + primaryCollection = buildInTools + break + } + + const collectionsToSearch = [ + primaryCollection, + buildInTools, + customTools, + workflowTools, + mcpTools, + ] as Array<ToolWithProvider[] | undefined> + + const seen = new Set<ToolWithProvider[]>() + for (const collection of collectionsToSearch) { + if (!collection || seen.has(collection)) + continue + seen.add(collection) + const matched = collection.find((toolWithProvider) => { + if (canFindTool(toolWithProvider.id, data.provider_id)) + return true + if (data.plugin_id && toolWithProvider.plugin_id === data.plugin_id) + return true + return data.provider_name === toolWithProvider.name + }) + if (matched?.icon) + return matched.icon + } + + if (data.provider_icon) + return data.provider_icon + + return '' + } + + if (isDataSourceNode(data)) + return dataSourceList?.find(toolWithProvider => toolWithProvider.plugin_id === data.plugin_id)?.icon || '' + + return '' + }, [data, dataSourceList, buildInTools, customTools, workflowTools, mcpTools, triggerPlugins]) return toolIcon } export const useGetToolIcon = () => { + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() + const { data: triggerPlugins } = useAllTriggerPlugins() const workflowStore = useWorkflowStore() + const getToolIcon = useCallback((data: Node['data']) => { const { - buildInTools, - customTools, - workflowTools, + buildInTools: storeBuiltInTools, + customTools: storeCustomTools, + workflowTools: storeWorkflowTools, + mcpTools: storeMcpTools, dataSourceList, } = workflowStore.getState() - if (data.type === BlockEnum.Tool) { - // eslint-disable-next-line sonarjs/no-dead-store - let targetTools = buildInTools - if (data.provider_type === CollectionType.builtIn) - targetTools = buildInTools - else if (data.provider_type === CollectionType.custom) - targetTools = customTools - else - targetTools = workflowTools - return targetTools.find(toolWithProvider => canFindTool(toolWithProvider.id, data.provider_id))?.icon + if (isTriggerPluginNode(data)) { + return findTriggerPluginIcon( + [ + data.plugin_id, + data.provider_id, + data.provider_name, + ], + triggerPlugins, + ) } - if (data.type === BlockEnum.DataSource) + if (isToolNode(data)) { + const primaryCollection = (() => { + switch (data.provider_type) { + case CollectionType.custom: + return storeCustomTools ?? customTools + case CollectionType.mcp: + return storeMcpTools ?? mcpTools + case CollectionType.workflow: + return storeWorkflowTools ?? workflowTools + case CollectionType.builtIn: + default: + return storeBuiltInTools ?? buildInTools + } + })() + + const collectionsToSearch = [ + primaryCollection, + storeBuiltInTools ?? buildInTools, + storeCustomTools ?? customTools, + storeWorkflowTools ?? workflowTools, + storeMcpTools ?? mcpTools, + ] as Array<ToolWithProvider[] | undefined> + + const seen = new Set<ToolWithProvider[]>() + for (const collection of collectionsToSearch) { + if (!collection || seen.has(collection)) + continue + seen.add(collection) + const matched = collection.find((toolWithProvider) => { + if (canFindTool(toolWithProvider.id, data.provider_id)) + return true + if (data.plugin_id && toolWithProvider.plugin_id === data.plugin_id) + return true + return data.provider_name === toolWithProvider.name + }) + if (matched?.icon) + return matched.icon + } + + if (data.provider_icon) + return data.provider_icon + + return undefined + } + + if (isDataSourceNode(data)) return dataSourceList?.find(toolWithProvider => toolWithProvider.plugin_id === data.plugin_id)?.icon - }, [workflowStore]) + + return undefined + }, [workflowStore, triggerPlugins, buildInTools, customTools, workflowTools, mcpTools]) return getToolIcon } diff --git a/web/app/components/workflow/hooks/use-workflow-interactions.ts b/web/app/components/workflow/hooks/use-workflow-interactions.ts index c080d6279e..e56c39d51e 100644 --- a/web/app/components/workflow/hooks/use-workflow-interactions.ts +++ b/web/app/components/workflow/hooks/use-workflow-interactions.ts @@ -316,7 +316,10 @@ export const useWorkflowUpdate = () => { edges: initialEdges(edges, nodes), }, } as any) - setViewport(viewport) + + // Only set viewport if it exists and is valid + if (viewport && typeof viewport.x === 'number' && typeof viewport.y === 'number' && typeof viewport.zoom === 'number') + setViewport(viewport) }, [eventEmitter, reactflow]) return { diff --git a/web/app/components/workflow/hooks/use-workflow-search.tsx b/web/app/components/workflow/hooks/use-workflow-search.tsx index 095ae4577a..68ad9873f9 100644 --- a/web/app/components/workflow/hooks/use-workflow-search.tsx +++ b/web/app/components/workflow/hooks/use-workflow-search.tsx @@ -8,11 +8,16 @@ import { workflowNodesAction } from '@/app/components/goto-anything/actions/work import BlockIcon from '@/app/components/workflow/block-icon' import { setupNodeSelectionListener } from '../utils/node-navigation' import { BlockEnum } from '../types' -import { useStore } from '../store' import type { Emoji } from '@/app/components/tools/types' import { CollectionType } from '@/app/components/tools/types' import { canFindTool } from '@/utils' import type { LLMNodeType } from '../nodes/llm/types' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' /** * Hook to register workflow nodes search functionality @@ -22,23 +27,23 @@ export const useWorkflowSearch = () => { const { handleNodeSelect } = useNodesInteractions() // Filter and process nodes for search - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() // Extract tool icon logic - clean separation of concerns const getToolIcon = useCallback((nodeData: CommonNodeType): string | Emoji | undefined => { if (nodeData?.type !== BlockEnum.Tool) return undefined const toolCollections: Record<string, any[]> = { - [CollectionType.builtIn]: buildInTools, - [CollectionType.custom]: customTools, - [CollectionType.mcp]: mcpTools, + [CollectionType.builtIn]: buildInTools || [], + [CollectionType.custom]: customTools || [], + [CollectionType.mcp]: mcpTools || [], } const targetTools = (nodeData.provider_type && toolCollections[nodeData.provider_type]) || workflowTools - return targetTools.find((tool: any) => canFindTool(tool.id, nodeData.provider_id))?.icon + return targetTools?.find((tool: any) => canFindTool(tool.id, nodeData.provider_id))?.icon }, [buildInTools, customTools, workflowTools, mcpTools]) // Extract model info logic - clean extraction diff --git a/web/app/components/workflow/hooks/use-workflow-start-run.tsx b/web/app/components/workflow/hooks/use-workflow-start-run.tsx index 0f4e68fe95..46fe5649c8 100644 --- a/web/app/components/workflow/hooks/use-workflow-start-run.tsx +++ b/web/app/components/workflow/hooks/use-workflow-start-run.tsx @@ -4,10 +4,17 @@ export const useWorkflowStartRun = () => { const handleStartWorkflowRun = useHooksStore(s => s.handleStartWorkflowRun) const handleWorkflowStartRunInWorkflow = useHooksStore(s => s.handleWorkflowStartRunInWorkflow) const handleWorkflowStartRunInChatflow = useHooksStore(s => s.handleWorkflowStartRunInChatflow) - + const handleWorkflowTriggerScheduleRunInWorkflow = useHooksStore(s => s.handleWorkflowTriggerScheduleRunInWorkflow) + const handleWorkflowTriggerWebhookRunInWorkflow = useHooksStore(s => s.handleWorkflowTriggerWebhookRunInWorkflow) + const handleWorkflowTriggerPluginRunInWorkflow = useHooksStore(s => s.handleWorkflowTriggerPluginRunInWorkflow) + const handleWorkflowRunAllTriggersInWorkflow = useHooksStore(s => s.handleWorkflowRunAllTriggersInWorkflow) return { handleStartWorkflowRun, handleWorkflowStartRunInWorkflow, handleWorkflowStartRunInChatflow, + handleWorkflowTriggerScheduleRunInWorkflow, + handleWorkflowTriggerWebhookRunInWorkflow, + handleWorkflowTriggerPluginRunInWorkflow, + handleWorkflowRunAllTriggersInWorkflow, } } diff --git a/web/app/components/workflow/hooks/use-workflow-variables.ts b/web/app/components/workflow/hooks/use-workflow-variables.ts index 8422a7fd0d..871937365a 100644 --- a/web/app/components/workflow/hooks/use-workflow-variables.ts +++ b/web/app/components/workflow/hooks/use-workflow-variables.ts @@ -10,20 +10,25 @@ import type { } from '@/app/components/workflow/types' import { useIsChatMode } from './use-workflow' import { useStoreApi } from 'reactflow' -import { useStore } from '@/app/components/workflow/store' import type { Type } from '../nodes/llm/types' import useMatchSchemaType from '../nodes/_base/components/variable/use-match-schema-type' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' export const useWorkflowVariables = () => { const { t } = useTranslation() const workflowStore = useWorkflowStore() const { schemaTypeDefinitions } = useMatchSchemaType() - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) - const dataSourceList = useStore(s => s.dataSourceList) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() + const getNodeAvailableVars = useCallback(({ parentNode, beforeNodes, @@ -43,6 +48,7 @@ export const useWorkflowVariables = () => { conversationVariables, environmentVariables, ragPipelineVariables, + dataSourceList, } = workflowStore.getState() return toNodeAvailableVars({ parentNode, @@ -54,15 +60,15 @@ export const useWorkflowVariables = () => { ragVariables: ragPipelineVariables, filterVar, allPluginInfoList: { - buildInTools, - customTools, - workflowTools, - mcpTools, - dataSourceList: dataSourceList ?? [], + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], + dataSourceList: dataSourceList || [], }, schemaTypeDefinitions, }) - }, [t, workflowStore, schemaTypeDefinitions, buildInTools]) + }, [t, workflowStore, schemaTypeDefinitions, buildInTools, customTools, workflowTools, mcpTools]) const getCurrentVariableType = useCallback(({ parentNode, @@ -87,10 +93,6 @@ export const useWorkflowVariables = () => { conversationVariables, environmentVariables, ragPipelineVariables, - buildInTools, - customTools, - workflowTools, - mcpTools, dataSourceList, } = workflowStore.getState() return getVarType({ @@ -105,16 +107,16 @@ export const useWorkflowVariables = () => { conversationVariables, ragVariables: ragPipelineVariables, allPluginInfoList: { - buildInTools, - customTools, - workflowTools, - mcpTools, + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], dataSourceList: dataSourceList ?? [], }, schemaTypeDefinitions, preferSchemaType, }) - }, [workflowStore, getVarType, schemaTypeDefinitions]) + }, [workflowStore, getVarType, schemaTypeDefinitions, buildInTools, customTools, workflowTools, mcpTools]) return { getNodeAvailableVars, diff --git a/web/app/components/workflow/hooks/use-workflow.ts b/web/app/components/workflow/hooks/use-workflow.ts index 3f9f8106cf..e6746085b8 100644 --- a/web/app/components/workflow/hooks/use-workflow.ts +++ b/web/app/components/workflow/hooks/use-workflow.ts @@ -23,6 +23,10 @@ import { useStore, useWorkflowStore, } from '../store' +import { + getWorkflowEntryNode, + isWorkflowEntryNode, +} from '../utils/workflow-entry' import { SUPPORT_OUTPUT_VARS_NODE, } from '../constants' @@ -32,21 +36,16 @@ import { CUSTOM_NOTE_NODE } from '../note-node/constants' import { findUsedVarNodes, getNodeOutputVars, updateNodeVars } from '../nodes/_base/components/variable/utils' import { useAvailableBlocks } from './use-available-blocks' import { useStore as useAppStore } from '@/app/components/app/store' -import { - fetchAllBuiltInTools, - fetchAllCustomTools, - fetchAllMCPTools, - fetchAllWorkflowTools, -} from '@/service/tools' + import { CUSTOM_ITERATION_START_NODE } from '@/app/components/workflow/nodes/iteration-start/constants' import { CUSTOM_LOOP_START_NODE } from '@/app/components/workflow/nodes/loop-start/constants' -import { basePath } from '@/utils/var' import { useNodesMetaData } from '.' +import { AppModeEnum } from '@/types/app' export const useIsChatMode = () => { const appDetail = useAppStore(s => s.appDetail) - return appDetail?.mode === 'advanced-chat' + return appDetail?.mode === AppModeEnum.ADVANCED_CHAT } export const useWorkflow = () => { @@ -69,6 +68,7 @@ export const useWorkflow = () => { edges, } = store.getState() const nodes = getNodes() + // let startNode = getWorkflowEntryNode(nodes) const currentNode = nodes.find(node => node.id === nodeId) let startNodes = nodes.filter(node => nodesMap?.[node.data.type as BlockEnum]?.metaData.isStart) || [] @@ -238,6 +238,33 @@ export const useWorkflow = () => { return nodes.filter(node => node.parentId === nodeId) }, [store]) + const isFromStartNode = useCallback((nodeId: string) => { + const { getNodes } = store.getState() + const nodes = getNodes() + const currentNode = nodes.find(node => node.id === nodeId) + + if (!currentNode) + return false + + if (isWorkflowEntryNode(currentNode.data.type)) + return true + + const checkPreviousNodes = (node: Node) => { + const previousNodes = getBeforeNodeById(node.id) + + for (const prevNode of previousNodes) { + if (isWorkflowEntryNode(prevNode.data.type)) + return true + if (checkPreviousNodes(prevNode)) + return true + } + + return false + } + + return checkPreviousNodes(currentNode) + }, [store, getBeforeNodeById]) + const handleOutVarRenameChange = useCallback((nodeId: string, oldValeSelector: ValueSelector, newVarSelector: ValueSelector) => { const { getNodes, setNodes } = store.getState() const allNodes = getNodes() @@ -397,6 +424,13 @@ export const useWorkflow = () => { return !hasCycle(targetNode) }, [store, getAvailableBlocks]) + const getNode = useCallback((nodeId?: string) => { + const { getNodes } = store.getState() + const nodes = getNodes() + + return nodes.find(node => node.id === nodeId) || getWorkflowEntryNode(nodes) + }, [store]) + return { getNodeById, getTreeLeafNodes, @@ -413,51 +447,8 @@ export const useWorkflow = () => { getLoopNodeChildren, getRootNodesById, getStartNodes, - } -} - -export const useFetchToolsData = () => { - const workflowStore = useWorkflowStore() - - const handleFetchAllTools = useCallback(async (type: string) => { - if (type === 'builtin') { - const buildInTools = await fetchAllBuiltInTools() - - if (basePath) { - buildInTools.forEach((item) => { - if (typeof item.icon == 'string' && !item.icon.includes(basePath)) - item.icon = `${basePath}${item.icon}` - }) - } - workflowStore.setState({ - buildInTools: buildInTools || [], - }) - } - if (type === 'custom') { - const customTools = await fetchAllCustomTools() - - workflowStore.setState({ - customTools: customTools || [], - }) - } - if (type === 'workflow') { - const workflowTools = await fetchAllWorkflowTools() - - workflowStore.setState({ - workflowTools: workflowTools || [], - }) - } - if (type === 'mcp') { - const mcpTools = await fetchAllMCPTools() - - workflowStore.setState({ - mcpTools: mcpTools || [], - }) - } - }, [workflowStore]) - - return { - handleFetchAllTools, + isFromStartNode, + getNode, } } @@ -481,14 +472,14 @@ export const useNodesReadOnly = () => { const historyWorkflowData = useStore(s => s.historyWorkflowData) const isRestoring = useStore(s => s.isRestoring) - const getNodesReadOnly = useCallback(() => { + const getNodesReadOnly = useCallback((): boolean => { const { workflowRunningData, historyWorkflowData, isRestoring, } = workflowStore.getState() - return workflowRunningData?.result.status === WorkflowRunningStatus.Running || historyWorkflowData || isRestoring + return !!(workflowRunningData?.result.status === WorkflowRunningStatus.Running || historyWorkflowData || isRestoring) }, [workflowStore]) return { diff --git a/web/app/components/workflow/index.tsx b/web/app/components/workflow/index.tsx index b289cafefd..86c6bf153e 100644 --- a/web/app/components/workflow/index.tsx +++ b/web/app/components/workflow/index.tsx @@ -37,7 +37,6 @@ import { } from './types' import { useEdgesInteractions, - useFetchToolsData, useNodesInteractions, useNodesReadOnly, useNodesSyncDraft, @@ -92,6 +91,12 @@ import useMatchSchemaType from './nodes/_base/components/variable/use-match-sche import type { VarInInspect } from '@/types/workflow' import { fetchAllInspectVars } from '@/service/workflow' import cn from '@/utils/classnames' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' const Confirm = dynamic(() => import('@/app/components/base/confirm'), { ssr: false, @@ -242,13 +247,6 @@ export const Workflow: FC<WorkflowProps> = memo(({ }) } }) - const { handleFetchAllTools } = useFetchToolsData() - useEffect(() => { - handleFetchAllTools('builtin') - handleFetchAllTools('custom') - handleFetchAllTools('workflow') - handleFetchAllTools('mcp') - }, [handleFetchAllTools]) const { handleNodeDragStart, @@ -299,10 +297,10 @@ export const Workflow: FC<WorkflowProps> = memo(({ const { schemaTypeDefinitions } = useMatchSchemaType() const { fetchInspectVars } = useSetWorkflowVarsWithValue() - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() const dataSourceList = useStore(s => s.dataSourceList) // buildInTools, customTools, workflowTools, mcpTools, dataSourceList const configsMap = useHooksStore(s => s.configsMap) @@ -323,10 +321,10 @@ export const Workflow: FC<WorkflowProps> = memo(({ passInVars: true, vars, passedInAllPluginInfoList: { - buildInTools, - customTools, - workflowTools, - mcpTools, + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], dataSourceList: dataSourceList ?? [], }, passedInSchemaTypeDefinitions: schemaTypeDefinitions, diff --git a/web/app/components/workflow/nodes/_base/components/agent-strategy-selector.tsx b/web/app/components/workflow/nodes/_base/components/agent-strategy-selector.tsx index 0c24dcfd2c..fe6266dea3 100644 --- a/web/app/components/workflow/nodes/_base/components/agent-strategy-selector.tsx +++ b/web/app/components/workflow/nodes/_base/components/agent-strategy-selector.tsx @@ -12,7 +12,7 @@ import SearchInput from '@/app/components/base/search-input' import Tools from '../../../block-selector/tools' import { useTranslation } from 'react-i18next' import { useStrategyProviders } from '@/service/use-strategy' -import { PluginType, type StrategyPluginDetail } from '@/app/components/plugins/types' +import { PluginCategoryEnum, type StrategyPluginDetail } from '@/app/components/plugins/types' import type { ToolWithProvider } from '../../../types' import { CollectionType } from '@/app/components/tools/types' import useGetIcon from '@/app/components/plugins/install-plugin/base/use-get-icon' @@ -140,7 +140,7 @@ export const AgentStrategySelector = memo((props: AgentStrategySelectorProps) => if (query) { fetchPlugins({ query, - category: PluginType.agent, + category: PluginCategoryEnum.agent, }) } }, [query]) diff --git a/web/app/components/workflow/nodes/_base/components/agent-strategy.tsx b/web/app/components/workflow/nodes/_base/components/agent-strategy.tsx index b447c3f70e..4b15e57d5c 100644 --- a/web/app/components/workflow/nodes/_base/components/agent-strategy.tsx +++ b/web/app/components/workflow/nodes/_base/components/agent-strategy.tsx @@ -22,6 +22,7 @@ import type { Node } from 'reactflow' import type { PluginMeta } from '@/app/components/plugins/types' import { noop } from 'lodash-es' import { useDocLink } from '@/context/i18n' +import { AppModeEnum } from '@/types/app' export type Strategy = { agent_strategy_provider_name: string @@ -99,7 +100,7 @@ export const AgentStrategy = memo((props: AgentStrategyProps) => { modelConfig={ defaultModel.data ? { - mode: 'chat', + mode: AppModeEnum.CHAT, name: defaultModel.data.model, provider: defaultModel.data.provider.provider, completion_params: {}, diff --git a/web/app/components/workflow/nodes/_base/components/before-run-form/form-item.tsx b/web/app/components/workflow/nodes/_base/components/before-run-form/form-item.tsx index 041c8dab28..0bb8b3844f 100644 --- a/web/app/components/workflow/nodes/_base/components/before-run-form/form-item.tsx +++ b/web/app/components/workflow/nodes/_base/components/before-run-form/form-item.tsx @@ -140,7 +140,7 @@ const FormItem: FC<Props> = ({ <Input value={value || ''} onChange={e => onChange(e.target.value)} - placeholder={t('appDebug.variableConfig.inputPlaceholder')!} + placeholder={typeof payload.label === 'object' ? payload.label.variable : payload.label} autoFocus={autoFocus} /> ) @@ -152,7 +152,7 @@ const FormItem: FC<Props> = ({ type="number" value={value || ''} onChange={e => onChange(e.target.value)} - placeholder={t('appDebug.variableConfig.inputPlaceholder')!} + placeholder={typeof payload.label === 'object' ? payload.label.variable : payload.label} autoFocus={autoFocus} /> ) @@ -163,7 +163,7 @@ const FormItem: FC<Props> = ({ <Textarea value={value || ''} onChange={e => onChange(e.target.value)} - placeholder={t('appDebug.variableConfig.inputPlaceholder')!} + placeholder={typeof payload.label === 'object' ? payload.label.variable : payload.label} autoFocus={autoFocus} /> ) diff --git a/web/app/components/workflow/nodes/_base/components/code-generator-button.tsx b/web/app/components/workflow/nodes/_base/components/code-generator-button.tsx index a62ffeb55f..21b1cf0595 100644 --- a/web/app/components/workflow/nodes/_base/components/code-generator-button.tsx +++ b/web/app/components/workflow/nodes/_base/components/code-generator-button.tsx @@ -6,7 +6,7 @@ import cn from 'classnames' import type { CodeLanguage } from '../../code/types' import { Generator } from '@/app/components/base/icons/src/vender/other' import { ActionButton } from '@/app/components/base/action-button' -import { AppType } from '@/types/app' +import { AppModeEnum } from '@/types/app' import type { GenRes } from '@/service/debug' import { GetCodeGeneratorResModal } from '@/app/components/app/configuration/config/code-generator/get-code-generator-res' import { useHooksStore } from '../../../hooks-store' @@ -42,7 +42,7 @@ const CodeGenerateBtn: FC<Props> = ({ </ActionButton> {showAutomatic && ( <GetCodeGeneratorResModal - mode={AppType.chat} + mode={AppModeEnum.CHAT} isShow={showAutomatic} codeLanguages={codeLanguages} onClose={showAutomaticFalse} diff --git a/web/app/components/workflow/nodes/_base/components/entry-node-container.tsx b/web/app/components/workflow/nodes/_base/components/entry-node-container.tsx new file mode 100644 index 0000000000..b0cecdd0ae --- /dev/null +++ b/web/app/components/workflow/nodes/_base/components/entry-node-container.tsx @@ -0,0 +1,40 @@ +import type { FC, ReactNode } from 'react' +import { useMemo } from 'react' +import { useTranslation } from 'react-i18next' + +export enum StartNodeTypeEnum { + Start = 'start', + Trigger = 'trigger', +} + +type EntryNodeContainerProps = { + children: ReactNode + customLabel?: string + nodeType?: StartNodeTypeEnum +} + +const EntryNodeContainer: FC<EntryNodeContainerProps> = ({ + children, + customLabel, + nodeType = StartNodeTypeEnum.Trigger, +}) => { + const { t } = useTranslation() + + const label = useMemo(() => { + const translationKey = nodeType === StartNodeTypeEnum.Start ? 'entryNodeStatus' : 'triggerStatus' + return customLabel || t(`workflow.${translationKey}.enabled`) + }, [customLabel, nodeType, t]) + + return ( + <div className="w-fit min-w-[242px] rounded-2xl bg-workflow-block-wrapper-bg-1 px-0 pb-0 pt-0.5"> + <div className="mb-0.5 flex items-center px-1.5 pt-0.5"> + <span className="text-2xs font-semibold uppercase text-text-tertiary"> + {label} + </span> + </div> + {children} + </div> + ) +} + +export default EntryNodeContainer diff --git a/web/app/components/workflow/nodes/_base/components/form-input-item.tsx b/web/app/components/workflow/nodes/_base/components/form-input-item.tsx index 70212a8581..14a0f19317 100644 --- a/web/app/components/workflow/nodes/_base/components/form-input-item.tsx +++ b/web/app/components/workflow/nodes/_base/components/form-input-item.tsx @@ -1,38 +1,50 @@ 'use client' import type { FC } from 'react' -import type { ToolVarInputs } from '@/app/components/workflow/nodes/tool/types' -import type { CredentialFormSchema } from '@/app/components/header/account-setting/model-provider-page/declarations' +import { useEffect, useMemo, useState } from 'react' +import { type ResourceVarInputs, VarKindType } from '../types' +import type { CredentialFormSchema, FormOption } from '@/app/components/header/account-setting/model-provider-page/declarations' import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' import { FormTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' -import { VarType as VarKindType } from '@/app/components/workflow/nodes/tool/types' import { VarType } from '@/app/components/workflow/types' +import { useFetchDynamicOptions } from '@/service/use-plugins' +import { useTriggerPluginDynamicOptions } from '@/service/use-triggers' import type { ToolWithProvider, ValueSelector, Var } from '@/app/components/workflow/types' +import type { TriggerWithProvider } from '@/app/components/workflow/block-selector/types' +import type { Tool } from '@/app/components/tools/types' import FormInputTypeSwitch from './form-input-type-switch' import useAvailableVarList from '@/app/components/workflow/nodes/_base/hooks/use-available-var-list' import Input from '@/app/components/base/input' import { SimpleSelect } from '@/app/components/base/select' import MixedVariableTextInput from '@/app/components/workflow/nodes/tool/components/mixed-variable-text-input' -import FormInputBoolean from './form-input-boolean' import AppSelector from '@/app/components/plugins/plugin-detail-panel/app-selector' import ModelParameterModal from '@/app/components/plugins/plugin-detail-panel/model-selector' import VarReferencePicker from '@/app/components/workflow/nodes/_base/components/variable/var-reference-picker' import CodeEditor from '@/app/components/workflow/nodes/_base/components/editor/code-editor' import { CodeLanguage } from '@/app/components/workflow/nodes/code/types' import cn from '@/utils/classnames' -import type { Tool } from '@/app/components/tools/types' +import { Listbox, ListboxButton, ListboxOption, ListboxOptions } from '@headlessui/react' +import { ChevronDownIcon } from '@heroicons/react/20/solid' +import { RiCheckLine, RiLoader4Line } from '@remixicon/react' +import type { Event } from '@/app/components/tools/types' +import { PluginCategoryEnum } from '@/app/components/plugins/types' +import CheckboxList from '@/app/components/base/checkbox-list' +import FormInputBoolean from './form-input-boolean' type Props = { readOnly: boolean nodeId: string schema: CredentialFormSchema - value: ToolVarInputs + value: ResourceVarInputs onChange: (value: any) => void inPanel?: boolean - currentTool?: Tool - currentProvider?: ToolWithProvider + currentTool?: Tool | Event + currentProvider?: ToolWithProvider | TriggerWithProvider showManageInputField?: boolean onManageInputField?: () => void + extraParams?: Record<string, any> + providerType?: string + disableVariableInsertion?: boolean } const FormInputItem: FC<Props> = ({ @@ -46,15 +58,22 @@ const FormInputItem: FC<Props> = ({ currentProvider, showManageInputField, onManageInputField, + extraParams, + providerType, + disableVariableInsertion = false, }) => { const language = useLanguage() + const [toolsOptions, setToolsOptions] = useState<FormOption[] | null>(null) + const [isLoadingToolsOptions, setIsLoadingToolsOptions] = useState(false) const { placeholder, variable, type, + _type, default: defaultValue, options, + multiple, scope, } = schema as any const varInput = value[variable] @@ -64,13 +83,16 @@ const FormInputItem: FC<Props> = ({ const isArray = type === FormTypeEnum.array const isShowJSONEditor = isObject || isArray const isFile = type === FormTypeEnum.file || type === FormTypeEnum.files - const isBoolean = type === FormTypeEnum.boolean - const isSelect = type === FormTypeEnum.select || type === FormTypeEnum.dynamicSelect + const isBoolean = _type === FormTypeEnum.boolean + const isCheckbox = _type === FormTypeEnum.checkbox + const isSelect = type === FormTypeEnum.select + const isDynamicSelect = type === FormTypeEnum.dynamicSelect const isAppSelector = type === FormTypeEnum.appSelector const isModelSelector = type === FormTypeEnum.modelSelector const showTypeSwitch = isNumber || isBoolean || isObject || isArray || isSelect const isConstant = varInput?.type === VarKindType.constant || !varInput?.type const showVariableSelector = isFile || varInput?.type === VarKindType.variable + const isMultipleSelect = multiple && (isSelect || isDynamicSelect) const { availableVars, availableNodesWithParent } = useAvailableVarList(nodeId, { onlyLeafNodeVar: false, @@ -123,12 +145,71 @@ const FormInputItem: FC<Props> = ({ const getVarKindType = () => { if (isFile) return VarKindType.variable - if (isSelect || isBoolean || isNumber || isArray || isObject) + if (isSelect || isDynamicSelect || isBoolean || isNumber || isArray || isObject) return VarKindType.constant if (isString) return VarKindType.mixed } + // Fetch dynamic options hook for tools + const { mutateAsync: fetchDynamicOptions } = useFetchDynamicOptions( + currentProvider?.plugin_id || '', + currentProvider?.name || '', + currentTool?.name || '', + variable || '', + providerType, + extraParams, + ) + + // Fetch dynamic options hook for triggers + const { data: triggerDynamicOptions, isLoading: isTriggerOptionsLoading } = useTriggerPluginDynamicOptions({ + plugin_id: currentProvider?.plugin_id || '', + provider: currentProvider?.name || '', + action: currentTool?.name || '', + parameter: variable || '', + extra: extraParams, + credential_id: currentProvider?.credential_id || '', + }, isDynamicSelect && providerType === PluginCategoryEnum.trigger && !!currentTool && !!currentProvider) + + // Computed values for dynamic options (unified for triggers and tools) + const triggerOptions = triggerDynamicOptions?.options + const dynamicOptions = providerType === PluginCategoryEnum.trigger + ? triggerOptions ?? toolsOptions + : toolsOptions + const isLoadingOptions = providerType === PluginCategoryEnum.trigger + ? (isTriggerOptionsLoading || isLoadingToolsOptions) + : isLoadingToolsOptions + + // Fetch dynamic options for tools only (triggers use hook directly) + useEffect(() => { + const fetchPanelDynamicOptions = async () => { + if (isDynamicSelect && currentTool && currentProvider && (providerType === PluginCategoryEnum.tool || providerType === PluginCategoryEnum.trigger)) { + setIsLoadingToolsOptions(true) + try { + const data = await fetchDynamicOptions() + setToolsOptions(data?.options || []) + } + catch (error) { + console.error('Failed to fetch dynamic options:', error) + setToolsOptions([]) + } + finally { + setIsLoadingToolsOptions(false) + } + } + } + + fetchPanelDynamicOptions() + }, [ + isDynamicSelect, + currentTool?.name, + currentProvider?.name, + variable, + extraParams, + providerType, + fetchDynamicOptions, + ]) + const handleTypeChange = (newType: string) => { if (newType === VarKindType.variable) { onChange({ @@ -163,6 +244,24 @@ const FormInputItem: FC<Props> = ({ }) } + const getSelectedLabels = (selectedValues: any[]) => { + if (!selectedValues || selectedValues.length === 0) + return '' + + const optionsList = isDynamicSelect ? (dynamicOptions || options || []) : (options || []) + const selectedOptions = optionsList.filter((opt: any) => + selectedValues.includes(opt.value), + ) + + if (selectedOptions.length <= 2) { + return selectedOptions + .map((opt: any) => opt.label?.[language] || opt.label?.en_US || opt.value) + .join(', ') + } + + return `${selectedOptions.length} selected` + } + const handleAppOrModelSelect = (newValue: any) => { onChange({ ...value, @@ -184,6 +283,45 @@ const FormInputItem: FC<Props> = ({ }) } + const availableCheckboxOptions = useMemo(() => ( + (options || []).filter((option: { show_on?: Array<{ variable: string; value: any }> }) => { + if (option.show_on?.length) + return option.show_on.every(showOnItem => value[showOnItem.variable]?.value === showOnItem.value || value[showOnItem.variable] === showOnItem.value) + return true + }) + ), [options, value]) + + const checkboxListOptions = useMemo(() => ( + availableCheckboxOptions.map((option: { value: string; label: Record<string, string> }) => ({ + value: option.value, + label: option.label?.[language] || option.label?.en_US || option.value, + })) + ), [availableCheckboxOptions, language]) + + const checkboxListValue = useMemo(() => { + let current: string[] = [] + if (Array.isArray(varInput?.value)) + current = varInput.value as string[] + else if (typeof varInput?.value === 'string') + current = [varInput.value as string] + else if (Array.isArray(defaultValue)) + current = defaultValue as string[] + + const allowedValues = new Set(availableCheckboxOptions.map((option: { value: string }) => option.value)) + return current.filter(item => allowedValues.has(item)) + }, [varInput?.value, defaultValue, availableCheckboxOptions]) + + const handleCheckboxListChange = (selected: string[]) => { + onChange({ + ...value, + [variable]: { + ...varInput, + type: VarKindType.constant, + value: selected, + }, + }) + } + return ( <div className={cn('gap-1', !(isShowJSONEditor && isConstant) && 'flex')}> {showTypeSwitch && ( @@ -198,6 +336,7 @@ const FormInputItem: FC<Props> = ({ availableNodes={availableNodesWithParent} showManageInputField={showManageInputField} onManageInputField={onManageInputField} + disableVariableInsertion={disableVariableInsertion} /> )} {isNumber && isConstant && ( @@ -209,13 +348,23 @@ const FormInputItem: FC<Props> = ({ placeholder={placeholder?.[language] || placeholder?.en_US} /> )} + {isCheckbox && isConstant && ( + <CheckboxList + title={schema.label?.[language] || schema.label?.en_US || variable} + value={checkboxListValue} + onChange={handleCheckboxListChange} + options={checkboxListOptions} + disabled={readOnly} + maxHeight='200px' + /> + )} {isBoolean && isConstant && ( <FormInputBoolean value={varInput?.value as boolean} onChange={handleValueChange} /> )} - {isSelect && isConstant && ( + {isSelect && isConstant && !isMultipleSelect && ( <SimpleSelect wrapperClassName='h-8 grow' disabled={readOnly} @@ -225,11 +374,175 @@ const FormInputItem: FC<Props> = ({ return option.show_on.every(showOnItem => value[showOnItem.variable] === showOnItem.value) return true - }).map((option: { value: any; label: { [x: string]: any; en_US: any } }) => ({ value: option.value, name: option.label[language] || option.label.en_US }))} + }).map((option: { value: any; label: { [x: string]: any; en_US: any }; icon?: string }) => ({ + value: option.value, + name: option.label[language] || option.label.en_US, + icon: option.icon, + }))} onSelect={item => handleValueChange(item.value as string)} placeholder={placeholder?.[language] || placeholder?.en_US} + renderOption={options.some((opt: any) => opt.icon) ? ({ item }) => ( + <div className="flex items-center"> + {item.icon && ( + <img src={item.icon} alt="" className="mr-2 h-4 w-4" /> + )} + <span>{item.name}</span> + </div> + ) : undefined} /> )} + {isSelect && isConstant && isMultipleSelect && ( + <Listbox + multiple + value={varInput?.value || []} + onChange={handleValueChange} + disabled={readOnly} + > + <div className="group/simple-select relative h-8 grow"> + <ListboxButton className="flex h-full w-full cursor-pointer items-center rounded-lg border-0 bg-components-input-bg-normal pl-3 pr-10 focus-visible:bg-state-base-hover-alt focus-visible:outline-none group-hover/simple-select:bg-state-base-hover-alt sm:text-sm sm:leading-6"> + <span className={cn('system-sm-regular block truncate text-left', + varInput?.value?.length > 0 ? 'text-components-input-text-filled' : 'text-components-input-text-placeholder', + )}> + {getSelectedLabels(varInput?.value) || placeholder?.[language] || placeholder?.en_US || 'Select options'} + </span> + <span className="absolute inset-y-0 right-0 flex items-center pr-2"> + <ChevronDownIcon + className="h-4 w-4 text-text-quaternary group-hover/simple-select:text-text-secondary" + aria-hidden="true" + /> + </span> + </ListboxButton> + <ListboxOptions className="absolute z-10 mt-1 max-h-60 w-full overflow-auto rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur px-1 py-1 text-base shadow-lg backdrop-blur-sm focus:outline-none sm:text-sm"> + {options.filter((option: { show_on: any[] }) => { + if (option.show_on?.length) + return option.show_on.every(showOnItem => value[showOnItem.variable] === showOnItem.value) + return true + }).map((option: { value: any; label: { [x: string]: any; en_US: any }; icon?: string }) => ( + <ListboxOption + key={option.value} + value={option.value} + className={({ focus }) => + cn('relative cursor-pointer select-none rounded-lg py-2 pl-3 pr-9 text-text-secondary hover:bg-state-base-hover', + focus && 'bg-state-base-hover', + ) + } + > + {({ selected }) => ( + <> + <div className="flex items-center"> + {option.icon && ( + <img src={option.icon} alt="" className="mr-2 h-4 w-4" /> + )} + <span className={cn('block truncate', selected && 'font-normal')}> + {option.label[language] || option.label.en_US} + </span> + </div> + {selected && ( + <span className="absolute inset-y-0 right-0 flex items-center pr-2 text-text-accent"> + <RiCheckLine className="h-4 w-4" aria-hidden="true" /> + </span> + )} + </> + )} + </ListboxOption> + ))} + </ListboxOptions> + </div> + </Listbox> + )} + {isDynamicSelect && !isMultipleSelect && ( + <SimpleSelect + wrapperClassName='h-8 grow' + disabled={readOnly || isLoadingOptions} + defaultValue={varInput?.value} + items={(dynamicOptions || options || []).filter((option: { show_on?: any[] }) => { + if (option.show_on?.length) + return option.show_on.every(showOnItem => value[showOnItem.variable] === showOnItem.value) + + return true + }).map((option: { value: any; label: { [x: string]: any; en_US: any }; icon?: string }) => ({ + value: option.value, + name: option.label[language] || option.label.en_US, + icon: option.icon, + }))} + onSelect={item => handleValueChange(item.value as string)} + placeholder={isLoadingOptions ? 'Loading...' : (placeholder?.[language] || placeholder?.en_US)} + renderOption={({ item }) => ( + <div className="flex items-center"> + {item.icon && ( + <img src={item.icon} alt="" className="mr-2 h-4 w-4" /> + )} + <span>{item.name}</span> + </div> + )} + /> + )} + {isDynamicSelect && isMultipleSelect && ( + <Listbox + multiple + value={varInput?.value || []} + onChange={handleValueChange} + disabled={readOnly || isLoadingOptions} + > + <div className="group/simple-select relative h-8 grow"> + <ListboxButton className="flex h-full w-full cursor-pointer items-center rounded-lg border-0 bg-components-input-bg-normal pl-3 pr-10 focus-visible:bg-state-base-hover-alt focus-visible:outline-none group-hover/simple-select:bg-state-base-hover-alt sm:text-sm sm:leading-6"> + <span className={cn('system-sm-regular block truncate text-left', + isLoadingOptions ? 'text-components-input-text-placeholder' + : varInput?.value?.length > 0 ? 'text-components-input-text-filled' : 'text-components-input-text-placeholder', + )}> + {isLoadingOptions + ? 'Loading...' + : getSelectedLabels(varInput?.value) || placeholder?.[language] || placeholder?.en_US || 'Select options'} + </span> + <span className="absolute inset-y-0 right-0 flex items-center pr-2"> + {isLoadingOptions ? ( + <RiLoader4Line className='h-3.5 w-3.5 animate-spin text-text-secondary' /> + ) : ( + <ChevronDownIcon + className="h-4 w-4 text-text-quaternary group-hover/simple-select:text-text-secondary" + aria-hidden="true" + /> + )} + </span> + </ListboxButton> + <ListboxOptions className="absolute z-10 mt-1 max-h-60 w-full overflow-auto rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur px-1 py-1 text-base shadow-lg backdrop-blur-sm focus:outline-none sm:text-sm"> + {(dynamicOptions || options || []).filter((option: { show_on?: any[] }) => { + if (option.show_on?.length) + return option.show_on.every(showOnItem => value[showOnItem.variable] === showOnItem.value) + return true + }).map((option: { value: any; label: { [x: string]: any; en_US: any }; icon?: string }) => ( + <ListboxOption + key={option.value} + value={option.value} + className={({ focus }) => + cn('relative cursor-pointer select-none rounded-lg py-2 pl-3 pr-9 text-text-secondary hover:bg-state-base-hover', + focus && 'bg-state-base-hover', + ) + } + > + {({ selected }) => ( + <> + <div className="flex items-center"> + {option.icon && ( + <img src={option.icon} alt="" className="mr-2 h-4 w-4" /> + )} + <span className={cn('block truncate', selected && 'font-normal')}> + {option.label[language] || option.label.en_US} + </span> + </div> + {selected && ( + <span className="absolute inset-y-0 right-0 flex items-center pr-2 text-text-accent"> + <RiCheckLine className="h-4 w-4" aria-hidden="true" /> + </span> + )} + </> + )} + </ListboxOption> + ))} + </ListboxOptions> + </div> + </Listbox> + )} {isShowJSONEditor && isConstant && ( <div className='mt-1 w-full'> <CodeEditor diff --git a/web/app/components/workflow/nodes/_base/components/install-plugin-button.tsx b/web/app/components/workflow/nodes/_base/components/install-plugin-button.tsx index 23119f0213..b0d878d53d 100644 --- a/web/app/components/workflow/nodes/_base/components/install-plugin-button.tsx +++ b/web/app/components/workflow/nodes/_base/components/install-plugin-button.tsx @@ -1,37 +1,96 @@ import Button from '@/app/components/base/button' import { RiInstallLine, RiLoader2Line } from '@remixicon/react' import type { ComponentProps, MouseEventHandler } from 'react' +import { useState } from 'react' import classNames from '@/utils/classnames' import { useTranslation } from 'react-i18next' +import checkTaskStatus from '@/app/components/plugins/install-plugin/base/check-task-status' +import { TaskStatus } from '@/app/components/plugins/types' import { useCheckInstalled, useInstallPackageFromMarketPlace } from '@/service/use-plugins' type InstallPluginButtonProps = Omit<ComponentProps<typeof Button>, 'children' | 'loading'> & { uniqueIdentifier: string + extraIdentifiers?: string[] onSuccess?: () => void } export const InstallPluginButton = (props: InstallPluginButtonProps) => { - const { className, uniqueIdentifier, onSuccess, ...rest } = props + const { + className, + uniqueIdentifier, + extraIdentifiers = [], + onSuccess, + ...rest + } = props const { t } = useTranslation() + const identifiers = Array.from(new Set( + [uniqueIdentifier, ...extraIdentifiers].filter((item): item is string => Boolean(item)), + )) const manifest = useCheckInstalled({ - pluginIds: [uniqueIdentifier], - enabled: !!uniqueIdentifier, + pluginIds: identifiers, + enabled: identifiers.length > 0, }) const install = useInstallPackageFromMarketPlace() - const isLoading = manifest.isLoading || install.isPending - // await for refetch to get the new installed plugin, when manifest refetch, this component will unmount - || install.isSuccess + const [isTracking, setIsTracking] = useState(false) + const isLoading = manifest.isLoading || install.isPending || isTracking const handleInstall: MouseEventHandler = (e) => { e.stopPropagation() + if (isLoading) + return + setIsTracking(true) install.mutate(uniqueIdentifier, { - onSuccess: async () => { - await manifest.refetch() - onSuccess?.() + onSuccess: async (response) => { + const finish = async () => { + await manifest.refetch() + onSuccess?.() + setIsTracking(false) + install.reset() + } + + if (!response) { + await finish() + return + } + + if (response.all_installed) { + await finish() + return + } + + const { check } = checkTaskStatus() + try { + const { status } = await check({ + taskId: response.task_id, + pluginUniqueIdentifier: uniqueIdentifier, + }) + + if (status === TaskStatus.failed) { + setIsTracking(false) + install.reset() + return + } + + await finish() + } + catch { + setIsTracking(false) + install.reset() + } + }, + onError: () => { + setIsTracking(false) + install.reset() }, }) } if (!manifest.data) return null - if (manifest.data.plugins.some(plugin => plugin.id === uniqueIdentifier)) return null + const identifierSet = new Set(identifiers) + const isInstalled = manifest.data.plugins.some(plugin => ( + identifierSet.has(plugin.id) + || (plugin.plugin_unique_identifier && identifierSet.has(plugin.plugin_unique_identifier)) + || (plugin.plugin_id && identifierSet.has(plugin.plugin_id)) + )) + if (isInstalled) return null return <Button variant={'secondary'} disabled={isLoading} diff --git a/web/app/components/workflow/nodes/_base/components/mixed-variable-text-input/index.tsx b/web/app/components/workflow/nodes/_base/components/mixed-variable-text-input/index.tsx new file mode 100644 index 0000000000..6680c8ebb6 --- /dev/null +++ b/web/app/components/workflow/nodes/_base/components/mixed-variable-text-input/index.tsx @@ -0,0 +1,62 @@ +import { + memo, +} from 'react' +import { useTranslation } from 'react-i18next' +import PromptEditor from '@/app/components/base/prompt-editor' +import Placeholder from './placeholder' +import type { + Node, + NodeOutPutVar, +} from '@/app/components/workflow/types' +import { BlockEnum } from '@/app/components/workflow/types' +import cn from '@/utils/classnames' + +type MixedVariableTextInputProps = { + readOnly?: boolean + nodesOutputVars?: NodeOutPutVar[] + availableNodes?: Node[] + value?: string + onChange?: (text: string) => void +} +const MixedVariableTextInput = ({ + readOnly = false, + nodesOutputVars, + availableNodes = [], + value = '', + onChange, +}: MixedVariableTextInputProps) => { + const { t } = useTranslation() + return ( + <PromptEditor + wrapperClassName={cn( + 'w-full rounded-lg border border-transparent bg-components-input-bg-normal px-2 py-1', + 'hover:border-components-input-border-hover hover:bg-components-input-bg-hover', + 'focus-within:border-components-input-border-active focus-within:bg-components-input-bg-active focus-within:shadow-xs', + )} + className='caret:text-text-accent' + editable={!readOnly} + value={value} + workflowVariableBlock={{ + show: true, + variables: nodesOutputVars || [], + workflowNodesMap: availableNodes.reduce((acc, node) => { + acc[node.id] = { + title: node.data.title, + type: node.data.type, + } + if (node.data.type === BlockEnum.Start) { + acc.sys = { + title: t('workflow.blocks.start'), + type: BlockEnum.Start, + } + } + return acc + }, {} as any), + }} + placeholder={<Placeholder />} + onChange={onChange} + /> + ) +} + +export default memo(MixedVariableTextInput) diff --git a/web/app/components/workflow/nodes/_base/components/mixed-variable-text-input/placeholder.tsx b/web/app/components/workflow/nodes/_base/components/mixed-variable-text-input/placeholder.tsx new file mode 100644 index 0000000000..75d4c91996 --- /dev/null +++ b/web/app/components/workflow/nodes/_base/components/mixed-variable-text-input/placeholder.tsx @@ -0,0 +1,52 @@ +import { useCallback } from 'react' +import { useTranslation } from 'react-i18next' +import { useLexicalComposerContext } from '@lexical/react/LexicalComposerContext' +import { FOCUS_COMMAND } from 'lexical' +import { $insertNodes } from 'lexical' +import { CustomTextNode } from '@/app/components/base/prompt-editor/plugins/custom-text/node' +import Badge from '@/app/components/base/badge' + +const Placeholder = () => { + const { t } = useTranslation() + const [editor] = useLexicalComposerContext() + + const handleInsert = useCallback((text: string) => { + editor.update(() => { + const textNode = new CustomTextNode(text) + $insertNodes([textNode]) + }) + editor.dispatchCommand(FOCUS_COMMAND, undefined as any) + }, [editor]) + + return ( + <div + className='pointer-events-auto flex h-full w-full cursor-text items-center px-2' + onClick={(e) => { + e.stopPropagation() + handleInsert('') + }} + > + <div className='flex grow items-center'> + {t('workflow.nodes.tool.insertPlaceholder1')} + <div className='system-kbd mx-0.5 flex h-4 w-4 items-center justify-center rounded bg-components-kbd-bg-gray text-text-placeholder'>/</div> + <div + className='system-sm-regular cursor-pointer text-components-input-text-placeholder underline decoration-dotted decoration-auto underline-offset-auto hover:text-text-tertiary' + onMouseDown={((e) => { + e.preventDefault() + e.stopPropagation() + handleInsert('/') + })} + > + {t('workflow.nodes.tool.insertPlaceholder2')} + </div> + </div> + <Badge + className='shrink-0' + text='String' + uppercase={false} + /> + </div> + ) +} + +export default Placeholder diff --git a/web/app/components/workflow/nodes/_base/components/next-step/add.tsx b/web/app/components/workflow/nodes/_base/components/next-step/add.tsx index 601bc8ea75..3001274c31 100644 --- a/web/app/components/workflow/nodes/_base/components/next-step/add.tsx +++ b/web/app/components/workflow/nodes/_base/components/next-step/add.tsx @@ -39,11 +39,11 @@ const Add = ({ const { nodesReadOnly } = useNodesReadOnly() const { availableNextBlocks } = useAvailableBlocks(nodeData.type, nodeData.isInIteration || nodeData.isInLoop) - const handleSelect = useCallback<OnSelectBlock>((type, toolDefaultValue) => { + const handleSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { handleNodeAdd( { nodeType: type, - toolDefaultValue, + pluginDefaultValue, }, { prevNodeId: nodeId, diff --git a/web/app/components/workflow/nodes/_base/components/next-step/operator.tsx b/web/app/components/workflow/nodes/_base/components/next-step/operator.tsx index c54a63d8ad..7143e6fe43 100644 --- a/web/app/components/workflow/nodes/_base/components/next-step/operator.tsx +++ b/web/app/components/workflow/nodes/_base/components/next-step/operator.tsx @@ -38,8 +38,8 @@ const ChangeItem = ({ availableNextBlocks, } = useAvailableBlocks(data.type, data.isInIteration || data.isInLoop) - const handleSelect = useCallback<OnSelectBlock>((type, toolDefaultValue) => { - handleNodeChange(nodeId, type, sourceHandle, toolDefaultValue) + const handleSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { + handleNodeChange(nodeId, type, sourceHandle, pluginDefaultValue) }, [nodeId, sourceHandle, handleNodeChange]) const renderTrigger = useCallback(() => { diff --git a/web/app/components/workflow/nodes/_base/components/node-control.tsx b/web/app/components/workflow/nodes/_base/components/node-control.tsx index 0e3f54f108..544e595ecf 100644 --- a/web/app/components/workflow/nodes/_base/components/node-control.tsx +++ b/web/app/components/workflow/nodes/_base/components/node-control.tsx @@ -9,7 +9,6 @@ import { RiPlayLargeLine, } from '@remixicon/react' import { - useNodeDataUpdate, useNodesInteractions, } from '../../../hooks' import { type Node, NodeRunningStatus } from '../../../types' @@ -19,6 +18,9 @@ import { Stop, } from '@/app/components/base/icons/src/vender/line/mediaAndDevices' import Tooltip from '@/app/components/base/tooltip' +import { useWorkflowStore } from '@/app/components/workflow/store' +import { useWorkflowRunValidation } from '@/app/components/workflow/hooks/use-checklist' +import Toast from '@/app/components/base/toast' type NodeControlProps = Pick<Node, 'id' | 'data'> const NodeControl: FC<NodeControlProps> = ({ @@ -27,9 +29,11 @@ const NodeControl: FC<NodeControlProps> = ({ }) => { const { t } = useTranslation() const [open, setOpen] = useState(false) - const { handleNodeDataUpdate } = useNodeDataUpdate() const { handleNodeSelect } = useNodesInteractions() + const workflowStore = useWorkflowStore() const isSingleRunning = data._singleRunningStatus === NodeRunningStatus.Running + const { warningNodes } = useWorkflowRunValidation() + const warningForNode = warningNodes.find(item => item.id === id) const handleOpenChange = useCallback((newOpen: boolean) => { setOpen(newOpen) }, []) @@ -38,7 +42,8 @@ const NodeControl: FC<NodeControlProps> = ({ return ( <div className={` - absolute -top-7 right-0 hidden h-7 pb-1 group-hover:flex + absolute -top-7 right-0 hidden h-7 pb-1 + ${!data._pluginInstallLocked && 'group-hover:flex'} ${data.selected && '!flex'} ${open && '!flex'} `} @@ -50,17 +55,20 @@ const NodeControl: FC<NodeControlProps> = ({ { canRunBySingle(data.type, isChildNode) && ( <div - className='flex h-5 w-5 cursor-pointer items-center justify-center rounded-md hover:bg-state-base-hover' + className={`flex h-5 w-5 items-center justify-center rounded-md ${isSingleRunning ? 'cursor-pointer hover:bg-state-base-hover' : warningForNode ? 'cursor-not-allowed text-text-disabled' : 'cursor-pointer hover:bg-state-base-hover'}`} onClick={() => { - const nextData: Record<string, any> = { - _isSingleRun: !isSingleRunning, + const action = isSingleRunning ? 'stop' : 'run' + if (!isSingleRunning && warningForNode) { + const message = warningForNode.errorMessage || t('workflow.panel.checklistTip') + Toast.notify({ type: 'error', message }) + return } - if(isSingleRunning) - nextData._singleRunningStatus = undefined - handleNodeDataUpdate({ - id, - data: nextData, + const store = workflowStore.getState() + store.setInitShowLastRunTab(true) + store.setPendingSingleRun({ + nodeId: id, + action, }) handleNodeSelect(id) }} @@ -70,7 +78,7 @@ const NodeControl: FC<NodeControlProps> = ({ ? <Stop className='h-3 w-3' /> : ( <Tooltip - popupContent={t('workflow.panel.runThisStep')} + popupContent={warningForNode ? warningForNode.errorMessage || t('workflow.panel.checklistTip') : t('workflow.panel.runThisStep')} asChild={false} > <RiPlayLargeLine className='h-3 w-3' /> diff --git a/web/app/components/workflow/nodes/_base/components/node-handle.tsx b/web/app/components/workflow/nodes/_base/components/node-handle.tsx index 90968a4580..6cfa7a7b9e 100644 --- a/web/app/components/workflow/nodes/_base/components/node-handle.tsx +++ b/web/app/components/workflow/nodes/_base/components/node-handle.tsx @@ -16,7 +16,7 @@ import { } from '../../../types' import type { Node } from '../../../types' import BlockSelector from '../../../block-selector' -import type { DataSourceDefaultValue, ToolDefaultValue } from '../../../block-selector/types' +import type { PluginDefaultValue } from '../../../block-selector/types' import { useAvailableBlocks, useIsChatMode, @@ -25,6 +25,7 @@ import { } from '../../../hooks' import { useStore, + useWorkflowStore, } from '../../../store' import cn from '@/utils/classnames' @@ -57,11 +58,11 @@ export const NodeTargetHandle = memo(({ if (!connected) setOpen(v => !v) }, [connected]) - const handleSelect = useCallback((type: BlockEnum, toolDefaultValue?: ToolDefaultValue | DataSourceDefaultValue) => { + const handleSelect = useCallback((type: BlockEnum, pluginDefaultValue?: PluginDefaultValue) => { handleNodeAdd( { nodeType: type, - toolDefaultValue, + pluginDefaultValue, }, { nextNodeId: id, @@ -84,7 +85,10 @@ export const NodeTargetHandle = memo(({ data._runningStatus === NodeRunningStatus.Failed && 'after:bg-workflow-link-line-error-handle', data._runningStatus === NodeRunningStatus.Exception && 'after:bg-workflow-link-line-failure-handle', !connected && 'after:opacity-0', - data.type === BlockEnum.Start && 'opacity-0', + (data.type === BlockEnum.Start + || data.type === BlockEnum.TriggerWebhook + || data.type === BlockEnum.TriggerSchedule + || data.type === BlockEnum.TriggerPlugin) && 'opacity-0', handleClassName, )} isConnectable={isConnectable} @@ -124,7 +128,10 @@ export const NodeSourceHandle = memo(({ showExceptionStatus, }: NodeHandleProps) => { const { t } = useTranslation() - const notInitialWorkflow = useStore(s => s.notInitialWorkflow) + const shouldAutoOpenStartNodeSelector = useStore(s => s.shouldAutoOpenStartNodeSelector) + const setShouldAutoOpenStartNodeSelector = useStore(s => s.setShouldAutoOpenStartNodeSelector) + const setHasSelectedStartNode = useStore(s => s.setHasSelectedStartNode) + const workflowStoreApi = useWorkflowStore() const [open, setOpen] = useState(false) const { handleNodeAdd } = useNodesInteractions() const { getNodesReadOnly } = useNodesReadOnly() @@ -140,11 +147,11 @@ export const NodeSourceHandle = memo(({ e.stopPropagation() setOpen(v => !v) }, []) - const handleSelect = useCallback((type: BlockEnum, toolDefaultValue?: ToolDefaultValue | DataSourceDefaultValue) => { + const handleSelect = useCallback((type: BlockEnum, pluginDefaultValue?: PluginDefaultValue) => { handleNodeAdd( { nodeType: type, - toolDefaultValue, + pluginDefaultValue, }, { prevNodeId: id, @@ -154,9 +161,27 @@ export const NodeSourceHandle = memo(({ }, [handleNodeAdd, id, handleId]) useEffect(() => { - if (notInitialWorkflow && data.type === BlockEnum.Start && !isChatMode) + if (!shouldAutoOpenStartNodeSelector) + return + + if (isChatMode) { + setShouldAutoOpenStartNodeSelector?.(false) + return + } + + if (data.type === BlockEnum.Start || data.type === BlockEnum.TriggerSchedule || data.type === BlockEnum.TriggerWebhook || data.type === BlockEnum.TriggerPlugin) { setOpen(true) - }, [notInitialWorkflow, data.type, isChatMode]) + if (setShouldAutoOpenStartNodeSelector) + setShouldAutoOpenStartNodeSelector(false) + else + workflowStoreApi?.setState?.({ shouldAutoOpenStartNodeSelector: false }) + + if (setHasSelectedStartNode) + setHasSelectedStartNode(false) + else + workflowStoreApi?.setState?.({ hasSelectedStartNode: false }) + } + }, [shouldAutoOpenStartNodeSelector, data.type, isChatMode, setShouldAutoOpenStartNodeSelector, setHasSelectedStartNode, workflowStoreApi]) return ( <Handle diff --git a/web/app/components/workflow/nodes/_base/components/node-position.tsx b/web/app/components/workflow/nodes/_base/components/node-position.tsx deleted file mode 100644 index e844726b4f..0000000000 --- a/web/app/components/workflow/nodes/_base/components/node-position.tsx +++ /dev/null @@ -1,63 +0,0 @@ -import { memo } from 'react' -import { useTranslation } from 'react-i18next' -import { useShallow } from 'zustand/react/shallow' -import { RiCrosshairLine } from '@remixicon/react' -import { useReactFlow, useStore } from 'reactflow' -import TooltipPlus from '@/app/components/base/tooltip' -import { useNodesSyncDraft } from '@/app/components/workflow-app/hooks' - -type NodePositionProps = { - nodeId: string -} -const NodePosition = ({ - nodeId, -}: NodePositionProps) => { - const { t } = useTranslation() - const reactflow = useReactFlow() - const { doSyncWorkflowDraft } = useNodesSyncDraft() - const { - nodePosition, - nodeWidth, - nodeHeight, - } = useStore(useShallow((s) => { - const nodes = s.getNodes() - const currentNode = nodes.find(node => node.id === nodeId)! - - return { - nodePosition: currentNode.position, - nodeWidth: currentNode.width, - nodeHeight: currentNode.height, - } - })) - const transform = useStore(s => s.transform) - - if (!nodePosition || !nodeWidth || !nodeHeight) return null - - const workflowContainer = document.getElementById('workflow-container') - const zoom = transform[2] - - const { clientWidth, clientHeight } = workflowContainer! - const { setViewport } = reactflow - - return ( - <TooltipPlus - popupContent={t('workflow.panel.moveToThisNode')} - > - <div - className='mr-1 flex h-6 w-6 cursor-pointer items-center justify-center rounded-md hover:bg-state-base-hover' - onClick={() => { - setViewport({ - x: (clientWidth - 400 - nodeWidth * zoom) / 2 - nodePosition.x * zoom, - y: (clientHeight - nodeHeight * zoom) / 2 - nodePosition.y * zoom, - zoom: transform[2], - }) - doSyncWorkflowDraft() - }} - > - <RiCrosshairLine className='h-4 w-4 text-text-tertiary' /> - </div> - </TooltipPlus> - ) -} - -export default memo(NodePosition) diff --git a/web/app/components/workflow/nodes/_base/components/panel-operator/change-block.tsx b/web/app/components/workflow/nodes/_base/components/panel-operator/change-block.tsx index d7b2188ed5..8b6d137127 100644 --- a/web/app/components/workflow/nodes/_base/components/panel-operator/change-block.tsx +++ b/web/app/components/workflow/nodes/_base/components/panel-operator/change-block.tsx @@ -8,12 +8,17 @@ import { intersection } from 'lodash-es' import BlockSelector from '@/app/components/workflow/block-selector' import { useAvailableBlocks, + useIsChatMode, useNodesInteractions, } from '@/app/components/workflow/hooks' +import { useHooksStore } from '@/app/components/workflow/hooks-store' import type { Node, OnSelectBlock, } from '@/app/components/workflow/types' +import { BlockEnum, isTriggerNode } from '@/app/components/workflow/types' + +import { FlowType } from '@/types/common' type ChangeBlockProps = { nodeId: string @@ -31,6 +36,14 @@ const ChangeBlock = ({ availablePrevBlocks, availableNextBlocks, } = useAvailableBlocks(nodeData.type, nodeData.isInIteration || nodeData.isInLoop) + const isChatMode = useIsChatMode() + const flowType = useHooksStore(s => s.configsMap?.flowType) + const showStartTab = flowType !== FlowType.ragPipeline && !isChatMode + const ignoreNodeIds = useMemo(() => { + if (isTriggerNode(nodeData.type as BlockEnum)) + return [nodeId] + return undefined + }, [nodeData.type, nodeId]) const availableNodes = useMemo(() => { if (availablePrevBlocks.length && availableNextBlocks.length) @@ -41,8 +54,8 @@ const ChangeBlock = ({ return availableNextBlocks }, [availablePrevBlocks, availableNextBlocks]) - const handleSelect = useCallback<OnSelectBlock>((type, toolDefaultValue) => { - handleNodeChange(nodeId, type, sourceHandle, toolDefaultValue) + const handleSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { + handleNodeChange(nodeId, type, sourceHandle, pluginDefaultValue) }, [handleNodeChange, nodeId, sourceHandle]) const renderTrigger = useCallback(() => { @@ -64,6 +77,9 @@ const ChangeBlock = ({ trigger={renderTrigger} popupClassName='min-w-[240px]' availableBlocksTypes={availableNodes} + showStartTab={showStartTab} + ignoreNodeIds={ignoreNodeIds} + forceEnableStartTab={nodeData.type === BlockEnum.Start} /> ) } diff --git a/web/app/components/workflow/nodes/_base/components/retry/retry-on-panel.tsx b/web/app/components/workflow/nodes/_base/components/retry/retry-on-panel.tsx index 0e5b807ff4..d074d0c60c 100644 --- a/web/app/components/workflow/nodes/_base/components/retry/retry-on-panel.tsx +++ b/web/app/components/workflow/nodes/_base/components/retry/retry-on-panel.tsx @@ -76,9 +76,11 @@ const RetryOnPanel = ({ /> <Input type='number' - wrapperClassName='w-[80px]' + wrapperClassName='w-[100px]' value={retry_config?.max_retries || 3} - onChange={e => handleMaxRetriesChange(e.target.value as any)} + onChange={e => + handleMaxRetriesChange(Number.parseInt(e.currentTarget.value, 10) || 3) + } min={1} max={10} unit={t('workflow.nodes.common.retry.times') || ''} @@ -96,9 +98,11 @@ const RetryOnPanel = ({ /> <Input type='number' - wrapperClassName='w-[80px]' + wrapperClassName='w-[100px]' value={retry_config?.retry_interval || 1000} - onChange={e => handleRetryIntervalChange(e.target.value as any)} + onChange={e => + handleRetryIntervalChange(Number.parseInt(e.currentTarget.value, 10) || 1000) + } min={100} max={5000} unit={t('workflow.nodes.common.retry.ms') || ''} diff --git a/web/app/components/workflow/nodes/_base/components/variable-tag.tsx b/web/app/components/workflow/nodes/_base/components/variable-tag.tsx index 4d3dfe217c..71af3ad4fd 100644 --- a/web/app/components/workflow/nodes/_base/components/variable-tag.tsx +++ b/web/app/components/workflow/nodes/_base/components/variable-tag.tsx @@ -8,7 +8,7 @@ import type { VarType, } from '@/app/components/workflow/types' import { BlockEnum } from '@/app/components/workflow/types' -import { getNodeInfoById, isConversationVar, isENV, isRagVariableVar, isSystemVar } from '@/app/components/workflow/nodes/_base/components/variable/utils' +import { getNodeInfoById, isConversationVar, isENV, isGlobalVar, isRagVariableVar, isSystemVar } from '@/app/components/workflow/nodes/_base/components/variable/utils' import { isExceptionVariable } from '@/app/components/workflow/utils' import { VariableLabelInSelect, @@ -39,7 +39,8 @@ const VariableTag = ({ const isEnv = isENV(valueSelector) const isChatVar = isConversationVar(valueSelector) - const isValid = Boolean(node) || isEnv || isChatVar || isRagVar + const isGlobal = isGlobalVar(valueSelector) + const isValid = Boolean(node) || isEnv || isChatVar || isRagVar || isGlobal const variableName = isSystemVar(valueSelector) ? valueSelector.slice(0).join('.') : valueSelector.slice(1).join('.') const isException = isExceptionVariable(variableName, node?.data.type) diff --git a/web/app/components/workflow/nodes/_base/components/variable/object-child-tree-panel/show/field.tsx b/web/app/components/workflow/nodes/_base/components/variable/object-child-tree-panel/show/field.tsx index 7862dc824c..62133f3212 100644 --- a/web/app/components/workflow/nodes/_base/components/variable/object-child-tree-panel/show/field.tsx +++ b/web/app/components/workflow/nodes/_base/components/variable/object-child-tree-panel/show/field.tsx @@ -1,14 +1,14 @@ 'use client' +import cn from '@/utils/classnames' +import { RiArrowDropDownLine } from '@remixicon/react' +import { useBoolean } from 'ahooks' import type { FC } from 'react' import React from 'react' +import { useTranslation } from 'react-i18next' +import type { Field as FieldType } from '../../../../../llm/types' import { Type } from '../../../../../llm/types' import { getFieldType } from '../../../../../llm/utils' -import type { Field as FieldType } from '../../../../../llm/types' -import cn from '@/utils/classnames' import TreeIndentLine from '../tree-indent-line' -import { useTranslation } from 'react-i18next' -import { useBoolean } from 'ahooks' -import { RiArrowDropDownLine } from '@remixicon/react' type Props = { name: string, @@ -28,6 +28,7 @@ const Field: FC<Props> = ({ const { t } = useTranslation() const isRoot = depth === 1 const hasChildren = payload.type === Type.object && payload.properties + const hasEnum = payload.enum && payload.enum.length > 0 const [fold, { toggle: toggleFold, }] = useBoolean(false) @@ -44,7 +45,10 @@ const Field: FC<Props> = ({ /> )} <div className={cn('system-sm-medium ml-[7px] h-6 truncate leading-6 text-text-secondary', isRoot && rootClassName)}>{name}</div> - <div className='system-xs-regular ml-3 shrink-0 leading-6 text-text-tertiary'>{getFieldType(payload)}{(payload.schemaType && payload.schemaType !== 'file' && ` (${payload.schemaType})`)}</div> + <div className='system-xs-regular ml-3 shrink-0 leading-6 text-text-tertiary'> + {getFieldType(payload)} + {(payload.schemaType && payload.schemaType !== 'file' && ` (${payload.schemaType})`)} + </div> {required && <div className='system-2xs-medium-uppercase ml-3 leading-6 text-text-warning'>{t('app.structOutput.required')}</div>} </div> {payload.description && ( @@ -52,6 +56,18 @@ const Field: FC<Props> = ({ <div className='system-xs-regular w-0 grow truncate text-text-tertiary'>{payload.description}</div> </div> )} + {hasEnum && ( + <div className='ml-[7px] flex'> + <div className='system-xs-regular w-0 grow text-text-quaternary'> + {payload.enum!.map((value, index) => ( + <span key={index}> + {typeof value === 'string' ? `"${value}"` : value} + {index < payload.enum!.length - 1 && ' | '} + </span> + ))} + </div> + </div> + )} </div> </div> diff --git a/web/app/components/workflow/nodes/_base/components/variable/utils.ts b/web/app/components/workflow/nodes/_base/components/variable/utils.ts index 715551cbff..3bd43bd29a 100644 --- a/web/app/components/workflow/nodes/_base/components/variable/utils.ts +++ b/web/app/components/workflow/nodes/_base/components/variable/utils.ts @@ -39,6 +39,9 @@ import type { import type { VariableAssignerNodeType } from '@/app/components/workflow/nodes/variable-assigner/types' import type { Field as StructField } from '@/app/components/workflow/nodes/llm/types' import type { RAGPipelineVariable } from '@/models/pipeline' +import type { WebhookTriggerNodeType } from '@/app/components/workflow/nodes/trigger-webhook/types' +import type { PluginTriggerNodeType } from '@/app/components/workflow/nodes/trigger-plugin/types' +import PluginTriggerNodeDefault from '@/app/components/workflow/nodes/trigger-plugin/default' import { AGENT_OUTPUT_STRUCT, @@ -51,6 +54,7 @@ import { SUPPORT_OUTPUT_VARS_NODE, TEMPLATE_TRANSFORM_OUTPUT_STRUCT, TOOL_OUTPUT_STRUCT, + getGlobalVars, } from '@/app/components/workflow/constants' import ToolNodeDefault from '@/app/components/workflow/nodes/tool/default' import DataSourceNodeDefault from '@/app/components/workflow/nodes/data-source/default' @@ -59,11 +63,21 @@ import type { PromptItem } from '@/models/debug' import { VAR_REGEX } from '@/config' import type { AgentNodeType } from '../../../agent/types' import type { SchemaTypeDefinition } from '@/service/use-common' +import { AppModeEnum } from '@/types/app' export const isSystemVar = (valueSelector: ValueSelector) => { return valueSelector[0] === 'sys' || valueSelector[1] === 'sys' } +export const isGlobalVar = (valueSelector: ValueSelector) => { + if(!isSystemVar(valueSelector)) return false + const second = valueSelector[1] + + if(['query', 'files'].includes(second)) + return false + return true +} + export const isENV = (valueSelector: ValueSelector) => { return valueSelector[0] === 'env' } @@ -348,34 +362,29 @@ const formatItem = ( variable: 'sys.query', type: VarType.string, }) - res.vars.push({ - variable: 'sys.dialogue_count', - type: VarType.number, - }) - res.vars.push({ - variable: 'sys.conversation_id', - type: VarType.string, - }) } - res.vars.push({ - variable: 'sys.user_id', - type: VarType.string, - }) res.vars.push({ variable: 'sys.files', type: VarType.arrayFile, }) - res.vars.push({ - variable: 'sys.app_id', - type: VarType.string, - }) - res.vars.push({ - variable: 'sys.workflow_id', - type: VarType.string, - }) - res.vars.push({ - variable: 'sys.workflow_run_id', - type: VarType.string, + break + } + + case BlockEnum.TriggerWebhook: { + const { + variables = [], + } = data as WebhookTriggerNodeType + res.vars = variables.map((v) => { + const type = v.value_type || VarType.string + const varRes: Var = { + variable: v.variable, + type, + isParagraph: false, + isSelect: false, + options: v.options, + required: v.required, + } + return varRes }) break @@ -612,6 +621,17 @@ const formatItem = ( break } + case BlockEnum.TriggerPlugin: { + const outputSchema = PluginTriggerNodeDefault.getOutputVars?.( + data as PluginTriggerNodeType, + allPluginInfoList, + [], + { schemaTypeDefinitions }, + ) || [] + res.vars = outputSchema + break + } + case 'env': { res.vars = data.envList.map((env: EnvironmentVariable) => { return { @@ -634,6 +654,11 @@ const formatItem = ( break } + case 'global': { + res.vars = data.globalVarList + break + } + case 'rag': { res.vars = data.ragVariables.map((ragVar: RAGPipelineVariable) => { return { @@ -774,6 +799,15 @@ export const toNodeOutputVars = ( chatVarList: conversationVariables, }, } + // GLOBAL_VAR_NODE data format + const GLOBAL_VAR_NODE = { + id: 'global', + data: { + title: 'SYSTEM', + type: 'global', + globalVarList: getGlobalVars(isChatMode), + }, + } // RAG_PIPELINE_NODE data format const RAG_PIPELINE_NODE = { id: 'rag', @@ -793,6 +827,8 @@ export const toNodeOutputVars = ( if (b.data.type === 'env') return -1 if (a.data.type === 'conversation') return 1 if (b.data.type === 'conversation') return -1 + if (a.data.type === 'global') return 1 + if (b.data.type === 'global') return -1 // sort nodes by x position return (b.position?.x || 0) - (a.position?.x || 0) }) @@ -803,6 +839,7 @@ export const toNodeOutputVars = ( ), ...(environmentVariables.length > 0 ? [ENV_NODE] : []), ...(isChatMode && conversationVariables.length > 0 ? [CHAT_VAR_NODE] : []), + GLOBAL_VAR_NODE, ...(RAG_PIPELINE_NODE.data.ragVariables.length > 0 ? [RAG_PIPELINE_NODE] : []), @@ -1026,7 +1063,8 @@ export const getVarType = ({ if (valueSelector[1] === 'index') return VarType.number } - const isSystem = isSystemVar(valueSelector) + const isGlobal = isGlobalVar(valueSelector) + const isInStartNodeSysVar = isSystemVar(valueSelector) && !isGlobal const isEnv = isENV(valueSelector) const isChatVar = isConversationVar(valueSelector) const isSharedRagVariable @@ -1039,7 +1077,8 @@ export const getVarType = ({ }) const targetVarNodeId = (() => { - if (isSystem) return startNode?.id + if (isInStartNodeSysVar) return startNode?.id + if (isGlobal) return 'global' if (isInNodeRagVariable) return valueSelector[1] return valueSelector[0] })() @@ -1052,7 +1091,7 @@ export const getVarType = ({ let type: VarType = VarType.string let curr: any = targetVar.vars - if (isSystem || isEnv || isChatVar || isSharedRagVariable) { + if (isInStartNodeSysVar || isEnv || isChatVar || isSharedRagVariable || isGlobal) { return curr.find( (v: any) => v.variable === (valueSelector as ValueSelector).join('.'), )?.type @@ -1242,7 +1281,7 @@ export const getNodeUsedVars = (node: Node): ValueSelector[] => { } case BlockEnum.LLM: { const payload = data as LLMNodeType - const isChatModel = payload.model?.mode === 'chat' + const isChatModel = payload.model?.mode === AppModeEnum.CHAT let prompts: string[] = [] if (isChatModel) { prompts @@ -1545,7 +1584,7 @@ export const updateNodeVars = ( } case BlockEnum.LLM: { const payload = data as LLMNodeType - const isChatModel = payload.model?.mode === 'chat' + const isChatModel = payload.model?.mode === AppModeEnum.CHAT if (isChatModel) { payload.prompt_template = ( payload.prompt_template as PromptItem[] diff --git a/web/app/components/workflow/nodes/_base/components/variable/var-reference-picker.tsx b/web/app/components/workflow/nodes/_base/components/variable/var-reference-picker.tsx index 85424cdaf4..82c2dfd470 100644 --- a/web/app/components/workflow/nodes/_base/components/variable/var-reference-picker.tsx +++ b/web/app/components/workflow/nodes/_base/components/variable/var-reference-picker.tsx @@ -18,10 +18,11 @@ import { import RemoveButton from '../remove-button' import useAvailableVarList from '../../hooks/use-available-var-list' import VarReferencePopup from './var-reference-popup' -import { getNodeInfoById, isConversationVar, isENV, isRagVariableVar, isSystemVar, removeFileVars, varTypeToStructType } from './utils' +import { getNodeInfoById, isConversationVar, isENV, isGlobalVar, isRagVariableVar, isSystemVar, removeFileVars, varTypeToStructType } from './utils' import ConstantField from './constant-field' import cn from '@/utils/classnames' import type { CommonNodeType, Node, NodeOutPutVar, ToolWithProvider, ValueSelector, Var } from '@/app/components/workflow/types' +import type { TriggerWithProvider } from '@/app/components/workflow/block-selector/types' import type { CredentialFormSchemaSelect } from '@/app/components/header/account-setting/model-provider-page/declarations' import { type CredentialFormSchema, type FormOption, FormTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' import { BlockEnum } from '@/app/components/workflow/types' @@ -38,6 +39,7 @@ import { useWorkflowVariables, } from '@/app/components/workflow/hooks' import { VarType as VarKindType } from '@/app/components/workflow/nodes/tool/types' +// import type { BaseResource, BaseResourceProvider } from '@/app/components/workflow/nodes/_base/types' import TypeSelector from '@/app/components/workflow/nodes/_base/components/selector' import AddButton from '@/app/components/base/button/add-button' import Badge from '@/app/components/base/badge' @@ -45,9 +47,10 @@ import Tooltip from '@/app/components/base/tooltip' import { isExceptionVariable } from '@/app/components/workflow/utils' import VarFullPathPanel from './var-full-path-panel' import { noop } from 'lodash-es' -import { useFetchDynamicOptions } from '@/service/use-plugins' import type { Tool } from '@/app/components/tools/types' +import { useFetchDynamicOptions } from '@/service/use-plugins' import { VariableIconWithColor } from '@/app/components/workflow/nodes/_base/components/variable/variable-label' +import { VAR_SHOW_NAME_MAP } from '@/app/components/workflow/constants' const TRIGGER_DEFAULT_WIDTH = 227 @@ -78,7 +81,7 @@ type Props = { popupFor?: 'assigned' | 'toAssigned' zIndex?: number currentTool?: Tool - currentProvider?: ToolWithProvider + currentProvider?: ToolWithProvider | TriggerWithProvider preferSchemaType?: boolean } @@ -203,6 +206,9 @@ const VarReferencePicker: FC<Props> = ({ const varName = useMemo(() => { if (!hasValue) return '' + const showName = VAR_SHOW_NAME_MAP[(value as ValueSelector).join('.')] + if(showName) + return showName const isSystem = isSystemVar(value as ValueSelector) const varName = Array.isArray(value) ? value[(value as ValueSelector).length - 1] : '' @@ -291,15 +297,17 @@ const VarReferencePicker: FC<Props> = ({ preferSchemaType, }) - const { isEnv, isChatVar, isRagVar, isValidVar, isException } = useMemo(() => { + const { isEnv, isChatVar, isGlobal, isRagVar, isValidVar, isException } = useMemo(() => { const isEnv = isENV(value as ValueSelector) const isChatVar = isConversationVar(value as ValueSelector) + const isGlobal = isGlobalVar(value as ValueSelector) const isRagVar = isRagVariableVar(value as ValueSelector) - const isValidVar = Boolean(outputVarNode) || isEnv || isChatVar || isRagVar + const isValidVar = Boolean(outputVarNode) || isEnv || isChatVar || isGlobal || isRagVar const isException = isExceptionVariable(varName, outputVarNode?.type) return { isEnv, isChatVar, + isGlobal, isRagVar, isValidVar, isException, @@ -392,10 +400,11 @@ const VarReferencePicker: FC<Props> = ({ const variableCategory = useMemo(() => { if (isEnv) return 'environment' if (isChatVar) return 'conversation' + if (isGlobal) return 'global' if (isLoopVar) return 'loop' if (isRagVar) return 'rag' return 'system' - }, [isEnv, isChatVar, isLoopVar, isRagVar]) + }, [isEnv, isChatVar, isGlobal, isLoopVar, isRagVar]) return ( <div className={cn(className, !readonly && 'cursor-pointer')}> @@ -473,7 +482,7 @@ const VarReferencePicker: FC<Props> = ({ {hasValue ? ( <> - {isShowNodeName && !isEnv && !isChatVar && !isRagVar && ( + {isShowNodeName && !isEnv && !isChatVar && !isGlobal && !isRagVar && ( <div className='flex items-center' onClick={(e) => { if (e.metaKey || e.ctrlKey) { e.stopPropagation() @@ -501,10 +510,11 @@ const VarReferencePicker: FC<Props> = ({ <div className='flex items-center text-text-accent'> {isLoading && <RiLoader4Line className='h-3.5 w-3.5 animate-spin text-text-secondary' />} <VariableIconWithColor + variables={value as ValueSelector} variableCategory={variableCategory} isExceptionVariable={isException} /> - <div className={cn('ml-0.5 truncate text-xs font-medium', isEnv && '!text-text-secondary', isChatVar && 'text-util-colors-teal-teal-700', isException && 'text-text-warning')} title={varName} style={{ + <div className={cn('ml-0.5 truncate text-xs font-medium', isEnv && '!text-text-secondary', isChatVar && 'text-util-colors-teal-teal-700', isException && 'text-text-warning', isGlobal && 'text-util-colors-orange-orange-600')} title={varName} style={{ maxWidth: maxVarNameWidth, }}>{varName}</div> </div> diff --git a/web/app/components/workflow/nodes/_base/components/variable/var-reference-vars.tsx b/web/app/components/workflow/nodes/_base/components/variable/var-reference-vars.tsx index e70cfed97c..ced4b7c65f 100644 --- a/web/app/components/workflow/nodes/_base/components/variable/var-reference-vars.tsx +++ b/web/app/components/workflow/nodes/_base/components/variable/var-reference-vars.tsx @@ -23,6 +23,7 @@ import { CodeAssistant, MagicEdit } from '@/app/components/base/icons/src/vender import ManageInputField from './manage-input-field' import { VariableIconWithColor } from '@/app/components/workflow/nodes/_base/components/variable/variable-label' import { Variable02 } from '@/app/components/base/icons/src/vender/solid/development' +import { VAR_SHOW_NAME_MAP } from '@/app/components/workflow/constants' type ItemProps = { nodeId: string @@ -82,10 +83,14 @@ const Item: FC<ItemProps> = ({ }, [isFlat, isInCodeGeneratorInstructionEditor, itemData.variable]) const varName = useMemo(() => { + if(VAR_SHOW_NAME_MAP[itemData.variable]) + return VAR_SHOW_NAME_MAP[itemData.variable] + if (!isFlat) return itemData.variable if (itemData.variable === 'current') return isInCodeGeneratorInstructionEditor ? 'current_code' : 'current_prompt' + return itemData.variable }, [isFlat, isInCodeGeneratorInstructionEditor, itemData.variable]) @@ -182,6 +187,7 @@ const Item: FC<ItemProps> = ({ > <div className='flex w-0 grow items-center'> {!isFlat && <VariableIconWithColor + variables={itemData.variable.split('.')} variableCategory={variableCategory} isExceptionVariable={isException} />} diff --git a/web/app/components/workflow/nodes/_base/components/variable/variable-label/base/variable-label.tsx b/web/app/components/workflow/nodes/_base/components/variable/variable-label/base/variable-label.tsx index 99f080f545..a8acda7e2c 100644 --- a/web/app/components/workflow/nodes/_base/components/variable/variable-label/base/variable-label.tsx +++ b/web/app/components/workflow/nodes/_base/components/variable/variable-label/base/variable-label.tsx @@ -11,6 +11,7 @@ import VariableIcon from './variable-icon' import VariableName from './variable-name' import cn from '@/utils/classnames' import Tooltip from '@/app/components/base/tooltip' +import { isConversationVar, isENV, isGlobalVar, isRagVariableVar } from '../../utils' const VariableLabel = ({ nodeType, @@ -26,6 +27,7 @@ const VariableLabel = ({ rightSlot, }: VariablePayload) => { const varColorClassName = useVarColor(variables, isExceptionVariable) + const isHideNodeLabel = !(isENV(variables) || isConversationVar(variables) || isGlobalVar(variables) || isRagVariableVar(variables)) return ( <div className={cn( @@ -35,10 +37,12 @@ const VariableLabel = ({ onClick={onClick} ref={ref} > - <VariableNodeLabel - nodeType={nodeType} - nodeTitle={nodeTitle} - /> + { isHideNodeLabel && ( + <VariableNodeLabel + nodeType={nodeType} + nodeTitle={nodeTitle} + /> + )} { notShowFullPath && ( <> diff --git a/web/app/components/workflow/nodes/_base/components/variable/variable-label/hooks.ts b/web/app/components/workflow/nodes/_base/components/variable/variable-label/hooks.ts index fef6d8c396..bb388d429a 100644 --- a/web/app/components/workflow/nodes/_base/components/variable/variable-label/hooks.ts +++ b/web/app/components/workflow/nodes/_base/components/variable/variable-label/hooks.ts @@ -1,15 +1,17 @@ import { useMemo } from 'react' import { Variable02 } from '@/app/components/base/icons/src/vender/solid/development' -import { BubbleX, Env } from '@/app/components/base/icons/src/vender/line/others' +import { BubbleX, Env, GlobalVariable } from '@/app/components/base/icons/src/vender/line/others' import { Loop } from '@/app/components/base/icons/src/vender/workflow' import { InputField } from '@/app/components/base/icons/src/vender/pipeline' import { isConversationVar, isENV, + isGlobalVar, isRagVariableVar, isSystemVar, } from '../utils' import { VarInInspectType } from '@/types/workflow' +import { VAR_SHOW_NAME_MAP } from '@/app/components/workflow/constants' export const useVarIcon = (variables: string[], variableCategory?: VarInInspectType | string) => { if (variableCategory === 'loop') @@ -24,6 +26,9 @@ export const useVarIcon = (variables: string[], variableCategory?: VarInInspectT if (isConversationVar(variables) || variableCategory === VarInInspectType.conversation || variableCategory === 'conversation') return BubbleX + if (isGlobalVar(variables) || variableCategory === VarInInspectType.system) + return GlobalVariable + return Variable02 } @@ -41,13 +46,22 @@ export const useVarColor = (variables: string[], isExceptionVariable?: boolean, if (isConversationVar(variables) || variableCategory === VarInInspectType.conversation || variableCategory === 'conversation') return 'text-util-colors-teal-teal-700' + if (isGlobalVar(variables) || variableCategory === VarInInspectType.system) + return 'text-util-colors-orange-orange-600' + return 'text-text-accent' }, [variables, isExceptionVariable, variableCategory]) } export const useVarName = (variables: string[], notShowFullPath?: boolean) => { + const showName = VAR_SHOW_NAME_MAP[variables.join('.')] + let variableFullPathName = variables.slice(1).join('.') + + if (isRagVariableVar(variables)) + variableFullPathName = variables.slice(2).join('.') + const varName = useMemo(() => { - let variableFullPathName = variables.slice(1).join('.') + variableFullPathName = variables.slice(1).join('.') if (isRagVariableVar(variables)) variableFullPathName = variables.slice(2).join('.') @@ -58,6 +72,8 @@ export const useVarName = (variables: string[], notShowFullPath?: boolean) => { return `${isSystem ? 'sys.' : ''}${varName}` }, [variables, notShowFullPath]) + if (showName) + return showName return varName } diff --git a/web/app/components/workflow/nodes/_base/components/workflow-panel/index.tsx b/web/app/components/workflow/nodes/_base/components/workflow-panel/index.tsx index 03b142ba43..eaafab550e 100644 --- a/web/app/components/workflow/nodes/_base/components/workflow-panel/index.tsx +++ b/web/app/components/workflow/nodes/_base/components/workflow-panel/index.tsx @@ -1,36 +1,18 @@ -import type { - FC, - ReactNode, -} from 'react' -import React, { - cloneElement, - memo, - useCallback, - useEffect, - useMemo, - useRef, - useState, -} from 'react' +import { useStore as useAppStore } from '@/app/components/app/store' +import { Stop } from '@/app/components/base/icons/src/vender/line/mediaAndDevices' +import Tooltip from '@/app/components/base/tooltip' +import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' import { - RiCloseLine, - RiPlayLargeLine, -} from '@remixicon/react' -import { useShallow } from 'zustand/react/shallow' -import { useTranslation } from 'react-i18next' -import NextStep from '../next-step' -import PanelOperator from '../panel-operator' -import NodePosition from '@/app/components/workflow/nodes/_base/components/node-position' -import HelpLink from '../help-link' -import { - DescriptionInput, - TitleInput, -} from '../title-description-input' -import ErrorHandleOnPanel from '../error-handle/error-handle-on-panel' -import RetryOnPanel from '../retry/retry-on-panel' -import { useResizePanel } from '../../hooks/use-resize-panel' -import cn from '@/utils/classnames' + AuthCategory, + AuthorizedInDataSourceNode, + AuthorizedInNode, + PluginAuth, + PluginAuthInDataSourceNode, +} from '@/app/components/plugins/plugin-auth' +import { usePluginStore } from '@/app/components/plugins/plugin-detail-panel/store' +import type { SimpleSubscription } from '@/app/components/plugins/plugin-detail-panel/subscription-list' +import { ReadmeEntrance } from '@/app/components/plugins/readme-panel/entrance' import BlockIcon from '@/app/components/workflow/block-icon' -import Split from '@/app/components/workflow/nodes/_base/components/split' import { WorkflowHistoryEvent, useAvailableBlocks, @@ -41,40 +23,59 @@ import { useToolIcon, useWorkflowHistory, } from '@/app/components/workflow/hooks' +import { useHooksStore } from '@/app/components/workflow/hooks-store' +import useInspectVarsCrud from '@/app/components/workflow/hooks/use-inspect-vars-crud' +import Split from '@/app/components/workflow/nodes/_base/components/split' +import DataSourceBeforeRunForm from '@/app/components/workflow/nodes/data-source/before-run-form' +import type { CustomRunFormProps } from '@/app/components/workflow/nodes/data-source/types' +import { DataSourceClassification } from '@/app/components/workflow/nodes/data-source/types' +import { useLogs } from '@/app/components/workflow/run/hooks' +import SpecialResultPanel from '@/app/components/workflow/run/special-result-panel' +import { useStore } from '@/app/components/workflow/store' +import { BlockEnum, type Node, NodeRunningStatus } from '@/app/components/workflow/types' import { canRunBySingle, hasErrorHandleNode, hasRetryNode, isSupportCustomRunForm, } from '@/app/components/workflow/utils' -import Tooltip from '@/app/components/base/tooltip' -import { BlockEnum, type Node, NodeRunningStatus } from '@/app/components/workflow/types' -import { useStore as useAppStore } from '@/app/components/app/store' -import { useStore } from '@/app/components/workflow/store' -import Tab, { TabType } from './tab' +import { useModalContext } from '@/context/modal-context' +import { useAllBuiltInTools } from '@/service/use-tools' +import { useAllTriggerPlugins } from '@/service/use-triggers' +import { FlowType } from '@/types/common' +import { canFindTool } from '@/utils' +import cn from '@/utils/classnames' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' +import { + RiCloseLine, + RiPlayLargeLine, +} from '@remixicon/react' +import { debounce } from 'lodash-es' +import type { FC, ReactNode } from 'react' +import React, { + cloneElement, + memo, + useCallback, + useEffect, + useMemo, + useRef, + useState, +} from 'react' +import { useTranslation } from 'react-i18next' +import { useShallow } from 'zustand/react/shallow' +import { useResizePanel } from '../../hooks/use-resize-panel' +import BeforeRunForm from '../before-run-form' +import PanelWrap from '../before-run-form/panel-wrap' +import ErrorHandleOnPanel from '../error-handle/error-handle-on-panel' +import HelpLink from '../help-link' +import NextStep from '../next-step' +import PanelOperator from '../panel-operator' +import RetryOnPanel from '../retry/retry-on-panel' +import { DescriptionInput, TitleInput } from '../title-description-input' import LastRun from './last-run' import useLastRun from './last-run/use-last-run' -import BeforeRunForm from '../before-run-form' -import { debounce } from 'lodash-es' -import { useLogs } from '@/app/components/workflow/run/hooks' -import PanelWrap from '../before-run-form/panel-wrap' -import SpecialResultPanel from '@/app/components/workflow/run/special-result-panel' -import { Stop } from '@/app/components/base/icons/src/vender/line/mediaAndDevices' -import { useHooksStore } from '@/app/components/workflow/hooks-store' -import { FlowType } from '@/types/common' -import { - AuthorizedInDataSourceNode, - AuthorizedInNode, - PluginAuth, - PluginAuthInDataSourceNode, -} from '@/app/components/plugins/plugin-auth' -import { AuthCategory } from '@/app/components/plugins/plugin-auth' -import { canFindTool } from '@/utils' -import type { CustomRunFormProps } from '@/app/components/workflow/nodes/data-source/types' -import { DataSourceClassification } from '@/app/components/workflow/nodes/data-source/types' -import { useModalContext } from '@/context/modal-context' -import DataSourceBeforeRunForm from '@/app/components/workflow/nodes/data-source/before-run-form' -import useInspectVarsCrud from '@/app/components/workflow/hooks/use-inspect-vars-crud' +import Tab, { TabType } from './tab' +import { TriggerSubscription } from './trigger-subscription' const getCustomRunForm = (params: CustomRunFormProps): React.JSX.Element => { const nodeType = params.payload.type @@ -85,6 +86,7 @@ const getCustomRunForm = (params: CustomRunFormProps): React.JSX.Element => { return <div>Custom Run Form: {nodeType} not found</div> } } + type BasePanelProps = { children: ReactNode id: Node['id'] @@ -97,6 +99,7 @@ const BasePanel: FC<BasePanelProps> = ({ children, }) => { const { t } = useTranslation() + const language = useLanguage() const { showMessageLogModal } = useAppStore(useShallow(state => ({ showMessageLogModal: state.showMessageLogModal, }))) @@ -107,6 +110,13 @@ const BasePanel: FC<BasePanelProps> = ({ const nodePanelWidth = useStore(s => s.nodePanelWidth) const otherPanelWidth = useStore(s => s.otherPanelWidth) const setNodePanelWidth = useStore(s => s.setNodePanelWidth) + const { + pendingSingleRun, + setPendingSingleRun, + } = useStore(s => ({ + pendingSingleRun: s.pendingSingleRun, + setPendingSingleRun: s.setPendingSingleRun, + })) const reservedCanvasWidth = 400 // Reserve the minimum visible width for the canvas @@ -211,6 +221,7 @@ const BasePanel: FC<BasePanelProps> = ({ useEffect(() => { hasClickRunning.current = false }, [id]) + const { nodesMap, } = useNodesMetaData() @@ -234,6 +245,7 @@ const BasePanel: FC<BasePanelProps> = ({ singleRunParams, nodeInfo, setRunInputData, + handleStop, handleSingleRun, handleRunWithParams, getExistVarValuesInForms, @@ -251,26 +263,65 @@ const BasePanel: FC<BasePanelProps> = ({ setIsPaused(false) }, [tabType]) + useEffect(() => { + if (!pendingSingleRun || pendingSingleRun.nodeId !== id) + return + + if (pendingSingleRun.action === 'run') + handleSingleRun() + else + handleStop() + + setPendingSingleRun(undefined) + }, [pendingSingleRun, id, handleSingleRun, handleStop, setPendingSingleRun]) + const logParams = useLogs() - const passedLogParams = (() => { - if ([BlockEnum.Tool, BlockEnum.Agent, BlockEnum.Iteration, BlockEnum.Loop].includes(data.type)) - return logParams + const passedLogParams = useMemo(() => [BlockEnum.Tool, BlockEnum.Agent, BlockEnum.Iteration, BlockEnum.Loop].includes(data.type) ? logParams : {}, [data.type, logParams]) - return {} - })() + const storeBuildInTools = useStore(s => s.buildInTools) + const { data: buildInTools } = useAllBuiltInTools() + const currToolCollection = useMemo(() => { + const candidates = buildInTools ?? storeBuildInTools + return candidates?.find(item => canFindTool(item.id, data.provider_id)) + }, [buildInTools, storeBuildInTools, data.provider_id]) + const needsToolAuth = useMemo(() => { + return data.type === BlockEnum.Tool && currToolCollection?.allow_delete + }, [data.type, currToolCollection?.allow_delete]) + + // only fetch trigger plugins when the node is a trigger plugin + const { data: triggerPlugins = [] } = useAllTriggerPlugins(data.type === BlockEnum.TriggerPlugin) + const currentTriggerPlugin = useMemo(() => { + if (data.type !== BlockEnum.TriggerPlugin || !data.plugin_id || !triggerPlugins?.length) + return undefined + return triggerPlugins?.find(p => p.plugin_id === data.plugin_id) + }, [data.type, data.plugin_id, triggerPlugins]) + const { setDetail } = usePluginStore() + + useEffect(() => { + if (currentTriggerPlugin?.subscription_constructor) { + setDetail({ + name: currentTriggerPlugin.label[language], + plugin_id: currentTriggerPlugin.plugin_id || '', + plugin_unique_identifier: currentTriggerPlugin.plugin_unique_identifier || '', + id: currentTriggerPlugin.id, + provider: currentTriggerPlugin.name, + declaration: { + trigger: { + subscription_schema: currentTriggerPlugin.subscription_schema || [], + subscription_constructor: currentTriggerPlugin.subscription_constructor, + }, + }, + }) + } + }, [currentTriggerPlugin, language, setDetail]) - const buildInTools = useStore(s => s.buildInTools) - const currCollection = useMemo(() => { - return buildInTools.find(item => canFindTool(item.id, data.provider_id)) - }, [buildInTools, data.provider_id]) - const showPluginAuth = useMemo(() => { - return data.type === BlockEnum.Tool && currCollection?.allow_delete - }, [currCollection, data.type]) const dataSourceList = useStore(s => s.dataSourceList) + const currentDataSource = useMemo(() => { if (data.type === BlockEnum.DataSource && data.provider_type !== DataSourceClassification.localFile) return dataSourceList?.find(item => item.plugin_id === data.plugin_id) - }, [dataSourceList, data.plugin_id, data.type, data.provider_type]) + }, [dataSourceList, data.provider_id, data.type, data.provider_type]) + const handleAuthorizationItemClick = useCallback((credential_id: string) => { handleNodeDataUpdateWithSyncDraft({ id, @@ -279,15 +330,46 @@ const BasePanel: FC<BasePanelProps> = ({ }, }) }, [handleNodeDataUpdateWithSyncDraft, id]) + const { setShowAccountSettingModal } = useModalContext() + const handleJumpToDataSourcePage = useCallback(() => { - setShowAccountSettingModal({ payload: 'data-source' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.DATA_SOURCE }) }, [setShowAccountSettingModal]) const { appendNodeInspectVars, } = useInspectVarsCrud() + const handleSubscriptionChange = useCallback((v: SimpleSubscription, callback?: () => void) => { + handleNodeDataUpdateWithSyncDraft( + { id, data: { subscription_id: v.id } }, + { + sync: true, + callback: { onSettled: callback }, + }, + ) + }, [handleNodeDataUpdateWithSyncDraft, id]) + + const readmeEntranceComponent = useMemo(() => { + let pluginDetail + switch (data.type) { + case BlockEnum.Tool: + pluginDetail = currToolCollection + break + case BlockEnum.DataSource: + pluginDetail = currentDataSource + break + case BlockEnum.TriggerPlugin: + pluginDetail = currentTriggerPlugin + break + + default: + break + } + return !pluginDetail ? null : <ReadmeEntrance pluginDetail={pluginDetail as any} className='mt-auto' /> + }, [data.type, currToolCollection, currentDataSource, currentTriggerPlugin]) + if (logParams.showSpecialResultPanel) { return ( <div className={cn( @@ -404,18 +486,10 @@ const BasePanel: FC<BasePanelProps> = ({ <div className='mr-1 flex h-6 w-6 cursor-pointer items-center justify-center rounded-md hover:bg-state-base-hover' onClick={() => { - if (isSingleRunning) { - handleNodeDataUpdate({ - id, - data: { - _isSingleRun: false, - _singleRunningStatus: undefined, - }, - }) - } - else { + if (isSingleRunning) + handleStop() + else handleSingleRun() - } }} > { @@ -426,7 +500,6 @@ const BasePanel: FC<BasePanelProps> = ({ </Tooltip> ) } - <NodePosition nodeId={id}></NodePosition> <HelpLink nodeType={data.type} /> <PanelOperator id={id} data={data} showHelpLink={false} /> <div className='mx-3 h-3.5 w-[1px] bg-divider-regular' /> @@ -445,12 +518,14 @@ const BasePanel: FC<BasePanelProps> = ({ /> </div> { - showPluginAuth && ( + needsToolAuth && ( <PluginAuth className='px-4 pb-2' pluginPayload={{ - provider: currCollection?.name || '', + provider: currToolCollection?.name || '', + providerType: currToolCollection?.type || '', category: AuthCategory.tool, + detail: currToolCollection as any, }} > <div className='flex items-center justify-between pl-4 pr-3'> @@ -460,8 +535,10 @@ const BasePanel: FC<BasePanelProps> = ({ /> <AuthorizedInNode pluginPayload={{ - provider: currCollection?.name || '', + provider: currToolCollection?.name || '', + providerType: currToolCollection?.type || '', category: AuthCategory.tool, + detail: currToolCollection as any, }} onAuthorizationItemClick={handleAuthorizationItemClick} credentialId={data.credential_id} @@ -490,7 +567,20 @@ const BasePanel: FC<BasePanelProps> = ({ ) } { - !showPluginAuth && !currentDataSource && ( + currentTriggerPlugin && ( + <TriggerSubscription + subscriptionIdSelected={data.subscription_id} + onSubscriptionChange={handleSubscriptionChange} + > + <Tab + value={tabType} + onChange={setTabType} + /> + </TriggerSubscription> + ) + } + { + !needsToolAuth && !currentDataSource && !currentTriggerPlugin && ( <div className='flex items-center justify-between pl-4 pr-3'> <Tab value={tabType} @@ -502,7 +592,7 @@ const BasePanel: FC<BasePanelProps> = ({ <Split /> </div> {tabType === TabType.settings && ( - <div className='flex-1 overflow-y-auto'> + <div className='flex flex-1 flex-col overflow-y-auto'> <div> {cloneElement(children as any, { id, @@ -547,6 +637,7 @@ const BasePanel: FC<BasePanelProps> = ({ </div> ) } + {readmeEntranceComponent} </div> )} @@ -565,6 +656,7 @@ const BasePanel: FC<BasePanelProps> = ({ {...passedLogParams} /> )} + </div> </div> ) diff --git a/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/index.tsx b/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/index.tsx index b26dd74714..43dab49ed8 100644 --- a/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/index.tsx +++ b/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/index.tsx @@ -60,6 +60,19 @@ const LastRun: FC<Props> = ({ const noLastRun = (error as any)?.status === 404 const runResult = (canRunLastRun ? lastRunResult : singleRunResult) || lastRunResult || {} + const resolvedStatus = useMemo(() => { + if (isPaused) + return NodeRunningStatus.Stopped + + if (oneStepRunRunningStatus === NodeRunningStatus.Stopped) + return NodeRunningStatus.Stopped + + if (oneStepRunRunningStatus === NodeRunningStatus.Listening) + return NodeRunningStatus.Listening + + return (runResult as any).status || otherResultPanelProps.status + }, [isPaused, oneStepRunRunningStatus, runResult, otherResultPanelProps.status]) + const resetHidePageStatus = useCallback(() => { setPageHasHide(false) setPageShowed(false) @@ -104,18 +117,18 @@ const LastRun: FC<Props> = ({ if (isRunning) return <ResultPanel status='running' showSteps={false} /> - if (!isPaused && (noLastRun || !runResult)) { return ( <NoData canSingleRun={canSingleRun} onSingleRun={onSingleRunClicked} /> ) } + return ( <div> <ResultPanel {...runResult as any} {...otherResultPanelProps} - status={isPaused ? NodeRunningStatus.Stopped : ((runResult as any).status || otherResultPanelProps.status)} + status={resolvedStatus} total_tokens={(runResult as any)?.execution_metadata?.total_tokens || otherResultPanelProps?.total_tokens} created_by={(runResult as any)?.created_by_account?.created_by || otherResultPanelProps?.created_by} nodeInfo={runResult as NodeTracing} diff --git a/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/use-last-run.ts b/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/use-last-run.ts index 21462de939..ac9f2051c3 100644 --- a/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/use-last-run.ts +++ b/web/app/components/workflow/nodes/_base/components/workflow-panel/last-run/use-last-run.ts @@ -22,6 +22,7 @@ import useVariableAssignerSingleRunFormParams from '@/app/components/workflow/no import useKnowledgeBaseSingleRunFormParams from '@/app/components/workflow/nodes/knowledge-base/use-single-run-form-params' import useToolGetDataForCheckMore from '@/app/components/workflow/nodes/tool/use-get-data-for-check-more' +import useTriggerPluginGetDataForCheckMore from '@/app/components/workflow/nodes/trigger-plugin/use-check-params' import { VALUE_SELECTOR_DELIMITER as DELIMITER } from '@/config' // import @@ -30,10 +31,12 @@ import { BlockEnum } from '@/app/components/workflow/types' import { useNodesSyncDraft, } from '@/app/components/workflow/hooks' +import { useWorkflowRunValidation } from '@/app/components/workflow/hooks/use-checklist' import useInspectVarsCrud from '@/app/components/workflow/hooks/use-inspect-vars-crud' import { useInvalidLastRun } from '@/service/use-workflow' import { useStore, useWorkflowStore } from '@/app/components/workflow/store' import { isSupportCustomRunForm } from '@/app/components/workflow/utils' +import Toast from '@/app/components/base/toast' const singleRunFormParamsHooks: Record<BlockEnum, any> = { [BlockEnum.LLM]: useLLMSingleRunFormParams, @@ -62,6 +65,9 @@ const singleRunFormParamsHooks: Record<BlockEnum, any> = { [BlockEnum.LoopEnd]: undefined, [BlockEnum.DataSource]: undefined, [BlockEnum.DataSourceEmpty]: undefined, + [BlockEnum.TriggerWebhook]: undefined, + [BlockEnum.TriggerSchedule]: undefined, + [BlockEnum.TriggerPlugin]: undefined, } const useSingleRunFormParamsHooks = (nodeType: BlockEnum) => { @@ -97,6 +103,9 @@ const getDataForCheckMoreHooks: Record<BlockEnum, any> = { [BlockEnum.DataSource]: undefined, [BlockEnum.DataSourceEmpty]: undefined, [BlockEnum.KnowledgeBase]: undefined, + [BlockEnum.TriggerWebhook]: undefined, + [BlockEnum.TriggerSchedule]: undefined, + [BlockEnum.TriggerPlugin]: useTriggerPluginGetDataForCheckMore, } const useGetDataForCheckMoreHooks = <T>(nodeType: BlockEnum) => { @@ -139,6 +148,17 @@ const useLastRun = <T>({ isRunAfterSingleRun, }) + const { warningNodes } = useWorkflowRunValidation() + const blockIfChecklistFailed = useCallback(() => { + const warningForNode = warningNodes.find(item => item.id === id) + if (!warningForNode) + return false + + const message = warningForNode.errorMessage || 'This node has unresolved checklist issues' + Toast.notify({ type: 'error', message }) + return true + }, [warningNodes, id]) + const { hideSingleRun, handleRun: doCallRunApi, @@ -199,7 +219,7 @@ const useLastRun = <T>({ }) } const workflowStore = useWorkflowStore() - const { setInitShowLastRunTab } = workflowStore.getState() + const { setInitShowLastRunTab, setShowVariableInspectPanel } = workflowStore.getState() const initShowLastRunTab = useStore(s => s.initShowLastRunTab) const [tabType, setTabType] = useState<TabType>(initShowLastRunTab ? TabType.lastRun : TabType.settings) useEffect(() => { @@ -211,6 +231,8 @@ const useLastRun = <T>({ const invalidLastRun = useInvalidLastRun(flowType, flowId, id) const handleRunWithParams = async (data: Record<string, any>) => { + if (blockIfChecklistFailed()) + return const { isValid } = checkValid() if (!isValid) return @@ -309,9 +331,13 @@ const useLastRun = <T>({ } const handleSingleRun = () => { + if (blockIfChecklistFailed()) + return const { isValid } = checkValid() if (!isValid) return + if (blockType === BlockEnum.TriggerWebhook || blockType === BlockEnum.TriggerPlugin || blockType === BlockEnum.TriggerSchedule) + setShowVariableInspectPanel(true) if (isCustomRunNode) { showSingleRun() return diff --git a/web/app/components/workflow/nodes/_base/components/workflow-panel/trigger-subscription.tsx b/web/app/components/workflow/nodes/_base/components/workflow-panel/trigger-subscription.tsx new file mode 100644 index 0000000000..811516df3d --- /dev/null +++ b/web/app/components/workflow/nodes/_base/components/workflow-panel/trigger-subscription.tsx @@ -0,0 +1,26 @@ +import type { SimpleSubscription } from '@/app/components/plugins/plugin-detail-panel/subscription-list' +import { CreateButtonType, CreateSubscriptionButton } from '@/app/components/plugins/plugin-detail-panel/subscription-list/create' +import { SubscriptionSelectorEntry } from '@/app/components/plugins/plugin-detail-panel/subscription-list/selector-entry' +import { useSubscriptionList } from '@/app/components/plugins/plugin-detail-panel/subscription-list/use-subscription-list' +import cn from '@/utils/classnames' +import type { FC } from 'react' + +type TriggerSubscriptionProps = { + subscriptionIdSelected?: string + onSubscriptionChange: (v: SimpleSubscription, callback?: () => void) => void + children: React.ReactNode +} + +export const TriggerSubscription: FC<TriggerSubscriptionProps> = ({ subscriptionIdSelected, onSubscriptionChange, children }) => { + const { subscriptions } = useSubscriptionList() + const subscriptionCount = subscriptions?.length || 0 + + return <div className={cn('px-4', subscriptionCount > 0 && 'flex items-center justify-between pr-3')}> + {!subscriptionCount && <CreateSubscriptionButton buttonType={CreateButtonType.FULL_BUTTON} />} + {children} + {subscriptionCount > 0 && <SubscriptionSelectorEntry + selectedId={subscriptionIdSelected} + onSelect={onSubscriptionChange} + />} + </div> +} diff --git a/web/app/components/workflow/nodes/_base/hooks/use-one-step-run.ts b/web/app/components/workflow/nodes/_base/hooks/use-one-step-run.ts index 908609c975..77d75ccc4f 100644 --- a/web/app/components/workflow/nodes/_base/hooks/use-one-step-run.ts +++ b/web/app/components/workflow/nodes/_base/hooks/use-one-step-run.ts @@ -10,7 +10,15 @@ import { import { getNodeInfoById, isConversationVar, isENV, isSystemVar, toNodeOutputVars } from '@/app/components/workflow/nodes/_base/components/variable/utils' import type { CommonNodeType, InputVar, ValueSelector, Var, Variable } from '@/app/components/workflow/types' -import { BlockEnum, InputVarType, NodeRunningStatus, VarType } from '@/app/components/workflow/types' +import { + BlockEnum, + InputVarType, + NodeRunningStatus, + VarType, + WorkflowRunningStatus, +} from '@/app/components/workflow/types' +import type { TriggerNodeType } from '@/app/components/workflow/types' +import { EVENT_WORKFLOW_STOP } from '@/app/components/workflow/variable-inspect/types' import { useStore, useWorkflowStore } from '@/app/components/workflow/store' import { fetchNodeInspectVars, getIterationSingleNodeRunUrl, getLoopSingleNodeRunUrl, singleNodeRun } from '@/service/workflow' import Toast from '@/app/components/base/toast' @@ -28,7 +36,7 @@ import ParameterExtractorDefault from '@/app/components/workflow/nodes/parameter import IterationDefault from '@/app/components/workflow/nodes/iteration/default' import DocumentExtractorDefault from '@/app/components/workflow/nodes/document-extractor/default' import LoopDefault from '@/app/components/workflow/nodes/loop/default' -import { ssePost } from '@/service/base' +import { post, ssePost } from '@/service/base' import { noop } from 'lodash-es' import { getInputVars as doGetInputVars } from '@/app/components/base/prompt-editor/constants' import type { NodeRunResult, NodeTracing } from '@/types/workflow' @@ -50,11 +58,19 @@ import { useStoreApi, } from 'reactflow' import { useInvalidLastRun } from '@/service/use-workflow' -import useInspectVarsCrud from '../../../hooks/use-inspect-vars-crud' +import useInspectVarsCrud from '@/app/components/workflow/hooks/use-inspect-vars-crud' import type { FlowType } from '@/types/common' import useMatchSchemaType from '../components/variable/use-match-schema-type' +import { useEventEmitterContextContext } from '@/context/event-emitter' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' + // eslint-disable-next-line ts/no-unsafe-function-type -const checkValidFns: Record<BlockEnum, Function> = { +const checkValidFns: Partial<Record<BlockEnum, Function>> = { [BlockEnum.LLM]: checkLLMValid, [BlockEnum.KnowledgeRetrieval]: checkKnowledgeRetrievalValid, [BlockEnum.IfElse]: checkIfElseValid, @@ -69,7 +85,12 @@ const checkValidFns: Record<BlockEnum, Function> = { [BlockEnum.Iteration]: checkIterationValid, [BlockEnum.DocExtractor]: checkDocumentExtractorValid, [BlockEnum.Loop]: checkLoopValid, -} as any +} + +type RequestError = { + message: string + status: string +} export type Params<T> = { id: string @@ -133,21 +154,23 @@ const useOneStepRun = <T>({ const availableNodesIncludeParent = getBeforeNodesInSameBranchIncludeParent(id) const workflowStore = useWorkflowStore() const { schemaTypeDefinitions } = useMatchSchemaType() + + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() + const getVar = (valueSelector: ValueSelector): Var | undefined => { const isSystem = valueSelector[0] === 'sys' const { - buildInTools, - customTools, - workflowTools, - mcpTools, dataSourceList, } = workflowStore.getState() const allPluginInfoList = { - buildInTools, - customTools, - workflowTools, - mcpTools, - dataSourceList: dataSourceList ?? [], + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], + dataSourceList: dataSourceList || [], } const allOutputVars = toNodeOutputVars(availableNodes, isChatMode, undefined, undefined, conversationVariables, [], allPluginInfoList, schemaTypeDefinitions) @@ -189,7 +212,52 @@ const useOneStepRun = <T>({ const store = useStoreApi() const { setShowSingleRunPanel, + setIsListening, + setListeningTriggerType, + setListeningTriggerNodeId, + setListeningTriggerNodeIds, + setListeningTriggerIsAll, + setShowVariableInspectPanel, } = workflowStore.getState() + const updateNodeInspectRunningState = useCallback((nodeId: string, isRunning: boolean) => { + const { + nodesWithInspectVars, + setNodesWithInspectVars, + } = workflowStore.getState() + + let hasChanges = false + const nodes = produce(nodesWithInspectVars, (draft) => { + const index = draft.findIndex(node => node.nodeId === nodeId) + if (index !== -1) { + const targetNode = draft[index] + if (targetNode.isSingRunRunning !== isRunning) { + targetNode.isSingRunRunning = isRunning + if (isRunning) + targetNode.isValueFetched = false + hasChanges = true + } + } + else if (isRunning) { + const { getNodes } = store.getState() + const target = getNodes().find(node => node.id === nodeId) + if (target) { + draft.unshift({ + nodeId, + nodeType: target.data.type, + title: target.data.title, + vars: [], + nodePayload: target.data, + isSingRunRunning: true, + isValueFetched: false, + }) + hasChanges = true + } + } + }) + + if (hasChanges) + setNodesWithInspectVars(nodes) + }, [workflowStore, store]) const invalidLastRun = useInvalidLastRun(flowType, flowId!, id) const [runResult, doSetRunResult] = useState<NodeRunResult | null>(null) const { @@ -198,10 +266,26 @@ const useOneStepRun = <T>({ invalidateConversationVarValues, } = useInspectVarsCrud() const runningStatus = data._singleRunningStatus || NodeRunningStatus.NotStart + const webhookSingleRunActiveRef = useRef(false) + const webhookSingleRunAbortRef = useRef<AbortController | null>(null) + const webhookSingleRunTimeoutRef = useRef<number | undefined>(undefined) + const webhookSingleRunTokenRef = useRef(0) + const webhookSingleRunDelayResolveRef = useRef<(() => void) | null>(null) + const pluginSingleRunActiveRef = useRef(false) + const pluginSingleRunAbortRef = useRef<AbortController | null>(null) + const pluginSingleRunTimeoutRef = useRef<number | undefined>(undefined) + const pluginSingleRunTokenRef = useRef(0) + const pluginSingleRunDelayResolveRef = useRef<(() => void) | null>(null) const isPausedRef = useRef(isPaused) useEffect(() => { isPausedRef.current = isPaused }, [isPaused]) + const { eventEmitter } = useEventEmitterContextContext() + + const isScheduleTriggerNode = data.type === BlockEnum.TriggerSchedule + const isWebhookTriggerNode = data.type === BlockEnum.TriggerWebhook + const isPluginTriggerNode = data.type === BlockEnum.TriggerPlugin + const isTriggerNode = isWebhookTriggerNode || isPluginTriggerNode || isScheduleTriggerNode const setRunResult = useCallback(async (data: NodeRunResult | null) => { const isPaused = isPausedRef.current @@ -221,13 +305,27 @@ const useOneStepRun = <T>({ const { getNodes } = store.getState() const nodes = getNodes() appendNodeInspectVars(id, vars, nodes) + updateNodeInspectRunningState(id, false) if (data?.status === NodeRunningStatus.Succeeded) { invalidLastRun() - if (isStartNode) + if (isStartNode || isTriggerNode) invalidateSysVarValues() invalidateConversationVarValues() // loop, iteration, variable assigner node can update the conversation variables, but to simple the logic(some nodes may also can update in the future), all nodes refresh. } - }, [isRunAfterSingleRun, runningStatus, flowId, id, store, appendNodeInspectVars, invalidLastRun, isStartNode, invalidateSysVarValues, invalidateConversationVarValues]) + }, [ + isRunAfterSingleRun, + runningStatus, + flowId, + id, + store, + appendNodeInspectVars, + updateNodeInspectRunningState, + invalidLastRun, + isStartNode, + isTriggerNode, + invalidateSysVarValues, + invalidateConversationVarValues, + ]) const { handleNodeDataUpdate }: { handleNodeDataUpdate: (data: any) => void } = useNodeDataUpdate() const setNodeRunning = () => { @@ -239,6 +337,299 @@ const useOneStepRun = <T>({ }, }) } + + const cancelWebhookSingleRun = useCallback(() => { + webhookSingleRunActiveRef.current = false + webhookSingleRunTokenRef.current += 1 + if (webhookSingleRunAbortRef.current) + webhookSingleRunAbortRef.current.abort() + webhookSingleRunAbortRef.current = null + if (webhookSingleRunTimeoutRef.current !== undefined) { + window.clearTimeout(webhookSingleRunTimeoutRef.current) + webhookSingleRunTimeoutRef.current = undefined + } + if (webhookSingleRunDelayResolveRef.current) { + webhookSingleRunDelayResolveRef.current() + webhookSingleRunDelayResolveRef.current = null + } + }, []) + + const cancelPluginSingleRun = useCallback(() => { + pluginSingleRunActiveRef.current = false + pluginSingleRunTokenRef.current += 1 + if (pluginSingleRunAbortRef.current) + pluginSingleRunAbortRef.current.abort() + pluginSingleRunAbortRef.current = null + if (pluginSingleRunTimeoutRef.current !== undefined) { + window.clearTimeout(pluginSingleRunTimeoutRef.current) + pluginSingleRunTimeoutRef.current = undefined + } + if (pluginSingleRunDelayResolveRef.current) { + pluginSingleRunDelayResolveRef.current() + pluginSingleRunDelayResolveRef.current = null + } + }, []) + + const startTriggerListening = useCallback(() => { + if (!isTriggerNode) + return + + setIsListening(true) + setShowVariableInspectPanel(true) + setListeningTriggerType(data.type as TriggerNodeType) + setListeningTriggerNodeId(id) + setListeningTriggerNodeIds([id]) + setListeningTriggerIsAll(false) + }, [ + isTriggerNode, + setIsListening, + setShowVariableInspectPanel, + setListeningTriggerType, + data.type, + setListeningTriggerNodeId, + id, + setListeningTriggerNodeIds, + setListeningTriggerIsAll, + ]) + + const stopTriggerListening = useCallback(() => { + if (!isTriggerNode) + return + + setIsListening(false) + setListeningTriggerType(null) + setListeningTriggerNodeId(null) + setListeningTriggerNodeIds([]) + setListeningTriggerIsAll(false) + }, [ + isTriggerNode, + setIsListening, + setListeningTriggerType, + setListeningTriggerNodeId, + setListeningTriggerNodeIds, + setListeningTriggerIsAll, + ]) + + const runScheduleSingleRun = useCallback(async (): Promise<NodeRunResult | null> => { + const urlPath = `/apps/${flowId}/workflows/draft/nodes/${id}/trigger/run` + + try { + const response: any = await post(urlPath, { + body: JSON.stringify({}), + }) + + if (!response) { + const message = 'Schedule trigger run failed' + Toast.notify({ type: 'error', message }) + throw new Error(message) + } + + if (response?.status === 'error') { + const message = response?.message || 'Schedule trigger run failed' + Toast.notify({ type: 'error', message }) + throw new Error(message) + } + + handleNodeDataUpdate({ + id, + data: { + ...data, + _isSingleRun: false, + _singleRunningStatus: NodeRunningStatus.Succeeded, + }, + }) + + return response as NodeRunResult + } + catch (error) { + console.error('handleRun: schedule trigger single run error', error) + handleNodeDataUpdate({ + id, + data: { + ...data, + _isSingleRun: false, + _singleRunningStatus: NodeRunningStatus.Failed, + }, + }) + Toast.notify({ type: 'error', message: 'Schedule trigger run failed' }) + throw error + } + }, [flowId, id, handleNodeDataUpdate, data]) + + const runWebhookSingleRun = useCallback(async (): Promise<any | null> => { + const urlPath = `/apps/${flowId}/workflows/draft/nodes/${id}/trigger/run` + + webhookSingleRunActiveRef.current = true + const token = ++webhookSingleRunTokenRef.current + + while (webhookSingleRunActiveRef.current && token === webhookSingleRunTokenRef.current) { + const controller = new AbortController() + webhookSingleRunAbortRef.current = controller + + try { + const response: any = await post(urlPath, { + body: JSON.stringify({}), + signal: controller.signal, + }) + + if (!webhookSingleRunActiveRef.current || token !== webhookSingleRunTokenRef.current) + return null + + if (!response) { + const message = response?.message || 'Webhook debug failed' + Toast.notify({ type: 'error', message }) + cancelWebhookSingleRun() + throw new Error(message) + } + + if (response?.status === 'waiting') { + const delay = Number(response.retry_in) || 2000 + webhookSingleRunAbortRef.current = null + if (!webhookSingleRunActiveRef.current || token !== webhookSingleRunTokenRef.current) + return null + + await new Promise<void>((resolve) => { + const timeoutId = window.setTimeout(resolve, delay) + webhookSingleRunTimeoutRef.current = timeoutId + webhookSingleRunDelayResolveRef.current = resolve + controller.signal.addEventListener('abort', () => { + window.clearTimeout(timeoutId) + resolve() + }, { once: true }) + }) + + webhookSingleRunTimeoutRef.current = undefined + webhookSingleRunDelayResolveRef.current = null + continue + } + + if (response?.status === 'error') { + const message = response.message || 'Webhook debug failed' + Toast.notify({ type: 'error', message }) + cancelWebhookSingleRun() + throw new Error(message) + } + + handleNodeDataUpdate({ + id, + data: { + ...data, + _isSingleRun: false, + _singleRunningStatus: NodeRunningStatus.Listening, + }, + }) + + cancelWebhookSingleRun() + return response + } + catch (error) { + if (controller.signal.aborted && (!webhookSingleRunActiveRef.current || token !== webhookSingleRunTokenRef.current)) + return null + if (controller.signal.aborted) + return null + + Toast.notify({ type: 'error', message: 'Webhook debug request failed' }) + cancelWebhookSingleRun() + if (error instanceof Error) + throw error + throw new Error(String(error)) + } + finally { + webhookSingleRunAbortRef.current = null + } + } + + return null + }, [flowId, id, data, handleNodeDataUpdate, cancelWebhookSingleRun]) + + const runPluginSingleRun = useCallback(async (): Promise<any | null> => { + const urlPath = `/apps/${flowId}/workflows/draft/nodes/${id}/trigger/run` + + pluginSingleRunActiveRef.current = true + const token = ++pluginSingleRunTokenRef.current + + while (pluginSingleRunActiveRef.current && token === pluginSingleRunTokenRef.current) { + const controller = new AbortController() + pluginSingleRunAbortRef.current = controller + + let requestError: RequestError | undefined + const response: any = await post(urlPath, { + body: JSON.stringify({}), + signal: controller.signal, + }).catch(async (error: Response) => { + const data = await error.clone().json() as Record<string, any> + const { error: respError, status } = data || {} + requestError = { + message: respError, + status, + } + return null + }).finally(() => { + pluginSingleRunAbortRef.current = null + }) + + if (!pluginSingleRunActiveRef.current || token !== pluginSingleRunTokenRef.current) + return null + + if (requestError) { + if (controller.signal.aborted) + return null + + Toast.notify({ type: 'error', message: requestError.message }) + cancelPluginSingleRun() + throw requestError + } + + if (!response) { + const message = 'Plugin debug failed' + Toast.notify({ type: 'error', message }) + cancelPluginSingleRun() + throw new Error(message) + } + + if (response?.status === 'waiting') { + const delay = Number(response.retry_in) || 2000 + if (!pluginSingleRunActiveRef.current || token !== pluginSingleRunTokenRef.current) + return null + + await new Promise<void>((resolve) => { + const timeoutId = window.setTimeout(resolve, delay) + pluginSingleRunTimeoutRef.current = timeoutId + pluginSingleRunDelayResolveRef.current = resolve + controller.signal.addEventListener('abort', () => { + window.clearTimeout(timeoutId) + resolve() + }, { once: true }) + }) + + pluginSingleRunTimeoutRef.current = undefined + pluginSingleRunDelayResolveRef.current = null + continue + } + + if (response?.status === 'error') { + const message = response.message || 'Plugin debug failed' + Toast.notify({ type: 'error', message }) + cancelPluginSingleRun() + throw new Error(message) + } + + handleNodeDataUpdate({ + id, + data: { + ...data, + _isSingleRun: false, + _singleRunningStatus: NodeRunningStatus.Listening, + }, + }) + + cancelPluginSingleRun() + return response + } + + return null + }, [flowId, id, data, handleNodeDataUpdate, cancelPluginSingleRun]) + const checkValidWrap = () => { if (!checkValid) return { isValid: true, errorMessage: '' } @@ -253,7 +644,7 @@ const useOneStepRun = <T>({ }) Toast.notify({ type: 'error', - message: res.errorMessage, + message: res.errorMessage || '', }) } return res @@ -300,33 +691,84 @@ const useOneStepRun = <T>({ const isCompleted = runningStatus === NodeRunningStatus.Succeeded || runningStatus === NodeRunningStatus.Failed const handleRun = async (submitData: Record<string, any>) => { + if (isWebhookTriggerNode) + cancelWebhookSingleRun() + if (isPluginTriggerNode) + cancelPluginSingleRun() + + updateNodeInspectRunningState(id, true) + + if (isTriggerNode) + startTriggerListening() + else + stopTriggerListening() + handleNodeDataUpdate({ id, data: { ...data, _isSingleRun: false, - _singleRunningStatus: NodeRunningStatus.Running, + _singleRunningStatus: isTriggerNode + ? NodeRunningStatus.Listening + : NodeRunningStatus.Running, }, }) let res: any let hasError = false try { if (!isIteration && !isLoop) { - const isStartNode = data.type === BlockEnum.Start - const postData: Record<string, any> = {} - if (isStartNode) { - const { '#sys.query#': query, '#sys.files#': files, ...inputs } = submitData - if (isChatMode) - postData.conversation_id = '' - - postData.inputs = inputs - postData.query = query - postData.files = files || [] + if (isScheduleTriggerNode) { + res = await runScheduleSingleRun() + } + else if (isWebhookTriggerNode) { + res = await runWebhookSingleRun() + if (!res) { + if (webhookSingleRunActiveRef.current) { + handleNodeDataUpdate({ + id, + data: { + ...data, + _isSingleRun: false, + _singleRunningStatus: NodeRunningStatus.Stopped, + }, + }) + } + return false + } + } + else if (isPluginTriggerNode) { + res = await runPluginSingleRun() + if (!res) { + if (pluginSingleRunActiveRef.current) { + handleNodeDataUpdate({ + id, + data: { + ...data, + _isSingleRun: false, + _singleRunningStatus: NodeRunningStatus.Stopped, + }, + }) + } + return false + } } else { - postData.inputs = submitData + const isStartNode = data.type === BlockEnum.Start + const postData: Record<string, any> = {} + if (isStartNode) { + const { '#sys.query#': query, '#sys.files#': files, ...inputs } = submitData + if (isChatMode) + postData.conversation_id = '' + + postData.inputs = inputs + postData.query = query + postData.files = files || [] + } + else { + postData.inputs = submitData + } + res = await singleNodeRun(flowType, flowId!, id, postData) as any } - res = await singleNodeRun(flowType, flowId!, id, postData) as any } else if (isIteration) { setIterationRunResult([]) @@ -557,6 +999,14 @@ const useOneStepRun = <T>({ } } finally { + if (isWebhookTriggerNode) + cancelWebhookSingleRun() + if (isPluginTriggerNode) + cancelPluginSingleRun() + if (isTriggerNode) + stopTriggerListening() + if (!isIteration && !isLoop) + updateNodeInspectRunningState(id, false) if (!isPausedRef.current && !isIteration && !isLoop && res) { setRunResult({ ...res, @@ -582,15 +1032,55 @@ const useOneStepRun = <T>({ } } - const handleStop = () => { + const handleStop = useCallback(() => { + if (isTriggerNode) { + const isTriggerActive = runningStatus === NodeRunningStatus.Listening + || webhookSingleRunActiveRef.current + || pluginSingleRunActiveRef.current + if (!isTriggerActive) + return + } + else if (runningStatus !== NodeRunningStatus.Running) { + return + } + + cancelWebhookSingleRun() + cancelPluginSingleRun() handleNodeDataUpdate({ id, data: { - ...data, - _singleRunningStatus: NodeRunningStatus.NotStart, + _isSingleRun: false, + _singleRunningStatus: NodeRunningStatus.Stopped, }, }) - } + stopTriggerListening() + updateNodeInspectRunningState(id, false) + const { + workflowRunningData, + setWorkflowRunningData, + nodesWithInspectVars, + deleteNodeInspectVars, + } = workflowStore.getState() + if (workflowRunningData) { + setWorkflowRunningData(produce(workflowRunningData, (draft) => { + draft.result.status = WorkflowRunningStatus.Stopped + })) + } + + const inspectNode = nodesWithInspectVars.find(node => node.nodeId === id) + if (inspectNode && !inspectNode.isValueFetched && (!inspectNode.vars || inspectNode.vars.length === 0)) + deleteNodeInspectVars(id) + }, [ + isTriggerNode, + runningStatus, + cancelWebhookSingleRun, + cancelPluginSingleRun, + handleNodeDataUpdate, + id, + stopTriggerListening, + updateNodeInspectRunningState, + workflowStore, + ]) const toVarInputs = (variables: Variable[]): InputVar[] => { if (!variables) @@ -653,6 +1143,11 @@ const useOneStepRun = <T>({ }) } + eventEmitter?.useSubscription((v: any) => { + if (v.type === EVENT_WORKFLOW_STOP) + handleStop() + }) + return { isShowSingleRun, hideSingleRun, diff --git a/web/app/components/workflow/nodes/_base/node.tsx b/web/app/components/workflow/nodes/_base/node.tsx index 4725f86ad5..73f78401ac 100644 --- a/web/app/components/workflow/nodes/_base/node.tsx +++ b/web/app/components/workflow/nodes/_base/node.tsx @@ -16,23 +16,18 @@ import { RiLoader2Line, } from '@remixicon/react' import { useTranslation } from 'react-i18next' -import type { NodeProps } from '../../types' +import type { NodeProps } from '@/app/components/workflow/types' import { BlockEnum, NodeRunningStatus, -} from '../../types' -import { - useNodesReadOnly, - useToolIcon, -} from '../../hooks' -import { - hasErrorHandleNode, - hasRetryNode, -} from '../../utils' -import { useNodeIterationInteractions } from '../iteration/use-interactions' -import { useNodeLoopInteractions } from '../loop/use-interactions' -import type { IterationNodeType } from '../iteration/types' -import CopyID from '../tool/components/copy-id' + isTriggerNode, +} from '@/app/components/workflow/types' +import { useNodesReadOnly, useToolIcon } from '@/app/components/workflow/hooks' +import { hasErrorHandleNode, hasRetryNode } from '@/app/components/workflow/utils' +import { useNodeIterationInteractions } from '@/app/components/workflow/nodes/iteration/use-interactions' +import { useNodeLoopInteractions } from '@/app/components/workflow/nodes/loop/use-interactions' +import type { IterationNodeType } from '@/app/components/workflow/nodes/iteration/types' +import CopyID from '@/app/components/workflow/nodes/tool/components/copy-id' import { NodeSourceHandle, NodeTargetHandle, @@ -42,11 +37,12 @@ import NodeControl from './components/node-control' import ErrorHandleOnNode from './components/error-handle/error-handle-on-node' import RetryOnNode from './components/retry/retry-on-node' import AddVariablePopupWithPosition from './components/add-variable-popup-with-position' +import EntryNodeContainer, { StartNodeTypeEnum } from './components/entry-node-container' import cn from '@/utils/classnames' import BlockIcon from '@/app/components/workflow/block-icon' import Tooltip from '@/app/components/base/tooltip' -import useInspectVarsCrud from '../../hooks/use-inspect-vars-crud' -import { ToolTypeEnum } from '../../block-selector/types' +import useInspectVarsCrud from '@/app/components/workflow/hooks/use-inspect-vars-crud' +import { ToolTypeEnum } from '@/app/components/workflow/block-selector/types' type NodeChildProps = { id: string @@ -67,6 +63,7 @@ const BaseNode: FC<BaseNodeProps> = ({ const { t } = useTranslation() const nodeRef = useRef<HTMLDivElement>(null) const { nodesReadOnly } = useNodesReadOnly() + const { handleNodeIterationChildSizeChange } = useNodeIterationInteractions() const { handleNodeLoopChildSizeChange } = useNodeLoopInteractions() const toolIcon = useToolIcon(data) @@ -141,13 +138,13 @@ const BaseNode: FC<BaseNodeProps> = ({ return null }, [data._loopIndex, data._runningStatus, t]) - return ( + const nodeContent = ( <div className={cn( 'relative flex rounded-2xl border', showSelectedBorder ? 'border-components-option-card-option-selected-border' : 'border-transparent', data._waitingRun && 'opacity-70', - data._dimmed && 'opacity-30', + data._pluginInstallLocked && 'cursor-not-allowed', )} ref={nodeRef} style={{ @@ -155,6 +152,17 @@ const BaseNode: FC<BaseNodeProps> = ({ height: (data.type === BlockEnum.Iteration || data.type === BlockEnum.Loop) ? data.height : 'auto', }} > + {(data._dimmed || data._pluginInstallLocked) && ( + <div + className={cn( + 'absolute inset-0 rounded-2xl transition-opacity', + data._pluginInstallLocked + ? 'pointer-events-auto z-30 bg-workflow-block-parma-bg opacity-80 backdrop-blur-[2px]' + : 'pointer-events-none z-20 bg-workflow-block-parma-bg opacity-50', + )} + data-testid='workflow-node-install-overlay' + /> + )} { data.type === BlockEnum.DataSource && ( <div className='absolute inset-[-2px] top-[-22px] z-[-1] rounded-[18px] bg-node-data-source-bg p-0.5 backdrop-blur-[6px]'> @@ -297,13 +305,13 @@ const BaseNode: FC<BaseNodeProps> = ({ </div> { data.type !== BlockEnum.Iteration && data.type !== BlockEnum.Loop && ( - cloneElement(children, { id, data }) + cloneElement(children, { id, data } as any) ) } { (data.type === BlockEnum.Iteration || data.type === BlockEnum.Loop) && ( <div className='grow pb-1 pl-1 pr-1'> - {cloneElement(children, { id, data })} + {cloneElement(children, { id, data } as any)} </div> ) } @@ -338,6 +346,17 @@ const BaseNode: FC<BaseNodeProps> = ({ </div> </div> ) + + const isStartNode = data.type === BlockEnum.Start + const isEntryNode = isTriggerNode(data.type as any) || isStartNode + + return isEntryNode ? ( + <EntryNodeContainer + nodeType={isStartNode ? StartNodeTypeEnum.Start : StartNodeTypeEnum.Trigger} + > + {nodeContent} + </EntryNodeContainer> + ) : nodeContent } export default memo(BaseNode) diff --git a/web/app/components/workflow/nodes/_base/types.ts b/web/app/components/workflow/nodes/_base/types.ts new file mode 100644 index 0000000000..18ad9c4e71 --- /dev/null +++ b/web/app/components/workflow/nodes/_base/types.ts @@ -0,0 +1,27 @@ +import type { ValueSelector } from '@/app/components/workflow/types' + +// Generic variable types for all resource forms +export enum VarKindType { + variable = 'variable', + constant = 'constant', + mixed = 'mixed', +} + +// Generic resource variable inputs +export type ResourceVarInputs = Record<string, { + type: VarKindType + value?: string | ValueSelector | any +}> + +// Base resource interface +export type BaseResource = { + name: string + [key: string]: any +} + +// Base resource provider interface +export type BaseResourceProvider = { + plugin_id?: string + name: string + [key: string]: any +} diff --git a/web/app/components/workflow/nodes/agent/node.tsx b/web/app/components/workflow/nodes/agent/node.tsx index a2190317af..fe87bc7cda 100644 --- a/web/app/components/workflow/nodes/agent/node.tsx +++ b/web/app/components/workflow/nodes/agent/node.tsx @@ -65,7 +65,7 @@ const AgentNode: FC<NodeProps<AgentNodeType>> = (props) => { }) return tools }, [currentStrategy?.parameters, inputs.agent_parameters]) - return <div className='mb-1 space-y-1 px-3 py-1'> + return <div className='mb-1 space-y-1 px-3'> {inputs.agent_strategy_name ? <SettingItem label={t('workflow.nodes.agent.strategy.shortLabel')} diff --git a/web/app/components/workflow/nodes/assigner/components/var-list/index.tsx b/web/app/components/workflow/nodes/assigner/components/var-list/index.tsx index bfb48d4eb2..b81277d740 100644 --- a/web/app/components/workflow/nodes/assigner/components/var-list/index.tsx +++ b/web/app/components/workflow/nodes/assigner/components/var-list/index.tsx @@ -68,7 +68,7 @@ const VarList: FC<Props> = ({ draft[index].value = '' // Clear value when operation changes if (item.value === WriteMode.set || item.value === WriteMode.increment || item.value === WriteMode.decrement || item.value === WriteMode.multiply || item.value === WriteMode.divide) { - if(varType === VarType.boolean) + if (varType === VarType.boolean) draft[index].value = false draft[index].input_type = AssignerNodeInputType.constant } diff --git a/web/app/components/workflow/nodes/components.ts b/web/app/components/workflow/nodes/components.ts index cdf3a21598..d8da8b9dae 100644 --- a/web/app/components/workflow/nodes/components.ts +++ b/web/app/components/workflow/nodes/components.ts @@ -42,6 +42,12 @@ import DataSourceNode from './data-source/node' import DataSourcePanel from './data-source/panel' import KnowledgeBaseNode from './knowledge-base/node' import KnowledgeBasePanel from './knowledge-base/panel' +import TriggerScheduleNode from './trigger-schedule/node' +import TriggerSchedulePanel from './trigger-schedule/panel' +import TriggerWebhookNode from './trigger-webhook/node' +import TriggerWebhookPanel from './trigger-webhook/panel' +import TriggerPluginNode from './trigger-plugin/node' +import TriggerPluginPanel from './trigger-plugin/panel' export const NodeComponentMap: Record<string, ComponentType<any>> = { [BlockEnum.Start]: StartNode, @@ -66,6 +72,9 @@ export const NodeComponentMap: Record<string, ComponentType<any>> = { [BlockEnum.Agent]: AgentNode, [BlockEnum.DataSource]: DataSourceNode, [BlockEnum.KnowledgeBase]: KnowledgeBaseNode, + [BlockEnum.TriggerSchedule]: TriggerScheduleNode, + [BlockEnum.TriggerWebhook]: TriggerWebhookNode, + [BlockEnum.TriggerPlugin]: TriggerPluginNode, } export const PanelComponentMap: Record<string, ComponentType<any>> = { @@ -91,4 +100,7 @@ export const PanelComponentMap: Record<string, ComponentType<any>> = { [BlockEnum.Agent]: AgentPanel, [BlockEnum.DataSource]: DataSourcePanel, [BlockEnum.KnowledgeBase]: KnowledgeBasePanel, + [BlockEnum.TriggerSchedule]: TriggerSchedulePanel, + [BlockEnum.TriggerWebhook]: TriggerWebhookPanel, + [BlockEnum.TriggerPlugin]: TriggerPluginPanel, } diff --git a/web/app/components/workflow/nodes/constants.ts b/web/app/components/workflow/nodes/constants.ts index 78684577f2..b09b27343a 100644 --- a/web/app/components/workflow/nodes/constants.ts +++ b/web/app/components/workflow/nodes/constants.ts @@ -1,5 +1,7 @@ import { TransferMethod } from '@/types/app' +export const CUSTOM_NODE_TYPE = 'custom' + export const FILE_TYPE_OPTIONS = [ { value: 'image', i18nKey: 'image' }, { value: 'document', i18nKey: 'doc' }, diff --git a/web/app/components/workflow/nodes/data-source-empty/hooks.ts b/web/app/components/workflow/nodes/data-source-empty/hooks.ts index e22e87485c..a17f0b2acb 100644 --- a/web/app/components/workflow/nodes/data-source-empty/hooks.ts +++ b/web/app/components/workflow/nodes/data-source-empty/hooks.ts @@ -11,7 +11,7 @@ export const useReplaceDataSourceNode = (id: string) => { const handleReplaceNode = useCallback<OnSelectBlock>(( type, - toolDefaultValue, + pluginDefaultValue, ) => { const { getNodes, @@ -28,7 +28,7 @@ export const useReplaceDataSourceNode = (id: string) => { const { newNode } = generateNewNode({ data: { ...(defaultValue as any), - ...toolDefaultValue, + ...pluginDefaultValue, }, position: { x: emptyNode.position.x, diff --git a/web/app/components/workflow/nodes/data-source/node.tsx b/web/app/components/workflow/nodes/data-source/node.tsx index f97098e52f..b490aea2a9 100644 --- a/web/app/components/workflow/nodes/data-source/node.tsx +++ b/web/app/components/workflow/nodes/data-source/node.tsx @@ -1,10 +1,57 @@ import type { FC } from 'react' -import { memo } from 'react' -import type { DataSourceNodeType } from './types' +import { memo, useEffect } from 'react' import type { NodeProps } from '@/app/components/workflow/types' -const Node: FC<NodeProps<DataSourceNodeType>> = () => { +import { InstallPluginButton } from '@/app/components/workflow/nodes/_base/components/install-plugin-button' +import { useNodePluginInstallation } from '@/app/components/workflow/hooks/use-node-plugin-installation' +import { useNodeDataUpdate } from '@/app/components/workflow/hooks/use-node-data-update' +import type { DataSourceNodeType } from './types' + +const Node: FC<NodeProps<DataSourceNodeType>> = ({ + id, + data, +}) => { + const { + isChecking, + isMissing, + uniqueIdentifier, + canInstall, + onInstallSuccess, + shouldDim, + } = useNodePluginInstallation(data) + const { handleNodeDataUpdate } = useNodeDataUpdate() + const shouldLock = !isChecking && isMissing && canInstall && Boolean(uniqueIdentifier) + + useEffect(() => { + if (data._pluginInstallLocked === shouldLock && data._dimmed === shouldDim) + return + handleNodeDataUpdate({ + id, + data: { + _pluginInstallLocked: shouldLock, + _dimmed: shouldDim, + }, + }) + }, [data._pluginInstallLocked, data._dimmed, handleNodeDataUpdate, id, shouldDim, shouldLock]) + + const showInstallButton = !isChecking && isMissing && canInstall && uniqueIdentifier + + if (!showInstallButton) + return null + return ( - <div> + <div className='relative mb-1 px-3 py-1'> + <div className='pointer-events-auto absolute right-3 top-[-32px] z-40'> + <InstallPluginButton + size='small' + extraIdentifiers={[ + data.plugin_id, + data.provider_name, + ].filter(Boolean) as string[]} + className='!font-medium !text-text-accent' + uniqueIdentifier={uniqueIdentifier!} + onSuccess={onInstallSuccess} + /> + </div> </div> ) } diff --git a/web/app/components/workflow/nodes/data-source/types.ts b/web/app/components/workflow/nodes/data-source/types.ts index da887244b8..d0bc034b89 100644 --- a/web/app/components/workflow/nodes/data-source/types.ts +++ b/web/app/components/workflow/nodes/data-source/types.ts @@ -1,13 +1,9 @@ -import type { CommonNodeType, Node, ValueSelector } from '@/app/components/workflow/types' +import type { CommonNodeType, Node } from '@/app/components/workflow/types' import type { FlowType } from '@/types/common' import type { NodeRunResult, VarInInspect } from '@/types/workflow' import type { Dispatch, SetStateAction } from 'react' - -export enum VarType { - variable = 'variable', - constant = 'constant', - mixed = 'mixed', -} +import type { ResourceVarInputs } from '../_base/types' +export { VarKindType as VarType } from '../_base/types' export enum DataSourceClassification { localFile = 'local_file', @@ -16,10 +12,7 @@ export enum DataSourceClassification { onlineDrive = 'online_drive', } -export type ToolVarInputs = Record<string, { - type: VarType - value?: string | ValueSelector | any -}> +export type ToolVarInputs = ResourceVarInputs export type DataSourceNodeType = CommonNodeType & { fileExtensions?: string[] @@ -30,6 +23,7 @@ export type DataSourceNodeType = CommonNodeType & { datasource_label: string datasource_parameters: ToolVarInputs datasource_configurations: Record<string, any> + plugin_unique_identifier?: string } export type CustomRunFormProps = { diff --git a/web/app/components/workflow/nodes/document-extractor/node.tsx b/web/app/components/workflow/nodes/document-extractor/node.tsx index ab7fe9a9a6..a0437a4f54 100644 --- a/web/app/components/workflow/nodes/document-extractor/node.tsx +++ b/web/app/components/workflow/nodes/document-extractor/node.tsx @@ -25,7 +25,7 @@ const NodeComponent: FC<NodeProps<DocExtractorNodeType>> = ({ const isSystem = isSystemVar(variable) const node = isSystem ? nodes.find(node => node.data.type === BlockEnum.Start) : nodes.find(node => node.id === variable[0]) return ( - <div className='relative px-3'> + <div className='relative mb-1 px-3 py-1'> <div className='system-2xs-medium-uppercase mb-1 text-text-tertiary'>{t(`${i18nPrefix}.inputVar`)}</div> <VariableLabelInNode variables={variable} diff --git a/web/app/components/workflow/nodes/end/default.ts b/web/app/components/workflow/nodes/end/default.ts index cadb580c34..881c16986b 100644 --- a/web/app/components/workflow/nodes/end/default.ts +++ b/web/app/components/workflow/nodes/end/default.ts @@ -6,17 +6,34 @@ import { BlockEnum } from '@/app/components/workflow/types' const metaData = genNodeMetaData({ sort: 2.1, type: BlockEnum.End, - isRequired: true, + isRequired: false, }) const nodeDefault: NodeDefault<EndNodeType> = { metaData, defaultValue: { outputs: [], }, - checkValid() { + checkValid(payload: EndNodeType, t: any) { + const outputs = payload.outputs || [] + + let errorMessage = '' + if (!outputs.length) { + errorMessage = t('workflow.errorMsg.fieldRequired', { field: t('workflow.nodes.end.output.variable') }) + } + else { + const invalidOutput = outputs.find((output) => { + const variableName = output.variable?.trim() + const hasSelector = Array.isArray(output.value_selector) && output.value_selector.length > 0 + return !variableName || !hasSelector + }) + + if (invalidOutput) + errorMessage = t('workflow.errorMsg.fieldRequired', { field: t('workflow.nodes.end.output.variable') }) + } + return { - isValid: true, - errorMessage: '', + isValid: !errorMessage, + errorMessage, } }, } diff --git a/web/app/components/workflow/nodes/end/panel.tsx b/web/app/components/workflow/nodes/end/panel.tsx index 2ad90ff5ac..420280d7c5 100644 --- a/web/app/components/workflow/nodes/end/panel.tsx +++ b/web/app/components/workflow/nodes/end/panel.tsx @@ -30,6 +30,7 @@ const Panel: FC<NodePanelProps<EndNodeType>> = ({ <Field title={t(`${i18nPrefix}.output.variable`)} + required operations={ !readOnly ? <AddButton onClick={handleAddVariable} /> : undefined } diff --git a/web/app/components/workflow/nodes/if-else/components/condition-list/condition-item.tsx b/web/app/components/workflow/nodes/if-else/components/condition-list/condition-item.tsx index 9bcd4b9671..65dac6f5be 100644 --- a/web/app/components/workflow/nodes/if-else/components/condition-list/condition-item.tsx +++ b/web/app/components/workflow/nodes/if-else/components/condition-list/condition-item.tsx @@ -42,6 +42,12 @@ import BoolValue from '@/app/components/workflow/panel/chat-variable-panel/compo import { getVarType } from '@/app/components/workflow/nodes/_base/components/variable/utils' import { useIsChatMode } from '@/app/components/workflow/hooks/use-workflow' import useMatchSchemaType from '../../../_base/components/variable/use-match-schema-type' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' const optionNameI18NPrefix = 'workflow.nodes.ifElse.optionName' type ConditionItemProps = { @@ -91,15 +97,12 @@ const ConditionItem = ({ const [isHovered, setIsHovered] = useState(false) const [open, setOpen] = useState(false) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() + const workflowStore = useWorkflowStore() - const { - setControlPromptEditorRerenderKey, - buildInTools, - customTools, - mcpTools, - workflowTools, - dataSourceList, - } = workflowStore.getState() const doUpdateCondition = useCallback((newCondition: Condition) => { if (isSubVariableKey) @@ -213,6 +216,8 @@ const ConditionItem = ({ const handleVarChange = useCallback((valueSelector: ValueSelector, _varItem: Var) => { const { conversationVariables, + setControlPromptEditorRerenderKey, + dataSourceList, } = workflowStore.getState() const resolvedVarType = getVarType({ valueSelector, @@ -220,11 +225,11 @@ const ConditionItem = ({ availableNodes, isChatMode, allPluginInfoList: { - buildInTools, - customTools, - mcpTools, - workflowTools, - dataSourceList: dataSourceList ?? [], + buildInTools: buildInTools || [], + customTools: customTools || [], + mcpTools: mcpTools || [], + workflowTools: workflowTools || [], + dataSourceList: dataSourceList || [], }, schemaTypeDefinitions, }) @@ -241,12 +246,12 @@ const ConditionItem = ({ }) doUpdateCondition(newCondition) setOpen(false) - }, [condition, doUpdateCondition, availableNodes, isChatMode, setControlPromptEditorRerenderKey, schemaTypeDefinitions]) + }, [condition, doUpdateCondition, availableNodes, isChatMode, schemaTypeDefinitions, buildInTools, customTools, mcpTools, workflowTools]) const showBooleanInput = useMemo(() => { if(condition.varType === VarType.boolean) return true - // eslint-disable-next-line sonarjs/prefer-single-boolean-return + if(condition.varType === VarType.arrayBoolean && [ComparisonOperator.contains, ComparisonOperator.notContains].includes(condition.comparison_operator!)) return true return false diff --git a/web/app/components/workflow/nodes/iteration/add-block.tsx b/web/app/components/workflow/nodes/iteration/add-block.tsx index 10aa8bb3e2..05d69caef4 100644 --- a/web/app/components/workflow/nodes/iteration/add-block.tsx +++ b/web/app/components/workflow/nodes/iteration/add-block.tsx @@ -33,11 +33,11 @@ const AddBlock = ({ const { handleNodeAdd } = useNodesInteractions() const { availableNextBlocks } = useAvailableBlocks(BlockEnum.Start, true) - const handleSelect = useCallback<OnSelectBlock>((type, toolDefaultValue) => { + const handleSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { handleNodeAdd( { nodeType: type, - toolDefaultValue, + pluginDefaultValue, }, { prevNodeId: iterationNodeData.start_node_id, diff --git a/web/app/components/workflow/nodes/iteration/default.ts b/web/app/components/workflow/nodes/iteration/default.ts index 450379ec6b..c375dbdcbf 100644 --- a/web/app/components/workflow/nodes/iteration/default.ts +++ b/web/app/components/workflow/nodes/iteration/default.ts @@ -22,6 +22,7 @@ const nodeDefault: NodeDefault<IterationNodeType> = { is_parallel: false, parallel_nums: 10, error_handle_mode: ErrorHandleMode.Terminated, + flatten_output: true, }, checkValid(payload: IterationNodeType, t: any) { let errorMessages = '' diff --git a/web/app/components/workflow/nodes/iteration/panel.tsx b/web/app/components/workflow/nodes/iteration/panel.tsx index 23e93b0dd5..63e0d5f8cd 100644 --- a/web/app/components/workflow/nodes/iteration/panel.tsx +++ b/web/app/components/workflow/nodes/iteration/panel.tsx @@ -46,6 +46,7 @@ const Panel: FC<NodePanelProps<IterationNodeType>> = ({ changeParallel, changeErrorResponseMode, changeParallelNums, + changeFlattenOutput, } = useConfig(id, data) return ( @@ -117,6 +118,18 @@ const Panel: FC<NodePanelProps<IterationNodeType>> = ({ <Select items={responseMethod} defaultValue={inputs.error_handle_mode} onSelect={changeErrorResponseMode} allowSearch={false} /> </Field> </div> + + <Split /> + + <div className='px-4 py-2'> + <Field + title={t(`${i18nPrefix}.flattenOutput`)} + tooltip={<div className='w-[230px]'>{t(`${i18nPrefix}.flattenOutputDesc`)}</div>} + inline + > + <Switch defaultValue={inputs.flatten_output} onChange={changeFlattenOutput} /> + </Field> + </div> </div> ) } diff --git a/web/app/components/workflow/nodes/iteration/types.ts b/web/app/components/workflow/nodes/iteration/types.ts index 9a68050d6f..f35b838b83 100644 --- a/web/app/components/workflow/nodes/iteration/types.ts +++ b/web/app/components/workflow/nodes/iteration/types.ts @@ -17,5 +17,6 @@ export type IterationNodeType = CommonNodeType & { is_parallel: boolean // open the parallel mode or not parallel_nums: number // the numbers of parallel error_handle_mode: ErrorHandleMode // how to handle error in the iteration + flatten_output: boolean // whether to flatten the output array if all elements are lists _isShowTips: boolean // when answer node in parallel mode iteration show tips } diff --git a/web/app/components/workflow/nodes/iteration/use-config.ts b/web/app/components/workflow/nodes/iteration/use-config.ts index 5d449a4756..2e47bb3740 100644 --- a/web/app/components/workflow/nodes/iteration/use-config.ts +++ b/web/app/components/workflow/nodes/iteration/use-config.ts @@ -15,6 +15,12 @@ import type { Item } from '@/app/components/base/select' import useInspectVarsCrud from '../../hooks/use-inspect-vars-crud' import { isEqual } from 'lodash-es' import { useStore } from '../../store' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' const useConfig = (id: string, payload: IterationNodeType) => { const { @@ -40,17 +46,17 @@ const useConfig = (id: string, payload: IterationNodeType) => { // output const { getIterationNodeChildren } = useWorkflow() const iterationChildrenNodes = getIterationNodeChildren(id) - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() const dataSourceList = useStore(s => s.dataSourceList) const allPluginInfoList = { - buildInTools, - customTools, - workflowTools, - mcpTools, - dataSourceList: dataSourceList ?? [], + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], + dataSourceList: dataSourceList || [], } const childrenNodeVars = toNodeOutputVars(iterationChildrenNodes, isChatMode, undefined, [], [], [], allPluginInfoList) @@ -98,6 +104,14 @@ const useConfig = (id: string, payload: IterationNodeType) => { }) setInputs(newInputs) }, [inputs, setInputs]) + + const changeFlattenOutput = useCallback((value: boolean) => { + const newInputs = produce(inputs, (draft) => { + draft.flatten_output = value + }) + setInputs(newInputs) + }, [inputs, setInputs]) + return { readOnly, inputs, @@ -109,6 +123,7 @@ const useConfig = (id: string, payload: IterationNodeType) => { changeParallel, changeErrorResponseMode, changeParallelNums, + changeFlattenOutput, } } diff --git a/web/app/components/workflow/nodes/knowledge-retrieval/components/metadata/metadata-filter/index.tsx b/web/app/components/workflow/nodes/knowledge-retrieval/components/metadata/metadata-filter/index.tsx index 8ea313dd26..1c6158a60e 100644 --- a/web/app/components/workflow/nodes/knowledge-retrieval/components/metadata/metadata-filter/index.tsx +++ b/web/app/components/workflow/nodes/knowledge-retrieval/components/metadata/metadata-filter/index.tsx @@ -84,7 +84,6 @@ const MetadataFilter = ({ popupClassName='!w-[387px]' isInWorkflow isAdvancedMode={true} - mode={metadataModelConfig?.mode || 'chat'} provider={metadataModelConfig?.provider || ''} completionParams={metadataModelConfig?.completion_params || { temperature: 0.7 }} modelId={metadataModelConfig?.name || ''} diff --git a/web/app/components/workflow/nodes/knowledge-retrieval/use-config.ts b/web/app/components/workflow/nodes/knowledge-retrieval/use-config.ts index 60789e6863..73d1c15872 100644 --- a/web/app/components/workflow/nodes/knowledge-retrieval/use-config.ts +++ b/web/app/components/workflow/nodes/knowledge-retrieval/use-config.ts @@ -32,7 +32,7 @@ import { getMultipleRetrievalConfig, getSelectedDatasetsMode, } from './utils' -import { RETRIEVE_TYPE } from '@/types/app' +import { AppModeEnum, RETRIEVE_TYPE } from '@/types/app' import { DATASET_DEFAULT } from '@/config' import type { DataSet } from '@/models/datasets' import { fetchDatasets } from '@/service/datasets' @@ -344,7 +344,7 @@ const useConfig = (id: string, payload: KnowledgeRetrievalNodeType) => { draft.metadata_model_config = { provider: model.provider, name: model.modelId, - mode: model.mode || 'chat', + mode: model.mode || AppModeEnum.CHAT, completion_params: draft.metadata_model_config?.completion_params || { temperature: 0.7 }, } }) diff --git a/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/json-schema-generator/prompt-editor.tsx b/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/json-schema-generator/prompt-editor.tsx index 9387813ee5..62d156253a 100644 --- a/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/json-schema-generator/prompt-editor.tsx +++ b/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/json-schema-generator/prompt-editor.tsx @@ -65,7 +65,6 @@ const PromptEditor: FC<PromptEditorProps> = ({ portalToFollowElemContentClassName='z-[1000]' isAdvancedMode={true} provider={model.provider} - mode={model.mode} completionParams={model.completion_params} modelId={model.name} setModel={onModelChange} diff --git a/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/add-field.tsx b/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/add-field.tsx index ab28233841..eba94e85dd 100644 --- a/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/add-field.tsx +++ b/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/add-field.tsx @@ -12,7 +12,10 @@ const AddField = () => { const handleAddField = useCallback(() => { setIsAddingNewField(true) - emit('addField', { path: [] }) + // fix: when user change the last property type, the 'hoveringProperty' value will be reset by 'setHoveringPropertyDebounced(null)', that cause the EditCard not showing + setTimeout(() => { + emit('addField', { path: [] }) + }, 100) }, [setIsAddingNewField, emit]) return ( diff --git a/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/edit-card/index.tsx b/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/edit-card/index.tsx index 4aa0f99d3f..2733fcc11f 100644 --- a/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/edit-card/index.tsx +++ b/web/app/components/workflow/nodes/llm/components/json-schema-config-modal/visual-editor/edit-card/index.tsx @@ -122,7 +122,8 @@ const EditCard: FC<EditCardProps> = ({ }, [emit, path, parentPath, fields, currentFields]) const handlePropertyNameChange = useCallback((e: React.ChangeEvent<HTMLInputElement>) => { - setCurrentFields(prev => ({ ...prev, name: e.target.value })) + // fix: when user add name contains space, the variable reference will not work + setCurrentFields(prev => ({ ...prev, name: e.target.value?.trim() })) }, []) const handlePropertyNameBlur = useCallback(() => { diff --git a/web/app/components/workflow/nodes/llm/components/prompt-generator-btn.tsx b/web/app/components/workflow/nodes/llm/components/prompt-generator-btn.tsx index c3c0483bec..a2b96535fa 100644 --- a/web/app/components/workflow/nodes/llm/components/prompt-generator-btn.tsx +++ b/web/app/components/workflow/nodes/llm/components/prompt-generator-btn.tsx @@ -6,7 +6,7 @@ import cn from 'classnames' import { Generator } from '@/app/components/base/icons/src/vender/other' import { ActionButton } from '@/app/components/base/action-button' import GetAutomaticResModal from '@/app/components/app/configuration/config/automatic/get-automatic-res' -import { AppType } from '@/types/app' +import { AppModeEnum } from '@/types/app' import type { GenRes } from '@/service/debug' import type { ModelConfig } from '@/app/components/workflow/types' import { useHooksStore } from '../../../hooks-store' @@ -42,7 +42,7 @@ const PromptGeneratorBtn: FC<Props> = ({ </ActionButton> {showAutomatic && ( <GetAutomaticResModal - mode={AppType.chat} + mode={AppModeEnum.CHAT} isShow={showAutomatic} onClose={showAutomaticFalse} onFinished={handleAutomaticRes} diff --git a/web/app/components/workflow/nodes/llm/default.ts b/web/app/components/workflow/nodes/llm/default.ts index a4ea0ef683..57033d26a1 100644 --- a/web/app/components/workflow/nodes/llm/default.ts +++ b/web/app/components/workflow/nodes/llm/default.ts @@ -1,4 +1,5 @@ // import { RETRIEVAL_OUTPUT_STRUCT } from '../../constants' +import { AppModeEnum } from '@/types/app' import { BlockEnum, EditionType } from '../../types' import { type NodeDefault, type PromptItem, PromptRole } from '../../types' import type { LLMNodeType } from './types' @@ -36,7 +37,7 @@ const nodeDefault: NodeDefault<LLMNodeType> = { model: { provider: '', name: '', - mode: 'chat', + mode: AppModeEnum.CHAT, completion_params: { temperature: 0.7, }, @@ -63,7 +64,7 @@ const nodeDefault: NodeDefault<LLMNodeType> = { errorMessages = t(`${i18nPrefix}.fieldRequired`, { field: t(`${i18nPrefix}.fields.model`) }) if (!errorMessages && !payload.memory) { - const isChatModel = payload.model.mode === 'chat' + const isChatModel = payload.model.mode === AppModeEnum.CHAT const isPromptEmpty = isChatModel ? !(payload.prompt_template as PromptItem[]).some((t) => { if (t.edition_type === EditionType.jinja2) @@ -77,14 +78,14 @@ const nodeDefault: NodeDefault<LLMNodeType> = { } if (!errorMessages && !!payload.memory) { - const isChatModel = payload.model.mode === 'chat' + const isChatModel = payload.model.mode === AppModeEnum.CHAT // payload.memory.query_prompt_template not pass is default: {{#sys.query#}} if (isChatModel && !!payload.memory.query_prompt_template && !payload.memory.query_prompt_template.includes('{{#sys.query#}}')) errorMessages = t('workflow.nodes.llm.sysQueryInUser') } if (!errorMessages) { - const isChatModel = payload.model.mode === 'chat' + const isChatModel = payload.model.mode === AppModeEnum.CHAT const isShowVars = (() => { if (isChatModel) return (payload.prompt_template as PromptItem[]).some(item => item.edition_type === EditionType.jinja2) diff --git a/web/app/components/workflow/nodes/llm/panel.tsx b/web/app/components/workflow/nodes/llm/panel.tsx index cd79b9f3d9..bb893b0da7 100644 --- a/web/app/components/workflow/nodes/llm/panel.tsx +++ b/web/app/components/workflow/nodes/llm/panel.tsx @@ -94,7 +94,6 @@ const Panel: FC<NodePanelProps<LLMNodeType>> = ({ } })() }, [inputs.model.completion_params]) - return ( <div className='mt-2'> <div className='space-y-4 px-4 pb-4'> @@ -106,7 +105,6 @@ const Panel: FC<NodePanelProps<LLMNodeType>> = ({ popupClassName='!w-[387px]' isInWorkflow isAdvancedMode={true} - mode={model?.mode} provider={model?.provider} completionParams={model?.completion_params} modelId={model?.name} diff --git a/web/app/components/workflow/nodes/llm/types.ts b/web/app/components/workflow/nodes/llm/types.ts index 987fb75fef..70dc4d9cc7 100644 --- a/web/app/components/workflow/nodes/llm/types.ts +++ b/web/app/components/workflow/nodes/llm/types.ts @@ -30,6 +30,7 @@ export enum Type { arrayNumber = 'array[number]', arrayObject = 'array[object]', file = 'file', + enumType = 'enum', } export enum ArrayType { diff --git a/web/app/components/workflow/nodes/llm/use-config.ts b/web/app/components/workflow/nodes/llm/use-config.ts index 44c7096744..d9b811bb85 100644 --- a/web/app/components/workflow/nodes/llm/use-config.ts +++ b/web/app/components/workflow/nodes/llm/use-config.ts @@ -18,6 +18,7 @@ import { import useNodeCrud from '@/app/components/workflow/nodes/_base/hooks/use-node-crud' import { checkHasContextBlock, checkHasHistoryBlock, checkHasQueryBlock } from '@/app/components/base/prompt-editor/constants' import useInspectVarsCrud from '@/app/components/workflow/hooks/use-inspect-vars-crud' +import { AppModeEnum } from '@/types/app' const useConfig = (id: string, payload: LLMNodeType) => { const { nodesReadOnly: readOnly } = useNodesReadOnly() @@ -27,6 +28,9 @@ const useConfig = (id: string, payload: LLMNodeType) => { const [defaultRolePrefix, setDefaultRolePrefix] = useState<{ user: string; assistant: string }>({ user: '', assistant: '' }) const { inputs, setInputs: doSetInputs } = useNodeCrud<LLMNodeType>(id, payload) const inputRef = useRef(inputs) + useEffect(() => { + inputRef.current = inputs + }, [inputs]) const { deleteNodeInspectorVars } = useInspectVarsCrud() @@ -46,7 +50,7 @@ const useConfig = (id: string, payload: LLMNodeType) => { // model const model = inputs.model const modelMode = inputs.model?.mode - const isChatModel = modelMode === 'chat' + const isChatModel = modelMode === AppModeEnum.CHAT const isCompletionModel = !isChatModel @@ -117,7 +121,7 @@ const useConfig = (id: string, payload: LLMNodeType) => { } = useConfigVision(model, { payload: inputs.vision, onChange: (newPayload) => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { draft.vision = newPayload }) setInputs(newInputs) @@ -131,7 +135,7 @@ const useConfig = (id: string, payload: LLMNodeType) => { draft.model.mode = model.mode! const isModeChange = model.mode !== inputRef.current.model.mode if (isModeChange && defaultConfig && Object.keys(defaultConfig).length > 0) - appendDefaultPromptConfig(draft, defaultConfig, model.mode === 'chat') + appendDefaultPromptConfig(draft, defaultConfig, model.mode === AppModeEnum.CHAT) }) setInputs(newInputs) setModelChanged(true) @@ -148,11 +152,11 @@ const useConfig = (id: string, payload: LLMNodeType) => { }, [model.provider, currentProvider, currentModel, handleModelChanged]) const handleCompletionParamsChange = useCallback((newParams: Record<string, any>) => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { draft.model.completion_params = newParams }) setInputs(newInputs) - }, [inputs, setInputs]) + }, [setInputs]) // change to vision model to set vision enabled, else disabled useEffect(() => { @@ -238,29 +242,29 @@ const useConfig = (id: string, payload: LLMNodeType) => { // context const handleContextVarChange = useCallback((newVar: ValueSelector | string) => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { draft.context.variable_selector = newVar as ValueSelector || [] draft.context.enabled = !!(newVar && newVar.length > 0) }) setInputs(newInputs) - }, [inputs, setInputs]) + }, [setInputs]) const handlePromptChange = useCallback((newPrompt: PromptItem[] | PromptItem) => { const newInputs = produce(inputRef.current, (draft) => { draft.prompt_template = newPrompt }) setInputs(newInputs) - }, [inputs, setInputs]) + }, [setInputs]) const handleMemoryChange = useCallback((newMemory?: Memory) => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { draft.memory = newMemory }) setInputs(newInputs) - }, [inputs, setInputs]) + }, [setInputs]) const handleSyeQueryChange = useCallback((newQuery: string) => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { if (!draft.memory) { draft.memory = { window: { @@ -275,7 +279,7 @@ const useConfig = (id: string, payload: LLMNodeType) => { } }) setInputs(newInputs) - }, [inputs, setInputs]) + }, [setInputs]) // structure output const { data: modelList } = useModelList(ModelTypeEnum.textGeneration) @@ -286,22 +290,22 @@ const useConfig = (id: string, payload: LLMNodeType) => { const [structuredOutputCollapsed, setStructuredOutputCollapsed] = useState(true) const handleStructureOutputEnableChange = useCallback((enabled: boolean) => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { draft.structured_output_enabled = enabled }) setInputs(newInputs) if (enabled) setStructuredOutputCollapsed(false) deleteNodeInspectorVars(id) - }, [inputs, setInputs, deleteNodeInspectorVars, id]) + }, [setInputs, deleteNodeInspectorVars, id]) const handleStructureOutputChange = useCallback((newOutput: StructuredOutput) => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { draft.structured_output = newOutput }) setInputs(newInputs) deleteNodeInspectorVars(id) - }, [inputs, setInputs, deleteNodeInspectorVars, id]) + }, [setInputs, deleteNodeInspectorVars, id]) const filterInputVar = useCallback((varPayload: Var) => { return [VarType.number, VarType.string, VarType.secret, VarType.arrayString, VarType.arrayNumber, VarType.file, VarType.arrayFile].includes(varPayload.type) @@ -317,11 +321,11 @@ const useConfig = (id: string, payload: LLMNodeType) => { // reasoning format const handleReasoningFormatChange = useCallback((reasoningFormat: 'tagged' | 'separated') => { - const newInputs = produce(inputs, (draft) => { + const newInputs = produce(inputRef.current, (draft) => { draft.reasoning_format = reasoningFormat }) setInputs(newInputs) - }, [inputs, setInputs]) + }, [setInputs]) const { availableVars, diff --git a/web/app/components/workflow/nodes/llm/use-single-run-form-params.ts b/web/app/components/workflow/nodes/llm/use-single-run-form-params.ts index aaa12be0c2..8d539dfc15 100644 --- a/web/app/components/workflow/nodes/llm/use-single-run-form-params.ts +++ b/web/app/components/workflow/nodes/llm/use-single-run-form-params.ts @@ -12,6 +12,7 @@ import useConfigVision from '../../hooks/use-config-vision' import { noop } from 'lodash-es' import { findVariableWhenOnLLMVision } from '../utils' import useAvailableVarList from '../_base/hooks/use-available-var-list' +import { AppModeEnum } from '@/types/app' const i18nPrefix = 'workflow.nodes.llm' type Params = { @@ -56,7 +57,7 @@ const useSingleRunFormParams = ({ // model const model = inputs.model const modelMode = inputs.model?.mode - const isChatModel = modelMode === 'chat' + const isChatModel = modelMode === AppModeEnum.CHAT const { isVisionModel, } = useConfigVision(model, { diff --git a/web/app/components/workflow/nodes/llm/utils.ts b/web/app/components/workflow/nodes/llm/utils.ts index 10c287f86b..1652d511d0 100644 --- a/web/app/components/workflow/nodes/llm/utils.ts +++ b/web/app/components/workflow/nodes/llm/utils.ts @@ -9,8 +9,9 @@ export const checkNodeValid = (_payload: LLMNodeType) => { } export const getFieldType = (field: Field) => { - const { type, items } = field - if(field.schemaType === 'file') return Type.file + const { type, items, enum: enums } = field + if (field.schemaType === 'file') return Type.file + if (enums && enums.length > 0) return Type.enumType if (type !== Type.array || !items) return type diff --git a/web/app/components/workflow/nodes/loop/add-block.tsx b/web/app/components/workflow/nodes/loop/add-block.tsx index a9c1429269..9e2fa5b555 100644 --- a/web/app/components/workflow/nodes/loop/add-block.tsx +++ b/web/app/components/workflow/nodes/loop/add-block.tsx @@ -34,11 +34,11 @@ const AddBlock = ({ const { handleNodeAdd } = useNodesInteractions() const { availableNextBlocks } = useAvailableBlocks(BlockEnum.Start, true) - const handleSelect = useCallback<OnSelectBlock>((type, toolDefaultValue) => { + const handleSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { handleNodeAdd( { nodeType: type, - toolDefaultValue, + pluginDefaultValue, }, { prevNodeId: loopNodeData.start_node_id, diff --git a/web/app/components/workflow/nodes/loop/insert-block.tsx b/web/app/components/workflow/nodes/loop/insert-block.tsx index c4f4348d8e..66d51956ba 100644 --- a/web/app/components/workflow/nodes/loop/insert-block.tsx +++ b/web/app/components/workflow/nodes/loop/insert-block.tsx @@ -25,11 +25,11 @@ const InsertBlock = ({ const handleOpenChange = useCallback((v: boolean) => { setOpen(v) }, []) - const handleInsert = useCallback<OnSelectBlock>((nodeType, toolDefaultValue) => { + const handleInsert = useCallback<OnSelectBlock>((nodeType, pluginDefaultValue) => { handleNodeAdd( { nodeType, - toolDefaultValue, + pluginDefaultValue, }, { nextNodeId: startNodeId, diff --git a/web/app/components/workflow/nodes/loop/use-config.ts b/web/app/components/workflow/nodes/loop/use-config.ts index fcf437eb96..e8504fb5e9 100644 --- a/web/app/components/workflow/nodes/loop/use-config.ts +++ b/web/app/components/workflow/nodes/loop/use-config.ts @@ -15,9 +15,24 @@ import useNodeCrud from '../_base/hooks/use-node-crud' import { toNodeOutputVars } from '../_base/components/variable/utils' import { getOperators } from './utils' import { LogicalOperator } from './types' -import type { HandleAddCondition, HandleAddSubVariableCondition, HandleRemoveCondition, HandleToggleConditionLogicalOperator, HandleToggleSubVariableConditionLogicalOperator, HandleUpdateCondition, HandleUpdateSubVariableCondition, LoopNodeType } from './types' +import type { + HandleAddCondition, + HandleAddSubVariableCondition, + HandleRemoveCondition, + HandleToggleConditionLogicalOperator, + HandleToggleSubVariableConditionLogicalOperator, + HandleUpdateCondition, + HandleUpdateSubVariableCondition, + LoopNodeType, +} from './types' import useIsVarFileAttribute from './use-is-var-file-attribute' import { useStore } from '@/app/components/workflow/store' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, +} from '@/service/use-tools' const useConfig = (id: string, payload: LoopNodeType) => { const { nodesReadOnly: readOnly } = useNodesReadOnly() @@ -38,17 +53,17 @@ const useConfig = (id: string, payload: LoopNodeType) => { // output const { getLoopNodeChildren } = useWorkflow() const loopChildrenNodes = [{ id, data: payload } as any, ...getLoopNodeChildren(id)] - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() const dataSourceList = useStore(s => s.dataSourceList) const allPluginInfoList = { - buildInTools, - customTools, - workflowTools, - mcpTools, - dataSourceList: dataSourceList ?? [], + buildInTools: buildInTools || [], + customTools: customTools || [], + workflowTools: workflowTools || [], + mcpTools: mcpTools || [], + dataSourceList: dataSourceList || [], } const childrenNodeVars = toNodeOutputVars(loopChildrenNodes, isChatMode, undefined, [], conversationVariables, [], allPluginInfoList) diff --git a/web/app/components/workflow/nodes/parameter-extractor/components/extract-parameter/import-from-tool.tsx b/web/app/components/workflow/nodes/parameter-extractor/components/extract-parameter/import-from-tool.tsx index d93d08a0ac..7b8354f6d5 100644 --- a/web/app/components/workflow/nodes/parameter-extractor/components/extract-parameter/import-from-tool.tsx +++ b/web/app/components/workflow/nodes/parameter-extractor/components/extract-parameter/import-from-tool.tsx @@ -8,9 +8,8 @@ import { useTranslation } from 'react-i18next' import BlockSelector from '../../../../block-selector' import type { Param, ParamType } from '../../types' import cn from '@/utils/classnames' -import { useStore } from '@/app/components/workflow/store' import type { - DataSourceDefaultValue, + PluginDefaultValue, ToolDefaultValue, } from '@/app/components/workflow/block-selector/types' import type { ToolParameter } from '@/app/components/tools/types' @@ -18,6 +17,11 @@ import { CollectionType } from '@/app/components/tools/types' import type { BlockEnum } from '@/app/components/workflow/types' import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' import { canFindTool } from '@/utils' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllWorkflowTools, +} from '@/service/use-tools' const i18nPrefix = 'workflow.nodes.parameterExtractor' @@ -42,23 +46,23 @@ const ImportFromTool: FC<Props> = ({ const { t } = useTranslation() const language = useLanguage() - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() - const handleSelectTool = useCallback((_type: BlockEnum, toolInfo?: ToolDefaultValue | DataSourceDefaultValue) => { - if (!toolInfo || 'datasource_name' in toolInfo) + const handleSelectTool = useCallback((_type: BlockEnum, toolInfo?: PluginDefaultValue) => { + if (!toolInfo || 'datasource_name' in toolInfo || !('tool_name' in toolInfo)) return - const { provider_id, provider_type, tool_name } = toolInfo + const { provider_id, provider_type, tool_name } = toolInfo as ToolDefaultValue const currentTools = (() => { switch (provider_type) { case CollectionType.builtIn: - return buildInTools + return buildInTools || [] case CollectionType.custom: - return customTools + return customTools || [] case CollectionType.workflow: - return workflowTools + return workflowTools || [] default: return [] } diff --git a/web/app/components/workflow/nodes/parameter-extractor/default.ts b/web/app/components/workflow/nodes/parameter-extractor/default.ts index a65306249d..5d2010122d 100644 --- a/web/app/components/workflow/nodes/parameter-extractor/default.ts +++ b/web/app/components/workflow/nodes/parameter-extractor/default.ts @@ -3,6 +3,7 @@ import { type ParameterExtractorNodeType, ReasoningModeType } from './types' import { genNodeMetaData } from '@/app/components/workflow/utils' import { BlockEnum } from '@/app/components/workflow/types' import { BlockClassificationEnum } from '@/app/components/workflow/block-selector/types' +import { AppModeEnum } from '@/types/app' const i18nPrefix = 'workflow' const metaData = genNodeMetaData({ @@ -17,7 +18,7 @@ const nodeDefault: NodeDefault<ParameterExtractorNodeType> = { model: { provider: '', name: '', - mode: 'chat', + mode: AppModeEnum.CHAT, completion_params: { temperature: 0.7, }, diff --git a/web/app/components/workflow/nodes/parameter-extractor/panel.tsx b/web/app/components/workflow/nodes/parameter-extractor/panel.tsx index a169217609..8faebfa547 100644 --- a/web/app/components/workflow/nodes/parameter-extractor/panel.tsx +++ b/web/app/components/workflow/nodes/parameter-extractor/panel.tsx @@ -67,7 +67,6 @@ const Panel: FC<NodePanelProps<ParameterExtractorNodeType>> = ({ popupClassName='!w-[387px]' isInWorkflow isAdvancedMode={true} - mode={model?.mode} provider={model?.provider} completionParams={model?.completion_params} modelId={model?.name} diff --git a/web/app/components/workflow/nodes/parameter-extractor/use-config.ts b/web/app/components/workflow/nodes/parameter-extractor/use-config.ts index 81dace1014..676d631a8a 100644 --- a/web/app/components/workflow/nodes/parameter-extractor/use-config.ts +++ b/web/app/components/workflow/nodes/parameter-extractor/use-config.ts @@ -17,6 +17,7 @@ import { checkHasQueryBlock } from '@/app/components/base/prompt-editor/constant import useAvailableVarList from '@/app/components/workflow/nodes/_base/hooks/use-available-var-list' import { supportFunctionCall } from '@/utils/tool-call' import useInspectVarsCrud from '../../hooks/use-inspect-vars-crud' +import { AppModeEnum } from '@/types/app' const useConfig = (id: string, payload: ParameterExtractorNodeType) => { const { @@ -86,13 +87,13 @@ const useConfig = (id: string, payload: ParameterExtractorNodeType) => { const model = inputs.model || { provider: '', name: '', - mode: 'chat', + mode: AppModeEnum.CHAT, completion_params: { temperature: 0.7, }, } const modelMode = inputs.model?.mode - const isChatModel = modelMode === 'chat' + const isChatModel = modelMode === AppModeEnum.CHAT const isCompletionModel = !isChatModel const { @@ -133,7 +134,7 @@ const useConfig = (id: string, payload: ParameterExtractorNodeType) => { draft.model.mode = model.mode! const isModeChange = model.mode !== inputRef.current.model?.mode if (isModeChange && defaultConfig && Object.keys(defaultConfig).length > 0) - appendDefaultPromptConfig(draft, defaultConfig, model.mode === 'chat') + appendDefaultPromptConfig(draft, defaultConfig, model.mode === AppModeEnum.CHAT) }) setInputs(newInputs) setModelChanged(true) diff --git a/web/app/components/workflow/nodes/question-classifier/default.ts b/web/app/components/workflow/nodes/question-classifier/default.ts index d34c854916..90ae3fd586 100644 --- a/web/app/components/workflow/nodes/question-classifier/default.ts +++ b/web/app/components/workflow/nodes/question-classifier/default.ts @@ -3,6 +3,7 @@ import type { QuestionClassifierNodeType } from './types' import { genNodeMetaData } from '@/app/components/workflow/utils' import { BlockEnum } from '@/app/components/workflow/types' import { BlockClassificationEnum } from '@/app/components/workflow/block-selector/types' +import { AppModeEnum } from '@/types/app' const i18nPrefix = 'workflow' @@ -18,7 +19,7 @@ const nodeDefault: NodeDefault<QuestionClassifierNodeType> = { model: { provider: '', name: '', - mode: 'chat', + mode: AppModeEnum.CHAT, completion_params: { temperature: 0.7, }, diff --git a/web/app/components/workflow/nodes/question-classifier/panel.tsx b/web/app/components/workflow/nodes/question-classifier/panel.tsx index 8e27f5dceb..8b6bc533f2 100644 --- a/web/app/components/workflow/nodes/question-classifier/panel.tsx +++ b/web/app/components/workflow/nodes/question-classifier/panel.tsx @@ -56,7 +56,6 @@ const Panel: FC<NodePanelProps<QuestionClassifierNodeType>> = ({ popupClassName='!w-[387px]' isInWorkflow isAdvancedMode={true} - mode={model?.mode} provider={model?.provider} completionParams={model.completion_params} modelId={model.name} diff --git a/web/app/components/workflow/nodes/question-classifier/use-config.ts b/web/app/components/workflow/nodes/question-classifier/use-config.ts index dc197a079e..28a6fa0314 100644 --- a/web/app/components/workflow/nodes/question-classifier/use-config.ts +++ b/web/app/components/workflow/nodes/question-classifier/use-config.ts @@ -15,6 +15,7 @@ import { useModelListAndDefaultModelAndCurrentProviderAndModel } from '@/app/com import { ModelTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' import { checkHasQueryBlock } from '@/app/components/base/prompt-editor/constants' import { useUpdateNodeInternals } from 'reactflow' +import { AppModeEnum } from '@/types/app' const useConfig = (id: string, payload: QuestionClassifierNodeType) => { const updateNodeInternals = useUpdateNodeInternals() @@ -38,7 +39,7 @@ const useConfig = (id: string, payload: QuestionClassifierNodeType) => { const model = inputs.model const modelMode = inputs.model?.mode - const isChatModel = modelMode === 'chat' + const isChatModel = modelMode === AppModeEnum.CHAT const { isVisionModel, diff --git a/web/app/components/workflow/nodes/start/default.ts b/web/app/components/workflow/nodes/start/default.ts index 3b98b57a73..60584b5144 100644 --- a/web/app/components/workflow/nodes/start/default.ts +++ b/web/app/components/workflow/nodes/start/default.ts @@ -7,10 +7,10 @@ const metaData = genNodeMetaData({ sort: 0.1, type: BlockEnum.Start, isStart: true, - isRequired: true, - isUndeletable: true, + isRequired: false, isSingleton: true, - isTypeFixed: true, + isTypeFixed: false, // support node type change for start node(user input) + helpLinkUri: 'user-input', }) const nodeDefault: NodeDefault<StartNodeType> = { metaData, diff --git a/web/app/components/workflow/nodes/start/panel.tsx b/web/app/components/workflow/nodes/start/panel.tsx index 0a1efd444f..a560bd2e63 100644 --- a/web/app/components/workflow/nodes/start/panel.tsx +++ b/web/app/components/workflow/nodes/start/panel.tsx @@ -62,7 +62,7 @@ const Panel: FC<NodePanelProps<StartNodeType>> = ({ <VarItem readonly payload={{ - variable: 'sys.query', + variable: 'userinput.query', } as any} rightContent={ <div className='text-xs font-normal text-text-tertiary'> @@ -76,7 +76,7 @@ const Panel: FC<NodePanelProps<StartNodeType>> = ({ readonly showLegacyBadge={!isChatMode} payload={{ - variable: 'sys.files', + variable: 'userinput.files', } as any} rightContent={ <div className='text-xs font-normal text-text-tertiary'> @@ -84,80 +84,7 @@ const Panel: FC<NodePanelProps<StartNodeType>> = ({ </div> } /> - { - isChatMode && ( - <> - <VarItem - readonly - payload={{ - variable: 'sys.dialogue_count', - } as any} - rightContent={ - <div className='text-xs font-normal text-text-tertiary'> - Number - </div> - } - /> - <VarItem - readonly - payload={{ - variable: 'sys.conversation_id', - } as any} - rightContent={ - <div className='text-xs font-normal text-text-tertiary'> - String - </div> - } - /> - </> - ) - } - <VarItem - readonly - payload={{ - variable: 'sys.user_id', - } as any} - rightContent={ - <div className='text-xs font-normal text-text-tertiary'> - String - </div> - } - /> - <VarItem - readonly - payload={{ - variable: 'sys.app_id', - } as any} - rightContent={ - <div className='text-xs font-normal text-text-tertiary'> - String - </div> - } - /> - <VarItem - readonly - payload={{ - variable: 'sys.workflow_id', - } as any} - rightContent={ - <div className='text-xs font-normal text-text-tertiary'> - String - </div> - } - /> - <VarItem - readonly - payload={{ - variable: 'sys.workflow_run_id', - } as any} - rightContent={ - <div className='text-xs font-normal text-text-tertiary'> - String - </div> - } - /> </div> - </> </Field> </div> diff --git a/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/index.tsx b/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/index.tsx index ec35f9a60a..fa50727123 100644 --- a/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/index.tsx +++ b/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/index.tsx @@ -20,6 +20,7 @@ type MixedVariableTextInputProps = { onChange?: (text: string) => void showManageInputField?: boolean onManageInputField?: () => void + disableVariableInsertion?: boolean } const MixedVariableTextInput = ({ readOnly = false, @@ -29,6 +30,7 @@ const MixedVariableTextInput = ({ onChange, showManageInputField, onManageInputField, + disableVariableInsertion = false, }: MixedVariableTextInputProps) => { const { t } = useTranslation() const controlPromptEditorRerenderKey = useStore(s => s.controlPromptEditorRerenderKey) @@ -37,7 +39,7 @@ const MixedVariableTextInput = ({ <PromptEditor key={controlPromptEditorRerenderKey} wrapperClassName={cn( - 'w-full rounded-lg border border-transparent bg-components-input-bg-normal px-2 py-1', + 'min-h-8 w-full rounded-lg border border-transparent bg-components-input-bg-normal px-2 py-1', 'hover:border-components-input-border-hover hover:bg-components-input-bg-hover', 'focus-within:border-components-input-border-active focus-within:bg-components-input-bg-active focus-within:shadow-xs', )} @@ -45,7 +47,7 @@ const MixedVariableTextInput = ({ editable={!readOnly} value={value} workflowVariableBlock={{ - show: true, + show: !disableVariableInsertion, variables: nodesOutputVars || [], workflowNodesMap: availableNodes.reduce((acc, node) => { acc[node.id] = { @@ -63,7 +65,7 @@ const MixedVariableTextInput = ({ showManageInputField, onManageInputField, }} - placeholder={<Placeholder />} + placeholder={<Placeholder disableVariableInsertion={disableVariableInsertion} />} onChange={onChange} /> ) diff --git a/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/placeholder.tsx b/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/placeholder.tsx index 75d4c91996..d6e0bbc059 100644 --- a/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/placeholder.tsx +++ b/web/app/components/workflow/nodes/tool/components/mixed-variable-text-input/placeholder.tsx @@ -6,7 +6,11 @@ import { $insertNodes } from 'lexical' import { CustomTextNode } from '@/app/components/base/prompt-editor/plugins/custom-text/node' import Badge from '@/app/components/base/badge' -const Placeholder = () => { +type PlaceholderProps = { + disableVariableInsertion?: boolean +} + +const Placeholder = ({ disableVariableInsertion = false }: PlaceholderProps) => { const { t } = useTranslation() const [editor] = useLexicalComposerContext() @@ -28,17 +32,21 @@ const Placeholder = () => { > <div className='flex grow items-center'> {t('workflow.nodes.tool.insertPlaceholder1')} - <div className='system-kbd mx-0.5 flex h-4 w-4 items-center justify-center rounded bg-components-kbd-bg-gray text-text-placeholder'>/</div> - <div - className='system-sm-regular cursor-pointer text-components-input-text-placeholder underline decoration-dotted decoration-auto underline-offset-auto hover:text-text-tertiary' - onMouseDown={((e) => { - e.preventDefault() - e.stopPropagation() - handleInsert('/') - })} - > - {t('workflow.nodes.tool.insertPlaceholder2')} - </div> + {(!disableVariableInsertion) && ( + <> + <div className='system-kbd mx-0.5 flex h-4 w-4 items-center justify-center rounded bg-components-kbd-bg-gray text-text-placeholder'>/</div> + <div + className='system-sm-regular cursor-pointer text-components-input-text-placeholder underline decoration-dotted decoration-auto underline-offset-auto hover:text-text-tertiary' + onMouseDown={((e) => { + e.preventDefault() + e.stopPropagation() + handleInsert('/') + })} + > + {t('workflow.nodes.tool.insertPlaceholder2')} + </div> + </> + )} </div> <Badge className='shrink-0' diff --git a/web/app/components/workflow/nodes/tool/components/tool-form/index.tsx b/web/app/components/workflow/nodes/tool/components/tool-form/index.tsx index 747790ac58..ade29beddb 100644 --- a/web/app/components/workflow/nodes/tool/components/tool-form/index.tsx +++ b/web/app/components/workflow/nodes/tool/components/tool-form/index.tsx @@ -18,6 +18,7 @@ type Props = { currentProvider?: ToolWithProvider showManageInputField?: boolean onManageInputField?: () => void + extraParams?: Record<string, any> } const ToolForm: FC<Props> = ({ @@ -31,6 +32,7 @@ const ToolForm: FC<Props> = ({ currentProvider, showManageInputField, onManageInputField, + extraParams, }) => { return ( <div className='space-y-1'> @@ -48,6 +50,8 @@ const ToolForm: FC<Props> = ({ currentProvider={currentProvider} showManageInputField={showManageInputField} onManageInputField={onManageInputField} + extraParams={extraParams} + providerType='tool' /> )) } diff --git a/web/app/components/workflow/nodes/tool/components/tool-form/item.tsx b/web/app/components/workflow/nodes/tool/components/tool-form/item.tsx index c70a039b5b..567266abde 100644 --- a/web/app/components/workflow/nodes/tool/components/tool-form/item.tsx +++ b/web/app/components/workflow/nodes/tool/components/tool-form/item.tsx @@ -26,6 +26,8 @@ type Props = { currentProvider?: ToolWithProvider showManageInputField?: boolean onManageInputField?: () => void + extraParams?: Record<string, any> + providerType?: 'tool' | 'trigger' } const ToolFormItem: FC<Props> = ({ @@ -39,6 +41,8 @@ const ToolFormItem: FC<Props> = ({ currentProvider, showManageInputField, onManageInputField, + extraParams, + providerType = 'tool', }) => { const language = useLanguage() const { name, label, type, required, tooltip, input_schema } = schema @@ -95,6 +99,8 @@ const ToolFormItem: FC<Props> = ({ currentProvider={currentProvider} showManageInputField={showManageInputField} onManageInputField={onManageInputField} + extraParams={extraParams} + providerType={providerType} /> {isShowSchema && ( diff --git a/web/app/components/workflow/nodes/tool/node.tsx b/web/app/components/workflow/nodes/tool/node.tsx index 8cc3ec580d..6aa483e8b0 100644 --- a/web/app/components/workflow/nodes/tool/node.tsx +++ b/web/app/components/workflow/nodes/tool/node.tsx @@ -1,46 +1,90 @@ import type { FC } from 'react' -import React from 'react' -import type { ToolNodeType } from './types' +import React, { useEffect } from 'react' import type { NodeProps } from '@/app/components/workflow/types' import { FormTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' +import { InstallPluginButton } from '@/app/components/workflow/nodes/_base/components/install-plugin-button' +import { useNodePluginInstallation } from '@/app/components/workflow/hooks/use-node-plugin-installation' +import { useNodeDataUpdate } from '@/app/components/workflow/hooks/use-node-data-update' +import type { ToolNodeType } from './types' const Node: FC<NodeProps<ToolNodeType>> = ({ + id, data, }) => { const { tool_configurations, paramSchemas } = data const toolConfigs = Object.keys(tool_configurations || {}) + const { + isChecking, + isMissing, + uniqueIdentifier, + canInstall, + onInstallSuccess, + shouldDim, + } = useNodePluginInstallation(data) + const showInstallButton = !isChecking && isMissing && canInstall && uniqueIdentifier + const { handleNodeDataUpdate } = useNodeDataUpdate() + const shouldLock = !isChecking && isMissing && canInstall && Boolean(uniqueIdentifier) - if (!toolConfigs.length) + useEffect(() => { + if (data._pluginInstallLocked === shouldLock && data._dimmed === shouldDim) + return + handleNodeDataUpdate({ + id, + data: { + _pluginInstallLocked: shouldLock, + _dimmed: shouldDim, + }, + }) + }, [data._pluginInstallLocked, data._dimmed, handleNodeDataUpdate, id, shouldDim, shouldLock]) + + const hasConfigs = toolConfigs.length > 0 + + if (!showInstallButton && !hasConfigs) return null return ( - <div className='mb-1 px-3 py-1'> - <div className='space-y-0.5'> - {toolConfigs.map((key, index) => ( - <div key={index} className='flex h-6 items-center justify-between space-x-1 rounded-md bg-workflow-block-parma-bg px-1 text-xs font-normal text-text-secondary'> - <div title={key} className='max-w-[100px] shrink-0 truncate text-xs font-medium uppercase text-text-tertiary'> - {key} + <div className='relative mb-1 px-3 py-1'> + {showInstallButton && ( + <div className='pointer-events-auto absolute right-3 top-[-32px] z-40'> + <InstallPluginButton + size='small' + className='!font-medium !text-text-accent' + extraIdentifiers={[ + data.plugin_id, + data.provider_id, + data.provider_name, + ].filter(Boolean) as string[]} + uniqueIdentifier={uniqueIdentifier!} + onSuccess={onInstallSuccess} + /> + </div> + )} + {hasConfigs && ( + <div className='space-y-0.5' aria-disabled={shouldDim}> + {toolConfigs.map((key, index) => ( + <div key={index} className='flex h-6 items-center justify-between space-x-1 rounded-md bg-workflow-block-parma-bg px-1 text-xs font-normal text-text-secondary'> + <div title={key} className='max-w-[100px] shrink-0 truncate text-xs font-medium uppercase text-text-tertiary'> + {key} + </div> + {typeof tool_configurations[key].value === 'string' && ( + <div title={tool_configurations[key].value} className='w-0 shrink-0 grow truncate text-right text-xs font-normal text-text-secondary'> + {paramSchemas?.find(i => i.name === key)?.type === FormTypeEnum.secretInput ? '********' : tool_configurations[key].value} + </div> + )} + {typeof tool_configurations[key].value === 'number' && ( + <div title={Number.isNaN(tool_configurations[key].value) ? '' : tool_configurations[key].value} className='w-0 shrink-0 grow truncate text-right text-xs font-normal text-text-secondary'> + {Number.isNaN(tool_configurations[key].value) ? '' : tool_configurations[key].value} + </div> + )} + {typeof tool_configurations[key] !== 'string' && tool_configurations[key]?.type === FormTypeEnum.modelSelector && ( + <div title={tool_configurations[key].model} className='w-0 shrink-0 grow truncate text-right text-xs font-normal text-text-secondary'> + {tool_configurations[key].model} + </div> + )} </div> - {typeof tool_configurations[key].value === 'string' && ( - <div title={tool_configurations[key].value} className='w-0 shrink-0 grow truncate text-right text-xs font-normal text-text-secondary'> - {paramSchemas?.find(i => i.name === key)?.type === FormTypeEnum.secretInput ? '********' : tool_configurations[key].value} - </div> - )} - {typeof tool_configurations[key].value === 'number' && ( - <div title={Number.isNaN(tool_configurations[key].value) ? '' : tool_configurations[key].value} className='w-0 shrink-0 grow truncate text-right text-xs font-normal text-text-secondary'> - {Number.isNaN(tool_configurations[key].value) ? '' : tool_configurations[key].value} - </div> - )} - {typeof tool_configurations[key] !== 'string' && tool_configurations[key]?.type === FormTypeEnum.modelSelector && ( - <div title={tool_configurations[key].model} className='w-0 shrink-0 grow truncate text-right text-xs font-normal text-text-secondary'> - {tool_configurations[key].model} - </div> - )} - </div> - - ))} - - </div> + ))} + </div> + )} </div> ) } diff --git a/web/app/components/workflow/nodes/tool/types.ts b/web/app/components/workflow/nodes/tool/types.ts index 8bed5076d3..6e6ef858dc 100644 --- a/web/app/components/workflow/nodes/tool/types.ts +++ b/web/app/components/workflow/nodes/tool/types.ts @@ -1,16 +1,10 @@ -import type { CollectionType } from '@/app/components/tools/types' -import type { CommonNodeType, ValueSelector } from '@/app/components/workflow/types' +import type { Collection, CollectionType } from '@/app/components/tools/types' +import type { CommonNodeType } from '@/app/components/workflow/types' +import type { ResourceVarInputs } from '../_base/types' -export enum VarType { - variable = 'variable', - constant = 'constant', - mixed = 'mixed', -} - -export type ToolVarInputs = Record<string, { - type: VarType - value?: string | ValueSelector | any -}> +// Use base types directly +export { VarKindType as VarType } from '../_base/types' +export type ToolVarInputs = ResourceVarInputs export type ToolNodeType = CommonNodeType & { provider_id: string @@ -26,4 +20,7 @@ export type ToolNodeType = CommonNodeType & { tool_description?: string is_team_authorization?: boolean params?: Record<string, any> + plugin_id?: string + provider_icon?: Collection['icon'] + plugin_unique_identifier?: string } diff --git a/web/app/components/workflow/nodes/tool/use-config.ts b/web/app/components/workflow/nodes/tool/use-config.ts index 5b8827936c..fe3fe543e9 100644 --- a/web/app/components/workflow/nodes/tool/use-config.ts +++ b/web/app/components/workflow/nodes/tool/use-config.ts @@ -2,7 +2,7 @@ import { useCallback, useEffect, useMemo, useState } from 'react' import { useTranslation } from 'react-i18next' import { produce } from 'immer' import { useBoolean } from 'ahooks' -import { useStore, useWorkflowStore } from '../../store' +import { useWorkflowStore } from '../../store' import type { ToolNodeType, ToolVarInputs } from './types' import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' import useNodeCrud from '@/app/components/workflow/nodes/_base/hooks/use-node-crud' @@ -15,15 +15,20 @@ import { import Toast from '@/app/components/base/toast' import type { InputVar } from '@/app/components/workflow/types' import { - useFetchToolsData, useNodesReadOnly, } from '@/app/components/workflow/hooks' import { canFindTool } from '@/utils' +import { + useAllBuiltInTools, + useAllCustomTools, + useAllMCPTools, + useAllWorkflowTools, + useInvalidToolsByType, +} from '@/service/use-tools' const useConfig = (id: string, payload: ToolNodeType) => { const workflowStore = useWorkflowStore() const { nodesReadOnly: readOnly } = useNodesReadOnly() - const { handleFetchAllTools } = useFetchToolsData() const { t } = useTranslation() const language = useLanguage() @@ -43,21 +48,21 @@ const useConfig = (id: string, payload: ToolNodeType) => { tool_parameters, } = inputs const isBuiltIn = provider_type === CollectionType.builtIn - const buildInTools = useStore(s => s.buildInTools) - const customTools = useStore(s => s.customTools) - const workflowTools = useStore(s => s.workflowTools) - const mcpTools = useStore(s => s.mcpTools) + const { data: buildInTools } = useAllBuiltInTools() + const { data: customTools } = useAllCustomTools() + const { data: workflowTools } = useAllWorkflowTools() + const { data: mcpTools } = useAllMCPTools() const currentTools = useMemo(() => { switch (provider_type) { case CollectionType.builtIn: - return buildInTools + return buildInTools || [] case CollectionType.custom: - return customTools + return customTools || [] case CollectionType.workflow: - return workflowTools + return workflowTools || [] case CollectionType.mcp: - return mcpTools + return mcpTools || [] default: return [] } @@ -75,6 +80,7 @@ const useConfig = (id: string, payload: ToolNodeType) => { { setTrue: showSetAuthModal, setFalse: hideSetAuthModal }, ] = useBoolean(false) + const invalidToolsByType = useInvalidToolsByType(provider_type) const handleSaveAuth = useCallback( async (value: any) => { await updateBuiltInToolCredential(currCollection?.name as string, value) @@ -83,14 +89,14 @@ const useConfig = (id: string, payload: ToolNodeType) => { type: 'success', message: t('common.api.actionSuccess'), }) - handleFetchAllTools(provider_type) + invalidToolsByType() hideSetAuthModal() }, [ currCollection?.name, hideSetAuthModal, t, - handleFetchAllTools, + invalidToolsByType, provider_type, ], ) @@ -241,17 +247,15 @@ const useConfig = (id: string, payload: ToolNodeType) => { name: outputKey, type: output.type === 'array' - ? `Array[${ - output.items?.type - ? output.items.type.slice(0, 1).toLocaleUpperCase() - + output.items.type.slice(1) - : 'Unknown' + ? `Array[${output.items?.type + ? output.items.type.slice(0, 1).toLocaleUpperCase() + + output.items.type.slice(1) + : 'Unknown' }]` - : `${ - output.type - ? output.type.slice(0, 1).toLocaleUpperCase() - + output.type.slice(1) - : 'Unknown' + : `${output.type + ? output.type.slice(0, 1).toLocaleUpperCase() + + output.type.slice(1) + : 'Unknown' }`, description: output.description, }) diff --git a/web/app/components/workflow/nodes/trigger-plugin/components/trigger-form/index.tsx b/web/app/components/workflow/nodes/trigger-plugin/components/trigger-form/index.tsx new file mode 100644 index 0000000000..93bf788c34 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/components/trigger-form/index.tsx @@ -0,0 +1,57 @@ +'use client' +import type { CredentialFormSchema } from '@/app/components/header/account-setting/model-provider-page/declarations' +import type { Event } from '@/app/components/tools/types' +import type { FC } from 'react' +import type { PluginTriggerVarInputs } from '@/app/components/workflow/nodes/trigger-plugin/types' +import TriggerFormItem from './item' +import type { TriggerWithProvider } from '@/app/components/workflow/block-selector/types' + +type Props = { + readOnly: boolean + nodeId: string + schema: CredentialFormSchema[] + value: PluginTriggerVarInputs + onChange: (value: PluginTriggerVarInputs) => void + onOpen?: (index: number) => void + inPanel?: boolean + currentEvent?: Event + currentProvider?: TriggerWithProvider + extraParams?: Record<string, any> + disableVariableInsertion?: boolean +} + +const TriggerForm: FC<Props> = ({ + readOnly, + nodeId, + schema, + value, + onChange, + inPanel, + currentEvent, + currentProvider, + extraParams, + disableVariableInsertion = false, +}) => { + return ( + <div className='space-y-1'> + { + schema.map((schema, index) => ( + <TriggerFormItem + key={index} + readOnly={readOnly} + nodeId={nodeId} + schema={schema} + value={value} + onChange={onChange} + inPanel={inPanel} + currentEvent={currentEvent} + currentProvider={currentProvider} + extraParams={extraParams} + disableVariableInsertion={disableVariableInsertion} + /> + )) + } + </div> + ) +} +export default TriggerForm diff --git a/web/app/components/workflow/nodes/trigger-plugin/components/trigger-form/item.tsx b/web/app/components/workflow/nodes/trigger-plugin/components/trigger-form/item.tsx new file mode 100644 index 0000000000..678c12f02a --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/components/trigger-form/item.tsx @@ -0,0 +1,112 @@ +'use client' +import type { FC } from 'react' +import { + RiBracesLine, +} from '@remixicon/react' +import type { PluginTriggerVarInputs } from '@/app/components/workflow/nodes/trigger-plugin/types' +import type { CredentialFormSchema } from '@/app/components/header/account-setting/model-provider-page/declarations' +import { FormTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' +import { useLanguage } from '@/app/components/header/account-setting/model-provider-page/hooks' +import Button from '@/app/components/base/button' +import Tooltip from '@/app/components/base/tooltip' +import FormInputItem from '@/app/components/workflow/nodes/_base/components/form-input-item' +import { useBoolean } from 'ahooks' +import SchemaModal from '@/app/components/plugins/plugin-detail-panel/tool-selector/schema-modal' +import type { Event } from '@/app/components/tools/types' +import type { TriggerWithProvider } from '@/app/components/workflow/block-selector/types' + +type Props = { + readOnly: boolean + nodeId: string + schema: CredentialFormSchema + value: PluginTriggerVarInputs + onChange: (value: PluginTriggerVarInputs) => void + inPanel?: boolean + currentEvent?: Event + currentProvider?: TriggerWithProvider + extraParams?: Record<string, any> + disableVariableInsertion?: boolean +} + +const TriggerFormItem: FC<Props> = ({ + readOnly, + nodeId, + schema, + value, + onChange, + inPanel, + currentEvent, + currentProvider, + extraParams, + disableVariableInsertion = false, +}) => { + const language = useLanguage() + const { name, label, type, required, tooltip, input_schema } = schema + const showSchemaButton = type === FormTypeEnum.object || type === FormTypeEnum.array + const showDescription = type === FormTypeEnum.textInput || type === FormTypeEnum.secretInput + const [isShowSchema, { + setTrue: showSchema, + setFalse: hideSchema, + }] = useBoolean(false) + return ( + <div className='space-y-0.5 py-1'> + <div> + <div className='flex h-6 items-center'> + <div className='system-sm-medium text-text-secondary'>{label[language] || label.en_US}</div> + {required && ( + <div className='system-xs-regular ml-1 text-text-destructive-secondary'>*</div> + )} + {!showDescription && tooltip && ( + <Tooltip + popupContent={<div className='w-[200px]'> + {tooltip[language] || tooltip.en_US} + </div>} + triggerClassName='ml-1 w-4 h-4' + asChild={false} + /> + )} + {showSchemaButton && ( + <> + <div className='system-xs-regular ml-1 mr-0.5 text-text-quaternary'>·</div> + <Button + variant='ghost' + size='small' + onClick={showSchema} + className='system-xs-regular px-1 text-text-tertiary' + > + <RiBracesLine className='mr-1 size-3.5' /> + <span>JSON Schema</span> + </Button> + </> + )} + </div> + {showDescription && tooltip && ( + <div className='body-xs-regular pb-0.5 text-text-tertiary'>{tooltip[language] || tooltip.en_US}</div> + )} + </div> + <FormInputItem + readOnly={readOnly} + nodeId={nodeId} + schema={schema} + value={value} + onChange={onChange} + inPanel={inPanel} + currentTool={currentEvent} + currentProvider={currentProvider} + providerType='trigger' + extraParams={extraParams} + disableVariableInsertion={disableVariableInsertion} + /> + + {isShowSchema && ( + <SchemaModal + isShow + onClose={hideSchema} + rootName={name} + schema={input_schema!} + /> + )} + </div> + ) +} +export default TriggerFormItem diff --git a/web/app/components/workflow/nodes/trigger-plugin/default.ts b/web/app/components/workflow/nodes/trigger-plugin/default.ts new file mode 100644 index 0000000000..928534e07c --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/default.ts @@ -0,0 +1,297 @@ +import type { SchemaTypeDefinition } from '@/service/use-common' +import type { NodeDefault, Var } from '../../types' +import { BlockEnum, VarType } from '../../types' +import { genNodeMetaData } from '../../utils' +import { VarKindType } from '../_base/types' +import { type Field, type StructuredOutput, Type } from '../llm/types' +import type { PluginTriggerNodeType } from './types' + +const normalizeJsonSchemaType = (schema: any): string | undefined => { + if (!schema) return undefined + const { type, properties, items, oneOf, anyOf, allOf } = schema + + if (Array.isArray(type)) + return type.find((item: string | null) => item && item !== 'null') || type[0] + + if (typeof type === 'string') + return type + + const compositeCandidates = [oneOf, anyOf, allOf] + .filter((entry): entry is any[] => Array.isArray(entry)) + .flat() + + for (const candidate of compositeCandidates) { + const normalized = normalizeJsonSchemaType(candidate) + if (normalized) + return normalized + } + + if (properties) + return 'object' + + if (items) + return 'array' + + return undefined +} + +const pickItemSchema = (schema: any) => { + if (!schema || !schema.items) + return undefined + return Array.isArray(schema.items) ? schema.items[0] : schema.items +} + +const extractSchemaType = (schema: any, _schemaTypeDefinitions?: SchemaTypeDefinition[]): string | undefined => { + if (!schema) + return undefined + + const schemaTypeFromSchema = schema.schema_type || schema.schemaType + if (typeof schemaTypeFromSchema === 'string' && schemaTypeFromSchema.trim().length > 0) + return schemaTypeFromSchema + + return undefined +} + +const resolveVarType = ( + schema: any, + schemaTypeDefinitions?: SchemaTypeDefinition[], +): { type: VarType; schemaType?: string } => { + const schemaType = extractSchemaType(schema, schemaTypeDefinitions) + const normalizedType = normalizeJsonSchemaType(schema) + + switch (normalizedType) { + case 'string': + return { type: VarType.string, schemaType } + case 'number': + return { type: VarType.number, schemaType } + case 'integer': + return { type: VarType.integer, schemaType } + case 'boolean': + return { type: VarType.boolean, schemaType } + case 'object': + return { type: VarType.object, schemaType } + case 'array': { + const itemSchema = pickItemSchema(schema) + if (!itemSchema) + return { type: VarType.array, schemaType } + + const { type: itemType, schemaType: itemSchemaType } = resolveVarType(itemSchema, schemaTypeDefinitions) + const resolvedSchemaType = schemaType || itemSchemaType + + if (itemSchemaType === 'file') + return { type: VarType.arrayFile, schemaType: resolvedSchemaType } + + switch (itemType) { + case VarType.string: + return { type: VarType.arrayString, schemaType: resolvedSchemaType } + case VarType.number: + case VarType.integer: + return { type: VarType.arrayNumber, schemaType: resolvedSchemaType } + case VarType.boolean: + return { type: VarType.arrayBoolean, schemaType: resolvedSchemaType } + case VarType.object: + return { type: VarType.arrayObject, schemaType: resolvedSchemaType } + case VarType.file: + return { type: VarType.arrayFile, schemaType: resolvedSchemaType } + default: + return { type: VarType.array, schemaType: resolvedSchemaType } + } + } + default: + return { type: VarType.any, schemaType } + } +} + +const toFieldType = (normalizedType: string | undefined, schemaType?: string): Type => { + if (schemaType === 'file') + return normalizedType === 'array' ? Type.array : Type.file + + switch (normalizedType) { + case 'number': + case 'integer': + return Type.number + case 'boolean': + return Type.boolean + case 'object': + return Type.object + case 'array': + return Type.array + case 'string': + default: + return Type.string + } +} + +const toArrayItemType = (type: Type): Exclude<Type, Type.array> => { + if (type === Type.array) + return Type.object + return type as Exclude<Type, Type.array> +} + +const convertJsonSchemaToField = (schema: any, schemaTypeDefinitions?: SchemaTypeDefinition[]): Field => { + const schemaType = extractSchemaType(schema, schemaTypeDefinitions) + const normalizedType = normalizeJsonSchemaType(schema) + const fieldType = toFieldType(normalizedType, schemaType) + + const field: Field = { + type: fieldType, + } + + if (schema?.description) + field.description = schema.description + + if (schemaType) + field.schemaType = schemaType + + if (Array.isArray(schema?.enum)) + field.enum = schema.enum + + if (fieldType === Type.object) { + const properties = schema?.properties || {} + field.properties = Object.entries(properties).reduce((acc, [key, value]) => { + acc[key] = convertJsonSchemaToField(value, schemaTypeDefinitions) + return acc + }, {} as Record<string, Field>) + + const required = Array.isArray(schema?.required) ? schema.required.filter(Boolean) : undefined + field.required = required && required.length > 0 ? required : undefined + field.additionalProperties = false + } + + if (fieldType === Type.array) { + const itemSchema = pickItemSchema(schema) + if (itemSchema) { + const itemField = convertJsonSchemaToField(itemSchema, schemaTypeDefinitions) + const { type, ...rest } = itemField + field.items = { + ...rest, + type: toArrayItemType(type), + } + } + } + + return field +} + +const buildOutputVars = (schema: Record<string, any>, schemaTypeDefinitions?: SchemaTypeDefinition[]): Var[] => { + if (!schema || typeof schema !== 'object') + return [] + + const properties = schema.properties as Record<string, any> | undefined + if (!properties) + return [] + + return Object.entries(properties).map(([name, propertySchema]) => { + const { type, schemaType } = resolveVarType(propertySchema, schemaTypeDefinitions) + const normalizedType = normalizeJsonSchemaType(propertySchema) + + const varItem: Var = { + variable: name, + type, + des: propertySchema?.description, + ...(schemaType ? { schemaType } : {}), + } + + if (normalizedType === 'object') { + const childProperties = propertySchema?.properties + ? Object.entries(propertySchema.properties).reduce((acc, [key, value]) => { + acc[key] = convertJsonSchemaToField(value, schemaTypeDefinitions) + return acc + }, {} as Record<string, Field>) + : {} + + const required = Array.isArray(propertySchema?.required) ? propertySchema.required.filter(Boolean) : undefined + + varItem.children = { + schema: { + type: Type.object, + properties: childProperties, + required: required && required.length > 0 ? required : undefined, + additionalProperties: false, + }, + } as StructuredOutput + } + + return varItem + }) +} + +const metaData = genNodeMetaData({ + sort: 1, + type: BlockEnum.TriggerPlugin, + helpLinkUri: 'plugin-trigger', + isStart: true, +}) + +const nodeDefault: NodeDefault<PluginTriggerNodeType> = { + metaData, + defaultValue: { + plugin_id: '', + event_name: '', + event_parameters: {}, + // event_type: '', + config: {}, + }, + checkValid(payload: PluginTriggerNodeType, t: any, moreDataForCheckValid: { + triggerInputsSchema?: Array<{ + variable: string + label: string + required?: boolean + }> + isReadyForCheckValid?: boolean + } = {}) { + let errorMessage = '' + + if (!payload.subscription_id) + errorMessage = t('workflow.nodes.triggerPlugin.subscriptionRequired') + + const { + triggerInputsSchema = [], + isReadyForCheckValid = true, + } = moreDataForCheckValid || {} + + if (!errorMessage && isReadyForCheckValid) { + triggerInputsSchema.filter(field => field.required).forEach((field) => { + if (errorMessage) + return + + const rawParam = payload.event_parameters?.[field.variable] + ?? (payload.config as Record<string, any> | undefined)?.[field.variable] + if (!rawParam) { + errorMessage = t('workflow.errorMsg.fieldRequired', { field: field.label }) + return + } + + const targetParam = typeof rawParam === 'object' && rawParam !== null && 'type' in rawParam + ? rawParam as { type: VarKindType; value: any } + : { type: VarKindType.constant, value: rawParam } + + const { type, value } = targetParam + if (type === VarKindType.variable) { + if (!value || (Array.isArray(value) && value.length === 0)) + errorMessage = t('workflow.errorMsg.fieldRequired', { field: field.label }) + } + else { + if ( + value === undefined + || value === null + || value === '' + || (Array.isArray(value) && value.length === 0) + ) + errorMessage = t('workflow.errorMsg.fieldRequired', { field: field.label }) + } + }) + } + + return { + isValid: !errorMessage, + errorMessage, + } + }, + getOutputVars(payload, _allPluginInfoList, _ragVars, { schemaTypeDefinitions } = { schemaTypeDefinitions: [] }) { + const schema = payload.output_schema || {} + return buildOutputVars(schema, schemaTypeDefinitions) + }, +} + +export default nodeDefault diff --git a/web/app/components/workflow/nodes/trigger-plugin/hooks/use-trigger-auth-flow.ts b/web/app/components/workflow/nodes/trigger-plugin/hooks/use-trigger-auth-flow.ts new file mode 100644 index 0000000000..983b8512de --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/hooks/use-trigger-auth-flow.ts @@ -0,0 +1,162 @@ +import { useCallback, useState } from 'react' +import { + useBuildTriggerSubscription, + useCreateTriggerSubscriptionBuilder, + useUpdateTriggerSubscriptionBuilder, + useVerifyTriggerSubscriptionBuilder, +} from '@/service/use-triggers' +import type { TriggerWithProvider } from '@/app/components/workflow/block-selector/types' + +// Helper function to serialize complex values to strings for backend encryption +const serializeFormValues = (values: Record<string, any>): Record<string, string> => { + const result: Record<string, string> = {} + + for (const [key, value] of Object.entries(values)) { + if (value === null || value === undefined) + result[key] = '' + else if (typeof value === 'object') + result[key] = JSON.stringify(value) + else + result[key] = String(value) + } + + return result +} + +export type AuthFlowStep = 'auth' | 'params' | 'complete' + +export type AuthFlowState = { + step: AuthFlowStep + builderId: string + isLoading: boolean + error: string | null +} + +export type AuthFlowActions = { + startAuth: () => Promise<void> + verifyAuth: (credentials: Record<string, any>) => Promise<void> + completeConfig: (parameters: Record<string, any>, properties?: Record<string, any>, name?: string) => Promise<void> + reset: () => void +} + +export const useTriggerAuthFlow = (provider: TriggerWithProvider): AuthFlowState & AuthFlowActions => { + const [step, setStep] = useState<AuthFlowStep>('auth') + const [builderId, setBuilderId] = useState<string>('') + const [isLoading, setIsLoading] = useState(false) + const [error, setError] = useState<string | null>(null) + + const createBuilder = useCreateTriggerSubscriptionBuilder() + const updateBuilder = useUpdateTriggerSubscriptionBuilder() + const verifyBuilder = useVerifyTriggerSubscriptionBuilder() + const buildSubscription = useBuildTriggerSubscription() + + const startAuth = useCallback(async () => { + if (builderId) return // Prevent multiple calls if already started + + setIsLoading(true) + setError(null) + + try { + const response = await createBuilder.mutateAsync({ + provider: provider.name, + }) + setBuilderId(response.subscription_builder.id) + setStep('auth') + } + catch (err: any) { + setError(err.message || 'Failed to start authentication flow') + throw err + } + finally { + setIsLoading(false) + } + }, [provider.name, createBuilder, builderId]) + + const verifyAuth = useCallback(async (credentials: Record<string, any>) => { + if (!builderId) { + setError('No builder ID available') + return + } + + setIsLoading(true) + setError(null) + + try { + await updateBuilder.mutateAsync({ + provider: provider.name, + subscriptionBuilderId: builderId, + credentials: serializeFormValues(credentials), + }) + + await verifyBuilder.mutateAsync({ + provider: provider.name, + subscriptionBuilderId: builderId, + }) + + setStep('params') + } + catch (err: any) { + setError(err.message || 'Authentication verification failed') + throw err + } + finally { + setIsLoading(false) + } + }, [provider.name, builderId, updateBuilder, verifyBuilder]) + + const completeConfig = useCallback(async ( + parameters: Record<string, any>, + properties: Record<string, any> = {}, + name?: string, + ) => { + if (!builderId) { + setError('No builder ID available') + return + } + + setIsLoading(true) + setError(null) + + try { + await updateBuilder.mutateAsync({ + provider: provider.name, + subscriptionBuilderId: builderId, + parameters: serializeFormValues(parameters), + properties: serializeFormValues(properties), + name, + }) + + await buildSubscription.mutateAsync({ + provider: provider.name, + subscriptionBuilderId: builderId, + }) + + setStep('complete') + } + catch (err: any) { + setError(err.message || 'Configuration failed') + throw err + } + finally { + setIsLoading(false) + } + }, [provider.name, builderId, updateBuilder, buildSubscription]) + + const reset = useCallback(() => { + setStep('auth') + setBuilderId('') + setIsLoading(false) + setError(null) + }, []) + + return { + step, + builderId, + isLoading, + error, + startAuth, + verifyAuth, + completeConfig, + reset, + } +} diff --git a/web/app/components/workflow/nodes/trigger-plugin/node.tsx b/web/app/components/workflow/nodes/trigger-plugin/node.tsx new file mode 100644 index 0000000000..0eee4cb8b4 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/node.tsx @@ -0,0 +1,126 @@ +import NodeStatus, { NodeStatusEnum } from '@/app/components/base/node-status' +import type { NodeProps } from '@/app/components/workflow/types' +import type { FC } from 'react' +import React, { useEffect, useMemo } from 'react' +import { useTranslation } from 'react-i18next' +import { InstallPluginButton } from '@/app/components/workflow/nodes/_base/components/install-plugin-button' +import { useNodePluginInstallation } from '@/app/components/workflow/hooks/use-node-plugin-installation' +import { useNodeDataUpdate } from '@/app/components/workflow/hooks/use-node-data-update' +import type { PluginTriggerNodeType } from './types' +import useConfig from './use-config' + +const formatConfigValue = (rawValue: any): string => { + if (rawValue === null || rawValue === undefined) + return '' + + if (typeof rawValue === 'string' || typeof rawValue === 'number' || typeof rawValue === 'boolean') + return String(rawValue) + + if (Array.isArray(rawValue)) + return rawValue.join('.') + + if (typeof rawValue === 'object') { + const { value } = rawValue as { value?: any } + if (value === null || value === undefined) + return '' + if (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') + return String(value) + if (Array.isArray(value)) + return value.join('.') + try { + return JSON.stringify(value) + } + catch { + return '' + } + } + + return '' +} + +const Node: FC<NodeProps<PluginTriggerNodeType>> = ({ + id, + data, +}) => { + const { subscriptions } = useConfig(id, data) + const { config = {}, subscription_id } = data + const configKeys = Object.keys(config) + const { + isChecking, + isMissing, + uniqueIdentifier, + canInstall, + onInstallSuccess, + shouldDim, + } = useNodePluginInstallation(data) + const { handleNodeDataUpdate } = useNodeDataUpdate() + const showInstallButton = !isChecking && isMissing && canInstall && uniqueIdentifier + const shouldLock = !isChecking && isMissing && canInstall && Boolean(uniqueIdentifier) + + useEffect(() => { + if (data._pluginInstallLocked === shouldLock && data._dimmed === shouldDim) + return + handleNodeDataUpdate({ + id, + data: { + _pluginInstallLocked: shouldLock, + _dimmed: shouldDim, + }, + }) + }, [data._pluginInstallLocked, data._dimmed, handleNodeDataUpdate, id, shouldDim, shouldLock]) + + const { t } = useTranslation() + + const isValidSubscription = useMemo(() => { + return subscription_id && subscriptions?.some(sub => sub.id === subscription_id) + }, [subscription_id, subscriptions]) + + return ( + <div className="relative mb-1 px-3 py-1"> + {showInstallButton && ( + <div className="pointer-events-auto absolute right-3 top-[-32px] z-40"> + <InstallPluginButton + size="small" + extraIdentifiers={[ + data.plugin_id, + data.provider_id, + data.provider_name, + ].filter(Boolean) as string[]} + className="!font-medium !text-text-accent" + uniqueIdentifier={uniqueIdentifier!} + onSuccess={onInstallSuccess} + /> + </div> + )} + <div className="space-y-0.5" aria-disabled={shouldDim}> + {!isValidSubscription && <NodeStatus status={NodeStatusEnum.warning} message={t('pluginTrigger.node.status.warning')} />} + {isValidSubscription && configKeys.map((key, index) => ( + <div + key={index} + className="flex h-6 items-center justify-between space-x-1 rounded-md bg-workflow-block-parma-bg px-1 text-xs font-normal text-text-secondary" + > + <div + title={key} + className="max-w-[100px] shrink-0 truncate text-xs font-medium uppercase text-text-tertiary" + > + {key} + </div> + <div + title={formatConfigValue(config[key])} + className="w-0 shrink-0 grow truncate text-right text-xs font-normal text-text-secondary" + > + {(() => { + const displayValue = formatConfigValue(config[key]) + if (displayValue.includes('secret')) + return '********' + return displayValue + })()} + </div> + </div> + ))} + </div> + </div> + ) +} + +export default React.memo(Node) diff --git a/web/app/components/workflow/nodes/trigger-plugin/panel.tsx b/web/app/components/workflow/nodes/trigger-plugin/panel.tsx new file mode 100644 index 0000000000..9b4d8058b1 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/panel.tsx @@ -0,0 +1,94 @@ +import type { FC } from 'react' +import React from 'react' +import type { PluginTriggerNodeType } from './types' +import Split from '@/app/components/workflow/nodes/_base/components/split' +import OutputVars, { VarItem } from '@/app/components/workflow/nodes/_base/components/output-vars' +import type { NodePanelProps } from '@/app/components/workflow/types' +import useConfig from './use-config' +import TriggerForm from './components/trigger-form' +import StructureOutputItem from '@/app/components/workflow/nodes/_base/components/variable/object-child-tree-panel/show' +import { Type } from '../llm/types' +import { BlockEnum } from '@/app/components/workflow/types' + +const Panel: FC<NodePanelProps<PluginTriggerNodeType>> = ({ + id, + data, +}) => { + const { + readOnly, + triggerParameterSchema, + triggerParameterValue, + setTriggerParameterValue, + outputSchema, + hasObjectOutput, + currentProvider, + currentEvent, + subscriptionSelected, + } = useConfig(id, data) + const disableVariableInsertion = data.type === BlockEnum.TriggerPlugin + + // Convert output schema to VarItem format + const outputVars = Object.entries(outputSchema.properties || {}).map(([name, schema]: [string, any]) => ({ + name, + type: schema.type || 'string', + description: schema.description || '', + })) + + return ( + <div className='mt-2'> + {/* Dynamic Parameters Form - Only show when authenticated */} + {triggerParameterSchema.length > 0 && subscriptionSelected && ( + <> + <div className='px-4 pb-4'> + <TriggerForm + readOnly={readOnly} + nodeId={id} + schema={triggerParameterSchema as any} + value={triggerParameterValue} + onChange={setTriggerParameterValue} + currentProvider={currentProvider} + currentEvent={currentEvent} + disableVariableInsertion={disableVariableInsertion} + /> + </div> + <Split /> + </> + )} + + {/* Output Variables - Always show */} + <OutputVars> + <> + {outputVars.map(varItem => ( + <VarItem + key={varItem.name} + name={varItem.name} + type={varItem.type} + description={varItem.description} + isIndent={hasObjectOutput} + /> + ))} + {Object.entries(outputSchema.properties || {}).map(([name, schema]: [string, any]) => ( + <div key={name}> + {schema.type === 'object' ? ( + <StructureOutputItem + rootClassName='code-sm-semibold text-text-secondary' + payload={{ + schema: { + type: Type.object, + properties: { + [name]: schema, + }, + additionalProperties: false, + }, + }} + /> + ) : null} + </div> + ))} + </> + </OutputVars> + </div> + ) +} + +export default React.memo(Panel) diff --git a/web/app/components/workflow/nodes/trigger-plugin/types.ts b/web/app/components/workflow/nodes/trigger-plugin/types.ts new file mode 100644 index 0000000000..6dba97d795 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/types.ts @@ -0,0 +1,24 @@ +import type { CommonNodeType } from '@/app/components/workflow/types' +import type { CollectionType } from '@/app/components/tools/types' +import type { ResourceVarInputs } from '../_base/types' + +export type PluginTriggerNodeType = CommonNodeType & { + provider_id: string + provider_type: CollectionType + provider_name: string + event_name: string + event_label: string + event_parameters: PluginTriggerVarInputs + event_configurations: Record<string, any> + output_schema: Record<string, any> + parameters_schema?: Record<string, any>[] + version?: string + event_node_version?: string + plugin_id?: string + config?: Record<string, any> + plugin_unique_identifier?: string +} + +// Use base types directly +export { VarKindType as PluginTriggerVarType } from '../_base/types' +export type PluginTriggerVarInputs = ResourceVarInputs diff --git a/web/app/components/workflow/nodes/trigger-plugin/use-check-params.ts b/web/app/components/workflow/nodes/trigger-plugin/use-check-params.ts new file mode 100644 index 0000000000..16b763f11a --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/use-check-params.ts @@ -0,0 +1,27 @@ +import { useCallback } from 'react' +import type { PluginTriggerNodeType } from './types' +import { useAllTriggerPlugins } from '@/service/use-triggers' +import { useGetLanguage } from '@/context/i18n' +import { getTriggerCheckParams } from '@/app/components/workflow/utils/trigger' + +type Params = { + id: string + payload: PluginTriggerNodeType +} + +const useGetDataForCheckMore = ({ + payload, +}: Params) => { + const { data: triggerPlugins } = useAllTriggerPlugins() + const language = useGetLanguage() + + const getData = useCallback(() => { + return getTriggerCheckParams(payload, triggerPlugins, language) + }, [payload, triggerPlugins, language]) + + return { + getData, + } +} + +export default useGetDataForCheckMore diff --git a/web/app/components/workflow/nodes/trigger-plugin/use-config.ts b/web/app/components/workflow/nodes/trigger-plugin/use-config.ts new file mode 100644 index 0000000000..cf66913e58 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/use-config.ts @@ -0,0 +1,233 @@ +import { useCallback, useEffect, useMemo } from 'react' +import { produce } from 'immer' +import type { PluginTriggerNodeType } from './types' +import type { PluginTriggerVarInputs } from './types' +import useNodeCrud from '@/app/components/workflow/nodes/_base/hooks/use-node-crud' +import { useNodesReadOnly } from '@/app/components/workflow/hooks' +import { + useAllTriggerPlugins, + useTriggerSubscriptions, +} from '@/service/use-triggers' +import { + getConfiguredValue, + toolParametersToFormSchemas, +} from '@/app/components/tools/utils/to-form-schema' +import type { InputVar } from '@/app/components/workflow/types' +import type { TriggerWithProvider } from '@/app/components/workflow/block-selector/types' +import type { Event } from '@/app/components/tools/types' +import { VarKindType } from '../_base/types' + +const normalizeEventParameters = ( + params: PluginTriggerVarInputs | Record<string, unknown> | null | undefined, + { allowScalars = false }: { allowScalars?: boolean } = {}, +): PluginTriggerVarInputs => { + if (!params || typeof params !== 'object' || Array.isArray(params)) + return {} as PluginTriggerVarInputs + + return Object.entries(params).reduce((acc, [key, entry]) => { + if (!entry && entry !== 0 && entry !== false) + return acc + + if ( + typeof entry === 'object' + && !Array.isArray(entry) + && 'type' in entry + && 'value' in entry + ) { + const normalizedEntry = { ...(entry as PluginTriggerVarInputs[string]) } + if (normalizedEntry.type === VarKindType.mixed) + normalizedEntry.type = VarKindType.constant + acc[key] = normalizedEntry + return acc + } + + if (!allowScalars) + return acc + + if (typeof entry === 'string') { + acc[key] = { + type: VarKindType.constant, + value: entry, + } + return acc + } + + if (typeof entry === 'number' || typeof entry === 'boolean') { + acc[key] = { + type: VarKindType.constant, + value: entry, + } + return acc + } + + if (Array.isArray(entry) && entry.every(item => typeof item === 'string')) { + acc[key] = { + type: VarKindType.variable, + value: entry, + } + } + + return acc + }, {} as PluginTriggerVarInputs) +} + +const useConfig = (id: string, payload: PluginTriggerNodeType) => { + const { nodesReadOnly: readOnly } = useNodesReadOnly() + const { data: triggerPlugins = [] } = useAllTriggerPlugins() + + const { inputs, setInputs: doSetInputs } = useNodeCrud<PluginTriggerNodeType>( + id, + payload, + ) + + const { + provider_id, + provider_name, + event_name, + config = {}, + event_parameters: rawEventParameters = {}, + subscription_id, + } = inputs + + const event_parameters = useMemo( + () => normalizeEventParameters(rawEventParameters as PluginTriggerVarInputs), + [rawEventParameters], + ) + const legacy_config_parameters = useMemo( + () => normalizeEventParameters(config as PluginTriggerVarInputs, { allowScalars: true }), + [config], + ) + + const currentProvider = useMemo<TriggerWithProvider | undefined>(() => { + return triggerPlugins.find( + provider => + provider.name === provider_name + || provider.id === provider_id + || (provider_id && provider.plugin_id === provider_id), + ) + }, [triggerPlugins, provider_name, provider_id]) + + const { data: subscriptions = [] } = useTriggerSubscriptions(provider_id || '') + + const subscriptionSelected = useMemo(() => { + return subscriptions?.find(s => s.id === subscription_id) + }, [subscriptions, subscription_id]) + + const currentEvent = useMemo<Event | undefined>(() => { + return currentProvider?.events.find( + event => event.name === event_name, + ) + }, [currentProvider, event_name]) + + // Dynamic trigger parameters (from specific trigger.parameters) + const triggerSpecificParameterSchema = useMemo(() => { + if (!currentEvent) return [] + return toolParametersToFormSchemas(currentEvent.parameters) + }, [currentEvent]) + + // Combined parameter schema (subscription + trigger specific) + const triggerParameterSchema = useMemo(() => { + const schemaMap = new Map() + + triggerSpecificParameterSchema.forEach((schema) => { + schemaMap.set(schema.variable || schema.name, schema) + }) + + return Array.from(schemaMap.values()) + }, [triggerSpecificParameterSchema]) + + const triggerParameterValue = useMemo(() => { + if (!triggerParameterSchema.length) + return {} as PluginTriggerVarInputs + + const hasStoredParameters = event_parameters && Object.keys(event_parameters).length > 0 + const baseValue = hasStoredParameters ? event_parameters : legacy_config_parameters + + const configuredValue = getConfiguredValue(baseValue, triggerParameterSchema) as PluginTriggerVarInputs + return normalizeEventParameters(configuredValue) + }, [triggerParameterSchema, event_parameters, legacy_config_parameters]) + + useEffect(() => { + if (!triggerParameterSchema.length) + return + + if (event_parameters && Object.keys(event_parameters).length > 0) + return + + if (!triggerParameterValue || Object.keys(triggerParameterValue).length === 0) + return + + const newInputs = produce(inputs, (draft) => { + draft.event_parameters = triggerParameterValue + draft.config = triggerParameterValue + }) + doSetInputs(newInputs) + }, [ + doSetInputs, + event_parameters, + inputs, + triggerParameterSchema, + triggerParameterValue, + ]) + + const setTriggerParameterValue = useCallback( + (value: PluginTriggerVarInputs) => { + const sanitizedValue = normalizeEventParameters(value) + const newInputs = produce(inputs, (draft) => { + draft.event_parameters = sanitizedValue + draft.config = sanitizedValue + }) + doSetInputs(newInputs) + }, + [inputs, doSetInputs], + ) + + const setInputVar = useCallback( + (variable: InputVar, varDetail: InputVar) => { + const newInputs = produce(inputs, (draft) => { + const nextEventParameters = normalizeEventParameters({ + ...draft.event_parameters, + [variable.variable]: { + type: VarKindType.variable, + value: varDetail.variable, + }, + } as PluginTriggerVarInputs) + + draft.event_parameters = nextEventParameters + draft.config = nextEventParameters + }) + doSetInputs(newInputs) + }, + [inputs, doSetInputs], + ) + + // Get output schema + const outputSchema = useMemo(() => { + return currentEvent?.output_schema || {} + }, [currentEvent]) + + // Check if trigger has complex output structure + const hasObjectOutput = useMemo(() => { + const properties = outputSchema.properties || {} + return Object.values(properties).some( + (prop: any) => prop.type === 'object', + ) + }, [outputSchema]) + + return { + readOnly, + inputs, + currentProvider, + currentEvent, + triggerParameterSchema, + triggerParameterValue, + setTriggerParameterValue, + setInputVar, + outputSchema, + hasObjectOutput, + subscriptions, + subscriptionSelected, + } +} + +export default useConfig diff --git a/web/app/components/workflow/nodes/trigger-plugin/utils/__tests__/form-helpers.test.ts b/web/app/components/workflow/nodes/trigger-plugin/utils/__tests__/form-helpers.test.ts new file mode 100644 index 0000000000..c75ffc0a59 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/utils/__tests__/form-helpers.test.ts @@ -0,0 +1,308 @@ +import { deepSanitizeFormValues, findMissingRequiredField, sanitizeFormValues } from '../form-helpers' + +describe('Form Helpers', () => { + describe('sanitizeFormValues', () => { + it('should convert null values to empty strings', () => { + const input = { field1: null, field2: 'value', field3: undefined } + const result = sanitizeFormValues(input) + + expect(result).toEqual({ + field1: '', + field2: 'value', + field3: '', + }) + }) + + it('should convert undefined values to empty strings', () => { + const input = { field1: undefined, field2: 'test' } + const result = sanitizeFormValues(input) + + expect(result).toEqual({ + field1: '', + field2: 'test', + }) + }) + + it('should convert non-string values to strings', () => { + const input = { number: 123, boolean: true, string: 'test' } + const result = sanitizeFormValues(input) + + expect(result).toEqual({ + number: '123', + boolean: 'true', + string: 'test', + }) + }) + + it('should handle empty objects', () => { + const result = sanitizeFormValues({}) + expect(result).toEqual({}) + }) + + it('should handle objects with mixed value types', () => { + const input = { + null_field: null, + undefined_field: undefined, + zero: 0, + false_field: false, + empty_string: '', + valid_string: 'test', + } + const result = sanitizeFormValues(input) + + expect(result).toEqual({ + null_field: '', + undefined_field: '', + zero: '0', + false_field: 'false', + empty_string: '', + valid_string: 'test', + }) + }) + }) + + describe('deepSanitizeFormValues', () => { + it('should handle nested objects', () => { + const input = { + level1: { + field1: null, + field2: 'value', + level2: { + field3: undefined, + field4: 'nested', + }, + }, + simple: 'test', + } + const result = deepSanitizeFormValues(input) + + expect(result).toEqual({ + level1: { + field1: '', + field2: 'value', + level2: { + field3: '', + field4: 'nested', + }, + }, + simple: 'test', + }) + }) + + it('should handle arrays correctly', () => { + const input = { + array: [1, 2, 3], + nested: { + array: ['a', null, 'c'], + }, + } + const result = deepSanitizeFormValues(input) + + expect(result).toEqual({ + array: [1, 2, 3], + nested: { + array: ['a', null, 'c'], + }, + }) + }) + + it('should handle null and undefined at root level', () => { + const input = { + null_field: null, + undefined_field: undefined, + nested: { + null_nested: null, + }, + } + const result = deepSanitizeFormValues(input) + + expect(result).toEqual({ + null_field: '', + undefined_field: '', + nested: { + null_nested: '', + }, + }) + }) + + it('should handle deeply nested structures', () => { + const input = { + level1: { + level2: { + level3: { + field: null, + }, + }, + }, + } + const result = deepSanitizeFormValues(input) + + expect(result).toEqual({ + level1: { + level2: { + level3: { + field: '', + }, + }, + }, + }) + }) + + it('should preserve non-null values in nested structures', () => { + const input = { + config: { + client_id: 'valid_id', + client_secret: null, + options: { + timeout: 5000, + enabled: true, + message: undefined, + }, + }, + } + const result = deepSanitizeFormValues(input) + + expect(result).toEqual({ + config: { + client_id: 'valid_id', + client_secret: '', + options: { + timeout: 5000, + enabled: true, + message: '', + }, + }, + }) + }) + }) + + describe('findMissingRequiredField', () => { + const requiredFields = [ + { name: 'client_id', label: 'Client ID' }, + { name: 'client_secret', label: 'Client Secret' }, + { name: 'scope', label: 'Scope' }, + ] + + it('should return null when all required fields are present', () => { + const formData = { + client_id: 'test_id', + client_secret: 'test_secret', + scope: 'read', + optional_field: 'optional', + } + + const result = findMissingRequiredField(formData, requiredFields) + expect(result).toBeNull() + }) + + it('should return the first missing field', () => { + const formData = { + client_id: 'test_id', + scope: 'read', + } + + const result = findMissingRequiredField(formData, requiredFields) + expect(result).toEqual({ name: 'client_secret', label: 'Client Secret' }) + }) + + it('should treat empty strings as missing fields', () => { + const formData = { + client_id: '', + client_secret: 'test_secret', + scope: 'read', + } + + const result = findMissingRequiredField(formData, requiredFields) + expect(result).toEqual({ name: 'client_id', label: 'Client ID' }) + }) + + it('should treat null values as missing fields', () => { + const formData = { + client_id: 'test_id', + client_secret: null, + scope: 'read', + } + + const result = findMissingRequiredField(formData, requiredFields) + expect(result).toEqual({ name: 'client_secret', label: 'Client Secret' }) + }) + + it('should treat undefined values as missing fields', () => { + const formData = { + client_id: 'test_id', + client_secret: 'test_secret', + scope: undefined, + } + + const result = findMissingRequiredField(formData, requiredFields) + expect(result).toEqual({ name: 'scope', label: 'Scope' }) + }) + + it('should handle empty required fields array', () => { + const formData = { + client_id: 'test_id', + } + + const result = findMissingRequiredField(formData, []) + expect(result).toBeNull() + }) + + it('should handle empty form data', () => { + const result = findMissingRequiredField({}, requiredFields) + expect(result).toEqual({ name: 'client_id', label: 'Client ID' }) + }) + + it('should handle multilingual labels', () => { + const multilingualFields = [ + { name: 'field1', label: { en_US: 'Field 1 EN', zh_Hans: 'Field 1 CN' } }, + ] + const formData = {} + + const result = findMissingRequiredField(formData, multilingualFields) + expect(result).toEqual({ + name: 'field1', + label: { en_US: 'Field 1 EN', zh_Hans: 'Field 1 CN' }, + }) + }) + + it('should return null for form data with extra fields', () => { + const formData = { + client_id: 'test_id', + client_secret: 'test_secret', + scope: 'read', + extra_field1: 'extra1', + extra_field2: 'extra2', + } + + const result = findMissingRequiredField(formData, requiredFields) + expect(result).toBeNull() + }) + }) + + describe('Edge cases', () => { + it('should handle objects with non-string keys', () => { + const input = { [Symbol('test')]: 'value', regular: 'field' } as any + const result = sanitizeFormValues(input) + + expect(result.regular).toBe('field') + }) + + it('should handle objects with getter properties', () => { + const obj = {} + Object.defineProperty(obj, 'getter', { + get: () => 'computed_value', + enumerable: true, + }) + + const result = sanitizeFormValues(obj) + expect(result.getter).toBe('computed_value') + }) + + it('should handle circular references in deepSanitizeFormValues gracefully', () => { + const obj: any = { field: 'value' } + obj.circular = obj + + expect(() => deepSanitizeFormValues(obj)).not.toThrow() + }) + }) +}) diff --git a/web/app/components/workflow/nodes/trigger-plugin/utils/form-helpers.ts b/web/app/components/workflow/nodes/trigger-plugin/utils/form-helpers.ts new file mode 100644 index 0000000000..36090d9771 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-plugin/utils/form-helpers.ts @@ -0,0 +1,55 @@ +/** + * Utility functions for form data handling in trigger plugin components + */ + +/** + * Sanitizes form values by converting null/undefined to empty strings + * This ensures React form inputs don't receive null values which can cause warnings + */ +export const sanitizeFormValues = (values: Record<string, any>): Record<string, string> => { + return Object.fromEntries( + Object.entries(values).map(([key, value]) => [ + key, + value === null || value === undefined ? '' : String(value), + ]), + ) +} + +/** + * Deep sanitizes form values while preserving nested objects structure + * Useful for complex form schemas with nested properties + */ +export const deepSanitizeFormValues = (values: Record<string, any>, visited = new WeakSet()): Record<string, any> => { + if (visited.has(values)) + return {} + + visited.add(values) + + const result: Record<string, any> = {} + + for (const [key, value] of Object.entries(values)) { + if (value === null || value === undefined) + result[key] = '' + else if (typeof value === 'object' && !Array.isArray(value)) + result[key] = deepSanitizeFormValues(value, visited) + else + result[key] = value + } + + return result +} + +/** + * Validates required fields in form data + * Returns the first missing required field or null if all are present + */ +export const findMissingRequiredField = ( + formData: Record<string, any>, + requiredFields: Array<{ name: string; label: any }>, +): { name: string; label: any } | null => { + for (const field of requiredFields) { + if (!formData[field.name] || formData[field.name] === '') + return field + } + return null +} diff --git a/web/app/components/workflow/nodes/trigger-schedule/components/frequency-selector.tsx b/web/app/components/workflow/nodes/trigger-schedule/components/frequency-selector.tsx new file mode 100644 index 0000000000..d0de74a6ef --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/components/frequency-selector.tsx @@ -0,0 +1,38 @@ +import React, { useMemo } from 'react' +import { useTranslation } from 'react-i18next' +import { SimpleSelect } from '@/app/components/base/select' +import type { ScheduleFrequency } from '../types' + +type FrequencySelectorProps = { + frequency: ScheduleFrequency + onChange: (frequency: ScheduleFrequency) => void +} + +const FrequencySelector = ({ frequency, onChange }: FrequencySelectorProps) => { + const { t } = useTranslation() + + const frequencies = useMemo(() => [ + { value: 'frequency-header', name: t('workflow.nodes.triggerSchedule.frequency.label'), isGroup: true }, + { value: 'hourly', name: t('workflow.nodes.triggerSchedule.frequency.hourly') }, + { value: 'daily', name: t('workflow.nodes.triggerSchedule.frequency.daily') }, + { value: 'weekly', name: t('workflow.nodes.triggerSchedule.frequency.weekly') }, + { value: 'monthly', name: t('workflow.nodes.triggerSchedule.frequency.monthly') }, + ], [t]) + + return ( + <SimpleSelect + key={`${frequency}-${frequencies[0]?.name}`} // Include translation in key to force re-render + items={frequencies} + defaultValue={frequency} + onSelect={item => onChange(item.value as ScheduleFrequency)} + placeholder={t('workflow.nodes.triggerSchedule.selectFrequency')} + className="w-full py-2" + wrapperClassName="h-auto" + optionWrapClassName="min-w-40" + notClearable={true} + allowSearch={false} + /> + ) +} + +export default FrequencySelector diff --git a/web/app/components/workflow/nodes/trigger-schedule/components/mode-switcher.tsx b/web/app/components/workflow/nodes/trigger-schedule/components/mode-switcher.tsx new file mode 100644 index 0000000000..6dc88c85bf --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/components/mode-switcher.tsx @@ -0,0 +1,37 @@ +import React from 'react' +import { useTranslation } from 'react-i18next' +import { RiCalendarLine, RiCodeLine } from '@remixicon/react' +import { SegmentedControl } from '@/app/components/base/segmented-control' +import type { ScheduleMode } from '../types' + +type ModeSwitcherProps = { + mode: ScheduleMode + onChange: (mode: ScheduleMode) => void +} + +const ModeSwitcher = ({ mode, onChange }: ModeSwitcherProps) => { + const { t } = useTranslation() + + const options = [ + { + Icon: RiCalendarLine, + text: t('workflow.nodes.triggerSchedule.mode.visual'), + value: 'visual' as const, + }, + { + Icon: RiCodeLine, + text: t('workflow.nodes.triggerSchedule.mode.cron'), + value: 'cron' as const, + }, + ] + + return ( + <SegmentedControl + options={options} + value={mode} + onChange={onChange} + /> + ) +} + +export default ModeSwitcher diff --git a/web/app/components/workflow/nodes/trigger-schedule/components/mode-toggle.tsx b/web/app/components/workflow/nodes/trigger-schedule/components/mode-toggle.tsx new file mode 100644 index 0000000000..6ae5d2cf67 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/components/mode-toggle.tsx @@ -0,0 +1,37 @@ +import React from 'react' +import { useTranslation } from 'react-i18next' +import { Asterisk, CalendarCheckLine } from '@/app/components/base/icons/src/vender/workflow' +import type { ScheduleMode } from '../types' + +type ModeToggleProps = { + mode: ScheduleMode + onChange: (mode: ScheduleMode) => void +} + +const ModeToggle = ({ mode, onChange }: ModeToggleProps) => { + const { t } = useTranslation() + + const handleToggle = () => { + const newMode = mode === 'visual' ? 'cron' : 'visual' + onChange(newMode) + } + + const currentText = mode === 'visual' + ? t('workflow.nodes.triggerSchedule.useCronExpression') + : t('workflow.nodes.triggerSchedule.useVisualPicker') + + const currentIcon = mode === 'visual' ? Asterisk : CalendarCheckLine + + return ( + <button + type="button" + onClick={handleToggle} + className="flex cursor-pointer items-center gap-1 rounded-lg px-2 py-1 text-sm text-text-secondary hover:bg-state-base-hover" + > + {React.createElement(currentIcon, { className: 'w-4 h-4' })} + <span>{currentText}</span> + </button> + ) +} + +export default ModeToggle diff --git a/web/app/components/workflow/nodes/trigger-schedule/components/monthly-days-selector.tsx b/web/app/components/workflow/nodes/trigger-schedule/components/monthly-days-selector.tsx new file mode 100644 index 0000000000..d7cce79328 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/components/monthly-days-selector.tsx @@ -0,0 +1,90 @@ +import React from 'react' +import { useTranslation } from 'react-i18next' +import { RiQuestionLine } from '@remixicon/react' +import Tooltip from '@/app/components/base/tooltip' + +type MonthlyDaysSelectorProps = { + selectedDays: (number | 'last')[] + onChange: (days: (number | 'last')[]) => void +} + +const MonthlyDaysSelector = ({ selectedDays, onChange }: MonthlyDaysSelectorProps) => { + const { t } = useTranslation() + + const handleDayClick = (day: number | 'last') => { + const current = selectedDays || [] + const newSelected = current.includes(day) + ? current.filter(d => d !== day) + : [...current, day] + // Ensure at least one day is selected (consistent with WeekdaySelector) + onChange(newSelected.length > 0 ? newSelected : [day]) + } + + const isDaySelected = (day: number | 'last') => selectedDays?.includes(day) || false + + const days = Array.from({ length: 31 }, (_, i) => i + 1) + const rows = [ + days.slice(0, 7), + days.slice(7, 14), + days.slice(14, 21), + days.slice(21, 28), + [29, 30, 31, 'last' as const], + ] + + return ( + <div className="space-y-2"> + <label className="mb-2 block text-xs font-medium text-text-tertiary"> + {t('workflow.nodes.triggerSchedule.days')} + </label> + + <div className="space-y-1.5"> + {rows.map((row, rowIndex) => ( + <div key={rowIndex} className="grid grid-cols-7 gap-1.5"> + {row.map(day => ( + <button + key={day} + type="button" + onClick={() => handleDayClick(day)} + className={`rounded-lg border bg-components-option-card-option-bg py-1 text-xs transition-colors ${ + day === 'last' ? 'col-span-2 min-w-0' : '' + } ${ + isDaySelected(day) + ? 'border-util-colors-blue-brand-blue-brand-600 text-text-secondary' + : 'border-divider-subtle text-text-tertiary hover:border-divider-regular hover:text-text-secondary' + }`} + > + {day === 'last' ? ( + <div className="flex items-center justify-center gap-1"> + <span>{t('workflow.nodes.triggerSchedule.lastDay')}</span> + <Tooltip + popupContent={t('workflow.nodes.triggerSchedule.lastDayTooltip')} + > + <RiQuestionLine className="h-3 w-3 text-text-quaternary" /> + </Tooltip> + </div> + ) : ( + day + )} + </button> + ))} + {/* Fill empty cells in the last row (Last day takes 2 cols, so need 1 less) */} + {rowIndex === rows.length - 1 && Array.from({ length: 7 - row.length - 1 }, (_, i) => ( + <div key={`empty-${i}`} className="invisible"></div> + ))} + </div> + ))} + </div> + + {/* Warning message for day 31 - aligned with grid */} + {selectedDays?.includes(31) && ( + <div className="mt-1.5 grid grid-cols-7 gap-1.5"> + <div className="col-span-7 text-xs text-gray-500"> + {t('workflow.nodes.triggerSchedule.lastDayTooltip')} + </div> + </div> + )} + </div> + ) +} + +export default MonthlyDaysSelector diff --git a/web/app/components/workflow/nodes/trigger-schedule/components/next-execution-times.tsx b/web/app/components/workflow/nodes/trigger-schedule/components/next-execution-times.tsx new file mode 100644 index 0000000000..02e85e2724 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/components/next-execution-times.tsx @@ -0,0 +1,42 @@ +import React from 'react' +import { useTranslation } from 'react-i18next' +import type { ScheduleTriggerNodeType } from '../types' +import { getFormattedExecutionTimes } from '../utils/execution-time-calculator' + +type NextExecutionTimesProps = { + data: ScheduleTriggerNodeType +} + +const NextExecutionTimes = ({ data }: NextExecutionTimesProps) => { + const { t } = useTranslation() + + if (!data.frequency) + return null + + const executionTimes = getFormattedExecutionTimes(data, 5) + + if (executionTimes.length === 0) + return null + + return ( + <div className="space-y-2"> + <label className="block text-xs font-medium text-gray-500"> + {t('workflow.nodes.triggerSchedule.nextExecutionTimes')} + </label> + <div className="flex min-h-[80px] flex-col rounded-xl bg-components-input-bg-normal py-2"> + {executionTimes.map((time, index) => ( + <div key={index} className="flex items-baseline text-xs"> + <span className="w-6 select-none text-right font-mono font-normal leading-[150%] tracking-wider text-text-quaternary"> + {String(index + 1).padStart(2, '0')} + </span> + <span className="pl-2 pr-3 font-mono font-normal leading-[150%] tracking-wider text-text-secondary"> + {time} + </span> + </div> + ))} + </div> + </div> + ) +} + +export default NextExecutionTimes diff --git a/web/app/components/workflow/nodes/trigger-schedule/components/on-minute-selector.tsx b/web/app/components/workflow/nodes/trigger-schedule/components/on-minute-selector.tsx new file mode 100644 index 0000000000..992a111d19 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/components/on-minute-selector.tsx @@ -0,0 +1,38 @@ +import React from 'react' +import { useTranslation } from 'react-i18next' +import Slider from '@/app/components/base/slider' + +type OnMinuteSelectorProps = { + value?: number + onChange: (value: number) => void +} + +const OnMinuteSelector = ({ value = 0, onChange }: OnMinuteSelectorProps) => { + const { t } = useTranslation() + + return ( + <div> + <label className="mb-2 block text-xs font-medium text-gray-500"> + {t('workflow.nodes.triggerSchedule.onMinute')} + </label> + <div className="relative flex h-8 items-center rounded-lg bg-components-input-bg-normal"> + <div className="flex h-full w-12 shrink-0 items-center justify-center text-[13px] text-components-input-text-filled"> + {value} + </div> + <div className="absolute left-12 top-0 h-full w-px bg-components-panel-bg"></div> + <div className="flex h-full grow items-center pl-4 pr-3"> + <Slider + className="w-full" + value={value} + min={0} + max={59} + step={1} + onChange={onChange} + /> + </div> + </div> + </div> + ) +} + +export default OnMinuteSelector diff --git a/web/app/components/workflow/nodes/trigger-schedule/components/weekday-selector.tsx b/web/app/components/workflow/nodes/trigger-schedule/components/weekday-selector.tsx new file mode 100644 index 0000000000..348fd53454 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/components/weekday-selector.tsx @@ -0,0 +1,57 @@ +import React from 'react' +import { useTranslation } from 'react-i18next' + +type WeekdaySelectorProps = { + selectedDays: string[] + onChange: (days: string[]) => void +} + +const WeekdaySelector = ({ selectedDays, onChange }: WeekdaySelectorProps) => { + const { t } = useTranslation() + + const weekdays = [ + { key: 'sun', label: 'Sun' }, + { key: 'mon', label: 'Mon' }, + { key: 'tue', label: 'Tue' }, + { key: 'wed', label: 'Wed' }, + { key: 'thu', label: 'Thu' }, + { key: 'fri', label: 'Fri' }, + { key: 'sat', label: 'Sat' }, + ] + + const handleDaySelect = (dayKey: string) => { + const current = selectedDays || [] + const newSelected = current.includes(dayKey) + ? current.filter(d => d !== dayKey) + : [...current, dayKey] + onChange(newSelected.length > 0 ? newSelected : [dayKey]) + } + + const isDaySelected = (dayKey: string) => selectedDays.includes(dayKey) + + return ( + <div className="space-y-2"> + <label className="mb-2 block text-xs font-medium text-text-tertiary"> + {t('workflow.nodes.triggerSchedule.weekdays')} + </label> + <div className="flex gap-1.5"> + {weekdays.map(day => ( + <button + key={day.key} + type="button" + className={`flex-1 rounded-lg border bg-components-option-card-option-bg py-1 text-xs transition-colors ${ + isDaySelected(day.key) + ? 'border-util-colors-blue-brand-blue-brand-600 text-text-secondary' + : 'border-divider-subtle text-text-tertiary hover:border-divider-regular hover:text-text-secondary' + }`} + onClick={() => handleDaySelect(day.key)} + > + {day.label} + </button> + ))} + </div> + </div> + ) +} + +export default WeekdaySelector diff --git a/web/app/components/workflow/nodes/trigger-schedule/constants.ts b/web/app/components/workflow/nodes/trigger-schedule/constants.ts new file mode 100644 index 0000000000..ab6b8842bf --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/constants.ts @@ -0,0 +1,19 @@ +import type { ScheduleTriggerNodeType } from './types' + +export const getDefaultScheduleConfig = (): Partial<ScheduleTriggerNodeType> => ({ + mode: 'visual', + frequency: 'daily', + visual_config: { + time: '12:00 AM', + weekdays: ['sun'], + on_minute: 0, + monthly_days: [1], + }, +}) + +export const getDefaultVisualConfig = () => ({ + time: '12:00 AM', + weekdays: ['sun'], + on_minute: 0, + monthly_days: [1], +}) diff --git a/web/app/components/workflow/nodes/trigger-schedule/default.ts b/web/app/components/workflow/nodes/trigger-schedule/default.ts new file mode 100644 index 0000000000..69f93c33f4 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/default.ts @@ -0,0 +1,167 @@ +import { BlockEnum } from '../../types' +import type { NodeDefault } from '../../types' +import type { ScheduleTriggerNodeType } from './types' +import { isValidCronExpression } from './utils/cron-parser' +import { getNextExecutionTimes } from './utils/execution-time-calculator' +import { getDefaultScheduleConfig } from './constants' +import { genNodeMetaData } from '../../utils' + +const isValidTimeFormat = (time: string): boolean => { + const timeRegex = /^(0?\d|1[0-2]):[0-5]\d (AM|PM)$/ + if (!timeRegex.test(time)) return false + + const [timePart, period] = time.split(' ') + const [hour, minute] = timePart.split(':') + const hourNum = Number.parseInt(hour, 10) + const minuteNum = Number.parseInt(minute, 10) + + return hourNum >= 1 && hourNum <= 12 + && minuteNum >= 0 && minuteNum <= 59 + && ['AM', 'PM'].includes(period) +} + +const validateHourlyConfig = (config: any, t: any): string => { + if (config.on_minute === undefined || config.on_minute < 0 || config.on_minute > 59) + return t('workflow.nodes.triggerSchedule.invalidOnMinute') + + return '' +} + +const validateDailyConfig = (config: any, t: any): string => { + const i18nPrefix = 'workflow.errorMsg' + + if (!config.time) + return t(`${i18nPrefix}.fieldRequired`, { field: t('workflow.nodes.triggerSchedule.time') }) + + if (!isValidTimeFormat(config.time)) + return t('workflow.nodes.triggerSchedule.invalidTimeFormat') + + return '' +} + +const validateWeeklyConfig = (config: any, t: any): string => { + const dailyError = validateDailyConfig(config, t) + if (dailyError) return dailyError + + const i18nPrefix = 'workflow.errorMsg' + + if (!config.weekdays || config.weekdays.length === 0) + return t(`${i18nPrefix}.fieldRequired`, { field: t('workflow.nodes.triggerSchedule.weekdays') }) + + const validWeekdays = ['sun', 'mon', 'tue', 'wed', 'thu', 'fri', 'sat'] + for (const day of config.weekdays) { + if (!validWeekdays.includes(day)) + return t('workflow.nodes.triggerSchedule.invalidWeekday', { weekday: day }) + } + + return '' +} + +const validateMonthlyConfig = (config: any, t: any): string => { + const dailyError = validateDailyConfig(config, t) + if (dailyError) return dailyError + + const i18nPrefix = 'workflow.errorMsg' + + const getMonthlyDays = (): (number | 'last')[] => { + if (Array.isArray(config.monthly_days) && config.monthly_days.length > 0) + return config.monthly_days + + return [] + } + + const monthlyDays = getMonthlyDays() + + if (monthlyDays.length === 0) + return t(`${i18nPrefix}.fieldRequired`, { field: t('workflow.nodes.triggerSchedule.monthlyDay') }) + + for (const day of monthlyDays) { + if (day !== 'last' && (typeof day !== 'number' || day < 1 || day > 31)) + return t('workflow.nodes.triggerSchedule.invalidMonthlyDay') + } + + return '' +} + +const validateVisualConfig = (payload: ScheduleTriggerNodeType, t: any): string => { + const i18nPrefix = 'workflow.errorMsg' + const { visual_config } = payload + + if (!visual_config) + return t(`${i18nPrefix}.fieldRequired`, { field: t('workflow.nodes.triggerSchedule.visualConfig') }) + + switch (payload.frequency) { + case 'hourly': + return validateHourlyConfig(visual_config, t) + case 'daily': + return validateDailyConfig(visual_config, t) + case 'weekly': + return validateWeeklyConfig(visual_config, t) + case 'monthly': + return validateMonthlyConfig(visual_config, t) + default: + return t('workflow.nodes.triggerSchedule.invalidFrequency') + } +} + +const metaData = genNodeMetaData({ + sort: 2, + type: BlockEnum.TriggerSchedule, + helpLinkUri: 'schedule-trigger', + isStart: true, +}) + +const nodeDefault: NodeDefault<ScheduleTriggerNodeType> = { + metaData, + defaultValue: { + ...getDefaultScheduleConfig(), + cron_expression: '', + } as ScheduleTriggerNodeType, + checkValid(payload: ScheduleTriggerNodeType, t: any) { + const i18nPrefix = 'workflow.errorMsg' + let errorMessages = '' + if (!errorMessages && !payload.mode) + errorMessages = t(`${i18nPrefix}.fieldRequired`, { field: t('workflow.nodes.triggerSchedule.mode') }) + + // Validate timezone format if provided (timezone will be auto-filled by use-config.ts if undefined) + if (!errorMessages && payload.timezone) { + try { + Intl.DateTimeFormat(undefined, { timeZone: payload.timezone }) + } + catch { + errorMessages = t('workflow.nodes.triggerSchedule.invalidTimezone') + } + } + if (!errorMessages) { + if (payload.mode === 'cron') { + if (!payload.cron_expression || payload.cron_expression.trim() === '') + errorMessages = t(`${i18nPrefix}.fieldRequired`, { field: t('workflow.nodes.triggerSchedule.cronExpression') }) + else if (!isValidCronExpression(payload.cron_expression)) + errorMessages = t('workflow.nodes.triggerSchedule.invalidCronExpression') + } + else if (payload.mode === 'visual') { + if (!payload.frequency) + errorMessages = t(`${i18nPrefix}.fieldRequired`, { field: t('workflow.nodes.triggerSchedule.frequency') }) + else + errorMessages = validateVisualConfig(payload, t) + } + } + if (!errorMessages) { + try { + const nextTimes = getNextExecutionTimes(payload, 1) + if (nextTimes.length === 0) + errorMessages = t('workflow.nodes.triggerSchedule.noValidExecutionTime') + } + catch { + errorMessages = t('workflow.nodes.triggerSchedule.executionTimeCalculationError') + } + } + + return { + isValid: !errorMessages, + errorMessage: errorMessages, + } + }, +} + +export default nodeDefault diff --git a/web/app/components/workflow/nodes/trigger-schedule/node.tsx b/web/app/components/workflow/nodes/trigger-schedule/node.tsx new file mode 100644 index 0000000000..9870ef211a --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/node.tsx @@ -0,0 +1,31 @@ +import type { FC } from 'react' +import React from 'react' +import { useTranslation } from 'react-i18next' +import type { ScheduleTriggerNodeType } from './types' +import type { NodeProps } from '@/app/components/workflow/types' +import { getNextExecutionTime } from './utils/execution-time-calculator' + +const i18nPrefix = 'workflow.nodes.triggerSchedule' + +const Node: FC<NodeProps<ScheduleTriggerNodeType>> = ({ + data, +}) => { + const { t } = useTranslation() + + return ( + <div className="mb-1 px-3 py-1"> + <div className="mb-1 text-[10px] font-medium uppercase tracking-wide text-text-tertiary"> + {t(`${i18nPrefix}.nextExecutionTime`)} + </div> + <div className="flex h-[26px] items-center rounded-md bg-workflow-block-parma-bg px-2 text-xs text-text-secondary"> + <div className="w-0 grow"> + <div className="truncate" title={getNextExecutionTime(data)}> + {getNextExecutionTime(data)} + </div> + </div> + </div> + </div> + ) +} + +export default React.memo(Node) diff --git a/web/app/components/workflow/nodes/trigger-schedule/panel.tsx b/web/app/components/workflow/nodes/trigger-schedule/panel.tsx new file mode 100644 index 0000000000..2a7c661339 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/panel.tsx @@ -0,0 +1,146 @@ +import type { FC } from 'react' +import React from 'react' +import { useTranslation } from 'react-i18next' +import type { ScheduleTriggerNodeType } from './types' +import Field from '@/app/components/workflow/nodes/_base/components/field' +import type { NodePanelProps } from '@/app/components/workflow/types' +import ModeToggle from './components/mode-toggle' +import FrequencySelector from './components/frequency-selector' +import WeekdaySelector from './components/weekday-selector' +import TimePicker from '@/app/components/base/date-and-time-picker/time-picker' +import NextExecutionTimes from './components/next-execution-times' +import MonthlyDaysSelector from './components/monthly-days-selector' +import OnMinuteSelector from './components/on-minute-selector' +import Input from '@/app/components/base/input' +import useConfig from './use-config' + +const i18nPrefix = 'workflow.nodes.triggerSchedule' + +const Panel: FC<NodePanelProps<ScheduleTriggerNodeType>> = ({ + id, + data, +}) => { + const { t } = useTranslation() + const { + inputs, + setInputs, + handleModeChange, + handleFrequencyChange, + handleCronExpressionChange, + handleWeekdaysChange, + handleTimeChange, + handleOnMinuteChange, + } = useConfig(id, data) + + return ( + <div className='mt-2'> + <div className='space-y-4 px-4 pb-3 pt-2'> + <Field + title={t(`${i18nPrefix}.title`)} + operations={ + <ModeToggle + mode={inputs.mode} + onChange={handleModeChange} + /> + } + > + <div className="space-y-3"> + + {inputs.mode === 'visual' && ( + <div className="space-y-3"> + <div className="grid grid-cols-3 gap-3"> + <div> + <label className="mb-2 block text-xs font-medium text-gray-500"> + {t('workflow.nodes.triggerSchedule.frequencyLabel')} + </label> + <FrequencySelector + frequency={inputs.frequency || 'daily'} + onChange={handleFrequencyChange} + /> + </div> + <div className="col-span-2"> + {inputs.frequency === 'hourly' ? ( + <OnMinuteSelector + value={inputs.visual_config?.on_minute} + onChange={handleOnMinuteChange} + /> + ) : ( + <> + <label className="mb-2 block text-xs font-medium text-gray-500"> + {t('workflow.nodes.triggerSchedule.time')} + </label> + <TimePicker + notClearable={true} + timezone={inputs.timezone} + value={inputs.visual_config?.time || '12:00 AM'} + triggerFullWidth={true} + onChange={(time) => { + if (time) { + const timeString = time.format('h:mm A') + handleTimeChange(timeString) + } + }} + onClear={() => { + handleTimeChange('12:00 AM') + }} + placeholder={t('workflow.nodes.triggerSchedule.selectTime')} + showTimezone={true} + /> + </> + )} + </div> + </div> + + {inputs.frequency === 'weekly' && ( + <WeekdaySelector + selectedDays={inputs.visual_config?.weekdays || []} + onChange={handleWeekdaysChange} + /> + )} + + {inputs.frequency === 'monthly' && ( + <MonthlyDaysSelector + selectedDays={inputs.visual_config?.monthly_days || [1]} + onChange={(days) => { + const newInputs = { + ...inputs, + visual_config: { + ...inputs.visual_config, + monthly_days: days, + }, + } + setInputs(newInputs) + }} + /> + )} + </div> + )} + + {inputs.mode === 'cron' && ( + <div className="space-y-2"> + <div> + <label className="mb-2 block text-xs font-medium text-gray-500"> + {t('workflow.nodes.triggerSchedule.cronExpression')} + </label> + <Input + value={inputs.cron_expression || ''} + onChange={e => handleCronExpressionChange(e.target.value)} + placeholder="0 0 * * *" + className="font-mono" + /> + </div> + </div> + )} + </div> + </Field> + + <div className="border-t border-divider-subtle"></div> + + <NextExecutionTimes data={inputs} /> + + </div> + </div> + ) +} + +export default React.memo(Panel) diff --git a/web/app/components/workflow/nodes/trigger-schedule/types.ts b/web/app/components/workflow/nodes/trigger-schedule/types.ts new file mode 100644 index 0000000000..3d82709199 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/types.ts @@ -0,0 +1,20 @@ +import type { CommonNodeType } from '@/app/components/workflow/types' + +export type ScheduleMode = 'visual' | 'cron' + +export type ScheduleFrequency = 'hourly' | 'daily' | 'weekly' | 'monthly' + +export type VisualConfig = { + time?: string + weekdays?: string[] + on_minute?: number + monthly_days?: (number | 'last')[] +} + +export type ScheduleTriggerNodeType = CommonNodeType & { + mode: ScheduleMode + frequency?: ScheduleFrequency + cron_expression?: string + visual_config?: VisualConfig + timezone?: string +} diff --git a/web/app/components/workflow/nodes/trigger-schedule/use-config.ts b/web/app/components/workflow/nodes/trigger-schedule/use-config.ts new file mode 100644 index 0000000000..06e29ccd84 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/use-config.ts @@ -0,0 +1,110 @@ +import { useCallback, useMemo } from 'react' +import type { ScheduleFrequency, ScheduleMode, ScheduleTriggerNodeType } from './types' +import useNodeCrud from '@/app/components/workflow/nodes/_base/hooks/use-node-crud' +import { useNodesReadOnly } from '@/app/components/workflow/hooks' +import { useAppContext } from '@/context/app-context' +import { getDefaultVisualConfig } from './constants' + +const useConfig = (id: string, payload: ScheduleTriggerNodeType) => { + const { nodesReadOnly: readOnly } = useNodesReadOnly() + + const { userProfile } = useAppContext() + + const frontendPayload = useMemo(() => { + return { + ...payload, + mode: payload.mode || 'visual', + frequency: payload.frequency || 'daily', + timezone: payload.timezone || userProfile.timezone || 'UTC', + visual_config: { + ...getDefaultVisualConfig(), + ...payload.visual_config, + }, + } + }, [payload, userProfile.timezone]) + + const { inputs, setInputs } = useNodeCrud<ScheduleTriggerNodeType>(id, frontendPayload) + + const handleModeChange = useCallback((mode: ScheduleMode) => { + const newInputs = { + ...inputs, + mode, + } + setInputs(newInputs) + }, [inputs, setInputs]) + + const handleFrequencyChange = useCallback((frequency: ScheduleFrequency) => { + const newInputs = { + ...inputs, + frequency, + visual_config: { + ...inputs.visual_config, + ...(frequency === 'hourly') && { + on_minute: inputs.visual_config?.on_minute ?? 0, + }, + }, + cron_expression: undefined, + } + setInputs(newInputs) + }, [inputs, setInputs]) + + const handleCronExpressionChange = useCallback((value: string) => { + const newInputs = { + ...inputs, + cron_expression: value, + frequency: undefined, + visual_config: undefined, + } + setInputs(newInputs) + }, [inputs, setInputs]) + + const handleWeekdaysChange = useCallback((weekdays: string[]) => { + const newInputs = { + ...inputs, + visual_config: { + ...inputs.visual_config, + weekdays, + }, + cron_expression: undefined, + } + setInputs(newInputs) + }, [inputs, setInputs]) + + const handleTimeChange = useCallback((time: string) => { + const newInputs = { + ...inputs, + visual_config: { + ...inputs.visual_config, + time, + }, + cron_expression: undefined, + } + setInputs(newInputs) + }, [inputs, setInputs]) + + const handleOnMinuteChange = useCallback((on_minute: number) => { + const newInputs = { + ...inputs, + visual_config: { + ...inputs.visual_config, + on_minute, + }, + cron_expression: undefined, + } + setInputs(newInputs) + }, [inputs, setInputs]) + + return { + readOnly, + inputs, + setInputs, + handleModeChange, + handleFrequencyChange, + handleCronExpressionChange, + handleWeekdaysChange, + handleTimeChange, + handleOnMinuteChange, + } +} + +export default useConfig diff --git a/web/app/components/workflow/nodes/trigger-schedule/utils/cron-parser.ts b/web/app/components/workflow/nodes/trigger-schedule/utils/cron-parser.ts new file mode 100644 index 0000000000..90f65db0aa --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/utils/cron-parser.ts @@ -0,0 +1,84 @@ +import { CronExpressionParser } from 'cron-parser' + +// Convert a UTC date from cron-parser to user timezone representation +// This ensures consistency with other execution time calculations +const convertToUserTimezoneRepresentation = (utcDate: Date, timezone: string): Date => { + // Get the time string in the target timezone + const userTimeStr = utcDate.toLocaleString('en-CA', { + timeZone: timezone, + hour12: false, + }) + const [dateStr, timeStr] = userTimeStr.split(', ') + const [year, month, day] = dateStr.split('-').map(Number) + const [hour, minute, second] = timeStr.split(':').map(Number) + + // Create a new Date object representing this time as "local" time + // This matches the behavior expected by the execution-time-calculator + return new Date(year, month - 1, day, hour, minute, second) +} + +/** + * Parse a cron expression and return the next 5 execution times + * + * @param cronExpression - Standard 5-field cron expression (minute hour day month dayOfWeek) + * @param timezone - IANA timezone identifier (e.g., 'UTC', 'America/New_York') + * @returns Array of Date objects representing the next 5 execution times + */ +export const parseCronExpression = (cronExpression: string, timezone: string = 'UTC'): Date[] => { + if (!cronExpression || cronExpression.trim() === '') + return [] + + const parts = cronExpression.trim().split(/\s+/) + + // Support both 5-field format and predefined expressions + if (parts.length !== 5 && !cronExpression.startsWith('@')) + return [] + + try { + // Parse the cron expression with timezone support + // Use the actual current time for cron-parser to handle properly + const interval = CronExpressionParser.parse(cronExpression, { + tz: timezone, + }) + + // Get the next 5 execution times using the take() method + const nextCronDates = interval.take(5) + + // Convert CronDate objects to Date objects and ensure they represent + // the time in user timezone (consistent with execution-time-calculator.ts) + return nextCronDates.map((cronDate) => { + const utcDate = cronDate.toDate() + return convertToUserTimezoneRepresentation(utcDate, timezone) + }) + } + catch { + // Return empty array if parsing fails + return [] + } +} + +/** + * Validate a cron expression format and syntax + * + * @param cronExpression - Standard 5-field cron expression to validate + * @returns boolean indicating if the cron expression is valid + */ +export const isValidCronExpression = (cronExpression: string): boolean => { + if (!cronExpression || cronExpression.trim() === '') + return false + + const parts = cronExpression.trim().split(/\s+/) + + // Support both 5-field format and predefined expressions + if (parts.length !== 5 && !cronExpression.startsWith('@')) + return false + + try { + // Use cron-parser to validate the expression + CronExpressionParser.parse(cronExpression) + return true + } + catch { + return false + } +} diff --git a/web/app/components/workflow/nodes/trigger-schedule/utils/execution-time-calculator.ts b/web/app/components/workflow/nodes/trigger-schedule/utils/execution-time-calculator.ts new file mode 100644 index 0000000000..aef122ba25 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/utils/execution-time-calculator.ts @@ -0,0 +1,295 @@ +import type { ScheduleTriggerNodeType } from '../types' +import { isValidCronExpression, parseCronExpression } from './cron-parser' +import { convertTimezoneToOffsetStr } from '@/app/components/base/date-and-time-picker/utils/dayjs' + +const DEFAULT_TIMEZONE = 'UTC' + +const resolveTimezone = (timezone?: string): string => { + if (timezone) + return timezone + + try { + return Intl.DateTimeFormat().resolvedOptions().timeZone || DEFAULT_TIMEZONE + } + catch { + return DEFAULT_TIMEZONE + } +} + +// Get current time completely in user timezone, no browser timezone involved +const getUserTimezoneCurrentTime = (timezone?: string): Date => { + const targetTimezone = resolveTimezone(timezone) + const now = new Date() + const userTimeStr = now.toLocaleString('en-CA', { + timeZone: targetTimezone, + hour12: false, + }) + const [dateStr, timeStr] = userTimeStr.split(', ') + const [year, month, day] = dateStr.split('-').map(Number) + const [hour, minute, second] = timeStr.split(':').map(Number) + return new Date(year, month - 1, day, hour, minute, second) +} + +// Format date that is already in user timezone, no timezone conversion +const formatUserTimezoneDate = (date: Date, timezone: string, includeWeekday: boolean = true, includeTimezone: boolean = true): string => { + const dateOptions: Intl.DateTimeFormatOptions = { + year: 'numeric', + month: 'long', + day: 'numeric', + } + + if (includeWeekday) + dateOptions.weekday = 'long' // Changed from 'short' to 'long' for full weekday name + + const timeOptions: Intl.DateTimeFormatOptions = { + hour: 'numeric', + minute: '2-digit', + hour12: true, + } + + const dateStr = date.toLocaleDateString('en-US', dateOptions) + const timeStr = date.toLocaleTimeString('en-US', timeOptions) + + if (includeTimezone) { + const timezoneOffset = convertTimezoneToOffsetStr(timezone) + return `${dateStr}, ${timeStr} (${timezoneOffset})` + } + + return `${dateStr}, ${timeStr}` +} + +// Helper function to get default datetime - consistent with base DatePicker +export const getDefaultDateTime = (): Date => { + const defaultDate = new Date(2024, 0, 2, 11, 30, 0, 0) + return defaultDate +} + +export const getNextExecutionTimes = (data: ScheduleTriggerNodeType, count: number = 5): Date[] => { + const timezone = resolveTimezone(data.timezone) + + if (data.mode === 'cron') { + if (!data.cron_expression || !isValidCronExpression(data.cron_expression)) + return [] + return parseCronExpression(data.cron_expression, timezone).slice(0, count) + } + + const times: Date[] = [] + const defaultTime = data.visual_config?.time || '12:00 AM' + + // Get "today" in user's timezone for display purposes + const now = new Date() + const userTodayStr = now.toLocaleDateString('en-CA', { timeZone: timezone }) + const [year, month, day] = userTodayStr.split('-').map(Number) + const userToday = new Date(year, month - 1, day, 0, 0, 0, 0) + + if (data.frequency === 'hourly') { + const onMinute = data.visual_config?.on_minute ?? 0 + + // Get current time completely in user timezone + const userCurrentTime = getUserTimezoneCurrentTime(timezone) + + let hour = userCurrentTime.getHours() + if (userCurrentTime.getMinutes() >= onMinute) + hour += 1 // Start from next hour if current minute has passed + + for (let i = 0; i < count; i++) { + const execution = new Date(userToday) + execution.setHours(hour + i, onMinute, 0, 0) + // Handle day overflow + if (hour + i >= 24) { + execution.setDate(userToday.getDate() + Math.floor((hour + i) / 24)) + execution.setHours((hour + i) % 24, onMinute, 0, 0) + } + times.push(execution) + } + } + else if (data.frequency === 'daily') { + const [time, period] = defaultTime.split(' ') + const [hour, minute] = time.split(':') + let displayHour = Number.parseInt(hour) + if (period === 'PM' && displayHour !== 12) displayHour += 12 + if (period === 'AM' && displayHour === 12) displayHour = 0 + + // Check if today's configured time has already passed + const todayExecution = new Date(userToday) + todayExecution.setHours(displayHour, Number.parseInt(minute), 0, 0) + + const userCurrentTime = getUserTimezoneCurrentTime(timezone) + + const startOffset = todayExecution <= userCurrentTime ? 1 : 0 + + for (let i = 0; i < count; i++) { + const execution = new Date(userToday) + execution.setDate(userToday.getDate() + startOffset + i) + execution.setHours(displayHour, Number.parseInt(minute), 0, 0) + times.push(execution) + } + } + else if (data.frequency === 'weekly') { + const selectedDays = data.visual_config?.weekdays || ['sun'] + const dayMap = { sun: 0, mon: 1, tue: 2, wed: 3, thu: 4, fri: 5, sat: 6 } + + const [time, period] = defaultTime.split(' ') + const [hour, minute] = time.split(':') + let displayHour = Number.parseInt(hour) + if (period === 'PM' && displayHour !== 12) displayHour += 12 + if (period === 'AM' && displayHour === 12) displayHour = 0 + + // Get current time completely in user timezone + const userCurrentTime = getUserTimezoneCurrentTime(timezone) + + let executionCount = 0 + let weekOffset = 0 + + while (executionCount < count) { + let hasValidDays = false + + for (const selectedDay of selectedDays) { + if (executionCount >= count) break + + const targetDay = dayMap[selectedDay as keyof typeof dayMap] + if (targetDay === undefined) continue + + hasValidDays = true + + const currentDayOfWeek = userToday.getDay() + const daysUntilTarget = (targetDay - currentDayOfWeek + 7) % 7 + + // Check if today's configured time has already passed + const todayAtTargetTime = new Date(userToday) + todayAtTargetTime.setHours(displayHour, Number.parseInt(minute), 0, 0) + + let adjustedDays = daysUntilTarget + if (daysUntilTarget === 0 && todayAtTargetTime <= userCurrentTime) + adjustedDays = 7 + + const execution = new Date(userToday) + execution.setDate(userToday.getDate() + adjustedDays + (weekOffset * 7)) + execution.setHours(displayHour, Number.parseInt(minute), 0, 0) + + // Only add if execution time is in the future + if (execution > userCurrentTime) { + times.push(execution) + executionCount++ + } + } + + if (!hasValidDays) break + weekOffset++ + } + + times.sort((a, b) => a.getTime() - b.getTime()) + } + else if (data.frequency === 'monthly') { + const getSelectedDays = (): (number | 'last')[] => { + if (data.visual_config?.monthly_days && data.visual_config.monthly_days.length > 0) + return data.visual_config.monthly_days + + return [1] + } + + const selectedDays = [...new Set(getSelectedDays())] + const [time, period] = defaultTime.split(' ') + const [hour, minute] = time.split(':') + let displayHour = Number.parseInt(hour) + if (period === 'PM' && displayHour !== 12) displayHour += 12 + if (period === 'AM' && displayHour === 12) displayHour = 0 + + // Get current time completely in user timezone + const userCurrentTime = getUserTimezoneCurrentTime(timezone) + + let executionCount = 0 + let monthOffset = 0 + + while (executionCount < count) { + const targetMonth = new Date(userToday.getFullYear(), userToday.getMonth() + monthOffset, 1) + const daysInMonth = new Date(targetMonth.getFullYear(), targetMonth.getMonth() + 1, 0).getDate() + + const monthlyExecutions: Date[] = [] + const processedDays = new Set<number>() + + for (const selectedDay of selectedDays) { + let targetDay: number + + if (selectedDay === 'last') { + targetDay = daysInMonth + } + else { + const dayNumber = selectedDay as number + if (dayNumber > daysInMonth) + continue + + targetDay = dayNumber + } + + if (processedDays.has(targetDay)) + continue + + processedDays.add(targetDay) + + const execution = new Date(targetMonth.getFullYear(), targetMonth.getMonth(), targetDay, displayHour, Number.parseInt(minute), 0, 0) + + // Only add if execution time is in the future + if (execution > userCurrentTime) + monthlyExecutions.push(execution) + } + + monthlyExecutions.sort((a, b) => a.getTime() - b.getTime()) + + for (const execution of monthlyExecutions) { + if (executionCount >= count) break + times.push(execution) + executionCount++ + } + + monthOffset++ + } + } + else { + for (let i = 0; i < count; i++) { + const execution = new Date(userToday) + execution.setDate(userToday.getDate() + i) + times.push(execution) + } + } + + return times +} + +export const formatExecutionTime = (date: Date, timezone: string | undefined, includeWeekday: boolean = true, includeTimezone: boolean = true): string => { + const resolvedTimezone = resolveTimezone(timezone) + return formatUserTimezoneDate(date, resolvedTimezone, includeWeekday, includeTimezone) +} + +export const getFormattedExecutionTimes = (data: ScheduleTriggerNodeType, count: number = 5): string[] => { + const timezone = resolveTimezone(data.timezone) + const times = getNextExecutionTimes(data, count) + + return times.map((date) => { + const includeWeekday = data.mode === 'visual' && data.frequency === 'weekly' + return formatExecutionTime(date, timezone, includeWeekday, true) // Panel shows timezone + }) +} + +export const getNextExecutionTime = (data: ScheduleTriggerNodeType): string => { + const timezone = resolveTimezone(data.timezone) + + // Return placeholder for cron mode with empty or invalid expression + if (data.mode === 'cron') { + if (!data.cron_expression || !isValidCronExpression(data.cron_expression)) + return '--' + } + + // Get Date objects (not formatted strings) + const times = getNextExecutionTimes(data, 1) + if (times.length === 0) { + const userCurrentTime = getUserTimezoneCurrentTime(timezone) + const fallbackDate = new Date(userCurrentTime.getFullYear(), userCurrentTime.getMonth(), userCurrentTime.getDate(), 12, 0, 0, 0) + const includeWeekday = data.mode === 'visual' && data.frequency === 'weekly' + return formatExecutionTime(fallbackDate, timezone, includeWeekday, false) // Node doesn't show timezone + } + + // Format the first execution time without timezone for node display + const includeWeekday = data.mode === 'visual' && data.frequency === 'weekly' + return formatExecutionTime(times[0], timezone, includeWeekday, false) // Node doesn't show timezone +} diff --git a/web/app/components/workflow/nodes/trigger-schedule/utils/integration.spec.ts b/web/app/components/workflow/nodes/trigger-schedule/utils/integration.spec.ts new file mode 100644 index 0000000000..1b7d374d33 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-schedule/utils/integration.spec.ts @@ -0,0 +1,349 @@ +import { isValidCronExpression, parseCronExpression } from './cron-parser' +import { getNextExecutionTime, getNextExecutionTimes } from './execution-time-calculator' +import type { ScheduleTriggerNodeType } from '../types' + +// Comprehensive integration tests for cron-parser and execution-time-calculator compatibility +describe('cron-parser + execution-time-calculator integration', () => { + beforeAll(() => { + jest.useFakeTimers() + jest.setSystemTime(new Date('2024-01-15T10:00:00Z')) + }) + + afterAll(() => { + jest.useRealTimers() + }) + + const createCronData = (overrides: Partial<ScheduleTriggerNodeType> = {}): ScheduleTriggerNodeType => ({ + id: 'test-cron', + type: 'schedule-trigger', + mode: 'cron', + frequency: 'daily', + timezone: 'UTC', + ...overrides, + }) + + describe('backward compatibility validation', () => { + it('maintains exact behavior for legacy cron expressions', () => { + const legacyExpressions = [ + '15 10 1 * *', // Monthly 1st at 10:15 + '0 0 * * 0', // Weekly Sunday midnight + '*/5 * * * *', // Every 5 minutes + '0 9-17 * * 1-5', // Business hours weekdays + '30 14 * * 1', // Monday 14:30 + '0 0 1,15 * *', // 1st and 15th midnight + ] + + legacyExpressions.forEach((expression) => { + // Test direct cron-parser usage + const directResult = parseCronExpression(expression, 'UTC') + expect(directResult).toHaveLength(5) + expect(isValidCronExpression(expression)).toBe(true) + + // Test through execution-time-calculator + const data = createCronData({ cron_expression: expression }) + const calculatorResult = getNextExecutionTimes(data, 5) + + expect(calculatorResult).toHaveLength(5) + + // Results should be identical + directResult.forEach((directDate, index) => { + const calcDate = calculatorResult[index] + expect(calcDate.getTime()).toBe(directDate.getTime()) + expect(calcDate.getHours()).toBe(directDate.getHours()) + expect(calcDate.getMinutes()).toBe(directDate.getMinutes()) + }) + }) + }) + + it('validates timezone handling consistency', () => { + const timezones = ['UTC', 'America/New_York', 'Asia/Tokyo', 'Europe/London'] + const expression = '0 12 * * *' // Daily noon + + timezones.forEach((timezone) => { + // Direct cron-parser call + const directResult = parseCronExpression(expression, timezone) + + // Through execution-time-calculator + const data = createCronData({ cron_expression: expression, timezone }) + const calculatorResult = getNextExecutionTimes(data, 5) + + expect(directResult).toHaveLength(5) + expect(calculatorResult).toHaveLength(5) + + // All results should show noon (12:00) in their respective timezone + directResult.forEach(date => expect(date.getHours()).toBe(12)) + calculatorResult.forEach(date => expect(date.getHours()).toBe(12)) + + // Cross-validation: results should be identical + directResult.forEach((directDate, index) => { + expect(calculatorResult[index].getTime()).toBe(directDate.getTime()) + }) + }) + }) + + it('error handling consistency', () => { + const invalidExpressions = [ + '', // Empty string + ' ', // Whitespace only + '60 10 1 * *', // Invalid minute + '15 25 1 * *', // Invalid hour + '15 10 32 * *', // Invalid day + '15 10 1 13 *', // Invalid month + '15 10 1', // Too few fields + '15 10 1 * * *', // Too many fields + 'invalid expression', // Completely invalid + ] + + invalidExpressions.forEach((expression) => { + // Direct cron-parser calls + expect(isValidCronExpression(expression)).toBe(false) + expect(parseCronExpression(expression, 'UTC')).toEqual([]) + + // Through execution-time-calculator + const data = createCronData({ cron_expression: expression }) + const result = getNextExecutionTimes(data, 5) + expect(result).toEqual([]) + + // getNextExecutionTime should return '--' for invalid cron + const timeString = getNextExecutionTime(data) + expect(timeString).toBe('--') + }) + }) + }) + + describe('enhanced features integration', () => { + it('month and day abbreviations work end-to-end', () => { + const enhancedExpressions = [ + { expr: '0 9 1 JAN *', month: 0, day: 1, hour: 9 }, // January 1st 9 AM + { expr: '0 15 * * MON', weekday: 1, hour: 15 }, // Monday 3 PM + { expr: '30 10 15 JUN,DEC *', month: [5, 11], day: 15, hour: 10, minute: 30 }, // Jun/Dec 15th + { expr: '0 12 * JAN-MAR *', month: [0, 1, 2], hour: 12 }, // Q1 noon + ] + + enhancedExpressions.forEach(({ expr, month, day, weekday, hour, minute = 0 }) => { + // Validate through both paths + expect(isValidCronExpression(expr)).toBe(true) + + const directResult = parseCronExpression(expr, 'UTC') + const data = createCronData({ cron_expression: expr }) + const calculatorResult = getNextExecutionTimes(data, 3) + + expect(directResult.length).toBeGreaterThan(0) + expect(calculatorResult.length).toBeGreaterThan(0) + + // Validate expected properties + const validateDate = (date: Date) => { + expect(date.getHours()).toBe(hour) + expect(date.getMinutes()).toBe(minute) + + if (month !== undefined) { + if (Array.isArray(month)) + expect(month).toContain(date.getMonth()) + else + expect(date.getMonth()).toBe(month) + } + + if (day !== undefined) + expect(date.getDate()).toBe(day) + + if (weekday !== undefined) + expect(date.getDay()).toBe(weekday) + } + + directResult.forEach(validateDate) + calculatorResult.forEach(validateDate) + }) + }) + + it('predefined expressions work through execution-time-calculator', () => { + const predefExpressions = [ + { expr: '@daily', hour: 0, minute: 0 }, + { expr: '@weekly', hour: 0, minute: 0, weekday: 0 }, // Sunday + { expr: '@monthly', hour: 0, minute: 0, day: 1 }, // 1st of month + { expr: '@yearly', hour: 0, minute: 0, month: 0, day: 1 }, // Jan 1st + ] + + predefExpressions.forEach(({ expr, hour, minute, weekday, day, month }) => { + expect(isValidCronExpression(expr)).toBe(true) + + const data = createCronData({ cron_expression: expr }) + const result = getNextExecutionTimes(data, 3) + + expect(result.length).toBeGreaterThan(0) + + result.forEach((date) => { + expect(date.getHours()).toBe(hour) + expect(date.getMinutes()).toBe(minute) + + if (weekday !== undefined) expect(date.getDay()).toBe(weekday) + if (day !== undefined) expect(date.getDate()).toBe(day) + if (month !== undefined) expect(date.getMonth()).toBe(month) + }) + }) + }) + + it('special characters integration', () => { + const specialExpressions = [ + '0 9 ? * 1', // ? wildcard for day + '0 12 * * 7', // Sunday as 7 + '0 15 L * *', // Last day of month + ] + + specialExpressions.forEach((expr) => { + // Should validate and parse successfully + expect(isValidCronExpression(expr)).toBe(true) + + const directResult = parseCronExpression(expr, 'UTC') + const data = createCronData({ cron_expression: expr }) + const calculatorResult = getNextExecutionTimes(data, 2) + + expect(directResult.length).toBeGreaterThan(0) + expect(calculatorResult.length).toBeGreaterThan(0) + + // Results should be consistent + expect(calculatorResult[0].getHours()).toBe(directResult[0].getHours()) + expect(calculatorResult[0].getMinutes()).toBe(directResult[0].getMinutes()) + }) + }) + }) + + describe('DST and timezone edge cases', () => { + it('handles DST transitions consistently', () => { + // Test around DST spring forward (March 2024) + jest.setSystemTime(new Date('2024-03-08T10:00:00Z')) + + const expression = '0 2 * * *' // 2 AM daily (problematic during DST) + const timezone = 'America/New_York' + + const directResult = parseCronExpression(expression, timezone) + const data = createCronData({ cron_expression: expression, timezone }) + const calculatorResult = getNextExecutionTimes(data, 5) + + expect(directResult.length).toBeGreaterThan(0) + expect(calculatorResult.length).toBeGreaterThan(0) + + // Both should handle DST gracefully + // During DST spring forward, 2 AM becomes 3 AM - this is correct behavior + directResult.forEach(date => expect([2, 3]).toContain(date.getHours())) + calculatorResult.forEach(date => expect([2, 3]).toContain(date.getHours())) + + // Results should be identical + directResult.forEach((directDate, index) => { + expect(calculatorResult[index].getTime()).toBe(directDate.getTime()) + }) + }) + + it('complex timezone scenarios', () => { + const scenarios = [ + { tz: 'Asia/Kolkata', expr: '30 14 * * *', expectedHour: 14, expectedMinute: 30 }, // UTC+5:30 + { tz: 'Australia/Adelaide', expr: '0 8 * * *', expectedHour: 8, expectedMinute: 0 }, // UTC+9:30/+10:30 + { tz: 'Pacific/Kiritimati', expr: '0 12 * * *', expectedHour: 12, expectedMinute: 0 }, // UTC+14 + ] + + scenarios.forEach(({ tz, expr, expectedHour, expectedMinute }) => { + const directResult = parseCronExpression(expr, tz) + const data = createCronData({ cron_expression: expr, timezone: tz }) + const calculatorResult = getNextExecutionTimes(data, 2) + + expect(directResult.length).toBeGreaterThan(0) + expect(calculatorResult.length).toBeGreaterThan(0) + + // Validate expected time + directResult.forEach((date) => { + expect(date.getHours()).toBe(expectedHour) + expect(date.getMinutes()).toBe(expectedMinute) + }) + + calculatorResult.forEach((date) => { + expect(date.getHours()).toBe(expectedHour) + expect(date.getMinutes()).toBe(expectedMinute) + }) + + // Cross-validate consistency + expect(calculatorResult[0].getTime()).toBe(directResult[0].getTime()) + }) + }) + }) + + describe('performance and reliability', () => { + it('handles high-frequency expressions efficiently', () => { + const highFreqExpressions = [ + '*/1 * * * *', // Every minute + '*/5 * * * *', // Every 5 minutes + '0,15,30,45 * * * *', // Every 15 minutes + ] + + highFreqExpressions.forEach((expr) => { + const start = performance.now() + + // Test both direct and through calculator + const directResult = parseCronExpression(expr, 'UTC') + const data = createCronData({ cron_expression: expr }) + const calculatorResult = getNextExecutionTimes(data, 5) + + const end = performance.now() + + expect(directResult).toHaveLength(5) + expect(calculatorResult).toHaveLength(5) + expect(end - start).toBeLessThan(100) // Should be fast + + // Results should be consistent + directResult.forEach((directDate, index) => { + expect(calculatorResult[index].getTime()).toBe(directDate.getTime()) + }) + }) + }) + + it('stress test with complex expressions', () => { + const complexExpressions = [ + '15,45 8-18 1,15 JAN-MAR MON-FRI', // Business hours, specific days, Q1, weekdays + '0 */2 ? * SUN#1,SUN#3', // First and third Sunday, every 2 hours + '30 9 L * *', // Last day of month, 9:30 AM + ] + + complexExpressions.forEach((expr) => { + if (isValidCronExpression(expr)) { + const directResult = parseCronExpression(expr, 'America/New_York') + const data = createCronData({ + cron_expression: expr, + timezone: 'America/New_York', + }) + const calculatorResult = getNextExecutionTimes(data, 3) + + expect(directResult.length).toBeGreaterThan(0) + expect(calculatorResult.length).toBeGreaterThan(0) + + // Validate consistency where results exist + const minLength = Math.min(directResult.length, calculatorResult.length) + for (let i = 0; i < minLength; i++) + expect(calculatorResult[i].getTime()).toBe(directResult[i].getTime()) + } + }) + }) + }) + + describe('format compatibility', () => { + it('getNextExecutionTime formatting consistency', () => { + const testCases = [ + { expr: '0 9 * * *', timezone: 'UTC' }, + { expr: '30 14 * * 1-5', timezone: 'America/New_York' }, + { expr: '@daily', timezone: 'Asia/Tokyo' }, + ] + + testCases.forEach(({ expr, timezone }) => { + const data = createCronData({ cron_expression: expr, timezone }) + const timeString = getNextExecutionTime(data) + + // Should return a formatted time string, not '--' + expect(timeString).not.toBe('--') + expect(typeof timeString).toBe('string') + expect(timeString.length).toBeGreaterThan(0) + + // Should contain expected format elements + expect(timeString).toMatch(/\d+:\d+/) // Time format + expect(timeString).toMatch(/AM|PM/) // 12-hour format + expect(timeString).toMatch(/\d{4}/) // Year + }) + }) + }) +}) diff --git a/web/app/components/workflow/nodes/trigger-webhook/components/generic-table.tsx b/web/app/components/workflow/nodes/trigger-webhook/components/generic-table.tsx new file mode 100644 index 0000000000..235593d7f3 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/components/generic-table.tsx @@ -0,0 +1,297 @@ +'use client' +import type { FC, ReactNode } from 'react' +import React, { useCallback, useMemo } from 'react' +import { RiDeleteBinLine } from '@remixicon/react' +import Input from '@/app/components/base/input' +import Checkbox from '@/app/components/base/checkbox' +import { SimpleSelect } from '@/app/components/base/select' +import { replaceSpaceWithUnderscoreInVarNameInput } from '@/utils/var' +import cn from '@/utils/classnames' + +// Tiny utility to judge whether a cell value is effectively present +const isPresent = (v: unknown): boolean => { + if (typeof v === 'string') return v.trim() !== '' + return !(v === '' || v === null || v === undefined || v === false) +} +// Column configuration types for table components +export type ColumnType = 'input' | 'select' | 'switch' | 'custom' + +export type SelectOption = { + name: string + value: string +} + +export type ColumnConfig = { + key: string + title: string + type: ColumnType + width?: string // CSS class for width (e.g., 'w-1/2', 'w-[140px]') + placeholder?: string + options?: SelectOption[] // For select type + render?: (value: unknown, row: GenericTableRow, index: number, onChange: (value: unknown) => void) => ReactNode + required?: boolean +} + +export type GenericTableRow = { + [key: string]: unknown +} + +type GenericTableProps = { + title: string + columns: ColumnConfig[] + data: GenericTableRow[] + onChange: (data: GenericTableRow[]) => void + readonly?: boolean + placeholder?: string + emptyRowData: GenericTableRow // Template for new empty rows + className?: string + showHeader?: boolean // Whether to show column headers +} + +// Internal type for stable mapping between rendered rows and data indices +type DisplayRow = { + row: GenericTableRow + dataIndex: number | null // null indicates the trailing UI-only row + isVirtual: boolean // whether this row is the extra empty row for adding new items +} + +const GenericTable: FC<GenericTableProps> = ({ + title, + columns, + data, + onChange, + readonly = false, + placeholder, + emptyRowData, + className, + showHeader = false, +}) => { + // Build the rows to display while keeping a stable mapping to original data + const displayRows = useMemo<DisplayRow[]>(() => { + // Helper to check empty + const isEmptyRow = (r: GenericTableRow) => + Object.values(r).every(v => v === '' || v === null || v === undefined || v === false) + + if (readonly) + return data.map((r, i) => ({ row: r, dataIndex: i, isVirtual: false })) + + const hasData = data.length > 0 + const rows: DisplayRow[] = [] + + if (!hasData) { + // Initialize with exactly one empty row when there is no data + rows.push({ row: { ...emptyRowData }, dataIndex: null, isVirtual: true }) + return rows + } + + // Add configured rows, hide intermediate empty ones, keep mapping + data.forEach((r, i) => { + const isEmpty = isEmptyRow(r) + // Skip empty rows except the very last configured row + if (isEmpty && i < data.length - 1) + return + rows.push({ row: r, dataIndex: i, isVirtual: false }) + }) + + // If the last configured row has content, append a trailing empty row + const lastHasContent = !isEmptyRow(data[data.length - 1]) + if (lastHasContent) + rows.push({ row: { ...emptyRowData }, dataIndex: null, isVirtual: true }) + + return rows + }, [data, emptyRowData, readonly]) + + const removeRow = useCallback((dataIndex: number) => { + if (readonly) return + if (dataIndex < 0 || dataIndex >= data.length) return // ignore virtual rows + const newData = data.filter((_, i) => i !== dataIndex) + onChange(newData) + }, [data, readonly, onChange]) + + const updateRow = useCallback((dataIndex: number | null, key: string, value: unknown) => { + if (readonly) return + + if (dataIndex !== null && dataIndex < data.length) { + // Editing existing configured row + const newData = [...data] + newData[dataIndex] = { ...newData[dataIndex], [key]: value } + onChange(newData) + return + } + + // Editing the trailing UI-only empty row: create a new configured row + const newRow = { ...emptyRowData, [key]: value } + const next = [...data, newRow] + onChange(next) + }, [data, emptyRowData, onChange, readonly]) + + // Determine the primary identifier column just once + const primaryKey = useMemo(() => ( + columns.find(col => col.key === 'key' || col.key === 'name')?.key ?? 'key' + ), [columns]) + + const renderCell = (column: ColumnConfig, row: GenericTableRow, dataIndex: number | null) => { + const value = row[column.key] + const handleChange = (newValue: unknown) => updateRow(dataIndex, column.key, newValue) + + switch (column.type) { + case 'input': + return ( + <Input + value={(value as string) || ''} + onChange={(e) => { + // Format variable names (replace spaces with underscores) + if (column.key === 'key' || column.key === 'name') + replaceSpaceWithUnderscoreInVarNameInput(e.target) + handleChange(e.target.value) + }} + onKeyDown={(e) => { + if (e.key === 'Enter') { + e.preventDefault() + e.currentTarget.blur() + } + }} + placeholder={column.placeholder} + disabled={readonly} + wrapperClassName="w-full min-w-0" + className={cn( + // Ghost/inline style: looks like plain text until focus/hover + 'h-6 rounded-none border-0 bg-transparent px-0 py-0 shadow-none', + 'hover:border-transparent hover:bg-transparent focus:border-transparent focus:bg-transparent', + 'system-sm-regular text-text-secondary placeholder:text-text-quaternary', + )} + /> + ) + + case 'select': + return ( + <SimpleSelect + items={column.options || []} + defaultValue={value as string | undefined} + onSelect={item => handleChange(item.value)} + disabled={readonly} + placeholder={column.placeholder} + hideChecked={false} + notClearable={true} + // wrapper provides compact height, trigger is transparent like text + wrapperClassName="h-6 w-full min-w-0" + className={cn( + 'h-6 rounded-none bg-transparent pl-0 pr-6 text-text-secondary', + 'hover:bg-transparent focus-visible:bg-transparent group-hover/simple-select:bg-transparent', + )} + optionWrapClassName="w-26 min-w-26 z-[60] -ml-3" + /> + ) + + case 'switch': + return ( + <div className="flex h-7 items-center"> + <Checkbox + id={`${column.key}-${String(dataIndex ?? 'v')}`} + checked={Boolean(value)} + onCheck={() => handleChange(!value)} + disabled={readonly} + /> + </div> + ) + + case 'custom': + return column.render ? column.render(value, row, (dataIndex ?? -1), handleChange) : null + + default: + return null + } + } + + const renderTable = () => { + return ( + <div className="rounded-lg border border-divider-regular"> + {showHeader && ( + <div className="system-xs-medium-uppercase flex h-7 items-center leading-7 text-text-tertiary"> + {columns.map((column, index) => ( + <div + key={column.key} + className={cn( + 'h-full pl-3', + column.width && column.width.startsWith('w-') ? 'shrink-0' : 'flex-1', + column.width, + // Add right border except for last column + index < columns.length - 1 && 'border-r border-divider-regular', + )} + > + {column.title} + </div> + ))} + </div> + )} + <div className="divide-y divide-divider-subtle"> + {displayRows.map(({ row, dataIndex, isVirtual: _isVirtual }, renderIndex) => { + const rowKey = `row-${renderIndex}` + + // Check if primary identifier column has content + const primaryValue = row[primaryKey] + const hasContent = isPresent(primaryValue) + + return ( + <div + key={rowKey} + className={cn( + 'group relative flex border-t border-divider-regular', + hasContent ? 'hover:bg-state-destructive-hover' : 'hover:bg-state-base-hover', + )} + style={{ minHeight: '28px' }} + > + {columns.map((column, columnIndex) => ( + <div + key={column.key} + className={cn( + 'shrink-0 pl-3', + column.width, + // Add right border except for last column + columnIndex < columns.length - 1 && 'border-r border-divider-regular', + )} + > + {renderCell(column, row, dataIndex)} + </div> + ))} + {!readonly && dataIndex !== null && hasContent && ( + <div className="absolute right-2 top-1/2 -translate-y-1/2 opacity-0 group-hover:opacity-100"> + <button + type="button" + onClick={() => removeRow(dataIndex)} + className="p-1" + aria-label="Delete row" + > + <RiDeleteBinLine className="h-3.5 w-3.5 text-text-destructive" /> + </button> + </div> + )} + </div> + ) + })} + </div> + </div> + ) + } + + // Show placeholder only when readonly and there is no data configured + const showPlaceholder = readonly && data.length === 0 + + return ( + <div className={className}> + <div className="mb-3 flex items-center justify-between"> + <h4 className="system-sm-semibold-uppercase text-text-secondary">{title}</h4> + </div> + + {showPlaceholder ? ( + <div className="flex h-7 items-center justify-center rounded-lg border border-divider-regular bg-components-panel-bg text-xs font-normal leading-[18px] text-text-quaternary"> + {placeholder} + </div> + ) : ( + renderTable() + )} + </div> + ) +} + +export default React.memo(GenericTable) diff --git a/web/app/components/workflow/nodes/trigger-webhook/components/header-table.tsx b/web/app/components/workflow/nodes/trigger-webhook/components/header-table.tsx new file mode 100644 index 0000000000..25e3cd4137 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/components/header-table.tsx @@ -0,0 +1,78 @@ +'use client' +import type { FC } from 'react' +import React from 'react' +import { useTranslation } from 'react-i18next' +import GenericTable from './generic-table' +import type { ColumnConfig, GenericTableRow } from './generic-table' +import type { WebhookHeader } from '../types' + +type HeaderTableProps = { + readonly?: boolean + headers?: WebhookHeader[] + onChange: (headers: WebhookHeader[]) => void +} + +const HeaderTable: FC<HeaderTableProps> = ({ + readonly = false, + headers = [], + onChange, +}) => { + const { t } = useTranslation() + + // Define columns for header table - matching prototype design + const columns: ColumnConfig[] = [ + { + key: 'name', + title: t('workflow.nodes.triggerWebhook.varName'), + type: 'input', + width: 'flex-1', + placeholder: t('workflow.nodes.triggerWebhook.varNamePlaceholder'), + }, + { + key: 'required', + title: t('workflow.nodes.triggerWebhook.required'), + type: 'switch', + width: 'w-[88px]', + }, + ] + + // No default prefilled row; table initializes with one empty row + + // Empty row template for new rows + const emptyRowData: GenericTableRow = { + name: '', + required: false, + } + + // Convert WebhookHeader[] to GenericTableRow[] + const tableData: GenericTableRow[] = headers.map(header => ({ + name: header.name, + required: header.required, + })) + + // Handle data changes + const handleDataChange = (data: GenericTableRow[]) => { + const newHeaders: WebhookHeader[] = data + .filter(row => row.name && typeof row.name === 'string' && row.name.trim() !== '') + .map(row => ({ + name: (row.name as string) || '', + required: !!row.required, + })) + onChange(newHeaders) + } + + return ( + <GenericTable + title="Header Parameters" + columns={columns} + data={tableData} + onChange={handleDataChange} + readonly={readonly} + placeholder={t('workflow.nodes.triggerWebhook.noHeaders')} + emptyRowData={emptyRowData} + showHeader={true} + /> + ) +} + +export default React.memo(HeaderTable) diff --git a/web/app/components/workflow/nodes/trigger-webhook/components/paragraph-input.tsx b/web/app/components/workflow/nodes/trigger-webhook/components/paragraph-input.tsx new file mode 100644 index 0000000000..f3946f5d3d --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/components/paragraph-input.tsx @@ -0,0 +1,57 @@ +'use client' +import type { FC } from 'react' +import React, { useRef } from 'react' +import cn from '@/utils/classnames' + +type ParagraphInputProps = { + value: string + onChange: (value: string) => void + placeholder?: string + disabled?: boolean + className?: string +} + +const ParagraphInput: FC<ParagraphInputProps> = ({ + value, + onChange, + placeholder, + disabled = false, + className, +}) => { + const textareaRef = useRef<HTMLTextAreaElement>(null) + + const lines = value ? value.split('\n') : [''] + const lineCount = Math.max(3, lines.length) + + return ( + <div className={cn('rounded-xl bg-components-input-bg-normal px-3 pb-2 pt-3', className)}> + <div className="relative"> + <div className="pointer-events-none absolute left-0 top-0 flex flex-col"> + {Array.from({ length: lineCount }, (_, index) => ( + <span + key={index} + className="flex h-[20px] select-none items-center font-mono text-xs leading-[20px] text-text-quaternary" + > + {String(index + 1).padStart(2, '0')} + </span> + ))} + </div> + <textarea + ref={textareaRef} + value={value} + onChange={e => onChange(e.target.value)} + placeholder={placeholder} + disabled={disabled} + className="w-full resize-none border-0 bg-transparent pl-6 font-mono text-xs leading-[20px] text-text-secondary outline-none placeholder:text-text-quaternary" + style={{ + minHeight: `${Math.max(3, lineCount) * 20}px`, + lineHeight: '20px', + }} + rows={Math.max(3, lineCount)} + /> + </div> + </div> + ) +} + +export default React.memo(ParagraphInput) diff --git a/web/app/components/workflow/nodes/trigger-webhook/components/parameter-table.tsx b/web/app/components/workflow/nodes/trigger-webhook/components/parameter-table.tsx new file mode 100644 index 0000000000..bf030c4340 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/components/parameter-table.tsx @@ -0,0 +1,112 @@ +'use client' +import type { FC } from 'react' +import React, { useMemo } from 'react' +import { useTranslation } from 'react-i18next' +import GenericTable from './generic-table' +import type { ColumnConfig, GenericTableRow } from './generic-table' +import type { WebhookParameter } from '../types' +import { createParameterTypeOptions, normalizeParameterType } from '../utils/parameter-type-utils' +import { VarType } from '@/app/components/workflow/types' + +type ParameterTableProps = { + title: string + parameters: WebhookParameter[] + onChange: (params: WebhookParameter[]) => void + readonly?: boolean + placeholder?: string + contentType?: string +} + +const ParameterTable: FC<ParameterTableProps> = ({ + title, + parameters, + onChange, + readonly, + placeholder, + contentType, +}) => { + const { t } = useTranslation() + + // Memoize typeOptions to prevent unnecessary re-renders that cause SimpleSelect state resets + const typeOptions = useMemo(() => + createParameterTypeOptions(contentType), + [contentType], + ) + + // Define columns based on component type - matching prototype design + const columns: ColumnConfig[] = [ + { + key: 'key', + title: t('workflow.nodes.triggerWebhook.varName'), + type: 'input', + width: 'flex-1', + placeholder: t('workflow.nodes.triggerWebhook.varNamePlaceholder'), + }, + { + key: 'type', + title: t('workflow.nodes.triggerWebhook.varType'), + type: 'select', + width: 'w-[120px]', + placeholder: t('workflow.nodes.triggerWebhook.varType'), + options: typeOptions, + }, + { + key: 'required', + title: t('workflow.nodes.triggerWebhook.required'), + type: 'switch', + width: 'w-[88px]', + }, + ] + + // Choose sensible default type for new rows according to content type + const defaultTypeValue: VarType = typeOptions[0]?.value || 'string' + + // Empty row template for new rows + const emptyRowData: GenericTableRow = { + key: '', + type: defaultTypeValue, + required: false, + } + + const tableData: GenericTableRow[] = parameters.map(param => ({ + key: param.name, + type: param.type, + required: param.required, + })) + + const handleDataChange = (data: GenericTableRow[]) => { + // For text/plain, enforce single text body semantics: keep only first non-empty row and force string type + // For application/octet-stream, enforce single file body semantics: keep only first non-empty row and force file type + const isTextPlain = (contentType || '').toLowerCase() === 'text/plain' + const isOctetStream = (contentType || '').toLowerCase() === 'application/octet-stream' + + const normalized = data + .filter(row => typeof row.key === 'string' && (row.key as string).trim() !== '') + .map(row => ({ + name: String(row.key), + type: isTextPlain ? VarType.string : isOctetStream ? VarType.file : normalizeParameterType((row.type as string)), + required: Boolean(row.required), + })) + + const newParams: WebhookParameter[] = (isTextPlain || isOctetStream) + ? normalized.slice(0, 1) + : normalized + + onChange(newParams) + } + + return ( + <GenericTable + title={title} + columns={columns} + data={tableData} + onChange={handleDataChange} + readonly={readonly} + placeholder={placeholder || t('workflow.nodes.triggerWebhook.noParameters')} + emptyRowData={emptyRowData} + showHeader={true} + /> + ) +} + +export default ParameterTable diff --git a/web/app/components/workflow/nodes/trigger-webhook/default.ts b/web/app/components/workflow/nodes/trigger-webhook/default.ts new file mode 100644 index 0000000000..5071a79913 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/default.ts @@ -0,0 +1,64 @@ +import { BlockEnum } from '../../types' +import type { NodeDefault } from '../../types' +import { genNodeMetaData } from '../../utils' +import type { WebhookTriggerNodeType } from './types' +import { isValidParameterType } from './utils/parameter-type-utils' +import { createWebhookRawVariable } from './utils/raw-variable' + +const metaData = genNodeMetaData({ + sort: 3, + type: BlockEnum.TriggerWebhook, + helpLinkUri: 'webhook-trigger', + isStart: true, +}) + +const nodeDefault: NodeDefault<WebhookTriggerNodeType> = { + metaData, + defaultValue: { + webhook_url: '', + method: 'POST', + content_type: 'application/json', + headers: [], + params: [], + body: [], + async_mode: true, + status_code: 200, + response_body: '', + variables: [createWebhookRawVariable()], + }, + checkValid(payload: WebhookTriggerNodeType, t: any) { + // Require webhook_url to be configured + if (!payload.webhook_url || payload.webhook_url.trim() === '') { + return { + isValid: false, + errorMessage: t('workflow.nodes.triggerWebhook.validation.webhookUrlRequired'), + } + } + + // Validate parameter types for params and body + const parametersWithTypes = [ + ...(payload.params || []), + ...(payload.body || []), + ] + + for (const param of parametersWithTypes) { + // Validate parameter type is valid + if (!isValidParameterType(param.type)) { + return { + isValid: false, + errorMessage: t('workflow.nodes.triggerWebhook.validation.invalidParameterType', { + name: param.name, + type: param.type, + }), + } + } + } + + return { + isValid: true, + errorMessage: '', + } + }, +} + +export default nodeDefault diff --git a/web/app/components/workflow/nodes/trigger-webhook/node.tsx b/web/app/components/workflow/nodes/trigger-webhook/node.tsx new file mode 100644 index 0000000000..40c3b441da --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/node.tsx @@ -0,0 +1,25 @@ +import type { FC } from 'react' +import React from 'react' +import type { WebhookTriggerNodeType } from './types' +import type { NodeProps } from '@/app/components/workflow/types' + +const Node: FC<NodeProps<WebhookTriggerNodeType>> = ({ + data, +}) => { + return ( + <div className="mb-1 px-3 py-1"> + <div className="mb-1 text-[10px] font-medium uppercase tracking-wide text-text-tertiary"> + URL + </div> + <div className="flex h-[26px] items-center rounded-md bg-workflow-block-parma-bg px-2 text-xs text-text-secondary"> + <div className="w-0 grow"> + <div className="truncate" title={data.webhook_url || '--'}> + {data.webhook_url || '--'} + </div> + </div> + </div> + </div> + ) +} + +export default React.memo(Node) diff --git a/web/app/components/workflow/nodes/trigger-webhook/panel.tsx b/web/app/components/workflow/nodes/trigger-webhook/panel.tsx new file mode 100644 index 0000000000..1de18bd806 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/panel.tsx @@ -0,0 +1,240 @@ +import type { FC } from 'react' +import React, { useEffect, useRef, useState } from 'react' +import { useTranslation } from 'react-i18next' + +import type { HttpMethod, WebhookTriggerNodeType } from './types' +import useConfig from './use-config' +import ParameterTable from './components/parameter-table' +import HeaderTable from './components/header-table' +import ParagraphInput from './components/paragraph-input' +import { OutputVariablesContent } from './utils/render-output-vars' +import Field from '@/app/components/workflow/nodes/_base/components/field' +import Split from '@/app/components/workflow/nodes/_base/components/split' +import OutputVars from '@/app/components/workflow/nodes/_base/components/output-vars' +import type { NodePanelProps } from '@/app/components/workflow/types' +import InputWithCopy from '@/app/components/base/input-with-copy' +import { InputNumber } from '@/app/components/base/input-number' +import { SimpleSelect } from '@/app/components/base/select' +import Toast from '@/app/components/base/toast' +import Tooltip from '@/app/components/base/tooltip' +import copy from 'copy-to-clipboard' +import { isPrivateOrLocalAddress } from '@/utils/urlValidation' + +const i18nPrefix = 'workflow.nodes.triggerWebhook' + +const HTTP_METHODS = [ + { name: 'GET', value: 'GET' }, + { name: 'POST', value: 'POST' }, + { name: 'PUT', value: 'PUT' }, + { name: 'DELETE', value: 'DELETE' }, + { name: 'PATCH', value: 'PATCH' }, + { name: 'HEAD', value: 'HEAD' }, +] + +const CONTENT_TYPES = [ + { name: 'application/json', value: 'application/json' }, + { name: 'application/x-www-form-urlencoded', value: 'application/x-www-form-urlencoded' }, + { name: 'text/plain', value: 'text/plain' }, + { name: 'application/octet-stream', value: 'application/octet-stream' }, + { name: 'multipart/form-data', value: 'multipart/form-data' }, +] + +const Panel: FC<NodePanelProps<WebhookTriggerNodeType>> = ({ + id, + data, +}) => { + const { t } = useTranslation() + const [debugUrlCopied, setDebugUrlCopied] = React.useState(false) + const [outputVarsCollapsed, setOutputVarsCollapsed] = useState(false) + const { + readOnly, + inputs, + handleMethodChange, + handleContentTypeChange, + handleHeadersChange, + handleParamsChange, + handleBodyChange, + handleStatusCodeChange, + handleStatusCodeBlur, + handleResponseBodyChange, + generateWebhookUrl, + } = useConfig(id, data) + + // Ensure we only attempt to generate URL once for a newly created node without url + const hasRequestedUrlRef = useRef(false) + useEffect(() => { + if (!readOnly && !inputs.webhook_url && !hasRequestedUrlRef.current) { + hasRequestedUrlRef.current = true + void generateWebhookUrl() + } + }, [readOnly, inputs.webhook_url, generateWebhookUrl]) + + return ( + <div className='mt-2'> + <div className='space-y-4 px-4 pb-3 pt-2'> + {/* Webhook URL Section */} + <Field title={t(`${i18nPrefix}.webhookUrl`)}> + <div className="space-y-1"> + <div className="flex gap-1" style={{ height: '32px' }}> + <div className="w-26 shrink-0"> + <SimpleSelect + items={HTTP_METHODS} + defaultValue={inputs.method} + onSelect={item => handleMethodChange(item.value as HttpMethod)} + disabled={readOnly} + className="h-8 pr-8 text-sm" + wrapperClassName="h-8" + optionWrapClassName="w-26 min-w-26 z-[5]" + allowSearch={false} + notClearable={true} + /> + </div> + <div className="flex-1" style={{ width: '284px' }}> + <InputWithCopy + value={inputs.webhook_url || ''} + placeholder={t(`${i18nPrefix}.webhookUrlPlaceholder`)} + readOnly + onCopy={() => { + Toast.notify({ + type: 'success', + message: t(`${i18nPrefix}.urlCopied`), + }) + }} + /> + </div> + </div> + {inputs.webhook_debug_url && ( + <div className="space-y-2"> + <Tooltip + popupContent={debugUrlCopied ? t(`${i18nPrefix}.debugUrlCopied`) : t(`${i18nPrefix}.debugUrlCopy`)} + popupClassName="system-xs-regular text-text-primary bg-components-tooltip-bg border border-components-panel-border shadow-lg backdrop-blur-sm rounded-md px-1.5 py-1" + position="top" + offset={{ mainAxis: -20 }} + needsDelay={true} + > + <div + className="flex cursor-pointer gap-1.5 rounded-lg px-1 py-1.5 transition-colors" + style={{ width: '368px', height: '38px' }} + onClick={() => { + copy(inputs.webhook_debug_url || '') + setDebugUrlCopied(true) + setTimeout(() => setDebugUrlCopied(false), 2000) + }} + > + <div className="mt-0.5 w-0.5 bg-divider-regular" style={{ height: '28px' }}></div> + <div className="flex-1" style={{ width: '352px', height: '32px' }}> + <div className="text-xs leading-4 text-text-tertiary"> + {t(`${i18nPrefix}.debugUrlTitle`)} + </div> + <div className="truncate text-xs leading-4 text-text-primary"> + {inputs.webhook_debug_url} + </div> + </div> + </div> + </Tooltip> + {isPrivateOrLocalAddress(inputs.webhook_debug_url) && ( + <div className="system-xs-regular mt-1 px-0 py-[2px] text-text-warning"> + {t(`${i18nPrefix}.debugUrlPrivateAddressWarning`)} + </div> + )} + </div> + )} + </div> + </Field> + + {/* Content Type */} + <Field title={t(`${i18nPrefix}.contentType`)}> + <div className="w-full"> + <SimpleSelect + items={CONTENT_TYPES} + defaultValue={inputs.content_type} + onSelect={item => handleContentTypeChange(item.value as string)} + disabled={readOnly} + className="h-8 text-sm" + wrapperClassName="h-8" + optionWrapClassName="min-w-48 z-[5]" + allowSearch={false} + notClearable={true} + /> + </div> + </Field> + + {/* Query Parameters */} + <ParameterTable + readonly={readOnly} + title="Query Parameters" + parameters={inputs.params} + onChange={handleParamsChange} + placeholder={t(`${i18nPrefix}.noQueryParameters`)} + /> + + {/* Header Parameters */} + <HeaderTable + readonly={readOnly} + headers={inputs.headers} + onChange={handleHeadersChange} + /> + + {/* Request Body Parameters */} + <ParameterTable + readonly={readOnly} + title="Request Body Parameters" + parameters={inputs.body} + onChange={handleBodyChange} + placeholder={t(`${i18nPrefix}.noBodyParameters`)} + contentType={inputs.content_type} + /> + + <Split /> + + {/* Response Configuration */} + <Field title={t(`${i18nPrefix}.responseConfiguration`)}> + <div className="space-y-3"> + <div className="flex items-center justify-between"> + <label className="system-sm-medium text-text-tertiary"> + {t(`${i18nPrefix}.statusCode`)} + </label> + <InputNumber + value={inputs.status_code} + onChange={(value) => { + handleStatusCodeChange(value || 200) + }} + disabled={readOnly} + wrapClassName="w-[120px]" + className="h-8" + defaultValue={200} + onBlur={() => { + handleStatusCodeBlur(inputs.status_code) + }} + /> + </div> + <div> + <label className="system-sm-medium mb-2 block text-text-tertiary"> + {t(`${i18nPrefix}.responseBody`)} + </label> + <ParagraphInput + value={inputs.response_body} + onChange={handleResponseBodyChange} + placeholder={t(`${i18nPrefix}.responseBodyPlaceholder`)} + disabled={readOnly} + /> + </div> + </div> + </Field> + </div> + + <Split /> + + <div className=''> + <OutputVars + collapsed={outputVarsCollapsed} + onCollapse={setOutputVarsCollapsed} + > + <OutputVariablesContent variables={inputs.variables} /> + </OutputVars> + </div> + </div> + ) +} + +export default Panel diff --git a/web/app/components/workflow/nodes/trigger-webhook/types.ts b/web/app/components/workflow/nodes/trigger-webhook/types.ts new file mode 100644 index 0000000000..d9632f20e1 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/types.ts @@ -0,0 +1,35 @@ +import type { CommonNodeType, VarType, Variable } from '@/app/components/workflow/types' + +export type HttpMethod = 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH' | 'HEAD' + +export type ArrayElementType = 'string' | 'number' | 'boolean' | 'object' + +export const getArrayElementType = (arrayType: `array[${ArrayElementType}]`): ArrayElementType => { + const match = arrayType.match(/^array\[(.+)\]$/) + return (match?.[1] as ArrayElementType) || 'string' +} + +export type WebhookParameter = { + name: string + type: VarType + required: boolean +} + +export type WebhookHeader = { + name: string + required: boolean +} + +export type WebhookTriggerNodeType = CommonNodeType & { + webhook_url?: string + webhook_debug_url?: string + method: HttpMethod + content_type: string + headers: WebhookHeader[] + params: WebhookParameter[] + body: WebhookParameter[] + async_mode: boolean + status_code: number + response_body: string + variables: Variable[] +} diff --git a/web/app/components/workflow/nodes/trigger-webhook/use-config.ts b/web/app/components/workflow/nodes/trigger-webhook/use-config.ts new file mode 100644 index 0000000000..9b525ec758 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/use-config.ts @@ -0,0 +1,251 @@ +import { useCallback } from 'react' +import { produce } from 'immer' +import { useTranslation } from 'react-i18next' +import type { HttpMethod, WebhookHeader, WebhookParameter, WebhookTriggerNodeType } from './types' + +import { useNodesReadOnly, useWorkflow } from '@/app/components/workflow/hooks' +import useNodeCrud from '@/app/components/workflow/nodes/_base/hooks/use-node-crud' +import { useStore as useAppStore } from '@/app/components/app/store' +import { fetchWebhookUrl } from '@/service/apps' +import type { Variable } from '@/app/components/workflow/types' +import { VarType } from '@/app/components/workflow/types' +import Toast from '@/app/components/base/toast' +import { checkKeys, hasDuplicateStr } from '@/utils/var' +import { WEBHOOK_RAW_VARIABLE_NAME } from './utils/raw-variable' + +const useConfig = (id: string, payload: WebhookTriggerNodeType) => { + const { t } = useTranslation() + const { nodesReadOnly: readOnly } = useNodesReadOnly() + const { inputs, setInputs } = useNodeCrud<WebhookTriggerNodeType>(id, payload) + const appId = useAppStore.getState().appDetail?.id + const { isVarUsedInNodes, removeUsedVarInNodes } = useWorkflow() + + const handleMethodChange = useCallback((method: HttpMethod) => { + setInputs(produce(inputs, (draft) => { + draft.method = method + })) + }, [inputs, setInputs]) + + const handleContentTypeChange = useCallback((contentType: string) => { + setInputs(produce(inputs, (draft) => { + const previousContentType = draft.content_type + draft.content_type = contentType + + // If the content type changes, reset body parameters and their variables, as the variable types might differ. + // However, we could consider retaining variables that are compatible with the new content type later. + if (previousContentType !== contentType) { + draft.body = [] + if (draft.variables) { + const bodyVariables = draft.variables.filter(v => v.label === 'body') + bodyVariables.forEach((v) => { + if (isVarUsedInNodes([id, v.variable])) + removeUsedVarInNodes([id, v.variable]) + }) + + draft.variables = draft.variables.filter(v => v.label !== 'body') + } + } + })) + }, [inputs, setInputs, id, isVarUsedInNodes, removeUsedVarInNodes]) + + const syncVariablesInDraft = useCallback(( + draft: WebhookTriggerNodeType, + newData: (WebhookParameter | WebhookHeader)[], + sourceType: 'param' | 'header' | 'body', + ) => { + if (!draft.variables) + draft.variables = [] + + const sanitizedEntries = newData.map(item => ({ + item, + sanitizedName: sourceType === 'header' ? item.name.replace(/-/g, '_') : item.name, + })) + + const hasReservedConflict = sanitizedEntries.some(entry => entry.sanitizedName === WEBHOOK_RAW_VARIABLE_NAME) + if (hasReservedConflict) { + Toast.notify({ + type: 'error', + message: t('appDebug.varKeyError.keyAlreadyExists', { + key: t('appDebug.variableConfig.varName'), + }), + }) + return false + } + const existingOtherVarNames = new Set( + draft.variables + .filter(v => v.label !== sourceType && v.variable !== WEBHOOK_RAW_VARIABLE_NAME) + .map(v => v.variable), + ) + + const crossScopeConflict = sanitizedEntries.find(entry => existingOtherVarNames.has(entry.sanitizedName)) + if (crossScopeConflict) { + Toast.notify({ + type: 'error', + message: t('appDebug.varKeyError.keyAlreadyExists', { + key: crossScopeConflict.sanitizedName, + }), + }) + return false + } + + if(hasDuplicateStr(sanitizedEntries.map(entry => entry.sanitizedName))) { + Toast.notify({ + type: 'error', + message: t('appDebug.varKeyError.keyAlreadyExists', { + key: t('appDebug.variableConfig.varName'), + }), + }) + return false + } + + for (const { sanitizedName } of sanitizedEntries) { + const { isValid, errorMessageKey } = checkKeys([sanitizedName], false) + if (!isValid) { + Toast.notify({ + type: 'error', + message: t(`appDebug.varKeyError.${errorMessageKey}`, { + key: t('appDebug.variableConfig.varName'), + }), + }) + return false + } + } + + // Create set of new variable names for this source + const newVarNames = new Set(sanitizedEntries.map(entry => entry.sanitizedName)) + + // Find variables from current source that will be deleted and clean up references + draft.variables + .filter(v => v.label === sourceType && !newVarNames.has(v.variable)) + .forEach((v) => { + // Clean up references if variable is used in other nodes + if (isVarUsedInNodes([id, v.variable])) + removeUsedVarInNodes([id, v.variable]) + }) + + // Remove variables that no longer exist in newData for this specific source type + draft.variables = draft.variables.filter((v) => { + // Keep variables from other sources + if (v.label !== sourceType) return true + return newVarNames.has(v.variable) + }) + + // Add or update variables + sanitizedEntries.forEach(({ item, sanitizedName }) => { + const existingVarIndex = draft.variables.findIndex(v => v.variable === sanitizedName) + + const inputVarType = 'type' in item + ? item.type + : VarType.string // Default to string for headers + + const newVar: Variable = { + value_type: inputVarType, + label: sourceType, // Use sourceType as label to identify source + variable: sanitizedName, + value_selector: [], + required: item.required, + } + + if (existingVarIndex >= 0) + draft.variables[existingVarIndex] = newVar + else + draft.variables.push(newVar) + }) + return true + }, [t, id, isVarUsedInNodes, removeUsedVarInNodes]) + + const handleParamsChange = useCallback((params: WebhookParameter[]) => { + setInputs(produce(inputs, (draft) => { + draft.params = params + syncVariablesInDraft(draft, params, 'param') + })) + }, [inputs, setInputs, syncVariablesInDraft]) + + const handleHeadersChange = useCallback((headers: WebhookHeader[]) => { + setInputs(produce(inputs, (draft) => { + draft.headers = headers + syncVariablesInDraft(draft, headers, 'header') + })) + }, [inputs, setInputs, syncVariablesInDraft]) + + const handleBodyChange = useCallback((body: WebhookParameter[]) => { + setInputs(produce(inputs, (draft) => { + draft.body = body + syncVariablesInDraft(draft, body, 'body') + })) + }, [inputs, setInputs, syncVariablesInDraft]) + + const handleAsyncModeChange = useCallback((asyncMode: boolean) => { + setInputs(produce(inputs, (draft) => { + draft.async_mode = asyncMode + })) + }, [inputs, setInputs]) + + const handleStatusCodeChange = useCallback((statusCode: number) => { + setInputs(produce(inputs, (draft) => { + draft.status_code = statusCode + })) + }, [inputs, setInputs]) + + const handleStatusCodeBlur = useCallback((statusCode: number) => { + // Only clamp when user finishes editing (on blur) + const clampedStatusCode = Math.min(Math.max(statusCode, 200), 399) + + setInputs(produce(inputs, (draft) => { + draft.status_code = clampedStatusCode + })) + }, [inputs, setInputs]) + + const handleResponseBodyChange = useCallback((responseBody: string) => { + setInputs(produce(inputs, (draft) => { + draft.response_body = responseBody + })) + }, [inputs, setInputs]) + + const generateWebhookUrl = useCallback(async () => { + // Idempotency: if we already have a URL, just return it. + if (inputs.webhook_url && inputs.webhook_url.length > 0) + return + + if (!appId) + return + + try { + // Call backend to generate or fetch webhook url for this node + const response = await fetchWebhookUrl({ appId, nodeId: id }) + + const newInputs = produce(inputs, (draft) => { + draft.webhook_url = response.webhook_url + draft.webhook_debug_url = response.webhook_debug_url + }) + setInputs(newInputs) + } + catch (error: unknown) { + // Fallback to mock URL when API is not ready or request fails + // Keep the UI unblocked and allow users to proceed in local/dev environments. + console.error('Failed to generate webhook URL:', error) + const newInputs = produce(inputs, (draft) => { + draft.webhook_url = '' + }) + setInputs(newInputs) + } + }, [appId, id, inputs, setInputs]) + + return { + readOnly, + inputs, + setInputs, + handleMethodChange, + handleContentTypeChange, + handleHeadersChange, + handleParamsChange, + handleBodyChange, + handleAsyncModeChange, + handleStatusCodeChange, + handleStatusCodeBlur, + handleResponseBodyChange, + generateWebhookUrl, + } +} + +export default useConfig diff --git a/web/app/components/workflow/nodes/trigger-webhook/utils/parameter-type-utils.ts b/web/app/components/workflow/nodes/trigger-webhook/utils/parameter-type-utils.ts new file mode 100644 index 0000000000..10f61a5e22 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/utils/parameter-type-utils.ts @@ -0,0 +1,125 @@ +import { VarType } from '@/app/components/workflow/types' + +// Constants for better maintainability and reusability +const BASIC_TYPES = [VarType.string, VarType.number, VarType.boolean, VarType.object, VarType.file] as const +const ARRAY_ELEMENT_TYPES = [VarType.arrayString, VarType.arrayNumber, VarType.arrayBoolean, VarType.arrayObject] as const + +// Generate all valid parameter types programmatically +const VALID_PARAMETER_TYPES: readonly VarType[] = [ + ...BASIC_TYPES, + ...ARRAY_ELEMENT_TYPES, +] as const + +// Type display name mappings +const TYPE_DISPLAY_NAMES: Record<VarType, string> = { + [VarType.string]: 'String', + [VarType.number]: 'Number', + [VarType.boolean]: 'Boolean', + [VarType.object]: 'Object', + [VarType.file]: 'File', + [VarType.arrayString]: 'Array[String]', + [VarType.arrayNumber]: 'Array[Number]', + [VarType.arrayBoolean]: 'Array[Boolean]', + [VarType.arrayObject]: 'Array[Object]', + [VarType.secret]: 'Secret', + [VarType.array]: 'Array', + 'array[file]': 'Array[File]', + [VarType.any]: 'Any', + 'array[any]': 'Array[Any]', + [VarType.integer]: 'Integer', +} as const + +// Content type configurations +const CONTENT_TYPE_CONFIGS = { + 'application/json': { + supportedTypes: [...BASIC_TYPES.filter(t => t !== 'file'), ...ARRAY_ELEMENT_TYPES], + description: 'JSON supports all types including arrays', + }, + 'text/plain': { + supportedTypes: [VarType.string] as const, + description: 'Plain text only supports string', + }, + 'application/x-www-form-urlencoded': { + supportedTypes: [VarType.string, VarType.number, VarType.boolean] as const, + description: 'Form data supports basic types', + }, + 'application/octet-stream': { + supportedTypes: [VarType.file] as const, + description: 'octet-stream supports only binary data', + }, + 'multipart/form-data': { + supportedTypes: [VarType.string, VarType.number, VarType.boolean, VarType.file] as const, + description: 'Multipart supports basic types plus files', + }, +} as const + +/** + * Type guard to check if a string is a valid parameter type + */ +export const isValidParameterType = (type: string): type is VarType => { + return (VALID_PARAMETER_TYPES as readonly string[]).includes(type) +} + +export const normalizeParameterType = (input: string | undefined | null): VarType => { + if (!input || typeof input !== 'string') + return VarType.string + + const trimmed = input.trim().toLowerCase() + if (trimmed === 'array[string]') + return VarType.arrayString + else if (trimmed === 'array[number]') + return VarType.arrayNumber + else if (trimmed === 'array[boolean]') + return VarType.arrayBoolean + else if (trimmed === 'array[object]') + return VarType.arrayObject + else if (trimmed === 'array') + // Migrate legacy 'array' type to 'array[string]' + return VarType.arrayString + else if (trimmed === 'number') + return VarType.number + else if (trimmed === 'boolean') + return VarType.boolean + else if (trimmed === 'object') + return VarType.object + else if (trimmed === 'file') + return VarType.file + + return VarType.string +} + +/** + * Gets display name for parameter types in UI components + */ +export const getParameterTypeDisplayName = (type: VarType): string => { + return TYPE_DISPLAY_NAMES[type] +} + +/** + * Gets available parameter types based on content type + * Provides context-aware type filtering for different webhook content types + */ +export const getAvailableParameterTypes = (contentType?: string): VarType[] => { + if (!contentType) + return [VarType.string, VarType.number, VarType.boolean] + + const normalizedContentType = (contentType || '').toLowerCase() + const configKey = normalizedContentType in CONTENT_TYPE_CONFIGS + ? normalizedContentType as keyof typeof CONTENT_TYPE_CONFIGS + : 'application/json' + + const config = CONTENT_TYPE_CONFIGS[configKey] + return [...config.supportedTypes] +} + +/** + * Creates type options for UI select components + */ +export const createParameterTypeOptions = (contentType?: string) => { + const availableTypes = getAvailableParameterTypes(contentType) + + return availableTypes.map(type => ({ + name: getParameterTypeDisplayName(type), + value: type, + })) +} diff --git a/web/app/components/workflow/nodes/trigger-webhook/utils/raw-variable.ts b/web/app/components/workflow/nodes/trigger-webhook/utils/raw-variable.ts new file mode 100644 index 0000000000..2be7d4c65f --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/utils/raw-variable.ts @@ -0,0 +1,12 @@ +import { VarType, type Variable } from '@/app/components/workflow/types' + +export const WEBHOOK_RAW_VARIABLE_NAME = '_webhook_raw' +export const WEBHOOK_RAW_VARIABLE_LABEL = 'raw' + +export const createWebhookRawVariable = (): Variable => ({ + variable: WEBHOOK_RAW_VARIABLE_NAME, + label: WEBHOOK_RAW_VARIABLE_LABEL, + value_type: VarType.object, + value_selector: [], + required: true, +}) diff --git a/web/app/components/workflow/nodes/trigger-webhook/utils/render-output-vars.tsx b/web/app/components/workflow/nodes/trigger-webhook/utils/render-output-vars.tsx new file mode 100644 index 0000000000..0e9cb8a309 --- /dev/null +++ b/web/app/components/workflow/nodes/trigger-webhook/utils/render-output-vars.tsx @@ -0,0 +1,75 @@ +import type { FC } from 'react' +import React from 'react' +import type { Variable } from '@/app/components/workflow/types' + +type OutputVariablesContentProps = { + variables?: Variable[] +} + +// Define the display order for variable labels to match the table order in the UI +const LABEL_ORDER = { raw: 0, param: 1, header: 2, body: 3 } as const + +const getLabelPrefix = (label: string): string => { + const prefixMap: Record<string, string> = { + raw: 'payload', + param: 'query_params', + header: 'header_params', + body: 'req_body_params', + } + return prefixMap[label] || label +} + +type VarItemProps = { + prefix: string + name: string + type: string +} + +const VarItem: FC<VarItemProps> = ({ prefix, name, type }) => { + return ( + <div className='py-1'> + <div className='flex items-center leading-[18px]'> + <span className='code-sm-regular text-text-tertiary'>{prefix}.</span> + <span className='code-sm-semibold text-text-secondary'>{name}</span> + <span className='system-xs-regular ml-2 text-text-tertiary'>{type}</span> + </div> + </div> + ) +} + +export const OutputVariablesContent: FC<OutputVariablesContentProps> = ({ variables = [] }) => { + if (!variables || variables.length === 0) { + return ( + <div className="system-sm-regular py-2 text-text-tertiary"> + No output variables + </div> + ) + } + + // Sort variables by label to match the table display order: param → header → body + // Unknown labels are placed at the end (order value 999) + const sortedVariables = [...variables].sort((a, b) => { + const labelA = typeof a.label === 'string' ? a.label : '' + const labelB = typeof b.label === 'string' ? b.label : '' + return (LABEL_ORDER[labelA as keyof typeof LABEL_ORDER] || 999) + - (LABEL_ORDER[labelB as keyof typeof LABEL_ORDER] || 999) + }) + + return ( + <div> + {sortedVariables.map((variable, index) => { + const label = typeof variable.label === 'string' ? variable.label : '' + const varName = typeof variable.variable === 'string' ? variable.variable : '' + + return ( + <VarItem + key={`${label}-${varName}-${index}`} + prefix={getLabelPrefix(label)} + name={varName} + type={variable.value_type || 'string'} + /> + ) + })} + </div> + ) +} diff --git a/web/app/components/workflow/note-node/index.tsx b/web/app/components/workflow/note-node/index.tsx index 7f2cde42d6..5a0b2677c1 100644 --- a/web/app/components/workflow/note-node/index.tsx +++ b/web/app/components/workflow/note-node/index.tsx @@ -1,6 +1,5 @@ import { memo, - useCallback, useRef, } from 'react' import { useTranslation } from 'react-i18next' @@ -51,10 +50,6 @@ const NoteNode = ({ } = useNodesInteractions() const { handleNodeDataUpdateWithSyncDraft } = useNodeDataUpdate() - const handleDeleteNode = useCallback(() => { - handleNodeDelete(id) - }, [id, handleNodeDelete]) - useClickAway(() => { handleNodeDataUpdateWithSyncDraft({ id, data: { selected: false } }) }, ref) @@ -102,9 +97,9 @@ const NoteNode = ({ <NoteEditorToolbar theme={theme} onThemeChange={handleThemeChange} - onCopy={handleNodesCopy} - onDuplicate={handleNodesDuplicate} - onDelete={handleDeleteNode} + onCopy={() => handleNodesCopy(id)} + onDuplicate={() => handleNodesDuplicate(id)} + onDelete={() => handleNodeDelete(id)} showAuthor={data.showAuthor} onShowAuthorChange={handleShowAuthorChange} /> diff --git a/web/app/components/workflow/operator/add-block.tsx b/web/app/components/workflow/operator/add-block.tsx index 3c11dbac32..c40f2277bb 100644 --- a/web/app/components/workflow/operator/add-block.tsx +++ b/web/app/components/workflow/operator/add-block.tsx @@ -13,10 +13,12 @@ import { } from '../utils' import { useAvailableBlocks, + useIsChatMode, useNodesMetaData, useNodesReadOnly, usePanelInteractions, } from '../hooks' +import { useHooksStore } from '../hooks-store' import { useWorkflowStore } from '../store' import TipPopup from './tip-popup' import cn from '@/utils/classnames' @@ -27,6 +29,7 @@ import type { import { BlockEnum, } from '@/app/components/workflow/types' +import { FlowType } from '@/types/common' type AddBlockProps = { renderTrigger?: (open: boolean) => React.ReactNode @@ -39,11 +42,14 @@ const AddBlock = ({ const { t } = useTranslation() const store = useStoreApi() const workflowStore = useWorkflowStore() + const isChatMode = useIsChatMode() const { nodesReadOnly } = useNodesReadOnly() const { handlePaneContextmenuCancel } = usePanelInteractions() const [open, setOpen] = useState(false) const { availableNextBlocks } = useAvailableBlocks(BlockEnum.Start, false) const { nodesMap: nodesMetaDataMap } = useNodesMetaData() + const flowType = useHooksStore(s => s.configsMap?.flowType) + const showStartTab = flowType !== FlowType.ragPipeline && !isChatMode const handleOpenChange = useCallback((open: boolean) => { setOpen(open) @@ -51,7 +57,7 @@ const AddBlock = ({ handlePaneContextmenuCancel() }, [handlePaneContextmenuCancel]) - const handleSelect = useCallback<OnSelectBlock>((type, toolDefaultValue) => { + const handleSelect = useCallback<OnSelectBlock>((type, pluginDefaultValue) => { const { getNodes, } = store.getState() @@ -65,7 +71,7 @@ const AddBlock = ({ data: { ...(defaultValue as any), title: nodesWithSameType.length > 0 ? `${defaultValue.title} ${nodesWithSameType.length + 1}` : defaultValue.title, - ...toolDefaultValue, + ...pluginDefaultValue, _isCandidate: true, }, position: { @@ -108,6 +114,7 @@ const AddBlock = ({ trigger={renderTrigger || renderTriggerElement} popupClassName='!min-w-[256px]' availableBlocksTypes={availableNextBlocks} + showStartTab={showStartTab} /> ) } diff --git a/web/app/components/workflow/operator/control.tsx b/web/app/components/workflow/operator/control.tsx index cfc32bbc30..7f1225de86 100644 --- a/web/app/components/workflow/operator/control.tsx +++ b/web/app/components/workflow/operator/control.tsx @@ -24,7 +24,7 @@ import { useStore } from '../store' import Divider from '../../base/divider' import AddBlock from './add-block' import TipPopup from './tip-popup' -import ExportImage from './export-image' +import MoreActions from './more-actions' import { useOperator } from './hooks' import cn from '@/utils/classnames' @@ -89,7 +89,6 @@ const Control = () => { </div> </TipPopup> <Divider className='my-1 w-3.5' /> - <ExportImage /> <TipPopup title={t('workflow.panel.organizeBlocks')} shortcuts={['ctrl', 'o']}> <div className={cn( @@ -114,6 +113,7 @@ const Control = () => { {!maximizeCanvas && <RiAspectRatioLine className='h-4 w-4' />} </div> </TipPopup> + <MoreActions /> </div> ) } diff --git a/web/app/components/workflow/operator/index.tsx b/web/app/components/workflow/operator/index.tsx index 1100a7a905..b4fcf184a7 100644 --- a/web/app/components/workflow/operator/index.tsx +++ b/web/app/components/workflow/operator/index.tsx @@ -1,4 +1,5 @@ -import { memo, useEffect, useMemo, useRef } from 'react' +import { memo, useCallback, useEffect, useMemo, useRef } from 'react' +import type { Node } from 'reactflow' import { MiniMap } from 'reactflow' import UndoRedo from '../header/undo-redo' import ZoomInOut from './zoom-in-out' @@ -24,6 +25,12 @@ const Operator = ({ handleUndo, handleRedo }: OperatorProps) => { return Math.max((workflowCanvasWidth - rightPanelWidth), 400) }, [workflowCanvasWidth, rightPanelWidth]) + const getMiniMapNodeClassName = useCallback((node: Node) => { + return node.data?.selected + ? 'bg-workflow-minimap-block border-components-option-card-option-selected-border' + : 'bg-workflow-minimap-block' + }, []) + // update bottom panel height useEffect(() => { if (bottomPanelRef.current) { @@ -65,6 +72,8 @@ const Operator = ({ handleUndo, handleRedo }: OperatorProps) => { height: 72, }} maskColor='var(--color-workflow-minimap-bg)' + nodeClassName={getMiniMapNodeClassName} + nodeStrokeWidth={3} className='!absolute !bottom-10 z-[9] !m-0 !h-[73px] !w-[103px] !rounded-lg !border-[0.5px] !border-divider-subtle !bg-background-default-subtle !shadow-md !shadow-shadow-shadow-5' /> diff --git a/web/app/components/workflow/operator/export-image.tsx b/web/app/components/workflow/operator/more-actions.tsx similarity index 87% rename from web/app/components/workflow/operator/export-image.tsx rename to web/app/components/workflow/operator/more-actions.tsx index 9b85847fd6..100df29560 100644 --- a/web/app/components/workflow/operator/export-image.tsx +++ b/web/app/components/workflow/operator/more-actions.tsx @@ -2,13 +2,15 @@ import type { FC } from 'react' import { memo, useCallback, + useMemo, useState, } from 'react' +import { useShallow } from 'zustand/react/shallow' import { useTranslation } from 'react-i18next' +import { RiExportLine, RiMoreFill } from '@remixicon/react' import { toJpeg, toPng, toSvg } from 'html-to-image' import { useNodesReadOnly } from '../hooks' import TipPopup from './tip-popup' -import { RiExportLine } from '@remixicon/react' import cn from '@/utils/classnames' import { PortalToFollowElem, @@ -18,8 +20,9 @@ import { import { getNodesBounds, useReactFlow } from 'reactflow' import ImagePreview from '@/app/components/base/image-uploader/image-preview' import { useStore } from '@/app/components/workflow/store' +import { useStore as useAppStore } from '@/app/components/app/store' -const ExportImage: FC = () => { +const MoreActions: FC = () => { const { t } = useTranslation() const { getNodesReadOnly } = useNodesReadOnly() const reactFlow = useReactFlow() @@ -29,6 +32,15 @@ const ExportImage: FC = () => { const [previewTitle, setPreviewTitle] = useState('') const knowledgeName = useStore(s => s.knowledgeName) const appName = useStore(s => s.appName) + const maximizeCanvas = useStore(s => s.maximizeCanvas) + const { appSidebarExpand } = useAppStore(useShallow(state => ({ + appSidebarExpand: state.appSidebarExpand, + }))) + + const crossAxisOffset = useMemo(() => { + if (maximizeCanvas) return 40 + return appSidebarExpand === 'expand' ? 188 : 40 + }, [appSidebarExpand, maximizeCanvas]) const handleExportImage = useCallback(async (type: 'png' | 'jpeg' | 'svg', currentWorkflow = false) => { if (!appName && !knowledgeName) @@ -53,14 +65,11 @@ const ExportImage: FC = () => { let dataUrl if (currentWorkflow) { - // Get all nodes and their bounds const nodes = reactFlow.getNodes() const nodesBounds = getNodesBounds(nodes) - // Save current viewport const currentViewport = reactFlow.getViewport() - // Calculate the required zoom to fit all nodes const viewportWidth = window.innerWidth const viewportHeight = window.innerHeight const zoom = Math.min( @@ -69,30 +78,25 @@ const ExportImage: FC = () => { 1, ) - // Calculate center position const centerX = nodesBounds.x + nodesBounds.width / 2 const centerY = nodesBounds.y + nodesBounds.height / 2 - // Set viewport to show all nodes reactFlow.setViewport({ x: viewportWidth / 2 - centerX * zoom, y: viewportHeight / 2 - centerY * zoom, zoom, }) - // Wait for the transition to complete await new Promise(resolve => setTimeout(resolve, 300)) - // Calculate actual content size with padding - const padding = 50 // More padding for better visualization + const padding = 50 const contentWidth = nodesBounds.width + padding * 2 const contentHeight = nodesBounds.height + padding * 2 - // Export with higher quality for whole workflow const exportOptions = { filter, - backgroundColor: '#1a1a1a', // Dark background to match previous style - pixelRatio: 2, // Higher resolution for better zoom + backgroundColor: '#1a1a1a', + pixelRatio: 2, width: contentWidth, height: contentHeight, style: { @@ -119,7 +123,6 @@ const ExportImage: FC = () => { filename += '-whole-workflow' - // Restore original viewport after a delay setTimeout(() => { reactFlow.setViewport(currentViewport) }, 500) @@ -142,11 +145,9 @@ const ExportImage: FC = () => { } if (currentWorkflow) { - // For whole workflow, show preview first setPreviewUrl(dataUrl) setPreviewTitle(`${filename}.${type}`) - // Also auto-download const link = document.createElement('a') link.href = dataUrl link.download = `${filename}.${type}` @@ -181,14 +182,14 @@ const ExportImage: FC = () => { <PortalToFollowElem open={open} onOpenChange={setOpen} - placement="top-start" + placement="bottom-end" offset={{ - mainAxis: 4, - crossAxis: -8, + mainAxis: -200, + crossAxis: crossAxisOffset, }} > <PortalToFollowElemTrigger> - <TipPopup title={t('workflow.common.exportImage')}> + <TipPopup title={t('workflow.common.moreActions')}> <div className={cn( 'flex h-8 w-8 cursor-pointer items-center justify-center rounded-lg hover:bg-state-base-hover hover:text-text-secondary', @@ -196,13 +197,17 @@ const ExportImage: FC = () => { )} onClick={handleTrigger} > - <RiExportLine className='h-4 w-4' /> + <RiMoreFill className='h-4 w-4' /> </div> </TipPopup> </PortalToFollowElemTrigger> <PortalToFollowElemContent className='z-10'> <div className='min-w-[180px] rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur text-text-secondary shadow-lg'> <div className='p-1'> + <div className='flex items-center gap-2 px-2 py-1 text-xs font-medium text-text-tertiary'> + <RiExportLine className='h-3 w-3' /> + {t('workflow.common.exportImage')} + </div> <div className='px-2 py-1 text-xs font-medium text-text-tertiary'> {t('workflow.common.currentView')} </div> @@ -264,4 +269,4 @@ const ExportImage: FC = () => { ) } -export default memo(ExportImage) +export default memo(MoreActions) diff --git a/web/app/components/workflow/panel/debug-and-preview/chat-wrapper.tsx b/web/app/components/workflow/panel/debug-and-preview/chat-wrapper.tsx index 1a97357da5..6fba10bf81 100644 --- a/web/app/components/workflow/panel/debug-and-preview/chat-wrapper.tsx +++ b/web/app/components/workflow/panel/debug-and-preview/chat-wrapper.tsx @@ -12,7 +12,7 @@ import ConversationVariableModal from './conversation-variable-modal' import { useChat } from './hooks' import type { ChatWrapperRefType } from './index' import Chat from '@/app/components/base/chat/chat' -import type { ChatItem, ChatItemInTree, OnSend } from '@/app/components/base/chat/types' +import type { ChatItem, OnSend } from '@/app/components/base/chat/types' import { useFeatures } from '@/app/components/base/features/hooks' import { fetchSuggestedQuestions, @@ -117,7 +117,7 @@ const ChatWrapper = ( ) }, [handleSend, workflowStore, conversationId, chatList, appDetail]) - const doRegenerate = useCallback((chatItem: ChatItemInTree, editedQuestion?: { message: string, files?: FileEntity[] }) => { + const doRegenerate = useCallback((chatItem: ChatItem, editedQuestion?: { message: string, files?: FileEntity[] }) => { const question = editedQuestion ? chatItem : chatList.find(item => item.id === chatItem.parentMessageId)! const parentAnswer = chatList.find(item => item.id === question.parentMessageId) doSend(editedQuestion ? editedQuestion.message : question.content, diff --git a/web/app/components/workflow/panel/global-variable-panel/index.tsx b/web/app/components/workflow/panel/global-variable-panel/index.tsx index ad7996ab0c..a421a1605a 100644 --- a/web/app/components/workflow/panel/global-variable-panel/index.tsx +++ b/web/app/components/workflow/panel/global-variable-panel/index.tsx @@ -8,16 +8,53 @@ import Item from './item' import { useStore } from '@/app/components/workflow/store' import cn from '@/utils/classnames' +import { useTranslation } from 'react-i18next' +import { useIsChatMode } from '../../hooks' +import { isInWorkflowPage } from '../../constants' const Panel = () => { + const { t } = useTranslation() + const isChatMode = useIsChatMode() const setShowPanel = useStore(s => s.setShowGlobalVariablePanel) + const isWorkflowPage = isInWorkflowPage() const globalVariableList: GlobalVariable[] = [ - { + ...(isChatMode ? [{ name: 'conversation_id', - value_type: 'string', - description: 'conversation id', + value_type: 'string' as const, + description: t('workflow.globalVar.fieldsDescription.conversationId'), }, + { + name: 'dialog_count', + value_type: 'number' as const, + description: t('workflow.globalVar.fieldsDescription.dialogCount'), + }] : []), + { + name: 'user_id', + value_type: 'string', + description: t('workflow.globalVar.fieldsDescription.userId'), + }, + { + name: 'app_id', + value_type: 'string', + description: t('workflow.globalVar.fieldsDescription.appId'), + }, + { + name: 'workflow_id', + value_type: 'string', + description: t('workflow.globalVar.fieldsDescription.workflowId'), + }, + { + name: 'workflow_run_id', + value_type: 'string', + description: t('workflow.globalVar.fieldsDescription.workflowRunId'), + }, + // is workflow + ...((isWorkflowPage && !isChatMode) ? [{ + name: 'timestamp', + value_type: 'number' as const, + description: t('workflow.globalVar.fieldsDescription.triggerTimestamp'), + }] : []), ] return ( @@ -27,7 +64,7 @@ const Panel = () => { )} > <div className='system-xl-semibold flex shrink-0 items-center justify-between p-4 pb-0 text-text-primary'> - Global Variables(Current not show) + {t('workflow.globalVar.title')} <div className='flex items-center'> <div className='flex h-6 w-6 cursor-pointer items-center justify-center' @@ -37,9 +74,9 @@ const Panel = () => { </div> </div> </div> - <div className='system-sm-regular shrink-0 px-4 py-1 text-text-tertiary'>...</div> + <div className='system-sm-regular shrink-0 px-4 py-1 text-text-tertiary'>{t('workflow.globalVar.description')}</div> - <div className='grow overflow-y-auto rounded-b-2xl px-4'> + <div className='mt-4 grow overflow-y-auto rounded-b-2xl px-4'> {globalVariableList.map(item => ( <Item key={item.name} diff --git a/web/app/components/workflow/panel/global-variable-panel/item.tsx b/web/app/components/workflow/panel/global-variable-panel/item.tsx index ddf9abe1d3..5185c1bead 100644 --- a/web/app/components/workflow/panel/global-variable-panel/item.tsx +++ b/web/app/components/workflow/panel/global-variable-panel/item.tsx @@ -1,6 +1,7 @@ import { memo } from 'react' import { capitalize } from 'lodash-es' -import { Env } from '@/app/components/base/icons/src/vender/line/others' +import { GlobalVariable as GlobalVariableIcon } from '@/app/components/base/icons/src/vender/line/others' + import type { GlobalVariable } from '@/app/components/workflow/types' import cn from '@/utils/classnames' @@ -17,12 +18,15 @@ const Item = ({ )}> <div className='flex items-center justify-between'> <div className='flex grow items-center gap-1'> - <Env className='h-4 w-4 text-util-colors-violet-violet-600' /> - <div className='system-sm-medium text-text-primary'>{payload.name}</div> + <GlobalVariableIcon className='h-4 w-4 text-util-colors-orange-orange-600' /> + <div className='system-sm-medium text-text-primary'> + <span className='text-text-tertiary'>sys.</span> + {payload.name} + </div> <div className='system-xs-medium text-text-tertiary'>{capitalize(payload.value_type)}</div> </div> </div> - <div className='system-xs-regular truncate text-text-tertiary'>{payload.description}</div> + <div className='system-xs-regular mt-1.5 truncate text-text-tertiary'>{payload.description}</div> </div> ) } diff --git a/web/app/components/workflow/panel/workflow-preview.tsx b/web/app/components/workflow/panel/workflow-preview.tsx index fdb1767df9..292a964b9e 100644 --- a/web/app/components/workflow/panel/workflow-preview.tsx +++ b/web/app/components/workflow/panel/workflow-preview.tsx @@ -31,6 +31,7 @@ const WorkflowPreview = () => { const { t } = useTranslation() const { handleCancelDebugAndPreviewPanel } = useWorkflowInteractions() const workflowRunningData = useStore(s => s.workflowRunningData) + const isListening = useStore(s => s.isListening) const showInputsPanel = useStore(s => s.showInputsPanel) const workflowCanvasWidth = useStore(s => s.workflowCanvasWidth) const panelWidth = useStore(s => s.previewPanelWidth) @@ -48,7 +49,16 @@ const WorkflowPreview = () => { }, [showDebugAndPreviewPanel, showInputsPanel]) useEffect(() => { - if ((workflowRunningData?.result.status === WorkflowRunningStatus.Succeeded || workflowRunningData?.result.status === WorkflowRunningStatus.Failed) && !workflowRunningData.resultText && !workflowRunningData.result.files?.length) + if (isListening) + switchTab('DETAIL') + }, [isListening]) + + useEffect(() => { + const status = workflowRunningData?.result.status + if (!workflowRunningData) + return + + if ((status === WorkflowRunningStatus.Succeeded || status === WorkflowRunningStatus.Failed) && !workflowRunningData.resultText && !workflowRunningData.result.files?.length) switchTab('DETAIL') }, [workflowRunningData]) diff --git a/web/app/components/workflow/run/index.tsx b/web/app/components/workflow/run/index.tsx index 2e9ae392a6..5f6b07033d 100644 --- a/web/app/components/workflow/run/index.tsx +++ b/web/app/components/workflow/run/index.tsx @@ -5,6 +5,8 @@ import { useContext } from 'use-context-selector' import { useTranslation } from 'react-i18next' import OutputPanel from './output-panel' import ResultPanel from './result-panel' +import StatusPanel from './status' +import { WorkflowRunningStatus } from '@/app/components/workflow/types' import TracingPanel from './tracing-panel' import cn from '@/utils/classnames' import { ToastContext } from '@/app/components/base/toast' @@ -12,6 +14,8 @@ import Loading from '@/app/components/base/loading' import { fetchRunDetail, fetchTracingList } from '@/service/log' import type { NodeTracing } from '@/types/workflow' import type { WorkflowRunDetailResponse } from '@/models/log' +import { useStore } from '../store' + export type RunProps = { hideResult?: boolean activeTab?: 'RESULT' | 'DETAIL' | 'TRACING' @@ -33,6 +37,7 @@ const RunPanel: FC<RunProps> = ({ const [loading, setLoading] = useState<boolean>(true) const [runDetail, setRunDetail] = useState<WorkflowRunDetailResponse>() const [list, setList] = useState<NodeTracing[]>([]) + const isListening = useStore(s => s.isListening) const executor = useMemo(() => { if (runDetail?.created_by_role === 'account') @@ -89,6 +94,11 @@ const RunPanel: FC<RunProps> = ({ await getTracingList() } + useEffect(() => { + if (isListening) + setCurrentTab('DETAIL') + }, [isListening]) + useEffect(() => { // fetch data if (runDetailUrl && tracingListUrl) @@ -166,6 +176,11 @@ const RunPanel: FC<RunProps> = ({ exceptionCounts={runDetail.exceptions_count} /> )} + {!loading && currentTab === 'DETAIL' && !runDetail && isListening && ( + <StatusPanel + status={WorkflowRunningStatus.Running} + /> + )} {!loading && currentTab === 'TRACING' && ( <TracingPanel className='bg-background-section-burn' diff --git a/web/app/components/workflow/run/iteration-log/iteration-log-trigger.tsx b/web/app/components/workflow/run/iteration-log/iteration-log-trigger.tsx index 8c9b43916b..d1926da15e 100644 --- a/web/app/components/workflow/run/iteration-log/iteration-log-trigger.tsx +++ b/web/app/components/workflow/run/iteration-log/iteration-log-trigger.tsx @@ -5,6 +5,7 @@ import type { IterationDurationMap, NodeTracing, } from '@/types/workflow' +import { NodeRunningStatus } from '@/app/components/workflow/types' import { Iteration } from '@/app/components/base/icons/src/vender/workflow' type IterationLogTriggerProps = { @@ -54,6 +55,30 @@ const IterationLogTrigger = ({ structuredList = instanceKeys .map(key => filterNodesForInstance(key)) .filter(branchNodes => branchNodes.length > 0) + + // Also include failed iterations that might not be in duration map + if (allExecutions && nodeInfo.details?.length) { + const existingIterationIndices = new Set<number>() + structuredList.forEach((iteration) => { + iteration.forEach((node) => { + if (node.execution_metadata?.iteration_index !== undefined) + existingIterationIndices.add(node.execution_metadata.iteration_index) + }) + }) + + // Find failed iterations that are not in the structured list + nodeInfo.details.forEach((iteration, index) => { + if (!existingIterationIndices.has(index) && iteration.some(node => node.status === NodeRunningStatus.Failed)) + structuredList.push(iteration) + }) + + // Sort by iteration index to maintain order + structuredList.sort((a, b) => { + const aIndex = a[0]?.execution_metadata?.iteration_index ?? 0 + const bIndex = b[0]?.execution_metadata?.iteration_index ?? 0 + return aIndex - bIndex + }) + } } else if (nodeInfo.details?.length) { structuredList = nodeInfo.details @@ -71,16 +96,36 @@ const IterationLogTrigger = ({ else if (nodeInfo.metadata?.iterator_length) displayIterationCount = nodeInfo.metadata.iterator_length - const getErrorCount = (details: NodeTracing[][] | undefined) => { + const getErrorCount = (details: NodeTracing[][] | undefined, iterationNodeMeta?: any) => { if (!details || details.length === 0) return 0 - return details.reduce((acc, iteration) => { - if (iteration.some(item => item.status === 'failed')) - acc++ - return acc - }, 0) + + // Use Set to track failed iteration indices to avoid duplicate counting + const failedIterationIndices = new Set<number>() + + // Collect failed iteration indices from details + details.forEach((iteration, index) => { + if (iteration.some(item => item.status === NodeRunningStatus.Failed)) { + // Try to get iteration index from first node, fallback to array index + const iterationIndex = iteration[0]?.execution_metadata?.iteration_index ?? index + failedIterationIndices.add(iterationIndex) + } + }) + + // If allExecutions exists, check for additional failed iterations + if (iterationNodeMeta?.iteration_duration_map && allExecutions) { + // Find all failed iteration nodes + allExecutions.forEach((exec) => { + if (exec.execution_metadata?.iteration_id === nodeInfo.node_id + && exec.status === NodeRunningStatus.Failed + && exec.execution_metadata?.iteration_index !== undefined) + failedIterationIndices.add(exec.execution_metadata.iteration_index) + }) + } + + return failedIterationIndices.size } - const errorCount = getErrorCount(nodeInfo.details) + const errorCount = getErrorCount(nodeInfo.details, nodeInfo.execution_metadata) return ( <Button diff --git a/web/app/components/workflow/run/status-container.tsx b/web/app/components/workflow/run/status-container.tsx index 47890da0b2..6837592c4e 100644 --- a/web/app/components/workflow/run/status-container.tsx +++ b/web/app/components/workflow/run/status-container.tsx @@ -14,6 +14,7 @@ const StatusContainer: FC<Props> = ({ children, }) => { const { theme } = useTheme() + return ( <div className={cn( diff --git a/web/app/components/workflow/run/status.tsx b/web/app/components/workflow/run/status.tsx index 253aaab3a1..fa9559fcf8 100644 --- a/web/app/components/workflow/run/status.tsx +++ b/web/app/components/workflow/run/status.tsx @@ -5,6 +5,7 @@ import cn from '@/utils/classnames' import Indicator from '@/app/components/header/indicator' import StatusContainer from '@/app/components/workflow/run/status-container' import { useDocLink } from '@/context/i18n' +import { useStore } from '../store' type ResultProps = { status: string @@ -23,6 +24,7 @@ const StatusPanel: FC<ResultProps> = ({ }) => { const { t } = useTranslation() const docLink = useDocLink() + const isListening = useStore(s => s.isListening) return ( <StatusContainer status={status}> @@ -45,7 +47,7 @@ const StatusPanel: FC<ResultProps> = ({ {status === 'running' && ( <> <Indicator color={'blue'} /> - <span>Running</span> + <span>{isListening ? 'Listening' : 'Running'}</span> </> )} {status === 'succeeded' && ( @@ -106,7 +108,7 @@ const StatusPanel: FC<ResultProps> = ({ {status === 'failed' && error && ( <> <div className='my-2 h-[0.5px] bg-divider-subtle'/> - <div className='system-xs-regular text-text-destructive'>{error}</div> + <div className='system-xs-regular whitespace-pre-wrap text-text-destructive'>{error}</div> { !!exceptionCounts && ( <> diff --git a/web/app/components/workflow/run/utils/format-log/parallel/index.ts b/web/app/components/workflow/run/utils/format-log/parallel/index.ts index f5a1136e3f..22c96918e9 100644 --- a/web/app/components/workflow/run/utils/format-log/parallel/index.ts +++ b/web/app/components/workflow/run/utils/format-log/parallel/index.ts @@ -148,7 +148,7 @@ const format = (list: NodeTracing[], t: any, isPrint?: boolean): NodeTracing[] = return false const isParallelStartNode = node.parallelDetail?.isParallelStartNode - // eslint-disable-next-line sonarjs/prefer-single-boolean-return + if (!isParallelStartNode) return false diff --git a/web/app/components/workflow/shortcuts-name.tsx b/web/app/components/workflow/shortcuts-name.tsx index e7122c5ad5..9dd8c4bcd1 100644 --- a/web/app/components/workflow/shortcuts-name.tsx +++ b/web/app/components/workflow/shortcuts-name.tsx @@ -5,10 +5,12 @@ import cn from '@/utils/classnames' type ShortcutsNameProps = { keys: string[] className?: string + textColor?: 'default' | 'secondary' } const ShortcutsName = ({ keys, className, + textColor = 'default', }: ShortcutsNameProps) => { return ( <div className={cn( @@ -19,7 +21,10 @@ const ShortcutsName = ({ keys.map(key => ( <div key={key} - className='system-kbd flex h-4 min-w-4 items-center justify-center rounded-[4px] bg-components-kbd-bg-gray capitalize' + className={cn( + 'system-kbd flex h-4 min-w-4 items-center justify-center rounded-[4px] bg-components-kbd-bg-gray capitalize', + textColor === 'secondary' && 'text-text-tertiary', + )} > {getKeyboardKeyNameBySystem(key)} </div> diff --git a/web/app/components/workflow/store/__tests__/trigger-status.test.ts b/web/app/components/workflow/store/__tests__/trigger-status.test.ts new file mode 100644 index 0000000000..d7e1284487 --- /dev/null +++ b/web/app/components/workflow/store/__tests__/trigger-status.test.ts @@ -0,0 +1,293 @@ +import { act, renderHook } from '@testing-library/react' +import { useTriggerStatusStore } from '../trigger-status' +import type { EntryNodeStatus } from '../trigger-status' + +describe('useTriggerStatusStore', () => { + beforeEach(() => { + // Clear the store state before each test + const { result } = renderHook(() => useTriggerStatusStore()) + act(() => { + result.current.clearTriggerStatuses() + }) + }) + + describe('Initial State', () => { + it('should initialize with empty trigger statuses', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + expect(result.current.triggerStatuses).toEqual({}) + }) + + it('should return "disabled" for non-existent trigger status', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + const status = result.current.getTriggerStatus('non-existent-id') + expect(status).toBe('disabled') + }) + }) + + describe('setTriggerStatus', () => { + it('should set trigger status for a single node', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + act(() => { + result.current.setTriggerStatus('node-1', 'enabled') + }) + + expect(result.current.triggerStatuses['node-1']).toBe('enabled') + expect(result.current.getTriggerStatus('node-1')).toBe('enabled') + }) + + it('should update existing trigger status', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + // Set initial status + act(() => { + result.current.setTriggerStatus('node-1', 'enabled') + }) + expect(result.current.getTriggerStatus('node-1')).toBe('enabled') + + // Update status + act(() => { + result.current.setTriggerStatus('node-1', 'disabled') + }) + expect(result.current.getTriggerStatus('node-1')).toBe('disabled') + }) + + it('should handle multiple nodes independently', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + act(() => { + result.current.setTriggerStatus('node-1', 'enabled') + result.current.setTriggerStatus('node-2', 'disabled') + }) + + expect(result.current.getTriggerStatus('node-1')).toBe('enabled') + expect(result.current.getTriggerStatus('node-2')).toBe('disabled') + }) + }) + + describe('setTriggerStatuses', () => { + it('should set multiple trigger statuses at once', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + const statuses = { + 'node-1': 'enabled' as EntryNodeStatus, + 'node-2': 'disabled' as EntryNodeStatus, + 'node-3': 'enabled' as EntryNodeStatus, + } + + act(() => { + result.current.setTriggerStatuses(statuses) + }) + + expect(result.current.triggerStatuses).toEqual(statuses) + expect(result.current.getTriggerStatus('node-1')).toBe('enabled') + expect(result.current.getTriggerStatus('node-2')).toBe('disabled') + expect(result.current.getTriggerStatus('node-3')).toBe('enabled') + }) + + it('should replace existing statuses completely', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + // Set initial statuses + act(() => { + result.current.setTriggerStatuses({ + 'node-1': 'enabled', + 'node-2': 'disabled', + }) + }) + + // Replace with new statuses + act(() => { + result.current.setTriggerStatuses({ + 'node-3': 'enabled', + 'node-4': 'disabled', + }) + }) + + expect(result.current.triggerStatuses).toEqual({ + 'node-3': 'enabled', + 'node-4': 'disabled', + }) + expect(result.current.getTriggerStatus('node-1')).toBe('disabled') // default + expect(result.current.getTriggerStatus('node-2')).toBe('disabled') // default + }) + + it('should handle empty object', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + // Set some initial data + act(() => { + result.current.setTriggerStatus('node-1', 'enabled') + }) + + // Clear with empty object + act(() => { + result.current.setTriggerStatuses({}) + }) + + expect(result.current.triggerStatuses).toEqual({}) + expect(result.current.getTriggerStatus('node-1')).toBe('disabled') + }) + }) + + describe('getTriggerStatus', () => { + it('should return the correct status for existing nodes', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + act(() => { + result.current.setTriggerStatuses({ + 'enabled-node': 'enabled', + 'disabled-node': 'disabled', + }) + }) + + expect(result.current.getTriggerStatus('enabled-node')).toBe('enabled') + expect(result.current.getTriggerStatus('disabled-node')).toBe('disabled') + }) + + it('should return "disabled" as default for non-existent nodes', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + expect(result.current.getTriggerStatus('non-existent')).toBe('disabled') + expect(result.current.getTriggerStatus('')).toBe('disabled') + expect(result.current.getTriggerStatus('undefined-node')).toBe('disabled') + }) + }) + + describe('clearTriggerStatuses', () => { + it('should clear all trigger statuses', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + // Set some statuses + act(() => { + result.current.setTriggerStatuses({ + 'node-1': 'enabled', + 'node-2': 'disabled', + 'node-3': 'enabled', + }) + }) + + expect(Object.keys(result.current.triggerStatuses)).toHaveLength(3) + + // Clear all + act(() => { + result.current.clearTriggerStatuses() + }) + + expect(result.current.triggerStatuses).toEqual({}) + expect(result.current.getTriggerStatus('node-1')).toBe('disabled') + expect(result.current.getTriggerStatus('node-2')).toBe('disabled') + expect(result.current.getTriggerStatus('node-3')).toBe('disabled') + }) + + it('should not throw when clearing empty statuses', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + expect(() => { + act(() => { + result.current.clearTriggerStatuses() + }) + }).not.toThrow() + + expect(result.current.triggerStatuses).toEqual({}) + }) + }) + + describe('Store Reactivity', () => { + it('should notify subscribers when status changes', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + const initialTriggerStatuses = result.current.triggerStatuses + + act(() => { + result.current.setTriggerStatus('reactive-node', 'enabled') + }) + + // The reference should change, indicating reactivity + expect(result.current.triggerStatuses).not.toBe(initialTriggerStatuses) + expect(result.current.triggerStatuses['reactive-node']).toBe('enabled') + }) + + it('should maintain immutability when updating statuses', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + act(() => { + result.current.setTriggerStatus('node-1', 'enabled') + }) + + const firstSnapshot = result.current.triggerStatuses + + act(() => { + result.current.setTriggerStatus('node-2', 'disabled') + }) + + const secondSnapshot = result.current.triggerStatuses + + // References should be different (immutable updates) + expect(firstSnapshot).not.toBe(secondSnapshot) + // But the first node status should remain + expect(secondSnapshot['node-1']).toBe('enabled') + expect(secondSnapshot['node-2']).toBe('disabled') + }) + }) + + describe('Edge Cases', () => { + it('should handle rapid consecutive updates', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + act(() => { + result.current.setTriggerStatus('rapid-node', 'enabled') + result.current.setTriggerStatus('rapid-node', 'disabled') + result.current.setTriggerStatus('rapid-node', 'enabled') + }) + + expect(result.current.getTriggerStatus('rapid-node')).toBe('enabled') + }) + + it('should handle setting the same status multiple times', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + act(() => { + result.current.setTriggerStatus('same-node', 'enabled') + }) + + const firstSnapshot = result.current.triggerStatuses + + act(() => { + result.current.setTriggerStatus('same-node', 'enabled') + }) + + const secondSnapshot = result.current.triggerStatuses + + expect(result.current.getTriggerStatus('same-node')).toBe('enabled') + // Should still create new reference (Zustand behavior) + expect(firstSnapshot).not.toBe(secondSnapshot) + }) + + it('should handle special node ID formats', () => { + const { result } = renderHook(() => useTriggerStatusStore()) + + const specialNodeIds = [ + 'node-with-dashes', + 'node_with_underscores', + 'nodeWithCamelCase', + 'node123', + 'node-123-abc', + ] + + act(() => { + specialNodeIds.forEach((nodeId, index) => { + const status = index % 2 === 0 ? 'enabled' : 'disabled' + result.current.setTriggerStatus(nodeId, status as EntryNodeStatus) + }) + }) + + specialNodeIds.forEach((nodeId, index) => { + const expectedStatus = index % 2 === 0 ? 'enabled' : 'disabled' + expect(result.current.getTriggerStatus(nodeId)).toBe(expectedStatus) + }) + }) + }) +}) diff --git a/web/app/components/workflow/store/index.ts b/web/app/components/workflow/store/index.ts index 61cd5773ce..5ca06d2ec3 100644 --- a/web/app/components/workflow/store/index.ts +++ b/web/app/components/workflow/store/index.ts @@ -1 +1,2 @@ export * from './workflow' +export * from './trigger-status' diff --git a/web/app/components/workflow/store/trigger-status.ts b/web/app/components/workflow/store/trigger-status.ts new file mode 100644 index 0000000000..2f472c79b9 --- /dev/null +++ b/web/app/components/workflow/store/trigger-status.ts @@ -0,0 +1,42 @@ +import { create } from 'zustand' +import { subscribeWithSelector } from 'zustand/middleware' + +export type EntryNodeStatus = 'enabled' | 'disabled' + +type TriggerStatusState = { + // Map of nodeId to trigger status + triggerStatuses: Record<string, EntryNodeStatus> + + // Actions + setTriggerStatus: (nodeId: string, status: EntryNodeStatus) => void + setTriggerStatuses: (statuses: Record<string, EntryNodeStatus>) => void + getTriggerStatus: (nodeId: string) => EntryNodeStatus + clearTriggerStatuses: () => void +} + +export const useTriggerStatusStore = create<TriggerStatusState>()( + subscribeWithSelector((set, get) => ({ + triggerStatuses: {}, + + setTriggerStatus: (nodeId: string, status: EntryNodeStatus) => { + set(state => ({ + triggerStatuses: { + ...state.triggerStatuses, + [nodeId]: status, + }, + })) + }, + + setTriggerStatuses: (statuses: Record<string, EntryNodeStatus>) => { + set({ triggerStatuses: statuses }) + }, + + getTriggerStatus: (nodeId: string): EntryNodeStatus => { + return get().triggerStatuses[nodeId] || 'disabled' + }, + + clearTriggerStatuses: () => { + set({ triggerStatuses: {} }) + }, + })), +) diff --git a/web/app/components/workflow/store/workflow/chat-variable-slice.ts b/web/app/components/workflow/store/workflow/chat-variable-slice.ts index 0d81446005..96fe8b00b8 100644 --- a/web/app/components/workflow/store/workflow/chat-variable-slice.ts +++ b/web/app/components/workflow/store/workflow/chat-variable-slice.ts @@ -20,7 +20,12 @@ export const createChatVariableSlice: StateCreator<ChatVariableSliceShape> = (se return ({ showChatVariablePanel: false, - setShowChatVariablePanel: showChatVariablePanel => set(() => ({ showChatVariablePanel })), + setShowChatVariablePanel: showChatVariablePanel => set(() => { + if (showChatVariablePanel) + return { ...hideAllPanel, showChatVariablePanel: true } + else + return { showChatVariablePanel: false } + }), showGlobalVariablePanel: false, setShowGlobalVariablePanel: showGlobalVariablePanel => set(() => { if (showGlobalVariablePanel) diff --git a/web/app/components/workflow/store/workflow/env-variable-slice.ts b/web/app/components/workflow/store/workflow/env-variable-slice.ts index de60e7dd5f..2ba6ce084a 100644 --- a/web/app/components/workflow/store/workflow/env-variable-slice.ts +++ b/web/app/components/workflow/store/workflow/env-variable-slice.ts @@ -10,11 +10,24 @@ export type EnvVariableSliceShape = { setEnvSecrets: (envSecrets: Record<string, string>) => void } -export const createEnvVariableSlice: StateCreator<EnvVariableSliceShape> = set => ({ - showEnvPanel: false, - setShowEnvPanel: showEnvPanel => set(() => ({ showEnvPanel })), - environmentVariables: [], - setEnvironmentVariables: environmentVariables => set(() => ({ environmentVariables })), - envSecrets: {}, - setEnvSecrets: envSecrets => set(() => ({ envSecrets })), -}) +export const createEnvVariableSlice: StateCreator<EnvVariableSliceShape> = (set) => { + const hideAllPanel = { + showDebugAndPreviewPanel: false, + showEnvPanel: false, + showChatVariablePanel: false, + showGlobalVariablePanel: false, + } + return ({ + showEnvPanel: false, + setShowEnvPanel: showEnvPanel => set(() => { + if (showEnvPanel) + return { ...hideAllPanel, showEnvPanel: true } + else + return { showEnvPanel: false } + }), + environmentVariables: [], + setEnvironmentVariables: environmentVariables => set(() => ({ environmentVariables })), + envSecrets: {}, + setEnvSecrets: envSecrets => set(() => ({ envSecrets })), + }) +} diff --git a/web/app/components/workflow/store/workflow/node-slice.ts b/web/app/components/workflow/store/workflow/node-slice.ts index 2068ee0ba1..3463fdee57 100644 --- a/web/app/components/workflow/store/workflow/node-slice.ts +++ b/web/app/components/workflow/store/workflow/node-slice.ts @@ -48,6 +48,11 @@ export type NodeSliceShape = { setLoopTimes: (loopTimes: number) => void iterParallelLogMap: Map<string, Map<string, NodeTracing[]>> setIterParallelLogMap: (iterParallelLogMap: Map<string, Map<string, NodeTracing[]>>) => void + pendingSingleRun?: { + nodeId: string + action: 'run' | 'stop' + } + setPendingSingleRun: (payload?: NodeSliceShape['pendingSingleRun']) => void } export const createNodeSlice: StateCreator<NodeSliceShape> = set => ({ @@ -73,4 +78,6 @@ export const createNodeSlice: StateCreator<NodeSliceShape> = set => ({ setLoopTimes: loopTimes => set(() => ({ loopTimes })), iterParallelLogMap: new Map<string, Map<string, NodeTracing[]>>(), setIterParallelLogMap: iterParallelLogMap => set(() => ({ iterParallelLogMap })), + pendingSingleRun: undefined, + setPendingSingleRun: payload => set(() => ({ pendingSingleRun: payload })), }) diff --git a/web/app/components/workflow/store/workflow/tool-slice.ts b/web/app/components/workflow/store/workflow/tool-slice.ts index d6d89abcf0..d5ff7743be 100644 --- a/web/app/components/workflow/store/workflow/tool-slice.ts +++ b/web/app/components/workflow/store/workflow/tool-slice.ts @@ -1,30 +1,24 @@ import type { StateCreator } from 'zustand' -import type { - ToolWithProvider, -} from '@/app/components/workflow/types' +import type { ToolWithProvider } from '../../types' export type ToolSliceShape = { - buildInTools: ToolWithProvider[] - setBuildInTools: (tools: ToolWithProvider[]) => void - customTools: ToolWithProvider[] - setCustomTools: (tools: ToolWithProvider[]) => void - workflowTools: ToolWithProvider[] - setWorkflowTools: (tools: ToolWithProvider[]) => void - mcpTools: ToolWithProvider[] - setMcpTools: (tools: ToolWithProvider[]) => void toolPublished: boolean setToolPublished: (toolPublished: boolean) => void + lastPublishedHasUserInput: boolean + setLastPublishedHasUserInput: (hasUserInput: boolean) => void + buildInTools?: ToolWithProvider[] + customTools?: ToolWithProvider[] + workflowTools?: ToolWithProvider[] + mcpTools?: ToolWithProvider[] } export const createToolSlice: StateCreator<ToolSliceShape> = set => ({ - buildInTools: [], - setBuildInTools: buildInTools => set(() => ({ buildInTools })), - customTools: [], - setCustomTools: customTools => set(() => ({ customTools })), - workflowTools: [], - setWorkflowTools: workflowTools => set(() => ({ workflowTools })), - mcpTools: [], - setMcpTools: mcpTools => set(() => ({ mcpTools })), toolPublished: false, setToolPublished: toolPublished => set(() => ({ toolPublished })), + lastPublishedHasUserInput: false, + setLastPublishedHasUserInput: hasUserInput => set(() => ({ lastPublishedHasUserInput: hasUserInput })), + buildInTools: undefined, + customTools: undefined, + workflowTools: undefined, + mcpTools: undefined, }) diff --git a/web/app/components/workflow/store/workflow/workflow-draft-slice.ts b/web/app/components/workflow/store/workflow/workflow-draft-slice.ts index a4048a9455..cae716dd52 100644 --- a/web/app/components/workflow/store/workflow/workflow-draft-slice.ts +++ b/web/app/components/workflow/store/workflow/workflow-draft-slice.ts @@ -21,6 +21,8 @@ export type WorkflowDraftSliceShape = { setSyncWorkflowDraftHash: (hash: string) => void isSyncingWorkflowDraft: boolean setIsSyncingWorkflowDraft: (isSyncingWorkflowDraft: boolean) => void + isWorkflowDataLoaded: boolean + setIsWorkflowDataLoaded: (loaded: boolean) => void } export const createWorkflowDraftSlice: StateCreator<WorkflowDraftSliceShape> = set => ({ @@ -33,4 +35,6 @@ export const createWorkflowDraftSlice: StateCreator<WorkflowDraftSliceShape> = s setSyncWorkflowDraftHash: syncWorkflowDraftHash => set(() => ({ syncWorkflowDraftHash })), isSyncingWorkflowDraft: false, setIsSyncingWorkflowDraft: isSyncingWorkflowDraft => set(() => ({ isSyncingWorkflowDraft })), + isWorkflowDataLoaded: false, + setIsWorkflowDataLoaded: loaded => set(() => ({ isWorkflowDataLoaded: loaded })), }) diff --git a/web/app/components/workflow/store/workflow/workflow-slice.ts b/web/app/components/workflow/store/workflow/workflow-slice.ts index 91dac42adb..35eeff07a7 100644 --- a/web/app/components/workflow/store/workflow/workflow-slice.ts +++ b/web/app/components/workflow/store/workflow/workflow-slice.ts @@ -1,6 +1,7 @@ import type { StateCreator } from 'zustand' import type { Node, + TriggerNodeType, WorkflowRunningData, } from '@/app/components/workflow/types' import type { FileUploadConfigResponse } from '@/models/common' @@ -13,6 +14,16 @@ type PreviewRunningData = WorkflowRunningData & { export type WorkflowSliceShape = { workflowRunningData?: PreviewRunningData setWorkflowRunningData: (workflowData: PreviewRunningData) => void + isListening: boolean + setIsListening: (listening: boolean) => void + listeningTriggerType: TriggerNodeType | null + setListeningTriggerType: (triggerType: TriggerNodeType | null) => void + listeningTriggerNodeId: string | null + setListeningTriggerNodeId: (nodeId: string | null) => void + listeningTriggerNodeIds: string[] + setListeningTriggerNodeIds: (nodeIds: string[]) => void + listeningTriggerIsAll: boolean + setListeningTriggerIsAll: (isAll: boolean) => void clipboardElements: Node[] setClipboardElements: (clipboardElements: Node[]) => void selection: null | { x1: number; y1: number; x2: number; y2: number } @@ -36,6 +47,16 @@ export type WorkflowSliceShape = { export const createWorkflowSlice: StateCreator<WorkflowSliceShape> = set => ({ workflowRunningData: undefined, setWorkflowRunningData: workflowRunningData => set(() => ({ workflowRunningData })), + isListening: false, + setIsListening: listening => set(() => ({ isListening: listening })), + listeningTriggerType: null, + setListeningTriggerType: triggerType => set(() => ({ listeningTriggerType: triggerType })), + listeningTriggerNodeId: null, + setListeningTriggerNodeId: nodeId => set(() => ({ listeningTriggerNodeId: nodeId })), + listeningTriggerNodeIds: [], + setListeningTriggerNodeIds: nodeIds => set(() => ({ listeningTriggerNodeIds: nodeIds })), + listeningTriggerIsAll: false, + setListeningTriggerIsAll: isAll => set(() => ({ listeningTriggerIsAll: isAll })), clipboardElements: [], setClipboardElements: clipboardElements => set(() => ({ clipboardElements })), selection: null, diff --git a/web/app/components/workflow/types.ts b/web/app/components/workflow/types.ts index f6a706a982..5ae8d530a8 100644 --- a/web/app/components/workflow/types.ts +++ b/web/app/components/workflow/types.ts @@ -5,10 +5,7 @@ import type { XYPosition, } from 'reactflow' import type { Resolution, TransferMethod } from '@/types/app' -import type { - DataSourceDefaultValue, - ToolDefaultValue, -} from '@/app/components/workflow/block-selector/types' +import type { PluginDefaultValue } from '@/app/components/workflow/block-selector/types' import type { VarType as VarKindType } from '@/app/components/workflow/nodes/tool/types' import type { FileResponse, NodeTracing, PanelProps } from '@/types/workflow' import type { Collection, Tool } from '@/app/components/tools/types' @@ -19,7 +16,7 @@ import type { } from '@/app/components/workflow/nodes/_base/components/error-handle/types' import type { WorkflowRetryConfig } from '@/app/components/workflow/nodes/_base/components/retry/types' import type { StructuredOutput } from '@/app/components/workflow/nodes/llm/types' -import type { PluginMeta } from '../plugins/types' +import type { Plugin, PluginMeta } from '@/app/components/plugins/types' import type { BlockClassificationEnum } from '@/app/components/workflow/block-selector/types' import type { SchemaTypeDefinition } from '@/service/use-common' @@ -50,6 +47,9 @@ export enum BlockEnum { DataSource = 'datasource', DataSourceEmpty = 'datasource-empty', KnowledgeBase = 'knowledge-index', + TriggerSchedule = 'trigger-schedule', + TriggerWebhook = 'trigger-webhook', + TriggerPlugin = 'trigger-plugin', } export enum ControlMode { @@ -103,9 +103,11 @@ export type CommonNodeType<T = {}> = { retry_config?: WorkflowRetryConfig default_value?: DefaultValueForm[] credential_id?: string + subscription_id?: string + provider_id?: string _dimmed?: boolean -} & T & Partial<Pick<ToolDefaultValue, 'provider_id' | 'provider_type' | 'provider_name' | 'tool_name'>> - & Partial<Pick<DataSourceDefaultValue, 'plugin_id' | 'provider_type' | 'provider_name' | 'datasource_name'>> + _pluginInstallLocked?: boolean +} & T & Partial<PluginDefaultValue> export type CommonEdgeType = { _hovering?: boolean @@ -176,7 +178,7 @@ export type ConversationVariable = { export type GlobalVariable = { name: string - value_type: 'string' | 'number' + value_type: 'string' | 'number' | 'integer' description: string } @@ -341,7 +343,7 @@ export type NodeDefault<T = {}> = { }) => Var[] } -export type OnSelectBlock = (type: BlockEnum, toolDefaultValue?: ToolDefaultValue | DataSourceDefaultValue) => void +export type OnSelectBlock = (type: BlockEnum, pluginDefaultValue?: PluginDefaultValue) => void export enum WorkflowRunningStatus { Waiting = 'waiting', @@ -359,6 +361,7 @@ export enum WorkflowVersion { export enum NodeRunningStatus { NotStart = 'not-start', Waiting = 'waiting', + Listening = 'listening', Running = 'running', Succeeded = 'succeeded', Failed = 'failed', @@ -372,7 +375,7 @@ export type OnNodeAdd = ( nodeType: BlockEnum sourceHandle?: string targetHandle?: string - toolDefaultValue?: ToolDefaultValue | DataSourceDefaultValue + pluginDefaultValue?: PluginDefaultValue }, oldNodesPayload: { prevNodeId?: string @@ -449,18 +452,12 @@ export type MoreInfo = { export type ToolWithProvider = Collection & { tools: Tool[] meta: PluginMeta -} - -export type UninstalledRecommendedPlugin = { - plugin_id: string - name: string - icon: string - plugin_unique_identifier: string + plugin_unique_identifier?: string } export type RAGRecommendedPlugins = { installed_recommended_plugins: ToolWithProvider[] - uninstalled_recommended_plugins: UninstalledRecommendedPlugin[] + uninstalled_recommended_plugins: Plugin[] } export enum SupportUploadFileTypes { @@ -500,3 +497,23 @@ export enum VersionHistoryContextMenuOptions { export type ChildNodeTypeCount = { [key: string]: number; } + +export const TRIGGER_NODE_TYPES = [ + BlockEnum.TriggerSchedule, + BlockEnum.TriggerWebhook, + BlockEnum.TriggerPlugin, +] as const + +// Type-safe trigger node type extracted from TRIGGER_NODE_TYPES array +export type TriggerNodeType = typeof TRIGGER_NODE_TYPES[number] + +export function isTriggerNode(nodeType: BlockEnum): boolean { + return TRIGGER_NODE_TYPES.includes(nodeType as any) +} + +export type Block = { + classification?: string + type: BlockEnum + title: string + description?: string +} diff --git a/web/app/components/workflow/utils/layout.ts b/web/app/components/workflow/utils/elk-layout.ts similarity index 97% rename from web/app/components/workflow/utils/layout.ts rename to web/app/components/workflow/utils/elk-layout.ts index b3cf3b0d88..69acbf9aff 100644 --- a/web/app/components/workflow/utils/layout.ts +++ b/web/app/components/workflow/utils/elk-layout.ts @@ -4,18 +4,18 @@ import { cloneDeep } from 'lodash-es' import type { Edge, Node, -} from '../types' +} from '@/app/components/workflow/types' import { BlockEnum, -} from '../types' +} from '@/app/components/workflow/types' import { CUSTOM_NODE, NODE_LAYOUT_HORIZONTAL_PADDING, NODE_LAYOUT_VERTICAL_PADDING, -} from '../constants' -import { CUSTOM_ITERATION_START_NODE } from '../nodes/iteration-start/constants' -import { CUSTOM_LOOP_START_NODE } from '../nodes/loop-start/constants' -import type { CaseItem, IfElseNodeType } from '../nodes/if-else/types' +} from '@/app/components/workflow/constants' +import { CUSTOM_ITERATION_START_NODE } from '@/app/components/workflow/nodes/iteration-start/constants' +import { CUSTOM_LOOP_START_NODE } from '@/app/components/workflow/nodes/loop-start/constants' +import type { CaseItem, IfElseNodeType } from '@/app/components/workflow/nodes/if-else/types' // Although the file name refers to Dagre, the implementation now relies on ELK's layered algorithm. // Keep the export signatures unchanged to minimise the blast radius while we migrate the layout stack. diff --git a/web/app/components/workflow/utils/index.ts b/web/app/components/workflow/utils/index.ts index e9ae2d1ef0..53a423de34 100644 --- a/web/app/components/workflow/utils/index.ts +++ b/web/app/components/workflow/utils/index.ts @@ -1,7 +1,7 @@ export * from './node' export * from './edge' export * from './workflow-init' -export * from './layout' +export * from './elk-layout' export * from './common' export * from './tool' export * from './workflow' diff --git a/web/app/components/workflow/utils/node-navigation.ts b/web/app/components/workflow/utils/node-navigation.ts index 5522764949..57106ae6ee 100644 --- a/web/app/components/workflow/utils/node-navigation.ts +++ b/web/app/components/workflow/utils/node-navigation.ts @@ -97,13 +97,14 @@ export function setupScrollToNodeListener( const node = nodes.find(n => n.id === nodeId) if (node) { // Use ReactFlow's fitView API to scroll to the node - reactflow.fitView({ - nodes: [node], - padding: 0.2, - duration: 800, - minZoom: 0.5, - maxZoom: 1, - }) + const nodePosition = { x: node.position.x, y: node.position.y } + + // Calculate position to place node in top-left area + // Move the center point right and down to show node in top-left + const targetX = nodePosition.x + window.innerWidth * 0.25 + const targetY = nodePosition.y + window.innerHeight * 0.25 + + reactflow.setCenter(targetX, targetY, { zoom: 1, duration: 800 }) } } } diff --git a/web/app/components/workflow/utils/trigger.ts b/web/app/components/workflow/utils/trigger.ts new file mode 100644 index 0000000000..f6d197c69c --- /dev/null +++ b/web/app/components/workflow/utils/trigger.ts @@ -0,0 +1,52 @@ +import type { TriggerWithProvider } from '@/app/components/workflow/block-selector/types' +import type { PluginTriggerNodeType } from '@/app/components/workflow/nodes/trigger-plugin/types' + +export type TriggerCheckParams = { + triggerInputsSchema: Array<{ + variable: string + label: string + required?: boolean + }> + isReadyForCheckValid: boolean +} + +export const getTriggerCheckParams = ( + triggerData: PluginTriggerNodeType, + triggerProviders: TriggerWithProvider[] | undefined, + language: string, +): TriggerCheckParams => { + if (!triggerProviders) { + return { + triggerInputsSchema: [], + isReadyForCheckValid: false, + } + } + + const { + provider_id, + provider_name, + event_name, + } = triggerData + + const provider = triggerProviders.find(item => + item.name === provider_name + || item.id === provider_id + || (provider_id && item.plugin_id === provider_id), + ) + + const currentEvent = provider?.events.find(event => event.name === event_name) + + const triggerInputsSchema = (currentEvent?.parameters || []).map((parameter) => { + const label = parameter.label?.[language] || parameter.label?.en_US || parameter.name + return { + variable: parameter.name, + label, + required: parameter.required, + } + }) + + return { + triggerInputsSchema, + isReadyForCheckValid: true, + } +} diff --git a/web/app/components/workflow/utils/workflow-entry.ts b/web/app/components/workflow/utils/workflow-entry.ts new file mode 100644 index 0000000000..724a68a85b --- /dev/null +++ b/web/app/components/workflow/utils/workflow-entry.ts @@ -0,0 +1,26 @@ +import { BlockEnum, type Node, isTriggerNode } from '../types' + +/** + * Get the workflow entry node + * Priority: trigger nodes > start node + */ +export function getWorkflowEntryNode(nodes: Node[]): Node | undefined { + const triggerNode = nodes.find(node => isTriggerNode(node.data.type)) + if (triggerNode) return triggerNode + + return nodes.find(node => node.data.type === BlockEnum.Start) +} + +/** + * Check if a node type is a workflow entry node + */ +export function isWorkflowEntryNode(nodeType: BlockEnum): boolean { + return nodeType === BlockEnum.Start || isTriggerNode(nodeType) +} + +/** + * Check if workflow is in trigger mode + */ +export function isTriggerWorkflow(nodes: Node[]): boolean { + return nodes.some(node => isTriggerNode(node.data.type)) +} diff --git a/web/app/components/workflow/utils/workflow.ts b/web/app/components/workflow/utils/workflow.ts index 48cb819086..14b1eb87d5 100644 --- a/web/app/components/workflow/utils/workflow.ts +++ b/web/app/components/workflow/utils/workflow.ts @@ -34,6 +34,9 @@ export const canRunBySingle = (nodeType: BlockEnum, isChildNode: boolean) => { || nodeType === BlockEnum.VariableAggregator || nodeType === BlockEnum.Assigner || nodeType === BlockEnum.DataSource + || nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin } export const isSupportCustomRunForm = (nodeType: BlockEnum) => { @@ -92,18 +95,29 @@ export const getNodesConnectedSourceOrTargetHandleIdsMap = (changes: ConnectedSo return nodesConnectedSourceOrTargetHandleIdsMap } -export const getValidTreeNodes = (startNode: Node, nodes: Node[], edges: Edge[]) => { - if (!startNode) { +export const getValidTreeNodes = (nodes: Node[], edges: Edge[]) => { + // Find all start nodes (Start and Trigger nodes) + const startNodes = nodes.filter(node => + node.data.type === BlockEnum.Start + || node.data.type === BlockEnum.TriggerSchedule + || node.data.type === BlockEnum.TriggerWebhook + || node.data.type === BlockEnum.TriggerPlugin, + ) + + if (startNodes.length === 0) { return { validNodes: [], maxDepth: 0, } } - const list: Node[] = [startNode] - let maxDepth = 1 + const list: Node[] = [] + let maxDepth = 0 const traverse = (root: Node, depth: number) => { + // Add the current node to the list + list.push(root) + if (depth > maxDepth) maxDepth = depth @@ -111,19 +125,19 @@ export const getValidTreeNodes = (startNode: Node, nodes: Node[], edges: Edge[]) if (outgoers.length) { outgoers.forEach((outgoer) => { - list.push(outgoer) + // Only traverse if we haven't processed this node yet (avoid cycles) + if (!list.find(n => n.id === outgoer.id)) { + if (outgoer.data.type === BlockEnum.Iteration) + list.push(...nodes.filter(node => node.parentId === outgoer.id)) + if (outgoer.data.type === BlockEnum.Loop) + list.push(...nodes.filter(node => node.parentId === outgoer.id)) - if (outgoer.data.type === BlockEnum.Iteration) - list.push(...nodes.filter(node => node.parentId === outgoer.id)) - if (outgoer.data.type === BlockEnum.Loop) - list.push(...nodes.filter(node => node.parentId === outgoer.id)) - - traverse(outgoer, depth + 1) + traverse(outgoer, depth + 1) + } }) } else { - list.push(root) - + // Leaf node - add iteration/loop children if any if (root.data.type === BlockEnum.Iteration) list.push(...nodes.filter(node => node.parentId === root.id)) if (root.data.type === BlockEnum.Loop) @@ -131,7 +145,11 @@ export const getValidTreeNodes = (startNode: Node, nodes: Node[], edges: Edge[]) } } - traverse(startNode, maxDepth) + // Start traversal from all start nodes + startNodes.forEach((startNode) => { + if (!list.find(n => n.id === startNode.id)) + traverse(startNode, 1) + }) return { validNodes: uniqBy(list, 'id'), diff --git a/web/app/components/workflow/variable-inspect/listening.tsx b/web/app/components/workflow/variable-inspect/listening.tsx new file mode 100644 index 0000000000..1f2577f150 --- /dev/null +++ b/web/app/components/workflow/variable-inspect/listening.tsx @@ -0,0 +1,219 @@ +import { type FC, useEffect, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { type Node, useStoreApi } from 'reactflow' +import Button from '@/app/components/base/button' +import BlockIcon from '@/app/components/workflow/block-icon' +import { BlockEnum } from '@/app/components/workflow/types' +import { StopCircle } from '@/app/components/base/icons/src/vender/line/mediaAndDevices' +import { useStore } from '../store' +import { useGetToolIcon } from '@/app/components/workflow/hooks/use-tool-icon' +import type { TFunction } from 'i18next' +import { getNextExecutionTime } from '@/app/components/workflow/nodes/trigger-schedule/utils/execution-time-calculator' +import type { ScheduleTriggerNodeType } from '@/app/components/workflow/nodes/trigger-schedule/types' +import type { WebhookTriggerNodeType } from '@/app/components/workflow/nodes/trigger-webhook/types' +import Tooltip from '@/app/components/base/tooltip' +import copy from 'copy-to-clipboard' + +const resolveListeningDescription = ( + message: string | undefined, + triggerNode: Node | undefined, + triggerType: BlockEnum, + t: TFunction, +): string => { + if (message) + return message + + if (triggerType === BlockEnum.TriggerSchedule) { + const scheduleData = triggerNode?.data as ScheduleTriggerNodeType | undefined + const nextTriggerTime = scheduleData ? getNextExecutionTime(scheduleData) : '' + return t('workflow.debug.variableInspect.listening.tipSchedule', { + nextTriggerTime: nextTriggerTime || t('workflow.debug.variableInspect.listening.defaultScheduleTime'), + }) + } + + if (triggerType === BlockEnum.TriggerPlugin) { + const pluginName = (triggerNode?.data as { provider_name?: string; title?: string })?.provider_name + || (triggerNode?.data as { title?: string })?.title + || t('workflow.debug.variableInspect.listening.defaultPluginName') + return t('workflow.debug.variableInspect.listening.tipPlugin', { pluginName }) + } + + if (triggerType === BlockEnum.TriggerWebhook) { + const nodeName = (triggerNode?.data as { title?: string })?.title || t('workflow.debug.variableInspect.listening.defaultNodeName') + return t('workflow.debug.variableInspect.listening.tip', { nodeName }) + } + + const nodeDescription = (triggerNode?.data as { desc?: string })?.desc + if (nodeDescription) + return nodeDescription + + return t('workflow.debug.variableInspect.listening.tipFallback') +} + +const resolveMultipleListeningDescription = ( + nodes: Node[], + t: TFunction, +): string => { + if (!nodes.length) + return t('workflow.debug.variableInspect.listening.tipFallback') + + const titles = nodes + .map(node => (node.data as { title?: string })?.title) + .filter((title): title is string => Boolean(title)) + + if (titles.length) + return t('workflow.debug.variableInspect.listening.tip', { nodeName: titles.join(', ') }) + + return t('workflow.debug.variableInspect.listening.tipFallback') +} + +export type ListeningProps = { + onStop: () => void + message?: string +} + +const Listening: FC<ListeningProps> = ({ + onStop, + message, +}) => { + const { t } = useTranslation() + const store = useStoreApi() + + // Get the current trigger type and node ID from store + const listeningTriggerType = useStore(s => s.listeningTriggerType) + const listeningTriggerNodeId = useStore(s => s.listeningTriggerNodeId) + const listeningTriggerNodeIds = useStore(s => s.listeningTriggerNodeIds) + const listeningTriggerIsAll = useStore(s => s.listeningTriggerIsAll) + + const getToolIcon = useGetToolIcon() + + // Get the trigger node data to extract icon information + const { getNodes } = store.getState() + const nodes = getNodes() + const triggerNode = listeningTriggerNodeId + ? nodes.find(node => node.id === listeningTriggerNodeId) + : undefined + const inferredTriggerType = (triggerNode?.data as { type?: BlockEnum })?.type + const triggerType = listeningTriggerType || inferredTriggerType || BlockEnum.TriggerWebhook + const webhookDebugUrl = triggerType === BlockEnum.TriggerWebhook + ? (triggerNode?.data as WebhookTriggerNodeType | undefined)?.webhook_debug_url + : undefined + const [debugUrlCopied, setDebugUrlCopied] = useState(false) + + useEffect(() => { + if (!debugUrlCopied) + return + + const timer = window.setTimeout(() => { + setDebugUrlCopied(false) + }, 2000) + + return () => { + window.clearTimeout(timer) + } + }, [debugUrlCopied]) + + let displayNodes: Node[] = [] + + if (listeningTriggerIsAll) { + if (listeningTriggerNodeIds.length > 0) { + displayNodes = nodes.filter(node => listeningTriggerNodeIds.includes(node.id)) + } + else { + displayNodes = nodes.filter((node) => { + const nodeType = (node.data as { type?: BlockEnum })?.type + return nodeType === BlockEnum.TriggerSchedule + || nodeType === BlockEnum.TriggerWebhook + || nodeType === BlockEnum.TriggerPlugin + }) + } + } + else if (triggerNode) { + displayNodes = [triggerNode] + } + + const iconsToRender = displayNodes.map((node) => { + const blockType = (node.data as { type?: BlockEnum })?.type || BlockEnum.TriggerWebhook + const icon = getToolIcon(node.data as any) + return { + key: node.id, + type: blockType, + toolIcon: icon, + } + }) + + if (iconsToRender.length === 0) { + iconsToRender.push({ + key: 'default', + type: listeningTriggerIsAll ? BlockEnum.TriggerWebhook : triggerType, + toolIcon: !listeningTriggerIsAll && triggerNode ? getToolIcon(triggerNode.data as any) : undefined, + }) + } + + const description = listeningTriggerIsAll + ? resolveMultipleListeningDescription(displayNodes, t) + : resolveListeningDescription(message, triggerNode, triggerType, t) + + return ( + <div className='flex h-full flex-col gap-4 rounded-xl bg-background-section p-8'> + <div className='flex flex-row flex-wrap items-center gap-3'> + {iconsToRender.map(icon => ( + <BlockIcon + key={icon.key} + type={icon.type} + toolIcon={icon.toolIcon} + size="md" + className="!h-10 !w-10 !rounded-xl [&_svg]:!h-7 [&_svg]:!w-7" + /> + ))} + </div> + <div className='flex flex-col gap-1'> + <div className='system-sm-semibold text-text-secondary'>{t('workflow.debug.variableInspect.listening.title')}</div> + <div className='system-xs-regular whitespace-pre-line text-text-tertiary'>{description}</div> + </div> + {webhookDebugUrl && ( + <div className='flex items-center gap-2'> + <div className='system-xs-regular shrink-0 whitespace-pre-line text-text-tertiary'> + {t('workflow.nodes.triggerWebhook.debugUrlTitle')} + </div> + <Tooltip + popupContent={debugUrlCopied + ? t('workflow.nodes.triggerWebhook.debugUrlCopied') + : t('workflow.nodes.triggerWebhook.debugUrlCopy')} + popupClassName="system-xs-regular text-text-primary bg-components-tooltip-bg border border-components-panel-border shadow-lg backdrop-blur-sm rounded-md px-1.5 py-1" + position="top" + offset={{ mainAxis: -4 }} + needsDelay={true} + > + <button + type='button' + aria-label={t('workflow.nodes.triggerWebhook.debugUrlCopy') || ''} + className={`inline-flex items-center rounded-[6px] border border-divider-regular bg-components-badge-white-to-dark px-1.5 py-[2px] font-mono text-[13px] leading-[18px] text-text-secondary transition-colors hover:bg-components-panel-on-panel-item-bg-hover focus:outline-none focus-visible:outline focus-visible:outline-2 focus-visible:outline-components-panel-border ${debugUrlCopied ? 'bg-components-panel-on-panel-item-bg-hover text-text-primary' : ''}`} + onClick={() => { + copy(webhookDebugUrl) + setDebugUrlCopied(true) + }} + > + <span className='whitespace-nowrap text-text-primary'> + {webhookDebugUrl} + </span> + </button> + </Tooltip> + </div> + )} + <div> + <Button + size='medium' + className='px-3' + variant='primary' + onClick={onStop} + > + <StopCircle className='mr-1 size-4' /> + {t('workflow.debug.variableInspect.listening.stopButton')} + </Button> + </div> + </div> + ) +} + +export default Listening diff --git a/web/app/components/workflow/variable-inspect/panel.tsx b/web/app/components/workflow/variable-inspect/panel.tsx index db0a6da8ab..c0ad4cd159 100644 --- a/web/app/components/workflow/variable-inspect/panel.tsx +++ b/web/app/components/workflow/variable-inspect/panel.tsx @@ -7,6 +7,7 @@ import { import { useStore } from '../store' import useCurrentVars from '../hooks/use-inspect-vars-crud' import Empty from './empty' +import Listening from './listening' import Left from './left' import Right from './right' import ActionButton from '@/app/components/base/action-button' @@ -16,6 +17,8 @@ import { VarInInspectType } from '@/types/workflow' import cn from '@/utils/classnames' import type { NodeProps } from '../types' import useMatchSchemaType from '../nodes/_base/components/variable/use-match-schema-type' +import { useEventEmitterContextContext } from '@/context/event-emitter' +import { EVENT_WORKFLOW_STOP } from '@/app/components/workflow/variable-inspect/types' export type currentVarType = { nodeId: string @@ -32,6 +35,7 @@ const Panel: FC = () => { const bottomPanelWidth = useStore(s => s.bottomPanelWidth) const setShowVariableInspectPanel = useStore(s => s.setShowVariableInspectPanel) const [showLeftPanel, setShowLeftPanel] = useState(true) + const isListening = useStore(s => s.isListening) const environmentVariables = useStore(s => s.environmentVariables) const currentFocusNodeId = useStore(s => s.currentFocusNodeId) @@ -135,6 +139,11 @@ const Panel: FC = () => { }, [setCurrentFocusNodeId, setCurrentVarId]) const { isLoading, schemaTypeDefinitions } = useMatchSchemaType() + const { eventEmitter } = useEventEmitterContextContext() + + const handleStopListening = useCallback(() => { + eventEmitter?.emit({ type: EVENT_WORKFLOW_STOP } as any) + }, [eventEmitter]) useEffect(() => { if (currentFocusNodeId && currentVarId && !isLoading) { @@ -144,6 +153,24 @@ const Panel: FC = () => { } }, [currentFocusNodeId, currentVarId, nodesWithInspectVars, fetchInspectVarValue, schemaTypeDefinitions, isLoading]) + if (isListening) { + return ( + <div className={cn('flex h-full flex-col')}> + <div className='flex shrink-0 items-center justify-between pl-4 pr-2 pt-2'> + <div className='system-sm-semibold-uppercase text-text-primary'>{t('workflow.debug.variableInspect.title')}</div> + <ActionButton onClick={() => setShowVariableInspectPanel(false)}> + <RiCloseLine className='h-4 w-4' /> + </ActionButton> + </div> + <div className='grow p-2'> + <Listening + onStop={handleStopListening} + /> + </div> + </div> + ) + } + if (isEmpty) { return ( <div className={cn('flex h-full flex-col')}> diff --git a/web/app/components/workflow/variable-inspect/right.tsx b/web/app/components/workflow/variable-inspect/right.tsx index 4e38e66269..9627a7ea43 100644 --- a/web/app/components/workflow/variable-inspect/right.tsx +++ b/web/app/components/workflow/variable-inspect/right.tsx @@ -24,7 +24,7 @@ import useNodeInfo from '../nodes/_base/hooks/use-node-info' import { useBoolean } from 'ahooks' import GetAutomaticResModal from '@/app/components/app/configuration/config/automatic/get-automatic-res' import GetCodeGeneratorResModal from '../../app/configuration/config/code-generator/get-code-generator-res' -import { AppType } from '@/types/app' +import { AppModeEnum } from '@/types/app' import { useHooksStore } from '../hooks-store' import { useCallback, useMemo } from 'react' import { useNodesInteractions, useToolIcon } from '../hooks' @@ -282,7 +282,7 @@ const Right = ({ isCodeBlock ? <GetCodeGeneratorResModal isShow - mode={AppType.chat} + mode={AppModeEnum.CHAT} onClose={handleHidePromptGenerator} flowId={configsMap?.flowId || ''} nodeId={nodeId} @@ -291,7 +291,7 @@ const Right = ({ onFinished={handleUpdatePrompt} /> : <GetAutomaticResModal - mode={AppType.chat} + mode={AppModeEnum.CHAT} isShow onClose={handleHidePromptGenerator} onFinished={handleUpdatePrompt} diff --git a/web/app/components/workflow/workflow-preview/components/node-handle.tsx b/web/app/components/workflow/workflow-preview/components/node-handle.tsx index 4ff08354be..2211e3397f 100644 --- a/web/app/components/workflow/workflow-preview/components/node-handle.tsx +++ b/web/app/components/workflow/workflow-preview/components/node-handle.tsx @@ -34,7 +34,10 @@ export const NodeTargetHandle = memo(({ 'after:absolute after:left-1.5 after:top-1 after:h-2 after:w-0.5 after:bg-workflow-link-line-handle', 'transition-all hover:scale-125', !connected && 'after:opacity-0', - data.type === BlockEnum.Start && 'opacity-0', + (data.type === BlockEnum.Start + || data.type === BlockEnum.TriggerWebhook + || data.type === BlockEnum.TriggerSchedule + || data.type === BlockEnum.TriggerPlugin) && 'opacity-0', handleClassName, )} > diff --git a/web/app/education-apply/hooks.ts b/web/app/education-apply/hooks.ts index df3ee38795..9d45a5ee69 100644 --- a/web/app/education-apply/hooks.ts +++ b/web/app/education-apply/hooks.ts @@ -20,6 +20,7 @@ import timezone from 'dayjs/plugin/timezone' import { useAppContext } from '@/context/app-context' import { useRouter } from 'next/navigation' import { useProviderContext } from '@/context/provider-context' +import { ACCOUNT_SETTING_TAB } from '@/app/components/header/account-setting/constants' dayjs.extend(utc) dayjs.extend(timezone) @@ -155,7 +156,7 @@ export const useEducationInit = () => { useEffect(() => { if (educationVerifying === 'yes' || educationVerifyAction === EDUCATION_VERIFY_URL_SEARCHPARAMS_ACTION) { - setShowAccountSettingModal({ payload: 'billing' }) + setShowAccountSettingModal({ payload: ACCOUNT_SETTING_TAB.BILLING }) if (educationVerifyAction === EDUCATION_VERIFY_URL_SEARCHPARAMS_ACTION) localStorage.setItem(EDUCATION_VERIFYING_LOCALSTORAGE_ITEM, 'yes') diff --git a/web/app/layout.tsx b/web/app/layout.tsx index 1be802460b..c83ea7fd85 100644 --- a/web/app/layout.tsx +++ b/web/app/layout.tsx @@ -57,6 +57,7 @@ const LocaleLayout = async ({ [DatasetAttr.DATA_PUBLIC_ENABLE_WEBSITE_JINAREADER]: process.env.NEXT_PUBLIC_ENABLE_WEBSITE_JINAREADER, [DatasetAttr.DATA_PUBLIC_ENABLE_WEBSITE_FIRECRAWL]: process.env.NEXT_PUBLIC_ENABLE_WEBSITE_FIRECRAWL, [DatasetAttr.DATA_PUBLIC_ENABLE_WEBSITE_WATERCRAWL]: process.env.NEXT_PUBLIC_ENABLE_WEBSITE_WATERCRAWL, + [DatasetAttr.DATA_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX]: process.env.NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX, [DatasetAttr.NEXT_PUBLIC_ZENDESK_WIDGET_KEY]: process.env.NEXT_PUBLIC_ZENDESK_WIDGET_KEY, [DatasetAttr.NEXT_PUBLIC_ZENDESK_FIELD_ID_ENVIRONMENT]: process.env.NEXT_PUBLIC_ZENDESK_FIELD_ID_ENVIRONMENT, [DatasetAttr.NEXT_PUBLIC_ZENDESK_FIELD_ID_VERSION]: process.env.NEXT_PUBLIC_ZENDESK_FIELD_ID_VERSION, diff --git a/web/app/signin/normal-form.tsx b/web/app/signin/normal-form.tsx index 920a992b4f..29e21b8ba2 100644 --- a/web/app/signin/normal-form.tsx +++ b/web/app/signin/normal-form.tsx @@ -135,8 +135,8 @@ const NormalForm = () => { {!systemFeatures.branding.enabled && <p className='body-md-regular mt-2 text-text-tertiary'>{t('login.joinTipStart')}{workspaceName}{t('login.joinTipEnd')}</p>} </div> : <div className="mx-auto w-full"> - <h2 className="title-4xl-semi-bold text-text-primary">{t('login.pageTitle')}</h2> - {!systemFeatures.branding.enabled && <p className='body-md-regular mt-2 text-text-tertiary'>{t('login.welcome')}</p>} + <h2 className="title-4xl-semi-bold text-text-primary">{systemFeatures.branding.enabled ? t('login.pageTitleForE') : t('login.pageTitle')}</h2> + <p className='body-md-regular mt-2 text-text-tertiary'>{t('login.welcome')}</p> </div>} <div className="relative"> <div className="mt-6 flex flex-col gap-3"> diff --git a/web/app/signin/utils/post-login-redirect.ts b/web/app/signin/utils/post-login-redirect.ts index 37ab122dfa..45e2c55941 100644 --- a/web/app/signin/utils/post-login-redirect.ts +++ b/web/app/signin/utils/post-login-redirect.ts @@ -1,4 +1,4 @@ -import { OAUTH_AUTHORIZE_PENDING_KEY, REDIRECT_URL_KEY } from '@/app/account/oauth/authorize/page' +import { OAUTH_AUTHORIZE_PENDING_KEY, REDIRECT_URL_KEY } from '@/app/account/oauth/authorize/constants' import dayjs from 'dayjs' import type { ReadonlyURLSearchParams } from 'next/navigation' diff --git a/web/app/styles/markdown.scss b/web/app/styles/markdown.scss index 005685f0e8..a4c24787a7 100644 --- a/web/app/styles/markdown.scss +++ b/web/app/styles/markdown.scss @@ -856,6 +856,18 @@ color: var(--color-prettylights-syntax-comment); } +.markdown-body .katex { + /* Allow long inline formulas to wrap instead of overflowing */ + white-space: normal !important; + overflow-wrap: break-word; /* better cross-browser support */ + word-break: break-word; /* non-standard fallback for older WebKit/Blink */ +} + +.markdown-body .katex-display { + /* Fallback for very long display equations */ + overflow-x: auto; +} + .markdown-body .pl-c1, .markdown-body .pl-s .pl-v { color: var(--color-prettylights-syntax-constant); diff --git a/web/assets/search-menu.svg b/web/assets/search-menu.svg new file mode 100644 index 0000000000..8f7131c2ce --- /dev/null +++ b/web/assets/search-menu.svg @@ -0,0 +1,7 @@ +<svg width="32" height="32" viewBox="0 0 32 32" fill="none" xmlns="http://www.w3.org/2000/svg"> +<path d="M28.0049 16C28.0049 20.4183 24.4231 24 20.0049 24C15.5866 24 12.0049 20.4183 12.0049 16C12.0049 11.5817 15.5866 8 20.0049 8C24.4231 8 28.0049 11.5817 28.0049 16Z" stroke="#676F83" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> +<path d="M4.00488 16H6.67155" stroke="#676F83" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> +<path d="M4.00488 9.33301H8.00488" stroke="#676F83" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> +<path d="M4.00488 22.667H8.00488" stroke="#676F83" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> +<path d="M26 22L29.3333 25.3333" stroke="#676F83" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> +</svg> diff --git a/web/config/index.ts b/web/config/index.ts index 0e876b800e..7b2b9e1084 100644 --- a/web/config/index.ts +++ b/web/config/index.ts @@ -144,7 +144,9 @@ export const getMaxToken = (modelId: string) => { export const LOCALE_COOKIE_NAME = 'locale' +const COOKIE_DOMAIN = (process.env.NEXT_PUBLIC_COOKIE_DOMAIN || '').trim() export const CSRF_COOKIE_NAME = () => { + if (COOKIE_DOMAIN) return 'csrf_token' const isSecure = API_PREFIX.startsWith('https://') return isSecure ? '__Host-csrf_token' : 'csrf_token' } @@ -375,6 +377,11 @@ export const ENABLE_WEBSITE_WATERCRAWL = getBooleanConfig( DatasetAttr.DATA_PUBLIC_ENABLE_WEBSITE_WATERCRAWL, false, ) +export const ENABLE_SINGLE_DOLLAR_LATEX = getBooleanConfig( + process.env.NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX, + DatasetAttr.DATA_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX, + false, +) export const VALUE_SELECTOR_DELIMITER = '@@@' @@ -414,6 +421,8 @@ export const ZENDESK_FIELD_IDS = { } export const APP_VERSION = pkg.version +export const IS_MARKETPLACE = globalThis.document?.body?.getAttribute('data-is-marketplace') === 'true' + export const RAG_PIPELINE_PREVIEW_CHUNK_NUM = 20 export const PROVIDER_WITH_PRESET_TONE = ['langgenius/openai/openai', 'langgenius/azure_openai/azure_openai'] diff --git a/web/context/app-context.tsx b/web/context/app-context.tsx index 9fd6481440..644a7a778f 100644 --- a/web/context/app-context.tsx +++ b/web/context/app-context.tsx @@ -10,6 +10,7 @@ import MaintenanceNotice from '@/app/components/header/maintenance-notice' import { noop } from 'lodash-es' import { setZendeskConversationFields } from '@/app/components/base/zendesk/utils' import { ZENDESK_FIELD_IDS } from '@/config' +import { useGlobalPublicStore } from './global-public-context' export type AppContextValue = { userProfile: UserProfileResponse @@ -77,6 +78,7 @@ export type AppContextProviderProps = { } export const AppContextProvider: FC<AppContextProviderProps> = ({ children }) => { + const systemFeatures = useGlobalPublicStore(s => s.systemFeatures) const { data: userProfileResponse, mutate: mutateUserProfile, error: userProfileError } = useSWR({ url: '/account/profile', params: {} }, fetchUserProfile) const { data: currentWorkspaceResponse, mutate: mutateCurrentWorkspace, isLoading: isLoadingCurrentWorkspace } = useSWR({ url: '/workspaces/current', params: {} }, fetchCurrentWorkspace) @@ -92,10 +94,12 @@ export const AppContextProvider: FC<AppContextProviderProps> = ({ children }) => try { const result = await userProfileResponse.json() setUserProfile(result) - const current_version = userProfileResponse.headers.get('x-version') - const current_env = process.env.NODE_ENV === 'development' ? 'DEVELOPMENT' : userProfileResponse.headers.get('x-env') - const versionData = await fetchLangGeniusVersion({ url: '/version', params: { current_version } }) - setLangGeniusVersionInfo({ ...versionData, current_version, latest_version: versionData.version, current_env }) + if (!systemFeatures.branding.enabled) { + const current_version = userProfileResponse.headers.get('x-version') + const current_env = process.env.NODE_ENV === 'development' ? 'DEVELOPMENT' : userProfileResponse.headers.get('x-env') + const versionData = await fetchLangGeniusVersion({ url: '/version', params: { current_version } }) + setLangGeniusVersionInfo({ ...versionData, current_version, latest_version: versionData.version, current_env }) + } } catch (error) { console.error('Failed to update user profile:', error) diff --git a/web/context/debug-configuration.ts b/web/context/debug-configuration.ts index 3ec82ee8ef..f77e0656d3 100644 --- a/web/context/debug-configuration.ts +++ b/web/context/debug-configuration.ts @@ -22,6 +22,7 @@ import type { import type { ExternalDataTool } from '@/models/common' import type { DataSet } from '@/models/datasets' import type { VisionSettings } from '@/types/app' +import { AppModeEnum } from '@/types/app' import { ModelModeType, RETRIEVE_TYPE, Resolution, TransferMethod } from '@/types/app' import { ANNOTATION_DEFAULT, DEFAULT_AGENT_SETTING, DEFAULT_CHAT_PROMPT_CONFIG, DEFAULT_COMPLETION_PROMPT_CONFIG } from '@/config' import type { FormValue } from '@/app/components/header/account-setting/model-provider-page/declarations' @@ -33,7 +34,7 @@ type IDebugConfiguration = { appId: string isAPIKeySet: boolean isTrailFinished: boolean - mode: string + mode: AppModeEnum modelModeType: ModelModeType promptMode: PromptMode setPromptMode: (promptMode: PromptMode) => void @@ -113,7 +114,7 @@ const DebugConfigurationContext = createContext<IDebugConfiguration>({ appId: '', isAPIKeySet: false, isTrailFinished: false, - mode: '', + mode: AppModeEnum.CHAT, modelModeType: ModelModeType.chat, promptMode: PromptMode.simple, setPromptMode: noop, @@ -212,6 +213,8 @@ const DebugConfigurationContext = createContext<IDebugConfiguration>({ prompt_template: '', prompt_variables: [], }, + chat_prompt_config: DEFAULT_CHAT_PROMPT_CONFIG, + completion_prompt_config: DEFAULT_COMPLETION_PROMPT_CONFIG, more_like_this: null, opening_statement: '', suggested_questions: [], @@ -222,6 +225,14 @@ const DebugConfigurationContext = createContext<IDebugConfiguration>({ suggested_questions_after_answer: null, retriever_resource: null, annotation_reply: null, + external_data_tools: [], + system_parameters: { + audio_file_size_limit: 0, + file_size_limit: 0, + image_file_size_limit: 0, + video_file_size_limit: 0, + workflow_file_upload_limit: 0, + }, dataSets: [], agentConfig: DEFAULT_AGENT_SETTING, }, diff --git a/web/context/modal-context.tsx b/web/context/modal-context.tsx index 5baadc934b..e0228b8ca8 100644 --- a/web/context/modal-context.tsx +++ b/web/context/modal-context.tsx @@ -1,9 +1,9 @@ 'use client' import type { Dispatch, SetStateAction } from 'react' -import { useCallback, useState } from 'react' +import { useCallback, useEffect, useState } from 'react' import { createContext, useContext, useContextSelector } from 'use-context-selector' -import { useRouter, useSearchParams } from 'next/navigation' +import { useSearchParams } from 'next/navigation' import type { ConfigurationMethodEnum, Credential, @@ -12,8 +12,15 @@ import type { ModelProvider, } from '@/app/components/header/account-setting/model-provider-page/declarations' import { + EDUCATION_PRICING_SHOW_ACTION, EDUCATION_VERIFYING_LOCALSTORAGE_ITEM, } from '@/app/education-apply/constants' +import type { AccountSettingTab } from '@/app/components/header/account-setting/constants' +import { + ACCOUNT_SETTING_MODAL_ACTION, + DEFAULT_ACCOUNT_SETTING_TAB, + isValidAccountSettingTab, +} from '@/app/components/header/account-setting/constants' import type { ModerationConfig, PromptVariable } from '@/models/debug' import type { ApiBasedExtension, @@ -90,7 +97,7 @@ export type ModelModalType = { } export type ModalContextState = { - setShowAccountSettingModal: Dispatch<SetStateAction<ModalState<string> | null>> + setShowAccountSettingModal: Dispatch<SetStateAction<ModalState<AccountSettingTab> | null>> setShowApiBasedExtensionModal: Dispatch<SetStateAction<ModalState<ApiBasedExtension> | null>> setShowModerationSettingModal: Dispatch<SetStateAction<ModalState<ModerationConfig> | null>> setShowExternalDataToolModal: Dispatch<SetStateAction<ModalState<ExternalDataTool> | null>> @@ -107,6 +114,9 @@ export type ModalContextState = { setShowUpdatePluginModal: Dispatch<SetStateAction<ModalState<UpdatePluginPayload> | null>> setShowEducationExpireNoticeModal: Dispatch<SetStateAction<ModalState<ExpireNoticeModalPayloadProps> | null>> } +const PRICING_MODAL_QUERY_PARAM = 'pricing' +const PRICING_MODAL_QUERY_VALUE = 'open' + const ModalContext = createContext<ModalContextState>({ setShowAccountSettingModal: noop, setShowApiBasedExtensionModal: noop, @@ -135,7 +145,16 @@ type ModalContextProviderProps = { export const ModalContextProvider = ({ children, }: ModalContextProviderProps) => { - const [showAccountSettingModal, setShowAccountSettingModal] = useState<ModalState<string> | null>(null) + const searchParams = useSearchParams() + + const [showAccountSettingModal, setShowAccountSettingModal] = useState<ModalState<AccountSettingTab> | null>(() => { + if (searchParams.get('action') === ACCOUNT_SETTING_MODAL_ACTION) { + const tabParam = searchParams.get('tab') + const tab = isValidAccountSettingTab(tabParam) ? tabParam : DEFAULT_ACCOUNT_SETTING_TAB + return { payload: tab } + } + return null + }) const [showApiBasedExtensionModal, setShowApiBasedExtensionModal] = useState<ModalState<ApiBasedExtension> | null>(null) const [showModerationSettingModal, setShowModerationSettingModal] = useState<ModalState<ModerationConfig> | null>(null) const [showExternalDataToolModal, setShowExternalDataToolModal] = useState<ModalState<ExternalDataTool> | null>(null) @@ -150,9 +169,9 @@ export const ModalContextProvider = ({ const [showUpdatePluginModal, setShowUpdatePluginModal] = useState<ModalState<UpdatePluginPayload> | null>(null) const [showEducationExpireNoticeModal, setShowEducationExpireNoticeModal] = useState<ModalState<ExpireNoticeModalPayloadProps> | null>(null) - const searchParams = useSearchParams() - const router = useRouter() - const [showPricingModal, setShowPricingModal] = useState(searchParams.get('show-pricing') === '1') + const [showPricingModal, setShowPricingModal] = useState( + searchParams.get(PRICING_MODAL_QUERY_PARAM) === PRICING_MODAL_QUERY_VALUE, + ) const [showAnnotationFullModal, setShowAnnotationFullModal] = useState(false) const handleCancelAccountSettingModal = () => { const educationVerifying = localStorage.getItem(EDUCATION_VERIFYING_LOCALSTORAGE_ITEM) @@ -161,11 +180,54 @@ export const ModalContextProvider = ({ localStorage.removeItem(EDUCATION_VERIFYING_LOCALSTORAGE_ITEM) removeSpecificQueryParam('action') + removeSpecificQueryParam('tab') setShowAccountSettingModal(null) if (showAccountSettingModal?.onCancelCallback) showAccountSettingModal?.onCancelCallback() } + const handleAccountSettingTabChange = useCallback((tab: AccountSettingTab) => { + setShowAccountSettingModal((prev) => { + if (!prev) + return { payload: tab } + if (prev.payload === tab) + return prev + return { ...prev, payload: tab } + }) + }, [setShowAccountSettingModal]) + + useEffect(() => { + if (typeof window === 'undefined') + return + const url = new URL(window.location.href) + if (!showAccountSettingModal?.payload) { + if (url.searchParams.get('action') !== ACCOUNT_SETTING_MODAL_ACTION) + return + url.searchParams.delete('action') + url.searchParams.delete('tab') + window.history.replaceState(null, '', url.toString()) + return + } + url.searchParams.set('action', ACCOUNT_SETTING_MODAL_ACTION) + url.searchParams.set('tab', showAccountSettingModal.payload) + window.history.replaceState(null, '', url.toString()) + }, [showAccountSettingModal]) + + useEffect(() => { + if (typeof window === 'undefined') + return + const url = new URL(window.location.href) + if (showPricingModal) { + url.searchParams.set(PRICING_MODAL_QUERY_PARAM, PRICING_MODAL_QUERY_VALUE) + } + else { + url.searchParams.delete(PRICING_MODAL_QUERY_PARAM) + if (url.searchParams.get('action') === EDUCATION_PRICING_SHOW_ACTION) + url.searchParams.delete('action') + } + window.history.replaceState(null, '', url.toString()) + }, [showPricingModal]) + const handleCancelModerationSettingModal = () => { setShowModerationSettingModal(null) if (showModerationSettingModal?.onCancelCallback) @@ -250,13 +312,21 @@ export const ModalContextProvider = ({ setShowOpeningModal(null) } + const handleShowPricingModal = useCallback(() => { + setShowPricingModal(true) + }, []) + + const handleCancelPricingModal = useCallback(() => { + setShowPricingModal(false) + }, []) + return ( <ModalContext.Provider value={{ setShowAccountSettingModal, setShowApiBasedExtensionModal, setShowModerationSettingModal, setShowExternalDataToolModal, - setShowPricingModal: () => setShowPricingModal(true), + setShowPricingModal: handleShowPricingModal, setShowAnnotationFullModal: () => setShowAnnotationFullModal(true), setShowModelModal, setShowExternalKnowledgeAPIModal, @@ -272,6 +342,7 @@ export const ModalContextProvider = ({ <AccountSetting activeTab={showAccountSettingModal.payload} onCancel={handleCancelAccountSettingModal} + onTabChange={handleAccountSettingTabChange} /> ) } @@ -307,12 +378,7 @@ export const ModalContextProvider = ({ { !!showPricingModal && ( - <Pricing onCancel={() => { - if (searchParams.get('show-pricing') === '1') - router.push(location.pathname, { forceOptimisticNavigation: true } as any) - removeSpecificQueryParam('action') - setShowPricingModal(false) - }} /> + <Pricing onCancel={handleCancelPricingModal} /> ) } diff --git a/web/context/provider-context.tsx b/web/context/provider-context.tsx index 755131c859..90233fbc21 100644 --- a/web/context/provider-context.tsx +++ b/web/context/provider-context.tsx @@ -17,7 +17,8 @@ import { } from '@/app/components/header/account-setting/model-provider-page/declarations' import type { Model, ModelProvider } from '@/app/components/header/account-setting/model-provider-page/declarations' import type { RETRIEVE_METHOD } from '@/types/app' -import { Plan, type UsagePlanInfo } from '@/app/components/billing/type' +import type { Plan } from '@/app/components/billing/type' +import type { UsagePlanInfo } from '@/app/components/billing/type' import { fetchCurrentPlanInfo } from '@/service/billing' import { parseCurrentPlan } from '@/app/components/billing/utils' import { defaultPlan } from '@/app/components/billing/config' @@ -70,23 +71,7 @@ const ProviderContext = createContext<ProviderContextState>({ textGenerationModelList: [], supportRetrievalMethods: [], isAPIKeySet: true, - plan: { - type: Plan.sandbox, - usage: { - vectorSpace: 32, - buildApps: 12, - teamMembers: 1, - annotatedResponse: 1, - documentsUploadQuota: 50, - }, - total: { - vectorSpace: 200, - buildApps: 50, - teamMembers: 1, - annotatedResponse: 10, - documentsUploadQuota: 500, - }, - }, + plan: defaultPlan, isFetchedPlan: false, enableBilling: false, onPlanInfoChanged: noop, diff --git a/web/context/web-app-context.tsx b/web/context/web-app-context.tsx index bcbd39b5fc..1b189cd452 100644 --- a/web/context/web-app-context.tsx +++ b/web/context/web-app-context.tsx @@ -9,6 +9,7 @@ import { usePathname, useSearchParams } from 'next/navigation' import type { FC, PropsWithChildren } from 'react' import { useEffect } from 'react' import { create } from 'zustand' +import { getProcessedSystemVariablesFromUrlParams } from '@/app/components/base/chat/utils' import { useGlobalPublicStore } from './global-public-context' type WebAppStore = { @@ -24,6 +25,10 @@ type WebAppStore = { updateWebAppMeta: (appMeta: AppMeta | null) => void userCanAccessApp: boolean updateUserCanAccessApp: (canAccess: boolean) => void + embeddedUserId: string | null + updateEmbeddedUserId: (userId: string | null) => void + embeddedConversationId: string | null + updateEmbeddedConversationId: (conversationId: string | null) => void } export const useWebAppStore = create<WebAppStore>(set => ({ @@ -39,6 +44,11 @@ export const useWebAppStore = create<WebAppStore>(set => ({ updateWebAppMeta: (appMeta: AppMeta | null) => set(() => ({ appMeta })), userCanAccessApp: false, updateUserCanAccessApp: (canAccess: boolean) => set(() => ({ userCanAccessApp: canAccess })), + embeddedUserId: null, + updateEmbeddedUserId: (userId: string | null) => set(() => ({ embeddedUserId: userId })), + embeddedConversationId: null, + updateEmbeddedConversationId: (conversationId: string | null) => + set(() => ({ embeddedConversationId: conversationId })), })) const getShareCodeFromRedirectUrl = (redirectUrl: string | null): string | null => { @@ -58,9 +68,12 @@ const WebAppStoreProvider: FC<PropsWithChildren> = ({ children }) => { const isGlobalPending = useGlobalPublicStore(s => s.isGlobalPending) const updateWebAppAccessMode = useWebAppStore(state => state.updateWebAppAccessMode) const updateShareCode = useWebAppStore(state => state.updateShareCode) + const updateEmbeddedUserId = useWebAppStore(state => state.updateEmbeddedUserId) + const updateEmbeddedConversationId = useWebAppStore(state => state.updateEmbeddedConversationId) const pathname = usePathname() const searchParams = useSearchParams() const redirectUrlParam = searchParams.get('redirect_url') + const searchParamsString = searchParams.toString() // Compute shareCode directly const shareCode = getShareCodeFromRedirectUrl(redirectUrlParam) || getShareCodeFromPathname(pathname) @@ -68,6 +81,29 @@ const WebAppStoreProvider: FC<PropsWithChildren> = ({ children }) => { updateShareCode(shareCode) }, [shareCode, updateShareCode]) + useEffect(() => { + let cancelled = false + const syncEmbeddedUserId = async () => { + try { + const { user_id, conversation_id } = await getProcessedSystemVariablesFromUrlParams() + if (!cancelled) { + updateEmbeddedUserId(user_id || null) + updateEmbeddedConversationId(conversation_id || null) + } + } + catch { + if (!cancelled) { + updateEmbeddedUserId(null) + updateEmbeddedConversationId(null) + } + } + } + syncEmbeddedUserId() + return () => { + cancelled = true + } + }, [searchParamsString, updateEmbeddedUserId, updateEmbeddedConversationId]) + const { isLoading, data: accessModeResult } = useGetWebAppAccessModeByCode(shareCode) useEffect(() => { diff --git a/web/docker/entrypoint.sh b/web/docker/entrypoint.sh index c12ebc0812..b32e648922 100755 --- a/web/docker/entrypoint.sh +++ b/web/docker/entrypoint.sh @@ -34,6 +34,7 @@ export NEXT_PUBLIC_MAX_TOOLS_NUM=${MAX_TOOLS_NUM} export NEXT_PUBLIC_ENABLE_WEBSITE_JINAREADER=${ENABLE_WEBSITE_JINAREADER:-true} export NEXT_PUBLIC_ENABLE_WEBSITE_FIRECRAWL=${ENABLE_WEBSITE_FIRECRAWL:-true} export NEXT_PUBLIC_ENABLE_WEBSITE_WATERCRAWL=${ENABLE_WEBSITE_WATERCRAWL:-true} +export NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX=${NEXT_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX:-false} export NEXT_PUBLIC_LOOP_NODE_MAX_COUNT=${LOOP_NODE_MAX_COUNT} export NEXT_PUBLIC_MAX_PARALLEL_LIMIT=${MAX_PARALLEL_LIMIT} export NEXT_PUBLIC_MAX_ITERATIONS_NUM=${MAX_ITERATIONS_NUM} diff --git a/web/global.d.ts b/web/global.d.ts index c5488a6cae..0ccadf7887 100644 --- a/web/global.d.ts +++ b/web/global.d.ts @@ -1,6 +1,7 @@ import './types/i18n' import './types/jsx' import './types/mdx' +import './types/assets' declare module 'lamejs'; declare module 'lamejs/src/js/MPEGMode'; @@ -8,4 +9,4 @@ declare module 'lamejs/src/js/Lame'; declare module 'lamejs/src/js/BitStream'; declare module 'react-18-input-autosize'; -export {} +export { } diff --git a/web/hooks/use-breakpoints.spec.ts b/web/hooks/use-breakpoints.spec.ts index 315e514f0f..8b29fe486c 100644 --- a/web/hooks/use-breakpoints.spec.ts +++ b/web/hooks/use-breakpoints.spec.ts @@ -1,10 +1,27 @@ +/** + * Test suite for useBreakpoints hook + * + * This hook provides responsive breakpoint detection based on window width. + * It listens to window resize events and returns the current media type. + * + * Breakpoint definitions: + * - mobile: width <= 640px + * - tablet: 640px < width <= 768px + * - pc: width > 768px + * + * The hook automatically updates when the window is resized and cleans up + * event listeners on unmount to prevent memory leaks. + */ import { act, renderHook } from '@testing-library/react' import useBreakpoints, { MediaType } from './use-breakpoints' describe('useBreakpoints', () => { const originalInnerWidth = window.innerWidth - // Mock the window resize event + /** + * Helper function to simulate window resize events + * Updates window.innerWidth and dispatches a resize event + */ const fireResize = (width: number) => { window.innerWidth = width act(() => { @@ -12,11 +29,18 @@ describe('useBreakpoints', () => { }) } - // Restore the original innerWidth after tests + /** + * Restore the original innerWidth after all tests + * Ensures tests don't affect each other or the test environment + */ afterAll(() => { window.innerWidth = originalInnerWidth }) + /** + * Test mobile breakpoint detection + * Mobile devices have width <= 640px + */ it('should return mobile for width <= 640px', () => { // Mock window.innerWidth for mobile Object.defineProperty(window, 'innerWidth', { @@ -29,6 +53,10 @@ describe('useBreakpoints', () => { expect(result.current).toBe(MediaType.mobile) }) + /** + * Test tablet breakpoint detection + * Tablet devices have width between 640px and 768px + */ it('should return tablet for width > 640px and <= 768px', () => { // Mock window.innerWidth for tablet Object.defineProperty(window, 'innerWidth', { @@ -41,6 +69,10 @@ describe('useBreakpoints', () => { expect(result.current).toBe(MediaType.tablet) }) + /** + * Test desktop/PC breakpoint detection + * Desktop devices have width > 768px + */ it('should return pc for width > 768px', () => { // Mock window.innerWidth for pc Object.defineProperty(window, 'innerWidth', { @@ -53,6 +85,10 @@ describe('useBreakpoints', () => { expect(result.current).toBe(MediaType.pc) }) + /** + * Test dynamic breakpoint updates on window resize + * The hook should react to window resize events and update the media type + */ it('should update media type when window resizes', () => { // Start with desktop Object.defineProperty(window, 'innerWidth', { @@ -73,6 +109,10 @@ describe('useBreakpoints', () => { expect(result.current).toBe(MediaType.mobile) }) + /** + * Test proper cleanup of event listeners + * Ensures no memory leaks by removing resize listeners on unmount + */ it('should clean up event listeners on unmount', () => { // Spy on addEventListener and removeEventListener const addEventListenerSpy = jest.spyOn(window, 'addEventListener') diff --git a/web/hooks/use-document-title.spec.ts b/web/hooks/use-document-title.spec.ts index a8d3d56cff..fbc82a0cdf 100644 --- a/web/hooks/use-document-title.spec.ts +++ b/web/hooks/use-document-title.spec.ts @@ -1,3 +1,15 @@ +/** + * Test suite for useDocumentTitle hook + * + * This hook manages the browser document title with support for: + * - Custom branding (when enabled in system features) + * - Default "Dify" branding + * - Pending state handling (prevents title flicker during loading) + * - Page-specific titles with automatic suffix + * + * Title format: "[Page Title] - [Brand Name]" + * If no page title: "[Brand Name]" + */ import { defaultSystemFeatures } from '@/types/feature' import { act, renderHook } from '@testing-library/react' import useDocumentTitle from './use-document-title' @@ -7,6 +19,10 @@ jest.mock('@/service/common', () => ({ getSystemFeatures: jest.fn(() => ({ ...defaultSystemFeatures })), })) +/** + * Test behavior when system features are still loading + * Title should remain empty to prevent flicker + */ describe('title should be empty if systemFeatures is pending', () => { act(() => { useGlobalPublicStore.setState({ @@ -14,16 +30,26 @@ describe('title should be empty if systemFeatures is pending', () => { isGlobalPending: true, }) }) + /** + * Test that title stays empty during loading even when a title is provided + */ it('document title should be empty if set title', () => { renderHook(() => useDocumentTitle('test')) expect(document.title).toBe('') }) + /** + * Test that title stays empty during loading when no title is provided + */ it('document title should be empty if not set title', () => { renderHook(() => useDocumentTitle('')) expect(document.title).toBe('') }) }) +/** + * Test default Dify branding behavior + * When custom branding is disabled, should use "Dify" as the brand name + */ describe('use default branding', () => { beforeEach(() => { act(() => { @@ -33,17 +59,29 @@ describe('use default branding', () => { }) }) }) + /** + * Test title format with page title and default branding + * Format: "[page] - Dify" + */ it('document title should be test-Dify if set title', () => { renderHook(() => useDocumentTitle('test')) expect(document.title).toBe('test - Dify') }) + /** + * Test title with only default branding (no page title) + * Format: "Dify" + */ it('document title should be Dify if not set title', () => { renderHook(() => useDocumentTitle('')) expect(document.title).toBe('Dify') }) }) +/** + * Test custom branding behavior + * When custom branding is enabled, should use the configured application_title + */ describe('use specific branding', () => { beforeEach(() => { act(() => { @@ -53,11 +91,19 @@ describe('use specific branding', () => { }) }) }) + /** + * Test title format with page title and custom branding + * Format: "[page] - [Custom Brand]" + */ it('document title should be test-Test if set title', () => { renderHook(() => useDocumentTitle('test')) expect(document.title).toBe('test - Test') }) + /** + * Test title with only custom branding (no page title) + * Format: "[Custom Brand]" + */ it('document title should be Test if not set title', () => { renderHook(() => useDocumentTitle('')) expect(document.title).toBe('Test') diff --git a/web/hooks/use-format-time-from-now.spec.ts b/web/hooks/use-format-time-from-now.spec.ts new file mode 100644 index 0000000000..92ed37515c --- /dev/null +++ b/web/hooks/use-format-time-from-now.spec.ts @@ -0,0 +1,376 @@ +/** + * Test suite for useFormatTimeFromNow hook + * + * This hook provides internationalized relative time formatting (e.g., "2 hours ago", "3 days ago") + * using dayjs with the relativeTime plugin. It automatically uses the correct locale based on + * the user's i18n settings. + * + * Key features: + * - Supports 20+ locales with proper translations + * - Automatically syncs with user's interface language + * - Uses dayjs for consistent time calculations + * - Returns human-readable relative time strings + */ +import { renderHook } from '@testing-library/react' +import { useFormatTimeFromNow } from './use-format-time-from-now' + +// Mock the i18n context +jest.mock('@/context/i18n', () => ({ + useI18N: jest.fn(() => ({ + locale: 'en-US', + })), +})) + +// Import after mock to get the mocked version +import { useI18N } from '@/context/i18n' + +describe('useFormatTimeFromNow', () => { + beforeEach(() => { + jest.clearAllMocks() + }) + + describe('Basic functionality', () => { + /** + * Test that the hook returns a formatTimeFromNow function + * This is the primary interface of the hook + */ + it('should return formatTimeFromNow function', () => { + const { result } = renderHook(() => useFormatTimeFromNow()) + + expect(result.current).toHaveProperty('formatTimeFromNow') + expect(typeof result.current.formatTimeFromNow).toBe('function') + }) + + /** + * Test basic relative time formatting with English locale + * Should return human-readable relative time strings + */ + it('should format time from now in English', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // Should contain "hour" or "hours" and "ago" + expect(formatted).toMatch(/hour|hours/) + expect(formatted).toMatch(/ago/) + }) + + /** + * Test that recent times are formatted as "a few seconds ago" + * Very recent timestamps should show seconds + */ + it('should format very recent times', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const fiveSecondsAgo = now - (5 * 1000) + const formatted = result.current.formatTimeFromNow(fiveSecondsAgo) + + expect(formatted).toMatch(/second|seconds|few seconds/) + }) + + /** + * Test formatting of times in the past (days ago) + * Should handle day-level granularity + */ + it('should format times from days ago', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const threeDaysAgo = now - (3 * 24 * 60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(threeDaysAgo) + + expect(formatted).toMatch(/day|days/) + expect(formatted).toMatch(/ago/) + }) + + /** + * Test formatting of future times + * dayjs fromNow also supports future times (e.g., "in 2 hours") + */ + it('should format future times', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const twoHoursFromNow = now + (2 * 60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(twoHoursFromNow) + + expect(formatted).toMatch(/in/) + expect(formatted).toMatch(/hour|hours/) + }) + }) + + describe('Locale support', () => { + /** + * Test Chinese (Simplified) locale formatting + * Should use Chinese characters for time units + */ + it('should format time in Chinese (Simplified)', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'zh-Hans' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // Chinese should contain Chinese characters + expect(formatted).toMatch(/[\u4E00-\u9FA5]/) + }) + + /** + * Test Spanish locale formatting + * Should use Spanish words for relative time + */ + it('should format time in Spanish', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'es-ES' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // Spanish should contain "hace" (ago) + expect(formatted).toMatch(/hace/) + }) + + /** + * Test French locale formatting + * Should use French words for relative time + */ + it('should format time in French', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'fr-FR' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // French should contain "il y a" (ago) + expect(formatted).toMatch(/il y a/) + }) + + /** + * Test Japanese locale formatting + * Should use Japanese characters + */ + it('should format time in Japanese', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'ja-JP' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // Japanese should contain Japanese characters + expect(formatted).toMatch(/[\u3040-\u309F\u30A0-\u30FF\u4E00-\u9FAF]/) + }) + + /** + * Test Portuguese (Brazil) locale formatting + * Should use pt-br locale mapping + */ + it('should format time in Portuguese (Brazil)', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'pt-BR' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // Portuguese should contain "há" (ago) + expect(formatted).toMatch(/há/) + }) + + /** + * Test fallback to English for unsupported locales + * Unknown locales should default to English + */ + it('should fallback to English for unsupported locale', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'xx-XX' as any }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // Should still return a valid string (in English) + expect(typeof formatted).toBe('string') + expect(formatted.length).toBeGreaterThan(0) + }) + }) + + describe('Edge cases', () => { + /** + * Test handling of timestamp 0 (Unix epoch) + * Should format as a very old date + */ + it('should handle timestamp 0', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const formatted = result.current.formatTimeFromNow(0) + + expect(typeof formatted).toBe('string') + expect(formatted.length).toBeGreaterThan(0) + expect(formatted).toMatch(/year|years/) + }) + + /** + * Test handling of very large timestamps + * Should handle dates far in the future + */ + it('should handle very large timestamps', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const farFuture = Date.now() + (365 * 24 * 60 * 60 * 1000) // 1 year from now + const formatted = result.current.formatTimeFromNow(farFuture) + + expect(typeof formatted).toBe('string') + expect(formatted).toMatch(/in/) + }) + + /** + * Test that the function is memoized based on locale + * Changing locale should update the function + */ + it('should update when locale changes', () => { + const { result, rerender } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + + // First render with English + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + rerender() + const englishResult = result.current.formatTimeFromNow(oneHourAgo) + + // Second render with Spanish + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'es-ES' }) + rerender() + const spanishResult = result.current.formatTimeFromNow(oneHourAgo) + + // Results should be different + expect(englishResult).not.toBe(spanishResult) + }) + }) + + describe('Time granularity', () => { + /** + * Test different time granularities (seconds, minutes, hours, days, months, years) + * dayjs should automatically choose the appropriate unit + */ + it('should use appropriate time units for different durations', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + + const now = Date.now() + + // Seconds + const seconds = result.current.formatTimeFromNow(now - 30 * 1000) + expect(seconds).toMatch(/second/) + + // Minutes + const minutes = result.current.formatTimeFromNow(now - 5 * 60 * 1000) + expect(minutes).toMatch(/minute/) + + // Hours + const hours = result.current.formatTimeFromNow(now - 3 * 60 * 60 * 1000) + expect(hours).toMatch(/hour/) + + // Days + const days = result.current.formatTimeFromNow(now - 5 * 24 * 60 * 60 * 1000) + expect(days).toMatch(/day/) + + // Months + const months = result.current.formatTimeFromNow(now - 60 * 24 * 60 * 60 * 1000) + expect(months).toMatch(/month/) + }) + }) + + describe('Locale mapping', () => { + /** + * Test that all supported locales in the localeMap are handled correctly + * This ensures the mapping from app locales to dayjs locales works + */ + it('should handle all mapped locales', () => { + const locales = [ + 'en-US', 'zh-Hans', 'zh-Hant', 'pt-BR', 'es-ES', 'fr-FR', + 'de-DE', 'ja-JP', 'ko-KR', 'ru-RU', 'it-IT', 'th-TH', + 'id-ID', 'uk-UA', 'vi-VN', 'ro-RO', 'pl-PL', 'hi-IN', + 'tr-TR', 'fa-IR', 'sl-SI', + ] + + const now = Date.now() + const oneHourAgo = now - (60 * 60 * 1000) + + locales.forEach((locale) => { + ;(useI18N as jest.Mock).mockReturnValue({ locale }) + + const { result } = renderHook(() => useFormatTimeFromNow()) + const formatted = result.current.formatTimeFromNow(oneHourAgo) + + // Should return a non-empty string for each locale + expect(typeof formatted).toBe('string') + expect(formatted.length).toBeGreaterThan(0) + }) + }) + }) + + describe('Performance', () => { + /** + * Test that the hook doesn't create new functions on every render + * The formatTimeFromNow function should be memoized with useCallback + */ + it('should memoize formatTimeFromNow function', () => { + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + + const { result, rerender } = renderHook(() => useFormatTimeFromNow()) + + const firstFunction = result.current.formatTimeFromNow + rerender() + const secondFunction = result.current.formatTimeFromNow + + // Same locale should return the same function reference + expect(firstFunction).toBe(secondFunction) + }) + + /** + * Test that changing locale creates a new function + * This ensures the memoization dependency on locale works correctly + */ + it('should create new function when locale changes', () => { + const { result, rerender } = renderHook(() => useFormatTimeFromNow()) + + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'en-US' }) + rerender() + const englishFunction = result.current.formatTimeFromNow + + ;(useI18N as jest.Mock).mockReturnValue({ locale: 'es-ES' }) + rerender() + const spanishFunction = result.current.formatTimeFromNow + + // Different locale should return different function reference + expect(englishFunction).not.toBe(spanishFunction) + }) + }) +}) diff --git a/web/hooks/use-oauth.ts b/web/hooks/use-oauth.ts index 9f21a476b3..34ed8bafb0 100644 --- a/web/hooks/use-oauth.ts +++ b/web/hooks/use-oauth.ts @@ -4,16 +4,38 @@ import { validateRedirectUrl } from '@/utils/urlValidation' export const useOAuthCallback = () => { useEffect(() => { + const urlParams = new URLSearchParams(window.location.search) + const subscriptionId = urlParams.get('subscription_id') + const error = urlParams.get('error') + const errorDescription = urlParams.get('error_description') + if (window.opener) { - window.opener.postMessage({ - type: 'oauth_callback', - }, '*') + if (subscriptionId) { + window.opener.postMessage({ + type: 'oauth_callback', + success: true, + subscriptionId, + }, '*') + } + else if (error) { + window.opener.postMessage({ + type: 'oauth_callback', + success: false, + error, + errorDescription, + }, '*') + } + else { + window.opener.postMessage({ + type: 'oauth_callback', + }, '*') + } window.close() } }, []) } -export const openOAuthPopup = (url: string, callback: () => void) => { +export const openOAuthPopup = (url: string, callback: (data?: any) => void) => { const width = 600 const height = 600 const left = window.screenX + (window.outerWidth - width) / 2 @@ -29,10 +51,20 @@ export const openOAuthPopup = (url: string, callback: () => void) => { const handleMessage = (event: MessageEvent) => { if (event.data?.type === 'oauth_callback') { window.removeEventListener('message', handleMessage) - callback() + callback(event.data) } } window.addEventListener('message', handleMessage) + + // Fallback for window close detection + const checkClosed = setInterval(() => { + if (popup?.closed) { + clearInterval(checkClosed) + window.removeEventListener('message', handleMessage) + callback() + } + }, 1000) + return popup } diff --git a/web/hooks/use-tab-searchparams.spec.ts b/web/hooks/use-tab-searchparams.spec.ts new file mode 100644 index 0000000000..62adea529f --- /dev/null +++ b/web/hooks/use-tab-searchparams.spec.ts @@ -0,0 +1,543 @@ +/** + * Test suite for useTabSearchParams hook + * + * This hook manages tab state through URL search parameters, enabling: + * - Bookmarkable tab states (users can share URLs with specific tabs active) + * - Browser history integration (back/forward buttons work with tabs) + * - Configurable routing behavior (push vs replace) + * - Optional search parameter syncing (can disable URL updates) + * + * The hook syncs a local tab state with URL search parameters, making tab + * navigation persistent and shareable across sessions. + */ +import { act, renderHook } from '@testing-library/react' +import { useTabSearchParams } from './use-tab-searchparams' + +// Mock Next.js navigation hooks +const mockPush = jest.fn() +const mockReplace = jest.fn() +const mockPathname = '/test-path' +const mockSearchParams = new URLSearchParams() + +jest.mock('next/navigation', () => ({ + usePathname: jest.fn(() => mockPathname), + useRouter: jest.fn(() => ({ + push: mockPush, + replace: mockReplace, + })), + useSearchParams: jest.fn(() => mockSearchParams), +})) + +// Import after mocks +import { usePathname } from 'next/navigation' + +describe('useTabSearchParams', () => { + beforeEach(() => { + jest.clearAllMocks() + mockSearchParams.delete('category') + mockSearchParams.delete('tab') + }) + + describe('Basic functionality', () => { + /** + * Test that the hook returns a tuple with activeTab and setActiveTab + * This is the primary interface matching React's useState pattern + */ + it('should return activeTab and setActiveTab function', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + const [activeTab, setActiveTab] = result.current + + expect(typeof activeTab).toBe('string') + expect(typeof setActiveTab).toBe('function') + }) + + /** + * Test that the hook initializes with the default tab + * When no search param is present, should use defaultTab + */ + it('should initialize with default tab when no search param exists', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + const [activeTab] = result.current + expect(activeTab).toBe('overview') + }) + + /** + * Test that the hook reads from URL search parameters + * When a search param exists, it should take precedence over defaultTab + */ + it('should initialize with search param value when present', () => { + mockSearchParams.set('category', 'settings') + + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + const [activeTab] = result.current + expect(activeTab).toBe('settings') + }) + + /** + * Test that setActiveTab updates the local state + * The active tab should change when setActiveTab is called + */ + it('should update active tab when setActiveTab is called', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings') + }) + + const [activeTab] = result.current + expect(activeTab).toBe('settings') + }) + }) + + describe('Routing behavior', () => { + /** + * Test default push routing behavior + * By default, tab changes should use router.push (adds to history) + */ + it('should use push routing by default', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings') + }) + + expect(mockPush).toHaveBeenCalledWith('/test-path?category=settings') + expect(mockReplace).not.toHaveBeenCalled() + }) + + /** + * Test replace routing behavior + * When routingBehavior is 'replace', should use router.replace (no history) + */ + it('should use replace routing when specified', () => { + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + routingBehavior: 'replace', + }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings') + }) + + expect(mockReplace).toHaveBeenCalledWith('/test-path?category=settings') + expect(mockPush).not.toHaveBeenCalled() + }) + + /** + * Test that URL encoding is applied to tab values + * Special characters in tab names should be properly encoded + */ + it('should encode special characters in tab values', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings & config') + }) + + expect(mockPush).toHaveBeenCalledWith( + '/test-path?category=settings%20%26%20config', + ) + }) + + /** + * Test that URL decoding is applied when reading from search params + * Encoded values in the URL should be properly decoded + */ + it('should decode encoded values from search params', () => { + mockSearchParams.set('category', 'settings%20%26%20config') + + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + const [activeTab] = result.current + expect(activeTab).toBe('settings & config') + }) + }) + + describe('Custom search parameter name', () => { + /** + * Test using a custom search parameter name + * Should support different param names instead of default 'category' + */ + it('should use custom search param name', () => { + mockSearchParams.set('tab', 'profile') + + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + searchParamName: 'tab', + }), + ) + + const [activeTab] = result.current + expect(activeTab).toBe('profile') + }) + + /** + * Test that setActiveTab uses the custom param name in the URL + */ + it('should update URL with custom param name', () => { + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + searchParamName: 'tab', + }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('profile') + }) + + expect(mockPush).toHaveBeenCalledWith('/test-path?tab=profile') + }) + }) + + describe('Disabled search params mode', () => { + /** + * Test that disableSearchParams prevents URL updates + * When disabled, tab state should be local only + */ + it('should not update URL when disableSearchParams is true', () => { + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + disableSearchParams: true, + }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings') + }) + + expect(mockPush).not.toHaveBeenCalled() + expect(mockReplace).not.toHaveBeenCalled() + }) + + /** + * Test that local state still updates when search params are disabled + * The tab state should work even without URL syncing + */ + it('should still update local state when search params disabled', () => { + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + disableSearchParams: true, + }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings') + }) + + const [activeTab] = result.current + expect(activeTab).toBe('settings') + }) + + /** + * Test that disabled mode always uses defaultTab + * Search params should be ignored when disabled + */ + it('should use defaultTab when search params disabled even if URL has value', () => { + mockSearchParams.set('category', 'settings') + + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + disableSearchParams: true, + }), + ) + + const [activeTab] = result.current + expect(activeTab).toBe('overview') + }) + }) + + describe('Edge cases', () => { + /** + * Test handling of empty string tab values + * Empty strings should be handled gracefully + */ + it('should handle empty string tab values', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('') + }) + + const [activeTab] = result.current + expect(activeTab).toBe('') + expect(mockPush).toHaveBeenCalledWith('/test-path?category=') + }) + + /** + * Test that special characters in tab names are properly encoded + * This ensures URLs remain valid even with unusual tab names + */ + it('should handle tabs with various special characters', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + // Test tab with slashes + act(() => result.current[1]('tab/with/slashes')) + expect(result.current[0]).toBe('tab/with/slashes') + + // Test tab with question marks + act(() => result.current[1]('tab?with?questions')) + expect(result.current[0]).toBe('tab?with?questions') + + // Test tab with hash symbols + act(() => result.current[1]('tab#with#hash')) + expect(result.current[0]).toBe('tab#with#hash') + + // Test tab with equals signs + act(() => result.current[1]('tab=with=equals')) + expect(result.current[0]).toBe('tab=with=equals') + }) + + /** + * Test fallback when pathname is not available + * Should use window.location.pathname as fallback + */ + it('should fallback to window.location.pathname when hook pathname is null', () => { + ;(usePathname as jest.Mock).mockReturnValue(null) + + // Mock window.location.pathname + Object.defineProperty(window, 'location', { + value: { pathname: '/fallback-path' }, + writable: true, + }) + + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings') + }) + + expect(mockPush).toHaveBeenCalledWith('/fallback-path?category=settings') + + // Restore mock + ;(usePathname as jest.Mock).mockReturnValue(mockPathname) + }) + }) + + describe('Multiple instances', () => { + /** + * Test that multiple instances with different param names work independently + * Different hooks should not interfere with each other + */ + it('should support multiple independent tab states', () => { + mockSearchParams.set('category', 'overview') + mockSearchParams.set('subtab', 'details') + + const { result: result1 } = renderHook(() => + useTabSearchParams({ + defaultTab: 'home', + searchParamName: 'category', + }), + ) + + const { result: result2 } = renderHook(() => + useTabSearchParams({ + defaultTab: 'info', + searchParamName: 'subtab', + }), + ) + + const [activeTab1] = result1.current + const [activeTab2] = result2.current + + expect(activeTab1).toBe('overview') + expect(activeTab2).toBe('details') + }) + }) + + describe('Integration scenarios', () => { + /** + * Test typical usage in a tabbed interface + * Simulates real-world tab switching behavior + */ + it('should handle sequential tab changes', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + // Change to settings tab + act(() => { + const [, setActiveTab] = result.current + setActiveTab('settings') + }) + + expect(result.current[0]).toBe('settings') + expect(mockPush).toHaveBeenCalledWith('/test-path?category=settings') + + // Change to profile tab + act(() => { + const [, setActiveTab] = result.current + setActiveTab('profile') + }) + + expect(result.current[0]).toBe('profile') + expect(mockPush).toHaveBeenCalledWith('/test-path?category=profile') + + // Verify push was called twice + expect(mockPush).toHaveBeenCalledTimes(2) + }) + + /** + * Test that the hook works with complex pathnames + * Should handle nested routes and existing query params + */ + it('should work with complex pathnames', () => { + ;(usePathname as jest.Mock).mockReturnValue('/app/123/settings') + + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('advanced') + }) + + expect(mockPush).toHaveBeenCalledWith('/app/123/settings?category=advanced') + + // Restore mock + ;(usePathname as jest.Mock).mockReturnValue(mockPathname) + }) + }) + + describe('Type safety', () => { + /** + * Test that the return type is a const tuple + * TypeScript should infer [string, (tab: string) => void] as const + */ + it('should return a const tuple type', () => { + const { result } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + // The result should be a tuple with exactly 2 elements + expect(result.current).toHaveLength(2) + expect(typeof result.current[0]).toBe('string') + expect(typeof result.current[1]).toBe('function') + }) + }) + + describe('Performance', () => { + /** + * Test that the hook creates a new function on each render + * Note: The current implementation doesn't use useCallback, + * so setActiveTab is recreated on each render. This could lead to + * unnecessary re-renders in child components that depend on this function. + * TODO: Consider memoizing setActiveTab with useCallback for better performance. + */ + it('should create new setActiveTab function on each render', () => { + const { result, rerender } = renderHook(() => + useTabSearchParams({ defaultTab: 'overview' }), + ) + + const [, firstSetActiveTab] = result.current + rerender() + const [, secondSetActiveTab] = result.current + + // Function reference changes on re-render (not memoized) + expect(firstSetActiveTab).not.toBe(secondSetActiveTab) + + // But both functions should work correctly + expect(typeof firstSetActiveTab).toBe('function') + expect(typeof secondSetActiveTab).toBe('function') + }) + }) + + describe('Browser history integration', () => { + /** + * Test that push behavior adds to browser history + * This enables back/forward navigation through tabs + */ + it('should add to history with push behavior', () => { + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + routingBehavior: 'push', + }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('tab1') + }) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('tab2') + }) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('tab3') + }) + + // Each tab change should create a history entry + expect(mockPush).toHaveBeenCalledTimes(3) + }) + + /** + * Test that replace behavior doesn't add to history + * This prevents cluttering browser history with tab changes + */ + it('should not add to history with replace behavior', () => { + const { result } = renderHook(() => + useTabSearchParams({ + defaultTab: 'overview', + routingBehavior: 'replace', + }), + ) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('tab1') + }) + + act(() => { + const [, setActiveTab] = result.current + setActiveTab('tab2') + }) + + // Should use replace instead of push + expect(mockReplace).toHaveBeenCalledTimes(2) + expect(mockPush).not.toHaveBeenCalled() + }) + }) +}) diff --git a/web/i18n-config/i18next-config.ts b/web/i18n-config/i18next-config.ts index af04802e42..360d2afb29 100644 --- a/web/i18n-config/i18next-config.ts +++ b/web/i18n-config/i18next-config.ts @@ -38,6 +38,7 @@ const NAMESPACES = [ 'oauth', 'pipeline', 'plugin-tags', + 'plugin-trigger', 'plugin', 'register', 'run-log', diff --git a/web/i18n/de-DE/app-debug.ts b/web/i18n/de-DE/app-debug.ts index fffb84bd61..badf27be59 100644 --- a/web/i18n/de-DE/app-debug.ts +++ b/web/i18n/de-DE/app-debug.ts @@ -259,7 +259,6 @@ const translation = { variableTable: { key: 'Variablenschlüssel', name: 'Name des Benutzereingabefelds', - optional: 'Optional', type: 'Eingabetyp', action: 'Aktionen', typeString: 'String', diff --git a/web/i18n/de-DE/app-log.ts b/web/i18n/de-DE/app-log.ts index 0fbdcca0bf..d9edd88d13 100644 --- a/web/i18n/de-DE/app-log.ts +++ b/web/i18n/de-DE/app-log.ts @@ -66,6 +66,8 @@ const translation = { quarterToDate: 'Quartal bis heute', yearToDate: 'Jahr bis heute', allTime: 'Gesamte Zeit', + last30days: 'Letzte 30 Tage', + custom: 'Benutzerdefiniert', }, annotation: { all: 'Alle', diff --git a/web/i18n/de-DE/billing.ts b/web/i18n/de-DE/billing.ts index fc45f3889c..6601bbb179 100644 --- a/web/i18n/de-DE/billing.ts +++ b/web/i18n/de-DE/billing.ts @@ -83,7 +83,7 @@ const translation = { cloud: 'Cloud-Dienst', apiRateLimitTooltip: 'Die API-Datenbeschränkung gilt für alle Anfragen, die über die Dify-API gemacht werden, einschließlich Textgenerierung, Chat-Konversationen, Workflow-Ausführungen und Dokumentenverarbeitung.', getStarted: 'Loslegen', - apiRateLimitUnit: '{{count,number}}/Tag', + apiRateLimitUnit: '{{count,number}}/Monat', documentsTooltip: 'Vorgabe für die Anzahl der Dokumente, die aus der Wissensdatenquelle importiert werden.', apiRateLimit: 'API-Datenlimit', documents: '{{count,number}} Wissensdokumente', diff --git a/web/i18n/de-DE/common.ts b/web/i18n/de-DE/common.ts index 9226e81d9b..a615974061 100644 --- a/web/i18n/de-DE/common.ts +++ b/web/i18n/de-DE/common.ts @@ -161,7 +161,6 @@ const translation = { workspace: 'Arbeitsbereich', createWorkspace: 'Arbeitsbereich erstellen', helpCenter: 'Hilfe', - communityFeedback: 'Rückmeldung', roadmap: 'Fahrplan', community: 'Gemeinschaft', about: 'Über', @@ -170,6 +169,7 @@ const translation = { support: 'Unterstützung', github: 'GitHub', contactUs: 'Kontaktieren Sie uns', + forum: 'Forum', }, settings: { accountGroup: 'KONTO', @@ -726,6 +726,7 @@ const translation = { uploadFromComputerLimit: 'Datei hochladen darf {{size}} nicht überschreiten', uploadFromComputerReadError: 'Lesen der Datei fehlgeschlagen, bitte versuchen Sie es erneut.', fileExtensionNotSupport: 'Dateiendung nicht bedient', + fileExtensionBlocked: 'Dieser Dateityp ist aus Sicherheitsgründen gesperrt', }, license: { expiring: 'Läuft an einem Tag ab', diff --git a/web/i18n/de-DE/login.ts b/web/i18n/de-DE/login.ts index a4c9165e23..4705a73087 100644 --- a/web/i18n/de-DE/login.ts +++ b/web/i18n/de-DE/login.ts @@ -120,6 +120,7 @@ const translation = { noAccount: 'Haben Sie kein Konto?', verifyMail: 'Fahren Sie mit dem Bestätigungscode fort', }, + pageTitleForE: 'Hey, lass uns anfangen!', } export default translation diff --git a/web/i18n/de-DE/tools.ts b/web/i18n/de-DE/tools.ts index 8cef76b732..4e93b4b71e 100644 --- a/web/i18n/de-DE/tools.ts +++ b/web/i18n/de-DE/tools.ts @@ -2,7 +2,6 @@ const translation = { title: 'Werkzeuge', createCustomTool: 'Eigenes Werkzeug erstellen', type: { - all: 'Alle', builtIn: 'Integriert', custom: 'Benutzerdefiniert', workflow: 'Arbeitsablauf', @@ -20,7 +19,6 @@ const translation = { setupModalTitleDescription: 'Nach der Konfiguration der Anmeldeinformationen können alle Mitglieder im Arbeitsbereich dieses Werkzeug beim Orchestrieren von Anwendungen nutzen.', }, includeToolNum: '{{num}} Werkzeuge inkludiert', - addTool: 'Werkzeug hinzufügen', createTool: { title: 'Eigenes Werkzeug erstellen', editAction: 'Konfigurieren', @@ -143,9 +141,7 @@ const translation = { addToolModal: { type: 'Art', category: 'Kategorie', - add: 'hinzufügen', added: 'zugefügt', - manageInTools: 'Verwalten in Tools', custom: { title: 'Kein benutzerdefiniertes Werkzeug verfügbar', tip: 'Benutzerdefiniertes Werkzeug erstellen', @@ -203,6 +199,12 @@ const translation = { noHeaders: 'Keine benutzerdefinierten Header konfiguriert', maskedHeadersTip: 'Headerwerte sind zum Schutz maskiert. Änderungen werden die tatsächlichen Werte aktualisieren.', headersTip: 'Zusätzliche HTTP-Header, die mit MCP-Serveranfragen gesendet werden sollen', + clientSecret: 'Client-Geheimnis', + clientSecretPlaceholder: 'Client-Geheimnis', + clientID: 'Kunden-ID', + authentication: 'Authentifizierung', + useDynamicClientRegistration: 'Dynamische Client-Registrierung verwenden', + configurations: 'Konfigurationen', }, delete: 'MCP-Server entfernen', deleteConfirmTitle: 'Möchten Sie {{mcp}} entfernen?', diff --git a/web/i18n/de-DE/workflow.ts b/web/i18n/de-DE/workflow.ts index 4353e5e10c..28aa8bdc19 100644 --- a/web/i18n/de-DE/workflow.ts +++ b/web/i18n/de-DE/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Veröffentlicht', publish: 'Veröffentlichen', update: 'Aktualisieren', - run: 'Ausführen', + run: 'Test ausführen', running: 'Wird ausgeführt', inRunMode: 'Im Ausführungsmodus', inPreview: 'In der Vorschau', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Ausführungsverlauf', goBackToEdit: 'Zurück zum Editor', conversationLog: 'Konversationsprotokoll', - features: 'Funktionen', debugAndPreview: 'Vorschau', restart: 'Neustarten', currentDraft: 'Aktueller Entwurf', @@ -91,10 +90,8 @@ const translation = { addParallelNode: 'Parallelen Knoten hinzufügen', parallel: 'PARALLEL', branch: 'ZWEIG', - featuresDocLink: 'Weitere Informationen', ImageUploadLegacyTip: 'Sie können jetzt Dateitypvariablen im Startformular erstellen. Wir werden die Funktion zum Hochladen von Bildern in Zukunft nicht mehr unterstützen.', fileUploadTip: 'Die Funktionen zum Hochladen von Bildern wurden auf das Hochladen von Dateien aktualisiert.', - featuresDescription: 'Verbessern Sie die Benutzererfahrung von Web-Apps', importWarning: 'Vorsicht', importWarningDetails: 'Der Unterschied zwischen den DSL-Versionen kann sich auf bestimmte Funktionen auswirken', openInExplore: 'In Explore öffnen', @@ -110,11 +107,12 @@ const translation = { exitVersions: 'Ausgangsversionen', exportPNG: 'Als PNG exportieren', addBlock: 'Knoten hinzufügen', - needEndNode: 'Der Endknoten muss hinzugefügt werden.', + needOutputNode: 'Der Ausgabeknoten muss hinzugefügt werden', needAnswerNode: 'Der Antwortknoten muss hinzugefügt werden.', tagBound: 'Anzahl der Apps, die dieses Tag verwenden', currentWorkflow: 'Aktueller Arbeitsablauf', currentView: 'Aktuelle Ansicht', + moreActions: 'Weitere Aktionen', }, env: { envPanelTitle: 'Umgebungsvariablen', @@ -139,6 +137,19 @@ const translation = { export: 'DSL mit geheimen Werten exportieren', }, }, + globalVar: { + title: 'Systemvariablen', + description: 'Systemvariablen sind globale Variablen, die von jedem Knoten ohne Verkabelung referenziert werden können, sofern der Typ passt, etwa Endnutzer-ID und Workflow-ID.', + fieldsDescription: { + conversationId: 'Konversations-ID', + dialogCount: 'Konversationsanzahl', + userId: 'Benutzer-ID', + triggerTimestamp: 'Zeitstempel des Anwendungsstarts', + appId: 'Anwendungs-ID', + workflowId: 'Workflow-ID', + workflowRunId: 'Workflow-Ausführungs-ID', + }, + }, chatVariable: { panelTitle: 'Gesprächsvariablen', panelDescription: 'Gesprächsvariablen werden verwendet, um interaktive Informationen zu speichern, die das LLM benötigt, einschließlich Gesprächsverlauf, hochgeladene Dateien und Benutzereinstellungen. Sie sind les- und schreibbar.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Start', - 'end': 'Ende', + 'end': 'Ausgabe', 'answer': 'Antwort', 'llm': 'LLM', 'knowledge-retrieval': 'Wissensabruf', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Definieren Sie die Anfangsparameter zum Starten eines Workflows', - 'end': 'Definieren Sie das Ende und den Ergebnistyp eines Workflows', + 'end': 'Definieren Sie die Ausgabe und den Ergebnistyp eines Workflows', 'answer': 'Definieren Sie den Antwortinhalt einer Chat-Konversation', 'llm': 'Große Sprachmodelle aufrufen, um Fragen zu beantworten oder natürliche Sprache zu verarbeiten', 'knowledge-retrieval': 'Ermöglicht das Abfragen von Textinhalten, die sich auf Benutzerfragen aus der Wissensdatenbank beziehen', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Benutzereingabefeld', - helpLink: 'Hilfelink', + helpLink: 'Hilfe', about: 'Über', createdBy: 'Erstellt von ', nextStep: 'Nächster Schritt', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Alle Probleme wurden gelöst', change: 'Ändern', optional: '(optional)', - moveToThisNode: 'Bewege zu diesem Knoten', selectNextStep: 'Nächsten Schritt auswählen', addNextStep: 'Fügen Sie den nächsten Schritt in diesem Arbeitsablauf hinzu.', organizeBlocks: 'Knoten organisieren', changeBlock: 'Knoten ändern', maximize: 'Maximiere die Leinwand', minimize: 'Vollbildmodus beenden', + scrollToSelectedNode: 'Zum ausgewählten Knoten scrollen', optional_and_hidden: '(optional & hidden)', }, nodes: { diff --git a/web/i18n/en-US/app-debug.ts b/web/i18n/en-US/app-debug.ts index 2ad705c563..9d1a824a88 100644 --- a/web/i18n/en-US/app-debug.ts +++ b/web/i18n/en-US/app-debug.ts @@ -346,7 +346,6 @@ const translation = { variableTable: { key: 'Variable Key', name: 'User Input Field Name', - optional: 'Optional', type: 'Input Type', action: 'Actions', typeString: 'String', diff --git a/web/i18n/en-US/app-log.ts b/web/i18n/en-US/app-log.ts index 946d8ffcb7..7c5024a68f 100644 --- a/web/i18n/en-US/app-log.ts +++ b/web/i18n/en-US/app-log.ts @@ -18,8 +18,9 @@ const translation = { status: 'STATUS', runtime: 'RUN TIME', tokens: 'TOKENS', - user: 'End User or Account', + user: 'END USER OR ACCOUNT', version: 'VERSION', + triggered_from: 'TRIGGER BY', }, pagination: { previous: 'Prev', @@ -59,6 +60,7 @@ const translation = { period: { today: 'Today', last7days: 'Last 7 Days', + last30days: 'Last 30 Days', last4weeks: 'Last 4 weeks', last3months: 'Last 3 months', last12months: 'Last 12 months', @@ -66,6 +68,7 @@ const translation = { quarterToDate: 'Quarter to date', yearToDate: 'Year to date', allTime: 'All time', + custom: 'Custom', }, annotation: { all: 'All', @@ -95,6 +98,15 @@ const translation = { iteration: 'Iteration', finalProcessing: 'Final Processing', }, + triggerBy: { + debugging: 'Debugging', + appRun: 'WebApp', + webhook: 'Webhook', + schedule: 'Schedule', + plugin: 'Plugin', + ragPipelineRun: 'RAG Pipeline', + ragPipelineDebugging: 'RAG Debugging', + }, } export default translation diff --git a/web/i18n/en-US/app-overview.ts b/web/i18n/en-US/app-overview.ts index feedc32e6b..4e88840b6d 100644 --- a/web/i18n/en-US/app-overview.ts +++ b/web/i18n/en-US/app-overview.ts @@ -30,6 +30,7 @@ const translation = { overview: { title: 'Overview', appInfo: { + title: 'Web App', explanation: 'Ready-to-use AI web app', accessibleAddress: 'Public URL', preview: 'Preview', @@ -37,6 +38,10 @@ const translation = { regenerate: 'Regenerate', regenerateNotice: 'Do you want to regenerate the public URL?', preUseReminder: 'Please enable web app before continuing.', + enableTooltip: { + description: 'To enable this feature, please add a User Input node to the canvas. (May already exist in draft, takes effect after publishing)', + learnMore: 'Learn more', + }, settings: { entry: 'Settings', title: 'Web App Settings', @@ -121,6 +126,14 @@ const translation = { accessibleAddress: 'Service API Endpoint', doc: 'API Reference', }, + triggerInfo: { + title: 'Triggers', + explanation: 'Workflow trigger management', + triggersAdded: '{{count}} Triggers added', + noTriggerAdded: 'No trigger added', + triggerStatusDescription: 'Trigger node status appears here. (May already exist in draft, takes effect after publishing)', + learnAboutTriggers: 'Learn about Triggers', + }, status: { running: 'In Service', disable: 'Disabled', diff --git a/web/i18n/en-US/app.ts b/web/i18n/en-US/app.ts index 7b3fead6e4..99bab2893c 100644 --- a/web/i18n/en-US/app.ts +++ b/web/i18n/en-US/app.ts @@ -254,6 +254,8 @@ const translation = { notSetDesc: 'Currently nobody can access the web app. Please set permissions.', }, noAccessPermission: 'No permission to access web app', + noUserInputNode: 'Missing user input node', + notPublishedYet: 'App is not published yet', maxActiveRequests: 'Max concurrent requests', maxActiveRequestsPlaceholder: 'Enter 0 for unlimited', maxActiveRequestsTip: 'Maximum number of concurrent active requests per app (0 for unlimited)', diff --git a/web/i18n/en-US/billing.ts b/web/i18n/en-US/billing.ts index 9169631281..0bd26c1075 100644 --- a/web/i18n/en-US/billing.ts +++ b/web/i18n/en-US/billing.ts @@ -7,6 +7,8 @@ const translation = { documentsUploadQuota: 'Documents Upload Quota', vectorSpace: 'Knowledge Data Storage', vectorSpaceTooltip: 'Documents with the High Quality indexing mode will consume Knowledge Data Storage resources. When Knowledge Data Storage reaches the limit, new documents will not be uploaded.', + triggerEvents: 'Trigger Events', + perMonth: 'per month', }, teamMembers: 'Team Members', upgradeBtn: { @@ -62,7 +64,7 @@ const translation = { documentsRequestQuota: '{{count,number}}/min Knowledge Request Rate Limit', documentsRequestQuotaTooltip: 'Specifies the total number of actions a workspace can perform per minute within the knowledge base, including dataset creation, deletion, updates, document uploads, modifications, archiving, and knowledge base queries. This metric is used to evaluate the performance of knowledge base requests. For example, if a Sandbox user performs 10 consecutive hit tests within one minute, their workspace will be temporarily restricted from performing the following actions for the next minute: dataset creation, deletion, updates, and document uploads or modifications. ', apiRateLimit: 'API Rate Limit', - apiRateLimitUnit: '{{count,number}}/day', + apiRateLimitUnit: '{{count,number}}/month', unlimitedApiRate: 'No API Rate Limit', apiRateLimitTooltip: 'API Rate Limit applies to all requests made through the Dify API, including text generation, chat conversations, workflow executions, and document processing.', documentProcessingPriority: ' Document Processing', @@ -72,6 +74,20 @@ const translation = { 'priority': 'Priority', 'top-priority': 'Top Priority', }, + triggerEvents: { + sandbox: '{{count,number}} Trigger Events', + professional: '{{count,number}} Trigger Events/month', + unlimited: 'Unlimited Trigger Events', + }, + workflowExecution: { + standard: 'Standard Workflow Execution', + faster: 'Faster Workflow Execution', + priority: 'Priority Workflow Execution', + }, + startNodes: { + limited: 'Up to {{count}} Start Nodes per Workflow', + unlimited: 'Unlimited Start Nodes per Workflow', + }, logsHistory: '{{days}} Log history', customTools: 'Custom Tools', unavailable: 'Unavailable', diff --git a/web/i18n/en-US/common.ts b/web/i18n/en-US/common.ts index ed0a596cbe..e0af44cc89 100644 --- a/web/i18n/en-US/common.ts +++ b/web/i18n/en-US/common.ts @@ -29,6 +29,11 @@ const translation = { refresh: 'Restart', reset: 'Reset', search: 'Search', + noSearchResults: 'No {{content}} were found', + resetKeywords: 'Reset keywords', + selectCount: '{{count}} Selected', + searchCount: 'Find {{count}} {{content}}', + noSearchCount: '0 {{content}}', change: 'Change', remove: 'Remove', send: 'Send', @@ -41,6 +46,7 @@ const translation = { downloadFailed: 'Download failed. Please try again later.', viewDetails: 'View Details', delete: 'Delete', + now: 'Now', deleteApp: 'Delete App', settings: 'Settings', setup: 'Setup', @@ -79,7 +85,9 @@ const translation = { placeholder: { input: 'Please enter', select: 'Please select', + search: 'Search...', }, + noData: 'No data', label: { optional: '(optional)', }, @@ -174,10 +182,10 @@ const translation = { emailSupport: 'Email Support', workspace: 'Workspace', createWorkspace: 'Create Workspace', - helpCenter: 'Docs', + helpCenter: 'View Docs', support: 'Support', compliance: 'Compliance', - communityFeedback: 'Feedback', + forum: 'Forum', roadmap: 'Roadmap', github: 'GitHub', community: 'Community', @@ -735,6 +743,7 @@ const translation = { uploadFromComputerLimit: 'Upload {{type}} cannot exceed {{size}}', pasteFileLinkInvalid: 'Invalid file link', fileExtensionNotSupport: 'File extension not supported', + fileExtensionBlocked: 'This file type is blocked for security reasons', }, tag: { placeholder: 'All Tags', @@ -769,6 +778,12 @@ const translation = { supportedFormats: 'Supports PNG, JPG, JPEG, WEBP and GIF', }, you: 'You', + dynamicSelect: { + error: 'Loading options failed', + noData: 'No options available', + loading: 'Loading options...', + selected: '{{count}} selected', + }, } export default translation diff --git a/web/i18n/en-US/dataset-creation.ts b/web/i18n/en-US/dataset-creation.ts index 54d5a54fb4..f32639a6b4 100644 --- a/web/i18n/en-US/dataset-creation.ts +++ b/web/i18n/en-US/dataset-creation.ts @@ -38,7 +38,7 @@ const translation = { button: 'Drag and drop file or folder, or', buttonSingleFile: 'Drag and drop file, or', browse: 'Browse', - tip: 'Supports {{supportTypes}}. Max {{batchCount}} in a batch and {{size}} MB each.', + tip: 'Supports {{supportTypes}}. Max {{batchCount}} in a batch and {{size}} MB each. Max total {{totalCount}} files.', validation: { typeError: 'File type not supported', size: 'File too large. Maximum is {{size}}MB', diff --git a/web/i18n/en-US/login.ts b/web/i18n/en-US/login.ts index 6015098022..dd923db217 100644 --- a/web/i18n/en-US/login.ts +++ b/web/i18n/en-US/login.ts @@ -1,5 +1,6 @@ const translation = { pageTitle: 'Log in to Dify', + pageTitleForE: 'Hey, let\'s get started!', welcome: '👋 Welcome! Please log in to get started.', email: 'Email address', emailPlaceholder: 'Your email', diff --git a/web/i18n/en-US/pipeline.ts b/web/i18n/en-US/pipeline.ts index 4b29bdbb00..8e5fd8a3e0 100644 --- a/web/i18n/en-US/pipeline.ts +++ b/web/i18n/en-US/pipeline.ts @@ -33,7 +33,7 @@ const translation = { }, ragToolSuggestions: { title: 'Suggestions for RAG', - noRecommendationPluginsInstalled: 'No recommended plugins installed, find more in <CustomLink>Marketplace</CustomLink>', + noRecommendationPlugins: 'No recommended plugins, find more in <CustomLink>Marketplace</CustomLink>', }, } diff --git a/web/i18n/en-US/plugin-trigger.ts b/web/i18n/en-US/plugin-trigger.ts new file mode 100644 index 0000000000..aedd0c6225 --- /dev/null +++ b/web/i18n/en-US/plugin-trigger.ts @@ -0,0 +1,186 @@ +const translation = { + subscription: { + title: 'Subscriptions', + listNum: '{{num}} subscriptions', + empty: { + title: 'No subscriptions', + button: 'New subscription', + }, + createButton: { + oauth: 'New subscription with OAuth', + apiKey: 'New subscription with API Key', + manual: 'Paste URL to create a new subscription', + }, + createSuccess: 'Subscription created successfully', + createFailed: 'Failed to create subscription', + maxCount: 'Max {{num}} subscriptions', + selectPlaceholder: 'Select subscription', + noSubscriptionSelected: 'No subscription selected', + subscriptionRemoved: 'Subscription removed', + list: { + title: 'Subscriptions', + addButton: 'Add', + tip: 'Receive events via Subscription', + item: { + enabled: 'Enabled', + disabled: 'Disabled', + credentialType: { + api_key: 'API Key', + oauth2: 'OAuth', + unauthorized: 'Manual', + }, + actions: { + delete: 'Delete', + deleteConfirm: { + title: 'Delete {{name}}?', + success: 'Subscription {{name}} deleted successfully', + error: 'Failed to delete subscription {{name}}', + content: 'Once deleted, this subscription cannot be recovered. Please confirm.', + contentWithApps: 'The current subscription is referenced by {{count}} applications. Deleting it will cause the configured applications to stop receiving subscription events.', + confirm: 'Confirm Delete', + cancel: 'Cancel', + confirmInputWarning: 'Please enter the correct name to confirm.', + confirmInputPlaceholder: 'Enter "{{name}}" to confirm.', + confirmInputTip: 'Please enter “{{name}}” to confirm.', + }, + }, + status: { + active: 'Active', + inactive: 'Inactive', + }, + usedByNum: 'Used by {{num}} workflows', + noUsed: 'No workflow used', + }, + }, + addType: { + title: 'Add subscription', + description: 'Choose how you want to create your trigger subscription', + options: { + apikey: { + title: 'Create with API Key', + description: 'Automatically create subscription using API credentials', + }, + oauth: { + title: 'Create with OAuth', + description: 'Authorize with third-party platform to create subscription', + clientSettings: 'OAuth Client Settings', + clientTitle: 'OAuth Client', + default: 'Default', + custom: 'Custom', + }, + manual: { + title: 'Manual Setup', + description: 'Paste URL to create a new subscription', + tip: 'Configure URL on third-party platform manually', + }, + }, + }, + }, + modal: { + steps: { + verify: 'Verify', + configuration: 'Configuration', + }, + common: { + cancel: 'Cancel', + back: 'Back', + next: 'Next', + create: 'Create', + verify: 'Verify', + authorize: 'Authorize', + creating: 'Creating...', + verifying: 'Verifying...', + authorizing: 'Authorizing...', + }, + oauthRedirectInfo: 'As no system client secrets found for this tool provider, setup it manually is required, for redirect_uri, please use', + apiKey: { + title: 'Create with API Key', + verify: { + title: 'Verify Credentials', + description: 'Please provide your API credentials to verify access', + error: 'Credential verification failed. Please check your API key.', + success: 'Credentials verified successfully', + }, + configuration: { + title: 'Configure Subscription', + description: 'Set up your subscription parameters', + }, + }, + oauth: { + title: 'Create with OAuth', + authorization: { + title: 'OAuth Authorization', + description: 'Authorize Dify to access your account', + redirectUrl: 'Redirect URL', + redirectUrlHelp: 'Use this URL in your OAuth app configuration', + authorizeButton: 'Authorize with {{provider}}', + waitingAuth: 'Waiting for authorization...', + authSuccess: 'Authorization successful', + authFailed: 'Failed to get OAuth authorization information', + waitingJump: 'Authorized, waiting for jump', + }, + configuration: { + title: 'Configure Subscription', + description: 'Set up your subscription parameters after authorization', + success: 'OAuth configuration successful', + failed: 'OAuth configuration failed', + }, + remove: { + success: 'OAuth remove successful', + failed: 'OAuth remove failed', + }, + save: { + success: 'OAuth configuration saved successfully', + }, + }, + manual: { + title: 'Manual Setup', + description: 'Configure your webhook subscription manually', + logs: { + title: 'Request Logs', + request: 'Request', + loading: 'Awaiting request from {{pluginName}}...', + }, + }, + form: { + subscriptionName: { + label: 'Subscription Name', + placeholder: 'Enter subscription name', + required: 'Subscription name is required', + }, + callbackUrl: { + label: 'Callback URL', + description: 'This URL will receive webhook events', + tooltip: 'Provide a publicly accessible endpoint that can receive callback requests from the trigger provider.', + placeholder: 'Generating...', + privateAddressWarning: 'This URL appears to be an internal address, which may cause webhook requests to fail. You may change TRIGGER_URL to a public address.', + }, + }, + errors: { + createFailed: 'Failed to create subscription', + verifyFailed: 'Failed to verify credentials', + authFailed: 'Authorization failed', + networkError: 'Network error, please try again', + }, + }, + events: { + title: 'Available Events', + description: 'Events that this trigger plugin can subscribe to', + empty: 'No events available', + event: 'Event', + events: 'Events', + actionNum: '{{num}} {{event}} INCLUDED', + item: { + parameters: '{{count}} parameters', + noParameters: 'No parameters', + }, + output: 'Output', + }, + node: { + status: { + warning: 'Disconnect', + }, + }, +} + +export default translation diff --git a/web/i18n/en-US/plugin.ts b/web/i18n/en-US/plugin.ts index 18fc69c841..62a5f35c0b 100644 --- a/web/i18n/en-US/plugin.ts +++ b/web/i18n/en-US/plugin.ts @@ -8,6 +8,7 @@ const translation = { tools: 'Tools', agents: 'Agent Strategies', extensions: 'Extensions', + triggers: 'Triggers', bundles: 'Bundles', datasources: 'Data Sources', }, @@ -16,6 +17,7 @@ const translation = { tool: 'Tool', agent: 'Agent Strategy', extension: 'Extension', + trigger: 'Trigger', bundle: 'Bundle', datasource: 'Data Source', }, @@ -62,6 +64,7 @@ const translation = { checkUpdate: 'Check Update', viewDetail: 'View Detail', remove: 'Remove', + back: 'Back', }, actionNum: '{{num}} {{action}} INCLUDED', strategyNum: '{{num}} {{strategy}} INCLUDED', @@ -306,6 +309,12 @@ const translation = { connectedWorkspace: 'Connected Workspace', emptyAuth: 'Please configure authentication', }, + readmeInfo: { + title: 'README', + needHelpCheckReadme: 'Need help? Check the README.', + noReadmeAvailable: 'No README available', + failedToFetch: 'Failed to fetch README', + }, } export default translation diff --git a/web/i18n/en-US/tools.ts b/web/i18n/en-US/tools.ts index 3fba10447f..308d4b2b05 100644 --- a/web/i18n/en-US/tools.ts +++ b/web/i18n/en-US/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'Create Custom Tool', customToolTip: 'Learn more about Dify custom tools', type: { - all: 'All', builtIn: 'Tools', custom: 'Custom', workflow: 'Workflow', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: 'After configuring credentials, all members within the workspace can use this tool when orchestrating applications.', }, includeToolNum: '{{num}} {{action}} included', - addTool: 'Add Tool', addToolModal: { type: 'type', category: 'category', - add: 'add', added: 'added', - manageInTools: 'Manage in Tools', custom: { title: 'No custom tool available', tip: 'Create a custom tool', @@ -203,6 +199,12 @@ const translation = { timeout: 'Timeout', sseReadTimeout: 'SSE Read Timeout', timeoutPlaceholder: '30', + authentication: 'Authentication', + useDynamicClientRegistration: 'Use Dynamic Client Registration', + clientID: 'Client ID', + clientSecret: 'Client Secret', + clientSecretPlaceholder: 'Client Secret', + configurations: 'Configurations', }, delete: 'Remove MCP Server', deleteConfirmTitle: 'Would you like to remove {{mcp}}?', diff --git a/web/i18n/en-US/workflow.ts b/web/i18n/en-US/workflow.ts index e07fc3f109..92a0b110c7 100644 --- a/web/i18n/en-US/workflow.ts +++ b/web/i18n/en-US/workflow.ts @@ -9,8 +9,11 @@ const translation = { publish: 'Publish', update: 'Update', publishUpdate: 'Publish Update', - run: 'Run', + run: 'Test Run', running: 'Running', + listening: 'Listening', + chooseStartNodeToRun: 'Choose the start node to run', + runAllTriggers: 'Run all triggers', inRunMode: 'In Run Mode', inPreview: 'In Preview', inPreviewMode: 'In Preview Mode', @@ -47,7 +50,8 @@ const translation = { needConnectTip: 'This step is not connected to anything', maxTreeDepth: 'Maximum limit of {{depth}} nodes per branch', needAdd: '{{node}} node must be added', - needEndNode: 'The End node must be added', + needOutputNode: 'The Output node must be added', + needStartNode: 'At least one start node must be added', needAnswerNode: 'The Answer node must be added', workflowProcess: 'Workflow Process', notRunning: 'Not running yet', @@ -77,12 +81,14 @@ const translation = { exportSVG: 'Export as SVG', currentView: 'Current View', currentWorkflow: 'Current Workflow', + moreActions: 'More Actions', model: 'Model', workflowAsTool: 'Workflow as Tool', configureRequired: 'Configure Required', configure: 'Configure', manageInTools: 'Manage in Tools', workflowAsToolTip: 'Tool reconfiguration is required after the workflow update.', + workflowAsToolDisabledHint: 'Publish the latest workflow and ensure a connected User Input node before configuring it as a tool.', viewDetailInTracingPanel: 'View details', syncingData: 'Syncing data, just a few seconds.', importDSL: 'Import DSL', @@ -140,6 +146,19 @@ const translation = { export: 'Export DSL with secret values ', }, }, + globalVar: { + title: 'System Variables', + description: 'System variables are global variables that can be referenced by any node without wiring when the type is correct, such as end-user ID and workflow ID.', + fieldsDescription: { + conversationId: 'Conversation ID', + dialogCount: 'Conversation Count', + userId: 'User ID', + triggerTimestamp: 'Application start timestamp', + appId: 'Application ID', + workflowId: 'Workflow ID', + workflowRunId: 'Workflow run ID', + }, + }, sidebar: { exportWarning: 'Export Current Saved Version', exportWarningDesc: 'This will export the current saved version of your workflow. If you have unsaved changes in the editor, please save them first by using the export option in the workflow canvas.', @@ -213,6 +232,16 @@ const translation = { invalidVariable: 'Invalid variable', noValidTool: '{{field}} no valid tool selected', toolParameterRequired: '{{field}}: parameter [{{param}}] is required', + startNodeRequired: 'Please add a start node first before {{operation}}', + }, + error: { + startNodeRequired: 'Please add a start node first before {{operation}}', + operations: { + connectingNodes: 'connecting nodes', + addingNodes: 'adding nodes', + modifyingWorkflow: 'modifying workflow', + updatingWorkflow: 'updating workflow', + }, }, singleRun: { testRun: 'Test Run', @@ -227,8 +256,11 @@ const translation = { }, tabs: { 'searchBlock': 'Search node', + 'start': 'Start', 'blocks': 'Nodes', 'searchTool': 'Search tool', + 'searchTrigger': 'Search triggers...', + 'allTriggers': 'All triggers', 'tools': 'Tools', 'allTool': 'All', 'plugin': 'Plugin', @@ -239,15 +271,28 @@ const translation = { 'transform': 'Transform', 'utilities': 'Utilities', 'noResult': 'No match found', + 'noPluginsFound': 'No plugins were found', + 'requestToCommunity': 'Requests to the community', 'agent': 'Agent Strategy', 'allAdded': 'All added', 'addAll': 'Add all', 'sources': 'Sources', 'searchDataSource': 'Search Data Source', + 'featuredTools': 'Featured', + 'showMoreFeatured': 'Show more', + 'showLessFeatured': 'Show less', + 'installed': 'Installed', + 'pluginByAuthor': 'By {{author}}', + 'usePlugin': 'Select tool', + 'hideActions': 'Hide tools', + 'noFeaturedPlugins': 'Discover more tools in Marketplace', + 'noFeaturedTriggers': 'Discover more triggers in Marketplace', + 'startDisabledTip': 'Trigger node and user input node are mutually exclusive.', }, blocks: { - 'start': 'Start', - 'end': 'End', + 'start': 'User Input', + 'originalStartNode': 'original start node', + 'end': 'Output', 'answer': 'Answer', 'llm': 'LLM', 'knowledge-retrieval': 'Knowledge Retrieval', @@ -270,10 +315,14 @@ const translation = { 'loop-end': 'Exit Loop', 'knowledge-index': 'Knowledge Base', 'datasource': 'Data Source', + 'trigger-schedule': 'Schedule Trigger', + 'trigger-webhook': 'Webhook Trigger', + 'trigger-plugin': 'Plugin Trigger', }, + customWebhook: 'Custom Webhook', blocksAbout: { 'start': 'Define the initial parameters for launching a workflow', - 'end': 'Define the end and result type of a workflow', + 'end': 'Define the output and result type of a workflow', 'answer': 'Define the reply content of a chat conversation', 'llm': 'Invoking large language models to answer questions or process natural language', 'knowledge-retrieval': 'Allows you to query text content related to user questions from the Knowledge', @@ -294,7 +343,11 @@ const translation = { 'agent': 'Invoking large language models to answer questions or process natural language', 'knowledge-index': 'Knowledge Base About', 'datasource': 'Data Source About', + 'trigger-schedule': 'Time-based workflow trigger that starts workflows on a schedule', + 'trigger-webhook': 'Webhook Trigger receives HTTP pushes from third-party systems to automatically trigger workflows.', + 'trigger-plugin': 'Third-party integration trigger that starts workflows from external platform events', }, + difyTeam: 'Dify Team', operator: { zoomIn: 'Zoom In', zoomOut: 'Zoom Out', @@ -324,22 +377,24 @@ const translation = { panel: { userInputField: 'User Input Field', changeBlock: 'Change Node', - helpLink: 'Help Link', + helpLink: 'View Docs', about: 'About', createdBy: 'Created By ', nextStep: 'Next Step', addNextStep: 'Add the next step in this workflow', selectNextStep: 'Select Next Step', runThisStep: 'Run this step', - moveToThisNode: 'Move to this node', checklist: 'Checklist', checklistTip: 'Make sure all issues are resolved before publishing', checklistResolved: 'All issues are resolved', + goTo: 'Go to', + startNode: 'Start Node', organizeBlocks: 'Organize nodes', change: 'Change', optional: '(optional)', maximize: 'Maximize Canvas', minimize: 'Exit Full Screen', + scrollToSelectedNode: 'Scroll to selected node', optional_and_hidden: '(optional & hidden)', }, nodes: { @@ -719,6 +774,50 @@ const translation = { json: 'tool generated json', }, }, + triggerPlugin: { + authorized: 'Authorized', + notConfigured: 'Not Configured', + notAuthorized: 'Not Authorized', + selectSubscription: 'Select Subscription', + availableSubscriptions: 'Available Subscriptions', + addSubscription: 'Add New Subscription', + removeSubscription: 'Remove Subscription', + subscriptionRemoved: 'Subscription removed successfully', + error: 'Error', + configuration: 'Configuration', + remove: 'Remove', + or: 'OR', + useOAuth: 'Use OAuth', + useApiKey: 'Use API Key', + authenticationFailed: 'Authentication failed', + authenticationSuccess: 'Authentication successful', + oauthConfigFailed: 'OAuth configuration failed', + configureOAuthClient: 'Configure OAuth Client', + oauthClientDescription: 'Configure OAuth client credentials to enable authentication', + oauthClientSaved: 'OAuth client configuration saved successfully', + configureApiKey: 'Configure API Key', + apiKeyDescription: 'Configure API key credentials for authentication', + apiKeyConfigured: 'API key configured successfully', + configurationFailed: 'Configuration failed', + failedToStart: 'Failed to start authentication flow', + credentialsVerified: 'Credentials verified successfully', + credentialVerificationFailed: 'Credential verification failed', + verifyAndContinue: 'Verify & Continue', + configureParameters: 'Configure Parameters', + parametersDescription: 'Configure trigger parameters and properties', + configurationComplete: 'Configuration Complete', + configurationCompleteDescription: 'Your trigger has been configured successfully', + configurationCompleteMessage: 'Your trigger configuration is now complete and ready to use.', + parameters: 'Parameters', + properties: 'Properties', + propertiesDescription: 'Additional configuration properties for this trigger', + noConfigurationRequired: 'No additional configuration required for this trigger.', + subscriptionName: 'Subscription Name', + subscriptionNameDescription: 'Enter a unique name for this trigger subscription', + subscriptionNamePlaceholder: 'Enter subscription name...', + subscriptionNameRequired: 'Subscription name is required', + subscriptionRequired: 'Subscription is required', + }, questionClassifiers: { model: 'model', inputVars: 'Input Variables', @@ -788,6 +887,8 @@ const translation = { removeAbnormalOutput: 'Remove Abnormal Output', }, answerNodeWarningDesc: 'Parallel mode warning: Answer nodes, conversation variable assignments, and persistent read/write operations within iterations may cause exceptions.', + flattenOutput: 'Flatten Output', + flattenOutputDesc: 'When enabled, if all iteration outputs are arrays, they will be flattened into a single array. When disabled, outputs will maintain a nested array structure.', }, loop: { deleteTitle: 'Delete Loop Node?', @@ -964,6 +1065,108 @@ const translation = { rerankingModelIsRequired: 'Reranking model is required', rerankingModelIsInvalid: 'Reranking model is invalid', }, + triggerSchedule: { + title: 'Schedule', + nodeTitle: 'Schedule Trigger', + notConfigured: 'Not configured', + useCronExpression: 'Use cron expression', + useVisualPicker: 'Use visual picker', + frequency: { + label: 'FREQUENCY', + hourly: 'Hourly', + daily: 'Daily', + weekly: 'Weekly', + monthly: 'Monthly', + }, + selectFrequency: 'Select frequency', + frequencyLabel: 'Frequency', + nextExecution: 'Next execution', + weekdays: 'Week days', + time: 'Time', + cronExpression: 'Cron expression', + nextExecutionTime: 'NEXT EXECUTION TIME', + nextExecutionTimes: 'Next 5 execution times', + startTime: 'Start Time', + executeNow: 'Execution now', + selectDateTime: 'Select Date & Time', + hours: 'Hours', + minutes: 'Minutes', + onMinute: 'On Minute', + days: 'Days', + lastDay: 'Last day', + lastDayTooltip: 'Not all months have 31 days. Use the \'last day\' option to select each month\'s final day.', + mode: 'Mode', + timezone: 'Timezone', + visualConfig: 'Visual Configuration', + monthlyDay: 'Monthly Day', + executionTime: 'Execution Time', + invalidTimezone: 'Invalid timezone', + invalidCronExpression: 'Invalid cron expression', + noValidExecutionTime: 'No valid execution time can be calculated', + executionTimeCalculationError: 'Failed to calculate execution times', + invalidFrequency: 'Invalid frequency', + invalidStartTime: 'Invalid start time', + startTimeMustBeFuture: 'Start time must be in the future', + invalidTimeFormat: 'Invalid time format (expected HH:MM AM/PM)', + invalidWeekday: 'Invalid weekday: {{weekday}}', + invalidMonthlyDay: 'Monthly day must be between 1-31 or "last"', + invalidOnMinute: 'On minute must be between 0-59', + invalidExecutionTime: 'Invalid execution time', + executionTimeMustBeFuture: 'Execution time must be in the future', + }, + triggerWebhook: { + title: 'Webhook Trigger', + nodeTitle: '🔗 Webhook Trigger', + configPlaceholder: 'Webhook trigger configuration will be implemented here', + webhookUrl: 'Webhook URL', + webhookUrlPlaceholder: 'Click generate to create webhook URL', + generate: 'Generate', + copy: 'Copy', + test: 'Test', + urlGenerated: 'Webhook URL generated successfully', + urlGenerationFailed: 'Failed to generate webhook URL', + urlCopied: 'URL copied to clipboard', + method: 'Method', + contentType: 'Content Type', + queryParameters: 'Query Parameters', + headerParameters: 'Header Parameters', + requestBodyParameters: 'Request Body Parameters', + parameterName: 'Variable name', + varName: 'Variable name', + varType: 'Type', + varNamePlaceholder: 'Enter variable name...', + required: 'Required', + addParameter: 'Add', + addHeader: 'Add', + noParameters: 'No parameters configured', + noQueryParameters: 'No query parameters configured', + noHeaders: 'No headers configured', + noBodyParameters: 'No body parameters configured', + debugUrlTitle: 'For test runs, always use this URL', + debugUrlCopy: 'Click to copy', + debugUrlCopied: 'Copied!', + debugUrlPrivateAddressWarning: 'This URL appears to be an internal address, which may cause webhook requests to fail. You may change TRIGGER_URL to a public address.', + errorHandling: 'Error Handling', + errorStrategy: 'Error Handling', + responseConfiguration: 'Response', + asyncMode: 'Async Mode', + statusCode: 'Status Code', + responseBody: 'Response Body', + responseBodyPlaceholder: 'Write your response body here', + headers: 'Headers', + validation: { + webhookUrlRequired: 'Webhook URL is required', + invalidParameterType: 'Invalid parameter type "{{type}}" for parameter "{{name}}"', + }, + }, + }, + triggerStatus: { + enabled: 'TRIGGER', + disabled: 'TRIGGER • DISABLED', + }, + entryNodeStatus: { + enabled: 'START', + disabled: 'START • DISABLED', }, tracing: { stopBy: 'Stop by {{user}}', @@ -1025,6 +1228,18 @@ const translation = { view: 'View log', edited: 'Edited', reset: 'Reset to last run value', + listening: { + title: 'Listening for events from triggers...', + tip: 'You can now simulate event triggers by sending test requests to HTTP {{nodeName}} endpoint or use it as a callback URL for live event debugging. All outputs can be viewed directly in the Variable Inspector.', + tipPlugin: 'Now you can create events in {{- pluginName}}, and retrieve outputs from these events in the Variable Inspector.', + tipSchedule: 'Listening for events from schedule triggers.\nNext scheduled run: {{nextTriggerTime}}', + tipFallback: 'Await incoming trigger events. Outputs will appear here.', + defaultNodeName: 'this trigger', + defaultPluginName: 'this plugin trigger', + defaultScheduleTime: 'Not configured', + selectedTriggers: 'selected triggers', + stopButton: 'Stop', + }, trigger: { normal: 'Variable Inspect', running: 'Caching running status', @@ -1050,6 +1265,22 @@ const translation = { noDependents: 'No dependents', }, }, + onboarding: { + title: 'Select a start node to begin', + description: 'Different start nodes have different capabilities. Don\'t worry, you can always change them later.', + userInputFull: 'User Input (original start node)', + userInputDescription: 'Start node that allows setting user input variables, with web app, service API, MCP server, and workflow as tool capabilities.', + trigger: 'Trigger', + triggerDescription: 'Triggers can serve as the start node of a workflow, such as scheduled tasks, custom webhooks, or integrations with other apps.', + back: 'Back', + learnMore: 'Learn more', + aboutStartNode: 'about start node.', + escTip: { + press: 'Press', + key: 'esc', + toDismiss: 'to dismiss', + }, + }, } export default translation diff --git a/web/i18n/es-ES/app-debug.ts b/web/i18n/es-ES/app-debug.ts index 6885c0d11f..76aa28d03f 100644 --- a/web/i18n/es-ES/app-debug.ts +++ b/web/i18n/es-ES/app-debug.ts @@ -255,7 +255,6 @@ const translation = { variableTable: { key: 'Clave de Variable', name: 'Nombre del Campo de Entrada del Usuario', - optional: 'Opcional', type: 'Tipo de Entrada', action: 'Acciones', typeString: 'Cadena', diff --git a/web/i18n/es-ES/app-log.ts b/web/i18n/es-ES/app-log.ts index 0044dee709..aeca32d420 100644 --- a/web/i18n/es-ES/app-log.ts +++ b/web/i18n/es-ES/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Trimestre hasta la fecha', yearToDate: 'Año hasta la fecha', allTime: 'Todo el tiempo', + custom: 'Personalizado', + last30days: 'Últimos 30 días', }, annotation: { all: 'Todos', diff --git a/web/i18n/es-ES/billing.ts b/web/i18n/es-ES/billing.ts index a8180e2d07..1632776e30 100644 --- a/web/i18n/es-ES/billing.ts +++ b/web/i18n/es-ES/billing.ts @@ -76,7 +76,7 @@ const translation = { priceTip: 'por espacio de trabajo/', teamMember_one: '{{count, número}} Miembro del Equipo', getStarted: 'Comenzar', - apiRateLimitUnit: '{{count, número}}/día', + apiRateLimitUnit: '{{count, número}}/mes', freeTrialTipSuffix: 'No se requiere tarjeta de crédito', unlimitedApiRate: 'Sin límite de tasa de API', apiRateLimit: 'Límite de tasa de API', diff --git a/web/i18n/es-ES/common.ts b/web/i18n/es-ES/common.ts index cd854f19fa..3e3ffbd8e3 100644 --- a/web/i18n/es-ES/common.ts +++ b/web/i18n/es-ES/common.ts @@ -165,7 +165,6 @@ const translation = { workspace: 'Espacio de trabajo', createWorkspace: 'Crear espacio de trabajo', helpCenter: 'Ayuda', - communityFeedback: 'Comentarios', roadmap: 'Hoja de ruta', community: 'Comunidad', about: 'Acerca de', @@ -174,6 +173,7 @@ const translation = { compliance: 'Cumplimiento', github: 'GitHub', contactUs: 'Contáctenos', + forum: 'Foro', }, settings: { accountGroup: 'CUENTA', @@ -726,6 +726,7 @@ const translation = { fileExtensionNotSupport: 'Extensión de archivo no compatible', pasteFileLinkInputPlaceholder: 'Introduzca la URL...', uploadFromComputerLimit: 'El archivo de carga no puede exceder {{size}}', + fileExtensionBlocked: 'Este tipo de archivo está bloqueado por motivos de seguridad', }, license: { expiring: 'Caduca en un día', diff --git a/web/i18n/es-ES/login.ts b/web/i18n/es-ES/login.ts index ba8ad292cc..cbc223e7da 100644 --- a/web/i18n/es-ES/login.ts +++ b/web/i18n/es-ES/login.ts @@ -120,6 +120,7 @@ const translation = { welcome: '👋 ¡Bienvenido! Por favor, completa los detalles para comenzar.', verifyMail: 'Continuar con el código de verificación', }, + pageTitleForE: '¡Hola, vamos a empezar!', } export default translation diff --git a/web/i18n/es-ES/tools.ts b/web/i18n/es-ES/tools.ts index 10584c41ca..f85a44882e 100644 --- a/web/i18n/es-ES/tools.ts +++ b/web/i18n/es-ES/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'Crear Herramienta Personalizada', customToolTip: 'Aprende más sobre las herramientas personalizadas de Dify', type: { - all: 'Todas', builtIn: 'Incorporadas', custom: 'Personalizadas', workflow: 'Flujo de Trabajo', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: 'Después de configurar las credenciales, todos los miembros dentro del espacio de trabajo pueden usar esta herramienta al orquestar aplicaciones.', }, includeToolNum: '{{num}} herramientas incluidas', - addTool: 'Agregar Herramienta', addToolModal: { type: 'tipo', category: 'categoría', - add: 'agregar', added: 'agregada', - manageInTools: 'Administrar en Herramientas', custom: { title: 'No hay herramienta personalizada disponible', tip: 'Crear una herramienta personalizada', @@ -203,6 +199,12 @@ const translation = { headerValue: 'Valor del encabezado', noHeaders: 'No se han configurado encabezados personalizados', headerKey: 'Nombre del encabezado', + authentication: 'Autenticación', + clientID: 'ID del Cliente', + clientSecretPlaceholder: 'Secreto del Cliente', + useDynamicClientRegistration: 'Usar registro dinámico de clientes', + clientSecret: 'Secreto del Cliente', + configurations: 'Configuraciones', }, delete: 'Eliminar servidor MCP', deleteConfirmTitle: '¿Eliminar {{mcp}}?', diff --git a/web/i18n/es-ES/workflow.ts b/web/i18n/es-ES/workflow.ts index d7a6bef9e7..dd9519b68f 100644 --- a/web/i18n/es-ES/workflow.ts +++ b/web/i18n/es-ES/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Publicado', publish: 'Publicar', update: 'Actualizar', - run: 'Ejecutar', + run: 'Ejecutar prueba', running: 'Ejecutando', inRunMode: 'En modo de ejecución', inPreview: 'En vista previa', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Historial de ejecución', goBackToEdit: 'Volver al editor', conversationLog: 'Registro de conversación', - features: 'Funcionalidades', debugAndPreview: 'Vista previa', restart: 'Reiniciar', currentDraft: 'Borrador actual', @@ -93,8 +92,6 @@ const translation = { branch: 'RAMA', fileUploadTip: 'Las funciones de carga de imágenes se han actualizado a la carga de archivos.', ImageUploadLegacyTip: 'Ahora puede crear variables de tipo de archivo en el formulario de inicio. Ya no admitiremos la función de carga de imágenes en el futuro.', - featuresDescription: 'Mejorar la experiencia del usuario de la aplicación web', - featuresDocLink: 'Aprende más', importWarning: 'Cautela', importWarningDetails: 'La diferencia de versión de DSL puede afectar a ciertas características', openInExplore: 'Abrir en Explorar', @@ -110,11 +107,12 @@ const translation = { publishUpdate: 'Publicar actualización', exportImage: 'Exportar imagen', needAnswerNode: 'Se debe agregar el nodo de respuesta', - needEndNode: 'Se debe agregar el nodo Final', + needOutputNode: 'Se debe agregar el nodo de Salida', addBlock: 'Agregar nodo', tagBound: 'Número de aplicaciones que utilizan esta etiqueta', currentView: 'Vista actual', currentWorkflow: 'Flujo de trabajo actual', + moreActions: 'Más acciones', }, env: { envPanelTitle: 'Variables de Entorno', @@ -139,6 +137,19 @@ const translation = { export: 'Exportar DSL con valores secretos', }, }, + globalVar: { + title: 'Variables del sistema', + description: 'Las variables del sistema son variables globales que cualquier nodo puede usar sin conexiones cuando el tipo es correcto, como el ID del usuario final y el ID del flujo de trabajo.', + fieldsDescription: { + conversationId: 'ID de la conversación', + dialogCount: 'Número de conversaciones', + userId: 'ID de usuario', + triggerTimestamp: 'Marca de tiempo de inicio de la aplicación', + appId: 'ID de la aplicación', + workflowId: 'ID del flujo de trabajo', + workflowRunId: 'ID de ejecución del flujo de trabajo', + }, + }, chatVariable: { panelTitle: 'Variables de Conversación', panelDescription: 'Las Variables de Conversación se utilizan para almacenar información interactiva que el LLM necesita recordar, incluyendo el historial de conversación, archivos subidos y preferencias del usuario. Son de lectura y escritura.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Inicio', - 'end': 'Fin', + 'end': 'Salida', 'answer': 'Respuesta', 'llm': 'LLM', 'knowledge-retrieval': 'Recuperación de conocimiento', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Define los parámetros iniciales para iniciar un flujo de trabajo', - 'end': 'Define el final y el tipo de resultado de un flujo de trabajo', + 'end': 'Define la salida y el tipo de resultado de un flujo de trabajo', 'answer': 'Define el contenido de respuesta de una conversación de chat', 'llm': 'Invoca modelos de lenguaje grandes para responder preguntas o procesar lenguaje natural', 'knowledge-retrieval': 'Te permite consultar contenido de texto relacionado con las preguntas de los usuarios desde el conocimiento', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Campo de entrada del usuario', - helpLink: 'Enlace de ayuda', + helpLink: 'Ayuda', about: 'Acerca de', createdBy: 'Creado por ', nextStep: 'Siguiente paso', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Se resolvieron todos los problemas', change: 'Cambiar', optional: '(opcional)', - moveToThisNode: 'Mueve a este nodo', organizeBlocks: 'Organizar nodos', addNextStep: 'Agrega el siguiente paso en este flujo de trabajo', changeBlock: 'Cambiar Nodo', selectNextStep: 'Seleccionar siguiente paso', maximize: 'Maximizar Canvas', minimize: 'Salir de pantalla completa', + scrollToSelectedNode: 'Desplácese hasta el nodo seleccionado', optional_and_hidden: '(opcional y oculto)', }, nodes: { diff --git a/web/i18n/fa-IR/app-debug.ts b/web/i18n/fa-IR/app-debug.ts index 141189032a..857dee9418 100644 --- a/web/i18n/fa-IR/app-debug.ts +++ b/web/i18n/fa-IR/app-debug.ts @@ -588,7 +588,6 @@ const translation = { typeString: 'رشته', name: 'نام فیلد ورودی کاربر', type: 'نوع ورودی', - optional: 'اختیاری', }, varKeyError: {}, otherError: { diff --git a/web/i18n/fa-IR/app-log.ts b/web/i18n/fa-IR/app-log.ts index 526fa01e76..93936e6c4e 100644 --- a/web/i18n/fa-IR/app-log.ts +++ b/web/i18n/fa-IR/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'از ابتدای فصل تاکنون', yearToDate: 'از ابتدای سال تاکنون', allTime: 'همه زمان‌ها', + last30days: '۳۰ روز گذشته', + custom: 'سفارشی', }, annotation: { all: 'همه', diff --git a/web/i18n/fa-IR/billing.ts b/web/i18n/fa-IR/billing.ts index 3749036f3c..e5121bb65b 100644 --- a/web/i18n/fa-IR/billing.ts +++ b/web/i18n/fa-IR/billing.ts @@ -73,7 +73,7 @@ const translation = { }, ragAPIRequestTooltip: 'به تعداد درخواست‌های API که فقط قابلیت‌های پردازش پایگاه دانش Dify را فراخوانی می‌کنند اشاره دارد.', receiptInfo: 'فقط صاحب تیم و مدیر تیم می‌توانند اشتراک تهیه کنند و اطلاعات صورتحساب را مشاهده کنند', - apiRateLimitUnit: '{{count,number}}/روز', + apiRateLimitUnit: '{{count,number}}/ماه', cloud: 'سرویس ابری', documents: '{{count,number}} سندهای دانش', self: 'خود میزبان', diff --git a/web/i18n/fa-IR/common.ts b/web/i18n/fa-IR/common.ts index dced7bfe82..fb81a9e98e 100644 --- a/web/i18n/fa-IR/common.ts +++ b/web/i18n/fa-IR/common.ts @@ -165,7 +165,6 @@ const translation = { workspace: 'فضای کاری', createWorkspace: 'ایجاد فضای کاری', helpCenter: 'راهنما', - communityFeedback: 'بازخورد', roadmap: 'نقشه راه', community: 'انجمن', about: 'درباره', @@ -174,6 +173,7 @@ const translation = { compliance: 'انطباق', support: 'پشتیبانی', contactUs: 'با ما تماس بگیرید', + forum: 'انجمن', }, settings: { accountGroup: 'حساب کاربری', @@ -726,6 +726,7 @@ const translation = { uploadFromComputerUploadError: 'آپلود فایل انجام نشد، لطفا دوباره آپلود کنید.', pasteFileLink: 'پیوند فایل را جایگذاری کنید', uploadFromComputerLimit: 'آپلود فایل نمی تواند از {{size}} تجاوز کند', + fileExtensionBlocked: 'این نوع فایل به دلایل امنیتی مسدود شده است', }, license: { expiring_plural: 'انقضا در {{count}} روز', diff --git a/web/i18n/fa-IR/login.ts b/web/i18n/fa-IR/login.ts index b57687cf5d..83382f3c9d 100644 --- a/web/i18n/fa-IR/login.ts +++ b/web/i18n/fa-IR/login.ts @@ -120,6 +120,7 @@ const translation = { noAccount: 'حساب کاربری ندارید؟', verifyMail: 'ادامه با کد تأیید', }, + pageTitleForE: 'هی، بیا شروع کنیم!', } export default translation diff --git a/web/i18n/fa-IR/tools.ts b/web/i18n/fa-IR/tools.ts index 587c16d960..bc0510341b 100644 --- a/web/i18n/fa-IR/tools.ts +++ b/web/i18n/fa-IR/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'ایجاد ابزار سفارشی', customToolTip: 'بیشتر در مورد ابزارهای سفارشی Dify بیاموزید', type: { - all: 'همه', builtIn: 'سفارشی شده', custom: 'سفارشی', workflow: 'جریان کار', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: 'پس از پیکربندی اعتبارنامه‌ها، همه اعضای موجود در فضای کاری می‌توانند از این ابزار هنگام هماهنگی برنامه‌ها استفاده کنند.', }, includeToolNum: '{{num}} ابزار شامل شد', - addTool: 'افزودن ابزار', addToolModal: { type: 'نوع', category: 'دسته‌بندی', - add: 'افزودن', added: 'افزوده شد', - manageInTools: 'مدیریت در ابزارها', custom: { title: 'هیچ ابزار سفارشی موجود نیست', tip: 'یک ابزار سفارشی ایجاد کنید', @@ -203,6 +199,12 @@ const translation = { noHeaders: 'هیچ هدر سفارشی پیکربندی نشده است', headersTip: 'هدرهای HTTP اضافی برای ارسال با درخواست‌های سرور MCP', maskedHeadersTip: 'مقدارهای هدر به خاطر امنیت مخفی شده‌اند. تغییرات مقادیر واقعی را به‌روزرسانی خواهد کرد.', + authentication: 'احراز هویت', + configurations: 'تنظیمات', + clientSecretPlaceholder: 'رمز مشتری', + clientID: 'شناسه مشتری', + clientSecret: 'رمز مشتری', + useDynamicClientRegistration: 'استفاده از ثبت‌نام پویا برای مشتری', }, delete: 'حذف سرور MCP', deleteConfirmTitle: 'آیا مایل به حذف {mcp} هستید؟', diff --git a/web/i18n/fa-IR/workflow.ts b/web/i18n/fa-IR/workflow.ts index aba3a25010..e27b8934e2 100644 --- a/web/i18n/fa-IR/workflow.ts +++ b/web/i18n/fa-IR/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'منتشر شده', publish: 'انتشار', update: 'به‌روزرسانی', - run: 'اجرا', + run: 'اجرای تست', running: 'در حال اجرا', inRunMode: 'در حالت اجرا', inPreview: 'در پیش‌نمایش', @@ -18,7 +18,6 @@ const translation = { runHistory: 'تاریخچه اجرا', goBackToEdit: 'بازگشت به ویرایشگر', conversationLog: 'گزارش مکالمات', - features: 'ویژگی‌ها', debugAndPreview: 'پیش‌نمایش', restart: 'راه‌اندازی مجدد', currentDraft: 'پیش‌نویس فعلی', @@ -91,8 +90,6 @@ const translation = { addParallelNode: 'افزودن گره موازی', parallel: 'موازی', branch: 'شاخه', - featuresDocLink: 'بیشتر بدانید', - featuresDescription: 'بهبود تجربه کاربری برنامه وب', ImageUploadLegacyTip: 'اکنون می توانید متغیرهای نوع فایل را در فرم شروع ایجاد کنید. ما دیگر از ویژگی آپلود تصویر در آینده پشتیبانی نخواهیم کرد.', fileUploadTip: 'ویژگی های آپلود تصویر برای آپلود فایل ارتقا یافته است.', importWarning: 'احتیاط', @@ -109,12 +106,13 @@ const translation = { exportImage: 'تصویر را صادر کنید', versionHistory: 'تاریخچه نسخه', publishUpdate: 'به‌روزرسانی منتشر کنید', - needEndNode: 'باید گره پایان اضافه شود', + needOutputNode: 'باید گره خروجی اضافه شود', needAnswerNode: 'باید گره پاسخ اضافه شود', addBlock: 'نود اضافه کنید', tagBound: 'تعداد برنامه‌هایی که از این برچسب استفاده می‌کنند', currentView: 'نمای فعلی', currentWorkflow: 'گردش کار فعلی', + moreActions: 'اقدامات بیشتر', }, env: { envPanelTitle: 'متغیرهای محیطی', @@ -139,6 +137,19 @@ const translation = { export: 'صادر کردن DSL با مقادیر مخفی', }, }, + globalVar: { + title: 'متغیرهای سیستمی', + description: 'متغیرهای سیستمی متغیرهای سراسری هستند که هر گره در صورت مطابقت نوع می‌تواند بدون سیم‌کشی از آن‌ها استفاده کند، مانند شناسه کاربر نهایی و شناسه گردش‌کار.', + fieldsDescription: { + conversationId: 'شناسه گفتگو', + dialogCount: 'تعداد گفتگو', + userId: 'شناسه کاربر', + triggerTimestamp: 'برچسب زمانی شروع اجرای برنامه', + appId: 'شناسه برنامه', + workflowId: 'شناسه گردش‌کار', + workflowRunId: 'شناسه اجرای گردش‌کار', + }, + }, chatVariable: { panelTitle: 'متغیرهای مکالمه', panelDescription: 'متغیرهای مکالمه برای ذخیره اطلاعات تعاملی که LLM نیاز به یادآوری دارد استفاده می‌شوند، از جمله تاریخچه مکالمه، فایل‌های آپلود شده و ترجیحات کاربر. آنها قابل خواندن و نوشتن هستند.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'شروع', - 'end': 'پایان', + 'end': 'خروجی', 'answer': 'پاسخ', 'llm': 'مدل زبان بزرگ', 'knowledge-retrieval': 'استخراج دانش', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'پارامترهای اولیه برای راه‌اندازی جریان کار را تعریف کنید', - 'end': 'پایان و نوع نتیجه یک جریان کار را تعریف کنید', + 'end': 'خروجی و نوع نتیجه یک جریان کار را تعریف کنید', 'answer': 'محتوای پاسخ مکالمه چت را تعریف کنید', 'llm': 'استفاده از مدل‌های زبان بزرگ برای پاسخ به سوالات یا پردازش زبان طبیعی', 'knowledge-retrieval': 'اجازه می‌دهد تا محتوای متنی مرتبط با سوالات کاربر از دانش استخراج شود', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'فیلد ورودی کاربر', - helpLink: 'لینک کمک', + helpLink: 'راهنما', about: 'درباره', createdBy: 'ساخته شده توسط', nextStep: 'مرحله بعدی', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'تمام مسائل حل شده‌اند', change: 'تغییر', optional: '(اختیاری)', - moveToThisNode: 'به این گره بروید', selectNextStep: 'گام بعدی را انتخاب کنید', changeBlock: 'تغییر گره', organizeBlocks: 'گره‌ها را سازماندهی کنید', addNextStep: 'مرحله بعدی را به این فرآیند اضافه کنید', minimize: 'خروج از حالت تمام صفحه', maximize: 'بیشینه‌سازی بوم', + scrollToSelectedNode: 'به گره انتخاب شده بروید', optional_and_hidden: '(اختیاری و پنهان)', }, nodes: { diff --git a/web/i18n/fr-FR/app-debug.ts b/web/i18n/fr-FR/app-debug.ts index d46735a71e..ca894192dc 100644 --- a/web/i18n/fr-FR/app-debug.ts +++ b/web/i18n/fr-FR/app-debug.ts @@ -259,7 +259,6 @@ const translation = { variableTable: { key: 'Clé Variable', name: 'Nom du champ d\'entrée de l\'utilisateur', - optional: 'Facultatif', type: 'Type d\'Entrée', action: 'Actions', typeString: 'Chaîne', diff --git a/web/i18n/fr-FR/app-log.ts b/web/i18n/fr-FR/app-log.ts index 42e25ba21f..061dad2eed 100644 --- a/web/i18n/fr-FR/app-log.ts +++ b/web/i18n/fr-FR/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Trimestre à ce jour', yearToDate: 'Année à ce jour', allTime: 'Tout le temps', + custom: 'Personnalisé', + last30days: 'Derniers 30 jours', }, annotation: { all: 'Tous', diff --git a/web/i18n/fr-FR/billing.ts b/web/i18n/fr-FR/billing.ts index a41eed7e23..9715a1e805 100644 --- a/web/i18n/fr-FR/billing.ts +++ b/web/i18n/fr-FR/billing.ts @@ -73,7 +73,7 @@ const translation = { ragAPIRequestTooltip: 'Fait référence au nombre d\'appels API invoquant uniquement les capacités de traitement de la base de connaissances de Dify.', receiptInfo: 'Seuls le propriétaire de l\'équipe et l\'administrateur de l\'équipe peuvent s\'abonner et consulter les informations de facturation', annotationQuota: 'Quota d’annotation', - apiRateLimitUnit: '{{count,number}}/jour', + apiRateLimitUnit: '{{count,number}}/mois', priceTip: 'par espace de travail/', freeTrialTipSuffix: 'Aucune carte de crédit requise', teamWorkspace: '{{count,number}} Espace de travail d\'équipe', diff --git a/web/i18n/fr-FR/common.ts b/web/i18n/fr-FR/common.ts index 96ecfe4151..3c8c4f0b78 100644 --- a/web/i18n/fr-FR/common.ts +++ b/web/i18n/fr-FR/common.ts @@ -161,7 +161,6 @@ const translation = { workspace: 'Espace de travail', createWorkspace: 'Créer un Espace de Travail', helpCenter: 'Aide', - communityFeedback: 'Retour d\'information', roadmap: 'Feuille de route', community: 'Communauté', about: 'À propos', @@ -170,6 +169,7 @@ const translation = { github: 'GitHub', compliance: 'Conformité', contactUs: 'Contactez-nous', + forum: 'Forum', }, settings: { accountGroup: 'COMPTE', @@ -727,6 +727,7 @@ const translation = { fileExtensionNotSupport: 'Extension de fichier non prise en charge', pasteFileLinkInvalid: 'Lien de fichier non valide', uploadFromComputerLimit: 'Le fichier de téléchargement ne peut pas dépasser {{size}}', + fileExtensionBlocked: 'Ce type de fichier est bloqué pour des raisons de sécurité', }, license: { expiring: 'Expirant dans un jour', diff --git a/web/i18n/fr-FR/login.ts b/web/i18n/fr-FR/login.ts index deae8e3ff4..3abb6fba2a 100644 --- a/web/i18n/fr-FR/login.ts +++ b/web/i18n/fr-FR/login.ts @@ -120,6 +120,7 @@ const translation = { verifyMail: 'Continuez avec le code de vérification', createAccount: 'Créez votre compte', }, + pageTitleForE: 'Hé, commençons !', } export default translation diff --git a/web/i18n/fr-FR/tools.ts b/web/i18n/fr-FR/tools.ts index c91952d6c5..9f296773f2 100644 --- a/web/i18n/fr-FR/tools.ts +++ b/web/i18n/fr-FR/tools.ts @@ -2,7 +2,6 @@ const translation = { title: 'Outils', createCustomTool: 'Créer un Outil Personnalisé', type: { - all: 'Tout', builtIn: 'Intégré', custom: 'Personnalisé', workflow: 'Flux de travail', @@ -20,7 +19,6 @@ const translation = { setupModalTitleDescription: 'Après avoir configuré les identifiants, tous les membres de l\'espace de travail peuvent utiliser cet outil lors de l\'orchestration des applications.', }, includeToolNum: '{{num}} outils inclus', - addTool: 'Ajouter un outil', createTool: { title: 'Créer un Outil Personnalisé', editAction: 'Configurer', @@ -143,9 +141,7 @@ const translation = { addToolModal: { type: 'type', added: 'supplémentaire', - add: 'ajouter', category: 'catégorie', - manageInTools: 'Gérer dans Outils', custom: { title: 'Aucun outil personnalisé disponible', tip: 'Créer un outil personnalisé', @@ -203,6 +199,12 @@ const translation = { headersTip: 'En-têtes HTTP supplémentaires à envoyer avec les requêtes au serveur MCP', addHeader: 'Ajouter un en-tête', maskedHeadersTip: 'Les valeurs d\'en-tête sont masquées pour des raisons de sécurité. Les modifications mettront à jour les valeurs réelles.', + clientSecretPlaceholder: 'Secret client', + configurations: 'Configurations', + clientID: 'ID client', + authentication: 'Authentification', + useDynamicClientRegistration: 'Utiliser l\'enregistrement dynamique des clients', + clientSecret: 'Secret client', }, delete: 'Supprimer le Serveur MCP', deleteConfirmTitle: 'Souhaitez-vous supprimer {mcp}?', diff --git a/web/i18n/fr-FR/workflow.ts b/web/i18n/fr-FR/workflow.ts index f6c1899cac..c6405e0851 100644 --- a/web/i18n/fr-FR/workflow.ts +++ b/web/i18n/fr-FR/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Publié', publish: 'Publier', update: 'Mettre à jour', - run: 'Exécuter', + run: 'Exécuter test', running: 'En cours d\'exécution', inRunMode: 'En mode exécution', inPreview: 'En aperçu', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Historique des exécutions', goBackToEdit: 'Retour à l\'éditeur', conversationLog: 'Journal de conversation', - features: 'Fonctionnalités', debugAndPreview: 'Aperçu', restart: 'Redémarrer', currentDraft: 'Brouillon actuel', @@ -91,10 +90,8 @@ const translation = { addParallelNode: 'Ajouter un nœud parallèle', parallel: 'PARALLÈLE', branch: 'BRANCHE', - featuresDocLink: 'Pour en savoir plus', ImageUploadLegacyTip: 'Vous pouvez désormais créer des variables de type de fichier dans le formulaire de démarrage. À l’avenir, nous ne prendrons plus en charge la fonctionnalité de téléchargement d’images.', fileUploadTip: 'Les fonctionnalités de téléchargement d’images ont été mises à niveau vers le téléchargement de fichiers.', - featuresDescription: 'Améliorer l’expérience utilisateur de l’application web', importWarning: 'Prudence', importWarningDetails: 'La différence de version DSL peut affecter certaines fonctionnalités', openInExplore: 'Ouvrir dans Explorer', @@ -109,12 +106,13 @@ const translation = { versionHistory: 'Historique des versions', exportImage: 'Exporter l\'image', exportJPEG: 'Exporter en JPEG', - needEndNode: 'Le nœud de fin doit être ajouté', + needOutputNode: 'Le nœud de sortie doit être ajouté', needAnswerNode: 'Le nœud de réponse doit être ajouté.', addBlock: 'Ajouter un nœud', tagBound: 'Nombre d\'applications utilisant cette étiquette', currentView: 'Vue actuelle', currentWorkflow: 'Flux de travail actuel', + moreActions: 'Plus d’actions', }, env: { envPanelTitle: 'Variables d\'Environnement', @@ -139,6 +137,19 @@ const translation = { export: 'Exporter les DSL avec des valeurs secrètes', }, }, + globalVar: { + title: 'Variables système', + description: 'Les variables système sont des variables globales que tout nœud peut référencer sans câblage lorsque le type correspond, comme l\'ID utilisateur final et l\'ID du workflow.', + fieldsDescription: { + conversationId: 'ID de conversation', + dialogCount: 'Nombre de conversations', + userId: 'ID utilisateur', + triggerTimestamp: 'Horodatage du lancement de l\'application', + appId: 'ID de l\'application', + workflowId: 'ID du workflow', + workflowRunId: 'ID d\'exécution du workflow', + }, + }, chatVariable: { panelTitle: 'Variables de Conversation', panelDescription: 'Les Variables de Conversation sont utilisées pour stocker des informations interactives dont le LLM a besoin de se souvenir, y compris l\'historique des conversations, les fichiers téléchargés et les préférences de l\'utilisateur. Elles sont en lecture-écriture.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Début', - 'end': 'Fin', + 'end': 'Sortie', 'answer': 'Réponse', 'llm': 'LLM', 'knowledge-retrieval': 'Récupération de connaissances', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Définir les paramètres initiaux pour lancer un flux de travail', - 'end': 'Définir la fin et le type de résultat d\'un flux de travail', + 'end': 'Définir la sortie et le type de résultat d\'un flux de travail', 'answer': 'Définir le contenu de la réponse d\'une conversation', 'llm': 'Inviter de grands modèles de langage pour répondre aux questions ou traiter le langage naturel', 'knowledge-retrieval': 'Permet de consulter le contenu textuel lié aux questions des utilisateurs à partir de la base de connaissances', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Champ de saisie de l\'utilisateur', - helpLink: 'Lien d\'aide', + helpLink: 'Aide', about: 'À propos', createdBy: 'Créé par', nextStep: 'Étape suivante', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Tous les problèmes ont été résolus', change: 'Modifier', optional: '(facultatif)', - moveToThisNode: 'Déplacer vers ce nœud', organizeBlocks: 'Organiser les nœuds', addNextStep: 'Ajoutez la prochaine étape dans ce flux de travail', selectNextStep: 'Sélectionner la prochaine étape', changeBlock: 'Changer de nœud', maximize: 'Maximiser le Canvas', minimize: 'Sortir du mode plein écran', + scrollToSelectedNode: 'Faites défiler jusqu’au nœud sélectionné', optional_and_hidden: '(optionnel et caché)', }, nodes: { diff --git a/web/i18n/hi-IN/app-debug.ts b/web/i18n/hi-IN/app-debug.ts index c1c37ac63b..4d2b006856 100644 --- a/web/i18n/hi-IN/app-debug.ts +++ b/web/i18n/hi-IN/app-debug.ts @@ -279,7 +279,6 @@ const translation = { variableTable: { key: 'वेरिएबल कुंजी', name: 'उपयोगकर्ता इनपुट फ़ील्ड नाम', - optional: 'वैकल्पिक', type: 'इनपुट प्रकार', action: 'क्रियाएँ', typeString: 'स्ट्रिंग', diff --git a/web/i18n/hi-IN/app-log.ts b/web/i18n/hi-IN/app-log.ts index 02e062df2e..f5241b23f1 100644 --- a/web/i18n/hi-IN/app-log.ts +++ b/web/i18n/hi-IN/app-log.ts @@ -67,6 +67,8 @@ const translation = { quarterToDate: 'तिमाही तक तिथि', yearToDate: 'वर्ष तक तिथि', allTime: 'सभी समय', + last30days: 'पिछले 30 दिन', + custom: 'कस्टम', }, annotation: { all: 'सभी', diff --git a/web/i18n/hi-IN/billing.ts b/web/i18n/hi-IN/billing.ts index fbc6dffc7c..7164a13d6f 100644 --- a/web/i18n/hi-IN/billing.ts +++ b/web/i18n/hi-IN/billing.ts @@ -96,7 +96,7 @@ const translation = { freeTrialTip: '200 ओपनएआई कॉल्स का मुफ्त परीक्षण।', documents: '{{count,number}} ज्ञान दस्तावेज़', freeTrialTipSuffix: 'कोई क्रेडिट कार्ड की आवश्यकता नहीं है', - apiRateLimitUnit: '{{count,number}}/दिन', + apiRateLimitUnit: '{{count,number}}/माह', teamWorkspace: '{{count,number}} टीम कार्यक्षेत्र', apiRateLimitTooltip: 'Dify API के माध्यम से की गई सभी अनुरोधों पर API दर सीमा लागू होती है, जिसमें टेक्स्ट जनरेशन, चैट वार्तालाप, कार्यप्रवाह निष्पादन और दस्तावेज़ प्रसंस्करण शामिल हैं।', teamMember_one: '{{count,number}} टीम सदस्य', diff --git a/web/i18n/hi-IN/common.ts b/web/i18n/hi-IN/common.ts index ae6ce65aea..e775946cb7 100644 --- a/web/i18n/hi-IN/common.ts +++ b/web/i18n/hi-IN/common.ts @@ -170,7 +170,6 @@ const translation = { workspace: 'वर्कस्पेस', createWorkspace: 'वर्कस्पेस बनाएं', helpCenter: 'सहायता', - communityFeedback: 'प्रतिक्रिया', roadmap: 'रोडमैप', community: 'समुदाय', about: 'के बारे में', @@ -179,6 +178,7 @@ const translation = { github: 'गिटहब', support: 'समर्थन', contactUs: 'संपर्क करें', + forum: 'फोरम', }, settings: { accountGroup: 'खाता', @@ -748,6 +748,7 @@ const translation = { pasteFileLink: 'फ़ाइल लिंक पेस्ट करें', fileExtensionNotSupport: 'फ़ाइल एक्सटेंशन समर्थित नहीं है', uploadFromComputer: 'स्थानीय अपलोड', + fileExtensionBlocked: 'सुरक्षा कारणों से इस फ़ाइल प्रकार को अवरुद्ध कर दिया गया है', }, license: { expiring: 'एक दिन में समाप्त हो रहा है', diff --git a/web/i18n/hi-IN/login.ts b/web/i18n/hi-IN/login.ts index fee51208c7..27b7df9849 100644 --- a/web/i18n/hi-IN/login.ts +++ b/web/i18n/hi-IN/login.ts @@ -125,6 +125,7 @@ const translation = { welcome: '👋 स्वागत है! कृपया शुरू करने के लिए विवरण भरें।', haveAccount: 'क्या आपका पहले से एक खाता है?', }, + pageTitleForE: 'अरे, चलो शुरू करें!', } export default translation diff --git a/web/i18n/hi-IN/tools.ts b/web/i18n/hi-IN/tools.ts index 7279d3bcbe..c606f5f0b3 100644 --- a/web/i18n/hi-IN/tools.ts +++ b/web/i18n/hi-IN/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'कस्टम उपकरण बनाएं', customToolTip: 'Dify कस्टम उपकरणों के बारे में और जानें', type: { - all: 'सभी', builtIn: 'निर्मित', custom: 'कस्टम', workflow: 'कार्यप्रवाह', @@ -22,13 +21,10 @@ const translation = { 'प्रमाणिकरण कॉन्फ़िगर करने के बाद, कार्यस्थान के सभी सदस्य इस उपकरण का उपयोग कर सकेंगे।', }, includeToolNum: '{{num}} उपकरण शामिल हैं', - addTool: 'उपकरण जोड़ें', addToolModal: { type: 'प्रकार', category: 'श्रेणी', - add: 'जोड़ें', added: 'जोड़ा गया', - manageInTools: 'उपकरणों में प्रबंधित करें', custom: { title: 'कोई कस्टम टूल उपलब्ध नहीं है', tip: 'एक कस्टम टूल बनाएं', @@ -208,6 +204,12 @@ const translation = { noHeaders: 'कोई कस्टम हेडर कॉन्फ़िगर नहीं किए गए हैं', maskedHeadersTip: 'सुरक्षा के लिए हेडर मानों को छिपाया गया है। परिवर्तन वास्तविक मानों को अपडेट करेगा।', headersTip: 'MCP सर्वर अनुरोधों के साथ भेजने के लिए अतिरिक्त HTTP हेडर्स', + clientSecretPlaceholder: 'क्लाइंट सीक्रेट', + clientSecret: 'क्लाइंट सीक्रेट', + clientID: 'क्लाइंट आईडी', + configurations: 'संरचनाएँ', + authentication: 'प्रमाणीकरण', + useDynamicClientRegistration: 'डायनामिक क्लाइंट पंजीकरण का उपयोग करें', }, delete: 'MCP सर्वर हटाएँ', deleteConfirmTitle: '{mcp} हटाना चाहते हैं?', diff --git a/web/i18n/hi-IN/workflow.ts b/web/i18n/hi-IN/workflow.ts index 224f3acaeb..f739f64cf0 100644 --- a/web/i18n/hi-IN/workflow.ts +++ b/web/i18n/hi-IN/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'प्रकाशित', publish: 'प्रकाशित करें', update: 'अपडेट करें', - run: 'चलाएं', + run: 'परीक्षण चलाएं', running: 'चल रहा है', inRunMode: 'रन मोड में', inPreview: 'पूर्वावलोकन में', @@ -18,7 +18,6 @@ const translation = { runHistory: 'रन इतिहास', goBackToEdit: 'संपादक पर वापस जाएं', conversationLog: 'वार्तालाप लॉग', - features: 'विशेषताएं', debugAndPreview: 'पूर्वावलोकन', restart: 'पुनः आरंभ करें', currentDraft: 'वर्तमान ड्राफ्ट', @@ -94,8 +93,6 @@ const translation = { addParallelNode: 'समानांतर नोड जोड़ें', parallel: 'समानांतर', branch: 'शाखा', - featuresDocLink: 'और जानो', - featuresDescription: 'वेब ऐप उपयोगकर्ता अनुभव को बेहतर बनाएं', fileUploadTip: 'छवि अपलोड सुविधाओं को फ़ाइल अपलोड में अपग्रेड किया गया है।', ImageUploadLegacyTip: 'अब आप प्रारंभ प्रपत्र में फ़ाइल प्रकार चर बना सकते हैं। हम अब भविष्य में छवि अपलोड सुविधा का समर्थन नहीं करेंगे।', importWarning: 'सावधानी', @@ -114,10 +111,11 @@ const translation = { versionHistory: 'संस्करण इतिहास', needAnswerNode: 'उत्तर नोड जोड़ा जाना चाहिए', addBlock: 'नोड जोड़ें', - needEndNode: 'अंत नोड जोड़ा जाना चाहिए', + needOutputNode: 'आउटपुट नोड जोड़ा जाना चाहिए', tagBound: 'इस टैग का उपयोग करने वाले ऐप्स की संख्या', currentView: 'वर्तमान दृश्य', currentWorkflow: 'वर्तमान कार्यप्रवाह', + moreActions: 'अधिक क्रियाएँ', }, env: { envPanelTitle: 'पर्यावरण चर', @@ -142,6 +140,19 @@ const translation = { export: 'गुप्त मानों के साथ DSL निर्यात करें', }, }, + globalVar: { + title: 'सिस्टम वेरिएबल्स', + description: 'सिस्टम वेरिएबल्स वैश्विक वेरिएबल्स हैं जिन्हें सही प्रकार होने पर किसी भी नोड द्वारा बिना वायरिंग के संदर्भित किया जा सकता है, जैसे एंड-यूज़र ID और वर्कफ़्लो ID.', + fieldsDescription: { + conversationId: 'संवाद ID', + dialogCount: 'संवाद गणना', + userId: 'उपयोगकर्ता ID', + triggerTimestamp: 'एप्लिकेशन शुरू होने का टाइमस्टैम्प', + appId: 'एप्लिकेशन ID', + workflowId: 'वर्कफ़्लो ID', + workflowRunId: 'वर्कफ़्लो रन ID', + }, + }, chatVariable: { panelTitle: 'वार्तालाप चर', panelDescription: 'वार्तालाप चर का उपयोग इंटरैक्टिव जानकारी संग्रहित करने के लिए किया जाता है जिसे LLM को याद रखने की आवश्यकता होती है, जिसमें वार्तालाप इतिहास, अपलोड की गई फाइलें, उपयोगकर्ता प्राथमिकताएं शामिल हैं। वे पठनीय और लेखनीय हैं।', @@ -245,7 +256,7 @@ const translation = { }, blocks: { 'start': 'प्रारंभ', - 'end': 'समाप्त', + 'end': 'आउटपुट', 'answer': 'उत्तर', 'llm': 'एलएलएम', 'knowledge-retrieval': 'ज्ञान पुनर्प्राप्ति', @@ -271,7 +282,7 @@ const translation = { }, blocksAbout: { 'start': 'वर्कफ़्लो लॉन्च करने के लिए प्रारंभिक पैरामीटर को परिभाषित करें', - 'end': 'वर्कफ़्लो का अंत और परिणाम प्रकार परिभाषित करें', + 'end': 'वर्कफ़्लो का आउटपुट और परिणाम प्रकार परिभाषित करें', 'answer': 'चैट संवाद के उत्तर सामग्री को परिभाषित करें', 'llm': 'प्रश्नों के उत्तर देने या प्राकृतिक भाषा को संसाधित करने के लिए बड़े भाषा मॉडल को आमंत्रित करना', 'knowledge-retrieval': @@ -322,7 +333,7 @@ const translation = { }, panel: { userInputField: 'उपयोगकर्ता इनपुट फ़ील्ड', - helpLink: 'सहायता लिंक', + helpLink: 'सहायता', about: 'के बारे में', createdBy: 'द्वारा बनाया गया ', nextStep: 'अगला कदम', @@ -333,13 +344,13 @@ const translation = { checklistResolved: 'सभी समस्याएं हल हो गई हैं', change: 'बदलें', optional: '(वैकल्पिक)', - moveToThisNode: 'इस नोड पर जाएं', changeBlock: 'नोड बदलें', addNextStep: 'इस कार्यप्रवाह में अगला कदम जोड़ें', selectNextStep: 'अगला कदम चुनें', organizeBlocks: 'नोड्स का आयोजन करें', minimize: 'पूर्ण स्क्रीन से बाहर निकलें', maximize: 'कैनवास का अधिकतम लाभ उठाएँ', + scrollToSelectedNode: 'चुने गए नोड पर स्क्रॉल करें', optional_and_hidden: '(वैकल्पिक और छिपा हुआ)', }, nodes: { diff --git a/web/i18n/id-ID/app-debug.ts b/web/i18n/id-ID/app-debug.ts index ae463eba11..8838fd13a9 100644 --- a/web/i18n/id-ID/app-debug.ts +++ b/web/i18n/id-ID/app-debug.ts @@ -325,7 +325,6 @@ const translation = { variableTable: { action: 'Tindakan', typeString: 'String', - optional: 'Fakultatif', typeSelect: 'Pilih', type: 'Jenis Masukan', key: 'Kunci Variabel', diff --git a/web/i18n/id-ID/app-log.ts b/web/i18n/id-ID/app-log.ts index 8192e1f40d..9cfe4ab166 100644 --- a/web/i18n/id-ID/app-log.ts +++ b/web/i18n/id-ID/app-log.ts @@ -60,6 +60,8 @@ const translation = { yearToDate: 'Tahun hingga saat ini', allTime: 'Sepanjang masa', last12months: '12 bulan terakhir', + custom: 'Kustom', + last30days: '30 Hari Terakhir', }, annotation: { all: 'Semua', diff --git a/web/i18n/id-ID/common.ts b/web/i18n/id-ID/common.ts index d4180ea218..732b04bb92 100644 --- a/web/i18n/id-ID/common.ts +++ b/web/i18n/id-ID/common.ts @@ -163,7 +163,6 @@ const translation = { helpCenter: 'Docs', compliance: 'Kepatuhan', community: 'Masyarakat', - communityFeedback: 'Umpan balik', roadmap: 'Peta jalan', logout: 'Keluar', settings: 'Pengaturan', @@ -173,6 +172,7 @@ const translation = { workspace: 'Workspace', createWorkspace: 'Membuat Ruang Kerja', contactUs: 'Hubungi Kami', + forum: 'Forum', }, compliance: { soc2Type2: 'Laporan SOC 2 Tipe II', @@ -701,6 +701,7 @@ const translation = { pasteFileLinkInvalid: 'Tautan file tidak valid', pasteFileLinkInputPlaceholder: 'Masukkan URL...', uploadFromComputerReadError: 'Pembacaan file gagal, silakan coba lagi.', + fileExtensionBlocked: 'Tipe file ini diblokir karena alasan keamanan', }, tag: { noTag: 'Tidak ada tag', diff --git a/web/i18n/id-ID/login.ts b/web/i18n/id-ID/login.ts index 41c7e04ec4..1590aa81a2 100644 --- a/web/i18n/id-ID/login.ts +++ b/web/i18n/id-ID/login.ts @@ -120,6 +120,7 @@ const translation = { noAccount: 'Tidak punya akun?', welcome: '👋 Selamat datang! Silakan isi detail untuk memulai.', }, + pageTitleForE: 'Hei, ayo kita mulai!', } export default translation diff --git a/web/i18n/id-ID/tools.ts b/web/i18n/id-ID/tools.ts index e3817e0111..d9866dfb58 100644 --- a/web/i18n/id-ID/tools.ts +++ b/web/i18n/id-ID/tools.ts @@ -1,6 +1,5 @@ const translation = { type: { - all: 'Semua', workflow: 'Alur Kerja', builtIn: 'Perkakas', custom: 'Adat', @@ -35,8 +34,6 @@ const translation = { category: 'golongan', type: 'jenis', added: 'Ditambahkan', - add: 'tambah', - manageInTools: 'Kelola di Alat', }, createTool: { exampleOptions: { @@ -185,6 +182,12 @@ const translation = { headerValuePlaceholder: 'Bearer 123', noHeaders: 'Tidak ada header kustom yang dikonfigurasi', maskedHeadersTip: 'Nilai header disembunyikan untuk keamanan. Perubahan akan memperbarui nilai yang sebenarnya.', + clientSecretPlaceholder: 'Rahasia Klien', + authentication: 'Otentikasi', + useDynamicClientRegistration: 'Gunakan Pendaftaran Klien Dinamis', + configurations: 'Konfigurasi', + clientSecret: 'Rahasia Klien', + clientID: 'ID Klien', }, operation: { edit: 'Mengedit', @@ -234,7 +237,6 @@ const translation = { title: 'Perkakas', createCustomTool: 'Buat Alat Kustom', customToolTip: 'Pelajari alat kustom Dify lebih lanjut', - addTool: 'Tambahkan Alat', author: 'Oleh', copyToolName: 'Salin Nama', howToGet: 'Cara mendapatkan', diff --git a/web/i18n/id-ID/workflow.ts b/web/i18n/id-ID/workflow.ts index 4ef6b2b832..506b17d925 100644 --- a/web/i18n/id-ID/workflow.ts +++ b/web/i18n/id-ID/workflow.ts @@ -90,7 +90,7 @@ const translation = { exportJPEG: 'Ekspor sebagai JPEG', addBlock: 'Tambahkan Node', processData: 'Proses Data', - needEndNode: 'Node Akhir harus ditambahkan', + needOutputNode: 'Node Output harus ditambahkan', manageInTools: 'Kelola di Alat', pointerMode: 'Mode Penunjuk', accessAPIReference: 'Referensi API Akses', @@ -137,6 +137,19 @@ const translation = { envPanelTitle: 'Variabel Lingkungan', envDescription: 'Variabel lingkungan dapat digunakan untuk menyimpan informasi pribadi dan kredensial. Mereka hanya baca dan dapat dipisahkan dari file DSL selama ekspor.', }, + globalVar: { + title: 'Variabel Sistem', + description: 'Variabel sistem adalah variabel global yang dapat dirujuk oleh node apa pun tanpa koneksi jika tipenya sesuai, seperti ID pengguna akhir dan ID alur kerja.', + fieldsDescription: { + conversationId: 'ID percakapan', + dialogCount: 'Jumlah percakapan', + userId: 'ID pengguna', + triggerTimestamp: 'Cap waktu saat aplikasi mulai berjalan', + appId: 'ID aplikasi', + workflowId: 'ID alur kerja', + workflowRunId: 'ID eksekusi alur kerja', + }, + }, chatVariable: { modal: { valuePlaceholder: 'Nilai default, biarkan kosong untuk tidak diatur', @@ -249,7 +262,7 @@ const translation = { 'answer': 'Jawaban', 'parameter-extractor': 'Ekstraktor Parameter', 'document-extractor': 'Ekstraktor Dokumen', - 'end': 'Ujung', + 'end': 'Keluaran', 'if-else': 'JIKA/LAIN', 'loop-start': 'Mulai Loop', 'variable-aggregator': 'Agregator Variabel', @@ -275,7 +288,7 @@ const translation = { 'variable-assigner': 'Agregatkan variabel multi-cabang menjadi satu variabel untuk konfigurasi terpadu simpul hilir.', 'loop': 'Jalankan perulangan logika hingga kondisi penghentian terpenuhi atau jumlah perulangan maksimum tercapai.', 'variable-aggregator': 'Agregatkan variabel multi-cabang menjadi satu variabel untuk konfigurasi terpadu simpul hilir.', - 'end': 'Menentukan jenis akhir dan hasil alur kerja', + 'end': 'Menentukan output dan jenis hasil alur kerja', 'list-operator': 'Digunakan untuk memfilter atau mengurutkan konten array.', 'datasource': 'Sumber Data Tentang', 'knowledge-index': 'Basis Pengetahuan Tentang', @@ -321,7 +334,7 @@ const translation = { userInputField: 'Bidang Input Pengguna', checklistResolved: 'Semua masalah terselesaikan', createdBy: 'Dibuat oleh', - helpLink: 'Tautan Bantuan', + helpLink: 'Docs', changeBlock: 'Ubah Node', runThisStep: 'Jalankan langkah ini', maximize: 'Maksimalkan Kanvas', diff --git a/web/i18n/it-IT/app-debug.ts b/web/i18n/it-IT/app-debug.ts index 94ff6182ca..02680a8bae 100644 --- a/web/i18n/it-IT/app-debug.ts +++ b/web/i18n/it-IT/app-debug.ts @@ -281,7 +281,6 @@ const translation = { variableTable: { key: 'Chiave Variabile', name: 'Nome Campo Input Utente', - optional: 'Opzionale', type: 'Tipo di Input', action: 'Azioni', typeString: 'Stringa', diff --git a/web/i18n/it-IT/app-log.ts b/web/i18n/it-IT/app-log.ts index 98cb6afd84..4862898a21 100644 --- a/web/i18n/it-IT/app-log.ts +++ b/web/i18n/it-IT/app-log.ts @@ -69,6 +69,8 @@ const translation = { quarterToDate: 'Trimestre corrente', yearToDate: 'Anno corrente', allTime: 'Tutto il tempo', + custom: 'Personalizzato', + last30days: 'Ultimi 30 giorni', }, annotation: { all: 'Tutti', diff --git a/web/i18n/it-IT/billing.ts b/web/i18n/it-IT/billing.ts index ef6b1943e3..fc5d67520b 100644 --- a/web/i18n/it-IT/billing.ts +++ b/web/i18n/it-IT/billing.ts @@ -88,7 +88,7 @@ const translation = { freeTrialTipPrefix: 'Iscriviti e ricevi un', teamMember_one: '{{count,number}} membro del team', documents: '{{count,number}} Documenti di Conoscenza', - apiRateLimitUnit: '{{count,number}}/giorno', + apiRateLimitUnit: '{{count,number}}/mese', documentsRequestQuota: '{{count,number}}/min Limite di richiesta di conoscenza', teamMember_other: '{{count,number}} membri del team', freeTrialTip: 'prova gratuita di 200 chiamate OpenAI.', diff --git a/web/i18n/it-IT/common.ts b/web/i18n/it-IT/common.ts index 306f24829e..4f14727b10 100644 --- a/web/i18n/it-IT/common.ts +++ b/web/i18n/it-IT/common.ts @@ -170,7 +170,6 @@ const translation = { workspace: 'Workspace', createWorkspace: 'Crea Workspace', helpCenter: 'Aiuto', - communityFeedback: 'Feedback', roadmap: 'Tabella di marcia', community: 'Comunità', about: 'Informazioni', @@ -179,6 +178,7 @@ const translation = { compliance: 'Conformità', github: 'GitHub', contactUs: 'Contattaci', + forum: 'Forum', }, settings: { accountGroup: 'ACCOUNT', @@ -756,6 +756,7 @@ const translation = { uploadFromComputerUploadError: 'Caricamento del file non riuscito, carica di nuovo.', pasteFileLink: 'Incolla il collegamento del file', uploadFromComputerReadError: 'Lettura del file non riuscita, riprovare.', + fileExtensionBlocked: 'Questo tipo di file è bloccato per motivi di sicurezza', }, license: { expiring_plural: 'Scadenza tra {{count}} giorni', diff --git a/web/i18n/it-IT/login.ts b/web/i18n/it-IT/login.ts index 5d6b040daf..e19baca6a3 100644 --- a/web/i18n/it-IT/login.ts +++ b/web/i18n/it-IT/login.ts @@ -130,6 +130,7 @@ const translation = { signUp: 'Iscriviti', welcome: '👋 Benvenuto! Per favore compila i dettagli per iniziare.', }, + pageTitleForE: 'Ehi, cominciamo!', } export default translation diff --git a/web/i18n/it-IT/tools.ts b/web/i18n/it-IT/tools.ts index 5e54b8f837..a81898eff2 100644 --- a/web/i18n/it-IT/tools.ts +++ b/web/i18n/it-IT/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'Crea Strumento Personalizzato', customToolTip: 'Scopri di più sugli strumenti personalizzati di Dify', type: { - all: 'Tutti', builtIn: 'Integrato', custom: 'Personalizzato', workflow: 'Flusso di lavoro', @@ -22,13 +21,10 @@ const translation = { 'Dopo aver configurato le credenziali, tutti i membri all\'interno del workspace possono utilizzare questo strumento durante l\'orchestrazione delle applicazioni.', }, includeToolNum: '{{num}} strumenti inclusi', - addTool: 'Aggiungi Strumento', addToolModal: { type: 'tipo', category: 'categoria', - add: 'aggiungi', added: 'aggiunto', - manageInTools: 'Gestisci in Strumenti', custom: { title: 'Nessuno strumento personalizzato disponibile', tip: 'Crea uno strumento personalizzato', @@ -213,6 +209,12 @@ const translation = { headerValuePlaceholder: 'ad esempio, Token di accesso123', headersTip: 'Intestazioni HTTP aggiuntive da inviare con le richieste al server MCP', maskedHeadersTip: 'I valori dell\'intestazione sono mascherati per motivi di sicurezza. Le modifiche aggiorneranno i valori effettivi.', + clientID: 'ID cliente', + clientSecret: 'Segreto del Cliente', + useDynamicClientRegistration: 'Usa la Registrazione Dinamica del Client', + clientSecretPlaceholder: 'Segreto del Cliente', + authentication: 'Autenticazione', + configurations: 'Configurazioni', }, delete: 'Rimuovi Server MCP', deleteConfirmTitle: 'Vuoi rimuovere {mcp}?', diff --git a/web/i18n/it-IT/workflow.ts b/web/i18n/it-IT/workflow.ts index 314b8e0c52..b188bc3666 100644 --- a/web/i18n/it-IT/workflow.ts +++ b/web/i18n/it-IT/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Pubblicato', publish: 'Pubblica', update: 'Aggiorna', - run: 'Esegui', + run: 'Esegui test', running: 'In esecuzione', inRunMode: 'In modalità di esecuzione', inPreview: 'In anteprima', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Cronologia esecuzioni', goBackToEdit: 'Torna all\'editor', conversationLog: 'Registro conversazioni', - features: 'Caratteristiche', debugAndPreview: 'Anteprima', restart: 'Riavvia', currentDraft: 'Bozza corrente', @@ -95,8 +94,6 @@ const translation = { addParallelNode: 'Aggiungi nodo parallelo', parallel: 'PARALLELO', branch: 'RAMO', - featuresDocLink: 'Ulteriori informazioni', - featuresDescription: 'Migliora l\'esperienza utente dell\'app Web', fileUploadTip: 'Le funzioni di caricamento delle immagini sono state aggiornate al caricamento dei file.', ImageUploadLegacyTip: 'Ora è possibile creare variabili di tipo file nel modulo iniziale. In futuro non supporteremo più la funzione di caricamento delle immagini.', importWarning: 'Cautela', @@ -113,12 +110,13 @@ const translation = { exportImage: 'Esporta immagine', exportJPEG: 'Esporta come JPEG', exportPNG: 'Esporta come PNG', - needEndNode: 'Deve essere aggiunto il nodo finale', + needOutputNode: 'Deve essere aggiunto il nodo di uscita', addBlock: 'Aggiungi nodo', needAnswerNode: 'Deve essere aggiunto il nodo di risposta', tagBound: 'Numero di app che utilizzano questo tag', currentWorkflow: 'Flusso di lavoro corrente', currentView: 'Vista corrente', + moreActions: 'Altre azioni', }, env: { envPanelTitle: 'Variabili d\'Ambiente', @@ -143,6 +141,19 @@ const translation = { export: 'Esporta DSL con valori segreti', }, }, + globalVar: { + title: 'Variabili di sistema', + description: 'Le variabili di sistema sono variabili globali che possono essere richiamate da qualsiasi nodo senza collegamenti quando il tipo è corretto, come l\'ID dell\'utente finale e l\'ID del workflow.', + fieldsDescription: { + conversationId: 'ID conversazione', + dialogCount: 'Conteggio conversazioni', + userId: 'ID utente', + triggerTimestamp: 'Timestamp di avvio dell\'applicazione', + appId: 'ID applicazione', + workflowId: 'ID workflow', + workflowRunId: 'ID esecuzione workflow', + }, + }, chatVariable: { panelTitle: 'Variabili di Conversazione', panelDescription: 'Le Variabili di Conversazione sono utilizzate per memorizzare informazioni interattive che il LLM deve ricordare, inclusi la cronologia delle conversazioni, i file caricati e le preferenze dell\'utente. Sono in lettura e scrittura.', @@ -247,7 +258,7 @@ const translation = { }, blocks: { 'start': 'Inizio', - 'end': 'Fine', + 'end': 'Uscita', 'answer': 'Risposta', 'llm': 'LLM', 'knowledge-retrieval': 'Recupero Conoscenza', @@ -273,7 +284,7 @@ const translation = { }, blocksAbout: { 'start': 'Definisci i parametri iniziali per l\'avvio di un flusso di lavoro', - 'end': 'Definisci la fine e il tipo di risultato di un flusso di lavoro', + 'end': 'Definisci l\'uscita e il tipo di risultato di un flusso di lavoro', 'answer': 'Definisci il contenuto della risposta di una conversazione chat', 'llm': 'Invoca modelli di linguaggio di grandi dimensioni per rispondere a domande o elaborare il linguaggio naturale', 'knowledge-retrieval': @@ -325,7 +336,7 @@ const translation = { }, panel: { userInputField: 'Campo di Input Utente', - helpLink: 'Link di Aiuto', + helpLink: 'Aiuto', about: 'Informazioni', createdBy: 'Creato da ', nextStep: 'Prossimo Passo', @@ -336,13 +347,13 @@ const translation = { checklistResolved: 'Tutti i problemi sono risolti', change: 'Cambia', optional: '(opzionale)', - moveToThisNode: 'Sposta a questo nodo', changeBlock: 'Cambia Nodo', selectNextStep: 'Seleziona il prossimo passo', organizeBlocks: 'Organizzare i nodi', addNextStep: 'Aggiungi il prossimo passo in questo flusso di lavoro', minimize: 'Esci dalla modalità schermo intero', maximize: 'Massimizza Canvas', + scrollToSelectedNode: 'Scorri fino al nodo selezionato', optional_and_hidden: '(opzionale e nascosto)', }, nodes: { diff --git a/web/i18n/ja-JP/app-debug.ts b/web/i18n/ja-JP/app-debug.ts index 4a6d8a81cb..f15119a5f5 100644 --- a/web/i18n/ja-JP/app-debug.ts +++ b/web/i18n/ja-JP/app-debug.ts @@ -340,7 +340,6 @@ const translation = { variableTable: { key: '変数キー', name: 'ユーザー入力フィールド名', - optional: 'オプション', type: '入力タイプ', action: 'アクション', typeString: '文字列', diff --git a/web/i18n/ja-JP/app-log.ts b/web/i18n/ja-JP/app-log.ts index 714481c8d1..aa23d8352d 100644 --- a/web/i18n/ja-JP/app-log.ts +++ b/web/i18n/ja-JP/app-log.ts @@ -20,6 +20,7 @@ const translation = { tokens: 'トークン', user: 'エンドユーザーまたはアカウント', version: 'バージョン', + triggered_from: 'トリガー方法', }, pagination: { previous: '前へ', @@ -59,6 +60,7 @@ const translation = { period: { today: '今日', last7days: '過去 7 日間', + last30days: '過去 30 日間', last4weeks: '過去 4 週間', last3months: '過去 3 ヶ月', last12months: '過去 12 ヶ月', @@ -66,6 +68,7 @@ const translation = { quarterToDate: '四半期初から今日まで', yearToDate: '年初から今日まで', allTime: 'すべての期間', + custom: 'カスタム', }, annotation: { all: 'すべて', @@ -95,6 +98,15 @@ const translation = { iteration: '反復', finalProcessing: '最終処理', }, + triggerBy: { + debugging: 'デバッグ', + appRun: 'ウェブアプリ', + webhook: 'Webhook', + schedule: 'スケジュール', + plugin: 'プラグイン', + ragPipelineRun: 'RAGパイプライン', + ragPipelineDebugging: 'RAGデバッグ', + }, } export default translation diff --git a/web/i18n/ja-JP/app-overview.ts b/web/i18n/ja-JP/app-overview.ts index d948bc3b28..ad1abb78fa 100644 --- a/web/i18n/ja-JP/app-overview.ts +++ b/web/i18n/ja-JP/app-overview.ts @@ -30,12 +30,17 @@ const translation = { overview: { title: '概要', appInfo: { + title: 'Web App', explanation: '使いやすい AI Web アプリ', accessibleAddress: '公開 URL', preview: 'プレビュー', regenerate: '再生成', regenerateNotice: '公開 URL を再生成しますか?', preUseReminder: '続行する前に Web アプリを有効にしてください。', + enableTooltip: { + description: 'この機能を有効にするには、キャンバスにユーザー入力ノードを追加してください。(下書きに既に存在する可能性があり、公開後に有効になります)', + learnMore: '詳細を見る', + }, settings: { entry: '設定', title: 'Web アプリの設定', @@ -113,7 +118,7 @@ const translation = { operation: 'ドキュメント', }, }, - launch: '発射', + launch: '公開', }, apiInfo: { title: 'バックエンドサービス API', @@ -121,6 +126,14 @@ const translation = { accessibleAddress: 'サービス API エンドポイント', doc: 'API リファレンス', }, + triggerInfo: { + title: 'トリガー', + explanation: 'ワークフロートリガー管理', + triggersAdded: '{{count}} 個のトリガーが追加されました', + noTriggerAdded: 'トリガーが追加されていません', + triggerStatusDescription: 'トリガーノードの状態がここに表示されます。(下書きに既に存在する可能性があり、公開後に有効になります)', + learnAboutTriggers: 'トリガーについて学ぶ', + }, status: { running: '稼働中', disable: '無効', diff --git a/web/i18n/ja-JP/billing.ts b/web/i18n/ja-JP/billing.ts index 6dbff60d5a..b679ae571a 100644 --- a/web/i18n/ja-JP/billing.ts +++ b/web/i18n/ja-JP/billing.ts @@ -7,6 +7,8 @@ const translation = { documentsUploadQuota: 'ドキュメント・アップロード・クォータ', vectorSpace: 'ナレッジベースのデータストレージ', vectorSpaceTooltip: '高品質インデックスモードのドキュメントは、ナレッジベースのデータストレージのリソースを消費します。ナレッジベースのデータストレージの上限に達すると、新しいドキュメントはアップロードされません。', + triggerEvents: 'トリガーイベント', + perMonth: '月あたり', }, upgradeBtn: { plain: 'プランをアップグレード', @@ -60,7 +62,7 @@ const translation = { documentsRequestQuota: '{{count,number}}/分のナレッジ リクエストのレート制限', documentsRequestQuotaTooltip: 'ナレッジベース内でワークスペースが 1 分間に実行できる操作の総数を示します。これには、データセットの作成、削除、更新、ドキュメントのアップロード、修正、アーカイブ、およびナレッジベースクエリが含まれます。この指標は、ナレッジベースリクエストのパフォーマンスを評価するために使用されます。例えば、Sandbox ユーザーが 1 分間に 10 回連続でヒットテストを実行した場合、そのワークスペースは次の 1 分間、データセットの作成、削除、更新、ドキュメントのアップロードや修正などの操作を一時的に実行できなくなります。', apiRateLimit: 'API レート制限', - apiRateLimitUnit: '{{count,number}}/日', + apiRateLimitUnit: '{{count,number}}/月', unlimitedApiRate: '無制限の API コール', apiRateLimitTooltip: 'API レート制限は、テキスト生成、チャットボット、ワークフロー、ドキュメント処理など、Dify API 経由のすべてのリクエストに適用されます。', documentProcessingPriority: '文書処理', diff --git a/web/i18n/ja-JP/common.ts b/web/i18n/ja-JP/common.ts index 25c9edac75..a4f858290a 100644 --- a/web/i18n/ja-JP/common.ts +++ b/web/i18n/ja-JP/common.ts @@ -66,6 +66,7 @@ const translation = { more: 'もっと', selectAll: 'すべて選択', deSelectAll: 'すべて選択解除', + now: '今', config: 'コンフィグ', yes: 'はい', no: 'いいえ', @@ -170,16 +171,16 @@ const translation = { emailSupport: 'サポート', workspace: 'ワークスペース', createWorkspace: 'ワークスペースを作成', - helpCenter: 'ヘルプ', + helpCenter: 'ドキュメントを見る', support: 'サポート', compliance: 'コンプライアンス', - communityFeedback: 'フィードバック', roadmap: 'ロードマップ', community: 'コミュニティ', about: 'Dify について', logout: 'ログアウト', github: 'GitHub', contactUs: 'お問い合わせ', + forum: 'フォーラム', }, compliance: { soc2Type1: 'SOC 2 Type I 報告書', @@ -741,6 +742,7 @@ const translation = { uploadFromComputerReadError: 'ファイルの読み取りに失敗しました。もう一度やり直してください。', fileExtensionNotSupport: 'ファイル拡張子はサポートされていません', pasteFileLinkInvalid: '無効なファイルリンク', + fileExtensionBlocked: 'このファイルタイプは、セキュリティ上の理由でブロックされています', }, license: { expiring_plural: '有効期限 {{count}} 日', diff --git a/web/i18n/ja-JP/login.ts b/web/i18n/ja-JP/login.ts index d1e9a9e0e2..7069315c9d 100644 --- a/web/i18n/ja-JP/login.ts +++ b/web/i18n/ja-JP/login.ts @@ -1,5 +1,6 @@ const translation = { pageTitle: 'Dify にログイン', + pageTitleForE: 'はじめましょう!', welcome: '👋 ようこそ!まずはログインしてご利用ください。', email: 'メールアドレス', emailPlaceholder: 'メールアドレスを入力してください', diff --git a/web/i18n/ja-JP/pipeline.ts b/web/i18n/ja-JP/pipeline.ts index 64700acc09..9ec1b68273 100644 --- a/web/i18n/ja-JP/pipeline.ts +++ b/web/i18n/ja-JP/pipeline.ts @@ -33,7 +33,7 @@ const translation = { }, ragToolSuggestions: { title: 'RAGのための提案', - noRecommendationPluginsInstalled: '推奨プラグインがインストールされていません。<CustomLink>マーケットプレイス</CustomLink>で詳細をご確認ください', + noRecommendationPlugins: '推奨プラグインがありません。<CustomLink>マーケットプレイス</CustomLink>で詳細をご確認ください', }, } diff --git a/web/i18n/ja-JP/plugin-trigger.ts b/web/i18n/ja-JP/plugin-trigger.ts new file mode 100644 index 0000000000..a4f0a8c5df --- /dev/null +++ b/web/i18n/ja-JP/plugin-trigger.ts @@ -0,0 +1,216 @@ +const translation = { + subscription: { + title: 'サブスクリプション', + listNum: '{{num}} サブスクリプション', + empty: { + title: 'サブスクリプションがありません', + description: 'イベントの受信を開始するために最初のサブスクリプションを作成してください', + button: '新しいサブスクリプション', + }, + createButton: { + oauth: 'OAuth で新しいサブスクリプション', + apiKey: 'API キーで新しいサブスクリプション', + manual: 'URL を貼り付けて新しいサブスクリプションを作成', + }, + list: { + title: 'サブスクリプション', + addButton: '追加', + tip: 'サブスクリプション経由でイベントを受信', + item: { + enabled: '有効', + disabled: '無効', + credentialType: { + api_key: 'API キー', + oauth2: 'OAuth', + unauthorized: '手動', + }, + actions: { + delete: '削除', + deleteConfirm: { + title: 'サブスクリプションを削除', + content: '「{{name}}」を削除してもよろしいですか?', + contentWithApps: 'このサブスクリプションは {{count}} 個のアプリで使用されています。「{{name}}」を削除してもよろしいですか?', + confirm: '削除', + cancel: 'キャンセル', + confirmInputWarning: '確認するために正しい名前を入力してください。', + }, + }, + status: { + active: 'アクティブ', + inactive: '非アクティブ', + }, + usedByNum: '{{num}} ワークフローで使用中', + noUsed: 'ワークフローで使用されていません', + }, + }, + addType: { + title: 'サブスクリプションを追加', + description: 'トリガーサブスクリプションの作成方法を選択してください', + options: { + apiKey: { + title: 'API キー経由', + description: 'API 認証情報を使用してサブスクリプションを自動作成', + }, + oauth: { + title: 'OAuth 経由', + description: 'サードパーティプラットフォームで認証してサブスクリプションを作成', + custom: 'カスタム', + default: 'デフォルト', + clientSettings: 'OAuthクライアント設定', + clientTitle: 'OAuth クライアント', + }, + manual: { + title: '手動設定', + description: 'URL を貼り付けて新しいサブスクリプションを作成', + tip: 'サードパーティプラットフォームで URL を手動設定', + }, + apikey: { + title: 'APIキーで作成', + description: 'API資格情報を使用してサブスクリプションを自動的に作成する', + }, + }, + }, + subscriptionRemoved: 'サブスクリプションが解除されました', + createSuccess: 'サブスクリプションが正常に作成されました', + noSubscriptionSelected: 'サブスクリプションが選択されていません', + selectPlaceholder: 'サブスクリプションを選択', + createFailed: 'サブスクリプションの作成に失敗しました', + }, + modal: { + steps: { + verify: '検証', + configuration: '設定', + }, + common: { + cancel: 'キャンセル', + back: '戻る', + next: '次へ', + create: '作成', + verify: '検証', + authorize: '認証', + creating: '作成中...', + verifying: '検証中...', + authorizing: '認証中...', + }, + oauthRedirectInfo: 'このツールプロバイダーのシステムクライアントシークレットが見つからないため、手動設定が必要です。redirect_uri には以下を使用してください', + apiKey: { + title: 'API キーで作成', + verify: { + title: '認証情報を検証', + description: 'アクセスを検証するために API 認証情報を提供してください', + error: '認証情報の検証に失敗しました。API キーをご確認ください。', + success: '認証情報が正常に検証されました', + }, + configuration: { + title: 'サブスクリプションを設定', + description: 'サブスクリプションパラメータを設定', + }, + }, + oauth: { + title: 'OAuth で作成', + authorization: { + title: 'OAuth 認証', + description: 'Dify があなたのアカウントにアクセスすることを認証', + redirectUrl: 'リダイレクト URL', + redirectUrlHelp: 'OAuth アプリ設定でこの URL を使用', + authorizeButton: '{{provider}} で認証', + waitingAuth: '認証を待機中...', + authSuccess: '認証が成功しました', + authFailed: '認証に失敗しました', + waitingJump: '承認済み、ジャンプ待機中', + }, + configuration: { + title: 'サブスクリプションを設定', + description: '認証後にサブスクリプションパラメータを設定', + success: 'OAuth設定が成功しました', + failed: 'OAuthの設定に失敗しました', + }, + remove: { + success: 'OAuthの削除に成功しました', + failed: 'OAuthの削除に失敗しました', + }, + save: { + success: 'OAuth の設定が正常に保存されました', + }, + }, + manual: { + title: '手動設定', + description: 'Webhook サブスクリプションを手動で設定', + instruction: { + title: '設定手順', + step1: '1. 以下のコールバック URL をコピー', + step2: '2. サードパーティプラットフォームの Webhook 設定に移動', + step3: '3. コールバック URL を Webhook エンドポイントとして追加', + step4: '4. 受信したいイベントを設定', + step5: '5. イベントをトリガーして Webhook をテスト', + step6: '6. ここに戻って Webhook が動作していることを確認し、設定を完了', + }, + logs: { + title: 'リクエストログ', + description: '受信 Webhook リクエストを監視', + empty: 'まだリクエストを受信していません。Webhook 設定をテストしてください。', + status: { + success: '成功', + error: 'エラー', + }, + expandAll: 'すべて展開', + collapseAll: 'すべて折りたたむ', + timestamp: 'タイムスタンプ', + method: 'メソッド', + path: 'パス', + headers: 'ヘッダー', + body: 'ボディ', + response: 'レスポンス', + request: 'リクエスト', + }, + }, + form: { + subscriptionName: { + label: 'サブスクリプション名', + placeholder: 'サブスクリプション名を入力', + required: 'サブスクリプション名は必須です', + }, + callbackUrl: { + label: 'コールバック URL', + description: 'この URL で Webhook イベントを受信します', + copy: 'コピー', + copied: 'コピーしました!', + placeholder: '生成中...', + privateAddressWarning: 'このURLは内部アドレスのようで、Webhookリクエストが失敗する可能性があります。', + tooltip: 'トリガープロバイダーからのコールバックリクエストを受信できる、公開アクセス可能なエンドポイントを提供してください。', + }, + }, + errors: { + createFailed: 'サブスクリプションの作成に失敗しました', + verifyFailed: '認証情報の検証に失敗しました', + authFailed: '認証に失敗しました', + networkError: 'ネットワークエラーです。再試行してください', + }, + }, + events: { + title: '利用可能なイベント', + description: 'このトリガープラグインが購読できるイベント', + empty: '利用可能なイベントがありません', + event: 'イベント', + events: 'イベント', + actionNum: '{{num}} {{event}} が含まれています', + item: { + parameters: '{{count}} パラメータ', + noParameters: 'パラメータなし', + }, + output: '出力', + }, + provider: { + github: 'GitHub', + gitlab: 'GitLab', + notion: 'Notion', + webhook: 'Webhook', + }, + node: { + status: { + warning: '切断', + }, + }, +} + +export default translation diff --git a/web/i18n/ja-JP/tools.ts b/web/i18n/ja-JP/tools.ts index 2fed3768c0..8df59af218 100644 --- a/web/i18n/ja-JP/tools.ts +++ b/web/i18n/ja-JP/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'カスタムツールを作成する', customToolTip: 'Dify カスタムツールの詳細', type: { - all: 'すべて', builtIn: 'ツール', custom: 'カスタム', workflow: 'ワークフロー', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: '資格情報を構成した後、ワークスペース内のすべてのメンバーがアプリケーションのオーケストレーション時にこのツールを使用できます。', }, includeToolNum: '{{num}}個のツールが含まれています', - addTool: 'ツールを追加する', addToolModal: { type: 'タイプ', category: 'カテゴリー', - add: '追加', added: '追加済', - manageInTools: 'ツールリストに移動して管理する', custom: { title: 'カスタムツールはありません', tip: 'カスタムツールを作成する', @@ -203,6 +199,12 @@ const translation = { noHeaders: 'カスタムヘッダーは設定されていません', headersTip: 'MCPサーバーへのリクエストに送信する追加のHTTPヘッダー', maskedHeadersTip: 'ヘッダー値はセキュリティのためマスクされています。変更は実際の値を更新します。', + configurations: '設定', + authentication: '認証', + clientID: 'クライアントID', + useDynamicClientRegistration: '動的クライアント登録を使用する', + clientSecretPlaceholder: 'クライアントシークレット', + clientSecret: 'クライアントシークレット', }, delete: 'MCP サーバーを削除', deleteConfirmTitle: '{{mcp}} を削除しますか?', diff --git a/web/i18n/ja-JP/workflow.ts b/web/i18n/ja-JP/workflow.ts index 3320f5a89f..07241b8c4f 100644 --- a/web/i18n/ja-JP/workflow.ts +++ b/web/i18n/ja-JP/workflow.ts @@ -9,8 +9,10 @@ const translation = { publish: '公開する', update: '更新', publishUpdate: '更新を公開', - run: '実行', + run: 'テスト実行', running: '実行中', + chooseStartNodeToRun: '実行する開始ノードを選択', + runAllTriggers: 'すべてのトリガーを実行', inRunMode: '実行モード中', inPreview: 'プレビュー中', inPreviewMode: 'プレビューモード中', @@ -19,11 +21,8 @@ const translation = { runHistory: '実行履歴', goBackToEdit: '編集に戻る', conversationLog: '会話ログ', - features: '機能', - featuresDescription: 'Web アプリの操作性を向上させる機能', ImageUploadLegacyTip: '開始フォームでファイル型変数が作成可能になりました。画像アップロード機能は今後サポート終了となります。', fileUploadTip: '画像アップロード機能がファイルアップロードに拡張されました', - featuresDocLink: '詳細を見る', debugAndPreview: 'プレビュー', restart: '再起動', currentDraft: '現在の下書き', @@ -47,7 +46,8 @@ const translation = { needConnectTip: '接続されていないステップがあります', maxTreeDepth: '1 ブランチあたりの最大ノード数:{{depth}}', needAdd: '{{node}}ノードを追加する必要があります', - needEndNode: '終了ブロックを追加する必要があります', + needOutputNode: '出力ノードを追加する必要があります', + needStartNode: '少なくとも1つのスタートノードを追加する必要があります', needAnswerNode: '回答ブロックを追加する必要があります', workflowProcess: 'ワークフロー処理', notRunning: 'まだ実行されていません', @@ -83,6 +83,7 @@ const translation = { configure: '設定', manageInTools: 'ツールページで管理', workflowAsToolTip: 'ワークフロー更新後はツールの再設定が必要です', + workflowAsToolDisabledHint: '最新のワークフローを公開し、接続済みの User Input ノードを用意してからツールとして設定してください。', viewDetailInTracingPanel: '詳細を表示', syncingData: 'データ同期中。。。', importDSL: 'DSL をインポート', @@ -116,6 +117,7 @@ const translation = { loadMore: 'さらに読み込む', noHistory: '履歴がありません', tagBound: 'このタグを使用しているアプリの数', + moreActions: 'さらにアクション', }, env: { envPanelTitle: '環境変数', @@ -140,6 +142,19 @@ const translation = { export: 'シークレット値付きでエクスポート', }, }, + globalVar: { + title: 'システム変数', + description: 'システム変数は、タイプが適合していれば配線なしで任意のノードから参照できるグローバル変数です。エンドユーザーIDやワークフローIDなどが含まれます。', + fieldsDescription: { + conversationId: '会話ID', + dialogCount: '会話数', + userId: 'ユーザーID', + triggerTimestamp: 'アプリケーションの起動タイムスタンプ', + appId: 'アプリケーションID', + workflowId: 'ワークフローID', + workflowRunId: 'ワークフロー実行ID', + }, + }, sidebar: { exportWarning: '現在保存されているバージョンをエクスポート', exportWarningDesc: 'これは現在保存されているワークフローのバージョンをエクスポートします。エディターで未保存の変更がある場合は、まずワークフローキャンバスのエクスポートオプションを使用して保存してください。', @@ -213,6 +228,16 @@ const translation = { invalidVariable: '無効な変数です', noValidTool: '{{field}} に利用可能なツールがありません', toolParameterRequired: '{{field}}: パラメータ [{{param}}] は必須です', + startNodeRequired: '{{operation}}前に開始ノードを追加してください', + }, + error: { + startNodeRequired: '{{operation}}前に開始ノードを追加してください', + operations: { + connectingNodes: 'ノード接続', + addingNodes: 'ノード追加', + modifyingWorkflow: 'ワークフロー変更', + updatingWorkflow: 'ワークフロー更新', + }, }, singleRun: { testRun: 'テスト実行', @@ -229,7 +254,9 @@ const translation = { 'searchBlock': 'ブロック検索', 'blocks': 'ブロック', 'searchTool': 'ツール検索', + 'searchTrigger': 'トリガー検索...', 'tools': 'ツール', + 'allTriggers': 'すべてのトリガー', 'allTool': 'すべて', 'customTool': 'カスタム', 'workflowTool': 'ワークフロー', @@ -238,16 +265,23 @@ const translation = { 'transform': '変換', 'utilities': 'ツール', 'noResult': '該当なし', + 'noPluginsFound': 'プラグインが見つかりません', + 'requestToCommunity': 'コミュニティにリクエスト', 'plugin': 'プラグイン', 'agent': 'エージェント戦略', + 'noFeaturedPlugins': 'マーケットプレイスでさらにツールを見つける', + 'noFeaturedTriggers': 'マーケットプレイスでさらにトリガーを見つける', 'addAll': 'すべてを追加する', 'allAdded': 'すべて追加されました', 'searchDataSource': 'データソースを検索', 'sources': 'ソース', + 'start': '始める', + 'startDisabledTip': 'トリガーノードとユーザー入力ノードは互いに排他です。', }, blocks: { - 'start': '開始', - 'end': '終了', + 'start': 'ユーザー入力', + 'originalStartNode': '元の開始ノード', + 'end': '出力', 'answer': '回答', 'llm': 'LLM', 'knowledge-retrieval': '知識検索', @@ -270,10 +304,14 @@ const translation = { 'loop-end': 'ループ完了', 'knowledge-index': '知識ベース', 'datasource': 'データソース', + 'trigger-plugin': 'プラグイントリガー', + 'trigger-webhook': 'Webhook トリガー', + 'trigger-schedule': 'スケジュールトリガー', }, + customWebhook: 'カスタムWebhook', blocksAbout: { 'start': 'ワークフロー開始時の初期パラメータを定義します。', - 'end': 'ワークフローの終了条件と結果のタイプを定義します。', + 'end': 'ワークフローの出力と結果のタイプを定義します', 'answer': 'チャットダイアログの返答内容を定義します。', 'llm': '大規模言語モデルを呼び出して質問回答や自然言語処理を実行します。', 'knowledge-retrieval': 'ナレッジベースからユーザー質問に関連するテキストを検索します。', @@ -294,7 +332,11 @@ const translation = { 'agent': '大規模言語モデルを活用した質問応答や自然言語処理を実行します。', 'knowledge-index': '知識ベースについて', 'datasource': 'データソースについて', + 'trigger-schedule': 'スケジュールに基づいてワークフローを開始する時間ベースのトリガー', + 'trigger-webhook': 'Webhook トリガーは第三者システムからの HTTP プッシュを受信してワークフローを自動的に開始します。', + 'trigger-plugin': 'サードパーティ統合トリガー、外部プラットフォームのイベントによってワークフローを開始します', }, + difyTeam: 'Dify チーム', operator: { zoomIn: '拡大', zoomOut: '縮小', @@ -324,7 +366,7 @@ const translation = { panel: { userInputField: 'ユーザー入力欄', changeBlock: 'ノード変更', - helpLink: 'ヘルプリンク', + helpLink: 'ドキュメントを見る', about: '詳細', createdBy: '作成者', nextStep: '次のステップ', @@ -334,12 +376,14 @@ const translation = { checklist: 'チェックリスト', checklistTip: '公開前に全ての項目を確認してください', checklistResolved: '全てのチェックが完了しました', + goTo: '移動', + startNode: '開始ノード', organizeBlocks: 'ノード整理', change: '変更', optional: '(任意)', - moveToThisNode: 'このノードに移動する', maximize: 'キャンバスを最大化する', minimize: '全画面を終了する', + scrollToSelectedNode: '選択したノードまでスクロール', optional_and_hidden: '(オプションおよび非表示)', }, nodes: { @@ -964,6 +1008,137 @@ const translation = { embeddingModelIsInvalid: '埋め込みモデルが無効です', rerankingModelIsInvalid: 'リランキングモデルは無効です', }, + triggerSchedule: { + frequency: { + label: '頻度', + monthly: '毎月', + weekly: '毎週', + daily: '毎日', + hourly: '毎時', + }, + frequencyLabel: '頻度', + days: '日', + title: 'スケジュール', + minutes: '分', + time: '時刻', + useCronExpression: 'Cron 式を使用', + nextExecutionTimes: '次の5回の実行時刻', + nextExecution: '次回実行', + notConfigured: '未設定', + startTime: '開始時刻', + hours: '時間', + onMinute: '分', + executeNow: '今すぐ実行', + weekdays: '曜日', + selectDateTime: '日時を選択', + cronExpression: 'Cron 式', + selectFrequency: '頻度を選択', + lastDay: '月末', + nextExecutionTime: '次回実行時刻', + lastDayTooltip: 'すべての月に31日があるわけではありません。「月末」オプションを使用して各月の最終日を選択してください。', + useVisualPicker: 'ビジュアル設定を使用', + nodeTitle: 'スケジュールトリガー', + mode: 'モード', + timezone: 'タイムゾーン', + visualConfig: 'ビジュアル設定', + monthlyDay: '月の日', + executionTime: '実行時間', + invalidTimezone: '無効なタイムゾーン', + invalidCronExpression: '無効なCron式', + noValidExecutionTime: '有効な実行時間を計算できません', + executionTimeCalculationError: '実行時間の計算に失敗しました', + invalidFrequency: '無効な頻度', + invalidStartTime: '無効な開始時間', + startTimeMustBeFuture: '開始時間は未来の時間である必要があります', + invalidTimeFormat: '無効な時間形式(期待される形式:HH:MM AM/PM)', + invalidWeekday: '無効な曜日:{{weekday}}', + invalidMonthlyDay: '月の日は1-31の間または"last"である必要があります', + invalidOnMinute: '分は0-59の間である必要があります', + invalidExecutionTime: '無効な実行時間', + executionTimeMustBeFuture: '実行時間は未来の時間である必要があります', + }, + triggerWebhook: { + title: 'Webhook トリガー', + nodeTitle: '🔗 Webhook トリガー', + configPlaceholder: 'Webhook トリガーの設定がここに実装されます', + webhookUrl: 'Webhook URL', + webhookUrlPlaceholder: '生成をクリックして Webhook URL を作成', + generate: '生成', + copy: 'コピー', + test: 'テスト', + urlGenerated: 'Webhook URL を生成しました', + urlGenerationFailed: 'Webhook URL の生成に失敗しました', + urlCopied: 'URL をクリップボードにコピーしました', + method: 'メソッド', + contentType: 'コンテンツタイプ', + queryParameters: 'クエリパラメータ', + headerParameters: 'ヘッダーパラメータ', + requestBodyParameters: 'リクエストボディパラメータ', + parameterName: '変数名', + varName: '変数名', + varType: 'タイプ', + varNamePlaceholder: '変数名を入力...', + headerName: '変数名', + required: '必須', + addParameter: '追加', + addHeader: '追加', + noParameters: '設定されたパラメータはありません', + noQueryParameters: 'クエリパラメータは設定されていません', + noHeaders: 'ヘッダーは設定されていません', + noBodyParameters: 'ボディパラメータは設定されていません', + debugUrlTitle: 'テスト実行には、常にこのURLを使用してください', + debugUrlCopy: 'クリックしてコピー', + debugUrlCopied: 'コピーしました!', + errorHandling: 'エラー処理', + errorStrategy: 'エラー処理', + responseConfiguration: 'レスポンス', + asyncMode: '非同期モード', + statusCode: 'ステータスコード', + responseBody: 'レスポンスボディ', + responseBodyPlaceholder: 'ここにレスポンスボディを入力してください', + headers: 'ヘッダー', + validation: { + webhookUrlRequired: 'Webhook URLが必要です', + invalidParameterType: 'パラメータ"{{name}}"の無効なパラメータタイプ"{{type}}"です', + }, + }, + triggerPlugin: { + authorized: '認可された', + notConfigured: '設定されていません', + error: 'エラー', + configuration: '構成', + remove: '削除する', + or: 'または', + useOAuth: 'OAuth を使用', + useApiKey: 'API キーを使用', + authenticationFailed: '認証に失敗しました', + authenticationSuccess: '認証に成功しました', + oauthConfigFailed: 'OAuth 設定に失敗しました', + configureOAuthClient: 'OAuth クライアントを設定', + oauthClientDescription: '認証を有効にするために OAuth クライアント認証情報を設定してください', + oauthClientSaved: 'OAuth クライアント設定が正常に保存されました', + configureApiKey: 'API キーを設定', + apiKeyDescription: '認証のための API キー認証情報を設定してください', + apiKeyConfigured: 'API キーが正常に設定されました', + configurationFailed: '設定に失敗しました', + failedToStart: '認証フローの開始に失敗しました', + credentialsVerified: '認証情報が正常に検証されました', + credentialVerificationFailed: '認証情報の検証に失敗しました', + verifyAndContinue: '検証して続行', + configureParameters: 'パラメーターを設定', + parametersDescription: 'トリガーのパラメーターとプロパティを設定してください', + configurationComplete: '設定完了', + configurationCompleteDescription: 'トリガーが正常に設定されました', + configurationCompleteMessage: 'トリガーの設定が完了し、使用する準備ができました。', + parameters: 'パラメーター', + properties: 'プロパティ', + propertiesDescription: 'このトリガーの追加設定プロパティ', + noConfigurationRequired: 'このトリガーには追加の設定は必要ありません。', + subscriptionName: 'サブスクリプション名', + subscriptionNameDescription: 'このトリガーサブスクリプションの一意な名前を入力してください', + subscriptionNamePlaceholder: 'サブスクリプション名を入力...', + subscriptionNameRequired: 'サブスクリプション名は必須です', + }, }, tracing: { stopBy: '{{user}}によって停止', @@ -1008,6 +1183,18 @@ const translation = { description: '最後の実行の結果がここに表示されます', }, variableInspect: { + listening: { + title: 'トリガーからのイベントを待機中…', + tip: 'HTTP {{nodeName}} エンドポイントにテストリクエストを送信するか、ライブイベントデバッグ用のコールバック URL として利用してイベントトリガーをシミュレートできます。すべての出力は Variable Inspector で直接確認できます。', + tipPlugin: '{{- pluginName}} でイベントを作成し、これらのイベントの出力を Variable Inspector で取得できます。', + tipSchedule: 'スケジュールトリガーからのイベントを待機しています。\n次回の予定実行: {{nextTriggerTime}}', + tipFallback: 'トリガーイベントを待機しています。出力はここに表示されます。', + defaultNodeName: 'このトリガー', + defaultPluginName: 'このプラグイントリガー', + defaultScheduleTime: '未設定', + selectedTriggers: '選択したトリガー', + stopButton: '停止', + }, trigger: { clear: 'クリア', running: 'キャッシング実行状況', @@ -1050,6 +1237,30 @@ const translation = { lastRunInputsCopied: '前回の実行から{{count}}個の入力をコピーしました', lastOutput: '最後の出力', }, + triggerStatus: { + enabled: 'トリガー', + disabled: 'トリガー • 無効', + }, + entryNodeStatus: { + enabled: 'スタート', + disabled: '開始 • 無効', + }, + onboarding: { + title: '開始するには開始ノードを選択してください', + description: '異なる開始ノードには異なる機能があります。心配しないでください、いつでも変更できます。', + userInputFull: 'ユーザー入力(元の開始ノード)', + userInputDescription: 'ユーザー入力変数の設定を可能にする開始ノードで、Webアプリ、サービスAPI、MCPサーバー、およびツールとしてのワークフロー機能を持ちます。', + trigger: 'トリガー', + triggerDescription: 'トリガーは、スケジュールされたタスク、カスタムwebhook、または他のアプリとの統合など、ワークフローの開始ノードとして機能できます。', + back: '戻る', + learnMore: '詳細を見る', + aboutStartNode: '開始ノードについて。', + escTip: { + press: '', + key: 'esc', + toDismiss: 'キーで閉じる', + }, + }, } export default translation diff --git a/web/i18n/ko-KR/app-debug.ts b/web/i18n/ko-KR/app-debug.ts index 7d59d681bc..0cd074a70f 100644 --- a/web/i18n/ko-KR/app-debug.ts +++ b/web/i18n/ko-KR/app-debug.ts @@ -255,7 +255,6 @@ const translation = { variableTable: { key: '변수 키', name: '사용자 입력 필드명', - optional: '옵션', type: '입력 타입', action: '액션', typeString: '문자열', diff --git a/web/i18n/ko-KR/app-log.ts b/web/i18n/ko-KR/app-log.ts index 1701d588b0..51cffcb374 100644 --- a/web/i18n/ko-KR/app-log.ts +++ b/web/i18n/ko-KR/app-log.ts @@ -66,6 +66,8 @@ const translation = { quarterToDate: '분기 초부터 오늘까지', yearToDate: '연 초부터 오늘까지', allTime: '모든 기간', + last30days: '최근 30일', + custom: '사용자 정의', }, annotation: { all: '모두', diff --git a/web/i18n/ko-KR/billing.ts b/web/i18n/ko-KR/billing.ts index c5f081d41b..112fa1bc63 100644 --- a/web/i18n/ko-KR/billing.ts +++ b/web/i18n/ko-KR/billing.ts @@ -88,7 +88,7 @@ const translation = { freeTrialTip: '200 회의 OpenAI 호출 무료 체험을 받으세요. ', annualBilling: '연간 청구', getStarted: '시작하기', - apiRateLimitUnit: '{{count,number}}/일', + apiRateLimitUnit: '{{count,number}}/월', freeTrialTipSuffix: '신용카드 없음', teamWorkspace: '{{count,number}} 팀 작업 공간', self: '자체 호스팅', diff --git a/web/i18n/ko-KR/common.ts b/web/i18n/ko-KR/common.ts index caff086f7c..22a4f918d5 100644 --- a/web/i18n/ko-KR/common.ts +++ b/web/i18n/ko-KR/common.ts @@ -157,7 +157,6 @@ const translation = { workspace: '작업 공간', createWorkspace: '작업 공간 만들기', helpCenter: '도움말 센터', - communityFeedback: '로드맵 및 피드백', roadmap: '로드맵', community: '커뮤니티', about: 'Dify 소개', @@ -166,6 +165,7 @@ const translation = { compliance: '컴플라이언스', support: '지원', contactUs: '문의하기', + forum: '포럼', }, settings: { accountGroup: '계정', @@ -722,6 +722,7 @@ const translation = { fileExtensionNotSupport: '지원되지 않는 파일 확장자', uploadFromComputerLimit: '업로드 파일은 {{size}}를 초과할 수 없습니다.', uploadFromComputerUploadError: '파일 업로드에 실패했습니다. 다시 업로드하십시오.', + fileExtensionBlocked: '보안상의 이유로 이 파일 형식은 차단되었습니다', }, license: { expiring_plural: '{{count}}일 후에 만료', diff --git a/web/i18n/ko-KR/login.ts b/web/i18n/ko-KR/login.ts index 8cde21472c..6d3d47a602 100644 --- a/web/i18n/ko-KR/login.ts +++ b/web/i18n/ko-KR/login.ts @@ -120,6 +120,7 @@ const translation = { noAccount: '계정이 없으신가요?', welcome: '👋 환영합니다! 시작하려면 세부 정보를 입력해 주세요.', }, + pageTitleForE: '이봐, 시작하자!', } export default translation diff --git a/web/i18n/ko-KR/tools.ts b/web/i18n/ko-KR/tools.ts index d8e975e61c..6bfed4e859 100644 --- a/web/i18n/ko-KR/tools.ts +++ b/web/i18n/ko-KR/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: '커스텀 도구 만들기', customToolTip: 'Dify 커스텀 도구에 대해 더 알아보기', type: { - all: '모두', builtIn: '내장', custom: '커스텀', workflow: '워크플로우', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: '자격 증명을 구성한 후에 워크스페이스의 모든 멤버가 이 도구를 사용하여 애플리케이션을 조작할 수 있습니다.', }, includeToolNum: '{{num}}개의 도구가 포함되어 있습니다', - addTool: '도구 추가', addToolModal: { type: '타입', category: '카테고리', - add: '추가', added: '추가됨', - manageInTools: '도구에서 관리', custom: { title: '사용자 정의 도구 없음', tip: '사용자 정의 도구 생성', @@ -203,6 +199,12 @@ const translation = { noHeaders: '사용자 정의 헤더가 구성되어 있지 않습니다.', headersTip: 'MCP 서버 요청과 함께 보낼 추가 HTTP 헤더', maskedHeadersTip: '헤더 값은 보안상 마스킹 처리되어 있습니다. 변경 사항은 실제 값에 업데이트됩니다.', + authentication: '인증', + configurations: '구성', + useDynamicClientRegistration: '동적 클라이언트 등록 사용', + clientSecret: '클라이언트 시크릿', + clientID: '클라이언트 ID', + clientSecretPlaceholder: '클라이언트 시크릿', }, delete: 'MCP 서버 제거', deleteConfirmTitle: '{mcp}를 제거하시겠습니까?', diff --git a/web/i18n/ko-KR/workflow.ts b/web/i18n/ko-KR/workflow.ts index 427452943a..e661b6b340 100644 --- a/web/i18n/ko-KR/workflow.ts +++ b/web/i18n/ko-KR/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: '게시됨', publish: '게시하기', update: '업데이트', - run: '실행', + run: '테스트 실행', running: '실행 중', inRunMode: '실행 모드', inPreview: '미리보기 중', @@ -18,7 +18,6 @@ const translation = { runHistory: '실행 기록', goBackToEdit: '편집기로 돌아가기', conversationLog: '대화 로그', - features: '기능', debugAndPreview: '미리보기', restart: '재시작', currentDraft: '현재 초안', @@ -93,9 +92,7 @@ const translation = { addParallelNode: '병렬 노드 추가', parallel: '병렬', branch: '브랜치', - featuresDocLink: '더 알아보세요', fileUploadTip: '이미지 업로드 기능이 파일 업로드로 업그레이드되었습니다.', - featuresDescription: '웹앱 사용자 경험 향상', ImageUploadLegacyTip: '이제 시작 양식에서 파일 형식 변수를 만들 수 있습니다. 앞으로 이미지 업로드 기능은 더 이상 지원되지 않습니다.', importWarning: '주의', @@ -115,10 +112,11 @@ const translation = { exportPNG: 'PNG 로 내보내기', addBlock: '노드 추가', needAnswerNode: '답변 노드를 추가해야 합니다.', - needEndNode: '종단 노드를 추가해야 합니다.', + needOutputNode: '출력 노드를 추가해야 합니다', tagBound: '이 태그를 사용하는 앱 수', currentView: '현재 보기', currentWorkflow: '현재 워크플로', + moreActions: '더 많은 작업', }, env: { envPanelTitle: '환경 변수', @@ -145,6 +143,19 @@ const translation = { export: '비밀 값이 포함된 DSL 내보내기', }, }, + globalVar: { + title: '시스템 변수', + description: '시스템 변수는 타입이 맞으면 배선 없이도 모든 노드에서 참조할 수 있는 전역 변수로, 엔드유저 ID와 워크플로 ID 등이 포함됩니다.', + fieldsDescription: { + conversationId: '대화 ID', + dialogCount: '대화 수', + userId: '사용자 ID', + triggerTimestamp: '애플리케이션 시작 타임스탬프', + appId: '애플리케이션 ID', + workflowId: '워크플로 ID', + workflowRunId: '워크플로 실행 ID', + }, + }, chatVariable: { panelTitle: '대화 변수', panelDescription: @@ -251,7 +262,7 @@ const translation = { }, blocks: { 'start': '시작', - 'end': '끝', + 'end': '출력', 'answer': '답변', 'llm': 'LLM', 'knowledge-retrieval': '지식 검색', @@ -277,7 +288,7 @@ const translation = { }, blocksAbout: { 'start': '워크플로우를 시작하기 위한 초기 매개변수를 정의합니다', - 'end': '워크플로우의 종료 및 결과 유형을 정의합니다', + 'end': '워크플로의 출력 및 결과 유형을 정의합니다', 'answer': '대화의 답변 내용을 정의합니다', 'llm': '질문에 답하거나 자연어를 처리하기 위해 대형 언어 모델을 호출합니다', 'knowledge-retrieval': @@ -332,7 +343,7 @@ const translation = { }, panel: { userInputField: '사용자 입력 필드', - helpLink: '도움말 링크', + helpLink: '도움말 센터', about: '정보', createdBy: '작성자 ', nextStep: '다음 단계', @@ -342,13 +353,13 @@ const translation = { checklistResolved: '모든 문제가 해결되었습니다', change: '변경', optional: '(선택사항)', - moveToThisNode: '이 노드로 이동', organizeBlocks: '노드 정리하기', selectNextStep: '다음 단계 선택', changeBlock: '노드 변경', addNextStep: '이 워크플로우에 다음 단계를 추가하세요.', minimize: '전체 화면 종료', maximize: '캔버스 전체 화면', + scrollToSelectedNode: '선택한 노드로 스크롤', optional_and_hidden: '(선택 사항 및 숨김)', }, nodes: { diff --git a/web/i18n/pl-PL/app-debug.ts b/web/i18n/pl-PL/app-debug.ts index 7b235f94de..ab4b0a06b0 100644 --- a/web/i18n/pl-PL/app-debug.ts +++ b/web/i18n/pl-PL/app-debug.ts @@ -277,7 +277,6 @@ const translation = { variableTable: { key: 'Klucz Zmiennej', name: 'Nazwa Pola Wejściowego Użytkownika', - optional: 'Opcjonalnie', type: 'Typ Wejścia', action: 'Akcje', typeString: 'String', diff --git a/web/i18n/pl-PL/app-log.ts b/web/i18n/pl-PL/app-log.ts index 90ad14ad0c..e341c7d331 100644 --- a/web/i18n/pl-PL/app-log.ts +++ b/web/i18n/pl-PL/app-log.ts @@ -69,6 +69,8 @@ const translation = { quarterToDate: 'Od początku kwartału', yearToDate: 'Od początku roku', allTime: 'Cały czas', + custom: 'Niestandardowy', + last30days: 'Ostatnie 30 dni', }, annotation: { all: 'Wszystkie', diff --git a/web/i18n/pl-PL/billing.ts b/web/i18n/pl-PL/billing.ts index cf0859468b..31aa337478 100644 --- a/web/i18n/pl-PL/billing.ts +++ b/web/i18n/pl-PL/billing.ts @@ -91,7 +91,7 @@ const translation = { freeTrialTipPrefix: 'Zarejestruj się i zdobądź', teamMember_other: '{{count,number}} członków zespołu', teamWorkspace: '{{count,number}} Zespół Workspace', - apiRateLimitUnit: '{{count,number}}/dzień', + apiRateLimitUnit: '{{count,number}}/miesiąc', cloud: 'Usługa chmurowa', teamMember_one: '{{count,number}} Członek zespołu', priceTip: 'na przestrzeń roboczą/', diff --git a/web/i18n/pl-PL/common.ts b/web/i18n/pl-PL/common.ts index b0ec44d963..b936080e8b 100644 --- a/web/i18n/pl-PL/common.ts +++ b/web/i18n/pl-PL/common.ts @@ -166,7 +166,6 @@ const translation = { workspace: 'Przestrzeń robocza', createWorkspace: 'Utwórz przestrzeń roboczą', helpCenter: 'Pomoc', - communityFeedback: 'Opinie', roadmap: 'Plan działania', community: 'Społeczność', about: 'O', @@ -175,6 +174,7 @@ const translation = { github: 'GitHub', compliance: 'Zgodność', contactUs: 'Skontaktuj się z nami', + forum: 'Forum', }, settings: { accountGroup: 'KONTO', @@ -744,6 +744,7 @@ const translation = { uploadFromComputerReadError: 'Odczyt pliku nie powiódł się, spróbuj ponownie.', fileExtensionNotSupport: 'Rozszerzenie pliku nie jest obsługiwane', uploadFromComputer: 'Przesyłanie lokalne', + fileExtensionBlocked: 'Ten typ pliku jest zablokowany ze względów bezpieczeństwa', }, license: { expiring_plural: 'Wygasa za {{count}} dni', diff --git a/web/i18n/pl-PL/login.ts b/web/i18n/pl-PL/login.ts index 394fe6c402..34519cd2b3 100644 --- a/web/i18n/pl-PL/login.ts +++ b/web/i18n/pl-PL/login.ts @@ -125,6 +125,7 @@ const translation = { haveAccount: 'Masz już konto?', welcome: '👋 Witaj! Proszę wypełnić szczegóły, aby rozpocząć.', }, + pageTitleForE: 'Hej, zaczynajmy!', } export default translation diff --git a/web/i18n/pl-PL/tools.ts b/web/i18n/pl-PL/tools.ts index dfa83d1231..fa6c5931e7 100644 --- a/web/i18n/pl-PL/tools.ts +++ b/web/i18n/pl-PL/tools.ts @@ -2,7 +2,6 @@ const translation = { title: 'Narzędzia', createCustomTool: 'Utwórz niestandardowe narzędzie', type: { - all: 'Wszystkie', builtIn: 'Wbudowane', custom: 'Niestandardowe', workflow: 'Przepływ pracy', @@ -21,7 +20,6 @@ const translation = { 'Po skonfigurowaniu poświadczeń wszyscy członkowie w przestrzeni roboczej mogą używać tego narzędzia podczas projektowania aplikacji.', }, includeToolNum: '{{num}} narzędzi zawarte', - addTool: 'Dodaj narzędzie', createTool: { title: 'Utwórz niestandardowe narzędzie', editAction: 'Konfiguruj', @@ -145,11 +143,9 @@ const translation = { notAuthorized: 'Narzędzie nieautoryzowane', howToGet: 'Jak uzyskać', addToolModal: { - manageInTools: 'Zarządzanie w Narzędziach', added: 'Dodane', type: 'typ', category: 'kategoria', - add: 'dodawać', custom: { title: 'Brak dostępnego narzędzia niestandardowego', tip: 'Utwórz narzędzie niestandardowe', @@ -207,6 +203,12 @@ const translation = { headerValue: 'Wartość nagłówka', noHeaders: 'Brak skonfigurowanych nagłówków niestandardowych', maskedHeadersTip: 'Wartości nagłówków są ukryte dla bezpieczeństwa. Zmiany zaktualizują rzeczywiste wartości.', + configurations: 'Konfiguracje', + authentication: 'Uwierzytelnianie', + clientSecretPlaceholder: 'Tajny klucz klienta', + clientSecret: 'Tajny klucz klienta', + useDynamicClientRegistration: 'Użyj dynamicznej rejestracji klienta', + clientID: 'ID klienta', }, delete: 'Usuń serwer MCP', deleteConfirmTitle: 'Usunąć {mcp}?', diff --git a/web/i18n/pl-PL/workflow.ts b/web/i18n/pl-PL/workflow.ts index 7c4d85e3ec..f30e9350f7 100644 --- a/web/i18n/pl-PL/workflow.ts +++ b/web/i18n/pl-PL/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Opublikowane', publish: 'Opublikuj', update: 'Aktualizuj', - run: 'Uruchom', + run: 'Uruchom test', running: 'Uruchamianie', inRunMode: 'W trybie uruchamiania', inPreview: 'W podglądzie', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Historia uruchomień', goBackToEdit: 'Wróć do edytora', conversationLog: 'Dziennik rozmów', - features: 'Funkcje', debugAndPreview: 'Podgląd', restart: 'Uruchom ponownie', currentDraft: 'Bieżący szkic', @@ -93,8 +92,6 @@ const translation = { branch: 'GAŁĄŹ', ImageUploadLegacyTip: 'Teraz można tworzyć zmienne typu pliku w formularzu startowym. W przyszłości nie będziemy już obsługiwać funkcji przesyłania obrazów.', fileUploadTip: 'Funkcje przesyłania obrazów zostały zaktualizowane do przesyłania plików.', - featuresDescription: 'Ulepszanie środowiska użytkownika aplikacji internetowej', - featuresDocLink: 'Dowiedz się więcej', importWarning: 'Ostrożność', importWarningDetails: 'Różnica w wersji DSL może mieć wpływ na niektóre funkcje', openInExplore: 'Otwieranie w obszarze Eksploruj', @@ -110,11 +107,12 @@ const translation = { exportPNG: 'Eksportuj jako PNG', publishUpdate: 'Opublikuj aktualizację', addBlock: 'Dodaj węzeł', - needEndNode: 'Należy dodać węzeł końcowy', + needOutputNode: 'Należy dodać węzeł wyjściowy', needAnswerNode: 'Węzeł odpowiedzi musi zostać dodany', tagBound: 'Liczba aplikacji korzystających z tego tagu', currentWorkflow: 'Bieżący przepływ pracy', currentView: 'Bieżący widok', + moreActions: 'Więcej akcji', }, env: { envPanelTitle: 'Zmienne Środowiskowe', @@ -139,6 +137,19 @@ const translation = { export: 'Eksportuj DSL z tajnymi wartościami', }, }, + globalVar: { + title: 'Zmienne systemowe', + description: 'Zmienne systemowe to zmienne globalne, do których może odwołać się każdy węzeł bez okablowania, jeśli typ jest zgodny, na przykład identyfikator użytkownika końcowego i identyfikator przepływu pracy.', + fieldsDescription: { + conversationId: 'ID konwersacji', + dialogCount: 'Liczba konwersacji', + userId: 'ID użytkownika', + triggerTimestamp: 'Znacznik czasu uruchomienia aplikacji', + appId: 'ID aplikacji', + workflowId: 'ID przepływu pracy', + workflowRunId: 'ID uruchomienia przepływu pracy', + }, + }, chatVariable: { panelTitle: 'Zmienne Konwersacji', panelDescription: 'Zmienne Konwersacji służą do przechowywania interaktywnych informacji, które LLM musi pamiętać, w tym historii konwersacji, przesłanych plików, preferencji użytkownika. Są one do odczytu i zapisu.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Start', - 'end': 'Koniec', + 'end': 'Wyjście', 'answer': 'Odpowiedź', 'llm': 'LLM', 'knowledge-retrieval': 'Wyszukiwanie wiedzy', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Zdefiniuj początkowe parametry uruchamiania przepływu pracy', - 'end': 'Zdefiniuj zakończenie i typ wyniku przepływu pracy', + 'end': 'Zdefiniuj wyjście i typ wyniku przepływu pracy', 'answer': 'Zdefiniuj treść odpowiedzi w rozmowie', 'llm': 'Wywołaj duże modele językowe do odpowiadania na pytania lub przetwarzania języka naturalnego', 'knowledge-retrieval': 'Pozwala na wyszukiwanie treści tekstowych związanych z pytaniami użytkowników z bazy wiedzy', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Pole wprowadzania użytkownika', - helpLink: 'Link do pomocy', + helpLink: 'Pomoc', about: 'O', createdBy: 'Stworzone przez ', nextStep: 'Następny krok', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Wszystkie problemy zostały rozwiązane', change: 'Zmień', optional: '(opcjonalne)', - moveToThisNode: 'Przenieś do tego węzła', selectNextStep: 'Wybierz następny krok', addNextStep: 'Dodaj następny krok w tym procesie roboczym', changeBlock: 'Zmień węzeł', organizeBlocks: 'Organizuj węzły', minimize: 'Wyjdź z trybu pełnoekranowego', maximize: 'Maksymalizuj płótno', + scrollToSelectedNode: 'Przewiń do wybranego węzła', optional_and_hidden: '(opcjonalne i ukryte)', }, nodes: { diff --git a/web/i18n/pt-BR/app-debug.ts b/web/i18n/pt-BR/app-debug.ts index ad46f8f0a9..1efec540df 100644 --- a/web/i18n/pt-BR/app-debug.ts +++ b/web/i18n/pt-BR/app-debug.ts @@ -261,7 +261,6 @@ const translation = { variableTable: { key: 'Chave da Variável', name: 'Nome do Campo de Entrada do Usuário', - optional: 'Opcional', type: 'Tipo de Entrada', action: 'Ações', typeString: 'Texto', diff --git a/web/i18n/pt-BR/app-log.ts b/web/i18n/pt-BR/app-log.ts index 9e2ff80759..9d7e318cfe 100644 --- a/web/i18n/pt-BR/app-log.ts +++ b/web/i18n/pt-BR/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Trimestre até hoje', yearToDate: 'Ano até hoje', allTime: 'Todo o tempo', + custom: 'Personalizado', + last30days: 'Últimos 30 Dias', }, annotation: { all: 'Tudo', diff --git a/web/i18n/pt-BR/billing.ts b/web/i18n/pt-BR/billing.ts index e4ca0a064a..9e58b24af4 100644 --- a/web/i18n/pt-BR/billing.ts +++ b/web/i18n/pt-BR/billing.ts @@ -80,7 +80,7 @@ const translation = { documentsRequestQuota: '{{count,number}}/min Limite de Taxa de Solicitação de Conhecimento', cloud: 'Serviço de Nuvem', teamWorkspace: '{{count,number}} Espaço de Trabalho da Equipe', - apiRateLimitUnit: '{{count,number}}/dia', + apiRateLimitUnit: '{{count,number}}/mês', freeTrialTipSuffix: 'Nenhum cartão de crédito necessário', teamMember_other: '{{count,number}} Membros da Equipe', comparePlanAndFeatures: 'Compare planos e recursos', diff --git a/web/i18n/pt-BR/common.ts b/web/i18n/pt-BR/common.ts index ccce7da2c3..e98d5d0748 100644 --- a/web/i18n/pt-BR/common.ts +++ b/web/i18n/pt-BR/common.ts @@ -161,7 +161,6 @@ const translation = { workspace: 'Espaço de trabalho', createWorkspace: 'Criar Espaço de Trabalho', helpCenter: 'Ajuda', - communityFeedback: 'Feedback', roadmap: 'Roteiro', community: 'Comunidade', about: 'Sobre', @@ -170,6 +169,7 @@ const translation = { support: 'Suporte', compliance: 'Conformidade', contactUs: 'Contate-Nos', + forum: 'Fórum', }, settings: { accountGroup: 'CONTA', @@ -726,6 +726,7 @@ const translation = { uploadFromComputerReadError: 'Falha na leitura do arquivo, tente novamente.', uploadFromComputerLimit: 'Carregar arquivo não pode exceder {{size}}', uploadFromComputerUploadError: 'Falha no upload do arquivo, faça o upload novamente.', + fileExtensionBlocked: 'Este tipo de arquivo está bloqueado por razões de segurança', }, license: { expiring: 'Expirando em um dia', diff --git a/web/i18n/pt-BR/login.ts b/web/i18n/pt-BR/login.ts index 200e7bf30c..4fa9f36146 100644 --- a/web/i18n/pt-BR/login.ts +++ b/web/i18n/pt-BR/login.ts @@ -120,6 +120,7 @@ const translation = { signUp: 'Inscreva-se', welcome: '👋 Bem-vindo! Por favor, preencha os detalhes para começar.', }, + pageTitleForE: 'Ei, vamos começar!', } export default translation diff --git a/web/i18n/pt-BR/tools.ts b/web/i18n/pt-BR/tools.ts index 401a81f615..6d5344b11b 100644 --- a/web/i18n/pt-BR/tools.ts +++ b/web/i18n/pt-BR/tools.ts @@ -2,7 +2,6 @@ const translation = { title: 'Ferramentas', createCustomTool: 'Criar Ferramenta Personalizada', type: { - all: 'Todas', builtIn: 'Integradas', custom: 'Personalizadas', workflow: 'Fluxo de trabalho', @@ -20,7 +19,6 @@ const translation = { setupModalTitleDescription: 'Após configurar as credenciais, todos os membros do espaço de trabalho podem usar essa ferramenta ao orquestrar aplicativos.', }, includeToolNum: '{{num}} ferramentas incluídas', - addTool: 'Adicionar Ferramenta', createTool: { title: 'Criar Ferramenta Personalizada', editAction: 'Configurar', @@ -143,9 +141,7 @@ const translation = { addToolModal: { category: 'categoria', type: 'tipo', - add: 'adicionar', added: 'Adicionado', - manageInTools: 'Gerenciar em Ferramentas', custom: { title: 'Nenhuma ferramenta personalizada disponível', tip: 'Crie uma ferramenta personalizada', @@ -203,6 +199,12 @@ const translation = { headerKey: 'Nome do Cabeçalho', noHeaders: 'Nenhum cabeçalho personalizado configurado', headerValuePlaceholder: 'ex: Token de portador 123', + useDynamicClientRegistration: 'Usar Registro Dinâmico de Cliente', + configurations: 'Configurações', + clientSecret: 'Segredo do Cliente', + authentication: 'Autenticação', + clientID: 'ID do Cliente', + clientSecretPlaceholder: 'Segredo do Cliente', }, delete: 'Remover Servidor MCP', deleteConfirmTitle: 'Você gostaria de remover {{mcp}}?', diff --git a/web/i18n/pt-BR/workflow.ts b/web/i18n/pt-BR/workflow.ts index bd5cf49ed7..265274c979 100644 --- a/web/i18n/pt-BR/workflow.ts +++ b/web/i18n/pt-BR/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Publicado', publish: 'Publicar', update: 'Atualizar', - run: 'Executar', + run: 'Executar teste', running: 'Executando', inRunMode: 'No modo de execução', inPreview: 'Em visualização', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Histórico de execução', goBackToEdit: 'Voltar para o editor', conversationLog: 'Registro de conversa', - features: 'Recursos', debugAndPreview: 'Visualizar', restart: 'Reiniciar', currentDraft: 'Rascunho atual', @@ -91,8 +90,6 @@ const translation = { addParallelNode: 'Adicionar nó paralelo', parallel: 'PARALELO', branch: 'RAMIFICAÇÃO', - featuresDocLink: 'Saiba Mais', - featuresDescription: 'Melhore a experiência do usuário do aplicativo Web', ImageUploadLegacyTip: 'Agora você pode criar variáveis de tipo de arquivo no formulário inicial. Não daremos mais suporte ao recurso de upload de imagens no futuro.', fileUploadTip: 'Os recursos de upload de imagens foram atualizados para upload de arquivos.', importWarning: 'Cuidado', @@ -110,11 +107,12 @@ const translation = { exportSVG: 'Exportar como SVG', exportJPEG: 'Exportar como JPEG', addBlock: 'Adicionar Nó', - needEndNode: 'O nó de Fim deve ser adicionado', + needOutputNode: 'O nó de Saída deve ser adicionado', needAnswerNode: 'O nó de resposta deve ser adicionado', tagBound: 'Número de aplicativos usando esta tag', currentView: 'Visualização atual', currentWorkflow: 'Fluxo de trabalho atual', + moreActions: 'Mais ações', }, env: { envPanelTitle: 'Variáveis de Ambiente', @@ -139,6 +137,19 @@ const translation = { export: 'Exportar DSL com valores secretos', }, }, + globalVar: { + title: 'Variáveis do sistema', + description: 'Variáveis do sistema são variáveis globais que qualquer nó pode referenciar sem conexões quando o tipo está correto, como o ID do usuário final e o ID do fluxo de trabalho.', + fieldsDescription: { + conversationId: 'ID da conversa', + dialogCount: 'Contagem de conversas', + userId: 'ID do usuário', + triggerTimestamp: 'Carimbo de data/hora do início da aplicação', + appId: 'ID da aplicação', + workflowId: 'ID do fluxo de trabalho', + workflowRunId: 'ID da execução do fluxo de trabalho', + }, + }, chatVariable: { panelTitle: 'Variáveis de Conversação', panelDescription: 'As Variáveis de Conversação são usadas para armazenar informações interativas que o LLM precisa lembrar, incluindo histórico de conversas, arquivos carregados, preferências do usuário. Elas são de leitura e escrita.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Iniciar', - 'end': 'Fim', + 'end': 'Saída', 'answer': 'Resposta', 'llm': 'LLM', 'knowledge-retrieval': 'Recuperação de conhecimento', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Definir os parâmetros iniciais para iniciar um fluxo de trabalho', - 'end': 'Definir o fim e o tipo de resultado de um fluxo de trabalho', + 'end': 'Definir a saída e o tipo de resultado de um fluxo de trabalho', 'answer': 'Definir o conteúdo da resposta de uma conversa', 'llm': 'Invocar grandes modelos de linguagem para responder perguntas ou processar linguagem natural', 'knowledge-retrieval': 'Permite consultar conteúdo de texto relacionado a perguntas do usuário a partir da base de conhecimento', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Campo de entrada do usuário', - helpLink: 'Link de ajuda', + helpLink: 'Ajuda', about: 'Sobre', createdBy: 'Criado por ', nextStep: 'Próximo passo', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Todos os problemas foram resolvidos', change: 'Mudar', optional: '(opcional)', - moveToThisNode: 'Mova-se para este nó', changeBlock: 'Mudar Nó', addNextStep: 'Adicione o próximo passo neste fluxo de trabalho', organizeBlocks: 'Organizar nós', selectNextStep: 'Selecione o próximo passo', maximize: 'Maximize Canvas', minimize: 'Sair do Modo Tela Cheia', + scrollToSelectedNode: 'Role até o nó selecionado', optional_and_hidden: '(opcional & oculto)', }, nodes: { diff --git a/web/i18n/ro-RO/app-debug.ts b/web/i18n/ro-RO/app-debug.ts index e3ecc1a8c0..fff56403a3 100644 --- a/web/i18n/ro-RO/app-debug.ts +++ b/web/i18n/ro-RO/app-debug.ts @@ -261,7 +261,6 @@ const translation = { variableTable: { key: 'Cheie variabilă', name: 'Nume câmp de intrare utilizator', - optional: 'Opțional', type: 'Tip intrare', action: 'Acțiuni', typeString: 'Șir', diff --git a/web/i18n/ro-RO/app-log.ts b/web/i18n/ro-RO/app-log.ts index 4a6e9bd96e..55f3ea2286 100644 --- a/web/i18n/ro-RO/app-log.ts +++ b/web/i18n/ro-RO/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Trimestrul curent', yearToDate: 'Anul curent', allTime: 'Tot timpul', + custom: 'Personalizat', + last30days: 'Ultimele 30 de zile', }, annotation: { all: 'Toate', diff --git a/web/i18n/ro-RO/billing.ts b/web/i18n/ro-RO/billing.ts index 3f5577dc32..0d787bb661 100644 --- a/web/i18n/ro-RO/billing.ts +++ b/web/i18n/ro-RO/billing.ts @@ -82,7 +82,7 @@ const translation = { documentsTooltip: 'Cota pe numărul de documente importate din Sursele de Date de Cunoștințe.', getStarted: 'Întrebați-vă', cloud: 'Serviciu de cloud', - apiRateLimitUnit: '{{count,number}}/zi', + apiRateLimitUnit: '{{count,number}}/lună', comparePlanAndFeatures: 'Compară planurile și caracteristicile', documentsRequestQuota: '{{count,number}}/min Limita de rată a cererilor de cunoștințe', documents: '{{count,number}} Documente de Cunoaștere', diff --git a/web/i18n/ro-RO/common.ts b/web/i18n/ro-RO/common.ts index 7e35a9fb26..f9b620e97e 100644 --- a/web/i18n/ro-RO/common.ts +++ b/web/i18n/ro-RO/common.ts @@ -161,7 +161,6 @@ const translation = { workspace: 'Spațiu de lucru', createWorkspace: 'Creează Spațiu de lucru', helpCenter: 'Ajutor', - communityFeedback: 'Feedback', roadmap: 'Plan de acțiune', community: 'Comunitate', about: 'Despre', @@ -170,6 +169,7 @@ const translation = { support: 'Suport', compliance: 'Conformitate', contactUs: 'Contactați-ne', + forum: 'Forum', }, settings: { accountGroup: 'CONT', @@ -726,6 +726,7 @@ const translation = { pasteFileLinkInvalid: 'Link fișier nevalid', uploadFromComputerLimit: 'Încărcarea fișierului nu poate depăși {{size}}', pasteFileLink: 'Lipiți linkul fișierului', + fileExtensionBlocked: 'Acest tip de fișier este blocat din motive de securitate', }, license: { expiring: 'Expiră într-o zi', diff --git a/web/i18n/ro-RO/login.ts b/web/i18n/ro-RO/login.ts index 34cd4a5ffd..f676b812cb 100644 --- a/web/i18n/ro-RO/login.ts +++ b/web/i18n/ro-RO/login.ts @@ -120,6 +120,7 @@ const translation = { createAccount: 'Creează-ți contul', welcome: '👋 Buna! Te rugăm să completezi detaliile pentru a începe.', }, + pageTitleForE: 'Hei, hai să începem!', } export default translation diff --git a/web/i18n/ro-RO/tools.ts b/web/i18n/ro-RO/tools.ts index b732128684..c9eeb29d97 100644 --- a/web/i18n/ro-RO/tools.ts +++ b/web/i18n/ro-RO/tools.ts @@ -2,7 +2,6 @@ const translation = { title: 'Instrumente', createCustomTool: 'Creează Instrument Personalizat', type: { - all: 'Toate', builtIn: 'Incorporat', custom: 'Personalizat', workflow: 'Flux de lucru', @@ -20,7 +19,6 @@ const translation = { setupModalTitleDescription: 'După configurarea credențialelor, toți membrii din spațiul de lucru pot utiliza acest instrument la orchestrarea aplicațiilor.', }, includeToolNum: '{{num}} instrumente incluse', - addTool: 'Adaugă Instrument', createTool: { title: 'Creează Instrument Personalizat', editAction: 'Configurează', @@ -143,8 +141,6 @@ const translation = { addToolModal: { added: 'adăugat', category: 'categorie', - manageInTools: 'Gestionați în Instrumente', - add: 'adăuga', type: 'tip', custom: { title: 'Niciun instrument personalizat disponibil', @@ -203,6 +199,12 @@ const translation = { maskedHeadersTip: 'Valorile de antet sunt mascate pentru securitate. Modificările vor actualiza valorile reale.', headersTip: 'Header-uri HTTP suplimentare de trimis cu cererile către serverul MCP', noHeaders: 'Nu sunt configurate antete personalizate.', + authentication: 'Autentificare', + configurations: 'Configurații', + clientSecretPlaceholder: 'Secretul Clientului', + clientID: 'ID client', + useDynamicClientRegistration: 'Utilizați înregistrarea dinamică a clientului', + clientSecret: 'Secretul Clientului', }, delete: 'Eliminare Server MCP', deleteConfirmTitle: 'Ștergeți {mcp}?', diff --git a/web/i18n/ro-RO/workflow.ts b/web/i18n/ro-RO/workflow.ts index ffa1282380..8d55033929 100644 --- a/web/i18n/ro-RO/workflow.ts +++ b/web/i18n/ro-RO/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Publicat', publish: 'Publică', update: 'Actualizează', - run: 'Rulează', + run: 'Rulează test', running: 'Rulând', inRunMode: 'În modul de rulare', inPreview: 'În previzualizare', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Istoric rulări', goBackToEdit: 'Înapoi la editor', conversationLog: 'Jurnal conversație', - features: 'Funcționalități', debugAndPreview: 'Previzualizare', restart: 'Repornește', currentDraft: 'Schimbare curentă', @@ -91,8 +90,6 @@ const translation = { addParallelNode: 'Adăugare nod paralel', parallel: 'PARALEL', branch: 'RAMURĂ', - featuresDescription: 'Îmbunătățiți experiența utilizatorului aplicației web', - featuresDocLink: 'Află mai multe', fileUploadTip: 'Funcțiile de încărcare a imaginilor au fost actualizate la încărcarea fișierelor.', ImageUploadLegacyTip: 'Acum puteți crea variabile de tip de fișier în formularul de pornire. Nu vom mai accepta funcția de încărcare a imaginilor în viitor.', importWarning: 'Prudență', @@ -111,10 +108,11 @@ const translation = { exportJPEG: 'Exportă ca JPEG', addBlock: 'Adaugă nod', needAnswerNode: 'Nodul de răspuns trebuie adăugat', - needEndNode: 'Nodul de sfârșit trebuie adăugat', + needOutputNode: 'Nodul de ieșire trebuie adăugat', tagBound: 'Numărul de aplicații care folosesc acest tag', currentView: 'Vizualizare curentă', currentWorkflow: 'Flux de lucru curent', + moreActions: 'Mai multe acțiuni', }, env: { envPanelTitle: 'Variabile de Mediu', @@ -139,6 +137,19 @@ const translation = { export: 'Exportă DSL cu valori secrete', }, }, + globalVar: { + title: 'Variabile de sistem', + description: 'Variabilele de sistem sunt variabile globale care pot fi folosite de orice nod fără conexiuni dacă tipul este corect, precum ID-ul utilizatorului final și ID-ul fluxului de lucru.', + fieldsDescription: { + conversationId: 'ID conversație', + dialogCount: 'Număr conversații', + userId: 'ID utilizator', + triggerTimestamp: 'Marcaj temporal al pornirii aplicației', + appId: 'ID aplicație', + workflowId: 'ID flux de lucru', + workflowRunId: 'ID rulare flux de lucru', + }, + }, chatVariable: { panelTitle: 'Variabile de Conversație', panelDescription: 'Variabilele de Conversație sunt utilizate pentru a stoca informații interactive pe care LLM trebuie să le rețină, inclusiv istoricul conversației, fișiere încărcate, preferințele utilizatorului. Acestea sunt citibile și inscriptibile.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Începe', - 'end': 'Sfârșit', + 'end': 'Ieșire', 'answer': 'Răspuns', 'llm': 'LLM', 'knowledge-retrieval': 'Recuperare de cunoștințe', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Definiți parametrii inițiali pentru lansarea unui flux de lucru', - 'end': 'Definiți sfârșitul și tipul rezultatului unui flux de lucru', + 'end': 'Definiți ieșirea și tipul rezultatului unui flux de lucru', 'answer': 'Definiți conținutul răspunsului unei conversații', 'llm': 'Invocarea modelelor de limbaj mari pentru a răspunde la întrebări sau pentru a procesa limbajul natural', 'knowledge-retrieval': 'Permite interogarea conținutului textului legat de întrebările utilizatorului din baza de cunoștințe', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Câmp de introducere utilizator', - helpLink: 'Link de ajutor', + helpLink: 'Ajutor', about: 'Despre', createdBy: 'Creat de ', nextStep: 'Pasul următor', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Toate problemele au fost rezolvate', change: 'Schimbă', optional: '(opțional)', - moveToThisNode: 'Mutați la acest nod', organizeBlocks: 'Organizează nodurile', addNextStep: 'Adăugați următorul pas în acest flux de lucru', changeBlock: 'Schimbă nodul', selectNextStep: 'Selectați Pasul Următor', maximize: 'Maximize Canvas', minimize: 'Iesi din modul pe tot ecranul', + scrollToSelectedNode: 'Derulați la nodul selectat', optional_and_hidden: '(opțional și ascuns)', }, nodes: { diff --git a/web/i18n/ru-RU/app-debug.ts b/web/i18n/ru-RU/app-debug.ts index 295bea3ba3..8d00994bef 100644 --- a/web/i18n/ru-RU/app-debug.ts +++ b/web/i18n/ru-RU/app-debug.ts @@ -327,7 +327,6 @@ const translation = { variableTable: { key: 'Ключ переменной', name: 'Имя поля пользовательского ввода', - optional: 'Необязательно', type: 'Тип ввода', action: 'Действия', typeString: 'Строка', diff --git a/web/i18n/ru-RU/app-log.ts b/web/i18n/ru-RU/app-log.ts index f874f5f523..03c5bee76b 100644 --- a/web/i18n/ru-RU/app-log.ts +++ b/web/i18n/ru-RU/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'С начала квартала', yearToDate: 'С начала года', allTime: 'Все время', + last30days: 'Последние 30 дней', + custom: 'Кастомный', }, annotation: { all: 'Все', diff --git a/web/i18n/ru-RU/billing.ts b/web/i18n/ru-RU/billing.ts index 7017f90cc2..1f3071a325 100644 --- a/web/i18n/ru-RU/billing.ts +++ b/web/i18n/ru-RU/billing.ts @@ -78,7 +78,7 @@ const translation = { apiRateLimit: 'Ограничение скорости API', self: 'Самостоятельно размещенный', teamMember_other: '{{count,number}} Члены команды', - apiRateLimitUnit: '{{count,number}}/день', + apiRateLimitUnit: '{{count,number}}/месяц', unlimitedApiRate: 'Нет ограничений на количество запросов к API', freeTrialTip: 'бесплатная пробная версия из 200 вызовов OpenAI.', freeTrialTipSuffix: 'Кредитная карта не требуется', diff --git a/web/i18n/ru-RU/common.ts b/web/i18n/ru-RU/common.ts index 2eb22284a5..ecfd241358 100644 --- a/web/i18n/ru-RU/common.ts +++ b/web/i18n/ru-RU/common.ts @@ -165,7 +165,6 @@ const translation = { workspace: 'Рабочее пространство', createWorkspace: 'Создать рабочее пространство', helpCenter: 'Помощь', - communityFeedback: 'Обратная связь', roadmap: 'План развития', community: 'Сообщество', about: 'О нас', @@ -174,6 +173,7 @@ const translation = { compliance: 'Соблюдение', support: 'Поддержка', contactUs: 'Свяжитесь с нами', + forum: 'Форум', }, settings: { accountGroup: 'АККАУНТ', @@ -726,6 +726,7 @@ const translation = { pasteFileLinkInvalid: 'Неверная ссылка на файл', uploadFromComputerLimit: 'Файл загрузки не может превышать {{size}}', uploadFromComputerUploadError: 'Загрузка файла не удалась, пожалуйста, загрузите еще раз.', + fileExtensionBlocked: 'Этот тип файла заблокирован по соображениям безопасности', }, license: { expiring: 'Срок действия истекает за один день', diff --git a/web/i18n/ru-RU/login.ts b/web/i18n/ru-RU/login.ts index bfb2860b57..f864bdb845 100644 --- a/web/i18n/ru-RU/login.ts +++ b/web/i18n/ru-RU/login.ts @@ -120,6 +120,7 @@ const translation = { verifyMail: 'Продолжите с кодом проверки', welcome: '👋 Добро пожаловать! Пожалуйста, заполните данные, чтобы начать.', }, + pageTitleForE: 'Привет, давай начнем!', } export default translation diff --git a/web/i18n/ru-RU/tools.ts b/web/i18n/ru-RU/tools.ts index 36d48affc2..48de76e383 100644 --- a/web/i18n/ru-RU/tools.ts +++ b/web/i18n/ru-RU/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'Создать пользовательский инструмент', customToolTip: 'Узнать больше о пользовательских инструментах Dify', type: { - all: 'Все', builtIn: 'Встроенные', custom: 'Пользовательские', workflow: 'Рабочий процесс', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: 'После настройки учетных данных все участники рабочего пространства смогут использовать этот инструмент при оркестровке приложений.', }, includeToolNum: 'Включено {{num}} инструментов', - addTool: 'Добавить инструмент', addToolModal: { type: 'тип', category: 'категория', - add: 'добавить', added: 'добавлено', - manageInTools: 'Управлять в инструментах', custom: { title: 'Нет доступного пользовательского инструмента', tip: 'Создать пользовательский инструмент', @@ -203,6 +199,12 @@ const translation = { noHeaders: 'Нет настроенных пользовательских заголовков', maskedHeadersTip: 'Значения заголовков скрыты для безопасности. Изменения обновят фактические значения.', headersTip: 'Дополнительные HTTP заголовки для отправки с запросами к серверу MCP', + configurations: 'Конфигурации', + clientID: 'Идентификатор клиента', + clientSecretPlaceholder: 'Секрет клиента', + useDynamicClientRegistration: 'Использовать динамическую регистрацию клиентов', + clientSecret: 'Секрет клиента', + authentication: 'Аутентификация', }, delete: 'Удалить MCP сервер', deleteConfirmTitle: 'Вы действительно хотите удалить {mcp}?', diff --git a/web/i18n/ru-RU/workflow.ts b/web/i18n/ru-RU/workflow.ts index 78be03ba91..9d7c99acea 100644 --- a/web/i18n/ru-RU/workflow.ts +++ b/web/i18n/ru-RU/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Опубликовано', publish: 'Опубликовать', update: 'Обновить', - run: 'Запустить', + run: 'Тестовый запуск', running: 'Выполняется', inRunMode: 'В режиме выполнения', inPreview: 'В режиме предпросмотра', @@ -18,7 +18,6 @@ const translation = { runHistory: 'История запусков', goBackToEdit: 'Вернуться к редактору', conversationLog: 'Журнал разговоров', - features: 'Функции', debugAndPreview: 'Предпросмотр', restart: 'Перезапустить', currentDraft: 'Текущий черновик', @@ -91,9 +90,7 @@ const translation = { addParallelNode: 'Добавить параллельный узел', parallel: 'ПАРАЛЛЕЛЬНЫЙ', branch: 'ВЕТКА', - featuresDocLink: 'Подробнее', fileUploadTip: 'Функции загрузки изображений были обновлены до загрузки файлов.', - featuresDescription: 'Улучшение взаимодействия с пользователем веб-приложения', ImageUploadLegacyTip: 'Теперь вы можете создавать переменные типа файла в стартовой форме. В будущем мы больше не будем поддерживать функцию загрузки изображений.', importWarning: 'Осторожность', importWarningDetails: 'Разница в версии DSL может повлиять на некоторые функции', @@ -111,10 +108,11 @@ const translation = { publishUpdate: 'Опубликовать обновление', addBlock: 'Добавить узел', needAnswerNode: 'В узел ответа необходимо добавить', - needEndNode: 'Узел конца должен быть добавлен', + needOutputNode: 'Необходимо добавить узел вывода', tagBound: 'Количество приложений, использующих этот тег', currentView: 'Текущий вид', currentWorkflow: 'Текущий рабочий процесс', + moreActions: 'Больше действий', }, env: { envPanelTitle: 'Переменные среды', @@ -139,6 +137,19 @@ const translation = { export: 'Экспортировать DSL с секретными значениями ', }, }, + globalVar: { + title: 'Системные переменные', + description: 'Системные переменные — это глобальные переменные, к которым любой узел может обращаться без соединений при корректном типе, например идентификатор конечного пользователя и идентификатор рабочего процесса.', + fieldsDescription: { + conversationId: 'ID беседы', + dialogCount: 'Количество бесед', + userId: 'ID пользователя', + triggerTimestamp: 'Отметка времени запуска приложения', + appId: 'ID приложения', + workflowId: 'ID рабочего процесса', + workflowRunId: 'ID запуска рабочего процесса', + }, + }, chatVariable: { panelTitle: 'Переменные разговора', panelDescription: 'Переменные разговора используются для хранения интерактивной информации, которую LLM необходимо запомнить, включая историю разговоров, загруженные файлы, пользовательские настройки. Они доступны для чтения и записи. ', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Начало', - 'end': 'Конец', + 'end': 'Вывод', 'answer': 'Ответ', 'llm': 'LLM', 'knowledge-retrieval': 'Поиск знаний', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Определите начальные параметры для запуска рабочего процесса', - 'end': 'Определите конец и тип результата рабочего процесса', + 'end': 'Определите вывод и тип результата рабочего процесса', 'answer': 'Определите содержимое ответа в чате', 'llm': 'Вызов больших языковых моделей для ответа на вопросы или обработки естественного языка', 'knowledge-retrieval': 'Позволяет запрашивать текстовый контент, связанный с вопросами пользователей, из базы знаний', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Поле ввода пользователя', - helpLink: 'Ссылка на справку', + helpLink: 'Помощь', about: 'О программе', createdBy: 'Создано ', nextStep: 'Следующий шаг', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Все проблемы решены', change: 'Изменить', optional: '(необязательно)', - moveToThisNode: 'Перейдите к этому узлу', selectNextStep: 'Выберите следующий шаг', organizeBlocks: 'Организовать узлы', addNextStep: 'Добавьте следующий шаг в этот рабочий процесс', changeBlock: 'Изменить узел', minimize: 'Выйти из полноэкранного режима', maximize: 'Максимизировать холст', + scrollToSelectedNode: 'Прокрутите до выбранного узла', optional_and_hidden: '(необязательно и скрыто)', }, nodes: { diff --git a/web/i18n/sl-SI/app-debug.ts b/web/i18n/sl-SI/app-debug.ts index 224c68e9e3..6642d79104 100644 --- a/web/i18n/sl-SI/app-debug.ts +++ b/web/i18n/sl-SI/app-debug.ts @@ -350,7 +350,6 @@ const translation = { }, variableTable: { action: 'Dejanja', - optional: 'Neobvezno', typeString: 'Niz', typeSelect: 'Izbrati', type: 'Vrsta vnosa', diff --git a/web/i18n/sl-SI/app-log.ts b/web/i18n/sl-SI/app-log.ts index 7f7cba0fa3..318ec731aa 100644 --- a/web/i18n/sl-SI/app-log.ts +++ b/web/i18n/sl-SI/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Četrtletje do danes', yearToDate: 'Leto do danes', allTime: 'Vse obdobje', + last30days: 'Zadnjih 30 dni', + custom: 'Po meri', }, annotation: { all: 'Vse', diff --git a/web/i18n/sl-SI/billing.ts b/web/i18n/sl-SI/billing.ts index fb9d9ec435..ef8c767090 100644 --- a/web/i18n/sl-SI/billing.ts +++ b/web/i18n/sl-SI/billing.ts @@ -86,7 +86,7 @@ const translation = { teamMember_one: '{{count,number}} član ekipe', teamMember_other: '{{count,number}} Članov ekipe', documentsRequestQuota: '{{count,number}}/min Omejitev stopnje zahtev po znanju', - apiRateLimitUnit: '{{count,number}}/dan', + apiRateLimitUnit: '{{count,number}}/mesec', priceTip: 'na delovnem prostoru/', freeTrialTipPrefix: 'Prijavite se in prejmite', cloud: 'Oblačna storitev', diff --git a/web/i18n/sl-SI/common.ts b/web/i18n/sl-SI/common.ts index 30400edf00..9a84513871 100644 --- a/web/i18n/sl-SI/common.ts +++ b/web/i18n/sl-SI/common.ts @@ -165,7 +165,6 @@ const translation = { workspace: 'Delovni prostor', createWorkspace: 'Ustvari delovni prostor', helpCenter: 'Pomoč', - communityFeedback: 'Povratne informacije', roadmap: 'Načrt razvoja', community: 'Skupnost', about: 'O nas', @@ -174,6 +173,7 @@ const translation = { github: 'GitHub', compliance: 'Skladnost', contactUs: 'Kontaktirajte nas', + forum: 'Forum', }, settings: { accountGroup: 'SPLOŠNO', @@ -792,6 +792,7 @@ const translation = { uploadFromComputer: 'Lokalno nalaganje', uploadFromComputerLimit: 'Nalaganje {{type}} ne sme presegati {{size}}', uploadFromComputerReadError: 'Branje datoteke ni uspelo, poskusite znova.', + fileExtensionBlocked: 'Ta vrsta datoteke je zaradi varnostnih razlogov blokirana', }, tag: { addTag: 'Dodajanje oznak', diff --git a/web/i18n/sl-SI/login.ts b/web/i18n/sl-SI/login.ts index 4e5b12689d..81f280666b 100644 --- a/web/i18n/sl-SI/login.ts +++ b/web/i18n/sl-SI/login.ts @@ -120,6 +120,7 @@ const translation = { noAccount: 'Nimate računa?', welcome: '👋 Dobrodošli! Prosimo, izpolnite podatke, da začnete.', }, + pageTitleForE: 'Hej, začnimo!', } export default translation diff --git a/web/i18n/sl-SI/tools.ts b/web/i18n/sl-SI/tools.ts index 8eb28c21bf..f8dd1dc831 100644 --- a/web/i18n/sl-SI/tools.ts +++ b/web/i18n/sl-SI/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'Ustvari prilagojeno orodje', customToolTip: 'Izvedite več o prilagojenih orodjih Dify', type: { - all: 'Vsa', builtIn: 'Vgrajena', custom: 'Prilagojena', workflow: 'Potek dela', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: 'Po konfiguraciji poverilnic bodo vsi člani znotraj delovnega prostora lahko uporabljali to orodje pri orkestraciji aplikacij.', }, includeToolNum: 'Vključeno {{num}} orodij', - addTool: 'Dodaj orodje', addToolModal: { type: 'tip', category: 'kategorija', - add: 'dodaj', added: 'dodano', - manageInTools: 'Upravljaj v Orodjih', custom: { title: 'Žiadne prispôsobené nástroje nie sú k dispozícii', tip: 'Vytvorte prispôsobený nástroj', @@ -203,6 +199,12 @@ const translation = { headerValuePlaceholder: 'npr., Bearer žeton123', noHeaders: 'Nobena prilagojena glava ni konfigurirana', maskedHeadersTip: 'Vrednosti glave so zakrite zaradi varnosti. Spremembe bodo posodobile dejanske vrednosti.', + authentication: 'Avtentikacija', + configurations: 'Konfiguracije', + clientSecret: 'Skrivnost odjemalca', + useDynamicClientRegistration: 'Uporabi dinamično registracijo odjemalca', + clientID: 'ID odjemalca', + clientSecretPlaceholder: 'Skrivnost odjemalca', }, delete: 'Odstrani strežnik MCP', deleteConfirmTitle: 'Odstraniti {mcp}?', diff --git a/web/i18n/sl-SI/workflow.ts b/web/i18n/sl-SI/workflow.ts index dbc4a75c43..fb1f709162 100644 --- a/web/i18n/sl-SI/workflow.ts +++ b/web/i18n/sl-SI/workflow.ts @@ -18,8 +18,7 @@ const translation = { }, versionHistory: 'Zgodovina različic', published: 'Objavljeno', - run: 'Teči', - featuresDocLink: 'Nauči se več', + run: 'Testni tek', notRunning: 'Še ne teče', exportImage: 'Izvozi sliko', openInExplore: 'Odpri v Raziskovanju', @@ -42,7 +41,7 @@ const translation = { inPreview: 'V predogledu', workflowAsToolTip: 'Zaradi posodobitve delovnega poteka je potrebna ponovna konfiguracija orodja.', variableNamePlaceholder: 'Ime spremenljivke', - needEndNode: 'Skrivnostna vozlišča je treba dodati.', + needOutputNode: 'Dodati je treba izhodiščno vozlišče', onFailure: 'O neuspehu', embedIntoSite: 'Vstavite v spletno stran', conversationLog: 'Pogovor Log', @@ -77,12 +76,10 @@ const translation = { fileUploadTip: 'Funkcije nalaganja slik so bile nadgrajene na nalaganje datotek.', backupCurrentDraft: 'Varnostno kopiraj trenutni osnutek', overwriteAndImport: 'Prepiši in uvozi', - features: 'Značilnosti', exportPNG: 'Izvozi kot PNG', chooseDSL: 'Izberi DSL datoteko', unpublished: 'Nepublikirano', pasteHere: 'Prilepite tukaj', - featuresDescription: 'Izboljšanje uporabniške izkušnje spletne aplikacije', exitVersions: 'Izhodne različice', editing: 'Urejanje', addFailureBranch: 'Dodaj neuspešno vejo', @@ -115,6 +112,7 @@ const translation = { tagBound: 'Število aplikacij, ki uporabljajo to oznako', currentView: 'Trenutni pogled', currentWorkflow: 'Trenutni potek dela', + moreActions: 'Več dejanj', }, env: { modal: { @@ -139,6 +137,19 @@ const translation = { envPanelButton: 'Dodaj spremenljivko', envDescription: 'Okoljske spremenljivke se lahko uporabljajo za shranjevanje zasebnih informacij in poverilnic. So samo za branje in jih je mogoče ločiti od DSL datoteke med izvozem.', }, + globalVar: { + title: 'Sistemske spremenljivke', + description: 'Sistemske spremenljivke so globalne spremenljivke, do katerih lahko vsako vozlišče dostopa brez povezovanja, če je tip pravilen, na primer ID končnega uporabnika in ID poteka dela.', + fieldsDescription: { + conversationId: 'ID pogovora', + dialogCount: 'Število pogovorov', + userId: 'ID uporabnika', + triggerTimestamp: 'Časovni žig začetka delovanja aplikacije', + appId: 'ID aplikacije', + workflowId: 'ID poteka dela', + workflowRunId: 'ID izvajanja poteka dela', + }, + }, chatVariable: { modal: { namePlaceholder: 'Ime spremenljivke', @@ -255,7 +266,7 @@ const translation = { 'code': 'Koda', 'template-transform': 'Predloga', 'answer': 'Odgovor', - 'end': 'Konec', + 'end': 'Izhod', 'iteration-start': 'Začetek iteracije', 'list-operator': 'Seznam operater', 'variable-aggregator': 'Spremenljivka agregator', @@ -275,7 +286,7 @@ const translation = { 'loop-end': 'Enakovredno „prekini“. Ta vozlišče nima konfiguracijskih elementov. Ko telo zanke doseže to vozlišče, zanka preneha.', 'document-extractor': 'Uporabljeno za razčlenitev prenesenih dokumentov v besedilno vsebino, ki jo je enostavno razumeti za LLM.', 'answer': 'Določi vsebino odgovora v pogovoru.', - 'end': 'Določite tip konca in rezultata delovnega toka', + 'end': 'Določite izhod in tip rezultata delovnega toka', 'knowledge-retrieval': 'Omogoča vam, da poizvedujete o besedilnih vsebinah, povezanih z vprašanji uporabnikov iz znanja.', 'http-request': 'Dovoli pošiljanje zahtevkov strežniku prek protokola HTTP', 'llm': 'Uporaba velikih jezikovnih modelov za odgovarjanje na vprašanja ali obdelavo naravnega jezika', @@ -324,10 +335,9 @@ const translation = { runThisStep: 'Izvedi ta korak', changeBlock: 'Spremeni vozlišče', addNextStep: 'Dodajte naslednji korak v ta delovni potek', - moveToThisNode: 'Premakni se na to vozlišče', checklistTip: 'Prepričajte se, da so vse težave rešene, preden objavite.', selectNextStep: 'Izberi naslednji korak', - helpLink: 'Pomočna povezava', + helpLink: 'Pomoč', checklist: 'Kontrolni seznam', checklistResolved: 'Vse težave so rešene', createdBy: 'Ustvarjeno z', @@ -335,6 +345,7 @@ const translation = { minimize: 'Izhod iz celotnega zaslona', maximize: 'Maksimiziraj platno', optional: '(neobvezno)', + scrollToSelectedNode: 'Pomaknite se do izbranega vozlišča', optional_and_hidden: '(neobvezno in skrito)', }, nodes: { diff --git a/web/i18n/th-TH/app-debug.ts b/web/i18n/th-TH/app-debug.ts index ce4b0f22e9..00704e76f5 100644 --- a/web/i18n/th-TH/app-debug.ts +++ b/web/i18n/th-TH/app-debug.ts @@ -323,7 +323,6 @@ const translation = { timeoutExceeded: 'ผลลัพธ์จะไม่แสดงเนื่องจากหมดเวลา โปรดดูบันทึกเพื่อรวบรวมผลลัพธ์ที่สมบูรณ์', }, variableTable: { - optional: 'เสริม', key: 'ปุ่มตัวแปร', typeString: 'เชือก', typeSelect: 'เลือก', diff --git a/web/i18n/th-TH/app-log.ts b/web/i18n/th-TH/app-log.ts index 8c47eaafdd..0e5758a8e1 100644 --- a/web/i18n/th-TH/app-log.ts +++ b/web/i18n/th-TH/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'ไตรมาสจนถึงปัจจุบัน', yearToDate: 'ปีจนถึงปัจจุบัน', allTime: 'ตลอดเวลา', + last30days: '30 วันที่ผ่านมา', + custom: 'กำหนดเอง', }, annotation: { all: 'ทั้งหมด', diff --git a/web/i18n/th-TH/billing.ts b/web/i18n/th-TH/billing.ts index 461e4a8240..a3bd5b85bc 100644 --- a/web/i18n/th-TH/billing.ts +++ b/web/i18n/th-TH/billing.ts @@ -82,7 +82,7 @@ const translation = { teamMember_one: '{{count,number}} สมาชิกทีม', unlimitedApiRate: 'ไม่มีข้อจำกัดอัตราการเรียก API', self: 'โฮสต์ด้วยตัวเอง', - apiRateLimitUnit: '{{count,number}}/วัน', + apiRateLimitUnit: '{{count,number}}/เดือน', teamMember_other: '{{count,number}} สมาชิกทีม', teamWorkspace: '{{count,number}} ทีมทำงาน', priceTip: 'ต่อพื้นที่ทำงาน/', diff --git a/web/i18n/th-TH/common.ts b/web/i18n/th-TH/common.ts index b7b8cfba53..62d9115ac9 100644 --- a/web/i18n/th-TH/common.ts +++ b/web/i18n/th-TH/common.ts @@ -160,7 +160,6 @@ const translation = { workspace: 'พื้นที่', createWorkspace: 'สร้างพื้นที่ทํางาน', helpCenter: 'วิธีใช้', - communityFeedback: 'การตอบสนอง', roadmap: 'แผนงาน', community: 'ชุมชน', about: 'ประมาณ', @@ -169,6 +168,7 @@ const translation = { compliance: 'การปฏิบัติตามข้อกำหนด', support: 'การสนับสนุน', contactUs: 'ติดต่อเรา', + forum: 'ฟอรั่ม', }, settings: { accountGroup: 'ทั่วไป', @@ -706,6 +706,7 @@ const translation = { uploadFromComputerLimit: 'อัปโหลด {{type}} ต้องไม่เกิน {{size}}', pasteFileLinkInvalid: 'ลิงก์ไฟล์ไม่ถูกต้อง', fileExtensionNotSupport: 'ไม่รองรับนามสกุลไฟล์', + fileExtensionBlocked: 'ประเภทไฟล์นี้ถูกบล็อกด้วยเหตุผลด้านความปลอดภัย', }, tag: { placeholder: 'แท็กทั้งหมด', diff --git a/web/i18n/th-TH/login.ts b/web/i18n/th-TH/login.ts index 732af8a875..517eee95a2 100644 --- a/web/i18n/th-TH/login.ts +++ b/web/i18n/th-TH/login.ts @@ -120,6 +120,7 @@ const translation = { verifyMail: 'โปรดดำเนินการต่อด้วยรหัสการตรวจสอบ', haveAccount: 'มีบัญชีอยู่แล้วใช่ไหม?', }, + pageTitleForE: 'เฮ้ เรามาเริ่มกันเถอะ!', } export default translation diff --git a/web/i18n/th-TH/tools.ts b/web/i18n/th-TH/tools.ts index 71175ff26c..47e160c9e9 100644 --- a/web/i18n/th-TH/tools.ts +++ b/web/i18n/th-TH/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'สร้างเครื่องมือที่กําหนดเอง', customToolTip: 'เรียนรู้เพิ่มเติมเกี่ยวกับเครื่องมือแบบกําหนดเองของ Dify', type: { - all: 'ทั้งหมด', builtIn: 'ในตัว', custom: 'ธรรมเนียม', workflow: 'เวิร์กโฟลว์', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: 'หลังจากกําหนดค่าข้อมูลประจําตัวแล้ว สมาชิกทั้งหมดภายในพื้นที่ทํางานสามารถใช้เครื่องมือนี้เมื่อประสานงานแอปพลิเคชันได้', }, includeToolNum: '{{num}} รวมเครื่องมือ', - addTool: 'เพิ่มเครื่องมือ', addToolModal: { type: 'ประเภท', category: 'ประเภท', - add: 'เพิ่ม', added: 'เพิ่ม', - manageInTools: 'จัดการในเครื่องมือ', custom: { title: 'ไม่มีเครื่องมือกำหนดเอง', tip: 'สร้างเครื่องมือกำหนดเอง', @@ -203,6 +199,12 @@ const translation = { noHeaders: 'ไม่มีการกำหนดหัวข้อที่กำหนดเอง', headersTip: 'HTTP header เพิ่มเติมที่จะส่งไปกับคำขอ MCP server', maskedHeadersTip: 'ค่าหัวถูกปกปิดเพื่อความปลอดภัย การเปลี่ยนแปลงจะปรับปรุงค่าที่แท้จริง', + clientSecret: 'รหัสลับของลูกค้า', + configurations: 'การตั้งค่า', + authentication: 'การตรวจสอบตัวตน', + clientSecretPlaceholder: 'รหัสลับของลูกค้า', + useDynamicClientRegistration: 'ใช้การลงทะเบียนลูกค้าแบบไดนามิก', + clientID: 'รหัสลูกค้า', }, delete: 'ลบเซิร์ฟเวอร์ MCP', deleteConfirmTitle: 'คุณต้องการลบ {mcp} หรือไม่?', diff --git a/web/i18n/th-TH/workflow.ts b/web/i18n/th-TH/workflow.ts index 419b577a02..51e9b4d088 100644 --- a/web/i18n/th-TH/workflow.ts +++ b/web/i18n/th-TH/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'เผย แพร่', publish: 'ตีพิมพ์', update: 'อัพเดต', - run: 'วิ่ง', + run: 'ทดสอบการทำงาน', running: 'กำลัง เรียก ใช้', inRunMode: 'ในโหมดเรียกใช้', inPreview: 'ในการแสดงตัวอย่าง', @@ -18,11 +18,8 @@ const translation = { runHistory: 'ประวัติการวิ่ง', goBackToEdit: 'กลับไปที่ตัวแก้ไข', conversationLog: 'บันทึกการสนทนา', - features: 'หน้าตา', - featuresDescription: 'ปรับปรุงประสบการณ์ผู้ใช้เว็บแอป', ImageUploadLegacyTip: 'ตอนนี้คุณสามารถสร้างตัวแปรชนิดไฟล์ในฟอร์มเริ่มต้นได้แล้ว เราจะไม่รองรับฟีเจอร์การอัปโหลดรูปภาพอีกต่อไปในอนาคต', fileUploadTip: 'ฟีเจอร์การอัปโหลดรูปภาพได้รับการอัปเกรดเป็นการอัปโหลดไฟล์', - featuresDocLink: 'ศึกษาเพิ่มเติม', debugAndPreview: 'ดูตัวอย่าง', restart: 'เริ่มใหม่', currentDraft: 'ร่างปัจจุบัน', @@ -111,10 +108,11 @@ const translation = { exportSVG: 'ส่งออกเป็น SVG', needAnswerNode: 'ต้องเพิ่มโหนดคำตอบ', addBlock: 'เพิ่มโนด', - needEndNode: 'ต้องเพิ่มโหนดจบ', + needOutputNode: 'ต้องเพิ่มโหนดเอาต์พุต', tagBound: 'จำนวนแอปพลิเคชันที่ใช้แท็กนี้', currentWorkflow: 'เวิร์กโฟลว์ปัจจุบัน', currentView: 'ปัจจุบัน View', + moreActions: 'การดําเนินการเพิ่มเติม', }, env: { envPanelTitle: 'ตัวแปรสภาพแวดล้อม', @@ -139,6 +137,19 @@ const translation = { export: 'ส่งออก DSL ด้วยค่าลับ', }, }, + globalVar: { + title: 'ตัวแปรระบบ', + description: 'ตัวแปรระบบเป็นตัวแปรแบบโกลบอลที่โหนดใด ๆ สามารถอ้างอิงได้โดยไม่ต้องเดินสายเมื่อชนิดข้อมูลถูกต้อง เช่น รหัสผู้ใช้ปลายทางและรหัสเวิร์กโฟลว์', + fieldsDescription: { + conversationId: 'รหัสการสนทนา', + dialogCount: 'จำนวนการสนทนา', + userId: 'รหัสผู้ใช้', + triggerTimestamp: 'ตราประทับเวลาที่แอปเริ่มทำงาน', + appId: 'รหัสแอปพลิเคชัน', + workflowId: 'รหัสเวิร์กโฟลว์', + workflowRunId: 'รหัสการรันเวิร์กโฟลว์', + }, + }, chatVariable: { panelTitle: 'ตัวแปรการสนทนา', panelDescription: 'ตัวแปรการสนทนาใช้เพื่อจัดเก็บข้อมูลแบบโต้ตอบที่ LLM จําเป็นต้องจดจํา รวมถึงประวัติการสนทนา ไฟล์ที่อัปโหลด การตั้งค่าของผู้ใช้ พวกเขาอ่าน-เขียน', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'เริ่ม', - 'end': 'ปลาย', + 'end': 'เอาต์พุต', 'answer': 'ตอบ', 'llm': 'นิติศาสตราจารย์', 'knowledge-retrieval': 'การดึงความรู้', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'กําหนดพารามิเตอร์เริ่มต้นสําหรับการเปิดใช้เวิร์กโฟลว์', - 'end': 'กําหนดชนิดสิ้นสุดและผลลัพธ์ของเวิร์กโฟลว์', + 'end': 'กำหนดเอาต์พุตและประเภทผลลัพธ์ของเวิร์กโฟลว์', 'answer': 'กําหนดเนื้อหาการตอบกลับของการสนทนาแชท', 'llm': 'การเรียกใช้โมเดลภาษาขนาดใหญ่เพื่อตอบคําถามหรือประมวลผลภาษาธรรมชาติ', 'knowledge-retrieval': 'ช่วยให้คุณสามารถสอบถามเนื้อหาข้อความที่เกี่ยวข้องกับคําถามของผู้ใช้จากความรู้', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'ฟิลด์ป้อนข้อมูลของผู้ใช้', - helpLink: 'ลิงค์ช่วยเหลือ', + helpLink: 'วิธีใช้', about: 'ประมาณ', createdBy: 'สร้างโดย', nextStep: 'ขั้นตอนถัดไป', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'ปัญหาทั้งหมดได้รับการแก้ไขแล้ว', change: 'เปลี่ยน', optional: '(ไม่บังคับ)', - moveToThisNode: 'ย้ายไปที่โหนดนี้', organizeBlocks: 'จัดระเบียบโหนด', addNextStep: 'เพิ่มขั้นตอนถัดไปในกระบวนการทำงานนี้', changeBlock: 'เปลี่ยนโหนด', selectNextStep: 'เลือกขั้นตอนถัดไป', minimize: 'ออกจากโหมดเต็มหน้าจอ', maximize: 'เพิ่มประสิทธิภาพผ้าใบ', + scrollToSelectedNode: 'เลื่อนไปยังโหนดที่เลือก', optional_and_hidden: '(ตัวเลือก & ซ่อน)', }, nodes: { diff --git a/web/i18n/tr-TR/app-debug.ts b/web/i18n/tr-TR/app-debug.ts index f707a691c2..d8ebc3d2df 100644 --- a/web/i18n/tr-TR/app-debug.ts +++ b/web/i18n/tr-TR/app-debug.ts @@ -327,7 +327,6 @@ const translation = { variableTable: { key: 'Değişken Anahtarı', name: 'Kullanıcı Giriş Alanı Adı', - optional: 'İsteğe Bağlı', type: 'Giriş Tipi', action: 'Aksiyonlar', typeString: 'Metin', diff --git a/web/i18n/tr-TR/app-log.ts b/web/i18n/tr-TR/app-log.ts index 380af8fd59..7b78fb452d 100644 --- a/web/i18n/tr-TR/app-log.ts +++ b/web/i18n/tr-TR/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Çeyrek Başlangıcından İtibaren', yearToDate: 'Yıl Başlangıcından İtibaren', allTime: 'Tüm Zamanlar', + custom: 'Özel', + last30days: 'Son 30 Gün', }, annotation: { all: 'Hepsi', diff --git a/web/i18n/tr-TR/billing.ts b/web/i18n/tr-TR/billing.ts index 6d01d9dd32..93c54fd1ed 100644 --- a/web/i18n/tr-TR/billing.ts +++ b/web/i18n/tr-TR/billing.ts @@ -78,7 +78,7 @@ const translation = { freeTrialTipPrefix: 'Kaydolun ve bir', priceTip: 'iş alanı başına/', documentsRequestQuota: '{{count,number}}/dakika Bilgi İsteği Oran Limiti', - apiRateLimitUnit: '{{count,number}}/gün', + apiRateLimitUnit: '{{count,number}}/ay', documents: '{{count,number}} Bilgi Belgesi', comparePlanAndFeatures: 'Planları ve özellikleri karşılaştır', self: 'Kendi Barındırılan', diff --git a/web/i18n/tr-TR/common.ts b/web/i18n/tr-TR/common.ts index b03b1423ae..3dc63a9ff8 100644 --- a/web/i18n/tr-TR/common.ts +++ b/web/i18n/tr-TR/common.ts @@ -165,7 +165,6 @@ const translation = { workspace: 'Çalışma Alanı', createWorkspace: 'Çalışma Alanı Oluştur', helpCenter: 'Yardım', - communityFeedback: 'Geri Bildirim', roadmap: 'Yol haritası', community: 'Topluluk', about: 'Hakkında', @@ -174,6 +173,7 @@ const translation = { compliance: 'Uygunluk', github: 'GitHub', contactUs: 'Bize Ulaşın', + forum: 'Forum', }, settings: { accountGroup: 'HESAP', @@ -726,6 +726,7 @@ const translation = { pasteFileLinkInputPlaceholder: 'URL\'yi giriniz...', pasteFileLinkInvalid: 'Geçersiz dosya bağlantısı', fileExtensionNotSupport: 'Dosya uzantısı desteklenmiyor', + fileExtensionBlocked: 'Bu dosya türü güvenlik nedenleriyle engellenmiştir', }, license: { expiring_plural: '{{count}} gün içinde sona eriyor', diff --git a/web/i18n/tr-TR/login.ts b/web/i18n/tr-TR/login.ts index b8bd6d74af..d6ada5f950 100644 --- a/web/i18n/tr-TR/login.ts +++ b/web/i18n/tr-TR/login.ts @@ -120,6 +120,7 @@ const translation = { haveAccount: 'Zaten bir hesabınız var mı?', welcome: '👋 Hoş geldiniz! Başlamak için lütfen detayları doldurun.', }, + pageTitleForE: 'Hey, haydi başlayalım!', } export default translation diff --git a/web/i18n/tr-TR/tools.ts b/web/i18n/tr-TR/tools.ts index d309b78689..12849b1879 100644 --- a/web/i18n/tr-TR/tools.ts +++ b/web/i18n/tr-TR/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: 'Özel Araç Oluştur', customToolTip: 'Dify özel araçları hakkında daha fazla bilgi edinin', type: { - all: 'Hepsi', builtIn: 'Yerleşik', custom: 'Özel', workflow: 'Workflow', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: 'Kimlik bilgilerini yapılandırdıktan sonra, çalışma alanındaki tüm üyeler uygulamaları düzenlerken bu aracı kullanabilir.', }, includeToolNum: '{{num}} araç dahil', - addTool: 'Araç Ekle', addToolModal: { type: 'Tür', category: 'Kategori', - add: 'Ekle', added: 'Eklendi', - manageInTools: 'Araçlarda Yönet', custom: { title: 'Mevcut özel araç yok', tip: 'Özel bir araç oluşturun', @@ -203,6 +199,12 @@ const translation = { headersTip: 'MCP sunucu istekleri ile gönderilecek ek HTTP başlıkları', headerValuePlaceholder: 'örneğin, Taşıyıcı jeton123', maskedHeadersTip: 'Başlık değerleri güvenlik amacıyla gizlenmiştir. Değişiklikler gerçek değerleri güncelleyecektir.', + clientID: 'Müşteri Kimliği', + configurations: 'Yapılandırmalar', + clientSecretPlaceholder: 'İstemci Sırrı', + clientSecret: 'İstemci Sırrı', + authentication: 'Kimlik Doğrulama', + useDynamicClientRegistration: 'Dinamik İstemci Kaydını Kullan', }, delete: 'MCP Sunucusunu Kaldır', deleteConfirmTitle: '{mcp} kaldırılsın mı?', diff --git a/web/i18n/tr-TR/workflow.ts b/web/i18n/tr-TR/workflow.ts index 930664ce57..df4f9c8093 100644 --- a/web/i18n/tr-TR/workflow.ts +++ b/web/i18n/tr-TR/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Yayınlandı', publish: 'Yayınla', update: 'Güncelle', - run: 'Çalıştır', + run: 'Test çalıştır', running: 'Çalışıyor', inRunMode: 'Çalıştırma Modunda', inPreview: 'Ön İzlemede', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Çalıştırma Geçmişi', goBackToEdit: 'Editöre geri dön', conversationLog: 'Konuşma Günlüğü', - features: 'Özellikler', debugAndPreview: 'Önizleme', restart: 'Yeniden Başlat', currentDraft: 'Geçerli Taslak', @@ -91,10 +90,8 @@ const translation = { disconnect: 'Ayırmak', parallel: 'PARALEL', branch: 'DAL', - featuresDocLink: 'Daha fazla bilgi edinin', fileUploadTip: 'Resim yükleme özellikleri, dosya yüklemeye yükseltildi.', ImageUploadLegacyTip: 'Artık başlangıç formunda dosya türü değişkenleri oluşturabilirsiniz. Gelecekte resim yükleme özelliğini artık desteklemeyeceğiz.', - featuresDescription: 'Web uygulaması kullanıcı deneyimini geliştirin', importWarningDetails: 'DSL sürüm farkı bazı özellikleri etkileyebilir', importWarning: 'Dikkat', openInExplore: 'Keşfet\'te Aç', @@ -111,10 +108,11 @@ const translation = { exportSVG: 'SVG olarak dışa aktar', addBlock: 'Düğüm Ekle', needAnswerNode: 'Cevap düğümü eklenmelidir.', - needEndNode: 'Son düğüm eklenmelidir', + needOutputNode: 'Çıktı düğümü eklenmelidir', tagBound: 'Bu etiketi kullanan uygulama sayısı', currentView: 'Geçerli Görünüm', currentWorkflow: 'Mevcut İş Akışı', + moreActions: 'Daha Fazla Eylem', }, env: { envPanelTitle: 'Çevre Değişkenleri', @@ -139,6 +137,19 @@ const translation = { export: 'Gizli değerlerle DSL\'yi dışa aktar', }, }, + globalVar: { + title: 'Sistem Değişkenleri', + description: 'Sistem değişkenleri, tipi uyumlu olduğunda herhangi bir düğümün bağlantı gerektirmeden başvurabileceği küresel değişkenlerdir; örneğin son kullanıcı kimliği ve iş akışı kimliği.', + fieldsDescription: { + conversationId: 'Konuşma Kimliği', + dialogCount: 'Konuşma Sayısı', + userId: 'Kullanıcı Kimliği', + triggerTimestamp: 'Uygulamanın çalışmaya başladığı zaman damgası', + appId: 'Uygulama Kimliği', + workflowId: 'İş Akışı Kimliği', + workflowRunId: 'İş akışı yürütme kimliği', + }, + }, chatVariable: { panelTitle: 'Konuşma Değişkenleri', panelDescription: 'Konuşma Değişkenleri, LLM\'nin hatırlaması gereken interaktif bilgileri (konuşma geçmişi, yüklenen dosyalar, kullanıcı tercihleri dahil) depolamak için kullanılır. Bunlar okunabilir ve yazılabilirdir.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Başlat', - 'end': 'Son', + 'end': 'Çıktı', 'answer': 'Yanıt', 'llm': 'LLM', 'knowledge-retrieval': 'Bilgi Geri Alımı', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Bir iş akışını başlatmak için başlangıç parametrelerini tanımlayın', - 'end': 'Bir iş akışının sonunu ve sonuç türünü tanımlayın', + 'end': 'Bir iş akışının çıktısını ve sonuç türünü tanımlayın', 'answer': 'Bir sohbet konuşmasının yanıt içeriğini tanımlayın', 'llm': 'Büyük dil modellerini soruları yanıtlamak veya doğal dili işlemek için çağırın', 'knowledge-retrieval': 'Kullanıcı sorularıyla ilgili metin içeriğini Bilgi\'den sorgulamanıza olanak tanır', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Kullanıcı Giriş Alanı', - helpLink: 'Yardım Linki', + helpLink: 'Yardım', about: 'Hakkında', createdBy: 'Oluşturan: ', nextStep: 'Sonraki Adım', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Tüm sorunlar çözüldü', change: 'Değiştir', optional: '(isteğe bağlı)', - moveToThisNode: 'Bu düğüme geç', changeBlock: 'Düğümü Değiştir', addNextStep: 'Bu iş akışına bir sonraki adımı ekleyin', organizeBlocks: 'Düğümleri düzenle', selectNextStep: 'Sonraki Adımı Seç', minimize: 'Tam Ekrandan Çık', maximize: 'Kanvası Maksimize Et', + scrollToSelectedNode: 'Seçili düğüme kaydırma', optional_and_hidden: '(isteğe bağlı ve gizli)', }, nodes: { diff --git a/web/i18n/uk-UA/app-debug.ts b/web/i18n/uk-UA/app-debug.ts index 3f7b6582bb..87b35168eb 100644 --- a/web/i18n/uk-UA/app-debug.ts +++ b/web/i18n/uk-UA/app-debug.ts @@ -273,7 +273,6 @@ const translation = { variableTable: { key: 'Ключ змінної', // Variable Key name: 'Назва поля для введення користувача', // User Input Field Name - optional: 'Додатково', // Optional type: 'Тип введення', // Input Type action: 'Дії', // Actions typeString: 'Рядок', // String diff --git a/web/i18n/uk-UA/app-log.ts b/web/i18n/uk-UA/app-log.ts index 8f8f3db5da..63fc63111a 100644 --- a/web/i18n/uk-UA/app-log.ts +++ b/web/i18n/uk-UA/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Квартал до сьогодні', yearToDate: 'Рік до сьогодні', allTime: 'За весь час', + last30days: 'Останні 30 днів', + custom: 'Користувацький', }, annotation: { all: 'Всі', diff --git a/web/i18n/uk-UA/billing.ts b/web/i18n/uk-UA/billing.ts index 03b743e4fe..e98b3e6091 100644 --- a/web/i18n/uk-UA/billing.ts +++ b/web/i18n/uk-UA/billing.ts @@ -84,7 +84,7 @@ const translation = { priceTip: 'за робочим простором/', unlimitedApiRate: 'Немає обмеження на швидкість API', freeTrialTipSuffix: 'Кредитна картка не потрібна', - apiRateLimitUnit: '{{count,number}}/день', + apiRateLimitUnit: '{{count,number}}/місяць', getStarted: 'Почати', freeTrialTip: 'безкоштовна пробна версія з 200 запитів до OpenAI.', documents: '{{count,number}} Документів знань', diff --git a/web/i18n/uk-UA/common.ts b/web/i18n/uk-UA/common.ts index a34cfc3725..941200d3a4 100644 --- a/web/i18n/uk-UA/common.ts +++ b/web/i18n/uk-UA/common.ts @@ -161,7 +161,6 @@ const translation = { workspace: 'Робочий простір', createWorkspace: 'Створити робочий простір', helpCenter: 'Довідковий центр', - communityFeedback: 'відгуки', roadmap: 'Дорожня карта', community: 'Спільнота', about: 'Про нас', @@ -170,6 +169,7 @@ const translation = { support: 'Підтримка', github: 'Гітхаб', contactUs: 'Зв’яжіться з нами', + forum: 'Форум', }, settings: { accountGroup: 'ОБЛІКОВИЙ ЗАПИС', @@ -727,6 +727,7 @@ const translation = { fileExtensionNotSupport: 'Розширення файлу не підтримується', uploadFromComputerReadError: 'Не вдалося прочитати файл, будь ласка, спробуйте ще раз.', uploadFromComputerUploadError: 'Не вдалося завантажити файл, будь ласка, завантажте ще раз.', + fileExtensionBlocked: 'Цей тип файлу заблоковано з міркувань безпеки', }, license: { expiring: 'Термін дії закінчується за один день', diff --git a/web/i18n/uk-UA/login.ts b/web/i18n/uk-UA/login.ts index 1fa4d414f7..1a1a6d7068 100644 --- a/web/i18n/uk-UA/login.ts +++ b/web/i18n/uk-UA/login.ts @@ -120,6 +120,7 @@ const translation = { noAccount: 'Не маєте облікового запису?', welcome: '👋 Ласкаво просимо! Будь ласка, заповніть деталі, щоб почати.', }, + pageTitleForE: 'Гей, давай почнемо!', } export default translation diff --git a/web/i18n/uk-UA/tools.ts b/web/i18n/uk-UA/tools.ts index 596153974f..3a3f72b5ba 100644 --- a/web/i18n/uk-UA/tools.ts +++ b/web/i18n/uk-UA/tools.ts @@ -2,7 +2,6 @@ const translation = { title: 'Інструменти', createCustomTool: 'Створити власний інструмент', type: { - all: 'Усі', builtIn: 'Вбудовані', custom: 'Користувацькі', workflow: 'Робочий процес', @@ -20,7 +19,6 @@ const translation = { setupModalTitleDescription: 'Після налаштування облікових даних усі члени робочого простору можуть використовувати цей інструмент під час оркестрування програм.', }, includeToolNum: '{{num}} інструмент(ів) включено', - addTool: 'Додати інструмент ', createTool: { title: 'Створити власний інструмент', editAction: 'Налаштування', @@ -142,10 +140,8 @@ const translation = { howToGet: 'Як отримати', addToolModal: { category: 'категорія', - add: 'Додати', added: 'Додано', type: 'тип', - manageInTools: 'Керування в інструментах', custom: { title: 'Немає доступного користувацького інструмента', tip: 'Створити користувацький інструмент', @@ -203,6 +199,12 @@ const translation = { headerKeyPlaceholder: 'наприклад, Авторизація', maskedHeadersTip: 'Значення заголовків маскуються для безпеки. Зміни оновлять фактичні значення.', headersTip: 'Додаткові HTTP заголовки для відправлення з запитами до сервера MCP', + clientSecret: 'Секрет клієнта', + clientSecretPlaceholder: 'Секрет клієнта', + clientID: 'Ідентифікатор клієнта', + authentication: 'Аутентифікація', + configurations: 'Конфігурації', + useDynamicClientRegistration: 'Використовувати динамічну реєстрацію клієнтів', }, delete: 'Видалити сервер MCP', deleteConfirmTitle: 'Видалити {mcp}?', diff --git a/web/i18n/uk-UA/workflow.ts b/web/i18n/uk-UA/workflow.ts index 2f4f298204..95425f0a32 100644 --- a/web/i18n/uk-UA/workflow.ts +++ b/web/i18n/uk-UA/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Опубліковано', publish: 'Опублікувати', update: 'Оновити', - run: 'Запустити', + run: 'Тестовий запуск', running: 'Запущено', inRunMode: 'У режимі запуску', inPreview: 'У режимі попереднього перегляду', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Історія запусків', goBackToEdit: 'Повернутися до редактора', conversationLog: 'Журнал розмов', - features: 'Функції', debugAndPreview: 'Попередній перегляд', restart: 'Перезапустити', currentDraft: 'Поточний чернетка', @@ -91,8 +90,6 @@ const translation = { addParallelNode: 'Додати паралельний вузол', parallel: 'ПАРАЛЕЛЬНИЙ', branch: 'ГІЛКА', - featuresDocLink: 'Дізнатися більше', - featuresDescription: 'Покращення взаємодії з користувачем веб-додатку', fileUploadTip: 'Функції завантаження зображень були оновлені для завантаження файлів.', ImageUploadLegacyTip: 'Тепер ви можете створювати змінні типу файлу у початковій формі. У майбутньому ми більше не підтримуватимемо функцію завантаження зображень.', importWarning: 'Обережність', @@ -110,11 +107,12 @@ const translation = { exportSVG: 'Експортувати як SVG', exportJPEG: 'Експортувати як JPEG', addBlock: 'Додати вузол', - needEndNode: 'Необхідно додати кінцевий вузол', + needOutputNode: 'Необхідно додати вихідний вузол', needAnswerNode: 'Вузол Відповіді повинен бути доданий', tagBound: 'Кількість додатків, що використовують цей тег', currentView: 'Поточний вигляд', currentWorkflow: 'Поточний робочий процес', + moreActions: 'Більше дій', }, env: { envPanelTitle: 'Змінні середовища', @@ -139,6 +137,19 @@ const translation = { export: 'Експортувати DSL з секретними значеннями', }, }, + globalVar: { + title: 'Системні змінні', + description: 'Системні змінні — це глобальні змінні, до яких будь-який вузол може звертатися без з’єднання, якщо тип відповідає, наприклад ID кінцевого користувача та ID робочого процесу.', + fieldsDescription: { + conversationId: 'ID розмови', + dialogCount: 'Кількість розмов', + userId: 'ID користувача', + triggerTimestamp: 'Мітка часу запуску застосунку', + appId: 'ID застосунку', + workflowId: 'ID робочого процесу', + workflowRunId: 'ID запуску робочого процесу', + }, + }, chatVariable: { panelTitle: 'Змінні розмови', panelDescription: 'Змінні розмови використовуються для зберігання інтерактивної інформації, яку LLM повинен пам\'ятати, включаючи історію розмови, завантажені файли, вподобання користувача. Вони доступні для читання та запису.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Початок', - 'end': 'Кінець', + 'end': 'Вивід', 'answer': 'Відповідь', 'llm': 'LLM', 'knowledge-retrieval': 'Отримання знань', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Визначте початкові параметри для запуску робочого потоку', - 'end': 'Визначте кінець і тип результату робочого потоку', + 'end': 'Визначте вивід і тип результату робочого потоку', 'answer': 'Визначте зміст відповіді у чаті', 'llm': 'Виклик великих мовних моделей для відповіді на запитання або обробки природної мови', 'knowledge-retrieval': 'Дозволяє виконувати запити текстового вмісту, пов\'язаного із запитаннями користувача, з бази знань', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Поле введення користувача', - helpLink: 'Посилання на допомогу', + helpLink: 'Довідковий центр', about: 'Про', createdBy: 'Створено ', nextStep: 'Наступний крок', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Всі проблеми вирішені', change: 'Змінити', optional: '(необов\'язково)', - moveToThisNode: 'Перемістіть до цього вузла', organizeBlocks: 'Організуйте вузли', changeBlock: 'Змінити вузол', selectNextStep: 'Виберіть наступний крок', addNextStep: 'Додайте наступний крок у цей робочий процес', minimize: 'Вийти з повноекранного режиму', maximize: 'Максимізувати полотно', + scrollToSelectedNode: 'Прокрутіть до вибраного вузла', optional_and_hidden: '(необов\'язково & приховано)', }, nodes: { diff --git a/web/i18n/vi-VN/app-debug.ts b/web/i18n/vi-VN/app-debug.ts index bd3823f1ff..9e71899b86 100644 --- a/web/i18n/vi-VN/app-debug.ts +++ b/web/i18n/vi-VN/app-debug.ts @@ -255,7 +255,6 @@ const translation = { variableTable: { key: 'Khóa biến', name: 'Tên trường nhập liệu người dùng', - optional: 'Tùy chọn', type: 'Loại nhập liệu', action: 'Hành động', typeString: 'Chuỗi', diff --git a/web/i18n/vi-VN/app-log.ts b/web/i18n/vi-VN/app-log.ts index 167b8747e2..121a06d8b6 100644 --- a/web/i18n/vi-VN/app-log.ts +++ b/web/i18n/vi-VN/app-log.ts @@ -65,6 +65,8 @@ const translation = { quarterToDate: 'Quý hiện tại', yearToDate: 'Năm hiện tại', allTime: 'Tất cả thời gian', + custom: 'Tùy chỉnh', + last30days: '30 Ngày Qua', }, annotation: { all: 'Tất cả', diff --git a/web/i18n/vi-VN/billing.ts b/web/i18n/vi-VN/billing.ts index 0166185e45..c6a7458164 100644 --- a/web/i18n/vi-VN/billing.ts +++ b/web/i18n/vi-VN/billing.ts @@ -90,7 +90,7 @@ const translation = { teamMember_other: '{{count,number}} thành viên trong nhóm', documents: '{{count,number}} Tài liệu Kiến thức', getStarted: 'Bắt đầu', - apiRateLimitUnit: '{{count,number}}/ngày', + apiRateLimitUnit: '{{count,number}}/tháng', freeTrialTipSuffix: 'Không cần thẻ tín dụng', documentsRequestQuotaTooltip: 'Chỉ định tổng số hành động mà một không gian làm việc có thể thực hiện mỗi phút trong cơ sở tri thức, bao gồm tạo mới tập dữ liệu, xóa, cập nhật, tải tài liệu lên, thay đổi, lưu trữ và truy vấn cơ sở tri thức. Chỉ số này được sử dụng để đánh giá hiệu suất của các yêu cầu cơ sở tri thức. Ví dụ, nếu một người dùng Sandbox thực hiện 10 lần kiểm tra liên tiếp trong một phút, không gian làm việc của họ sẽ bị hạn chế tạm thời không thực hiện các hành động sau trong phút tiếp theo: tạo mới tập dữ liệu, xóa, cập nhật và tải tài liệu lên hoặc thay đổi.', startBuilding: 'Bắt đầu xây dựng', diff --git a/web/i18n/vi-VN/common.ts b/web/i18n/vi-VN/common.ts index d0d6222d08..b929854555 100644 --- a/web/i18n/vi-VN/common.ts +++ b/web/i18n/vi-VN/common.ts @@ -161,7 +161,6 @@ const translation = { workspace: 'Không gian làm việc', createWorkspace: 'Tạo Không gian làm việc', helpCenter: 'Trung tâm trợ giúp', - communityFeedback: 'Phản hồi', roadmap: 'Lộ trình', community: 'Cộng đồng', about: 'Về chúng tôi', @@ -170,6 +169,7 @@ const translation = { github: 'GitHub', support: 'Hỗ trợ', contactUs: 'Liên hệ với chúng tôi', + forum: 'Diễn đàn', }, settings: { accountGroup: 'TÀI KHOẢN', @@ -726,6 +726,7 @@ const translation = { pasteFileLinkInvalid: 'Liên kết tệp không hợp lệ', uploadFromComputerUploadError: 'Tải lên tệp không thành công, vui lòng tải lên lại.', uploadFromComputerReadError: 'Đọc tệp không thành công, vui lòng thử lại.', + fileExtensionBlocked: 'Loại tệp này bị chặn vì lý do bảo mật', }, license: { expiring_plural: 'Hết hạn sau {{count}} ngày', diff --git a/web/i18n/vi-VN/login.ts b/web/i18n/vi-VN/login.ts index 6d877fffef..dec7eddee2 100644 --- a/web/i18n/vi-VN/login.ts +++ b/web/i18n/vi-VN/login.ts @@ -120,6 +120,7 @@ const translation = { verifyMail: 'Tiếp tục với mã xác minh', welcome: '👋 Chào mừng! Vui lòng điền vào các chi tiết để bắt đầu.', }, + pageTitleForE: 'Này, hãy bắt đầu nào!', } export default translation diff --git a/web/i18n/vi-VN/tools.ts b/web/i18n/vi-VN/tools.ts index 7c0826890e..a499a451a3 100644 --- a/web/i18n/vi-VN/tools.ts +++ b/web/i18n/vi-VN/tools.ts @@ -2,7 +2,6 @@ const translation = { title: 'Công cụ', createCustomTool: 'Tạo công cụ tùy chỉnh', type: { - all: 'Tất cả', builtIn: 'Tích hợp sẵn', custom: 'Tùy chỉnh', workflow: 'Quy trình làm việc', @@ -20,7 +19,6 @@ const translation = { setupModalTitleDescription: 'Sau khi cấu hình thông tin đăng nhập, tất cả thành viên trong không gian làm việc có thể sử dụng công cụ này khi triển khai ứng dụng.', }, includeToolNum: 'Bao gồm {{num}} công cụ', - addTool: 'Thêm công cụ', createTool: { title: 'Tạo công cụ tùy chỉnh', editAction: 'Cấu hình', @@ -142,9 +140,7 @@ const translation = { howToGet: 'Cách nhận', addToolModal: { category: 'loại', - manageInTools: 'Quản lý trong Công cụ', type: 'kiểu', - add: 'thêm', added: 'Thêm', custom: { title: 'Không có công cụ tùy chỉnh nào', @@ -203,6 +199,12 @@ const translation = { headerValue: 'Giá trị tiêu đề', maskedHeadersTip: 'Các giá trị tiêu đề được mã hóa để đảm bảo an ninh. Các thay đổi sẽ cập nhật các giá trị thực tế.', headersTip: 'Các tiêu đề HTTP bổ sung để gửi cùng với các yêu cầu máy chủ MCP', + authentication: 'Xác thực', + clientSecret: 'Bí mật của khách hàng', + clientID: 'ID khách hàng', + configurations: 'Cấu hình', + useDynamicClientRegistration: 'Sử dụng Đăng ký Khách hàng Động', + clientSecretPlaceholder: 'Bí mật của khách hàng', }, delete: 'Xóa Máy chủ MCP', deleteConfirmTitle: 'Xóa {mcp}?', diff --git a/web/i18n/vi-VN/workflow.ts b/web/i18n/vi-VN/workflow.ts index 4a3a720cb3..4fe45a8cc6 100644 --- a/web/i18n/vi-VN/workflow.ts +++ b/web/i18n/vi-VN/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: 'Đã xuất bản', publish: 'Xuất bản', update: 'Cập nhật', - run: 'Chạy', + run: 'Chạy thử nghiệm', running: 'Đang chạy', inRunMode: 'Chế độ chạy', inPreview: 'Trong chế độ xem trước', @@ -18,7 +18,6 @@ const translation = { runHistory: 'Lịch sử chạy', goBackToEdit: 'Quay lại trình chỉnh sửa', conversationLog: 'Nhật ký cuộc trò chuyện', - features: 'Tính năng', debugAndPreview: 'Xem trước', restart: 'Khởi động lại', currentDraft: 'Bản nháp hiện tại', @@ -91,9 +90,7 @@ const translation = { addParallelNode: 'Thêm nút song song', parallel: 'SONG SONG', branch: 'NHÁNH', - featuresDocLink: 'Tìm hiểu thêm', fileUploadTip: 'Các tính năng tải lên hình ảnh đã được nâng cấp để tải tệp lên.', - featuresDescription: 'Nâng cao trải nghiệm người dùng ứng dụng web', ImageUploadLegacyTip: 'Bây giờ bạn có thể tạo các biến loại tệp trong biểu mẫu bắt đầu. Chúng tôi sẽ không còn hỗ trợ tính năng tải lên hình ảnh trong tương lai.', importWarning: 'Thận trọng', importWarningDetails: 'Sự khác biệt về phiên bản DSL có thể ảnh hưởng đến một số tính năng nhất định', @@ -111,10 +108,11 @@ const translation = { exportJPEG: 'Xuất dưới dạng JPEG', needAnswerNode: 'Nút Trả lời phải được thêm vào', addBlock: 'Thêm Node', - needEndNode: 'Nút Kết thúc phải được thêm vào', + needOutputNode: 'Phải thêm nút Đầu ra', tagBound: 'Số lượng ứng dụng sử dụng thẻ này', currentWorkflow: 'Quy trình làm việc hiện tại', currentView: 'Hiện tại View', + moreActions: 'Hành động khác', }, env: { envPanelTitle: 'Biến Môi Trường', @@ -139,6 +137,19 @@ const translation = { export: 'Xuất DSL với giá trị bí mật', }, }, + globalVar: { + title: 'Biến hệ thống', + description: 'Biến hệ thống là biến toàn cục mà bất kỳ nút nào cũng có thể tham chiếu mà không cần nối dây khi kiểu dữ liệu phù hợp, chẳng hạn như ID người dùng cuối và ID quy trình làm việc.', + fieldsDescription: { + conversationId: 'ID cuộc trò chuyện', + dialogCount: 'Số lần trò chuyện', + userId: 'ID người dùng', + triggerTimestamp: 'Dấu thời gian ứng dụng bắt đầu chạy', + appId: 'ID ứng dụng', + workflowId: 'ID quy trình làm việc', + workflowRunId: 'ID lần chạy quy trình làm việc', + }, + }, chatVariable: { panelTitle: 'Biến Hội Thoại', panelDescription: 'Biến Hội Thoại được sử dụng để lưu trữ thông tin tương tác mà LLM cần ghi nhớ, bao gồm lịch sử hội thoại, tệp đã tải lên, tùy chọn người dùng. Chúng có thể đọc và ghi được.', @@ -242,7 +253,7 @@ const translation = { }, blocks: { 'start': 'Bắt đầu', - 'end': 'Kết thúc', + 'end': 'Đầu ra', 'answer': 'Trả lời', 'llm': 'LLM', 'knowledge-retrieval': 'Truy xuất kiến thức', @@ -268,7 +279,7 @@ const translation = { }, blocksAbout: { 'start': 'Định nghĩa các tham số ban đầu để khởi chạy quy trình làm việc', - 'end': 'Định nghĩa kết thúc và loại kết quả của quy trình làm việc', + 'end': 'Định nghĩa đầu ra và loại kết quả của quy trình làm việc', 'answer': 'Định nghĩa nội dung trả lời của cuộc trò chuyện', 'llm': 'Gọi các mô hình ngôn ngữ lớn để trả lời câu hỏi hoặc xử lý ngôn ngữ tự nhiên', 'knowledge-retrieval': 'Cho phép truy vấn nội dung văn bản liên quan đến câu hỏi của người dùng từ cơ sở kiến thức', @@ -311,7 +322,7 @@ const translation = { }, panel: { userInputField: 'Trường đầu vào của người dùng', - helpLink: 'Liên kết trợ giúp', + helpLink: 'Trung tâm trợ giúp', about: 'Giới thiệu', createdBy: 'Tạo bởi ', nextStep: 'Bước tiếp theo', @@ -321,13 +332,13 @@ const translation = { checklistResolved: 'Tất cả các vấn đề đã được giải quyết', change: 'Thay đổi', optional: '(tùy chọn)', - moveToThisNode: 'Di chuyển đến nút này', changeBlock: 'Thay đổi Node', selectNextStep: 'Chọn bước tiếp theo', organizeBlocks: 'Tổ chức các nút', addNextStep: 'Thêm bước tiếp theo trong quy trình này', maximize: 'Tối đa hóa Canvas', minimize: 'Thoát chế độ toàn màn hình', + scrollToSelectedNode: 'Cuộn đến nút đã chọn', optional_and_hidden: '(tùy chọn & ẩn)', }, nodes: { diff --git a/web/i18n/zh-Hans/app-api.ts b/web/i18n/zh-Hans/app-api.ts index 70b8413244..4fe97f8231 100644 --- a/web/i18n/zh-Hans/app-api.ts +++ b/web/i18n/zh-Hans/app-api.ts @@ -31,7 +31,7 @@ const translation = { }, completionMode: { title: '文本生成型应用 API', - info: '可用于生成高质量文本的应用,例如生成文章、摘要、翻译等,通过调用 completion-messages 接口,发送用户输入得到生成文本结果。用于生成文本的模型参数和提示词模版取决于开发者在 Dify 提示词编排页的设置。', + info: '可用于生成高质量文本的应用,例如生成文章、摘要、翻译等,通过调用 completion-messages 接口,发送用户输入得到生成文本结果。用于生成文本的模型参数和提示词模板取决于开发者在 Dify 提示词编排页的设置。', createCompletionApi: '创建文本补全消息', createCompletionApiTip: '创建文本补全消息,支持一问一答模式。', inputsTips: '(选填)以键值对方式提供用户输入字段,与提示词编排中的变量对应。Key 为变量名称,Value 是参数值。如果字段类型为 Select,传入的 Value 需为预设选项之一。', diff --git a/web/i18n/zh-Hans/app-debug.ts b/web/i18n/zh-Hans/app-debug.ts index 0cb3bf6b5a..a0759e9b8c 100644 --- a/web/i18n/zh-Hans/app-debug.ts +++ b/web/i18n/zh-Hans/app-debug.ts @@ -342,7 +342,6 @@ const translation = { variableTable: { key: '变量 Key', name: '字段名称', - optional: '可选', type: '类型', action: '操作', typeString: '文本', diff --git a/web/i18n/zh-Hans/app-log.ts b/web/i18n/zh-Hans/app-log.ts index 26c5c915c5..629d584642 100644 --- a/web/i18n/zh-Hans/app-log.ts +++ b/web/i18n/zh-Hans/app-log.ts @@ -20,6 +20,7 @@ const translation = { tokens: 'TOKENS', user: '用户或账户', version: '版本', + triggered_from: '触发方式', }, pagination: { previous: '上一页', @@ -30,7 +31,7 @@ const translation = { noOutput: '无输出', element: { title: '这里有人吗', - content: '在这里观测和标注最终用户和 AI 应用程序之间的交互,以不断提高 AI 的准确性。您可以<testLink>试试</testLink> web app 或<shareLink>分享</shareLink>出去,然后返回此页面。', + content: '在这里观测和标注最终用户和 AI 应用程序之间的交互,以不断提高 AI 的准确性。您可以尝试<shareLink>分享</shareLink>或<testLink>测试</testLink>此Web应用程序,然后返回此页面。', }, }, }, @@ -59,6 +60,7 @@ const translation = { period: { today: '今天', last7days: '过去 7 天', + last30days: '过去 30 天', last4weeks: '过去 4 周', last3months: '过去 3 月', last12months: '过去 12 月', @@ -66,6 +68,7 @@ const translation = { quarterToDate: '本季度至今', yearToDate: '本年至今', allTime: '所有时间', + custom: '自定义', }, annotation: { all: '全部', @@ -95,6 +98,15 @@ const translation = { iteration: '迭代', finalProcessing: '最终处理', }, + triggerBy: { + debugging: '调试', + appRun: '网页应用', + webhook: 'Webhook', + schedule: '定时任务', + plugin: '插件', + ragPipelineRun: 'RAG 流水线', + ragPipelineDebugging: 'RAG 调试', + }, } export default translation diff --git a/web/i18n/zh-Hans/app-overview.ts b/web/i18n/zh-Hans/app-overview.ts index a41a86975a..730240b9f7 100644 --- a/web/i18n/zh-Hans/app-overview.ts +++ b/web/i18n/zh-Hans/app-overview.ts @@ -30,6 +30,7 @@ const translation = { overview: { title: '概览', appInfo: { + title: 'Web App', explanation: '开箱即用的 AI web app', accessibleAddress: '公开访问 URL', preview: '预览', @@ -37,6 +38,10 @@ const translation = { regenerate: '重新生成', regenerateNotice: '您是否要重新生成公开访问 URL?', preUseReminder: '使用前请先打开开关', + enableTooltip: { + description: '要启用此功能,请在画布中添加用户输入节点。(草稿中可能已存在,发布后生效)', + learnMore: '了解更多', + }, settings: { entry: '设置', title: 'web app 设置', @@ -121,6 +126,14 @@ const translation = { accessibleAddress: 'API 访问凭据', doc: '查阅 API 文档', }, + triggerInfo: { + title: '触发器', + explanation: '工作流触发器管理', + triggersAdded: '已添加 {{count}} 个触发器', + noTriggerAdded: '未添加触发器', + triggerStatusDescription: '触发器节点状态显示在这里。(草稿中可能已存在,发布后生效)', + learnAboutTriggers: '了解触发器', + }, status: { running: '运行中', disable: '已停用', diff --git a/web/i18n/zh-Hans/app.ts b/web/i18n/zh-Hans/app.ts index cdc2ba1ad7..53b4ef784a 100644 --- a/web/i18n/zh-Hans/app.ts +++ b/web/i18n/zh-Hans/app.ts @@ -38,7 +38,7 @@ const translation = { newApp: { learnMore: '了解更多', startFromBlank: '创建空白应用', - startFromTemplate: '从应用模版创建', + startFromTemplate: '从应用模板创建', foundResult: '{{count}} 个结果', foundResults: '{{count}} 个结果', noAppsFound: '未找到应用', @@ -80,7 +80,7 @@ const translation = { Confirm: '确认', import: '导入', nameNotEmpty: '名称不能为空', - appTemplateNotSelected: '请选择应用模版', + appTemplateNotSelected: '请选择应用模板', appTypeRequired: '请选择应用类型', appCreated: '应用已创建', caution: '注意', @@ -95,7 +95,7 @@ const translation = { }, newAppFromTemplate: { byCategories: '分类', - searchAllTemplate: '搜索所有模版...', + searchAllTemplate: '搜索所有模板...', sidebar: { Recommended: '推荐', Agent: 'Agent', @@ -253,6 +253,8 @@ const translation = { notSetDesc: '当前任何人都无法访问 Web 应用。请设置访问权限。', }, noAccessPermission: '没有权限访问 web 应用', + noUserInputNode: '缺少用户输入节点', + notPublishedYet: '应用暂未发布', maxActiveRequests: '最大活跃请求数', maxActiveRequestsPlaceholder: '0 表示不限制', maxActiveRequestsTip: '当前应用的最大活跃请求数(0 表示不限制)', diff --git a/web/i18n/zh-Hans/billing.ts b/web/i18n/zh-Hans/billing.ts index 00a7dd909a..3c50abd01f 100644 --- a/web/i18n/zh-Hans/billing.ts +++ b/web/i18n/zh-Hans/billing.ts @@ -7,6 +7,8 @@ const translation = { documentsUploadQuota: '文档上传配额', vectorSpace: '知识库数据存储空间', vectorSpaceTooltip: '采用高质量索引模式的文档会消耗知识数据存储资源。当知识数据存储达到限制时,将不会上传新文档。', + triggerEvents: '触发事件', + perMonth: '每月', }, upgradeBtn: { plain: '查看套餐', @@ -61,7 +63,7 @@ const translation = { documentsRequestQuota: '{{count,number}}/分钟 知识库请求频率限制', documentsRequestQuotaTooltip: '指每分钟内,一个空间在知识库中可执行的操作总数,包括数据集的创建、删除、更新,文档的上传、修改、归档,以及知识库查询等,用于评估知识库请求的性能。例如,Sandbox 用户在 1 分钟内连续执行 10 次命中测试,其工作区将在接下来的 1 分钟内无法继续执行以下操作:数据集的创建、删除、更新,文档的上传、修改等操作。', apiRateLimit: 'API 请求频率限制', - apiRateLimitUnit: '{{count,number}} 次/天', + apiRateLimitUnit: '{{count,number}} 次/月', unlimitedApiRate: 'API 请求频率无限制', apiRateLimitTooltip: 'API 请求频率限制涵盖所有通过 Dify API 发起的调用,例如文本生成、聊天对话、工作流执行和文档处理等。', documentProcessingPriority: '文档处理', @@ -71,6 +73,20 @@ const translation = { 'priority': '优先', 'top-priority': '最高优先级', }, + triggerEvents: { + sandbox: '{{count,number}} 触发事件', + professional: '{{count,number}} 触发事件/月', + unlimited: '无限制触发事件', + }, + workflowExecution: { + standard: '标准工作流执行', + faster: '更快的工作流执行', + priority: '优先工作流执行', + }, + startNodes: { + limited: '每个工作流最多 {{count}} 个起始节点', + unlimited: '每个工作流无限制起始节点', + }, logsHistory: '{{days}}日志历史', customTools: '自定义工具', unavailable: '不可用', diff --git a/web/i18n/zh-Hans/common.ts b/web/i18n/zh-Hans/common.ts index f1a089aa57..8ad5f67d1b 100644 --- a/web/i18n/zh-Hans/common.ts +++ b/web/i18n/zh-Hans/common.ts @@ -29,6 +29,11 @@ const translation = { refresh: '重新开始', reset: '重置', search: '搜索', + noSearchResults: '没有找到{{content}}', + resetKeywords: '重置关键词', + selectCount: '已选择 {{count}} 项', + searchCount: '找到 {{count}} 个 {{content}}', + noSearchCount: '0 个 {{content}}', change: '更改', remove: '移除', send: '发送', @@ -71,6 +76,7 @@ const translation = { more: '更多', selectAll: '全选', deSelectAll: '取消全选', + now: '现在', }, errorMsg: { fieldRequired: '{{field}} 为必填项', @@ -79,7 +85,9 @@ const translation = { placeholder: { input: '请输入', select: '请选择', + search: '搜索...', }, + noData: '暂无数据', label: { optional: '(可选)', }, @@ -173,10 +181,10 @@ const translation = { emailSupport: '邮件支持', workspace: '工作空间', createWorkspace: '创建工作空间', - helpCenter: '帮助文档', + helpCenter: '查看帮助文档', support: '支持', compliance: '合规', - communityFeedback: '用户反馈', + forum: '论坛', roadmap: '路线图', github: 'GitHub', community: '社区', @@ -729,6 +737,7 @@ const translation = { uploadFromComputerLimit: '上传 {{type}} 不能超过 {{size}}', pasteFileLinkInvalid: '文件链接无效', fileExtensionNotSupport: '文件类型不支持', + fileExtensionBlocked: '出于安全考虑,该文件类型已被禁止上传', }, tag: { placeholder: '全部标签', @@ -769,6 +778,12 @@ const translation = { title: '提供反馈', placeholder: '请描述发生了什么问题或我们可以如何改进...', }, + dynamicSelect: { + error: '加载选项失败', + noData: '没有可用的选项', + loading: '加载选项...', + selected: '已选择 {{count}} 项', + }, } export default translation diff --git a/web/i18n/zh-Hans/dataset-creation.ts b/web/i18n/zh-Hans/dataset-creation.ts index 5b1ff2435c..f780269914 100644 --- a/web/i18n/zh-Hans/dataset-creation.ts +++ b/web/i18n/zh-Hans/dataset-creation.ts @@ -38,7 +38,7 @@ const translation = { button: '拖拽文件或文件夹至此,或者', buttonSingleFile: '拖拽文件至此,或者', browse: '选择文件', - tip: '已支持 {{supportTypes}},每批最多 {{batchCount}} 个文件,每个文件不超过 {{size}} MB。', + tip: '已支持 {{supportTypes}},每批最多 {{batchCount}} 个文件,每个文件不超过 {{size}} MB ,总数不超过 {{totalCount}} 个文件。', validation: { typeError: '文件类型不支持', size: '文件太大了,不能超过 {{size}}MB', diff --git a/web/i18n/zh-Hans/login.ts b/web/i18n/zh-Hans/login.ts index 82c6b355f9..13a75eaaaa 100644 --- a/web/i18n/zh-Hans/login.ts +++ b/web/i18n/zh-Hans/login.ts @@ -1,5 +1,6 @@ const translation = { pageTitle: '登录 Dify', + pageTitleForE: '嗨,近来可好', welcome: '👋 欢迎!请登录以开始使用。', email: '邮箱', emailPlaceholder: '输入邮箱地址', diff --git a/web/i18n/zh-Hans/pipeline.ts b/web/i18n/zh-Hans/pipeline.ts index 3c3a7a6506..1ae087fcfd 100644 --- a/web/i18n/zh-Hans/pipeline.ts +++ b/web/i18n/zh-Hans/pipeline.ts @@ -33,7 +33,7 @@ const translation = { }, ragToolSuggestions: { title: 'RAG 工具推荐', - noRecommendationPluginsInstalled: '暂无已安装的推荐插件,更多插件请在 <CustomLink>Marketplace</CustomLink> 中查找', + noRecommendationPlugins: '暂无推荐插件,更多插件请在 <CustomLink>Marketplace</CustomLink> 中查找', }, } diff --git a/web/i18n/zh-Hans/plugin-trigger.ts b/web/i18n/zh-Hans/plugin-trigger.ts new file mode 100644 index 0000000000..304cdd47bd --- /dev/null +++ b/web/i18n/zh-Hans/plugin-trigger.ts @@ -0,0 +1,186 @@ +const translation = { + subscription: { + title: '订阅', + listNum: '{{num}} 个订阅', + empty: { + title: '暂无订阅', + button: '新建订阅', + }, + createButton: { + oauth: '通过 OAuth 新建订阅', + apiKey: '通过 API Key 新建订阅', + manual: '粘贴 URL 以创建新订阅', + }, + createSuccess: '订阅创建成功', + createFailed: '订阅创建失败', + maxCount: '最多 {{num}} 个订阅', + selectPlaceholder: '选择订阅', + noSubscriptionSelected: '未选择订阅', + subscriptionRemoved: '订阅已移除', + list: { + title: '订阅列表', + addButton: '添加', + tip: '通过订阅接收事件', + item: { + enabled: '已启用', + disabled: '已禁用', + credentialType: { + api_key: 'API密钥', + oauth2: 'OAuth', + unauthorized: '手动', + }, + actions: { + delete: '删除', + deleteConfirm: { + title: '删除 {{name}}?', + success: '订阅 {{name}} 删除成功', + error: '订阅 {{name}} 删除失败', + content: '删除后,该订阅将无法恢复,请确认。', + contentWithApps: '该订阅正在被 {{count}} 个应用使用,删除它将导致这些应用停止接收订阅事件。', + confirm: '确认删除', + cancel: '取消', + confirmInputWarning: '请输入正确的名称确认。', + confirmInputPlaceholder: '输入 "{{name}}" 确认', + confirmInputTip: '请输入 “{{name}}” 确认:', + }, + }, + status: { + active: '活跃', + inactive: '非活跃', + }, + usedByNum: '被 {{num}} 个工作流使用', + noUsed: '未被工作流使用', + }, + }, + addType: { + title: '添加订阅', + description: '选择创建触发器订阅的方式', + options: { + apikey: { + title: '通过 API Key 创建', + description: '使用 API 凭据自动创建订阅', + }, + oauth: { + title: '通过 OAuth 创建', + description: '与第三方平台授权以创建订阅', + clientSettings: 'OAuth 客户端设置', + clientTitle: 'OAuth 客户端', + default: '默认', + custom: '自定义', + }, + manual: { + title: '手动设置', + description: '粘贴 URL 以创建新订阅', + tip: '手动配置 URL 到第三方平台', + }, + }, + }, + }, + modal: { + steps: { + verify: '验证', + configuration: '配置', + }, + common: { + cancel: '取消', + back: '返回', + next: '下一步', + create: '创建', + verify: '验证', + authorize: '授权', + creating: '创建中...', + verifying: '验证中...', + authorizing: '授权中...', + }, + oauthRedirectInfo: '由于未找到此工具提供方的系统客户端密钥,需要手动设置,对于 redirect_uri,请使用', + apiKey: { + title: '通过 API Key 创建', + verify: { + title: '验证凭据', + description: '请提供您的 API 凭据以验证访问权限', + error: '凭据验证失败,请检查您的 API 密钥。', + success: '凭据验证成功', + }, + configuration: { + title: '配置订阅', + description: '设置您的订阅参数', + }, + }, + oauth: { + title: '通过 OAuth 创建', + authorization: { + title: 'OAuth 授权', + description: '授权 Dify 访问您的账户', + redirectUrl: '重定向 URL', + redirectUrlHelp: '在您的 OAuth 应用配置中使用此 URL', + authorizeButton: '使用 {{provider}} 授权', + waitingAuth: '等待授权中...', + authSuccess: '授权成功', + authFailed: '获取 OAuth 授权信息失败', + waitingJump: '已授权,待跳转', + }, + configuration: { + title: '配置订阅', + description: '授权完成后设置您的订阅参数', + success: 'OAuth 配置成功', + failed: 'OAuth 配置失败', + }, + remove: { + success: 'OAuth 移除成功', + failed: 'OAuth 移除失败', + }, + save: { + success: 'OAuth 配置保存成功', + }, + }, + manual: { + title: '手动设置', + description: '手动配置您的 Webhook 订阅', + logs: { + title: '请求日志', + request: '请求', + loading: '等待 {{pluginName}} 的请求...', + }, + }, + form: { + subscriptionName: { + label: '订阅名称', + placeholder: '输入订阅名称', + required: '订阅名称为必填项', + }, + callbackUrl: { + label: '回调 URL', + description: '此 URL 将接收Webhook事件', + tooltip: '填写能被触发器提供方访问的公网地址,用于接收回调请求。', + placeholder: '生成中...', + privateAddressWarning: '此 URL 似乎是一个内部地址,可能会导致 Webhook 请求失败。', + }, + }, + errors: { + createFailed: '创建订阅失败', + verifyFailed: '验证凭据失败', + authFailed: '授权失败', + networkError: '网络错误,请重试', + }, + }, + events: { + title: '可用事件', + description: '此触发器插件可以订阅的事件', + empty: '没有可用事件', + event: '事件', + events: '事件', + actionNum: '包含 {{num}} 个 {{event}}', + item: { + parameters: '{{count}}个参数', + noParameters: '暂无参数', + }, + output: '输出', + }, + node: { + status: { + warning: '未连接', + }, + }, +} + +export default translation diff --git a/web/i18n/zh-Hans/plugin.ts b/web/i18n/zh-Hans/plugin.ts index adda0a3b8a..d648bccb85 100644 --- a/web/i18n/zh-Hans/plugin.ts +++ b/web/i18n/zh-Hans/plugin.ts @@ -8,6 +8,7 @@ const translation = { tools: '工具', agents: 'Agent 策略', extensions: '扩展', + triggers: '触发器', bundles: '插件集', datasources: '数据源', }, @@ -16,6 +17,7 @@ const translation = { tool: '工具', agent: 'Agent 策略', extension: '扩展', + trigger: '触发器', bundle: '插件集', datasource: '数据源', }, @@ -62,6 +64,7 @@ const translation = { checkUpdate: '检查更新', viewDetail: '查看详情', remove: '移除', + back: '返回', }, actionNum: '包含 {{num}} 个 {{action}}', strategyNum: '包含 {{num}} 个 {{strategy}}', @@ -77,7 +80,7 @@ const translation = { endpointModalDesc: '完成配置后可使用插件 API 端点提供的功能', serviceOk: '服务正常', disabled: '停用', - modelNum: '{{num}} 模型已包含', + modelNum: '包含 {{num}} 个模型', toolSelector: { title: '添加工具', toolSetting: '工具设置', @@ -306,6 +309,12 @@ const translation = { connectedWorkspace: '已连接的工作区', emptyAuth: '请配置凭据', }, + readmeInfo: { + title: 'README', + needHelpCheckReadme: '需要帮助?查看 README。', + noReadmeAvailable: 'README 文档不可用', + failedToFetch: '获取 README 文档失败', + }, } export default translation diff --git a/web/i18n/zh-Hans/tools.ts b/web/i18n/zh-Hans/tools.ts index 15b1c7f592..cab4b22164 100644 --- a/web/i18n/zh-Hans/tools.ts +++ b/web/i18n/zh-Hans/tools.ts @@ -3,7 +3,6 @@ const translation = { createCustomTool: '创建自定义工具', customToolTip: '了解更多关于 Dify 自定义工具的信息', type: { - all: '全部', builtIn: '工具', custom: '自定义', workflow: '工作流', @@ -21,13 +20,10 @@ const translation = { setupModalTitleDescription: '配置凭据后,工作区中的所有成员都可以在编排应用程序时使用此工具。', }, includeToolNum: '包含 {{num}} 个 {{action}}', - addTool: '添加工具', addToolModal: { type: '类型', category: '类别', - add: '添加', added: '已添加', - manageInTools: '去工具列表管理', custom: { title: '没有可用的自定义工具', tip: '创建自定义工具', @@ -65,7 +61,7 @@ const translation = { exampleOptions: { json: '天气 (JSON)', yaml: '宠物商店 (YAML)', - blankTemplate: '空白模版', + blankTemplate: '空白模板', }, availableTools: { title: '可用工具', @@ -203,6 +199,12 @@ const translation = { timeout: '超时时间', sseReadTimeout: 'SSE 读取超时时间', timeoutPlaceholder: '30', + authentication: '认证', + useDynamicClientRegistration: '使用动态客户端注册', + clientID: '客户端 ID', + clientSecret: '客户端密钥', + clientSecretPlaceholder: '客户端密钥', + configurations: '配置', }, delete: '删除 MCP 服务', deleteConfirmTitle: '你想要删除 {{mcp}} 吗?', diff --git a/web/i18n/zh-Hans/workflow.ts b/web/i18n/zh-Hans/workflow.ts index c5c72eb712..18e76caa64 100644 --- a/web/i18n/zh-Hans/workflow.ts +++ b/web/i18n/zh-Hans/workflow.ts @@ -9,8 +9,10 @@ const translation = { publish: '发布', update: '更新', publishUpdate: '发布更新', - run: '运行', + run: '测试运行', running: '运行中', + chooseStartNodeToRun: '选择启动节点进行运行', + runAllTriggers: '运行所有触发器', inRunMode: '在运行模式中', inPreview: '预览中', inPreviewMode: '预览中', @@ -46,7 +48,8 @@ const translation = { needConnectTip: '此节点尚未连接到其他节点', maxTreeDepth: '每个分支最大限制 {{depth}} 个节点', needAdd: '必须添加{{node}}节点', - needEndNode: '必须添加结束节点', + needOutputNode: '必须添加输出节点', + needStartNode: '必须添加至少一个开始节点', needAnswerNode: '必须添加直接回复节点', workflowProcess: '工作流', notRunning: '尚未运行', @@ -76,12 +79,14 @@ const translation = { exportSVG: '导出为 SVG', currentView: '当前视图', currentWorkflow: '整个工作流', + moreActions: '更多操作', model: '模型', workflowAsTool: '发布为工具', configureRequired: '需要进行配置', configure: '配置', manageInTools: '访问工具页', workflowAsToolTip: '工作流更新后需要重新配置工具参数', + workflowAsToolDisabledHint: '请先发布最新的工作流,并确保已连接的 User Input 节点后再配置为工具。', viewDetailInTracingPanel: '查看详细信息', syncingData: '同步数据中,只需几秒钟。', importDSL: '导入 DSL', @@ -140,6 +145,19 @@ const translation = { export: '导出包含 Secret 值的 DSL', }, }, + globalVar: { + title: '系统变量', + description: '系统变量是全局变量,在类型匹配时无需连线即可被任意节点引用,例如终端用户 ID 和工作流 ID。', + fieldsDescription: { + conversationId: '会话 ID', + dialogCount: '会话次数', + userId: '用户 ID', + triggerTimestamp: '应用开始运行的时间戳', + appId: '应用 ID', + workflowId: '工作流 ID', + workflowRunId: '工作流运行 ID', + }, + }, sidebar: { exportWarning: '导出当前已保存版本', exportWarningDesc: '这将导出您工作流的当前已保存版本。如果您在编辑器中有未保存的更改,请先使用工作流画布中的导出选项保存它们。', @@ -213,6 +231,16 @@ const translation = { invalidVariable: '无效的变量', noValidTool: '{{field}} 无可用工具', toolParameterRequired: '{{field}}: 参数 [{{param}}] 不能为空', + startNodeRequired: '请先添加开始节点,然后再{{operation}}', + }, + error: { + startNodeRequired: '请先添加开始节点,然后再{{operation}}', + operations: { + connectingNodes: '连接节点', + addingNodes: '添加节点', + modifyingWorkflow: '修改工作流', + updatingWorkflow: '更新工作流', + }, }, singleRun: { testRun: '测试运行', @@ -229,6 +257,8 @@ const translation = { 'searchBlock': '搜索节点', 'blocks': '节点', 'searchTool': '搜索工具', + 'searchTrigger': '搜索触发器...', + 'allTriggers': '全部触发器', 'tools': '工具', 'allTool': '全部', 'plugin': '插件', @@ -239,15 +269,29 @@ const translation = { 'transform': '转换', 'utilities': '工具', 'noResult': '未找到匹配项', + 'noPluginsFound': '未找到插件', + 'requestToCommunity': '向社区反馈', 'agent': 'Agent 策略', 'allAdded': '已添加全部', 'addAll': '添加全部', 'sources': '数据源', 'searchDataSource': '搜索数据源', + 'start': '开始', + 'featuredTools': '精选推荐', + 'showMoreFeatured': '查看更多', + 'showLessFeatured': '收起', + 'installed': '已安装', + 'pluginByAuthor': '来自 {{author}}', + 'usePlugin': '选择工具', + 'hideActions': '收起工具', + 'noFeaturedPlugins': '前往插件市场查看更多工具', + 'noFeaturedTriggers': '前往插件市场查看更多触发器', + 'startDisabledTip': '触发节点与用户输入节点互斥。', }, blocks: { - 'start': '开始', - 'end': '结束', + 'start': '用户输入', + 'originalStartNode': '原始开始节点', + 'end': '输出', 'answer': '直接回复', 'llm': 'LLM', 'knowledge-retrieval': '知识检索', @@ -270,10 +314,14 @@ const translation = { 'loop-end': '退出循环', 'knowledge-index': '知识库', 'datasource': '数据源', + 'trigger-webhook': 'Webhook 触发器', + 'trigger-schedule': '定时触发器', + 'trigger-plugin': '插件触发器', }, + customWebhook: '自定义 Webhook', blocksAbout: { 'start': '定义一个 workflow 流程启动的初始参数', - 'end': '定义一个 workflow 流程的结束和结果类型', + 'end': '定义一个 workflow 流程的输出和结果类型', 'answer': '定义一个聊天对话的回复内容', 'llm': '调用大语言模型回答问题或者对自然语言进行处理', 'knowledge-retrieval': '允许你从知识库中查询与用户问题相关的文本内容', @@ -294,7 +342,11 @@ const translation = { 'agent': '调用大型语言模型回答问题或处理自然语言', 'knowledge-index': '知识库节点', 'datasource': '数据源节点', + 'trigger-webhook': 'Webhook 触发器接收来自第三方系统的 HTTP 推送以自动触发工作流。', + 'trigger-schedule': '基于时间的工作流触发器,按计划启动工作流', + 'trigger-plugin': '从外部平台事件启动工作流的第三方集成触发器', }, + difyTeam: 'Dify 团队', operator: { zoomIn: '放大', zoomOut: '缩小', @@ -324,7 +376,7 @@ const translation = { panel: { userInputField: '用户输入字段', changeBlock: '更改节点', - helpLink: '帮助链接', + helpLink: '查看帮助文档', about: '关于', createdBy: '作者', nextStep: '下一步', @@ -334,12 +386,14 @@ const translation = { checklist: '检查清单', checklistTip: '发布前确保所有问题均已解决', checklistResolved: '所有问题均已解决', + goTo: '转到', + startNode: '开始节点', organizeBlocks: '整理节点', change: '更改', optional: '(选填)', - moveToThisNode: '定位至此节点', maximize: '最大化画布', minimize: '退出最大化', + scrollToSelectedNode: '滚动至选中节点', optional_and_hidden: '(选填 & 隐藏)', }, nodes: { @@ -788,6 +842,8 @@ const translation = { removeAbnormalOutput: '移除错误输出', }, answerNodeWarningDesc: '并行模式警告:在迭代中,回答节点、会话变量赋值和工具持久读/写操作可能会导致异常。', + flattenOutput: '扁平化输出', + flattenOutputDesc: '启用时,如果所有迭代输出都是数组,它们将被扁平化为单个数组。禁用时,输出将保持嵌套数组结构。', }, loop: { deleteTitle: '删除循环节点?', @@ -964,6 +1020,138 @@ const translation = { rerankingModelIsRequired: 'Reranking 模型是必需的', rerankingModelIsInvalid: '无效的 Reranking 模型', }, + triggerSchedule: { + frequency: { + label: '频率', + monthly: '每月', + daily: '每日', + hourly: '每小时', + weekly: '每周', + }, + title: '定时触发', + nodeTitle: '定时触发器', + useCronExpression: '使用 Cron 表达式', + selectFrequency: '选择频率', + nextExecutionTimes: '接下来 5 次执行时间', + hours: '小时', + minutes: '分钟', + onMinute: '分钟', + cronExpression: 'Cron 表达式', + weekdays: '星期', + executeNow: '立即执行', + frequencyLabel: '频率', + nextExecution: '下次执行', + time: '时间', + lastDay: '最后一天', + startTime: '开始时间', + selectDateTime: '选择日期和时间', + lastDayTooltip: '并非所有月份都有 31 天。使用"最后一天"选项来选择每个月的最后一天。', + nextExecutionTime: '下次执行时间', + useVisualPicker: '使用可视化配置', + days: '天', + notConfigured: '未配置', + mode: '模式', + timezone: '时区', + visualConfig: '可视化配置', + monthlyDay: '月份日期', + executionTime: '执行时间', + invalidTimezone: '无效的时区', + invalidCronExpression: '无效的 Cron 表达式', + noValidExecutionTime: '无法计算有效的执行时间', + executionTimeCalculationError: '执行时间计算失败', + invalidFrequency: '无效的频率', + invalidStartTime: '无效的开始时间', + startTimeMustBeFuture: '开始时间必须是将来的时间', + invalidTimeFormat: '无效的时间格式(预期格式:HH:MM AM/PM)', + invalidWeekday: '无效的工作日:{{weekday}}', + invalidMonthlyDay: '月份日期必须在 1-31 之间或为"last"', + invalidOnMinute: '分钟必须在 0-59 之间', + invalidExecutionTime: '无效的执行时间', + executionTimeMustBeFuture: '执行时间必须是将来的时间', + }, + triggerWebhook: { + configPlaceholder: 'Webhook 触发器配置将在此处实现', + title: 'Webhook 触发器', + nodeTitle: '🔗 Webhook 触发器', + webhookUrl: 'Webhook URL', + webhookUrlPlaceholder: '点击生成以创建 webhook URL', + generate: '生成', + copy: '复制', + test: '测试', + urlGenerated: 'Webhook URL 生成成功', + urlGenerationFailed: '生成 Webhook URL 失败', + urlCopied: 'URL 已复制到剪贴板', + method: '方法', + contentType: '内容类型', + queryParameters: '查询参数', + headerParameters: 'Header 参数', + requestBodyParameters: '请求体参数', + parameterName: '变量名', + varName: '变量名', + varType: '类型', + varNamePlaceholder: '输入变量名...', + headerName: '变量名', + required: '必填', + addParameter: '添加', + addHeader: '添加', + noParameters: '未配置任何参数', + noQueryParameters: '未配置查询参数', + noHeaders: '未配置 Header', + noBodyParameters: '未配置请求体参数', + debugUrlTitle: '测试运行时,请始终使用此URL', + debugUrlCopy: '点击复制', + debugUrlCopied: '已复制!', + errorHandling: '错误处理', + errorStrategy: '错误处理', + responseConfiguration: '响应', + asyncMode: '异步模式', + statusCode: '状态码', + responseBody: '响应体', + responseBodyPlaceholder: '在此输入您的响应体', + headers: 'Headers', + validation: { + webhookUrlRequired: '需要提供Webhook URL', + invalidParameterType: '参数"{{name}}"的参数类型"{{type}}"无效', + }, + }, + triggerPlugin: { + authorized: '已授权', + notConfigured: '未配置', + error: '错误', + configuration: '配置', + remove: '移除', + or: '或', + useOAuth: '使用 OAuth', + useApiKey: '使用 API Key', + authenticationFailed: '身份验证失败', + authenticationSuccess: '身份验证成功', + oauthConfigFailed: 'OAuth 配置失败', + configureOAuthClient: '配置 OAuth 客户端', + oauthClientDescription: '配置 OAuth 客户端凭据以启用身份验证', + oauthClientSaved: 'OAuth 客户端配置保存成功', + configureApiKey: '配置 API Key', + apiKeyDescription: '配置 API key 凭据进行身份验证', + apiKeyConfigured: 'API key 配置成功', + configurationFailed: '配置失败', + failedToStart: '启动身份验证流程失败', + credentialsVerified: '凭据验证成功', + credentialVerificationFailed: '凭据验证失败', + verifyAndContinue: '验证并继续', + configureParameters: '配置参数', + parametersDescription: '配置触发器参数和属性', + configurationComplete: '配置完成', + configurationCompleteDescription: '您的触发器已成功配置', + configurationCompleteMessage: '您的触发器配置已完成,现在可以使用了。', + parameters: '参数', + properties: '属性', + propertiesDescription: '此触发器的额外配置属性', + noConfigurationRequired: '此触发器不需要额外配置。', + subscriptionName: '订阅名称', + subscriptionNameDescription: '为此触发器订阅输入一个唯一名称', + subscriptionNamePlaceholder: '输入订阅名称...', + subscriptionNameRequired: '订阅名称是必需的', + subscriptionRequired: '需要配置订阅', + }, }, tracing: { stopBy: '由{{user}}终止', @@ -1025,6 +1213,18 @@ const translation = { view: '查看记录', edited: '已编辑', reset: '还原至上一次运行', + listening: { + title: '正在监听触发器事件…', + tip: '您现在可以向 HTTP {{nodeName}} 端点发送测试请求以模拟事件触发,或将其用作实时事件调试的回调 URL。所有输出都可以在变量检查器中直接查看。', + tipPlugin: '现在您可以在 {{- pluginName}} 中创建事件,并在变量检查器中查看这些事件的输出。', + tipSchedule: '正在监听计划触发器事件。\n下一次计划运行时间:{{nextTriggerTime}}', + tipFallback: '正在等待触发器事件,输出结果将在此显示。', + defaultNodeName: '此触发器', + defaultPluginName: '此插件触发器', + defaultScheduleTime: '未设置', + selectedTriggers: '所选触发器', + stopButton: '停止', + }, trigger: { normal: '变量检查', running: '缓存中', @@ -1050,6 +1250,30 @@ const translation = { noDependents: '无被依赖', }, }, + triggerStatus: { + enabled: '触发器', + disabled: '触发器 • 已禁用', + }, + entryNodeStatus: { + enabled: '开始', + disabled: '开始 • 已禁用', + }, + onboarding: { + title: '选择开始节点来开始', + description: '不同的开始节点具有不同的功能。不用担心,您随时可以更改它们。', + userInputFull: '用户输入(原始开始节点)', + userInputDescription: '允许设置用户输入变量的开始节点,具有Web应用程序、服务API、MCP服务器和工作流即工具功能。', + trigger: '触发器', + triggerDescription: '触发器可以作为工作流的开始节点,例如定时任务、自定义webhook或与其他应用程序的集成。', + back: '返回', + learnMore: '了解更多', + aboutStartNode: '关于开始节点。', + escTip: { + press: '按', + key: 'esc', + toDismiss: '键关闭', + }, + }, } export default translation diff --git a/web/i18n/zh-Hant/app-annotation.ts b/web/i18n/zh-Hant/app-annotation.ts index a86804afdd..c5dbf3c67a 100644 --- a/web/i18n/zh-Hant/app-annotation.ts +++ b/web/i18n/zh-Hant/app-annotation.ts @@ -4,7 +4,7 @@ const translation = { editBy: '{{author}}編輯的答案', noData: { title: '沒有標註', - description: '你可以在應用會話除錯中編輯標註,也可以在此批次匯入標註用於高質量回復。', + description: '你可以在應用會話除錯中編輯標註,也可以在此批次匯入標註用於高品質回復。', }, table: { header: { diff --git a/web/i18n/zh-Hant/app-api.ts b/web/i18n/zh-Hant/app-api.ts index db43cd8b77..30b80c53ba 100644 --- a/web/i18n/zh-Hant/app-api.ts +++ b/web/i18n/zh-Hant/app-api.ts @@ -30,7 +30,7 @@ const translation = { }, completionMode: { title: '文字生成型應用 API', - info: '可用於生成高質量文字的應用,例如生成文章、摘要、翻譯等,透過呼叫 completion-messages 介面,傳送使用者輸入得到生成文字結果。用於生成文字的模型引數和提示詞模版取決於開發者在 Dify 提示詞編排頁的設定。', + info: '可用於生成高品質文字的應用,例如生成文章、摘要、翻譯等,透過呼叫 completion-messages 介面,傳送使用者輸入得到生成文字結果。用於生成文字的模型引數和提示詞模版取決於開發者在 Dify 提示詞編排頁的設定。', createCompletionApi: '建立文字補全訊息', createCompletionApiTip: '建立文字補全訊息,支援一問一答模式。', inputsTips: '(選填)以鍵值對方式提供使用者輸入欄位,與提示詞編排中的變數對應。Key 為變數名稱,Value 是引數值。如果欄位型別為 Select,傳入的 Value 需為預設選項之一。', diff --git a/web/i18n/zh-Hant/app-debug.ts b/web/i18n/zh-Hant/app-debug.ts index 61b08a0354..ff3e131e89 100644 --- a/web/i18n/zh-Hant/app-debug.ts +++ b/web/i18n/zh-Hant/app-debug.ts @@ -255,7 +255,6 @@ const translation = { variableTable: { key: '變數 Key', name: '欄位名稱', - optional: '可選', type: '型別', action: '操作', typeString: '文字', @@ -390,7 +389,7 @@ const translation = { writeOpener: '編寫開場白', placeholder: '在這裡寫下你的開場白,你可以使用變數,嘗試輸入 {{variable}}。', openingQuestion: '開場問題', - openingQuestionPlaceholder: '可以使用變量,嘗試輸入 {{variable}}。', + openingQuestionPlaceholder: '可以使用變數,嘗試輸入 {{variable}}。', noDataPlaceHolder: '在對話型應用中,讓 AI 主動說第一段話可以拉近與使用者間的距離。', varTip: '你可以使用變數,試試輸入 {{variable}}', @@ -420,14 +419,14 @@ const translation = { }, result: '結果', datasetConfig: { - settingTitle: '召回設定', + settingTitle: '檢索設定', knowledgeTip: '點選“+”按鈕新增知識庫', retrieveOneWay: { - title: 'N 選 1 召回', + title: 'N 選 1 檢索', description: '根據使用者意圖和知識庫描述,由 Agent 自主判斷選擇最匹配的單個知識庫來查詢相關文字,適合知識庫區分度大且知識庫數量偏少的應用。', }, retrieveMultiWay: { - title: '多路召回', + title: '多路檢索', description: '根據使用者意圖同時匹配所有知識庫,從多路知識庫查詢相關文字片段,經過重排序步驟,從多路查詢結果中選擇匹配使用者問題的最佳結果,需配置 Rerank 模型 API。', }, rerankModelRequired: '請選擇 Rerank 模型', @@ -480,25 +479,25 @@ const translation = { }, }, codegen: { - resTitle: '生成的代碼', + resTitle: '生成的程式碼', apply: '應用', - overwriteConfirmMessage: '此作將覆蓋現有代碼。你想繼續嗎?', + overwriteConfirmMessage: '此作將覆蓋現有程式碼。你想繼續嗎?', instruction: '指示', - instructionPlaceholder: '輸入要生成的代碼的詳細說明。', + instructionPlaceholder: '輸入要生成的程式碼的詳細說明。', generate: '生成', - noDataLine2: '代碼預覽將在此處顯示。', + noDataLine2: '程式碼預覽將在此處顯示。', applyChanges: '應用更改', noDataLine1: '在左側描述您的用例,', - overwriteConfirmTitle: '覆蓋現有代碼?', - title: '代碼生成器', - generatedCodeTitle: '生成的代碼', - loading: '產生代碼...', - description: '代碼生成器使用配置的模型根據您的指令生成高質量的代碼。請提供清晰詳細的說明。', + overwriteConfirmTitle: '覆蓋現有程式碼?', + title: '程式碼生成器', + generatedCodeTitle: '生成的程式碼', + loading: '產生程式碼...', + description: '程式碼生成器使用配置的模型根據您的指令生成高品質的程式碼。請提供清晰詳細的說明。', }, generate: { template: { pythonDebugger: { - instruction: '可以根據您的指令生成和調試代碼的機器人', + instruction: '可以根據您的指令生成和調試程式碼的機器人', name: 'Python 調試器', }, translation: { @@ -561,7 +560,7 @@ const translation = { idealOutputPlaceholder: '描述您理想的回應格式、長度、語調和內容要求...', press: '新聞稿', newNoDataLine1: '在左側列寫入指示,然後點擊生成以查看回應.', - codeGenInstructionPlaceHolderLine: '反饋越詳細,例如輸入和輸出的數據類型以及變量的處理方式,代碼生成就會越準確。', + codeGenInstructionPlaceHolderLine: '反饋越詳細,例如輸入和輸出的數據類型以及變數的處理方式,程式碼生成就會越準確。', }, warningMessage: { timeoutExceeded: '由於超時,不顯示結果。請參閱日誌以收集完整結果。', diff --git a/web/i18n/zh-Hant/app-log.ts b/web/i18n/zh-Hant/app-log.ts index d24b4a1cce..7b3f33c3bc 100644 --- a/web/i18n/zh-Hant/app-log.ts +++ b/web/i18n/zh-Hant/app-log.ts @@ -29,7 +29,7 @@ const translation = { noOutput: '無輸出', element: { title: '這裡有人嗎', - content: '在這裡觀測和標註終端使用者和 AI 應用程式之間的互動,以不斷提高 AI 的準確性。您可以<testLink>試試</testLink> web app 或<shareLink>分享</shareLink>出去,然後返回此頁面。', + content: '在這裡觀測和標註終端使用者和 AI 應用程式之間的互動,以不斷提高 AI 的準確性。您可以嘗試<shareLink>分享</shareLink>或<testLink>測試</testLink>此Web應用程序,然後返回此頁面。', }, }, }, @@ -48,7 +48,7 @@ const translation = { dislike: '反對', addAnnotation: '標記改進回覆', editAnnotation: '編輯改進回覆', - annotationPlaceholder: '輸入你希望 AI 回覆的預期答案,這在今後可用於模型微調,持續改進文字生成質量。', + annotationPlaceholder: '輸入你希望 AI 回覆的預期答案,這在今後可用於模型微調,持續改進文字生成品質。', }, variables: '變數', uploadImages: '上傳的圖片', @@ -65,6 +65,8 @@ const translation = { quarterToDate: '本季度至今', yearToDate: '本年至今', allTime: '所有時間', + last30days: '過去30天', + custom: '自訂', }, annotation: { all: '全部', diff --git a/web/i18n/zh-Hant/billing.ts b/web/i18n/zh-Hant/billing.ts index 1b0b1f5e1f..38589179e7 100644 --- a/web/i18n/zh-Hant/billing.ts +++ b/web/i18n/zh-Hant/billing.ts @@ -56,7 +56,7 @@ const translation = { agentMode: '代理模式', workflow: '工作流', llmLoadingBalancing: 'LLM 負載均衡', - llmLoadingBalancingTooltip: '向模型添加多個 API 金鑰,從而有效地繞過 API 速率限制。', + llmLoadingBalancingTooltip: '向模型新增多個 API 金鑰,從而有效地繞過 API 速率限制。', }, comingSoon: '即將推出', member: '成員', @@ -68,13 +68,13 @@ const translation = { }, annotatedResponse: { title: '標註回覆數', - tooltip: '標註回覆功能透過人工編輯標註為應用提供了可定製的高質量問答回覆能力', + tooltip: '標註回覆功能透過人工編輯標註為應用提供了可定製的高品質問答回覆能力', }, ragAPIRequestTooltip: '指單獨呼叫 Dify 知識庫資料處理能力的 API。', receiptInfo: '只有團隊所有者和團隊管理員才能訂閱和檢視賬單資訊', annotationQuota: '註釋配額', self: '自我主持', - apiRateLimitUnit: '{{count,number}}/天', + apiRateLimitUnit: '{{count,number}}/月', freeTrialTipPrefix: '註冊並獲得一個', annualBilling: '年度計費', freeTrialTipSuffix: '無需信用卡', diff --git a/web/i18n/zh-Hant/common.ts b/web/i18n/zh-Hant/common.ts index c749966a71..51cdcf8a0b 100644 --- a/web/i18n/zh-Hant/common.ts +++ b/web/i18n/zh-Hant/common.ts @@ -160,7 +160,7 @@ const translation = { emailSupport: '電子郵件支援', workspace: '工作空間', createWorkspace: '建立工作空間', - helpCenter: '幫助文件', + helpCenter: '查看幫助文件', communityFeedback: '使用者反饋', roadmap: '路線圖', community: '社群', @@ -170,6 +170,7 @@ const translation = { github: 'GitHub', compliance: '合規', contactUs: '聯絡我們', + forum: '論壇', }, settings: { accountGroup: '賬戶', @@ -457,7 +458,7 @@ const translation = { defaultConfig: '默認配置', configLoadBalancing: '配置負載均衡', loadBalancingDescription: '使用多組憑證減輕壓力。', - addConfig: '添加配置', + addConfig: '新增配置', upgradeForLoadBalancing: '升級您的計劃以啟用 Load Balancing。', apiKey: 'API 金鑰', loadBalancing: '負載均衡', @@ -472,19 +473,19 @@ const translation = { installProvider: '安裝模型提供程式', toBeConfigured: '待配置', emptyProviderTitle: '未設置模型提供者', - configureTip: '設置 api-key 或添加要使用的模型', + configureTip: '設置 api-key 或新增要使用的模型', emptyProviderTip: '請先安裝模型提供者。', auth: { apiKeyModal: { - addModel: '添加模型', + addModel: '新增模型', title: 'API 金鑰授權配置', desc: '配置完憑證後,工作區內的所有成員在協調應用程式時都可以使用此模型。', }, authRemoved: '授權已被移除', configModel: '配置模型', - addApiKey: '添加 API 金鑰', - addCredential: '添加憑證', - addModelCredential: '添加模型憑證', + addApiKey: '新增 API 金鑰', + addCredential: '新增憑證', + addModelCredential: '新增模型憑證', modelCredentials: '模型憑證', providerManaged: '供應商管理', addNewModel: '新增模型', @@ -599,7 +600,7 @@ const translation = { }, datasetMenus: { documents: '文件', - hitTesting: '召回測試', + hitTesting: '檢索測試', settings: '設定', emptyTip: ' 知識庫尚未關聯,請前往應用程式或外掛完成關聯。', viewDoc: '檢視文件', @@ -632,9 +633,9 @@ const translation = { title: '引用', linkToDataset: '跳轉至知識庫', characters: '字元:', - hitCount: '召回次數:', + hitCount: '檢索次數:', vectorHash: '向量雜湊:', - hitScore: '召回得分:', + hitScore: '檢索得分:', }, inputPlaceholder: '與 {{botName}} 對話', thinking: '思維。。。', @@ -726,6 +727,7 @@ const translation = { uploadFromComputer: '本地上傳', fileExtensionNotSupport: '不支援檔擴展名', uploadFromComputerLimit: '上傳文件不能超過 {{size}}', + fileExtensionBlocked: '出於安全原因,此檔案類型被阻止', }, license: { expiring: '將在 1 天內過期', diff --git a/web/i18n/zh-Hant/dataset-creation.ts b/web/i18n/zh-Hant/dataset-creation.ts index c8b3277e96..2e551b58d9 100644 --- a/web/i18n/zh-Hant/dataset-creation.ts +++ b/web/i18n/zh-Hant/dataset-creation.ts @@ -101,7 +101,7 @@ const translation = { separatorPlaceholder: '例如換行符(\n)或特定的分隔符(如 "***")', maxLength: '分段最大長度', overlap: '分段重疊長度', - overlapTip: '設定分段之間的重疊長度可以保留分段之間的語義關係,提升召回效果。建議設定為最大分段長度的 10%-25%', + overlapTip: '設定分段之間的重疊長度可以保留分段之間的語義關係,提升檢索效果。建議設定為最大分段長度的 10%-25%', overlapCheck: '分段重疊長度不能大於分段最大長度', rules: '文字預處理規則', removeExtraSpaces: '替換掉連續的空格、換行符和製表符', @@ -110,7 +110,7 @@ const translation = { preview: '預覽', reset: '重置', indexMode: '索引方式', - qualified: '高質量', + qualified: '高品質', recommend: '推薦', qualifiedTip: '呼叫系統預設的嵌入介面進行處理,以在使用者查詢時提供更高的準確度', warning: '請先完成模型供應商的 API KEY 設定。.', @@ -135,7 +135,7 @@ const translation = { cancel: '取消', sideTipTitle: '為什麼要分段和預處理?', sideTipP1: '在處理文字資料時,分段和清洗是兩個重要的預處理步驟。', - sideTipP2: '分段的目的是將長文字拆分成較小的段落,以便模型更有效地處理和理解。這有助於提高模型生成的結果的質量和相關性。', + sideTipP2: '分段的目的是將長文字拆分成較小的段落,以便模型更有效地處理和理解。這有助於提高模型生成的結果的品質和相關性。', sideTipP3: '清洗則是對文字進行預處理,刪除不必要的字元、符號或格式,使知識庫更加乾淨、整潔,便於模型解析。', sideTipP4: '透過對知識庫進行適當的分段和清洗,可以提高模型在實際應用中的表現,從而為使用者提供更準確、更有價值的結果。', previewTitle: '分段預覽', diff --git a/web/i18n/zh-Hant/dataset-documents.ts b/web/i18n/zh-Hant/dataset-documents.ts index dfeab89d20..57a5eb1226 100644 --- a/web/i18n/zh-Hant/dataset-documents.ts +++ b/web/i18n/zh-Hant/dataset-documents.ts @@ -8,7 +8,7 @@ const translation = { header: { fileName: '檔名', words: '字元數', - hitCount: '召回次數', + hitCount: '檢索次數', uploadTime: '上傳時間', status: '狀態', action: '操作', @@ -79,7 +79,7 @@ const translation = { error: '匯入出錯', ok: '確定', }, - addUrl: '添加 URL', + addUrl: '新增 URL', learnMore: '瞭解更多資訊', }, metadata: { @@ -225,7 +225,7 @@ const translation = { segmentLength: '段落長度', avgParagraphLength: '平均段落長度', paragraphs: '段落數量', - hitCount: '召回次數', + hitCount: '檢索次數', embeddingTime: '嵌入時間', embeddedSpend: '嵌入花費', }, @@ -326,7 +326,7 @@ const translation = { segmentLength: '分段長度', textCleaning: '文字預定義與清洗', segments: '段落', - highQuality: '高質量模式', + highQuality: '高品質模式', economy: '經濟模式', estimate: '預估消耗', stop: '停止處理', @@ -345,7 +345,7 @@ const translation = { keywords: '關鍵詞', addKeyWord: '新增關鍵詞', keywordError: '關鍵詞最大長度為 20', - hitCount: '召回次數', + hitCount: '檢索次數', vectorHash: '向量雜湊:', questionPlaceholder: '在這裡新增問題', questionEmpty: '問題不能為空', @@ -357,20 +357,20 @@ const translation = { newQaSegment: '新問答分段', delete: '刪除這個分段?', characters_other: '字元', - addChunk: '添加數據塊', - addChildChunk: '添加子塊', - addAnother: '添加另一個', - childChunkAdded: '添加了 1 個子塊', + addChunk: '新增數據塊', + addChildChunk: '新增子塊', + addAnother: '新增另一個', + childChunkAdded: '新增了 1 個子塊', editParentChunk: '編輯父塊(Edit Parent Chunk)', editChildChunk: '編輯子塊', chunkDetail: '數據塊詳細資訊', regenerationConfirmTitle: '是否要重新生成子塊?', - regenerationConfirmMessage: '重新生成子數據塊將覆蓋當前子數據塊,包括已編輯的數據塊和新添加的數據塊。重新生成無法復原。', + regenerationConfirmMessage: '重新生成子數據塊將覆蓋當前子數據塊,包括已編輯的數據塊和新新增的數據塊。重新生成無法復原。', regeneratingTitle: '重新生成子塊', regenerationSuccessTitle: '再生完成', collapseChunks: '摺疊塊', expandChunks: '擴展塊', - chunkAdded: '添加了 1 個數據塊', + chunkAdded: '新增了 1 個數據塊', editedAt: '編輯於', dateTimeFormat: 'YYYY/MM/DD HH:mm', regeneratingMessage: '這可能需要一些時間,請稍候...', diff --git a/web/i18n/zh-Hant/dataset-hit-testing.ts b/web/i18n/zh-Hant/dataset-hit-testing.ts index 4b8cc5150a..079e985a09 100644 --- a/web/i18n/zh-Hant/dataset-hit-testing.ts +++ b/web/i18n/zh-Hant/dataset-hit-testing.ts @@ -1,6 +1,6 @@ const translation = { - title: '召回測試', - desc: '基於給定的查詢文字測試知識庫的召回效果。', + title: '檢索測試', + desc: '基於給定的查詢文字測試知識庫的檢索效果。', dateTimeFormat: 'YYYY-MM-DD HH:mm', table: { header: { @@ -10,15 +10,15 @@ const translation = { }, }, input: { - title: '源文字', + title: '來源文字', placeholder: '請輸入文字,建議使用簡短的陳述句。', countWarning: '不超過 200 個字元', - indexWarning: '僅支援高質量模式知識庫', + indexWarning: '僅支援高品質模式知識庫', testing: '測試', }, hit: { - title: '召回段落', - emptyTip: '召回測試結果將展示在這裡', + title: '檢索段落', + emptyTip: '檢索測試結果將展示在這裡', }, noRecentTip: '最近無查詢結果', viewChart: '查看向量圖表', @@ -26,8 +26,8 @@ const translation = { settingTitle: '檢索設置', open: '打開', records: '記錄', - chunkDetail: '數據塊詳細資訊', - hitChunks: '命中 {{num}} 個子塊', + chunkDetail: '資料區塊詳細資訊', + hitChunks: '命中 {{num}} 個子區塊', keyword: '關鍵字', } diff --git a/web/i18n/zh-Hant/dataset-settings.ts b/web/i18n/zh-Hant/dataset-settings.ts index 0c158e8748..30b779b4e6 100644 --- a/web/i18n/zh-Hant/dataset-settings.ts +++ b/web/i18n/zh-Hant/dataset-settings.ts @@ -13,7 +13,7 @@ const translation = { permissionsOnlyMe: '只有我', permissionsAllMember: '所有團隊成員', indexMethod: '索引模式', - indexMethodHighQuality: '高質量', + indexMethodHighQuality: '高品質', indexMethodHighQualityTip: '使用 Embedding 模型進行處理,以在使用者查詢時提供更高的準確度。', indexMethodEconomy: '經濟', indexMethodEconomyTip: '每個區塊使用 10 個關鍵字進行檢索,不會消耗 tokens,但可能會降低檢索的準確度。', diff --git a/web/i18n/zh-Hant/dataset.ts b/web/i18n/zh-Hant/dataset.ts index f037ec7a70..80ec728d56 100644 --- a/web/i18n/zh-Hant/dataset.ts +++ b/web/i18n/zh-Hant/dataset.ts @@ -50,7 +50,7 @@ const translation = { docsFailedNotice: '文件無法被索引', retry: '重試', indexingTechnique: { - high_quality: '高質量', + high_quality: '高品質', economy: '經濟', }, indexingMethod: { @@ -59,7 +59,7 @@ const translation = { hybrid_search: '混合', invertedIndex: '倒排索引', }, - mixtureHighQualityAndEconomicTip: '混合高質量和經濟知識庫需要重新排序模型。', + mixtureHighQualityAndEconomicTip: '混合高品質和經濟知識庫需要重新排序模型。', inconsistentEmbeddingModelTip: '如果選定知識庫的嵌入模型不一致,則需要重新排序模型。', retrievalSettings: '檢索設置', rerankSettings: '重新排序設置', @@ -135,7 +135,7 @@ const translation = { mixtureInternalAndExternalTip: 'Rerank 模型是內部和外部知識的混合所必需的。', connectDataset: '連接到外部知識庫', learnHowToWriteGoodKnowledgeDescription: '瞭解如何編寫良好的知識描述', - createExternalAPI: '添加外部知識 API', + createExternalAPI: '新增外部知識 API', externalAPIPanelTitle: '外部知識 API', createNewExternalAPI: '創建新的外部知識 API', externalKnowledgeDescriptionPlaceholder: '描述此知識庫中的內容(選擇)', @@ -179,7 +179,7 @@ const translation = { type: '類型', title: '新元數據', back: '返回', - namePlaceholder: '添加元數據名稱', + namePlaceholder: '新增元數據名稱', name: '名字', }, checkName: { @@ -201,7 +201,7 @@ const translation = { }, datasetMetadata: { rename: '重新命名', - addMetaData: '添加元數據', + addMetaData: '新增元數據', builtInDescription: '內建的元數據會自動提取和生成。在使用之前必須啟用,且無法編輯。', name: '名字', disabled: '禁用', @@ -216,11 +216,11 @@ const translation = { startLabeling: '開始標記', documentInformation: '文件資訊', technicalParameters: '技術參數', - metadataToolTip: '元數據作為一個關鍵的過濾器,提高了信息檢索的準確性和相關性。您可以在此處修改和添加此文檔的元數據。', + metadataToolTip: '元數據作為一個關鍵的過濾器,提高了信息檢索的準確性和相關性。您可以在此處修改和新增此文檔的元數據。', }, metadata: '元數據', chooseTime: '選擇一個時間...', - addMetadata: '添加元數據', + addMetadata: '新增元數據', }, embeddingModelNotAvailable: '嵌入模型無法使用。', externalKnowledgeBase: '外部知識庫', diff --git a/web/i18n/zh-Hant/education.ts b/web/i18n/zh-Hant/education.ts index bd2bd88249..36197996e0 100644 --- a/web/i18n/zh-Hant/education.ts +++ b/web/i18n/zh-Hant/education.ts @@ -55,7 +55,7 @@ const translation = { }, stillInEducation: { title: '仍在接受教育嗎?', - expired: '立即重新驗證,以獲得即將到來的學年新優惠券。我們會將其添加到您的帳戶中,您可以用於下一次升級。', + expired: '立即重新驗證,以獲得即將到來的學年新優惠券。我們會將其新增到您的帳戶中,您可以用於下一次升級。', isAboutToExpire: '現在重新驗證以獲得即將到來的學年新優惠券。它將保存在您的帳戶中,並在下次續訂時隨時可以使用。', }, alreadyGraduated: { diff --git a/web/i18n/zh-Hant/login.ts b/web/i18n/zh-Hant/login.ts index 0e7608140f..eae1ac9c75 100644 --- a/web/i18n/zh-Hant/login.ts +++ b/web/i18n/zh-Hant/login.ts @@ -1,5 +1,6 @@ const translation = { pageTitle: '嗨,近來可好', + pageTitleForE: '嗨,近來可好', welcome: '👋 歡迎來到 Dify, 登入以繼續', email: '郵箱', emailPlaceholder: '輸入郵箱地址', @@ -94,7 +95,7 @@ const translation = { changePasswordBtn: '設置密碼', enterYourName: '請輸入您的使用者名', backToLogin: '返回登錄', - noLoginMethodTip: '請聯繫系統管理員以添加身份驗證方法。', + noLoginMethodTip: '請聯繫系統管理員以新增身份驗證方法。', withSSO: '繼續使用 SSO', back: '返回', resetPasswordDesc: '輸入您用於註冊 Dify 的電子郵件,我們將向您發送一封密碼重置電子郵件。', @@ -107,7 +108,7 @@ const translation = { licenseLostTip: '無法連接 Dify 許可證伺服器。請聯繫您的管理員以繼續使用 Dify。', webapp: { noLoginMethod: '未為網絡應用程序配置身份驗證方法', - noLoginMethodTip: '請聯絡系統管理員以添加身份驗證方法。', + noLoginMethodTip: '請聯絡系統管理員以新增身份驗證方法。', disabled: '網頁應用程序身份驗證已被禁用。請聯繫系統管理員以啟用它。您可以嘗試直接使用應用程序。', login: '登入', }, diff --git a/web/i18n/zh-Hant/plugin.ts b/web/i18n/zh-Hant/plugin.ts index d979087c3a..ade3125a47 100644 --- a/web/i18n/zh-Hant/plugin.ts +++ b/web/i18n/zh-Hant/plugin.ts @@ -57,11 +57,11 @@ const translation = { placeholder: '選擇工具...', uninstalledTitle: '未安裝工具', auto: '自動', - title: '添加工具', + title: '新增工具', unsupportedContent: '已安裝的插件版本不提供此作。', settings: '用戶設置', uninstalledLink: '在插件中管理', - empty: '點擊 『+』 按鈕添加工具。您可以新增多個工具。', + empty: '點擊 『+』 按鈕新增工具。您可以新增多個工具。', unsupportedContent2: '按兩下以切換版本。', paramsTip1: '控制 LLM 推理參數。', toolSetting: '工具設定', @@ -74,7 +74,7 @@ const translation = { endpointDisableTip: '禁用端點', endpointsTip: '此插件通過終端節點提供特定功能,您可以為當前工作區配置多個終端節點集。', modelNum: '{{num}}包含的型號', - endpointsEmpty: '按兩下「+」按鈕添加端點', + endpointsEmpty: '按兩下「+」按鈕新增端點', endpointDisableContent: '您想禁用 {{name}} 嗎?', configureApp: '配置 App', endpointDeleteContent: '您想刪除 {{name}} 嗎?', @@ -233,9 +233,9 @@ const translation = { oauthClientSettings: 'OAuth 客戶端設置', setupOAuth: '設置 OAuth 客戶端', saveOnly: '僅保存', - addApi: '添加 API 金鑰', + addApi: '新增 API 金鑰', custom: '自訂', - addOAuth: '添加 OAuth', + addOAuth: '新增 OAuth', useOAuthAuth: '使用 OAuth 授權', authorization: '授權', oauthClient: 'OAuth 客戶端', diff --git a/web/i18n/zh-Hant/tools.ts b/web/i18n/zh-Hant/tools.ts index 3c53b87c72..246d2d9dd5 100644 --- a/web/i18n/zh-Hant/tools.ts +++ b/web/i18n/zh-Hant/tools.ts @@ -2,7 +2,6 @@ const translation = { title: '工具', createCustomTool: '建立自定義工具', type: { - all: '全部', builtIn: '內建', custom: '自定義', workflow: '工作流', @@ -20,7 +19,6 @@ const translation = { setupModalTitleDescription: '配置憑據後,工作區中的所有成員都可以在編排應用程式時使用此工具。', }, includeToolNum: '包含 {{num}} 個工具', - addTool: '新增工具', createTool: { title: '建立自定義工具', editAction: '編輯', @@ -141,10 +139,8 @@ const translation = { notAuthorized: '工具未授權', howToGet: '如何獲取', addToolModal: { - add: '加', type: '類型', - added: '添加', - manageInTools: '在工具中管理', + added: '新增', category: '類別', custom: { title: '沒有可用的自訂工具', @@ -198,11 +194,17 @@ const translation = { noHeaders: '沒有配置自定義標頭', timeoutPlaceholder: '三十', headerValuePlaceholder: '例如,承載者令牌123', - addHeader: '添加標題', + addHeader: '新增標題', headerKeyPlaceholder: '例如,授權', headersTip: '與 MCP 伺服器請求一同發送的附加 HTTP 標頭', maskedHeadersTip: '標頭值已被遮罩以保障安全。更改將更新實際值。', headers: '標題', + authentication: '身份驗證', + clientID: '客戶編號', + clientSecretPlaceholder: '客戶端密鑰', + configurations: '設定', + useDynamicClientRegistration: '使用動態客戶端註冊', + clientSecret: '客戶端密鑰', }, delete: '刪除 MCP 伺服器', deleteConfirmTitle: '您確定要刪除 {{mcp}} 嗎?', @@ -237,7 +239,7 @@ const translation = { description: '描述', descriptionPlaceholder: '說明此工具的用途及如何被 LLM 使用', parameters: '參數', - parametersTip: '為每個參數添加描述,以幫助 LLM 理解其目的和約束。', + parametersTip: '為每個參數新增描述,以幫助 LLM 理解其目的和約束。', parametersPlaceholder: '參數的目的和約束', confirm: '啟用 MCP 伺服器', }, diff --git a/web/i18n/zh-Hant/workflow.ts b/web/i18n/zh-Hant/workflow.ts index faa80b0fa4..ce053d6e5b 100644 --- a/web/i18n/zh-Hant/workflow.ts +++ b/web/i18n/zh-Hant/workflow.ts @@ -8,7 +8,7 @@ const translation = { published: '已發佈', publish: '發佈', update: '更新', - run: '運行', + run: '測試運行', running: '運行中', inRunMode: '在運行模式中', inPreview: '預覽中', @@ -18,7 +18,6 @@ const translation = { runHistory: '運行歷史', goBackToEdit: '返回編輯模式', conversationLog: '對話記錄', - features: '功能', debugAndPreview: '預覽', restart: '重新開始', currentDraft: '當前草稿', @@ -30,35 +29,37 @@ const translation = { batchRunApp: '批量運行', accessAPIReference: '訪問 API', embedIntoSite: '嵌入網站', - addTitle: '添加標題...', - addDescription: '添加描述...', - noVar: '沒有變量', - variableNamePlaceholder: '變量名', - searchVar: '搜索變量', - setVarValuePlaceholder: '設置變量值', + addTitle: '新增標題...', + addDescription: '新增描述...', + noVar: '沒有變數', + variableNamePlaceholder: '變數名', + searchVar: '搜索變數', + setVarValuePlaceholder: '設置變數值', needConnectTip: '此節點尚未連接到其他節點', maxTreeDepth: '每個分支最大限制 {{depth}} 個節點', - needEndNode: '必須添加結束節點', - needAnswerNode: '必須添加直接回覆節點', + needAdd: '必須新增{{node}}節點', + needOutputNode: '必須新增輸出節點', + needEndNode: '必須新增結束節點', + needAnswerNode: '必須新增直接回覆節點', workflowProcess: '工作流', notRunning: '尚未運行', - previewPlaceholder: '在下面的框中輸入內容開始調試聊天機器人', + previewPlaceholder: '在下面的框中輸入內容開始測試聊天機器人', effectVarConfirm: { - title: '移除變量', - content: '該變量在其他節點中使用。您是否仍要刪除它?', + title: '移除變數', + content: '該變數在其他節點中使用。您是否仍要刪除它?', }, insertVarTip: '按 \'/\' 鍵快速插入', - processData: '數據處理', + processData: '資料處理', input: '輸入', output: '輸出', - jinjaEditorPlaceholder: '輸入“/”或“{”插入變量', + jinjaEditorPlaceholder: '輸入“/”或“{”插入變數', viewOnly: '只讀', showRunHistory: '顯示運行歷史', enableJinja: '開啟支持 Jinja 模板', learnMore: '了解更多', copy: '拷貝', duplicate: '複製', - addBlock: '添加節點', + addBlock: '新增節點', pasteHere: '粘貼到這裡', pointerMode: '指針模式', handMode: '手模式', @@ -74,16 +75,16 @@ const translation = { overwriteAndImport: '覆蓋和導入', importSuccess: '導入成功', chooseDSL: '選擇 DSL(yml)檔', - syncingData: '同步數據,只需幾秒鐘。', + syncingData: '同步資料,只需幾秒鐘。', importDSLTip: '當前草稿將被覆蓋。在導入之前將工作流匯出為備份。', importFailure: '匯入失敗', parallelTip: { click: { title: '點擊', - desc: '添加', + desc: '新增', }, drag: { - title: '拖动', + title: '拖動', desc: '連接', }, limit: '並行度僅限於 {{num}} 個分支。', @@ -91,18 +92,16 @@ const translation = { }, disconnect: '斷開', jumpToNode: '跳轉到此節點', - addParallelNode: '添加並行節點', + addParallelNode: '新增並行節點', parallel: '並行', branch: '分支', - featuresDocLink: '瞭解更多資訊', fileUploadTip: '圖片上傳功能已升級為檔上傳。', ImageUploadLegacyTip: '現在,您可以在起始表單中創建檔案類型變數。我們將來不再支持圖片上傳功能。', - featuresDescription: '增強 Web 應用程式用戶體驗', importWarning: '謹慎', importWarningDetails: 'DSL 版本差異可能會影響某些功能', openInExplore: '在“探索”中打開', onFailure: '失敗時', - addFailureBranch: '添加 Fail Branch', + addFailureBranch: '新增 Fail Branch', loadMore: '載入更多工作流', noHistory: '無歷史記錄', publishUpdate: '發布更新', @@ -115,20 +114,21 @@ const translation = { tagBound: '使用此標籤的應用程式數量', currentView: '當前檢視', currentWorkflow: '當前工作流程', + moreActions: '更多動作', }, env: { envPanelTitle: '環境變數', envDescription: '環境變數可用於存儲私人信息和憑證。它們是唯讀的,並且可以在導出時與 DSL 文件分開。', - envPanelButton: '添加變數', + envPanelButton: '新增變數', modal: { - title: '添加環境變數', + title: '新增環境變數', editTitle: '編輯環境變數', type: '類型', name: '名稱', namePlaceholder: '環境名稱', value: '值', valuePlaceholder: '環境值', - secretTip: '用於定義敏感信息或數據,DSL 設置配置為防止洩露。', + secretTip: '用於定義敏感信息或資料,DSL 設置配置為防止洩露。', description: '描述', descriptionPlaceholder: '描述此變數', }, @@ -139,6 +139,19 @@ const translation = { export: '導出帶有機密值的 DSL', }, }, + globalVar: { + title: '系統變數', + description: '系統變數是全域變數,在類型符合時可由任意節點在無需連線的情況下引用,例如終端使用者 ID 與工作流程 ID。', + fieldsDescription: { + conversationId: '對話 ID', + dialogCount: '對話次數', + userId: '使用者 ID', + triggerTimestamp: '應用程式開始運行的時間戳', + appId: '應用程式 ID', + workflowId: '工作流程 ID', + workflowRunId: '工作流程執行 ID', + }, + }, chatVariable: { panelTitle: '對話變數', panelDescription: '對話變數用於儲存 LLM 需要記住的互動資訊,包括對話歷史、上傳的檔案、使用者偏好等。這些變數可讀寫。', @@ -185,9 +198,9 @@ const translation = { nodeConnect: '區塊已連接', nodePaste: '區塊已粘貼', nodeDelete: '區塊已刪除', - nodeAdd: '區塊已添加', + nodeAdd: '區塊已新增', nodeResize: '區塊已調整大小', - noteAdd: '註釋已添加', + noteAdd: '註釋已新增', noteChange: '註釋已更改', edgeDelete: '區塊已斷開連接', noteDelete: '註釋已刪除', @@ -197,14 +210,14 @@ const translation = { authRequired: '請先授權', invalidJson: '{{field}} 是非法的 JSON', fields: { - variable: '變量名', - variableValue: '變量值', - code: '代碼', + variable: '變數名', + variableValue: '變數值', + code: '程式碼', model: '模型', rerankModel: 'Rerank 模型', visionVariable: 'Vision Variable', }, - invalidVariable: '無效的變量', + invalidVariable: '無效的變數', rerankModelRequired: '在開啟 Rerank 模型之前,請在設置中確認模型配置成功。', toolParameterRequired: '{{field}}:参數 [{{param}}] 為必填項', noValidTool: '{{field}} 未選擇有效工具', @@ -224,6 +237,8 @@ const translation = { 'searchBlock': '搜索節點', 'blocks': '節點', 'tools': '工具', + 'searchTrigger': '搜尋觸發器...', + 'allTriggers': '所有觸發器', 'allTool': '全部', 'customTool': '自定義', 'workflowTool': '工作流', @@ -235,24 +250,26 @@ const translation = { 'searchTool': '搜索工具', 'agent': '代理策略', 'plugin': '插件', - 'allAdded': '所有已添加的', - 'addAll': '全部添加', + 'allAdded': '所有已新增的', + 'addAll': '全部新增', 'sources': '來源', 'searchDataSource': '搜尋資料來源', + 'noFeaturedPlugins': '前往 Marketplace 查看更多工具', + 'noFeaturedTriggers': '前往 Marketplace 查看更多觸發器', }, blocks: { 'start': '開始', - 'end': '結束', + 'end': '輸出', 'answer': '直接回覆', 'llm': 'LLM', 'knowledge-retrieval': '知識檢索', 'question-classifier': '問題分類器', 'if-else': '條件分支', - 'code': '代碼執行', + 'code': '程式碼執行', 'template-transform': '模板轉換', 'http-request': 'HTTP 請求', - 'variable-assigner': '變量聚合器', - 'variable-aggregator': '變量聚合器', + 'variable-assigner': '變數聚合器', + 'variable-aggregator': '變數聚合器', 'assigner': '變數分配器', 'iteration-start': '迭代開始', 'iteration': '迭代', @@ -268,18 +285,18 @@ const translation = { }, blocksAbout: { 'start': '定義一個 workflow 流程啟動的參數', - 'end': '定義一個 workflow 流程的結束和結果類型', + 'end': '定義一個 workflow 流程的輸出和結果類型', 'answer': '定義一個聊天對話的回覆內容', 'llm': '調用大語言模型回答問題或者對自然語言進行處理', 'knowledge-retrieval': '允許你從知識庫中查詢與用戶問題相關的文本內容', 'question-classifier': '定義用戶問題的分類條件,LLM 能夠根據分類描述定義對話的進展方式', 'if-else': '允許你根據 if/else 條件將 workflow 拆分成兩個分支', - 'code': '執行一段 Python 或 NodeJS 代碼實現自定義邏輯', - 'template-transform': '使用 Jinja 模板語法將數據轉換為字符串', + 'code': '執行一段 Python 或 NodeJS 程式碼實現自定義邏輯', + 'template-transform': '使用 Jinja 模板語法將資料轉換為字符串', 'http-request': '允許通過 HTTP 協議發送服務器請求', - 'variable-assigner': '將多路分支的變量聚合為一個變量,以實現下游節點統一配置。', + 'variable-assigner': '將多路分支的變數聚合為一個變數,以實現下游節點統一配置。', 'assigner': '變數分配節點用於為可寫入的變數(如對話變數)分配值。', - 'variable-aggregator': '將多路分支的變量聚合為一個變量,以實現下游節點統一配置。', + 'variable-aggregator': '將多路分支的變數聚合為一個變數,以實現下游節點統一配置。', 'iteration': '對列表對象執行多次步驟直至輸出所有結果。', 'parameter-extractor': '利用 LLM 從自然語言內推理提取出結構化參數,用於後置的工具調用或 HTTP 請求。', 'document-extractor': '用於將上傳的文件解析為 LLM 易於理解的文字內容。', @@ -312,11 +329,11 @@ const translation = { panel: { userInputField: '用戶輸入字段', changeBlock: '更改節點', - helpLink: '幫助鏈接', + helpLink: '查看幫助文件', about: '關於', createdBy: '作者', nextStep: '下一步', - addNextStep: '添加此工作流程中的下一個節點', + addNextStep: '新增此工作流程中的下一個節點', selectNextStep: '選擇下一個節點', runThisStep: '運行此步驟', checklist: '檢查清單', @@ -325,15 +342,15 @@ const translation = { organizeBlocks: '整理節點', change: '更改', optional: '(選擇性)', - moveToThisNode: '定位至此節點', minimize: '退出全螢幕', maximize: '最大化畫布', + scrollToSelectedNode: '捲動至選取的節點', optional_and_hidden: '(可選且隱藏)', }, nodes: { common: { - outputVars: '輸出變量', - insertVarTip: '插入變量', + outputVars: '輸出變數', + insertVarTip: '插入變數', memory: { memory: '記憶', memoryTip: '聊天記憶設置', @@ -395,7 +412,7 @@ const translation = { start: { required: '必填', inputField: '輸入字段', - builtInVar: '內置變量', + builtInVar: '內置變數', outputVars: { query: '用戶輸入', memories: { @@ -411,7 +428,7 @@ const translation = { outputs: '輸出', output: { type: '輸出類型', - variable: '輸出變量', + variable: '輸出變數', }, type: { 'none': '無', @@ -421,16 +438,16 @@ const translation = { }, answer: { answer: '回覆', - outputVars: '輸出變量', + outputVars: '輸出變數', }, llm: { model: '模型', - variables: '變量', + variables: '變數', context: '上下文', contextTooltip: '您可以導入知識庫作為上下文', - notSetContextInPromptTip: '要啟用上下文功能,請在提示中填寫上下文變量。', + notSetContextInPromptTip: '要啟用上下文功能,請在提示中填寫上下文變數。', prompt: '提示詞', - addMessage: '添加消息', + addMessage: '新增消息', roleDescription: { system: '為對話提供高層指導', user: '向模型提供指令、查詢或任何基於文本的輸入', @@ -449,7 +466,7 @@ const translation = { usage: '模型用量信息', }, singleRun: { - variable: '變量', + variable: '變數', }, sysQueryInUser: 'user message 中必須包含 sys.query', jsonSchema: { @@ -462,7 +479,7 @@ const translation = { promptPlaceholder: '描述你的 JSON 架構...', addField: '新增字段', generate: '生成', - descriptionPlaceholder: '添加描述', + descriptionPlaceholder: '新增描述', fieldNamePlaceholder: '欄位名稱', showAdvancedOptions: '顯示進階選項', import: '從 JSON 匯入', @@ -488,49 +505,49 @@ const translation = { }, }, knowledgeRetrieval: { - queryVariable: '查詢變量', + queryVariable: '查詢變數', knowledge: '知識庫', outputVars: { - output: '召回的分段', + output: '檢索的分段', content: '分段內容', title: '分段標題', icon: '分段圖標', url: '分段鏈接', - metadata: '其他元數據', + metadata: '其他元資料', }, metadata: { options: { disabled: { - subTitle: '不啟用元數據過濾', + subTitle: '不啟用元資料過濾', title: '禁用', }, automatic: { title: '自動的', - subTitle: '根據用戶查詢自動生成元數據過濾條件', - desc: '根據查詢變數自動生成元數據過濾條件', + subTitle: '根據用戶查詢自動生成元資料過濾條件', + desc: '根據查詢變數自動生成元資料過濾條件', }, manual: { title: '手動', - subTitle: '手動添加元數據過濾條件', + subTitle: '手動新增元資料過濾條件', }, }, panel: { - add: '添加條件', + add: '新增條件', datePlaceholder: '選擇一個時間...', - search: '搜尋元數據', + search: '搜尋元資料', conditions: '條件', - title: '元數據過濾條件', + title: '元資料過濾條件', select: '選擇變數...', placeholder: '輸入數值', }, - title: '元數據過濾', - tip: '元數據過濾是使用元數據屬性(如標籤、類別或訪問權限)來精煉和控制在系統內檢索相關信息的過程。', + title: '元資料過濾', + tip: '元資料過濾是使用元資料屬性(如標籤、類別或訪問權限)來精煉和控制在系統內檢索相關信息的過程。', }, }, http: { - inputVars: '輸入變量', + inputVars: '輸入變數', api: 'API', - apiPlaceholder: '輸入 URL,輸入變量時請鍵入‘/’', + apiPlaceholder: '輸入 URL,輸入變數時請鍵入‘/’', notStartWithHttp: 'API 應該以 http:// 或 https:// 開頭', key: '鍵', value: '值', @@ -557,7 +574,7 @@ const translation = { 'api-key-title': 'API Key', 'header': 'Header', }, - insertVarPlaceholder: '鍵入 \'/\' 鍵快速插入變量', + insertVarPlaceholder: '鍵入 \'/\' 鍵快速插入變數', timeout: { title: '超時設置', connectLabel: '連接超時', @@ -580,16 +597,16 @@ const translation = { }, }, code: { - inputVars: '輸入變量', - outputVars: '輸出變量', + inputVars: '輸入變數', + outputVars: '輸出變數', advancedDependencies: '高級依賴', - advancedDependenciesTip: '在這裡添加一些預加載需要消耗較多時間或非默認內置的依賴包', + advancedDependenciesTip: '在這裡新增一些預加載需要消耗較多時間或非默認內置的依賴包', searchDependencies: '搜索依賴', syncFunctionSignature: '同步函數簽名至代碼', }, templateTransform: { - inputVars: '輸入變量', - code: '代碼', + inputVars: '輸入變數', + code: '模板程式碼', codeSupportTip: '只支持 Jinja2', outputVars: { output: '轉換後內容', @@ -602,7 +619,7 @@ const translation = { and: 'and', or: 'or', operator: '操作符', - notSetVariable: '請先設置變量', + notSetVariable: '請先設置變數', comparisonOperator: { 'contains': '包含', 'not contains': '不包含', @@ -623,7 +640,7 @@ const translation = { 'before': '之前', }, enterValue: '輸入值', - addCondition: '添加條件', + addCondition: '新增條件', conditionNotSetup: '條件未設置', selectVariable: '選擇變數...', optionName: { @@ -638,10 +655,10 @@ const translation = { addSubVariable: '子變數', }, variableAssigner: { - title: '變量賦值', + title: '變數賦值', outputType: '輸出類型', - varNotSet: '未設置變量', - noVarTip: '添加需要賦值的變量', + varNotSet: '未設置變數', + noVarTip: '新增需要賦值的變數', type: { string: 'String', number: 'Number', @@ -649,12 +666,12 @@ const translation = { array: 'Array', }, aggregationGroup: '聚合分組', - aggregationGroupTip: '開啟該功能後,變量聚合器內可以同時聚合多組變量', - addGroup: '添加分組', + aggregationGroupTip: '開啟該功能後,變數聚合器內可以同時聚合多組變數', + addGroup: '新增分組', outputVars: { - varDescribe: '{{groupName}}的輸出變量', + varDescribe: '{{groupName}}的輸出變數', }, - setAssignVariable: '設置賦值變量', + setAssignVariable: '設置賦值變數', }, assigner: { 'assignedVariable': '已分配變數', @@ -685,13 +702,13 @@ const translation = { 'variables': '變數', 'selectAssignedVariable': '選擇配置的變數...', 'setParameter': '設定參數...', - 'noVarTip': '點擊「+」按鈕添加變數', + 'noVarTip': '點擊「+」按鈕新增變數', 'assignedVarsDescription': '分配的變數必須是可寫變數,例如對話變數。', 'varNotSet': '未設置變數', }, tool: { authorize: '授權', - inputVars: '輸入變量', + inputVars: '輸入變數', outputVars: { text: '工具生成的內容', files: { @@ -703,13 +720,13 @@ const translation = { }, json: '工具生成的 JSON', }, - insertPlaceholder2: '插入變量', + insertPlaceholder2: '插入變數', insertPlaceholder1: '輸入或按壓', settings: '設定', }, questionClassifiers: { model: '模型', - inputVars: '輸入變量', + inputVars: '輸入變數', outputVars: { className: '分類名稱', usage: '模型用量信息', @@ -719,13 +736,13 @@ const translation = { advancedSetting: '高級設置', topicName: '主題內容', topicPlaceholder: '在這裡輸入你的主題內容', - addClass: '添加分類', + addClass: '新增分類', instruction: '指令', instructionTip: '你可以輸入額外的附加指令,幫助問題分類器更好的理解如何分類', instructionPlaceholder: '在這裡輸入你的指令', }, parameterExtractor: { - inputVar: '輸入變量', + inputVar: '輸入變數', outputVars: { isSuccess: '是否成功。成功時值為 1,失敗時值為 0。', errorReason: '錯誤原因', @@ -733,7 +750,7 @@ const translation = { }, extractParameters: '提取參數', importFromTool: '從工具導入', - addExtractParameter: '添加提取參數', + addExtractParameter: '新增提取參數', addExtractParameterContent: { name: '名稱', namePlaceholder: '提取參數名稱', @@ -755,7 +772,7 @@ const translation = { deleteTitle: '刪除迭代節點?', deleteDesc: '刪除迭代節點將刪除所有子節點', input: '輸入', - output: '輸出變量', + output: '輸出變數', iteration_one: '{{count}}個迭代', iteration_other: '{{count}}個迭代', currentIteration: '當前迭代', @@ -794,7 +811,7 @@ const translation = { strikethrough: '刪除線', placeholder: '寫下您的筆記...', }, - addNote: '添加註釋', + addNote: '新增註解', }, docExtractor: { outputVars: { @@ -905,8 +922,8 @@ const translation = { error_one: '{{count}} 錯誤', loopMaxCount: '最大迴圈次數', input: '輸入', - loopVariables: '循環變量', - output: '輸出變量', + loopVariables: '循環變數', + output: '輸出變數', comma: ',', errorResponseMethod: '錯誤回應方法', breakCondition: '迴圈終止條件', @@ -1006,8 +1023,20 @@ const translation = { description: '上次運行的結果將顯示在這裡', }, variableInspect: { + listening: { + title: '正在監聽觸發器事件…', + tip: '您現在可以向 HTTP {{nodeName}} 端點發送測試請求來模擬事件觸發,或將其作為即時事件除錯的回呼 URL。所有輸出都可在變數檢視器中直接查看。', + tipPlugin: '您現在可以在 {{- pluginName}} 中建立事件,並在變數檢視器中檢視這些事件的輸出。', + tipSchedule: '正在監聽排程觸發器事件。\n下一次排程執行時間:{{nextTriggerTime}}', + tipFallback: '正在等待觸發器事件,輸出會顯示在此處。', + defaultNodeName: '此觸發器', + defaultPluginName: '此插件觸發器', + defaultScheduleTime: '未設定', + selectedTriggers: '已選觸發器', + stopButton: '停止', + }, trigger: { - cached: '查看緩存的變量', + cached: '查看快取的變數', stop: '停止跑步', clear: '清晰', running: '快取運行狀態', @@ -1026,7 +1055,7 @@ const translation = { emptyTip: '在畫布上逐步執行節點或逐步運行節點後,您可以在變數檢視中查看節點變數的當前值。', resetConversationVar: '將對話變數重置為默認值', export: '出口', - largeData: '大數據,唯讀預覽。匯出以檢視全部。', + largeData: '大資料,唯讀預覽。匯出以檢視全部。', exportToolTip: '將變數匯出為檔案', largeDataNoExport: '大型資料 - 僅部分預覽', }, diff --git a/web/models/app.ts b/web/models/app.ts index 454cc5d1e8..e0f31ff26e 100644 --- a/web/models/app.ts +++ b/web/models/app.ts @@ -1,5 +1,15 @@ -import type { AliyunConfig, ArizeConfig, LangFuseConfig, LangSmithConfig, OpikConfig, PhoenixConfig, TencentConfig, TracingProvider, WeaveConfig } from '@/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/tracing/type' -import type { App, AppMode, AppTemplate, SiteConfig } from '@/types/app' +import type { + AliyunConfig, + ArizeConfig, + LangFuseConfig, + LangSmithConfig, + OpikConfig, + PhoenixConfig, + TencentConfig, + TracingProvider, + WeaveConfig, +} from '@/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/tracing/type' +import type { App, AppModeEnum, AppTemplate, SiteConfig } from '@/types/app' import type { Dependency } from '@/app/components/plugins/types' export enum DSLImportMode { @@ -27,7 +37,7 @@ export type AppDetailResponse = App export type DSLImportResponse = { id: string status: DSLImportStatus - app_mode: AppMode + app_mode: AppModeEnum app_id?: string current_dsl_version?: string imported_dsl_version?: string @@ -111,3 +121,12 @@ export type TracingConfig = { tracing_provider: TracingProvider tracing_config: ArizeConfig | PhoenixConfig | LangSmithConfig | LangFuseConfig | OpikConfig | WeaveConfig | AliyunConfig | TencentConfig } + +export type WebhookTriggerResponse = { + id: string + webhook_id: string + webhook_url: string + webhook_debug_url: string + node_id: string + created_at: string +} diff --git a/web/models/common.ts b/web/models/common.ts index aa6372e36f..d83ae5fb98 100644 --- a/web/models/common.ts +++ b/web/models/common.ts @@ -236,6 +236,7 @@ export type FileUploadConfigResponse = { audio_file_size_limit?: number // default is 50MB video_file_size_limit?: number // default is 100MB workflow_file_upload_limit?: number // default is 10 + file_upload_limit: number // default is 5 } export type InvitationResult = { diff --git a/web/models/datasets.ts b/web/models/datasets.ts index aeeb5c161a..eb7b7de4a2 100644 --- a/web/models/datasets.ts +++ b/web/models/datasets.ts @@ -1,5 +1,5 @@ import type { DataSourceNotionPage, DataSourceProvider } from './common' -import type { AppIconType, AppMode, RetrievalConfig, TransferMethod } from '@/types/app' +import type { AppIconType, AppModeEnum, RetrievalConfig, TransferMethod } from '@/types/app' import type { Tag } from '@/app/components/base/tag-management/constant' import type { IndexingType } from '@/app/components/datasets/create/step-two' import type { MetadataFilteringVariableType } from '@/app/components/workflow/nodes/knowledge-retrieval/types' @@ -344,6 +344,8 @@ export type WebsiteCrawlInfo = { description: string source_url: string title: string + provider?: string + job_id?: string } export type OnlineDocumentInfo = { @@ -660,7 +662,7 @@ export type ExternalKnowledgeBaseHitTestingResponse = { export type RelatedApp = { id: string name: string - mode: AppMode + mode: AppModeEnum icon_type: AppIconType | null icon: string icon_background: string diff --git a/web/models/debug.ts b/web/models/debug.ts index 90995e72dc..344db6e5af 100644 --- a/web/models/debug.ts +++ b/web/models/debug.ts @@ -9,6 +9,7 @@ import type { MetadataFilteringModeEnum, } from '@/app/components/workflow/nodes/knowledge-retrieval/types' import type { ModelConfig as NodeModelConfig } from '@/app/components/workflow/types' +import type { ExternalDataTool } from '@/models/common' export type Inputs = Record<string, string | number | object | boolean> export enum PromptMode { @@ -133,9 +134,9 @@ export type ModelConfig = { model_id: string mode: ModelModeType prompt_type?: PromptMode + configs: PromptConfig chat_prompt_config?: ChatPromptConfig | null completion_prompt_config?: CompletionPromptConfig | null - configs: PromptConfig opening_statement: string | null more_like_this: MoreLikeThisConfig | null suggested_questions: string[] | null @@ -146,6 +147,14 @@ export type ModelConfig = { retriever_resource: RetrieverResourceConfig | null sensitive_word_avoidance: ModerationConfig | null annotation_reply: AnnotationReplyConfig | null + external_data_tools?: ExternalDataTool[] | null + system_parameters: { + audio_file_size_limit: number + file_size_limit: number + image_file_size_limit: number + video_file_size_limit: number + workflow_file_upload_limit: number + } dataSets: any[] agentConfig: AgentConfig } diff --git a/web/models/explore.ts b/web/models/explore.ts index 735aa54eab..01a946ff28 100644 --- a/web/models/explore.ts +++ b/web/models/explore.ts @@ -1,7 +1,7 @@ -import type { AppIconType, AppMode } from '@/types/app' +import type { AppIconType, AppModeEnum } from '@/types/app' export type AppBasicInfo = { id: string - mode: AppMode + mode: AppModeEnum icon_type: AppIconType | null icon: string icon_background: string diff --git a/web/models/log.ts b/web/models/log.ts index eff46372d0..baa07a59c4 100644 --- a/web/models/log.ts +++ b/web/models/log.ts @@ -229,11 +229,38 @@ export type AnnotationsCountResponse = { count: number } +export enum WorkflowRunTriggeredFrom { + DEBUGGING = 'debugging', + APP_RUN = 'app-run', + RAG_PIPELINE_RUN = 'rag-pipeline-run', + RAG_PIPELINE_DEBUGGING = 'rag-pipeline-debugging', + WEBHOOK = 'webhook', + SCHEDULE = 'schedule', + PLUGIN = 'plugin', +} + +export type TriggerMetadata = { + type?: string + endpoint_id?: string + plugin_unique_identifier?: string + provider_id?: string + event_name?: string + icon_filename?: string + icon_dark_filename?: string + icon?: string | null + icon_dark?: string | null +} + +export type WorkflowLogDetails = { + trigger_metadata?: TriggerMetadata +} + export type WorkflowRunDetail = { id: string version: string status: 'running' | 'succeeded' | 'failed' | 'stopped' error?: string + triggered_from?: WorkflowRunTriggeredFrom elapsed_time: number total_tokens: number total_price: number @@ -255,6 +282,7 @@ export type EndUserInfo = { export type WorkflowAppLogDetail = { id: string workflow_run: WorkflowRunDetail + details?: WorkflowLogDetails created_from: 'service-api' | 'web-app' | 'explore' created_by_role: 'account' | 'end_user' created_by_account?: AccountInfo diff --git a/web/next.config.js b/web/next.config.js index c4f6fc87b6..212bed0a9c 100644 --- a/web/next.config.js +++ b/web/next.config.js @@ -1,4 +1,7 @@ const { codeInspectorPlugin } = require('code-inspector-plugin') + +const isDev = process.env.NODE_ENV === 'development' + const withPWA = require('next-pwa')({ dest: 'public', register: true, @@ -137,6 +140,9 @@ const nextConfig = { ] }, output: 'standalone', + compiler: { + removeConsole: isDev ? false : { exclude: ['warn', 'error'] }, + } } module.exports = withPWA(withBundleAnalyzer(withMDX(nextConfig))) diff --git a/web/package.json b/web/package.json index f5c9607b61..fcb1fa47b5 100644 --- a/web/package.json +++ b/web/package.json @@ -1,8 +1,8 @@ { "name": "dify-web", - "version": "1.9.2", + "version": "1.10.0", "private": true, - "packageManager": "pnpm@10.18.3+sha512.bbd16e6d7286fd7e01f6b3c0b3c932cda2965c06a908328f74663f10a9aea51f1129eea615134bf992831b009eabe167ecb7008b597f40ff9bc75946aadfb08d", + "packageManager": "pnpm@10.22.0+sha512.bf049efe995b28f527fd2b41ae0474ce29186f7edcb3bf545087bd61fbbebb2bf75362d1307fda09c2d288e1e499787ac12d4fcb617a974718a6051f2eee741c", "engines": { "node": ">=v22.11.0" }, @@ -22,7 +22,7 @@ "dev": "cross-env NODE_OPTIONS='--inspect' next dev --turbopack", "build": "next build", "build:docker": "next build && node scripts/optimize-standalone.js", - "start": "cp -r .next/static .next/standalone/.next/static && cp -r public .next/standalone/public && cross-env PORT=$npm_config_port HOSTNAME=$npm_config_host node .next/standalone/server.js", + "start": "node ./scripts/copy-and-start.mjs", "lint": "eslint --cache --cache-location node_modules/.cache/eslint/.eslint-cache", "lint:fix": "eslint --cache --cache-location node_modules/.cache/eslint/.eslint-cache --fix", "lint:quiet": "eslint --cache --cache-location node_modules/.cache/eslint/.eslint-cache --quiet", @@ -73,7 +73,8 @@ "classnames": "^2.5.1", "cmdk": "^1.1.1", "copy-to-clipboard": "^3.3.3", - "dayjs": "^1.11.18", + "cron-parser": "^5.4.0", + "dayjs": "^1.11.19", "decimal.js": "^10.6.0", "dompurify": "^3.3.0", "echarts": "^5.6.0", @@ -146,7 +147,7 @@ "@babel/core": "^7.28.4", "@chromatic-com/storybook": "^4.1.1", "@eslint-react/eslint-plugin": "^1.53.1", - "@happy-dom/jest-environment": "^20.0.7", + "@happy-dom/jest-environment": "^20.0.8", "@mdx-js/loader": "^3.1.1", "@mdx-js/react": "^3.1.1", "@next/bundle-analyzer": "15.5.4", @@ -199,6 +200,7 @@ "sass": "^1.93.2", "storybook": "9.1.13", "tailwindcss": "^3.4.18", + "ts-node": "^10.9.2", "typescript": "^5.9.3", "uglify-js": "^3.19.3" }, @@ -210,7 +212,7 @@ "canvas": "^3.2.0", "esbuild": "~0.25.0", "pbkdf2": "~3.1.3", - "vite": "~6.2", + "vite": "~6.4.1", "prismjs": "~1.30", "brace-expansion": "~2.0" }, @@ -233,7 +235,7 @@ "esbuild@<0.25.0": "0.25.0", "pbkdf2@<3.1.3": "3.1.3", "prismjs@<1.30.0": "1.30.0", - "vite@<6.2.7": "6.2.7", + "vite@<6.4.1": "6.4.1", "array-includes": "npm:@nolyfill/array-includes@^1", "array.prototype.findlast": "npm:@nolyfill/array.prototype.findlast@^1", "array.prototype.findlastindex": "npm:@nolyfill/array.prototype.findlastindex@^1", diff --git a/web/pnpm-lock.yaml b/web/pnpm-lock.yaml index 89812542b1..9e99f990e0 100644 --- a/web/pnpm-lock.yaml +++ b/web/pnpm-lock.yaml @@ -12,7 +12,7 @@ overrides: canvas: ^3.2.0 esbuild: ~0.25.0 pbkdf2: ~3.1.3 - vite: ~6.2 + vite: ~6.4.1 prismjs: ~1.30 brace-expansion: ~2.0 lexical: 0.37.0 @@ -24,7 +24,7 @@ overrides: esbuild@<0.25.0: 0.25.0 pbkdf2@<3.1.3: 3.1.3 prismjs@<1.30.0: 1.30.0 - vite@<6.2.7: 6.2.7 + vite@<6.4.1: 6.4.1 array-includes: npm:@nolyfill/array-includes@^1 array.prototype.findlast: npm:@nolyfill/array.prototype.findlast@^1 array.prototype.findlastindex: npm:@nolyfill/array.prototype.findlastindex@^1 @@ -101,7 +101,7 @@ importers: version: 0.37.0 '@monaco-editor/react': specifier: ^4.7.0 - version: 4.7.0(monaco-editor@0.52.2)(react-dom@19.1.1(react@19.1.1))(react@19.1.1) + version: 4.7.0(monaco-editor@0.54.0)(react-dom@19.1.1(react@19.1.1))(react@19.1.1) '@octokit/core': specifier: ^6.1.6 version: 6.1.6 @@ -147,9 +147,12 @@ importers: copy-to-clipboard: specifier: ^3.3.3 version: 3.3.3 + cron-parser: + specifier: ^5.4.0 + version: 5.4.0 dayjs: - specifier: ^1.11.18 - version: 1.11.18 + specifier: ^1.11.19 + version: 1.11.19 decimal.js: specifier: ^10.6.0 version: 10.6.0 @@ -351,19 +354,19 @@ importers: devDependencies: '@antfu/eslint-config': specifier: ^5.4.1 - version: 5.4.1(@eslint-react/eslint-plugin@1.53.1(eslint@9.38.0(jiti@1.21.7))(ts-api-utils@2.1.0(typescript@5.9.3))(typescript@5.9.3))(@next/eslint-plugin-next@15.5.4)(@vue/compiler-sfc@3.5.17)(eslint-plugin-react-hooks@5.2.0(eslint@9.38.0(jiti@1.21.7)))(eslint-plugin-react-refresh@0.4.24(eslint@9.38.0(jiti@1.21.7)))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + version: 5.4.1(@eslint-react/eslint-plugin@1.53.1(eslint@9.38.0(jiti@1.21.7))(ts-api-utils@2.1.0(typescript@5.9.3))(typescript@5.9.3))(@next/eslint-plugin-next@15.5.4)(@vue/compiler-sfc@3.5.22)(eslint-plugin-react-hooks@5.2.0(eslint@9.38.0(jiti@1.21.7)))(eslint-plugin-react-refresh@0.4.24(eslint@9.38.0(jiti@1.21.7)))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@babel/core': specifier: ^7.28.4 version: 7.28.4 '@chromatic-com/storybook': specifier: ^4.1.1 - version: 4.1.1(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + version: 4.1.1(storybook@9.1.13(@testing-library/dom@10.4.1)) '@eslint-react/eslint-plugin': specifier: ^1.53.1 version: 1.53.1(eslint@9.38.0(jiti@1.21.7))(ts-api-utils@2.1.0(typescript@5.9.3))(typescript@5.9.3) '@happy-dom/jest-environment': - specifier: ^20.0.7 - version: 20.0.7(@jest/environment@29.7.0)(@jest/fake-timers@29.7.0)(@jest/types@29.6.3)(jest-mock@29.7.0)(jest-util@29.7.0) + specifier: ^20.0.8 + version: 20.0.8(@jest/environment@29.7.0)(@jest/fake-timers@29.7.0)(@jest/types@29.6.3)(jest-mock@29.7.0)(jest-util@29.7.0) '@mdx-js/loader': specifier: ^3.1.1 version: 3.1.1(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) @@ -384,22 +387,22 @@ importers: version: 4.2.0 '@storybook/addon-docs': specifier: 9.1.13 - version: 9.1.13(@types/react@19.1.17)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + version: 9.1.13(@types/react@19.1.17)(storybook@9.1.13(@testing-library/dom@10.4.1)) '@storybook/addon-links': specifier: 9.1.13 - version: 9.1.13(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + version: 9.1.13(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)) '@storybook/addon-onboarding': specifier: 9.1.13 - version: 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + version: 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)) '@storybook/addon-themes': specifier: 9.1.13 - version: 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + version: 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)) '@storybook/nextjs': specifier: 9.1.13 - version: 9.1.13(esbuild@0.25.0)(next@15.5.6(@babel/core@7.28.4)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2))(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(type-fest@4.2.0)(typescript@5.9.3)(uglify-js@3.19.3)(webpack-hot-middleware@2.26.1)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) + version: 9.1.13(esbuild@0.25.0)(next@15.5.6(@babel/core@7.28.4)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2))(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2)(storybook@9.1.13(@testing-library/dom@10.4.1))(type-fest@4.2.0)(typescript@5.9.3)(uglify-js@3.19.3)(webpack-hot-middleware@2.26.1)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) '@storybook/react': specifier: 9.1.13 - version: 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3) + version: 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3) '@testing-library/dom': specifier: ^10.4.1 version: 10.4.1 @@ -483,7 +486,7 @@ importers: version: 3.0.5(eslint@9.38.0(jiti@1.21.7)) eslint-plugin-storybook: specifier: ^9.1.13 - version: 9.1.13(eslint@9.38.0(jiti@1.21.7))(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3) + version: 9.1.13(eslint@9.38.0(jiti@1.21.7))(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3) eslint-plugin-tailwindcss: specifier: ^3.18.2 version: 3.18.2(tailwindcss@3.4.18(yaml@2.8.1)) @@ -498,7 +501,7 @@ importers: version: 29.7.0(@types/node@18.15.0)(ts-node@10.9.2(@types/node@18.15.0)(typescript@5.9.3)) knip: specifier: ^5.66.1 - version: 5.66.1(@types/node@18.15.0)(typescript@5.9.3) + version: 5.66.2(@types/node@18.15.0)(typescript@5.9.3) lint-staged: specifier: ^15.5.2 version: 15.5.2 @@ -516,10 +519,13 @@ importers: version: 1.93.2 storybook: specifier: 9.1.13 - version: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + version: 9.1.13(@testing-library/dom@10.4.1) tailwindcss: specifier: ^3.4.18 version: 3.4.18(yaml@2.8.1) + ts-node: + specifier: ^10.9.2 + version: 10.9.2(@types/node@18.15.0)(typescript@5.9.3) typescript: specifier: ^5.9.3 version: 5.9.3 @@ -694,6 +700,10 @@ packages: resolution: {integrity: sha512-D2hP9eA+Sqx1kBZgzxZh0y1trbuU+JoDkiEwqhQ36nodYqJwyEIhPSdMNd7lOm/4io72luTPWH20Yda0xOuUow==} engines: {node: '>=6.9.0'} + '@babel/helper-validator-identifier@7.28.5': + resolution: {integrity: sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==} + engines: {node: '>=6.9.0'} + '@babel/helper-validator-option@7.27.1': resolution: {integrity: sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==} engines: {node: '>=6.9.0'} @@ -711,6 +721,11 @@ packages: engines: {node: '>=6.0.0'} hasBin: true + '@babel/parser@7.28.5': + resolution: {integrity: sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==} + engines: {node: '>=6.0.0'} + hasBin: true + '@babel/plugin-bugfix-firefox-class-in-computed-class-key@7.27.1': resolution: {integrity: sha512-QPG3C9cCVRQLxAVwmefEmwdTanECuUBMQZ/ym5kiw3XKCGA7qkuQLcjWWHcrD/GKbn/WmJwaezfuuAOcyKlRPA==} engines: {node: '>=6.9.0'} @@ -1236,6 +1251,10 @@ packages: resolution: {integrity: sha512-bkFqkLhh3pMBUQQkpVgWDWq/lqzc2678eUyDlTBhRqhCHFguYYGM0Efga7tYk4TogG/3x0EEl66/OQ+WGbWB/Q==} engines: {node: '>=6.9.0'} + '@babel/types@7.28.5': + resolution: {integrity: sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==} + engines: {node: '>=6.9.0'} + '@bcoe/v8-coverage@0.2.3': resolution: {integrity: sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==} @@ -1295,11 +1314,11 @@ packages: resolution: {integrity: sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==} engines: {node: '>=10.0.0'} - '@emnapi/core@1.5.0': - resolution: {integrity: sha512-sbP8GzB1WDzacS8fgNPpHlp6C9VZe+SJP3F90W9rLemaQj2PzIuTEl1qDOYQf58YIpyjViI24y9aPWCjEzY2cg==} + '@emnapi/core@1.6.0': + resolution: {integrity: sha512-zq/ay+9fNIJJtJiZxdTnXS20PllcYMX3OE23ESc4HK/bdYu3cOWYVhsOhVnXALfU/uqJIxn5NBPd9z4v+SfoSg==} - '@emnapi/runtime@1.5.0': - resolution: {integrity: sha512-97/BJ3iXHww3djw6hYIfErCZFee7qCtrneuLa20UXFCOTCfBM2cvQHjWJ2EG0s0MtdNwInarqCTz35i4wWXHsQ==} + '@emnapi/runtime@1.6.0': + resolution: {integrity: sha512-obtUmAHTMjll499P+D9A3axeJFlhdjOWdKUNs/U6QIGT7V5RjcUW1xToAzjvmgTSQhDbYn/NwfTRoJcQ2rNBxA==} '@emnapi/wasi-threads@1.1.0': resolution: {integrity: sha512-WI0DdZ8xFSbgMjR1sFsKABJ/C5OnRrjT06JXbZKexJGrDuPTzZdDYfFlsgcCXCyf+suG5QU2e/y1Wo2V/OapLQ==} @@ -1324,306 +1343,150 @@ packages: cpu: [ppc64] os: [aix] - '@esbuild/aix-ppc64@0.25.11': - resolution: {integrity: sha512-Xt1dOL13m8u0WE8iplx9Ibbm+hFAO0GsU2P34UNoDGvZYkY8ifSiy6Zuc1lYxfG7svWE2fzqCUmFp5HCn51gJg==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [aix] - '@esbuild/android-arm64@0.25.0': resolution: {integrity: sha512-grvv8WncGjDSyUBjN9yHXNt+cq0snxXbDxy5pJtzMKGmmpPxeAmAhWxXI+01lU5rwZomDgD3kJwulEnhTRUd6g==} engines: {node: '>=18'} cpu: [arm64] os: [android] - '@esbuild/android-arm64@0.25.11': - resolution: {integrity: sha512-9slpyFBc4FPPz48+f6jyiXOx/Y4v34TUeDDXJpZqAWQn/08lKGeD8aDp9TMn9jDz2CiEuHwfhRmGBvpnd/PWIQ==} - engines: {node: '>=18'} - cpu: [arm64] - os: [android] - '@esbuild/android-arm@0.25.0': resolution: {integrity: sha512-PTyWCYYiU0+1eJKmw21lWtC+d08JDZPQ5g+kFyxP0V+es6VPPSUhM6zk8iImp2jbV6GwjX4pap0JFbUQN65X1g==} engines: {node: '>=18'} cpu: [arm] os: [android] - '@esbuild/android-arm@0.25.11': - resolution: {integrity: sha512-uoa7dU+Dt3HYsethkJ1k6Z9YdcHjTrSb5NUy66ZfZaSV8hEYGD5ZHbEMXnqLFlbBflLsl89Zke7CAdDJ4JI+Gg==} - engines: {node: '>=18'} - cpu: [arm] - os: [android] - '@esbuild/android-x64@0.25.0': resolution: {integrity: sha512-m/ix7SfKG5buCnxasr52+LI78SQ+wgdENi9CqyCXwjVR2X4Jkz+BpC3le3AoBPYTC9NHklwngVXvbJ9/Akhrfg==} engines: {node: '>=18'} cpu: [x64] os: [android] - '@esbuild/android-x64@0.25.11': - resolution: {integrity: sha512-Sgiab4xBjPU1QoPEIqS3Xx+R2lezu0LKIEcYe6pftr56PqPygbB7+szVnzoShbx64MUupqoE0KyRlN7gezbl8g==} - engines: {node: '>=18'} - cpu: [x64] - os: [android] - '@esbuild/darwin-arm64@0.25.0': resolution: {integrity: sha512-mVwdUb5SRkPayVadIOI78K7aAnPamoeFR2bT5nszFUZ9P8UpK4ratOdYbZZXYSqPKMHfS1wdHCJk1P1EZpRdvw==} engines: {node: '>=18'} cpu: [arm64] os: [darwin] - '@esbuild/darwin-arm64@0.25.11': - resolution: {integrity: sha512-VekY0PBCukppoQrycFxUqkCojnTQhdec0vevUL/EDOCnXd9LKWqD/bHwMPzigIJXPhC59Vd1WFIL57SKs2mg4w==} - engines: {node: '>=18'} - cpu: [arm64] - os: [darwin] - '@esbuild/darwin-x64@0.25.0': resolution: {integrity: sha512-DgDaYsPWFTS4S3nWpFcMn/33ZZwAAeAFKNHNa1QN0rI4pUjgqf0f7ONmXf6d22tqTY+H9FNdgeaAa+YIFUn2Rg==} engines: {node: '>=18'} cpu: [x64] os: [darwin] - '@esbuild/darwin-x64@0.25.11': - resolution: {integrity: sha512-+hfp3yfBalNEpTGp9loYgbknjR695HkqtY3d3/JjSRUyPg/xd6q+mQqIb5qdywnDxRZykIHs3axEqU6l1+oWEQ==} - engines: {node: '>=18'} - cpu: [x64] - os: [darwin] - '@esbuild/freebsd-arm64@0.25.0': resolution: {integrity: sha512-VN4ocxy6dxefN1MepBx/iD1dH5K8qNtNe227I0mnTRjry8tj5MRk4zprLEdG8WPyAPb93/e4pSgi1SoHdgOa4w==} engines: {node: '>=18'} cpu: [arm64] os: [freebsd] - '@esbuild/freebsd-arm64@0.25.11': - resolution: {integrity: sha512-CmKjrnayyTJF2eVuO//uSjl/K3KsMIeYeyN7FyDBjsR3lnSJHaXlVoAK8DZa7lXWChbuOk7NjAc7ygAwrnPBhA==} - engines: {node: '>=18'} - cpu: [arm64] - os: [freebsd] - '@esbuild/freebsd-x64@0.25.0': resolution: {integrity: sha512-mrSgt7lCh07FY+hDD1TxiTyIHyttn6vnjesnPoVDNmDfOmggTLXRv8Id5fNZey1gl/V2dyVK1VXXqVsQIiAk+A==} engines: {node: '>=18'} cpu: [x64] os: [freebsd] - '@esbuild/freebsd-x64@0.25.11': - resolution: {integrity: sha512-Dyq+5oscTJvMaYPvW3x3FLpi2+gSZTCE/1ffdwuM6G1ARang/mb3jvjxs0mw6n3Lsw84ocfo9CrNMqc5lTfGOw==} - engines: {node: '>=18'} - cpu: [x64] - os: [freebsd] - '@esbuild/linux-arm64@0.25.0': resolution: {integrity: sha512-9QAQjTWNDM/Vk2bgBl17yWuZxZNQIF0OUUuPZRKoDtqF2k4EtYbpyiG5/Dk7nqeK6kIJWPYldkOcBqjXjrUlmg==} engines: {node: '>=18'} cpu: [arm64] os: [linux] - '@esbuild/linux-arm64@0.25.11': - resolution: {integrity: sha512-Qr8AzcplUhGvdyUF08A1kHU3Vr2O88xxP0Tm8GcdVOUm25XYcMPp2YqSVHbLuXzYQMf9Bh/iKx7YPqECs6ffLA==} - engines: {node: '>=18'} - cpu: [arm64] - os: [linux] - '@esbuild/linux-arm@0.25.0': resolution: {integrity: sha512-vkB3IYj2IDo3g9xX7HqhPYxVkNQe8qTK55fraQyTzTX/fxaDtXiEnavv9geOsonh2Fd2RMB+i5cbhu2zMNWJwg==} engines: {node: '>=18'} cpu: [arm] os: [linux] - '@esbuild/linux-arm@0.25.11': - resolution: {integrity: sha512-TBMv6B4kCfrGJ8cUPo7vd6NECZH/8hPpBHHlYI3qzoYFvWu2AdTvZNuU/7hsbKWqu/COU7NIK12dHAAqBLLXgw==} - engines: {node: '>=18'} - cpu: [arm] - os: [linux] - '@esbuild/linux-ia32@0.25.0': resolution: {integrity: sha512-43ET5bHbphBegyeqLb7I1eYn2P/JYGNmzzdidq/w0T8E2SsYL1U6un2NFROFRg1JZLTzdCoRomg8Rvf9M6W6Gg==} engines: {node: '>=18'} cpu: [ia32] os: [linux] - '@esbuild/linux-ia32@0.25.11': - resolution: {integrity: sha512-TmnJg8BMGPehs5JKrCLqyWTVAvielc615jbkOirATQvWWB1NMXY77oLMzsUjRLa0+ngecEmDGqt5jiDC6bfvOw==} - engines: {node: '>=18'} - cpu: [ia32] - os: [linux] - '@esbuild/linux-loong64@0.25.0': resolution: {integrity: sha512-fC95c/xyNFueMhClxJmeRIj2yrSMdDfmqJnyOY4ZqsALkDrrKJfIg5NTMSzVBr5YW1jf+l7/cndBfP3MSDpoHw==} engines: {node: '>=18'} cpu: [loong64] os: [linux] - '@esbuild/linux-loong64@0.25.11': - resolution: {integrity: sha512-DIGXL2+gvDaXlaq8xruNXUJdT5tF+SBbJQKbWy/0J7OhU8gOHOzKmGIlfTTl6nHaCOoipxQbuJi7O++ldrxgMw==} - engines: {node: '>=18'} - cpu: [loong64] - os: [linux] - '@esbuild/linux-mips64el@0.25.0': resolution: {integrity: sha512-nkAMFju7KDW73T1DdH7glcyIptm95a7Le8irTQNO/qtkoyypZAnjchQgooFUDQhNAy4iu08N79W4T4pMBwhPwQ==} engines: {node: '>=18'} cpu: [mips64el] os: [linux] - '@esbuild/linux-mips64el@0.25.11': - resolution: {integrity: sha512-Osx1nALUJu4pU43o9OyjSCXokFkFbyzjXb6VhGIJZQ5JZi8ylCQ9/LFagolPsHtgw6himDSyb5ETSfmp4rpiKQ==} - engines: {node: '>=18'} - cpu: [mips64el] - os: [linux] - '@esbuild/linux-ppc64@0.25.0': resolution: {integrity: sha512-NhyOejdhRGS8Iwv+KKR2zTq2PpysF9XqY+Zk77vQHqNbo/PwZCzB5/h7VGuREZm1fixhs4Q/qWRSi5zmAiO4Fw==} engines: {node: '>=18'} cpu: [ppc64] os: [linux] - '@esbuild/linux-ppc64@0.25.11': - resolution: {integrity: sha512-nbLFgsQQEsBa8XSgSTSlrnBSrpoWh7ioFDUmwo158gIm5NNP+17IYmNWzaIzWmgCxq56vfr34xGkOcZ7jX6CPw==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [linux] - '@esbuild/linux-riscv64@0.25.0': resolution: {integrity: sha512-5S/rbP5OY+GHLC5qXp1y/Mx//e92L1YDqkiBbO9TQOvuFXM+iDqUNG5XopAnXoRH3FjIUDkeGcY1cgNvnXp/kA==} engines: {node: '>=18'} cpu: [riscv64] os: [linux] - '@esbuild/linux-riscv64@0.25.11': - resolution: {integrity: sha512-HfyAmqZi9uBAbgKYP1yGuI7tSREXwIb438q0nqvlpxAOs3XnZ8RsisRfmVsgV486NdjD7Mw2UrFSw51lzUk1ww==} - engines: {node: '>=18'} - cpu: [riscv64] - os: [linux] - '@esbuild/linux-s390x@0.25.0': resolution: {integrity: sha512-XM2BFsEBz0Fw37V0zU4CXfcfuACMrppsMFKdYY2WuTS3yi8O1nFOhil/xhKTmE1nPmVyvQJjJivgDT+xh8pXJA==} engines: {node: '>=18'} cpu: [s390x] os: [linux] - '@esbuild/linux-s390x@0.25.11': - resolution: {integrity: sha512-HjLqVgSSYnVXRisyfmzsH6mXqyvj0SA7pG5g+9W7ESgwA70AXYNpfKBqh1KbTxmQVaYxpzA/SvlB9oclGPbApw==} - engines: {node: '>=18'} - cpu: [s390x] - os: [linux] - '@esbuild/linux-x64@0.25.0': resolution: {integrity: sha512-9yl91rHw/cpwMCNytUDxwj2XjFpxML0y9HAOH9pNVQDpQrBxHy01Dx+vaMu0N1CKa/RzBD2hB4u//nfc+Sd3Cw==} engines: {node: '>=18'} cpu: [x64] os: [linux] - '@esbuild/linux-x64@0.25.11': - resolution: {integrity: sha512-HSFAT4+WYjIhrHxKBwGmOOSpphjYkcswF449j6EjsjbinTZbp8PJtjsVK1XFJStdzXdy/jaddAep2FGY+wyFAQ==} - engines: {node: '>=18'} - cpu: [x64] - os: [linux] - '@esbuild/netbsd-arm64@0.25.0': resolution: {integrity: sha512-RuG4PSMPFfrkH6UwCAqBzauBWTygTvb1nxWasEJooGSJ/NwRw7b2HOwyRTQIU97Hq37l3npXoZGYMy3b3xYvPw==} engines: {node: '>=18'} cpu: [arm64] os: [netbsd] - '@esbuild/netbsd-arm64@0.25.11': - resolution: {integrity: sha512-hr9Oxj1Fa4r04dNpWr3P8QKVVsjQhqrMSUzZzf+LZcYjZNqhA3IAfPQdEh1FLVUJSiu6sgAwp3OmwBfbFgG2Xg==} - engines: {node: '>=18'} - cpu: [arm64] - os: [netbsd] - '@esbuild/netbsd-x64@0.25.0': resolution: {integrity: sha512-jl+qisSB5jk01N5f7sPCsBENCOlPiS/xptD5yxOx2oqQfyourJwIKLRA2yqWdifj3owQZCL2sn6o08dBzZGQzA==} engines: {node: '>=18'} cpu: [x64] os: [netbsd] - '@esbuild/netbsd-x64@0.25.11': - resolution: {integrity: sha512-u7tKA+qbzBydyj0vgpu+5h5AeudxOAGncb8N6C9Kh1N4n7wU1Xw1JDApsRjpShRpXRQlJLb9wY28ELpwdPcZ7A==} - engines: {node: '>=18'} - cpu: [x64] - os: [netbsd] - '@esbuild/openbsd-arm64@0.25.0': resolution: {integrity: sha512-21sUNbq2r84YE+SJDfaQRvdgznTD8Xc0oc3p3iW/a1EVWeNj/SdUCbm5U0itZPQYRuRTW20fPMWMpcrciH2EJw==} engines: {node: '>=18'} cpu: [arm64] os: [openbsd] - '@esbuild/openbsd-arm64@0.25.11': - resolution: {integrity: sha512-Qq6YHhayieor3DxFOoYM1q0q1uMFYb7cSpLD2qzDSvK1NAvqFi8Xgivv0cFC6J+hWVw2teCYltyy9/m/14ryHg==} - engines: {node: '>=18'} - cpu: [arm64] - os: [openbsd] - '@esbuild/openbsd-x64@0.25.0': resolution: {integrity: sha512-2gwwriSMPcCFRlPlKx3zLQhfN/2WjJ2NSlg5TKLQOJdV0mSxIcYNTMhk3H3ulL/cak+Xj0lY1Ym9ysDV1igceg==} engines: {node: '>=18'} cpu: [x64] os: [openbsd] - '@esbuild/openbsd-x64@0.25.11': - resolution: {integrity: sha512-CN+7c++kkbrckTOz5hrehxWN7uIhFFlmS/hqziSFVWpAzpWrQoAG4chH+nN3Be+Kzv/uuo7zhX716x3Sn2Jduw==} - engines: {node: '>=18'} - cpu: [x64] - os: [openbsd] - - '@esbuild/openharmony-arm64@0.25.11': - resolution: {integrity: sha512-rOREuNIQgaiR+9QuNkbkxubbp8MSO9rONmwP5nKncnWJ9v5jQ4JxFnLu4zDSRPf3x4u+2VN4pM4RdyIzDty/wQ==} - engines: {node: '>=18'} - cpu: [arm64] - os: [openharmony] - '@esbuild/sunos-x64@0.25.0': resolution: {integrity: sha512-bxI7ThgLzPrPz484/S9jLlvUAHYMzy6I0XiU1ZMeAEOBcS0VePBFxh1JjTQt3Xiat5b6Oh4x7UC7IwKQKIJRIg==} engines: {node: '>=18'} cpu: [x64] os: [sunos] - '@esbuild/sunos-x64@0.25.11': - resolution: {integrity: sha512-nq2xdYaWxyg9DcIyXkZhcYulC6pQ2FuCgem3LI92IwMgIZ69KHeY8T4Y88pcwoLIjbed8n36CyKoYRDygNSGhA==} - engines: {node: '>=18'} - cpu: [x64] - os: [sunos] - '@esbuild/win32-arm64@0.25.0': resolution: {integrity: sha512-ZUAc2YK6JW89xTbXvftxdnYy3m4iHIkDtK3CLce8wg8M2L+YZhIvO1DKpxrd0Yr59AeNNkTiic9YLf6FTtXWMw==} engines: {node: '>=18'} cpu: [arm64] os: [win32] - '@esbuild/win32-arm64@0.25.11': - resolution: {integrity: sha512-3XxECOWJq1qMZ3MN8srCJ/QfoLpL+VaxD/WfNRm1O3B4+AZ/BnLVgFbUV3eiRYDMXetciH16dwPbbHqwe1uU0Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [win32] - '@esbuild/win32-ia32@0.25.0': resolution: {integrity: sha512-eSNxISBu8XweVEWG31/JzjkIGbGIJN/TrRoiSVZwZ6pkC6VX4Im/WV2cz559/TXLcYbcrDN8JtKgd9DJVIo8GA==} engines: {node: '>=18'} cpu: [ia32] os: [win32] - '@esbuild/win32-ia32@0.25.11': - resolution: {integrity: sha512-3ukss6gb9XZ8TlRyJlgLn17ecsK4NSQTmdIXRASVsiS2sQ6zPPZklNJT5GR5tE/MUarymmy8kCEf5xPCNCqVOA==} - engines: {node: '>=18'} - cpu: [ia32] - os: [win32] - '@esbuild/win32-x64@0.25.0': resolution: {integrity: sha512-ZENoHJBxA20C2zFzh6AI4fT6RraMzjYw4xKWemRTRmRVtN9c5DcH9r/f2ihEkMjOW5eGgrwCslG/+Y/3bL+DHQ==} engines: {node: '>=18'} cpu: [x64] os: [win32] - '@esbuild/win32-x64@0.25.11': - resolution: {integrity: sha512-D7Hpz6A2L4hzsRpPaCYkQnGOotdUpDzSGRIv9I+1ITdHROSFUWW95ZPZWQmGka1Fg7W3zFJowyn9WGwMJ0+KPA==} - engines: {node: '>=18'} - cpu: [x64] - os: [win32] - '@eslint-community/eslint-plugin-eslint-comments@4.5.0': resolution: {integrity: sha512-MAhuTKlr4y/CE3WYX26raZjy+I/kS2PLKSzvfmDCGrBLTFHOYwqROZdr4XwPgXwX3K9rjzMr4pSmUWGnzsUyMg==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} @@ -1707,8 +1570,8 @@ packages: resolution: {integrity: sha512-UZ1VpFvXf9J06YG9xQBdnzU+kthors6KjhMAl6f4gH4usHyh31rUf2DLGInT8RFYIReYXNSydgPY0V2LuWgl7A==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@eslint/markdown@7.4.0': - resolution: {integrity: sha512-VQykmMjBb4tQoJOXVWXa+oQbQeCZlE7W3rAsOpmtpKLvJd75saZZ04PVVs7+zgMDJGghd4/gyFV6YlvdJFaeNQ==} + '@eslint/markdown@7.4.1': + resolution: {integrity: sha512-fhcQcylVqgb7GLPr2+6hlDQXK4J3d/fPY6qzk9/i7IYtQkIr15NKI5Zg39Dv2cV/bn5J0Znm69rmu9vJI/7Tlw==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} '@eslint/object-schema@2.1.7': @@ -1719,10 +1582,6 @@ packages: resolution: {integrity: sha512-Ul5l+lHEcw3L5+k8POx6r74mxEYKG5kOb6Xpy2gCRW6zweT6TEhAf8vhxGgjhqrd/VO/Dirhsb+1hNpD1ue9hw==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@eslint/plugin-kit@0.3.5': - resolution: {integrity: sha512-Z5kJ+wU3oA7MMIqVR9tyZRtjYPr4OC004Q4Rw7pgOKUOKkJfZ3O24nz3WYfGRpMDNmcOi3TwQOmgm7B7Tpii0w==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@floating-ui/core@1.7.3': resolution: {integrity: sha512-sGnvb5dmrJaKEZ+LDIpguvdX3bDlEllmv4/ClQ9awcmCZrlx5jQyyMWFM5kBI+EyNOCDDiKk8il0zeuX3Zlg/w==} @@ -1753,8 +1612,8 @@ packages: '@formatjs/intl-localematcher@0.5.10': resolution: {integrity: sha512-af3qATX+m4Rnd9+wHcjJ4w2ijq+rAVP3CCinJQvFv1kgSu1W6jypUmvleJxcewdxmutM8dmIRZFxO/IQBZmP2Q==} - '@happy-dom/jest-environment@20.0.7': - resolution: {integrity: sha512-f7cvUghxPIUS8L21uSNab1GYXPr6+7FvltpsWyzrSzhSbjhDWr5Ixcy5bv2DqaQEhAKIQ7SYBYD5n4+SSHwfig==} + '@happy-dom/jest-environment@20.0.8': + resolution: {integrity: sha512-e8/c1EW+vUF7MFTZZtPbWrD3rStPnx3X8M4pAaOU++x+1lsXr/bsdoLoHs6bQ2kEZyPRhate3sC6MnpVD/O/9A==} engines: {node: '>=20.0.0'} peerDependencies: '@jest/environment': '>=25.0.0' @@ -2444,98 +2303,98 @@ packages: '@octokit/types@14.1.0': resolution: {integrity: sha512-1y6DgTy8Jomcpu33N+p5w58l6xyt55Ar2I91RPiIA0xCJBXyUAhXCcmZaDWSANiha7R9a6qJJ2CRomGPZ6f46g==} - '@oxc-resolver/binding-android-arm-eabi@11.10.0': - resolution: {integrity: sha512-qvSSjeeBvYh3KlpMwDbLr0m/bmEfEzaAv2yW4RnYDGrsFVgTHlNc3WzQSji0+Bf2g3kLgyZ5pwylaJpS9baUIA==} + '@oxc-resolver/binding-android-arm-eabi@11.11.0': + resolution: {integrity: sha512-aN0UJg1xr0N1dADQ135z4p3bP9AYAUN1Ey2VvLMK6IwWYIJGWpKT+cr1l3AiyBeLK8QZyFDb4IDU8LHgjO9TDQ==} cpu: [arm] os: [android] - '@oxc-resolver/binding-android-arm64@11.10.0': - resolution: {integrity: sha512-rjiCqkhH1di5Sb/KpOmuC/1OCGZVDdUyVIxxPsmzkdgrTgS6Of5cwOHTBVNxXuVdlIMz0swN8wrmqUM9jspPAQ==} + '@oxc-resolver/binding-android-arm64@11.11.0': + resolution: {integrity: sha512-FckvvMclo8CSJqQjKpHueIIbKrg9L638NKWQTiJQaD8W9F61h8hTjF8+QFLlCHh6R9RcE5roVHdkkiBKHlB2Zw==} cpu: [arm64] os: [android] - '@oxc-resolver/binding-darwin-arm64@11.10.0': - resolution: {integrity: sha512-qr2+vw0BKxZVuaw3Ssbzfe0999FYs5BkKqezP8ocwYE9pJUC4hNlWUWhGLDxj0tBSjMEFvWQNF7IxCeZk6nzKw==} + '@oxc-resolver/binding-darwin-arm64@11.11.0': + resolution: {integrity: sha512-7ZcpgaXSBnwRHM1YR8Vazq7mCTtGdYRvM7k46CscA+oipCVqmI4LbW2wLsc6HVjqX+SM/KPOfFGoGjEgmQPFTQ==} cpu: [arm64] os: [darwin] - '@oxc-resolver/binding-darwin-x64@11.10.0': - resolution: {integrity: sha512-2XFEd89yVnnkk7u0LACdXsiHDN3rMthzcdSHj2VROaItiAW6qfKy+SJwLK94lYCVv9nFjxJUVHiVJUsKIn70tQ==} + '@oxc-resolver/binding-darwin-x64@11.11.0': + resolution: {integrity: sha512-Wsd1JWORokMmOKrR4t4jxpwYEWG11+AHWu9bdzjCO5EIyi0AuNpPIAEcEFCP9FNd0h8c+VUYbMRU/GooD2zOIg==} cpu: [x64] os: [darwin] - '@oxc-resolver/binding-freebsd-x64@11.10.0': - resolution: {integrity: sha512-EHapmlf+bg92Pf3+0E0nYSKQgQ5u2V++KXB0WTushFJSU+k6gXEL/P/y1QwKqzJ986Q14YWHh7IiT/nQvpaz4Q==} + '@oxc-resolver/binding-freebsd-x64@11.11.0': + resolution: {integrity: sha512-YX+W10kHrMouu/+Y+rqJdCWO3dFBKM1DIils30PHsmXWp1v+ZZvhibaST2BP6zrWkWquZ8pMmsObD6N10lLgiA==} cpu: [x64] os: [freebsd] - '@oxc-resolver/binding-linux-arm-gnueabihf@11.10.0': - resolution: {integrity: sha512-NhSAeelg0EU4ymM8XrUfGJL74jBHs2Q3WdVbXIve+ROge0UAB7yXpk40u7quIOmbyqAEUp/QPlhtEmWc+lWcPg==} + '@oxc-resolver/binding-linux-arm-gnueabihf@11.11.0': + resolution: {integrity: sha512-UAhlhVkW2ui98bClmEkDLKQz4XBSccxMahG7rMeX2RepS2QByAWxYFFThaNbHtBSB+B4Rc1hudkihq8grQkU3g==} cpu: [arm] os: [linux] - '@oxc-resolver/binding-linux-arm-musleabihf@11.10.0': - resolution: {integrity: sha512-9rjZigo5/92O3jayjucIdhhq4eJBgf61K9UZZF1r1uoIhS4i0wz7W29gMWkCVYbwZAfkHxfmTn3zu8Vv34NvUQ==} + '@oxc-resolver/binding-linux-arm-musleabihf@11.11.0': + resolution: {integrity: sha512-5pEliabSEiimXz/YyPxzyBST82q8PbM6BoEMS8kOyaDbEBuzTr7pWU1U0F7ILGBFjJmHaj3N7IAhQgeXdpdySg==} cpu: [arm] os: [linux] - '@oxc-resolver/binding-linux-arm64-gnu@11.10.0': - resolution: {integrity: sha512-73pz+sYfPfMzl8OVdjsWJXu5LO868LBpy8M/a/m4a7HUREwBz1/CK59ifxhbIkIeAv2ZkhwKiouFxsKmCsQRrw==} + '@oxc-resolver/binding-linux-arm64-gnu@11.11.0': + resolution: {integrity: sha512-CiyufPFIOJrW/HovAMGsH0AbV7BSCb0oE0KDtt7z1+e+qsDo7HRlTSnqE3JbNuhJRg3Cz/j7qEYzgGqco9SE4Q==} cpu: [arm64] os: [linux] - '@oxc-resolver/binding-linux-arm64-musl@11.10.0': - resolution: {integrity: sha512-s8AMNkiguFn2XJtnAaSHl+ak97Zwkq6biouUNuApDRZh34ckAjWxPTQRhUZLCFybNxgZtwVbglVQv0BJYieIXg==} + '@oxc-resolver/binding-linux-arm64-musl@11.11.0': + resolution: {integrity: sha512-w07MfGtDLZV0rISdXl2cGASxD/sRrrR93Qd4q27O2Hsky4MGbLw94trbzhmAkc7OKoJI0iDg1217i3jfxmVk1Q==} cpu: [arm64] os: [linux] - '@oxc-resolver/binding-linux-ppc64-gnu@11.10.0': - resolution: {integrity: sha512-70eHfsX9Xw+wGqmwFhlIxT/LhzGDlnI4ECQ7w0VLZsYpAUjRiQPUQCDKkfP65ikzHPSLeY8pARKVIc2gdC0HEA==} + '@oxc-resolver/binding-linux-ppc64-gnu@11.11.0': + resolution: {integrity: sha512-gzM+ZfIjfcCofwX/m1eLCoTT+3T70QLWaKDOW5Hf3+ddLlxMEVRIQtUoRsp0e/VFanr7u7VKS57TxhkRubseNg==} cpu: [ppc64] os: [linux] - '@oxc-resolver/binding-linux-riscv64-gnu@11.10.0': - resolution: {integrity: sha512-geibi+L5hKmDwZ9iLEUzuvRG4o6gZWB8shlNBLiKnGtYD5SMAvCcJiHpz1Sf6ESm8laXjiIf6T/pTZZpaeStyw==} + '@oxc-resolver/binding-linux-riscv64-gnu@11.11.0': + resolution: {integrity: sha512-oCR0ImJQhIwmqwNShsRT0tGIgKF5/H4nhtIEkQAQ9bLzMgjtRqIrZ3DtGHqd7w58zhXWfIZdyPNF9IrSm+J/fQ==} cpu: [riscv64] os: [linux] - '@oxc-resolver/binding-linux-riscv64-musl@11.10.0': - resolution: {integrity: sha512-oL1B0jGu9vYoQKyJiMvjtuxDzmV9P8M/xdu6wjUjvaGC/gIwvhILzlHgD3SMtFJJhzLVf4HPmYAF7BsLWvTugA==} + '@oxc-resolver/binding-linux-riscv64-musl@11.11.0': + resolution: {integrity: sha512-MjCEqsUzXMfWPfsEUX+UXttzXz6xiNU11r7sj00C5og/UCyqYw1OjrbC/B1f/dloDpTn0rd4xy6c/LTvVQl2tg==} cpu: [riscv64] os: [linux] - '@oxc-resolver/binding-linux-s390x-gnu@11.10.0': - resolution: {integrity: sha512-Sj6ooR4RZ+04SSc/iV7oK8C2TxoWzJbD5yirsF64ULFukTvQHz99ImjtwgauBUnR+3loyca3s6o8DiAmqHaxAw==} + '@oxc-resolver/binding-linux-s390x-gnu@11.11.0': + resolution: {integrity: sha512-4TaTX7gT3357vWQsTe3IfDtWyJNe0FejypQ4ngwxB3v1IVaW6KAUt0huSvx/tmj+YWxd3zzXdWd8AzW0jo6dpg==} cpu: [s390x] os: [linux] - '@oxc-resolver/binding-linux-x64-gnu@11.10.0': - resolution: {integrity: sha512-wH5nPRgIaEhuOD9M70NujV91FscboRkNf38wKAYiy9xuKeVsc43JzFqvmgxU1vXsKwUJBc/qMt4nFNluLXwVzw==} + '@oxc-resolver/binding-linux-x64-gnu@11.11.0': + resolution: {integrity: sha512-ch1o3+tBra9vmrgXqrufVmYnvRPFlyUb7JWs/VXndBmyNSuP2KP+guAUrC0fr2aSGoOQOasAiZza7MTFU7Vrxg==} cpu: [x64] os: [linux] - '@oxc-resolver/binding-linux-x64-musl@11.10.0': - resolution: {integrity: sha512-rDrv1Joh6hAidV/hixAA1+6keNr1aJA3rUU6VD8mqTedbUMV1CdQJ55f9UmQZn0nO35tQvwF0eLBNmumErCNLw==} + '@oxc-resolver/binding-linux-x64-musl@11.11.0': + resolution: {integrity: sha512-llTdl2gJAqXaGV7iV1w5BVlqXACcoT1YD3o840pCQx1ZmKKAAz7ydPnTjYVdkGImXNWPOIWJixHW0ryDm4Mx7w==} cpu: [x64] os: [linux] - '@oxc-resolver/binding-wasm32-wasi@11.10.0': - resolution: {integrity: sha512-VE+fuYPMqObhwEoLOUp9UgebrMFBBCuvCBY+auk+o3bFWOYXLpvCa5PzC4ttF7gOotQD/TWqbVWtfOh0CdBSHw==} + '@oxc-resolver/binding-wasm32-wasi@11.11.0': + resolution: {integrity: sha512-cROavohP0nX91NtIVVgOTugqoxlUSNxI9j7MD+B7fmD3gEFl8CVyTamR0/p6loDxLv51bQYTHRKn/ZYTd3ENzw==} engines: {node: '>=14.0.0'} cpu: [wasm32] - '@oxc-resolver/binding-win32-arm64-msvc@11.10.0': - resolution: {integrity: sha512-M70Fr5P1SnQY4vm7ZTeodE27mDV6zqxLkQMHF4t43xt55dIFIlHiRTgCzykiI9ggan3M1YWffLeB97Q3X2yxSg==} + '@oxc-resolver/binding-win32-arm64-msvc@11.11.0': + resolution: {integrity: sha512-6amVs34yHmxE6Q3CtTPXnSvIYGqwQJ/lVVRYccLzg9smge3WJ1knyBV5jpKKayp0n316uPYzB4EgEbgcuRvrPw==} cpu: [arm64] os: [win32] - '@oxc-resolver/binding-win32-ia32-msvc@11.10.0': - resolution: {integrity: sha512-UJfRwzXAAIduNJa0cZlwT8L8eAOSX85VfKQ0i0NCJWNjwFzjeeOpvd/vNXMd1jmYU22a8fulFX3k8AzdwI7wYw==} + '@oxc-resolver/binding-win32-ia32-msvc@11.11.0': + resolution: {integrity: sha512-v/IZ5s2/3auHUoi0t6Ea1CDsWxrE9BvgvbDcJ04QX+nEbmTBazWPZeLsH8vWkRAh8EUKCZHXxjQsPhEH5Yk5pQ==} cpu: [ia32] os: [win32] - '@oxc-resolver/binding-win32-x64-msvc@11.10.0': - resolution: {integrity: sha512-Q8gwXHjDeEokECEFCECkJW1OEOEgfFUGoLZs88jDpZ/QmdBklH/SbMLKJdYeIPztQ6HD069GAVPnP3WcXyHoUA==} + '@oxc-resolver/binding-win32-x64-msvc@11.11.0': + resolution: {integrity: sha512-qvm+IQ6r2q4HZitSV69O+OmvCD1y4pH7SbhR6lPwLsfZS5QRHS8V20VHxmG1jJzSPPw7S8Bb1rdNcxDSqc4bYA==} cpu: [x64] os: [win32] @@ -2942,116 +2801,6 @@ packages: peerDependencies: rollup: ^1.20.0||^2.0.0 - '@rollup/rollup-android-arm-eabi@4.52.5': - resolution: {integrity: sha512-8c1vW4ocv3UOMp9K+gToY5zL2XiiVw3k7f1ksf4yO1FlDFQ1C2u72iACFnSOceJFsWskc2WZNqeRhFRPzv+wtQ==} - cpu: [arm] - os: [android] - - '@rollup/rollup-android-arm64@4.52.5': - resolution: {integrity: sha512-mQGfsIEFcu21mvqkEKKu2dYmtuSZOBMmAl5CFlPGLY94Vlcm+zWApK7F/eocsNzp8tKmbeBP8yXyAbx0XHsFNA==} - cpu: [arm64] - os: [android] - - '@rollup/rollup-darwin-arm64@4.52.5': - resolution: {integrity: sha512-takF3CR71mCAGA+v794QUZ0b6ZSrgJkArC+gUiG6LB6TQty9T0Mqh3m2ImRBOxS2IeYBo4lKWIieSvnEk2OQWA==} - cpu: [arm64] - os: [darwin] - - '@rollup/rollup-darwin-x64@4.52.5': - resolution: {integrity: sha512-W901Pla8Ya95WpxDn//VF9K9u2JbocwV/v75TE0YIHNTbhqUTv9w4VuQ9MaWlNOkkEfFwkdNhXgcLqPSmHy0fA==} - cpu: [x64] - os: [darwin] - - '@rollup/rollup-freebsd-arm64@4.52.5': - resolution: {integrity: sha512-QofO7i7JycsYOWxe0GFqhLmF6l1TqBswJMvICnRUjqCx8b47MTo46W8AoeQwiokAx3zVryVnxtBMcGcnX12LvA==} - cpu: [arm64] - os: [freebsd] - - '@rollup/rollup-freebsd-x64@4.52.5': - resolution: {integrity: sha512-jr21b/99ew8ujZubPo9skbrItHEIE50WdV86cdSoRkKtmWa+DDr6fu2c/xyRT0F/WazZpam6kk7IHBerSL7LDQ==} - cpu: [x64] - os: [freebsd] - - '@rollup/rollup-linux-arm-gnueabihf@4.52.5': - resolution: {integrity: sha512-PsNAbcyv9CcecAUagQefwX8fQn9LQ4nZkpDboBOttmyffnInRy8R8dSg6hxxl2Re5QhHBf6FYIDhIj5v982ATQ==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm-musleabihf@4.52.5': - resolution: {integrity: sha512-Fw4tysRutyQc/wwkmcyoqFtJhh0u31K+Q6jYjeicsGJJ7bbEq8LwPWV/w0cnzOqR2m694/Af6hpFayLJZkG2VQ==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm64-gnu@4.52.5': - resolution: {integrity: sha512-a+3wVnAYdQClOTlyapKmyI6BLPAFYs0JM8HRpgYZQO02rMR09ZcV9LbQB+NL6sljzG38869YqThrRnfPMCDtZg==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-arm64-musl@4.52.5': - resolution: {integrity: sha512-AvttBOMwO9Pcuuf7m9PkC1PUIKsfaAJ4AYhy944qeTJgQOqJYJ9oVl2nYgY7Rk0mkbsuOpCAYSs6wLYB2Xiw0Q==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-loong64-gnu@4.52.5': - resolution: {integrity: sha512-DkDk8pmXQV2wVrF6oq5tONK6UHLz/XcEVow4JTTerdeV1uqPeHxwcg7aFsfnSm9L+OO8WJsWotKM2JJPMWrQtA==} - cpu: [loong64] - os: [linux] - - '@rollup/rollup-linux-ppc64-gnu@4.52.5': - resolution: {integrity: sha512-W/b9ZN/U9+hPQVvlGwjzi+Wy4xdoH2I8EjaCkMvzpI7wJUs8sWJ03Rq96jRnHkSrcHTpQe8h5Tg3ZzUPGauvAw==} - cpu: [ppc64] - os: [linux] - - '@rollup/rollup-linux-riscv64-gnu@4.52.5': - resolution: {integrity: sha512-sjQLr9BW7R/ZiXnQiWPkErNfLMkkWIoCz7YMn27HldKsADEKa5WYdobaa1hmN6slu9oWQbB6/jFpJ+P2IkVrmw==} - cpu: [riscv64] - os: [linux] - - '@rollup/rollup-linux-riscv64-musl@4.52.5': - resolution: {integrity: sha512-hq3jU/kGyjXWTvAh2awn8oHroCbrPm8JqM7RUpKjalIRWWXE01CQOf/tUNWNHjmbMHg/hmNCwc/Pz3k1T/j/Lg==} - cpu: [riscv64] - os: [linux] - - '@rollup/rollup-linux-s390x-gnu@4.52.5': - resolution: {integrity: sha512-gn8kHOrku8D4NGHMK1Y7NA7INQTRdVOntt1OCYypZPRt6skGbddska44K8iocdpxHTMMNui5oH4elPH4QOLrFQ==} - cpu: [s390x] - os: [linux] - - '@rollup/rollup-linux-x64-gnu@4.52.5': - resolution: {integrity: sha512-hXGLYpdhiNElzN770+H2nlx+jRog8TyynpTVzdlc6bndktjKWyZyiCsuDAlpd+j+W+WNqfcyAWz9HxxIGfZm1Q==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-linux-x64-musl@4.52.5': - resolution: {integrity: sha512-arCGIcuNKjBoKAXD+y7XomR9gY6Mw7HnFBv5Rw7wQRvwYLR7gBAgV7Mb2QTyjXfTveBNFAtPt46/36vV9STLNg==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-openharmony-arm64@4.52.5': - resolution: {integrity: sha512-QoFqB6+/9Rly/RiPjaomPLmR/13cgkIGfA40LHly9zcH1S0bN2HVFYk3a1eAyHQyjs3ZJYlXvIGtcCs5tko9Cw==} - cpu: [arm64] - os: [openharmony] - - '@rollup/rollup-win32-arm64-msvc@4.52.5': - resolution: {integrity: sha512-w0cDWVR6MlTstla1cIfOGyl8+qb93FlAVutcor14Gf5Md5ap5ySfQ7R9S/NjNaMLSFdUnKGEasmVnu3lCMqB7w==} - cpu: [arm64] - os: [win32] - - '@rollup/rollup-win32-ia32-msvc@4.52.5': - resolution: {integrity: sha512-Aufdpzp7DpOTULJCuvzqcItSGDH73pF3ko/f+ckJhxQyHtp67rHw3HMNxoIdDMUITJESNE6a8uh4Lo4SLouOUg==} - cpu: [ia32] - os: [win32] - - '@rollup/rollup-win32-x64-gnu@4.52.5': - resolution: {integrity: sha512-UGBUGPFp1vkj6p8wCRraqNhqwX/4kNQPS57BCFc8wYh0g94iVIW33wJtQAx3G7vrjjNtRaxiMUylM0ktp/TRSQ==} - cpu: [x64] - os: [win32] - - '@rollup/rollup-win32-x64-msvc@4.52.5': - resolution: {integrity: sha512-TAcgQh2sSkykPRWLrdyy2AiceMckNf5loITqXxFI5VuQjS5tSuw3WlwdN8qv8vzjLAUTvYaH/mVjSFpbkFbpTg==} - cpu: [x64] - os: [win32] - '@sentry-internal/browser-utils@8.55.0': resolution: {integrity: sha512-ROgqtQfpH/82AQIpESPqPQe0UyWywKJsmVIqi3c5Fh+zkds5LUxnssTj3yNd1x+kxaPDVB023jAP+3ibNgeNDw==} engines: {node: '>=14.18'} @@ -3341,8 +3090,8 @@ packages: '@types/cacheable-request@6.0.3': resolution: {integrity: sha512-IQ3EbTzGxIigb1I3qPZc1rWJnH0BmSKv5QYTalEwweFvyBDLSAe24zP0le/hyi7ecGfZVlIVAg4BZqb8WBwKqw==} - '@types/chai@5.2.2': - resolution: {integrity: sha512-8kB30R7Hwqf40JPiKhVzodJs2Qc1ZJ5zuT3uzw5Hq/dhNCl3G3l83jfpdI1e20BP348+fV7VIL/+FxaXkqBmWg==} + '@types/chai@5.2.3': + resolution: {integrity: sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==} '@types/d3-array@3.2.2': resolution: {integrity: sha512-hOLWVbm7uRza0BYXpIIW5pxfrKe0W+D5lrFiAEYR+pb6w3N2SwSMaJbXdUfSEv+dT4MfHBLtn5js0LAWaO6otw==} @@ -3531,8 +3280,8 @@ packages: '@types/node@18.15.0': resolution: {integrity: sha512-z6nr0TTEOBGkzLGmbypWOGnpSpSIBorEhC4L+4HeQ2iezKCi4f77kyslRwvHeNitymGQ+oFyIWGP96l/DPSV9w==} - '@types/node@20.19.22': - resolution: {integrity: sha512-hRnu+5qggKDSyWHlnmThnUqg62l29Aj/6vcYgUaSFL9oc7DVjeWEQN3PRgdSc6F8d9QRMWkf36CLMch1Do/+RQ==} + '@types/node@20.19.23': + resolution: {integrity: sha512-yIdlVVVHXpmqRhtyovZAcSy0MiPcYWGkoO4CGe/+jpP0hmNuihm4XhHbADpK++MsiLHP5MVlv+bcgdF99kSiFQ==} '@types/papaparse@5.3.16': resolution: {integrity: sha512-T3VuKMC2H0lgsjI9buTB3uuKj3EMD2eap1MOuEQuBQ44EnDx/IkGhU6EwiTf9zG3za4SKlmwKAImdDKdNnCsXg==} @@ -3599,63 +3348,63 @@ packages: '@types/yargs@17.0.33': resolution: {integrity: sha512-WpxBCKWPLr4xSsHgz511rFJAM+wS28w2zEO1QDNY5zM/S8ok70NNfztH0xwhqKyaK0OHCbN98LDAZuy1ctxDkA==} - '@typescript-eslint/eslint-plugin@8.46.1': - resolution: {integrity: sha512-rUsLh8PXmBjdiPY+Emjz9NX2yHvhS11v0SR6xNJkm5GM1MO9ea/1GoDKlHHZGrOJclL/cZ2i/vRUYVtjRhrHVQ==} + '@typescript-eslint/eslint-plugin@8.46.2': + resolution: {integrity: sha512-ZGBMToy857/NIPaaCucIUQgqueOiq7HeAKkhlvqVV4lm089zUFW6ikRySx2v+cAhKeUCPuWVHeimyk6Dw1iY3w==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} peerDependencies: - '@typescript-eslint/parser': ^8.46.1 + '@typescript-eslint/parser': ^8.46.2 eslint: ^8.57.0 || ^9.0.0 typescript: '>=4.8.4 <6.0.0' - '@typescript-eslint/parser@8.46.1': - resolution: {integrity: sha512-6JSSaBZmsKvEkbRUkf7Zj7dru/8ZCrJxAqArcLaVMee5907JdtEbKGsZ7zNiIm/UAkpGUkaSMZEXShnN2D1HZA==} + '@typescript-eslint/parser@8.46.2': + resolution: {integrity: sha512-BnOroVl1SgrPLywqxyqdJ4l3S2MsKVLDVxZvjI1Eoe8ev2r3kGDo+PcMihNmDE+6/KjkTubSJnmqGZZjQSBq/g==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} peerDependencies: eslint: ^8.57.0 || ^9.0.0 typescript: '>=4.8.4 <6.0.0' - '@typescript-eslint/project-service@8.46.1': - resolution: {integrity: sha512-FOIaFVMHzRskXr5J4Jp8lFVV0gz5ngv3RHmn+E4HYxSJ3DgDzU7fVI1/M7Ijh1zf6S7HIoaIOtln1H5y8V+9Zg==} + '@typescript-eslint/project-service@8.46.2': + resolution: {integrity: sha512-PULOLZ9iqwI7hXcmL4fVfIsBi6AN9YxRc0frbvmg8f+4hQAjQ5GYNKK0DIArNo+rOKmR/iBYwkpBmnIwin4wBg==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} peerDependencies: typescript: '>=4.8.4 <6.0.0' - '@typescript-eslint/scope-manager@8.46.1': - resolution: {integrity: sha512-weL9Gg3/5F0pVQKiF8eOXFZp8emqWzZsOJuWRUNtHT+UNV2xSJegmpCNQHy37aEQIbToTq7RHKhWvOsmbM680A==} + '@typescript-eslint/scope-manager@8.46.2': + resolution: {integrity: sha512-LF4b/NmGvdWEHD2H4MsHD8ny6JpiVNDzrSZr3CsckEgCbAGZbYM4Cqxvi9L+WqDMT+51Ozy7lt2M+d0JLEuBqA==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@typescript-eslint/tsconfig-utils@8.46.1': - resolution: {integrity: sha512-X88+J/CwFvlJB+mK09VFqx5FE4H5cXD+H/Bdza2aEWkSb8hnWIQorNcscRl4IEo1Cz9VI/+/r/jnGWkbWPx54g==} + '@typescript-eslint/tsconfig-utils@8.46.2': + resolution: {integrity: sha512-a7QH6fw4S57+F5y2FIxxSDyi5M4UfGF+Jl1bCGd7+L4KsaUY80GsiF/t0UoRFDHAguKlBaACWJRmdrc6Xfkkag==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} peerDependencies: typescript: '>=4.8.4 <6.0.0' - '@typescript-eslint/type-utils@8.46.1': - resolution: {integrity: sha512-+BlmiHIiqufBxkVnOtFwjah/vrkF4MtKKvpXrKSPLCkCtAp8H01/VV43sfqA98Od7nJpDcFnkwgyfQbOG0AMvw==} + '@typescript-eslint/type-utils@8.46.2': + resolution: {integrity: sha512-HbPM4LbaAAt/DjxXaG9yiS9brOOz6fabal4uvUmaUYe6l3K1phQDMQKBRUrr06BQkxkvIZVVHttqiybM9nJsLA==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} peerDependencies: eslint: ^8.57.0 || ^9.0.0 typescript: '>=4.8.4 <6.0.0' - '@typescript-eslint/types@8.46.1': - resolution: {integrity: sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ==} + '@typescript-eslint/types@8.46.2': + resolution: {integrity: sha512-lNCWCbq7rpg7qDsQrd3D6NyWYu+gkTENkG5IKYhUIcxSb59SQC/hEQ+MrG4sTgBVghTonNWq42bA/d4yYumldQ==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@typescript-eslint/typescript-estree@8.46.1': - resolution: {integrity: sha512-uIifjT4s8cQKFQ8ZBXXyoUODtRoAd7F7+G8MKmtzj17+1UbdzFl52AzRyZRyKqPHhgzvXunnSckVu36flGy8cg==} + '@typescript-eslint/typescript-estree@8.46.2': + resolution: {integrity: sha512-f7rW7LJ2b7Uh2EiQ+7sza6RDZnajbNbemn54Ob6fRwQbgcIn+GWfyuHDHRYgRoZu1P4AayVScrRW+YfbTvPQoQ==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} peerDependencies: typescript: '>=4.8.4 <6.0.0' - '@typescript-eslint/utils@8.46.1': - resolution: {integrity: sha512-vkYUy6LdZS7q1v/Gxb2Zs7zziuXN0wxqsetJdeZdRe/f5dwJFglmuvZBfTUivCtjH725C1jWCDfpadadD95EDQ==} + '@typescript-eslint/utils@8.46.2': + resolution: {integrity: sha512-sExxzucx0Tud5tE0XqR0lT0psBQvEpnpiul9XbGUB1QwpWJJAps1O/Z7hJxLGiZLBKMCutjTzDgmd1muEhBnVg==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} peerDependencies: eslint: ^8.57.0 || ^9.0.0 typescript: '>=4.8.4 <6.0.0' - '@typescript-eslint/visitor-keys@8.46.1': - resolution: {integrity: sha512-ptkmIf2iDkNUjdeu2bQqhFPV1m6qTnFFjg7PPDjxKWaMaP0Z6I9l30Jr3g5QqbZGdw8YdYvLp+XnqnWWZOg/NA==} + '@typescript-eslint/visitor-keys@8.46.2': + resolution: {integrity: sha512-tUFMXI4gxzzMXt4xpGJEsBsTox0XbNQ1y94EwlD/CuZwFcQP79xfQqMhau9HsRc/J0cAPA/HZt1dZPtGn9V/7w==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} '@ungap/structured-clone@1.3.0': @@ -3681,7 +3430,7 @@ packages: resolution: {integrity: sha512-46ryTE9RZO/rfDd7pEqFl7etuyzekzEhUbTW3BvmeO/BcCMEgq59BKhek3dXDWgAj4oMK6OZi+vRr1wPW6qjEQ==} peerDependencies: msw: ^2.4.9 - vite: 6.2.7 + vite: 6.4.1 peerDependenciesMeta: msw: optional: true @@ -3697,26 +3446,17 @@ packages: '@vitest/utils@3.2.4': resolution: {integrity: sha512-fB2V0JFrQSMsCo9HiSq3Ezpdv4iYaXRG1Sx8edX3MwxfyNn83mKiGzOcH+Fkxt4MHxr3y42fQi1oeAInqgX2QA==} - '@vue/compiler-core@3.5.17': - resolution: {integrity: sha512-Xe+AittLbAyV0pabcN7cP7/BenRBNcteM4aSDCtRvGw0d9OL+HG1u/XHLY/kt1q4fyMeZYXyIYrsHuPSiDPosA==} - '@vue/compiler-core@3.5.22': resolution: {integrity: sha512-jQ0pFPmZwTEiRNSb+i9Ow/I/cHv2tXYqsnHKKyCQ08irI2kdF5qmYedmF8si8mA7zepUFmJ2hqzS8CQmNOWOkQ==} - '@vue/compiler-dom@3.5.17': - resolution: {integrity: sha512-+2UgfLKoaNLhgfhV5Ihnk6wB4ljyW1/7wUIog2puUqajiC29Lp5R/IKDdkebh9jTbTogTbsgB+OY9cEWzG95JQ==} - '@vue/compiler-dom@3.5.22': resolution: {integrity: sha512-W8RknzUM1BLkypvdz10OVsGxnMAuSIZs9Wdx1vzA3mL5fNMN15rhrSCLiTm6blWeACwUwizzPVqGJgOGBEN/hA==} - '@vue/compiler-sfc@3.5.17': - resolution: {integrity: sha512-rQQxbRJMgTqwRugtjw0cnyQv9cP4/4BxWfTdRBkqsTfLOHWykLzbOc3C4GGzAmdMDxhzU/1Ija5bTjMVrddqww==} + '@vue/compiler-sfc@3.5.22': + resolution: {integrity: sha512-tbTR1zKGce4Lj+JLzFXDq36K4vcSZbJ1RBu8FxcDv1IGRz//Dh2EBqksyGVypz3kXpshIfWKGOCcqpSbyGWRJQ==} - '@vue/compiler-ssr@3.5.17': - resolution: {integrity: sha512-hkDbA0Q20ZzGgpj5uZjb9rBzQtIHLS78mMilwrlpWk2Ep37DYntUz0PonQ6kr113vfOEdM+zTBuJDaceNIW0tQ==} - - '@vue/shared@3.5.17': - resolution: {integrity: sha512-CabR+UN630VnsJO/jHWYBC1YVXyMq94KKp6iF5MQgZJs5I8cmjw6oVMO1oDbtBkENSHSSn/UadWlW/OAgdmKrg==} + '@vue/compiler-ssr@3.5.22': + resolution: {integrity: sha512-GdgyLvg4R+7T8Nk2Mlighx7XGxq/fJf9jaVofc3IL0EPesTE86cP/8DD1lT3h1JeZr2ySBvyqKQJgbS54IX1Ww==} '@vue/shared@3.5.22': resolution: {integrity: sha512-F4yc6palwq3TT0u+FYf0Ns4Tfl9GRFURDN2gWG7L1ecIaS/4fCIuFOjMTnCyjsu/OK6vaDKLCrGAa+KvvH+h4w==} @@ -4464,6 +4204,10 @@ packages: create-require@1.1.1: resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==} + cron-parser@5.4.0: + resolution: {integrity: sha512-HxYB8vTvnQFx4dLsZpGRa0uHp6X3qIzS3ZJgJ9v6l/5TJMgeWQbLkR5yiJ5hOxGbc9+jCADDnydIe15ReLZnJA==} + engines: {node: '>=18'} + cross-env@10.1.0: resolution: {integrity: sha512-GsYosgnACZTADcmEyJctkJIoqAhHjttw7RsFrVoJNXbsWWqaq6Ym+7kZjq6mS45O0jij6vtiReppKQEtqWy6Dw==} engines: {node: '>=20'} @@ -4667,8 +4411,8 @@ packages: dagre-d3-es@7.0.11: resolution: {integrity: sha512-tvlJLyQf834SylNKax8Wkzco/1ias1OPw8DcUMDE7oUIoSEW25riQVuiu/0OWEFqT0cxHT3Pa9/D82Jr47IONw==} - dayjs@1.11.18: - resolution: {integrity: sha512-zFBQ7WFRvVRhKcWoUh+ZA1g2HVgUbsZm9sbddh8EC5iv93sui8DVVz1Npvz+r6meo9VKfa8NyLWBsQK1VvIKPA==} + dayjs@1.11.19: + resolution: {integrity: sha512-t5EcLVS6QPBNqM2z8fakk/NKel+Xzshgt8FFKAn+qwlD1pzZWxh0nVCrvFK7ZDb6XucZeF9z8C7CBWTRIVApAw==} debounce@1.2.1: resolution: {integrity: sha512-XRRe6Glud4rd/ZGQfiV1ruXSfbvfJedlV9Y6zOlP+2K04vBYiJEte6stfFkCP03aMnY5tsipamumUjL14fofug==} @@ -4817,6 +4561,9 @@ packages: resolution: {integrity: sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ==} engines: {node: '>= 4'} + dompurify@3.1.7: + resolution: {integrity: sha512-VaTstWtsneJY8xzy7DekmYWEOZcmzIe3Qb3zPd4STve1OBTa+e+WmS1ITQec1fZYXI3HCsOZZiSMpG6oxoWMWQ==} + dompurify@3.3.0: resolution: {integrity: sha512-r+f6MYR1gGN1eJv0TVQbhA7if/U7P87cdPl3HN5rikqaBSBxLiCb/b9O+2eG0cxz0ghyU+mU1QkbsOwERMYlWQ==} @@ -4946,11 +4693,6 @@ packages: engines: {node: '>=18'} hasBin: true - esbuild@0.25.11: - resolution: {integrity: sha512-KohQwyzrKTQmhXDW1PjCv3Tyspn9n5GcY2RTDqeORIdIJY8yKIF7sTSopFmn/wpMPW4rdPXI0UE5LJLuq3bx0Q==} - engines: {node: '>=18'} - hasBin: true - escalade@3.2.0: resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} engines: {node: '>=6'} @@ -5599,8 +5341,8 @@ packages: hachure-fill@0.5.2: resolution: {integrity: sha512-3GKBOn+m2LX9iq+JC1064cSFprJY4jL1jCXTcpnfER5HYE2l/4EfWSGzkPa/ZDBmYI0ZOEj5VHV/eKnPGkHuOg==} - happy-dom@20.0.7: - resolution: {integrity: sha512-CywLfzmYxP5OYpuAG0usFY0CpxJtwYR+w8Mms5J8W29Y2Pzf6rbfQS2M523tRZTb0oLA+URopPtnAQX2fupHZQ==} + happy-dom@20.0.8: + resolution: {integrity: sha512-TlYaNQNtzsZ97rNMBAm8U+e2cUQXNithgfCizkDgc11lgmN4j9CKMhO3FPGKWQYPwwkFcPpoXYF/CqEPLgzfOg==} engines: {node: '>=20.0.0'} has-flag@4.0.0: @@ -5827,6 +5569,7 @@ packages: intersection-observer@0.12.2: resolution: {integrity: sha512-7m1vEcPCxXYI8HqnL8CKI6siDyD+eIWSwgB3DZA+ZTogxk9I4CDnj4wilt9x/+/QbHI4YG5YZNmC6458/e9Ktg==} + deprecated: The Intersection Observer polyfill is no longer needed and can safely be removed. Intersection Observer has been Baseline since 2019. is-alphabetical@1.0.4: resolution: {integrity: sha512-DwzsA04LQ10FHTZuL0/grVDk4rFoVH1pjAToYwBrHSxcrBIGQuXrQMtD5U1b0U2XVgKZCTLLP8u2Qxqhy3l2Vg==} @@ -6229,8 +5972,8 @@ packages: resolution: {integrity: sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==} engines: {node: '>=6'} - knip@5.66.1: - resolution: {integrity: sha512-Ad3VUPIk9GZYovKuwKtGMheupek7IoPGaDEBAvnCYLKJXnwmqNLyXqMp+l5r3OOpFVjF7DdkFIZFVrXESDNylQ==} + knip@5.66.2: + resolution: {integrity: sha512-5wvsdc17C5bMxjuGfN9KVS/tW5KIvzP1RClfpTMdLYm8IXIsfWsiHlFkTvZIca9skwoVDyTyXmbRq4w1Poim+A==} engines: {node: '>=18.18.0'} hasBin: true peerDependencies: @@ -6368,6 +6111,10 @@ packages: lru-cache@5.1.1: resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} + luxon@3.7.2: + resolution: {integrity: sha512-vtEhXh/gNjI9Yg1u4jX/0YVPMvxzHuGgCm6tC5kZyb08yjGWGnqAjGJvcXbqQR2P3MyMEFnRbpcdFS6PBcLqew==} + engines: {node: '>=12'} + lz-string@1.5.0: resolution: {integrity: sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==} hasBin: true @@ -6378,6 +6125,9 @@ packages: magic-string@0.30.19: resolution: {integrity: sha512-2N21sPY9Ws53PZvsEpVtNuSW+ScYbQdp4b9qUaL+9QkHUrGFKo56Lg9Emg5s9V/qrtNBmiR01sYhUOwu3H+VOw==} + magic-string@0.30.21: + resolution: {integrity: sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==} + magicast@0.3.5: resolution: {integrity: sha512-L0WhttDl+2BOsybvEOLK7fW3UA0OQ0IQ2d6Zl2x/a6vVRs3bAY0ECOSHHeL5jD+SbOpOCUEi0y1DgHEn9Qn1AQ==} @@ -6402,6 +6152,11 @@ packages: markdown-table@3.0.4: resolution: {integrity: sha512-wiYz4+JrLyb/DqW2hkFJxP7Vd7JuTDm77fvbM8VfEQdmSMqcImWeeRbHwZjBjIFki/VaMK2BhFi7oUUZeM5bqw==} + marked@14.0.0: + resolution: {integrity: sha512-uIj4+faQ+MgHgwUW1l2PsPglZLOLOT1uErt06dAPtx2kjteLAkbsd/0FiYg/MGS+i7ZKLb7w2WClxHkzOOuryQ==} + engines: {node: '>= 18'} + hasBin: true + marked@15.0.12: resolution: {integrity: sha512-8dD6FusOQSrpv9Z1rdNMdlSgQOIP880DHqnohobOmYLElGEqAL/JvxvuxZO16r4HtjTlfPRDC1hbvxC9dPN2nA==} engines: {node: '>= 18'} @@ -6677,8 +6432,8 @@ packages: mlly@1.8.0: resolution: {integrity: sha512-l8D9ODSRWLe2KHJSifWGwBqpTZXIXTeo8mlKjY+E2HAakaTeNpqAyBZ8GSqLzHgw4XmHmC8whvpjJNMbFZN7/g==} - monaco-editor@0.52.2: - resolution: {integrity: sha512-GEQWEZmfkOGLdd3XK8ryrfWz3AIP8YymVXiPHEdewrUq7mh0qrKrfHLNCXcbB6sTnMLnOZ3ztSiKcciFUkIJwQ==} + monaco-editor@0.54.0: + resolution: {integrity: sha512-hx45SEUoLatgWxHKCmlLJH81xBo0uXP4sRkESUpmDQevfi+e7K1VuiSprK6UpQ8u4zOcKNiH0pMvHvlMWA/4cw==} mrmime@2.0.1: resolution: {integrity: sha512-Y3wQdFg2Va6etvQ5I82yUhGdsKrcYox6p7FfL1LbK2J4V01F9TGlepTIhnK24t7koZibmg82KGglhA1XK5IsLQ==} @@ -6839,8 +6594,8 @@ packages: os-browserify@0.3.0: resolution: {integrity: sha512-gjcpUc3clBf9+210TRaDWbf+rZZZEshZ+DlXMRCeAjp0xhTrnQsKHypIy1J3d5hKdUzj69t708EHtU8P6bUn0A==} - oxc-resolver@11.10.0: - resolution: {integrity: sha512-LNJkji0qsBvZ7+yze3S1qsWufZ3VBcyU1wAnC5bBP0QzHsKf4rrNhG5I4c0RIDQGKsKDpVWh8vhUAGE3cb53kA==} + oxc-resolver@11.11.0: + resolution: {integrity: sha512-vVeBJf77zBeqOA/LBCTO/pr0/ETHGSleCRsI5Kmsf2OsfB5opzhhZptt6VxkqjKWZH+eF1se88fYDG5DGRLjkg==} p-cancelable@2.1.1: resolution: {integrity: sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg==} @@ -6982,6 +6737,10 @@ packages: resolution: {integrity: sha512-wfRLBZ0feWRhCIkoMB6ete7czJcnNnqRpcoWQBLqatqXXmelSRqfdDK4F3u9T2s2cXas/hQJcryI/4lAL+XTlA==} engines: {node: '>=0.12'} + pbkdf2@3.1.5: + resolution: {integrity: sha512-Q3CG/cYvCO1ye4QKkuH7EXxs3VC/rI1/trd+qX2+PolbaKG0H+bgcZzrTt96mMyRtejk+JMCiLUn3y29W8qmFQ==} + engines: {node: '>= 0.10'} + pdfjs-dist@4.4.168: resolution: {integrity: sha512-MbkAjpwka/dMHaCfQ75RY1FXX3IewBVu6NGZOcxerRFlaBiIkZmUoR0jotX5VUzYZEXAGzSFtknWs5xRKliXPA==} engines: {node: '>=18'} @@ -7599,8 +7358,8 @@ packages: resolution: {integrity: sha512-OcXjMsGdhL4XnbShKpAcSqPMzQoYkYyhbEaeSko47MjRP9NfEQMhZkXL1DoFlt9LWQn4YttrdnV6X2OiyzBi+A==} engines: {node: '>=10'} - resolve@1.22.10: - resolution: {integrity: sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==} + resolve@1.22.11: + resolution: {integrity: sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==} engines: {node: '>= 0.4'} hasBin: true @@ -7649,11 +7408,6 @@ packages: engines: {node: '>=10.0.0'} hasBin: true - rollup@4.52.5: - resolution: {integrity: sha512-3GuObel8h7Kqdjt0gxkEzaifHTqLVW56Y/bjN7PSQtkKr0w3V/QYSdt6QWYtd7A1xUtYQigtdUfgj1RvWVtorw==} - engines: {node: '>=18.0.0', npm: '>=8.0.0'} - hasBin: true - roughjs@4.6.6: resolution: {integrity: sha512-ZUz/69+SYpFN/g/lUlo2FXcIjRkSu3nDarreVdGGndHEBJ6cXPdKguS8JGxwj5HA5xIbVKSmLgr5b3AWxtRfvQ==} @@ -8419,46 +8173,6 @@ packages: vfile@6.0.3: resolution: {integrity: sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q==} - vite@6.2.7: - resolution: {integrity: sha512-qg3LkeuinTrZoJHHF94coSaTfIPyBYoywp+ys4qu20oSJFbKMYoIJo0FWJT9q6Vp49l6z9IsJRbHdcGtiKbGoQ==} - engines: {node: ^18.0.0 || ^20.0.0 || >=22.0.0} - hasBin: true - peerDependencies: - '@types/node': ^18.0.0 || ^20.0.0 || >=22.0.0 - jiti: '>=1.21.0' - less: '*' - lightningcss: ^1.21.0 - sass: '*' - sass-embedded: '*' - stylus: '*' - sugarss: '*' - terser: ^5.16.0 - tsx: ^4.8.1 - yaml: ^2.4.2 - peerDependenciesMeta: - '@types/node': - optional: true - jiti: - optional: true - less: - optional: true - lightningcss: - optional: true - sass: - optional: true - sass-embedded: - optional: true - stylus: - optional: true - sugarss: - optional: true - terser: - optional: true - tsx: - optional: true - yaml: - optional: true - vm-browserify@1.1.2: resolution: {integrity: sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==} @@ -8752,15 +8466,15 @@ snapshots: '@alloc/quick-lru@5.2.0': {} - '@antfu/eslint-config@5.4.1(@eslint-react/eslint-plugin@1.53.1(eslint@9.38.0(jiti@1.21.7))(ts-api-utils@2.1.0(typescript@5.9.3))(typescript@5.9.3))(@next/eslint-plugin-next@15.5.4)(@vue/compiler-sfc@3.5.17)(eslint-plugin-react-hooks@5.2.0(eslint@9.38.0(jiti@1.21.7)))(eslint-plugin-react-refresh@0.4.24(eslint@9.38.0(jiti@1.21.7)))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': + '@antfu/eslint-config@5.4.1(@eslint-react/eslint-plugin@1.53.1(eslint@9.38.0(jiti@1.21.7))(ts-api-utils@2.1.0(typescript@5.9.3))(typescript@5.9.3))(@next/eslint-plugin-next@15.5.4)(@vue/compiler-sfc@3.5.22)(eslint-plugin-react-hooks@5.2.0(eslint@9.38.0(jiti@1.21.7)))(eslint-plugin-react-refresh@0.4.24(eslint@9.38.0(jiti@1.21.7)))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: '@antfu/install-pkg': 1.1.0 '@clack/prompts': 0.11.0 '@eslint-community/eslint-plugin-eslint-comments': 4.5.0(eslint@9.38.0(jiti@1.21.7)) - '@eslint/markdown': 7.4.0 + '@eslint/markdown': 7.4.1 '@stylistic/eslint-plugin': 5.5.0(eslint@9.38.0(jiti@1.21.7)) - '@typescript-eslint/eslint-plugin': 8.46.1(@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/parser': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/eslint-plugin': 8.46.2(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/parser': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@vitest/eslint-plugin': 1.3.23(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) ansis: 4.2.0 cac: 6.7.14 @@ -8780,10 +8494,10 @@ snapshots: eslint-plugin-regexp: 2.10.0(eslint@9.38.0(jiti@1.21.7)) eslint-plugin-toml: 0.12.0(eslint@9.38.0(jiti@1.21.7)) eslint-plugin-unicorn: 61.0.2(eslint@9.38.0(jiti@1.21.7)) - eslint-plugin-unused-imports: 4.3.0(@typescript-eslint/eslint-plugin@8.46.1(@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7)) - eslint-plugin-vue: 10.5.1(@stylistic/eslint-plugin@5.5.0(eslint@9.38.0(jiti@1.21.7)))(@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(vue-eslint-parser@10.2.0(eslint@9.38.0(jiti@1.21.7))) + eslint-plugin-unused-imports: 4.3.0(@typescript-eslint/eslint-plugin@8.46.2(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7)) + eslint-plugin-vue: 10.5.1(@stylistic/eslint-plugin@5.5.0(eslint@9.38.0(jiti@1.21.7)))(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(vue-eslint-parser@10.2.0(eslint@9.38.0(jiti@1.21.7))) eslint-plugin-yml: 1.19.0(eslint@9.38.0(jiti@1.21.7)) - eslint-processor-vue-blocks: 2.0.0(@vue/compiler-sfc@3.5.17)(eslint@9.38.0(jiti@1.21.7)) + eslint-processor-vue-blocks: 2.0.0(@vue/compiler-sfc@3.5.22)(eslint@9.38.0(jiti@1.21.7)) globals: 16.4.0 jsonc-eslint-parser: 2.4.1 local-pkg: 1.1.2 @@ -8892,7 +8606,7 @@ snapshots: '@babel/helper-plugin-utils': 7.27.1 debug: 4.4.3 lodash.debounce: 4.0.8 - resolve: 1.22.10 + resolve: 1.22.11 transitivePeerDependencies: - supports-color @@ -8956,6 +8670,8 @@ snapshots: '@babel/helper-validator-identifier@7.27.1': {} + '@babel/helper-validator-identifier@7.28.5': {} + '@babel/helper-validator-option@7.27.1': {} '@babel/helper-wrap-function@7.28.3': @@ -8975,6 +8691,10 @@ snapshots: dependencies: '@babel/types': 7.28.4 + '@babel/parser@7.28.5': + dependencies: + '@babel/types': 7.28.5 + '@babel/plugin-bugfix-firefox-class-in-computed-class-key@7.27.1(@babel/core@7.28.4)': dependencies: '@babel/core': 7.28.4 @@ -9631,6 +9351,11 @@ snapshots: '@babel/helper-string-parser': 7.27.1 '@babel/helper-validator-identifier': 7.27.1 + '@babel/types@7.28.5': + dependencies: + '@babel/helper-string-parser': 7.27.1 + '@babel/helper-validator-identifier': 7.28.5 + '@bcoe/v8-coverage@0.2.3': {} '@braintree/sanitize-url@7.1.1': {} @@ -9652,13 +9377,13 @@ snapshots: '@chevrotain/utils@11.0.3': {} - '@chromatic-com/storybook@4.1.1(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@chromatic-com/storybook@4.1.1(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: '@neoconfetti/react': 1.0.0 chromatic: 12.2.0 filesize: 10.1.6 jsonfile: 6.2.0 - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) strip-ansi: 7.1.2 transitivePeerDependencies: - '@chromatic-com/cypress' @@ -9720,17 +9445,16 @@ snapshots: '@cspotcode/source-map-support@0.8.1': dependencies: '@jridgewell/trace-mapping': 0.3.9 - optional: true '@discoveryjs/json-ext@0.5.7': {} - '@emnapi/core@1.5.0': + '@emnapi/core@1.6.0': dependencies: '@emnapi/wasi-threads': 1.1.0 tslib: 2.8.1 optional: true - '@emnapi/runtime@1.5.0': + '@emnapi/runtime@1.6.0': dependencies: tslib: 2.8.1 optional: true @@ -9747,7 +9471,7 @@ snapshots: '@es-joy/jsdoccomment@0.50.2': dependencies: '@types/estree': 1.0.8 - '@typescript-eslint/types': 8.46.1 + '@typescript-eslint/types': 8.46.2 comment-parser: 1.4.1 esquery: 1.6.0 jsdoc-type-pratt-parser: 4.1.0 @@ -9755,7 +9479,7 @@ snapshots: '@es-joy/jsdoccomment@0.58.0': dependencies: '@types/estree': 1.0.8 - '@typescript-eslint/types': 8.46.1 + '@typescript-eslint/types': 8.46.2 comment-parser: 1.4.1 esquery: 1.6.0 jsdoc-type-pratt-parser: 5.4.0 @@ -9763,156 +9487,78 @@ snapshots: '@esbuild/aix-ppc64@0.25.0': optional: true - '@esbuild/aix-ppc64@0.25.11': - optional: true - '@esbuild/android-arm64@0.25.0': optional: true - '@esbuild/android-arm64@0.25.11': - optional: true - '@esbuild/android-arm@0.25.0': optional: true - '@esbuild/android-arm@0.25.11': - optional: true - '@esbuild/android-x64@0.25.0': optional: true - '@esbuild/android-x64@0.25.11': - optional: true - '@esbuild/darwin-arm64@0.25.0': optional: true - '@esbuild/darwin-arm64@0.25.11': - optional: true - '@esbuild/darwin-x64@0.25.0': optional: true - '@esbuild/darwin-x64@0.25.11': - optional: true - '@esbuild/freebsd-arm64@0.25.0': optional: true - '@esbuild/freebsd-arm64@0.25.11': - optional: true - '@esbuild/freebsd-x64@0.25.0': optional: true - '@esbuild/freebsd-x64@0.25.11': - optional: true - '@esbuild/linux-arm64@0.25.0': optional: true - '@esbuild/linux-arm64@0.25.11': - optional: true - '@esbuild/linux-arm@0.25.0': optional: true - '@esbuild/linux-arm@0.25.11': - optional: true - '@esbuild/linux-ia32@0.25.0': optional: true - '@esbuild/linux-ia32@0.25.11': - optional: true - '@esbuild/linux-loong64@0.25.0': optional: true - '@esbuild/linux-loong64@0.25.11': - optional: true - '@esbuild/linux-mips64el@0.25.0': optional: true - '@esbuild/linux-mips64el@0.25.11': - optional: true - '@esbuild/linux-ppc64@0.25.0': optional: true - '@esbuild/linux-ppc64@0.25.11': - optional: true - '@esbuild/linux-riscv64@0.25.0': optional: true - '@esbuild/linux-riscv64@0.25.11': - optional: true - '@esbuild/linux-s390x@0.25.0': optional: true - '@esbuild/linux-s390x@0.25.11': - optional: true - '@esbuild/linux-x64@0.25.0': optional: true - '@esbuild/linux-x64@0.25.11': - optional: true - '@esbuild/netbsd-arm64@0.25.0': optional: true - '@esbuild/netbsd-arm64@0.25.11': - optional: true - '@esbuild/netbsd-x64@0.25.0': optional: true - '@esbuild/netbsd-x64@0.25.11': - optional: true - '@esbuild/openbsd-arm64@0.25.0': optional: true - '@esbuild/openbsd-arm64@0.25.11': - optional: true - '@esbuild/openbsd-x64@0.25.0': optional: true - '@esbuild/openbsd-x64@0.25.11': - optional: true - - '@esbuild/openharmony-arm64@0.25.11': - optional: true - '@esbuild/sunos-x64@0.25.0': optional: true - '@esbuild/sunos-x64@0.25.11': - optional: true - '@esbuild/win32-arm64@0.25.0': optional: true - '@esbuild/win32-arm64@0.25.11': - optional: true - '@esbuild/win32-ia32@0.25.0': optional: true - '@esbuild/win32-ia32@0.25.11': - optional: true - '@esbuild/win32-x64@0.25.0': optional: true - '@esbuild/win32-x64@0.25.11': - optional: true - '@eslint-community/eslint-plugin-eslint-comments@4.5.0(eslint@9.38.0(jiti@1.21.7))': dependencies: escape-string-regexp: 4.0.0 @@ -9929,9 +9575,9 @@ snapshots: '@eslint-react/ast@1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: '@eslint-react/eff': 1.53.1 - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/typescript-estree': 8.46.1(typescript@5.9.3) - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/typescript-estree': 8.46.2(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) string-ts: 2.2.1 ts-pattern: 5.8.0 transitivePeerDependencies: @@ -9946,10 +9592,10 @@ snapshots: '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/var': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) birecord: 0.1.1 ts-pattern: 5.8.0 transitivePeerDependencies: @@ -9964,10 +9610,10 @@ snapshots: '@eslint-react/eff': 1.53.1 '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) eslint-plugin-react-debug: 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint-plugin-react-dom: 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) @@ -9984,7 +9630,7 @@ snapshots: '@eslint-react/kit@1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: '@eslint-react/eff': 1.53.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) ts-pattern: 5.8.0 zod: 4.1.12 transitivePeerDependencies: @@ -9996,7 +9642,7 @@ snapshots: dependencies: '@eslint-react/eff': 1.53.1 '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) ts-pattern: 5.8.0 zod: 4.1.12 transitivePeerDependencies: @@ -10008,9 +9654,9 @@ snapshots: dependencies: '@eslint-react/ast': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/eff': 1.53.1 - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) string-ts: 2.2.1 ts-pattern: 5.8.0 transitivePeerDependencies: @@ -10060,10 +9706,10 @@ snapshots: '@eslint/js@9.38.0': {} - '@eslint/markdown@7.4.0': + '@eslint/markdown@7.4.1': dependencies: '@eslint/core': 0.16.0 - '@eslint/plugin-kit': 0.3.5 + '@eslint/plugin-kit': 0.3.4 github-slugger: 2.0.0 mdast-util-from-markdown: 2.0.2 mdast-util-frontmatter: 2.0.1 @@ -10081,11 +9727,6 @@ snapshots: '@eslint/core': 0.15.2 levn: 0.4.1 - '@eslint/plugin-kit@0.3.5': - dependencies: - '@eslint/core': 0.15.2 - levn: 0.4.1 - '@floating-ui/core@1.7.3': dependencies: '@floating-ui/utils': 0.2.10 @@ -10123,12 +9764,12 @@ snapshots: dependencies: tslib: 2.8.1 - '@happy-dom/jest-environment@20.0.7(@jest/environment@29.7.0)(@jest/fake-timers@29.7.0)(@jest/types@29.6.3)(jest-mock@29.7.0)(jest-util@29.7.0)': + '@happy-dom/jest-environment@20.0.8(@jest/environment@29.7.0)(@jest/fake-timers@29.7.0)(@jest/types@29.6.3)(jest-mock@29.7.0)(jest-util@29.7.0)': dependencies: '@jest/environment': 29.7.0 '@jest/fake-timers': 29.7.0 '@jest/types': 29.6.3 - happy-dom: 20.0.7 + happy-dom: 20.0.8 jest-mock: 29.7.0 jest-util: 29.7.0 @@ -10316,12 +9957,12 @@ snapshots: '@img/sharp-wasm32@0.33.5': dependencies: - '@emnapi/runtime': 1.5.0 + '@emnapi/runtime': 1.6.0 optional: true '@img/sharp-wasm32@0.34.4': dependencies: - '@emnapi/runtime': 1.5.0 + '@emnapi/runtime': 1.6.0 optional: true '@img/sharp-win32-arm64@0.34.4': @@ -10554,7 +10195,6 @@ snapshots: dependencies: '@jridgewell/resolve-uri': 3.1.2 '@jridgewell/sourcemap-codec': 1.5.5 - optional: true '@lexical/clipboard@0.36.2': dependencies: @@ -10815,17 +10455,17 @@ snapshots: dependencies: state-local: 1.0.7 - '@monaco-editor/react@4.7.0(monaco-editor@0.52.2)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)': + '@monaco-editor/react@4.7.0(monaco-editor@0.54.0)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)': dependencies: '@monaco-editor/loader': 1.5.0 - monaco-editor: 0.52.2 + monaco-editor: 0.54.0 react: 19.1.1 react-dom: 19.1.1(react@19.1.1) '@napi-rs/wasm-runtime@1.0.7': dependencies: - '@emnapi/core': 1.5.0 - '@emnapi/runtime': 1.5.0 + '@emnapi/core': 1.6.0 + '@emnapi/runtime': 1.6.0 '@tybys/wasm-util': 0.10.1 optional: true @@ -10978,63 +10618,63 @@ snapshots: dependencies: '@octokit/openapi-types': 25.1.0 - '@oxc-resolver/binding-android-arm-eabi@11.10.0': + '@oxc-resolver/binding-android-arm-eabi@11.11.0': optional: true - '@oxc-resolver/binding-android-arm64@11.10.0': + '@oxc-resolver/binding-android-arm64@11.11.0': optional: true - '@oxc-resolver/binding-darwin-arm64@11.10.0': + '@oxc-resolver/binding-darwin-arm64@11.11.0': optional: true - '@oxc-resolver/binding-darwin-x64@11.10.0': + '@oxc-resolver/binding-darwin-x64@11.11.0': optional: true - '@oxc-resolver/binding-freebsd-x64@11.10.0': + '@oxc-resolver/binding-freebsd-x64@11.11.0': optional: true - '@oxc-resolver/binding-linux-arm-gnueabihf@11.10.0': + '@oxc-resolver/binding-linux-arm-gnueabihf@11.11.0': optional: true - '@oxc-resolver/binding-linux-arm-musleabihf@11.10.0': + '@oxc-resolver/binding-linux-arm-musleabihf@11.11.0': optional: true - '@oxc-resolver/binding-linux-arm64-gnu@11.10.0': + '@oxc-resolver/binding-linux-arm64-gnu@11.11.0': optional: true - '@oxc-resolver/binding-linux-arm64-musl@11.10.0': + '@oxc-resolver/binding-linux-arm64-musl@11.11.0': optional: true - '@oxc-resolver/binding-linux-ppc64-gnu@11.10.0': + '@oxc-resolver/binding-linux-ppc64-gnu@11.11.0': optional: true - '@oxc-resolver/binding-linux-riscv64-gnu@11.10.0': + '@oxc-resolver/binding-linux-riscv64-gnu@11.11.0': optional: true - '@oxc-resolver/binding-linux-riscv64-musl@11.10.0': + '@oxc-resolver/binding-linux-riscv64-musl@11.11.0': optional: true - '@oxc-resolver/binding-linux-s390x-gnu@11.10.0': + '@oxc-resolver/binding-linux-s390x-gnu@11.11.0': optional: true - '@oxc-resolver/binding-linux-x64-gnu@11.10.0': + '@oxc-resolver/binding-linux-x64-gnu@11.11.0': optional: true - '@oxc-resolver/binding-linux-x64-musl@11.10.0': + '@oxc-resolver/binding-linux-x64-musl@11.11.0': optional: true - '@oxc-resolver/binding-wasm32-wasi@11.10.0': + '@oxc-resolver/binding-wasm32-wasi@11.11.0': dependencies: '@napi-rs/wasm-runtime': 1.0.7 optional: true - '@oxc-resolver/binding-win32-arm64-msvc@11.10.0': + '@oxc-resolver/binding-win32-arm64-msvc@11.11.0': optional: true - '@oxc-resolver/binding-win32-ia32-msvc@11.10.0': + '@oxc-resolver/binding-win32-ia32-msvc@11.11.0': optional: true - '@oxc-resolver/binding-win32-x64-msvc@11.10.0': + '@oxc-resolver/binding-win32-x64-msvc@11.11.0': optional: true '@parcel/watcher-android-arm64@2.5.1': @@ -11416,7 +11056,7 @@ snapshots: builtin-modules: 3.3.0 deepmerge: 4.3.1 is-module: 1.0.0 - resolve: 1.22.10 + resolve: 1.22.11 rollup: 2.79.2 '@rollup/plugin-replace@2.4.2(rollup@2.79.2)': @@ -11432,72 +11072,6 @@ snapshots: picomatch: 2.3.1 rollup: 2.79.2 - '@rollup/rollup-android-arm-eabi@4.52.5': - optional: true - - '@rollup/rollup-android-arm64@4.52.5': - optional: true - - '@rollup/rollup-darwin-arm64@4.52.5': - optional: true - - '@rollup/rollup-darwin-x64@4.52.5': - optional: true - - '@rollup/rollup-freebsd-arm64@4.52.5': - optional: true - - '@rollup/rollup-freebsd-x64@4.52.5': - optional: true - - '@rollup/rollup-linux-arm-gnueabihf@4.52.5': - optional: true - - '@rollup/rollup-linux-arm-musleabihf@4.52.5': - optional: true - - '@rollup/rollup-linux-arm64-gnu@4.52.5': - optional: true - - '@rollup/rollup-linux-arm64-musl@4.52.5': - optional: true - - '@rollup/rollup-linux-loong64-gnu@4.52.5': - optional: true - - '@rollup/rollup-linux-ppc64-gnu@4.52.5': - optional: true - - '@rollup/rollup-linux-riscv64-gnu@4.52.5': - optional: true - - '@rollup/rollup-linux-riscv64-musl@4.52.5': - optional: true - - '@rollup/rollup-linux-s390x-gnu@4.52.5': - optional: true - - '@rollup/rollup-linux-x64-gnu@4.52.5': - optional: true - - '@rollup/rollup-linux-x64-musl@4.52.5': - optional: true - - '@rollup/rollup-openharmony-arm64@4.52.5': - optional: true - - '@rollup/rollup-win32-arm64-msvc@4.52.5': - optional: true - - '@rollup/rollup-win32-ia32-msvc@4.52.5': - optional: true - - '@rollup/rollup-win32-x64-gnu@4.52.5': - optional: true - - '@rollup/rollup-win32-x64-msvc@4.52.5': - optional: true - '@sentry-internal/browser-utils@8.55.0': dependencies: '@sentry/core': 8.55.0 @@ -11545,38 +11119,38 @@ snapshots: dependencies: '@sinonjs/commons': 3.0.1 - '@storybook/addon-docs@9.1.13(@types/react@19.1.17)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@storybook/addon-docs@9.1.13(@types/react@19.1.17)(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: '@mdx-js/react': 3.1.1(@types/react@19.1.17)(react@19.1.1) - '@storybook/csf-plugin': 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + '@storybook/csf-plugin': 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)) '@storybook/icons': 1.6.0(react-dom@19.1.1(react@19.1.1))(react@19.1.1) - '@storybook/react-dom-shim': 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + '@storybook/react-dom-shim': 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)) react: 19.1.1 react-dom: 19.1.1(react@19.1.1) - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) ts-dedent: 2.2.0 transitivePeerDependencies: - '@types/react' - '@storybook/addon-links@9.1.13(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@storybook/addon-links@9.1.13(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: '@storybook/global': 5.0.0 - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) optionalDependencies: react: 19.1.1 - '@storybook/addon-onboarding@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@storybook/addon-onboarding@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) - '@storybook/addon-themes@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@storybook/addon-themes@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) ts-dedent: 2.2.0 - '@storybook/builder-webpack5@9.1.13(esbuild@0.25.0)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3)(uglify-js@3.19.3)': + '@storybook/builder-webpack5@9.1.13(esbuild@0.25.0)(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3)(uglify-js@3.19.3)': dependencies: - '@storybook/core-webpack': 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + '@storybook/core-webpack': 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)) case-sensitive-paths-webpack-plugin: 2.4.0 cjs-module-lexer: 1.4.3 css-loader: 6.11.0(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) @@ -11584,7 +11158,7 @@ snapshots: fork-ts-checker-webpack-plugin: 8.0.0(typescript@5.9.3)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) html-webpack-plugin: 5.6.4(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) magic-string: 0.30.19 - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) style-loader: 3.3.4(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) terser-webpack-plugin: 5.3.14(esbuild@0.25.0)(uglify-js@3.19.3)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) ts-dedent: 2.2.0 @@ -11601,14 +11175,14 @@ snapshots: - uglify-js - webpack-cli - '@storybook/core-webpack@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@storybook/core-webpack@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) ts-dedent: 2.2.0 - '@storybook/csf-plugin@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@storybook/csf-plugin@9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) unplugin: 1.16.1 '@storybook/global@5.0.0': {} @@ -11618,7 +11192,7 @@ snapshots: react: 19.1.1 react-dom: 19.1.1(react@19.1.1) - '@storybook/nextjs@9.1.13(esbuild@0.25.0)(next@15.5.6(@babel/core@7.28.4)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2))(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(type-fest@4.2.0)(typescript@5.9.3)(uglify-js@3.19.3)(webpack-hot-middleware@2.26.1)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3))': + '@storybook/nextjs@9.1.13(esbuild@0.25.0)(next@15.5.6(@babel/core@7.28.4)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2))(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(sass@1.93.2)(storybook@9.1.13(@testing-library/dom@10.4.1))(type-fest@4.2.0)(typescript@5.9.3)(uglify-js@3.19.3)(webpack-hot-middleware@2.26.1)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3))': dependencies: '@babel/core': 7.28.4 '@babel/plugin-syntax-bigint': 7.8.3(@babel/core@7.28.4) @@ -11634,9 +11208,9 @@ snapshots: '@babel/preset-typescript': 7.27.1(@babel/core@7.28.4) '@babel/runtime': 7.28.4 '@pmmmwh/react-refresh-webpack-plugin': 0.5.17(react-refresh@0.14.2)(type-fest@4.2.0)(webpack-hot-middleware@2.26.1)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) - '@storybook/builder-webpack5': 9.1.13(esbuild@0.25.0)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3)(uglify-js@3.19.3) - '@storybook/preset-react-webpack': 9.1.13(esbuild@0.25.0)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3)(uglify-js@3.19.3) - '@storybook/react': 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3) + '@storybook/builder-webpack5': 9.1.13(esbuild@0.25.0)(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3)(uglify-js@3.19.3) + '@storybook/preset-react-webpack': 9.1.13(esbuild@0.25.0)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3)(uglify-js@3.19.3) + '@storybook/react': 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3) '@types/semver': 7.7.1 babel-loader: 9.2.1(@babel/core@7.28.4)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) css-loader: 6.11.0(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) @@ -11652,7 +11226,7 @@ snapshots: resolve-url-loader: 5.0.0 sass-loader: 16.0.5(sass@1.93.2)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) semver: 7.7.3 - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) style-loader: 3.3.4(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) styled-jsx: 5.1.7(@babel/core@7.28.4)(react@19.1.1) tsconfig-paths: 4.2.0 @@ -11678,9 +11252,9 @@ snapshots: - webpack-hot-middleware - webpack-plugin-serve - '@storybook/preset-react-webpack@9.1.13(esbuild@0.25.0)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3)(uglify-js@3.19.3)': + '@storybook/preset-react-webpack@9.1.13(esbuild@0.25.0)(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3)(uglify-js@3.19.3)': dependencies: - '@storybook/core-webpack': 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + '@storybook/core-webpack': 9.1.13(storybook@9.1.13(@testing-library/dom@10.4.1)) '@storybook/react-docgen-typescript-plugin': 1.0.6--canary.9.0c3f3b7.0(typescript@5.9.3)(webpack@5.102.1(esbuild@0.25.0)(uglify-js@3.19.3)) '@types/semver': 7.7.1 find-up: 7.0.0 @@ -11688,9 +11262,9 @@ snapshots: react: 19.1.1 react-docgen: 7.1.1 react-dom: 19.1.1(react@19.1.1) - resolve: 1.22.10 + resolve: 1.22.11 semver: 7.7.3 - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) tsconfig-paths: 4.2.0 webpack: 5.102.1(esbuild@0.25.0)(uglify-js@3.19.3) optionalDependencies: @@ -11716,26 +11290,26 @@ snapshots: transitivePeerDependencies: - supports-color - '@storybook/react-dom-shim@9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))': + '@storybook/react-dom-shim@9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1))': dependencies: react: 19.1.1 react-dom: 19.1.1(react@19.1.1) - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) - '@storybook/react@9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3)': + '@storybook/react@9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3)': dependencies: '@storybook/global': 5.0.0 - '@storybook/react-dom-shim': 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))) + '@storybook/react-dom-shim': 9.1.13(react-dom@19.1.1(react@19.1.1))(react@19.1.1)(storybook@9.1.13(@testing-library/dom@10.4.1)) react: 19.1.1 react-dom: 19.1.1(react@19.1.1) - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) optionalDependencies: typescript: 5.9.3 '@stylistic/eslint-plugin@5.5.0(eslint@9.38.0(jiti@1.21.7))': dependencies: '@eslint-community/eslint-utils': 4.9.0(eslint@9.38.0(jiti@1.21.7)) - '@typescript-eslint/types': 8.46.1 + '@typescript-eslint/types': 8.46.2 eslint: 9.38.0(jiti@1.21.7) eslint-visitor-keys: 4.2.1 espree: 10.4.0 @@ -11851,17 +11425,13 @@ snapshots: dependencies: '@testing-library/dom': 10.4.1 - '@tsconfig/node10@1.0.11': - optional: true + '@tsconfig/node10@1.0.11': {} - '@tsconfig/node12@1.0.11': - optional: true + '@tsconfig/node12@1.0.11': {} - '@tsconfig/node14@1.0.3': - optional: true + '@tsconfig/node14@1.0.3': {} - '@tsconfig/node16@1.0.4': - optional: true + '@tsconfig/node16@1.0.4': {} '@tybys/wasm-util@0.10.1': dependencies: @@ -11898,9 +11468,10 @@ snapshots: '@types/node': 18.15.0 '@types/responselike': 1.0.3 - '@types/chai@5.2.2': + '@types/chai@5.2.3': dependencies: '@types/deep-eql': 4.0.2 + assertion-error: 2.0.1 '@types/d3-array@3.2.2': {} @@ -12115,7 +11686,7 @@ snapshots: '@types/node@18.15.0': {} - '@types/node@20.19.22': + '@types/node@20.19.23': dependencies: undici-types: 6.21.0 @@ -12179,14 +11750,14 @@ snapshots: dependencies: '@types/yargs-parser': 21.0.3 - '@typescript-eslint/eslint-plugin@8.46.1(@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': + '@typescript-eslint/eslint-plugin@8.46.2(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: '@eslint-community/regexpp': 4.12.1 - '@typescript-eslint/parser': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/visitor-keys': 8.46.1 + '@typescript-eslint/parser': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/visitor-keys': 8.46.2 eslint: 9.38.0(jiti@1.21.7) graphemer: 1.4.0 ignore: 7.0.5 @@ -12196,41 +11767,41 @@ snapshots: transitivePeerDependencies: - supports-color - '@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': + '@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/typescript-estree': 8.46.1(typescript@5.9.3) - '@typescript-eslint/visitor-keys': 8.46.1 + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/typescript-estree': 8.46.2(typescript@5.9.3) + '@typescript-eslint/visitor-keys': 8.46.2 debug: 4.4.3 eslint: 9.38.0(jiti@1.21.7) typescript: 5.9.3 transitivePeerDependencies: - supports-color - '@typescript-eslint/project-service@8.46.1(typescript@5.9.3)': + '@typescript-eslint/project-service@8.46.2(typescript@5.9.3)': dependencies: - '@typescript-eslint/tsconfig-utils': 8.46.1(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 + '@typescript-eslint/tsconfig-utils': 8.46.2(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 debug: 4.4.3 typescript: 5.9.3 transitivePeerDependencies: - supports-color - '@typescript-eslint/scope-manager@8.46.1': + '@typescript-eslint/scope-manager@8.46.2': dependencies: - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/visitor-keys': 8.46.1 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/visitor-keys': 8.46.2 - '@typescript-eslint/tsconfig-utils@8.46.1(typescript@5.9.3)': + '@typescript-eslint/tsconfig-utils@8.46.2(typescript@5.9.3)': dependencies: typescript: 5.9.3 - '@typescript-eslint/type-utils@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': + '@typescript-eslint/type-utils@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/typescript-estree': 8.46.1(typescript@5.9.3) - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/typescript-estree': 8.46.2(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) debug: 4.4.3 eslint: 9.38.0(jiti@1.21.7) ts-api-utils: 2.1.0(typescript@5.9.3) @@ -12238,14 +11809,14 @@ snapshots: transitivePeerDependencies: - supports-color - '@typescript-eslint/types@8.46.1': {} + '@typescript-eslint/types@8.46.2': {} - '@typescript-eslint/typescript-estree@8.46.1(typescript@5.9.3)': + '@typescript-eslint/typescript-estree@8.46.2(typescript@5.9.3)': dependencies: - '@typescript-eslint/project-service': 8.46.1(typescript@5.9.3) - '@typescript-eslint/tsconfig-utils': 8.46.1(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/visitor-keys': 8.46.1 + '@typescript-eslint/project-service': 8.46.2(typescript@5.9.3) + '@typescript-eslint/tsconfig-utils': 8.46.2(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/visitor-keys': 8.46.2 debug: 4.4.3 fast-glob: 3.3.3 is-glob: 4.0.3 @@ -12256,28 +11827,28 @@ snapshots: transitivePeerDependencies: - supports-color - '@typescript-eslint/utils@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': + '@typescript-eslint/utils@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: '@eslint-community/eslint-utils': 4.9.0(eslint@9.38.0(jiti@1.21.7)) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/typescript-estree': 8.46.1(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/typescript-estree': 8.46.2(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) typescript: 5.9.3 transitivePeerDependencies: - supports-color - '@typescript-eslint/visitor-keys@8.46.1': + '@typescript-eslint/visitor-keys@8.46.2': dependencies: - '@typescript-eslint/types': 8.46.1 + '@typescript-eslint/types': 8.46.2 eslint-visitor-keys: 4.2.1 '@ungap/structured-clone@1.3.0': {} '@vitest/eslint-plugin@1.3.23(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3)': dependencies: - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) optionalDependencies: typescript: 5.9.3 @@ -12286,19 +11857,17 @@ snapshots: '@vitest/expect@3.2.4': dependencies: - '@types/chai': 5.2.2 + '@types/chai': 5.2.3 '@vitest/spy': 3.2.4 '@vitest/utils': 3.2.4 chai: 5.3.3 tinyrainbow: 2.0.0 - '@vitest/mocker@3.2.4(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1))': + '@vitest/mocker@3.2.4': dependencies: '@vitest/spy': 3.2.4 estree-walker: 3.0.3 magic-string: 0.30.19 - optionalDependencies: - vite: 6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1) '@vitest/pretty-format@3.2.4': dependencies: @@ -12314,14 +11883,6 @@ snapshots: loupe: 3.2.1 tinyrainbow: 2.0.0 - '@vue/compiler-core@3.5.17': - dependencies: - '@babel/parser': 7.28.4 - '@vue/shared': 3.5.17 - entities: 4.5.0 - estree-walker: 2.0.2 - source-map-js: 1.2.1 - '@vue/compiler-core@3.5.22': dependencies: '@babel/parser': 7.28.4 @@ -12330,34 +11891,27 @@ snapshots: estree-walker: 2.0.2 source-map-js: 1.2.1 - '@vue/compiler-dom@3.5.17': - dependencies: - '@vue/compiler-core': 3.5.17 - '@vue/shared': 3.5.17 - '@vue/compiler-dom@3.5.22': dependencies: '@vue/compiler-core': 3.5.22 '@vue/shared': 3.5.22 - '@vue/compiler-sfc@3.5.17': + '@vue/compiler-sfc@3.5.22': dependencies: - '@babel/parser': 7.28.4 - '@vue/compiler-core': 3.5.17 - '@vue/compiler-dom': 3.5.17 - '@vue/compiler-ssr': 3.5.17 - '@vue/shared': 3.5.17 + '@babel/parser': 7.28.5 + '@vue/compiler-core': 3.5.22 + '@vue/compiler-dom': 3.5.22 + '@vue/compiler-ssr': 3.5.22 + '@vue/shared': 3.5.22 estree-walker: 2.0.2 - magic-string: 0.30.19 + magic-string: 0.30.21 postcss: 8.5.6 source-map-js: 1.2.1 - '@vue/compiler-ssr@3.5.17': + '@vue/compiler-ssr@3.5.22': dependencies: - '@vue/compiler-dom': 3.5.17 - '@vue/shared': 3.5.17 - - '@vue/shared@3.5.17': {} + '@vue/compiler-dom': 3.5.22 + '@vue/shared': 3.5.22 '@vue/shared@3.5.22': {} @@ -12470,7 +12024,7 @@ snapshots: dependencies: '@babel/runtime': 7.28.4 '@types/js-cookie': 3.0.6 - dayjs: 1.11.18 + dayjs: 1.11.19 intersection-observer: 0.12.2 js-cookie: 3.0.5 lodash: 4.17.21 @@ -12543,8 +12097,7 @@ snapshots: are-docs-informative@0.0.2: {} - arg@4.1.3: - optional: true + arg@4.1.3: {} arg@5.0.2: {} @@ -13168,8 +12721,11 @@ snapshots: - supports-color - ts-node - create-require@1.1.1: - optional: true + create-require@1.1.1: {} + + cron-parser@5.4.0: + dependencies: + luxon: 3.7.2 cross-env@10.1.0: dependencies: @@ -13412,7 +12968,7 @@ snapshots: d3: 7.9.0 lodash-es: 4.17.21 - dayjs@1.11.18: {} + dayjs@1.11.19: {} debounce@1.2.1: {} @@ -13491,8 +13047,7 @@ snapshots: diff-sequences@29.6.3: {} - diff@4.0.2: - optional: true + diff@4.0.2: {} diffie-hellman@5.0.3: dependencies: @@ -13532,6 +13087,8 @@ snapshots: dependencies: domelementtype: 2.3.0 + dompurify@3.1.7: {} + dompurify@3.3.0: optionalDependencies: '@types/trusted-types': 2.0.7 @@ -13691,36 +13248,6 @@ snapshots: '@esbuild/win32-ia32': 0.25.0 '@esbuild/win32-x64': 0.25.0 - esbuild@0.25.11: - optionalDependencies: - '@esbuild/aix-ppc64': 0.25.11 - '@esbuild/android-arm': 0.25.11 - '@esbuild/android-arm64': 0.25.11 - '@esbuild/android-x64': 0.25.11 - '@esbuild/darwin-arm64': 0.25.11 - '@esbuild/darwin-x64': 0.25.11 - '@esbuild/freebsd-arm64': 0.25.11 - '@esbuild/freebsd-x64': 0.25.11 - '@esbuild/linux-arm': 0.25.11 - '@esbuild/linux-arm64': 0.25.11 - '@esbuild/linux-ia32': 0.25.11 - '@esbuild/linux-loong64': 0.25.11 - '@esbuild/linux-mips64el': 0.25.11 - '@esbuild/linux-ppc64': 0.25.11 - '@esbuild/linux-riscv64': 0.25.11 - '@esbuild/linux-s390x': 0.25.11 - '@esbuild/linux-x64': 0.25.11 - '@esbuild/netbsd-arm64': 0.25.11 - '@esbuild/netbsd-x64': 0.25.11 - '@esbuild/openbsd-arm64': 0.25.11 - '@esbuild/openbsd-x64': 0.25.11 - '@esbuild/openharmony-arm64': 0.25.11 - '@esbuild/sunos-x64': 0.25.11 - '@esbuild/win32-arm64': 0.25.11 - '@esbuild/win32-ia32': 0.25.11 - '@esbuild/win32-x64': 0.25.11 - optional: true - escalade@3.2.0: {} escape-string-regexp@1.0.5: {} @@ -13779,7 +13306,7 @@ snapshots: eslint-plugin-import-lite@0.3.0(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3): dependencies: '@eslint-community/eslint-utils': 4.9.0(eslint@9.38.0(jiti@1.21.7)) - '@typescript-eslint/types': 8.46.1 + '@typescript-eslint/types': 8.46.2 eslint: 9.38.0(jiti@1.21.7) optionalDependencies: typescript: 5.9.3 @@ -13839,8 +13366,8 @@ snapshots: eslint-plugin-perfectionist@4.15.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3): dependencies: - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) natural-orderby: 5.0.0 transitivePeerDependencies: @@ -13865,10 +13392,10 @@ snapshots: '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/var': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) string-ts: 2.2.1 ts-pattern: 5.8.0 @@ -13885,9 +13412,9 @@ snapshots: '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/var': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) compare-versions: 6.1.1 eslint: 9.38.0(jiti@1.21.7) string-ts: 2.2.1 @@ -13905,10 +13432,10 @@ snapshots: '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/var': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) string-ts: 2.2.1 ts-pattern: 5.8.0 @@ -13929,10 +13456,10 @@ snapshots: '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/var': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) string-ts: 2.2.1 ts-pattern: 5.8.0 @@ -13953,9 +13480,9 @@ snapshots: '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/var': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) string-ts: 2.2.1 ts-pattern: 5.8.0 @@ -13972,10 +13499,10 @@ snapshots: '@eslint-react/kit': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/shared': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) '@eslint-react/var': 1.53.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.46.1 - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/types': 8.46.1 - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) compare-versions: 6.1.1 eslint: 9.38.0(jiti@1.21.7) is-immutable-type: 5.0.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) @@ -14012,11 +13539,11 @@ snapshots: semver: 7.7.2 typescript: 5.9.3 - eslint-plugin-storybook@9.1.13(eslint@9.38.0(jiti@1.21.7))(storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)))(typescript@5.9.3): + eslint-plugin-storybook@9.1.13(eslint@9.38.0(jiti@1.21.7))(storybook@9.1.13(@testing-library/dom@10.4.1))(typescript@5.9.3): dependencies: - '@typescript-eslint/utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) - storybook: 9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + storybook: 9.1.13(@testing-library/dom@10.4.1) transitivePeerDependencies: - supports-color - typescript @@ -14059,13 +13586,13 @@ snapshots: semver: 7.7.3 strip-indent: 4.1.1 - eslint-plugin-unused-imports@4.3.0(@typescript-eslint/eslint-plugin@8.46.1(@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7)): + eslint-plugin-unused-imports@4.3.0(@typescript-eslint/eslint-plugin@8.46.2(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7)): dependencies: eslint: 9.38.0(jiti@1.21.7) optionalDependencies: - '@typescript-eslint/eslint-plugin': 8.46.1(@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/eslint-plugin': 8.46.2(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) - eslint-plugin-vue@10.5.1(@stylistic/eslint-plugin@5.5.0(eslint@9.38.0(jiti@1.21.7)))(@typescript-eslint/parser@8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(vue-eslint-parser@10.2.0(eslint@9.38.0(jiti@1.21.7))): + eslint-plugin-vue@10.5.1(@stylistic/eslint-plugin@5.5.0(eslint@9.38.0(jiti@1.21.7)))(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3))(eslint@9.38.0(jiti@1.21.7))(vue-eslint-parser@10.2.0(eslint@9.38.0(jiti@1.21.7))): dependencies: '@eslint-community/eslint-utils': 4.9.0(eslint@9.38.0(jiti@1.21.7)) eslint: 9.38.0(jiti@1.21.7) @@ -14077,7 +13604,7 @@ snapshots: xml-name-validator: 4.0.0 optionalDependencies: '@stylistic/eslint-plugin': 5.5.0(eslint@9.38.0(jiti@1.21.7)) - '@typescript-eslint/parser': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/parser': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint-plugin-yml@1.19.0(eslint@9.38.0(jiti@1.21.7)): dependencies: @@ -14091,9 +13618,9 @@ snapshots: transitivePeerDependencies: - supports-color - eslint-processor-vue-blocks@2.0.0(@vue/compiler-sfc@3.5.17)(eslint@9.38.0(jiti@1.21.7)): + eslint-processor-vue-blocks@2.0.0(@vue/compiler-sfc@3.5.22)(eslint@9.38.0(jiti@1.21.7)): dependencies: - '@vue/compiler-sfc': 3.5.17 + '@vue/compiler-sfc': 3.5.22 eslint: 9.38.0(jiti@1.21.7) eslint-scope@5.1.1: @@ -14119,7 +13646,7 @@ snapshots: '@eslint/core': 0.16.0 '@eslint/eslintrc': 3.3.1 '@eslint/js': 9.38.0 - '@eslint/plugin-kit': 0.3.5 + '@eslint/plugin-kit': 0.3.4 '@humanfs/node': 0.16.7 '@humanwhocodes/module-importer': 1.0.1 '@humanwhocodes/retry': 0.4.3 @@ -14544,9 +14071,9 @@ snapshots: hachure-fill@0.5.2: {} - happy-dom@20.0.7: + happy-dom@20.0.8: dependencies: - '@types/node': 20.19.22 + '@types/node': 20.19.23 '@types/whatwg-mimetype': 3.0.2 whatwg-mimetype: 3.0.0 @@ -14900,7 +14427,7 @@ snapshots: is-immutable-type@5.0.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3): dependencies: - '@typescript-eslint/type-utils': 8.46.1(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@1.21.7))(typescript@5.9.3) eslint: 9.38.0(jiti@1.21.7) ts-api-utils: 2.1.0(typescript@5.9.3) ts-declaration-location: 1.0.7(typescript@5.9.3) @@ -15174,7 +14701,7 @@ snapshots: jest-pnp-resolver: 1.2.3(jest-resolve@29.7.0) jest-util: 29.7.0 jest-validate: 29.7.0 - resolve: 1.22.10 + resolve: 1.22.11 resolve.exports: 2.0.3 slash: 3.0.0 @@ -15392,7 +14919,7 @@ snapshots: kleur@3.0.3: {} - knip@5.66.1(@types/node@18.15.0)(typescript@5.9.3): + knip@5.66.2(@types/node@18.15.0)(typescript@5.9.3): dependencies: '@nodelib/fs.walk': 1.2.8 '@types/node': 18.15.0 @@ -15401,7 +14928,7 @@ snapshots: jiti: 2.6.1 js-yaml: 4.1.0 minimist: 1.2.8 - oxc-resolver: 11.10.0 + oxc-resolver: 11.11.0 picocolors: 1.1.1 picomatch: 4.0.3 smol-toml: 1.4.2 @@ -15548,6 +15075,8 @@ snapshots: dependencies: yallist: 3.1.1 + luxon@3.7.2: {} + lz-string@1.5.0: {} magic-string@0.25.9: @@ -15558,6 +15087,10 @@ snapshots: dependencies: '@jridgewell/sourcemap-codec': 1.5.5 + magic-string@0.30.21: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.5 + magicast@0.3.5: dependencies: '@babel/parser': 7.28.4 @@ -15572,8 +15105,7 @@ snapshots: dependencies: semver: 7.7.3 - make-error@1.3.6: - optional: true + make-error@1.3.6: {} makeerror@1.0.12: dependencies: @@ -15583,6 +15115,8 @@ snapshots: markdown-table@3.0.4: {} + marked@14.0.0: {} + marked@15.0.12: {} md5.js@1.3.5: @@ -15804,7 +15338,7 @@ snapshots: d3: 7.9.0 d3-sankey: 0.12.3 dagre-d3-es: 7.0.11 - dayjs: 1.11.18 + dayjs: 1.11.19 dompurify: 3.3.0 katex: 0.16.25 khroma: 2.1.0 @@ -16164,7 +15698,10 @@ snapshots: pkg-types: 1.3.1 ufo: 1.6.1 - monaco-editor@0.52.2: {} + monaco-editor@0.54.0: + dependencies: + dompurify: 3.1.7 + marked: 14.0.0 mrmime@2.0.1: {} @@ -16349,27 +15886,27 @@ snapshots: os-browserify@0.3.0: {} - oxc-resolver@11.10.0: + oxc-resolver@11.11.0: optionalDependencies: - '@oxc-resolver/binding-android-arm-eabi': 11.10.0 - '@oxc-resolver/binding-android-arm64': 11.10.0 - '@oxc-resolver/binding-darwin-arm64': 11.10.0 - '@oxc-resolver/binding-darwin-x64': 11.10.0 - '@oxc-resolver/binding-freebsd-x64': 11.10.0 - '@oxc-resolver/binding-linux-arm-gnueabihf': 11.10.0 - '@oxc-resolver/binding-linux-arm-musleabihf': 11.10.0 - '@oxc-resolver/binding-linux-arm64-gnu': 11.10.0 - '@oxc-resolver/binding-linux-arm64-musl': 11.10.0 - '@oxc-resolver/binding-linux-ppc64-gnu': 11.10.0 - '@oxc-resolver/binding-linux-riscv64-gnu': 11.10.0 - '@oxc-resolver/binding-linux-riscv64-musl': 11.10.0 - '@oxc-resolver/binding-linux-s390x-gnu': 11.10.0 - '@oxc-resolver/binding-linux-x64-gnu': 11.10.0 - '@oxc-resolver/binding-linux-x64-musl': 11.10.0 - '@oxc-resolver/binding-wasm32-wasi': 11.10.0 - '@oxc-resolver/binding-win32-arm64-msvc': 11.10.0 - '@oxc-resolver/binding-win32-ia32-msvc': 11.10.0 - '@oxc-resolver/binding-win32-x64-msvc': 11.10.0 + '@oxc-resolver/binding-android-arm-eabi': 11.11.0 + '@oxc-resolver/binding-android-arm64': 11.11.0 + '@oxc-resolver/binding-darwin-arm64': 11.11.0 + '@oxc-resolver/binding-darwin-x64': 11.11.0 + '@oxc-resolver/binding-freebsd-x64': 11.11.0 + '@oxc-resolver/binding-linux-arm-gnueabihf': 11.11.0 + '@oxc-resolver/binding-linux-arm-musleabihf': 11.11.0 + '@oxc-resolver/binding-linux-arm64-gnu': 11.11.0 + '@oxc-resolver/binding-linux-arm64-musl': 11.11.0 + '@oxc-resolver/binding-linux-ppc64-gnu': 11.11.0 + '@oxc-resolver/binding-linux-riscv64-gnu': 11.11.0 + '@oxc-resolver/binding-linux-riscv64-musl': 11.11.0 + '@oxc-resolver/binding-linux-s390x-gnu': 11.11.0 + '@oxc-resolver/binding-linux-x64-gnu': 11.11.0 + '@oxc-resolver/binding-linux-x64-musl': 11.11.0 + '@oxc-resolver/binding-wasm32-wasi': 11.11.0 + '@oxc-resolver/binding-win32-arm64-msvc': 11.11.0 + '@oxc-resolver/binding-win32-ia32-msvc': 11.11.0 + '@oxc-resolver/binding-win32-x64-msvc': 11.11.0 p-cancelable@2.1.1: {} @@ -16423,7 +15960,7 @@ snapshots: asn1.js: 4.10.1 browserify-aes: 1.2.0 evp_bytestokey: 1.0.3 - pbkdf2: 3.1.3 + pbkdf2: 3.1.5 safe-buffer: 5.2.1 parse-entities@2.0.0: @@ -16510,6 +16047,15 @@ snapshots: sha.js: 2.4.12 to-buffer: 1.2.2 + pbkdf2@3.1.5: + dependencies: + create-hash: 1.2.0 + create-hmac: 1.1.7 + ripemd160: 2.0.3 + safe-buffer: 5.2.1 + sha.js: 2.4.12 + to-buffer: 1.2.2 + pdfjs-dist@4.4.168: optionalDependencies: canvas: 3.2.0 @@ -16582,7 +16128,7 @@ snapshots: postcss: 8.5.6 postcss-value-parser: 4.2.0 read-cache: 1.0.0 - resolve: 1.22.10 + resolve: 1.22.11 postcss-js@4.1.0(postcss@8.5.6): dependencies: @@ -16804,7 +16350,7 @@ snapshots: '@types/doctrine': 0.0.9 '@types/resolve': 1.20.6 doctrine: 3.0.0 - resolve: 1.22.10 + resolve: 1.22.11 strip-indent: 4.1.1 transitivePeerDependencies: - supports-color @@ -17227,7 +16773,7 @@ snapshots: resolve.exports@2.0.3: {} - resolve@1.22.10: + resolve@1.22.11: dependencies: is-core-module: '@nolyfill/is-core-module@1.0.39' path-parse: 1.0.7 @@ -17278,35 +16824,6 @@ snapshots: optionalDependencies: fsevents: 2.3.3 - rollup@4.52.5: - dependencies: - '@types/estree': 1.0.8 - optionalDependencies: - '@rollup/rollup-android-arm-eabi': 4.52.5 - '@rollup/rollup-android-arm64': 4.52.5 - '@rollup/rollup-darwin-arm64': 4.52.5 - '@rollup/rollup-darwin-x64': 4.52.5 - '@rollup/rollup-freebsd-arm64': 4.52.5 - '@rollup/rollup-freebsd-x64': 4.52.5 - '@rollup/rollup-linux-arm-gnueabihf': 4.52.5 - '@rollup/rollup-linux-arm-musleabihf': 4.52.5 - '@rollup/rollup-linux-arm64-gnu': 4.52.5 - '@rollup/rollup-linux-arm64-musl': 4.52.5 - '@rollup/rollup-linux-loong64-gnu': 4.52.5 - '@rollup/rollup-linux-ppc64-gnu': 4.52.5 - '@rollup/rollup-linux-riscv64-gnu': 4.52.5 - '@rollup/rollup-linux-riscv64-musl': 4.52.5 - '@rollup/rollup-linux-s390x-gnu': 4.52.5 - '@rollup/rollup-linux-x64-gnu': 4.52.5 - '@rollup/rollup-linux-x64-musl': 4.52.5 - '@rollup/rollup-openharmony-arm64': 4.52.5 - '@rollup/rollup-win32-arm64-msvc': 4.52.5 - '@rollup/rollup-win32-ia32-msvc': 4.52.5 - '@rollup/rollup-win32-x64-gnu': 4.52.5 - '@rollup/rollup-win32-x64-msvc': 4.52.5 - fsevents: 2.3.3 - optional: true - roughjs@4.6.6: dependencies: hachure-fill: 0.5.2 @@ -17541,13 +17058,13 @@ snapshots: state-local@1.0.7: {} - storybook@9.1.13(@testing-library/dom@10.4.1)(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)): + storybook@9.1.13(@testing-library/dom@10.4.1): dependencies: '@storybook/global': 5.0.0 '@testing-library/jest-dom': 6.9.1 '@testing-library/user-event': 14.6.1(@testing-library/dom@10.4.1) '@vitest/expect': 3.2.4 - '@vitest/mocker': 3.2.4(vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1)) + '@vitest/mocker': 3.2.4 '@vitest/spy': 3.2.4 better-opn: 3.0.2 esbuild: 0.25.0 @@ -17724,7 +17241,7 @@ snapshots: postcss-load-config: 6.0.1(jiti@1.21.7)(postcss@8.5.6)(yaml@2.8.1) postcss-nested: 6.2.0(postcss@8.5.6) postcss-selector-parser: 6.1.2 - resolve: 1.22.10 + resolve: 1.22.11 sucrase: 3.35.0 transitivePeerDependencies: - tsx @@ -17876,7 +17393,6 @@ snapshots: typescript: 5.9.3 v8-compile-cache-lib: 3.0.1 yn: 3.1.1 - optional: true ts-pattern@5.8.0: {} @@ -18079,8 +17595,7 @@ snapshots: uuid@11.1.0: {} - v8-compile-cache-lib@3.0.1: - optional: true + v8-compile-cache-lib@3.0.1: {} v8-to-istanbul@9.3.0: dependencies: @@ -18103,20 +17618,6 @@ snapshots: '@types/unist': 3.0.3 vfile-message: 4.0.3 - vite@6.2.7(@types/node@18.15.0)(jiti@1.21.7)(sass@1.93.2)(terser@5.44.0)(yaml@2.8.1): - dependencies: - esbuild: 0.25.11 - postcss: 8.5.6 - rollup: 4.52.5 - optionalDependencies: - '@types/node': 18.15.0 - fsevents: 2.3.3 - jiti: 1.21.7 - sass: 1.93.2 - terser: 5.44.0 - yaml: 2.8.1 - optional: true - vm-browserify@1.1.2: {} void-elements@3.1.0: {} @@ -18442,8 +17943,7 @@ snapshots: dependencies: lib0: 0.2.114 - yn@3.1.1: - optional: true + yn@3.1.1: {} yocto-queue@0.1.0: {} diff --git a/web/scripts/copy-and-start.mjs b/web/scripts/copy-and-start.mjs new file mode 100644 index 0000000000..b23ce636a4 --- /dev/null +++ b/web/scripts/copy-and-start.mjs @@ -0,0 +1,115 @@ +#!/usr/bin/env node +/** + * This script copies static files to the target directory and starts the server. + * It is intended to be used as a replacement for `next start`. + */ + +import { cp, mkdir, stat } from 'node:fs/promises' +import { spawn } from 'node:child_process' +import path from 'node:path' + +// Configuration for directories to copy +const DIRS_TO_COPY = [ + { + src: path.join('.next', 'static'), + dest: path.join('.next', 'standalone', '.next', 'static'), + }, + { + src: 'public', + dest: path.join('.next', 'standalone', 'public'), + }, +] + +// Path to the server script +const SERVER_SCRIPT_PATH = path.join('.next', 'standalone', 'server.js') + +// Function to check if a path exists +const pathExists = async (path) => { + try { + console.debug(`Checking if path exists: ${path}`) + await stat(path) + console.debug(`Path exists: ${path}`) + return true + } + catch (err) { + if (err.code === 'ENOENT') { + console.warn(`Path does not exist: ${path}`) + return false + } + throw err + } +} + +// Function to recursively copy directories +const copyDir = async (src, dest) => { + console.debug(`Copying directory from ${src} to ${dest}`) + await cp(src, dest, { recursive: true }) + console.info(`Successfully copied ${src} to ${dest}`) +} + +// Process each directory copy operation +const copyAllDirs = async () => { + console.debug('Starting directory copy operations') + for (const { src, dest } of DIRS_TO_COPY) { + try { + // Instead of pre-creating destination directory, we ensure parent directory exists + const destParent = path.dirname(dest) + console.debug(`Ensuring destination parent directory exists: ${destParent}`) + await mkdir(destParent, { recursive: true }) + if (await pathExists(src)) { + await copyDir(src, dest) + } + else { + console.error(`Error: ${src} directory does not exist. This is a required build artifact.`) + process.exit(1) + } + } + catch (err) { + console.error(`Error processing ${src}:`, err.message) + process.exit(1) + } + } + console.debug('Finished directory copy operations') +} + +// Run copy operations and start server +const main = async () => { + console.debug('Starting copy-and-start script') + await copyAllDirs() + + // Start server + const port = process.env.npm_config_port || process.env.PORT || '3000' + const host = process.env.npm_config_host || process.env.HOSTNAME || '0.0.0.0' + + console.info(`Starting server on ${host}:${port}`) + console.debug(`Server script path: ${SERVER_SCRIPT_PATH}`) + console.debug(`Environment variables - PORT: ${port}, HOSTNAME: ${host}`) + + const server = spawn( + process.execPath, + [SERVER_SCRIPT_PATH], + { + env: { + ...process.env, + PORT: port, + HOSTNAME: host, + }, + stdio: 'inherit', + }, + ) + + server.on('error', (err) => { + console.error('Failed to start server:', err) + process.exit(1) + }) + + server.on('exit', (code) => { + console.debug(`Server exited with code: ${code}`) + process.exit(code || 0) + }) +} + +main().catch((err) => { + console.error('Unexpected error:', err) + process.exit(1) +}) diff --git a/web/service/_tools_util.spec.ts b/web/service/_tools_util.spec.ts index f06e5a1e34..658c276df1 100644 --- a/web/service/_tools_util.spec.ts +++ b/web/service/_tools_util.spec.ts @@ -14,3 +14,39 @@ describe('makeProviderQuery', () => { expect(buildProviderQuery('ABC?DEF')).toBe('provider=ABC%3FDEF') }) }) + +describe('Tools Utilities', () => { + describe('buildProviderQuery', () => { + it('should build query string with provider parameter', () => { + const result = buildProviderQuery('openai') + expect(result).toBe('provider=openai') + }) + + it('should handle provider names with special characters', () => { + const result = buildProviderQuery('provider-name') + expect(result).toBe('provider=provider-name') + }) + + it('should handle empty string', () => { + const result = buildProviderQuery('') + expect(result).toBe('provider=') + }) + + it('should URL encode special characters', () => { + const result = buildProviderQuery('provider name') + expect(result).toBe('provider=provider+name') + }) + + it('should handle Unicode characters', () => { + const result = buildProviderQuery('提供者') + expect(result).toContain('provider=') + expect(decodeURIComponent(result)).toBe('provider=提供者') + }) + + it('should handle provider names with slashes', () => { + const result = buildProviderQuery('langgenius/openai/gpt-4') + expect(result).toContain('provider=') + expect(decodeURIComponent(result)).toBe('provider=langgenius/openai/gpt-4') + }) + }) +}) diff --git a/web/service/apps.ts b/web/service/apps.ts index 5602f75791..b1124767ad 100644 --- a/web/service/apps.ts +++ b/web/service/apps.ts @@ -1,8 +1,8 @@ import type { Fetcher } from 'swr' import { del, get, patch, post, put } from './base' -import type { ApiKeysListResponse, AppDailyConversationsResponse, AppDailyEndUsersResponse, AppDailyMessagesResponse, AppDetailResponse, AppListResponse, AppStatisticsResponse, AppTemplatesResponse, AppTokenCostsResponse, AppVoicesListResponse, CreateApiKeyResponse, DSLImportMode, DSLImportResponse, GenerationIntroductionResponse, TracingConfig, TracingStatus, UpdateAppModelConfigResponse, UpdateAppSiteCodeResponse, UpdateOpenAIKeyResponse, ValidateOpenAIKeyResponse, WorkflowDailyConversationsResponse } from '@/models/app' +import type { ApiKeysListResponse, AppDailyConversationsResponse, AppDailyEndUsersResponse, AppDailyMessagesResponse, AppDetailResponse, AppListResponse, AppStatisticsResponse, AppTemplatesResponse, AppTokenCostsResponse, AppVoicesListResponse, CreateApiKeyResponse, DSLImportMode, DSLImportResponse, GenerationIntroductionResponse, TracingConfig, TracingStatus, UpdateAppModelConfigResponse, UpdateAppSiteCodeResponse, UpdateOpenAIKeyResponse, ValidateOpenAIKeyResponse, WebhookTriggerResponse, WorkflowDailyConversationsResponse } from '@/models/app' import type { CommonResponse } from '@/models/common' -import type { AppIconType, AppMode, ModelConfig } from '@/types/app' +import type { AppIconType, AppModeEnum, ModelConfig } from '@/types/app' import type { TracingProvider } from '@/app/(commonLayout)/app/(appDetailLayout)/[appId]/overview/tracing/type' export const fetchAppList: Fetcher<AppListResponse, { url: string; params?: Record<string, any> }> = ({ url, params }) => { @@ -22,7 +22,7 @@ export const fetchAppTemplates: Fetcher<AppTemplatesResponse, { url: string }> = return get<AppTemplatesResponse>(url) } -export const createApp: Fetcher<AppDetailResponse, { name: string; icon_type?: AppIconType; icon?: string; icon_background?: string; mode: AppMode; description?: string; config?: ModelConfig }> = ({ name, icon_type, icon, icon_background, mode, description, config }) => { +export const createApp: Fetcher<AppDetailResponse, { name: string; icon_type?: AppIconType; icon?: string; icon_background?: string; mode: AppModeEnum; description?: string; config?: ModelConfig }> = ({ name, icon_type, icon, icon_background, mode, description, config }) => { return post<AppDetailResponse>('apps', { body: { name, icon_type, icon, icon_background, mode, description, model_config: config } }) } @@ -31,7 +31,7 @@ export const updateAppInfo: Fetcher<AppDetailResponse, { appID: string; name: st return put<AppDetailResponse>(`apps/${appID}`, { body }) } -export const copyApp: Fetcher<AppDetailResponse, { appID: string; name: string; icon_type: AppIconType; icon: string; icon_background?: string | null; mode: AppMode; description?: string }> = ({ appID, name, icon_type, icon, icon_background, mode, description }) => { +export const copyApp: Fetcher<AppDetailResponse, { appID: string; name: string; icon_type: AppIconType; icon: string; icon_background?: string | null; mode: AppModeEnum; description?: string }> = ({ appID, name, icon_type, icon, icon_background, mode, description }) => { return post<AppDetailResponse>(`apps/${appID}/copy`, { body: { name, icon_type, icon, icon_background, mode, description } }) } @@ -162,6 +162,11 @@ export const updateTracingStatus: Fetcher<CommonResponse, { appId: string; body: return post(`/apps/${appId}/trace`, { body }) } +// Webhook Trigger +export const fetchWebhookUrl: Fetcher<WebhookTriggerResponse, { appId: string; nodeId: string }> = ({ appId, nodeId }) => { + return get<WebhookTriggerResponse>(`apps/${appId}/workflows/triggers/webhook`, { params: { node_id: nodeId } }) +} + export const fetchTracingConfig: Fetcher<TracingConfig & { has_not_configured: true }, { appId: string; provider: TracingProvider }> = ({ appId, provider }) => { return get(`/apps/${appId}/trace-config`, { params: { diff --git a/web/service/base.ts b/web/service/base.ts index ccb48bd46b..e966fa74aa 100644 --- a/web/service/base.ts +++ b/web/service/base.ts @@ -155,7 +155,7 @@ export function format(text: string) { return res.replaceAll('\n', '<br/>').replaceAll('```', '') } -const handleStream = ( +export const handleStream = ( response: Response, onData: IOnData, onCompleted?: IOnCompleted, diff --git a/web/service/debug.ts b/web/service/debug.ts index 61057c591f..c2862f2ddb 100644 --- a/web/service/debug.ts +++ b/web/service/debug.ts @@ -1,8 +1,9 @@ import { get, post, ssePost } from './base' import type { IOnCompleted, IOnData, IOnError, IOnMessageReplace } from './base' import type { ChatPromptConfig, CompletionPromptConfig } from '@/models/debug' -import type { ModelModeType } from '@/types/app' +import type { AppModeEnum, ModelModeType } from '@/types/app' import type { ModelParameterRule } from '@/app/components/header/account-setting/model-provider-page/declarations' + export type BasicAppFirstRes = { prompt: string variables: string[] @@ -87,7 +88,7 @@ export const fetchPromptTemplate = ({ mode, modelName, hasSetDataSet, -}: { appMode: string; mode: ModelModeType; modelName: string; hasSetDataSet: boolean }) => { +}: { appMode: AppModeEnum; mode: ModelModeType; modelName: string; hasSetDataSet: boolean }) => { return get<Promise<{ chat_prompt_config: ChatPromptConfig; completion_prompt_config: CompletionPromptConfig; stop: [] }>>('/app/prompt-templates', { params: { app_mode: appMode, diff --git a/web/service/demo/index.tsx b/web/service/demo/index.tsx index aa02968549..5cbfa7c52a 100644 --- a/web/service/demo/index.tsx +++ b/web/service/demo/index.tsx @@ -4,6 +4,8 @@ import React from 'react' import useSWR, { useSWRConfig } from 'swr' import { createApp, fetchAppDetail, fetchAppList, getAppDailyConversations, getAppDailyEndUsers, updateAppApiStatus, updateAppModelConfig, updateAppRateLimit, updateAppSiteAccessToken, updateAppSiteConfig, updateAppSiteStatus } from '../apps' import Loading from '@/app/components/base/loading' +import { AppModeEnum } from '@/types/app' + const Service: FC = () => { const { data: appList, error: appListError } = useSWR({ url: '/apps', params: { page: 1 } }, fetchAppList) const { data: firstApp, error: appDetailError } = useSWR({ url: '/apps', id: '1' }, fetchAppDetail) @@ -21,7 +23,7 @@ const Service: FC = () => { const handleCreateApp = async () => { await createApp({ name: `new app${Math.round(Math.random() * 100)}`, - mode: 'chat', + mode: AppModeEnum.CHAT, }) // reload app list mutate({ url: '/apps', params: { page: 1 } }) diff --git a/web/service/fetch.ts b/web/service/fetch.ts index 8d663c902b..030549bdab 100644 --- a/web/service/fetch.ts +++ b/web/service/fetch.ts @@ -2,7 +2,7 @@ import type { AfterResponseHook, BeforeErrorHook, BeforeRequestHook, Hooks } fro import ky from 'ky' import type { IOtherOptions } from './base' import Toast from '@/app/components/base/toast' -import { API_PREFIX, APP_VERSION, CSRF_COOKIE_NAME, CSRF_HEADER_NAME, MARKETPLACE_API_PREFIX, PASSPORT_HEADER_NAME, PUBLIC_API_PREFIX, WEB_APP_SHARE_CODE_HEADER_NAME } from '@/config' +import { API_PREFIX, APP_VERSION, CSRF_COOKIE_NAME, CSRF_HEADER_NAME, IS_MARKETPLACE, MARKETPLACE_API_PREFIX, PASSPORT_HEADER_NAME, PUBLIC_API_PREFIX, WEB_APP_SHARE_CODE_HEADER_NAME } from '@/config' import Cookies from 'js-cookie' import { getWebAppAccessToken, getWebAppPassport } from './webapp-auth' @@ -160,7 +160,7 @@ async function base<T>(url: string, options: FetchOptionType = {}, otherOptions: // ! For Marketplace API, help to filter tags added in new version if (isMarketplaceAPI) - (headers as any).set('X-Dify-Version', APP_VERSION) + (headers as any).set('X-Dify-Version', !IS_MARKETPLACE ? APP_VERSION : '999.0.0') const client = baseClient.extend({ hooks: { diff --git a/web/service/share.ts b/web/service/share.ts index fe2b9c2290..9129be6358 100644 --- a/web/service/share.ts +++ b/web/service/share.ts @@ -284,7 +284,8 @@ export const fetchAccessToken = async ({ userId, appCode }: { userId?: string, a if (accessToken) headers.append('Authorization', `Bearer ${accessToken}`) const params = new URLSearchParams() - userId && params.append('user_id', userId) + if (userId) + params.append('user_id', userId) const url = `/passport?${params.toString()}` return get<{ access_token: string }>(url, { headers }) as Promise<{ access_token: string }> } diff --git a/web/service/tools.ts b/web/service/tools.ts index 6a88d8d567..2897ccac12 100644 --- a/web/service/tools.ts +++ b/web/service/tools.ts @@ -8,8 +8,6 @@ import type { WorkflowToolProviderRequest, WorkflowToolProviderResponse, } from '@/app/components/tools/types' -import type { ToolWithProvider } from '@/app/components/workflow/types' -import type { Label } from '@/app/components/tools/labels/constant' import { buildProviderQuery } from './_tools_util' export const fetchCollectionList = () => { @@ -112,26 +110,6 @@ export const testAPIAvailable = (payload: any) => { }) } -export const fetchAllBuiltInTools = () => { - return get<ToolWithProvider[]>('/workspaces/current/tools/builtin') -} - -export const fetchAllCustomTools = () => { - return get<ToolWithProvider[]>('/workspaces/current/tools/api') -} - -export const fetchAllWorkflowTools = () => { - return get<ToolWithProvider[]>('/workspaces/current/tools/workflow') -} - -export const fetchAllMCPTools = () => { - return get<ToolWithProvider[]>('/workspaces/current/tools/mcp') -} - -export const fetchLabelList = () => { - return get<Label[]>('/workspaces/current/tool-labels') -} - export const createWorkflowToolProvider = (payload: WorkflowToolProviderRequest & { workflow_app_id: string }) => { return post('/workspaces/current/tool-provider/workflow/create', { body: { ...payload }, diff --git a/web/service/try-app.ts b/web/service/try-app.ts index 17624c7a1c..dc8c16b25b 100644 --- a/web/service/try-app.ts +++ b/web/service/try-app.ts @@ -1,11 +1,10 @@ -import type { AppMode } from '@/types/app' import { get, } from './base' import type { SiteInfo, } from '@/models/share' -import type { ModelConfig } from '@/types/app' +import type { AppModeEnum, ModelConfig } from '@/types/app' import qs from 'qs' import type { DataSetListResponse } from '@/models/datasets' import type { Edge, Node } from '@/app/components/workflow/types' @@ -14,7 +13,7 @@ import type { Viewport } from 'reactflow' export type TryAppInfo = { name: string description: string - mode: AppMode + mode: AppModeEnum site: SiteInfo model_config: ModelConfig deleted_tools: any[] diff --git a/web/service/use-base.ts b/web/service/use-base.ts index 37af55a74a..b6445f4baf 100644 --- a/web/service/use-base.ts +++ b/web/service/use-base.ts @@ -3,9 +3,11 @@ import { useQueryClient, } from '@tanstack/react-query' -export const useInvalid = (key: QueryKey) => { +export const useInvalid = (key?: QueryKey) => { const queryClient = useQueryClient() return () => { + if (!key) + return queryClient.invalidateQueries( { queryKey: key, @@ -14,9 +16,11 @@ export const useInvalid = (key: QueryKey) => { } } -export const useReset = (key: QueryKey) => { +export const useReset = (key?: QueryKey) => { const queryClient = useQueryClient() return () => { + if (!key) + return queryClient.resetQueries( { queryKey: key, diff --git a/web/service/use-pipeline.ts b/web/service/use-pipeline.ts index a7b9c89410..92a7542c56 100644 --- a/web/service/use-pipeline.ts +++ b/web/service/use-pipeline.ts @@ -39,13 +39,14 @@ import { useInvalid } from './use-base' const NAME_SPACE = 'pipeline' export const PipelineTemplateListQueryKeyPrefix = [NAME_SPACE, 'template-list'] -export const usePipelineTemplateList = (params: PipelineTemplateListParams) => { +export const usePipelineTemplateList = (params: PipelineTemplateListParams, enabled = true) => { const { type, language } = params return useQuery<PipelineTemplateListResponse>({ queryKey: [...PipelineTemplateListQueryKeyPrefix, type, language], queryFn: () => { return get<PipelineTemplateListResponse>('/rag/pipeline/templates', { params }) }, + enabled, }) } diff --git a/web/service/use-plugins.ts b/web/service/use-plugins.ts index f59e500792..f6dbecaeba 100644 --- a/web/service/use-plugins.ts +++ b/web/service/use-plugins.ts @@ -1,4 +1,4 @@ -import { useCallback, useEffect } from 'react' +import { useCallback, useEffect, useState } from 'react' import type { FormOption, ModelProvider, @@ -10,6 +10,7 @@ import type { Dependency, GitHubItemAndMarketPlaceDependency, InstallPackageResponse, + InstallStatusResponse, InstalledLatestVersionResponse, InstalledPluginListWithTotalResponse, PackageDependency, @@ -18,7 +19,6 @@ import type { PluginDetail, PluginInfoFromMarketPlace, PluginTask, - PluginType, PluginsFromMarketplaceByInfoResponse, PluginsFromMarketplaceResponse, ReferenceSetting, @@ -27,7 +27,7 @@ import type { uploadGitHubResponse, } from '@/app/components/plugins/types' import { TaskStatus } from '@/app/components/plugins/types' -import { PluginType as PluginTypeEnum } from '@/app/components/plugins/types' +import { PluginCategoryEnum } from '@/app/components/plugins/types' import type { PluginsSearchParams, } from '@/app/components/plugins/marketplace/types' @@ -39,11 +39,12 @@ import { useQuery, useQueryClient, } from '@tanstack/react-query' -import { useInvalidateAllBuiltInTools, useInvalidateRAGRecommendedPlugins } from './use-tools' +import { useInvalidateAllBuiltInTools } from './use-tools' import useReferenceSetting from '@/app/components/plugins/plugin-page/use-reference-setting' import { uninstallPlugin } from '@/service/plugins' import useRefreshPluginList from '@/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list' import { cloneDeep } from 'lodash-es' +import { getFormattedPlugin } from '@/app/components/plugins/marketplace/utils' const NAME_SPACE = 'plugins' @@ -67,6 +68,66 @@ export const useCheckInstalled = ({ }) } +const useRecommendedMarketplacePluginsKey = [NAME_SPACE, 'recommendedMarketplacePlugins'] +export const useRecommendedMarketplacePlugins = ({ + collection = '__recommended-plugins-tools', + enabled = true, + limit = 15, +}: { + collection?: string + enabled?: boolean + limit?: number +} = {}) => { + return useQuery<Plugin[]>({ + queryKey: [...useRecommendedMarketplacePluginsKey, collection, limit], + queryFn: async () => { + const response = await postMarketplace<{ data: { plugins: Plugin[] } }>( + `/collections/${collection}/plugins`, + { + body: { + limit, + }, + }, + ) + return response.data.plugins.map(plugin => getFormattedPlugin(plugin)) + }, + enabled, + staleTime: 60 * 1000, + }) +} + +export const useFeaturedToolsRecommendations = (enabled: boolean, limit = 15) => { + const { + data: plugins = [], + isLoading, + } = useRecommendedMarketplacePlugins({ + collection: '__recommended-plugins-tools', + enabled, + limit, + }) + + return { + plugins, + isLoading, + } +} + +export const useFeaturedTriggersRecommendations = (enabled: boolean, limit = 15) => { + const { + data: plugins = [], + isLoading, + } = useRecommendedMarketplacePlugins({ + collection: '__recommended-plugins-triggers', + enabled, + limit, + }) + + return { + plugins, + isLoading, + } +} + export const useInstalledPluginList = (disable?: boolean, pageSize = 100) => { const fetchPlugins = async ({ pageParam = 1 }) => { const response = await get<InstalledPluginListWithTotalResponse>( @@ -135,14 +196,12 @@ export const useInstalledLatestVersion = (pluginIds: string[]) => { export const useInvalidateInstalledPluginList = () => { const queryClient = useQueryClient() const invalidateAllBuiltInTools = useInvalidateAllBuiltInTools() - const invalidateRAGRecommendedPlugins = useInvalidateRAGRecommendedPlugins() return () => { queryClient.invalidateQueries( { queryKey: useInstalledPluginListKey, }) invalidateAllBuiltInTools() - invalidateRAGRecommendedPlugins() } } @@ -235,7 +294,7 @@ export const useUploadGitHub = (payload: { export const useInstallOrUpdate = ({ onSuccess, }: { - onSuccess?: (res: { success: boolean }[]) => void + onSuccess?: (res: InstallStatusResponse[]) => void }) => { const { mutateAsync: updatePackageFromMarketPlace } = useUpdatePackageFromMarketPlace() @@ -253,6 +312,8 @@ export const useInstallOrUpdate = ({ const installedPayload = installedInfo[orgAndName] const isInstalled = !!installedPayload let uniqueIdentifier = '' + let taskId = '' + let isFinishedInstallation = false if (item.type === 'github') { const data = item as GitHubItemAndMarketPlaceDependency @@ -270,12 +331,14 @@ export const useInstallOrUpdate = ({ // has the same version, but not installed if (uniqueIdentifier === installedPayload?.uniqueIdentifier) { return { - success: true, + status: TaskStatus.success, + taskId: '', + uniqueIdentifier: '', } } } if (!isInstalled) { - await post<InstallPackageResponse>('/workspaces/current/plugin/install/github', { + const { task_id, all_installed } = await post<InstallPackageResponse>('/workspaces/current/plugin/install/github', { body: { repo: data.value.repo!, version: data.value.release! || data.value.version!, @@ -283,6 +346,8 @@ export const useInstallOrUpdate = ({ plugin_unique_identifier: uniqueIdentifier, }, }) + taskId = task_id + isFinishedInstallation = all_installed } } if (item.type === 'marketplace') { @@ -290,15 +355,19 @@ export const useInstallOrUpdate = ({ uniqueIdentifier = data.value.marketplace_plugin_unique_identifier! || plugin[i]?.plugin_id if (uniqueIdentifier === installedPayload?.uniqueIdentifier) { return { - success: true, + status: TaskStatus.success, + taskId: '', + uniqueIdentifier: '', } } if (!isInstalled) { - await post<InstallPackageResponse>('/workspaces/current/plugin/install/marketplace', { + const { task_id, all_installed } = await post<InstallPackageResponse>('/workspaces/current/plugin/install/marketplace', { body: { plugin_unique_identifiers: [uniqueIdentifier], }, }) + taskId = task_id + isFinishedInstallation = all_installed } } if (item.type === 'package') { @@ -306,38 +375,59 @@ export const useInstallOrUpdate = ({ uniqueIdentifier = data.value.unique_identifier if (uniqueIdentifier === installedPayload?.uniqueIdentifier) { return { - success: true, + status: TaskStatus.success, + taskId: '', + uniqueIdentifier: '', } } if (!isInstalled) { - await post<InstallPackageResponse>('/workspaces/current/plugin/install/pkg', { + const { task_id, all_installed } = await post<InstallPackageResponse>('/workspaces/current/plugin/install/pkg', { body: { plugin_unique_identifiers: [uniqueIdentifier], }, }) + taskId = task_id + isFinishedInstallation = all_installed } } if (isInstalled) { if (item.type === 'package') { await uninstallPlugin(installedPayload.installedId) - await post<InstallPackageResponse>('/workspaces/current/plugin/install/pkg', { + const { task_id, all_installed } = await post<InstallPackageResponse>('/workspaces/current/plugin/install/pkg', { body: { plugin_unique_identifiers: [uniqueIdentifier], }, }) + taskId = task_id + isFinishedInstallation = all_installed } else { - await updatePackageFromMarketPlace({ + const { task_id, all_installed } = await updatePackageFromMarketPlace({ original_plugin_unique_identifier: installedPayload?.uniqueIdentifier, new_plugin_unique_identifier: uniqueIdentifier, }) + taskId = task_id + isFinishedInstallation = all_installed + } + } + if (isFinishedInstallation) { + return { + status: TaskStatus.success, + taskId: '', + uniqueIdentifier: '', + } + } + else { + return { + status: TaskStatus.running, + taskId, + uniqueIdentifier, } } - return ({ success: true }) } // eslint-disable-next-line unused-imports/no-unused-vars catch (e) { - return Promise.resolve({ success: false }) + return Promise.resolve({ status: TaskStatus.failed, taskId: '', uniqueIdentifier: '' }) } })) }, @@ -488,7 +578,8 @@ export const useFetchPluginsInMarketPlaceByInfo = (infos: Record<string, any>[]) } const usePluginTaskListKey = [NAME_SPACE, 'pluginTaskList'] -export const usePluginTaskList = (category?: PluginType) => { +export const usePluginTaskList = (category?: PluginCategoryEnum | string) => { + const [initialized, setInitialized] = useState(false) const { canManagement, } = useReferenceSetting() @@ -512,16 +603,21 @@ export const usePluginTaskList = (category?: PluginType) => { useEffect(() => { // After first fetch, refresh plugin list each time all tasks are done - if (!isRefetching) { - const lastData = cloneDeep(data) - const taskDone = lastData?.tasks.every(task => task.status === TaskStatus.success || task.status === TaskStatus.failed) - const taskAllFailed = lastData?.tasks.every(task => task.status === TaskStatus.failed) - if (taskDone) { - if (lastData?.tasks.length && !taskAllFailed) - refreshPluginList(category ? { category } as any : undefined, !category) - } - } - }, [isRefetching]) + // Skip initialization period, because the query cache is not updated yet + if (!initialized || isRefetching) + return + + const lastData = cloneDeep(data) + const taskDone = lastData?.tasks.every(task => task.status === TaskStatus.success || task.status === TaskStatus.failed) + const taskAllFailed = lastData?.tasks.every(task => task.status === TaskStatus.failed) + if (taskDone && lastData?.tasks.length && !taskAllFailed) + refreshPluginList(category ? { category } as any : undefined, !category) + }, [initialized, isRefetching, data, category, refreshPluginList]) + + useEffect(() => { + if (isFetched && !initialized) + setInitialized(true) + }, [isFetched, initialized]) const handleRefetch = useCallback(() => { refetch() @@ -605,7 +701,7 @@ export const usePluginInfo = (providerName?: string) => { const name = parts[1] try { const response = await fetchPluginInfoFromMarketPlace({ org, name }) - return response.data.plugin.category === PluginTypeEnum.model ? response.data.plugin : null + return response.data.plugin.category === PluginCategoryEnum.model ? response.data.plugin : null } catch { return null @@ -615,7 +711,7 @@ export const usePluginInfo = (providerName?: string) => { }) } -export const useFetchDynamicOptions = (plugin_id: string, provider: string, action: string, parameter: string, provider_type: 'tool') => { +export const useFetchDynamicOptions = (plugin_id: string, provider: string, action: string, parameter: string, provider_type?: string, extra?: Record<string, any>) => { return useMutation({ mutationFn: () => get<{ options: FormOption[] }>('/workspaces/current/plugin/parameters/dynamic-options', { params: { @@ -624,7 +720,26 @@ export const useFetchDynamicOptions = (plugin_id: string, provider: string, acti action, parameter, provider_type, + ...extra, }, }), }) } + +export const usePluginReadme = ({ plugin_unique_identifier, language }: { plugin_unique_identifier: string, language?: string }) => { + return useQuery({ + queryKey: ['pluginReadme', plugin_unique_identifier, language], + queryFn: () => get<{ readme: string }>('/workspaces/current/plugin/readme', { params: { plugin_unique_identifier, language } }, { silent: true }), + enabled: !!plugin_unique_identifier, + retry: 0, + }) +} + +export const usePluginReadmeAsset = ({ file_name, plugin_unique_identifier }: { file_name?: string, plugin_unique_identifier?: string }) => { + const normalizedFileName = file_name?.replace(/(^\.\/_assets\/|^_assets\/)/, '') + return useQuery({ + queryKey: ['pluginReadmeAsset', plugin_unique_identifier, file_name], + queryFn: () => get<Blob>('/workspaces/current/plugin/asset', { params: { plugin_unique_identifier, file_name: normalizedFileName } }, { silent: true }), + enabled: !!plugin_unique_identifier && !!file_name && /(^\.\/_assets|^_assets)/.test(file_name), + }) +} diff --git a/web/service/use-tools.ts b/web/service/use-tools.ts index a881441cd5..ad483bea11 100644 --- a/web/service/use-tools.ts +++ b/web/service/use-tools.ts @@ -4,9 +4,11 @@ import type { MCPServerDetail, Tool, } from '@/app/components/tools/types' +import { CollectionType } from '@/app/components/tools/types' import type { RAGRecommendedPlugins, ToolWithProvider } from '@/app/components/workflow/types' import type { AppIconType } from '@/types/app' import { useInvalid } from './use-base' +import type { QueryKey } from '@tanstack/react-query' import { useMutation, useQuery, @@ -76,6 +78,17 @@ export const useInvalidateAllMCPTools = () => { return useInvalid(useAllMCPToolsKey) } +const useInvalidToolsKeyMap: Record<string, QueryKey> = { + [CollectionType.builtIn]: useAllBuiltInToolsKey, + [CollectionType.custom]: useAllCustomToolsKey, + [CollectionType.workflow]: useAllWorkflowToolsKey, + [CollectionType.mcp]: useAllMCPToolsKey, +} +export const useInvalidToolsByType = (type?: CollectionType | string) => { + const queryKey = type ? useInvalidToolsKeyMap[type] : undefined + return useInvalid(queryKey) +} + export const useCreateMCP = () => { return useMutation({ mutationKey: [NAME_SPACE, 'create-mcp'], @@ -327,3 +340,53 @@ export const useRAGRecommendedPlugins = () => { export const useInvalidateRAGRecommendedPlugins = () => { return useInvalid(useRAGRecommendedPluginListKey) } + +// App Triggers API hooks +export type AppTrigger = { + id: string + trigger_type: 'trigger-webhook' | 'trigger-schedule' | 'trigger-plugin' + title: string + node_id: string + provider_name: string + icon: string + status: 'enabled' | 'disabled' | 'unauthorized' + created_at: string + updated_at: string +} + +export const useAppTriggers = (appId: string | undefined, options?: any) => { + return useQuery<{ data: AppTrigger[] }>({ + queryKey: [NAME_SPACE, 'app-triggers', appId], + queryFn: () => get<{ data: AppTrigger[] }>(`/apps/${appId}/triggers`), + enabled: !!appId, + ...options, // Merge additional options while maintaining backward compatibility + }) +} + +export const useInvalidateAppTriggers = () => { + const queryClient = useQueryClient() + return (appId: string) => { + queryClient.invalidateQueries({ + queryKey: [NAME_SPACE, 'app-triggers', appId], + }) + } +} + +export const useUpdateTriggerStatus = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'update-trigger-status'], + mutationFn: (payload: { + appId: string + triggerId: string + enableTrigger: boolean + }) => { + const { appId, triggerId, enableTrigger } = payload + return post<AppTrigger>(`/apps/${appId}/trigger-enable`, { + body: { + trigger_id: triggerId, + enable_trigger: enableTrigger, + }, + }) + }, + }) +} diff --git a/web/service/use-triggers.ts b/web/service/use-triggers.ts new file mode 100644 index 0000000000..cfb786e4a9 --- /dev/null +++ b/web/service/use-triggers.ts @@ -0,0 +1,320 @@ +import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query' +import { del, get, post } from './base' +import type { + TriggerLogEntity, + TriggerOAuthClientParams, + TriggerOAuthConfig, + TriggerProviderApiEntity, + TriggerSubscription, + TriggerSubscriptionBuilder, + TriggerWithProvider, +} from '@/app/components/workflow/block-selector/types' +import { CollectionType } from '@/app/components/tools/types' +import { useInvalid } from './use-base' + +const NAME_SPACE = 'triggers' + +// Trigger Provider Service - Provider ID Format: plugin_id/provider_name + +// Convert backend API response to frontend ToolWithProvider format +const convertToTriggerWithProvider = (provider: TriggerProviderApiEntity): TriggerWithProvider => { + return { + // Collection fields + id: provider.plugin_id || provider.name, + name: provider.name, + author: provider.author, + description: provider.description, + icon: provider.icon || '', + label: provider.label, + type: CollectionType.trigger, + team_credentials: {}, + is_team_authorization: false, + allow_delete: false, + labels: provider.tags || [], + plugin_id: provider.plugin_id, + plugin_unique_identifier: provider.plugin_unique_identifier || '', + events: provider.events.map(event => ({ + name: event.name, + author: provider.author, + label: event.identity.label, + description: event.description, + parameters: event.parameters.map(param => ({ + name: param.name, + label: param.label, + human_description: param.description || param.label, + type: param.type, + form: param.type, + llm_description: JSON.stringify(param.description || {}), + required: param.required || false, + default: param.default || '', + options: param.options?.map(option => ({ + label: option.label, + value: option.value, + })) || [], + multiple: param.multiple || false, + })), + labels: provider.tags || [], + output_schema: event.output_schema || {}, + })), + + // Trigger-specific schema fields + subscription_constructor: provider.subscription_constructor, + subscription_schema: provider.subscription_schema, + supported_creation_methods: provider.supported_creation_methods, + + meta: { + version: '1.0', + }, + } +} + +export const useAllTriggerPlugins = (enabled = true) => { + return useQuery<TriggerWithProvider[]>({ + queryKey: [NAME_SPACE, 'all'], + queryFn: async () => { + const response = await get<TriggerProviderApiEntity[]>('/workspaces/current/triggers') + return response.map(convertToTriggerWithProvider) + }, + enabled, + staleTime: 0, + gcTime: 0, + }) +} + +export const useTriggerPluginsByType = (triggerType: string, enabled = true) => { + return useQuery<TriggerWithProvider[]>({ + queryKey: [NAME_SPACE, 'byType', triggerType], + queryFn: async () => { + const response = await get<TriggerProviderApiEntity[]>(`/workspaces/current/triggers?type=${triggerType}`) + return response.map(convertToTriggerWithProvider) + }, + enabled: enabled && !!triggerType, + }) +} + +export const useInvalidateAllTriggerPlugins = () => { + return useInvalid([NAME_SPACE, 'all']) +} + +// ===== Trigger Subscriptions Management ===== + +export const useTriggerProviderInfo = (provider: string, enabled = true) => { + return useQuery<TriggerProviderApiEntity>({ + queryKey: [NAME_SPACE, 'provider-info', provider], + queryFn: () => get<TriggerProviderApiEntity>(`/workspaces/current/trigger-provider/${provider}/info`), + enabled: enabled && !!provider, + staleTime: 0, + gcTime: 0, + }) +} + +export const useTriggerSubscriptions = (provider: string, enabled = true) => { + return useQuery<TriggerSubscription[]>({ + queryKey: [NAME_SPACE, 'list-subscriptions', provider], + queryFn: () => get<TriggerSubscription[]>(`/workspaces/current/trigger-provider/${provider}/subscriptions/list`), + enabled: enabled && !!provider, + }) +} + +export const useInvalidateTriggerSubscriptions = () => { + const queryClient = useQueryClient() + return (provider: string) => { + queryClient.invalidateQueries({ + queryKey: [NAME_SPACE, 'subscriptions', provider], + }) + } +} + +export const useCreateTriggerSubscriptionBuilder = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'create-subscription-builder'], + mutationFn: (payload: { + provider: string + credential_type?: string + }) => { + const { provider, ...body } = payload + return post<{ subscription_builder: TriggerSubscriptionBuilder }>( + `/workspaces/current/trigger-provider/${provider}/subscriptions/builder/create`, + { body }, + ) + }, + }) +} + +export const useUpdateTriggerSubscriptionBuilder = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'update-subscription-builder'], + mutationFn: (payload: { + provider: string + subscriptionBuilderId: string + name?: string + properties?: Record<string, any> + parameters?: Record<string, any> + credentials?: Record<string, any> + }) => { + const { provider, subscriptionBuilderId, ...body } = payload + return post<TriggerSubscriptionBuilder>( + `/workspaces/current/trigger-provider/${provider}/subscriptions/builder/update/${subscriptionBuilderId}`, + { body }, + ) + }, + }) +} + +export const useVerifyTriggerSubscriptionBuilder = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'verify-subscription-builder'], + mutationFn: (payload: { + provider: string + subscriptionBuilderId: string + credentials?: Record<string, any> + }) => { + const { provider, subscriptionBuilderId, ...body } = payload + return post<{ verified: boolean }>( + `/workspaces/current/trigger-provider/${provider}/subscriptions/builder/verify/${subscriptionBuilderId}`, + { body }, + { silent: true }, + ) + }, + }) +} + +export type BuildTriggerSubscriptionPayload = { + provider: string + subscriptionBuilderId: string + name?: string + parameters?: Record<string, any> +} + +export const useBuildTriggerSubscription = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'build-subscription'], + mutationFn: (payload: BuildTriggerSubscriptionPayload) => { + const { provider, subscriptionBuilderId, ...body } = payload + return post( + `/workspaces/current/trigger-provider/${provider}/subscriptions/builder/build/${subscriptionBuilderId}`, + { body }, + ) + }, + }) +} + +export const useDeleteTriggerSubscription = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'delete-subscription'], + mutationFn: (subscriptionId: string) => { + return post<{ result: string }>( + `/workspaces/current/trigger-provider/${subscriptionId}/subscriptions/delete`, + ) + }, + }) +} + +export const useTriggerSubscriptionBuilderLogs = ( + provider: string, + subscriptionBuilderId: string, + options: { + enabled?: boolean + refetchInterval?: number | false + } = {}, +) => { + const { enabled = true, refetchInterval = false } = options + + return useQuery<{ logs: TriggerLogEntity[] }>({ + queryKey: [NAME_SPACE, 'subscription-builder-logs', provider, subscriptionBuilderId], + queryFn: () => get( + `/workspaces/current/trigger-provider/${provider}/subscriptions/builder/logs/${subscriptionBuilderId}`, + ), + enabled: enabled && !!provider && !!subscriptionBuilderId, + refetchInterval, + }) +} + +// ===== OAuth Management ===== +export const useTriggerOAuthConfig = (provider: string, enabled = true) => { + return useQuery<TriggerOAuthConfig>({ + queryKey: [NAME_SPACE, 'oauth-config', provider], + queryFn: () => get<TriggerOAuthConfig>(`/workspaces/current/trigger-provider/${provider}/oauth/client`), + enabled: enabled && !!provider, + }) +} + +export type ConfigureTriggerOAuthPayload = { + provider: string + client_params?: TriggerOAuthClientParams + enabled: boolean +} + +export const useConfigureTriggerOAuth = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'configure-oauth'], + mutationFn: (payload: ConfigureTriggerOAuthPayload) => { + const { provider, ...body } = payload + return post<{ result: string }>( + `/workspaces/current/trigger-provider/${provider}/oauth/client`, + { body }, + ) + }, + }) +} + +export const useDeleteTriggerOAuth = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'delete-oauth'], + mutationFn: (provider: string) => { + return del<{ result: string }>( + `/workspaces/current/trigger-provider/${provider}/oauth/client`, + ) + }, + }) +} + +export const useInitiateTriggerOAuth = () => { + return useMutation({ + mutationKey: [NAME_SPACE, 'initiate-oauth'], + mutationFn: (provider: string) => { + return get<{ authorization_url: string; subscription_builder: TriggerSubscriptionBuilder }>( + `/workspaces/current/trigger-provider/${provider}/subscriptions/oauth/authorize`, + {}, + { silent: true }, + ) + }, + }) +} + +// ===== Dynamic Options Support ===== +export const useTriggerPluginDynamicOptions = (payload: { + plugin_id: string + provider: string + action: string + parameter: string + credential_id: string + extra?: Record<string, any> +}, enabled = true) => { + return useQuery<{ options: Array<{ value: string; label: any }> }>({ + queryKey: [NAME_SPACE, 'dynamic-options', payload.plugin_id, payload.provider, payload.action, payload.parameter, payload.credential_id, payload.extra], + queryFn: () => get<{ options: Array<{ value: string; label: any }> }>( + '/workspaces/current/plugin/parameters/dynamic-options', + { + params: { + ...payload, + provider_type: 'trigger', // Add required provider_type parameter + }, + }, + { silent: true }, + ), + enabled: enabled && !!payload.plugin_id && !!payload.provider && !!payload.action && !!payload.parameter && !!payload.credential_id, + retry: 0, + }) +} + +// ===== Cache Invalidation Helpers ===== + +export const useInvalidateTriggerOAuthConfig = () => { + const queryClient = useQueryClient() + return (provider: string) => { + queryClient.invalidateQueries({ + queryKey: [NAME_SPACE, 'oauth-config', provider], + }) + } +} diff --git a/web/service/utils.spec.ts b/web/service/utils.spec.ts new file mode 100644 index 0000000000..fc5385c309 --- /dev/null +++ b/web/service/utils.spec.ts @@ -0,0 +1,170 @@ +/** + * Test suite for service utility functions + * + * This module provides utilities for working with different flow types in the application. + * Flow types determine the API endpoint prefix used for various operations. + * + * Key concepts: + * - FlowType.appFlow: Standard application workflows (prefix: 'apps') + * - FlowType.ragPipeline: RAG (Retrieval-Augmented Generation) pipelines (prefix: 'rag/pipelines') + * + * The getFlowPrefix function maps flow types to their corresponding API path prefixes, + * with a fallback to 'apps' for undefined or unknown flow types. + */ +import { flowPrefixMap, getFlowPrefix } from './utils' +import { FlowType } from '@/types/common' + +describe('Service Utils', () => { + describe('flowPrefixMap', () => { + /** + * Test that the flowPrefixMap object contains the expected mappings + * This ensures the mapping configuration is correct + */ + it('should have correct flow type to prefix mappings', () => { + expect(flowPrefixMap[FlowType.appFlow]).toBe('apps') + expect(flowPrefixMap[FlowType.ragPipeline]).toBe('rag/pipelines') + }) + + /** + * Test that the map only contains the expected flow types + * This helps catch unintended additions to the mapping + */ + it('should contain exactly two flow type mappings', () => { + const keys = Object.keys(flowPrefixMap) + expect(keys).toHaveLength(2) + }) + }) + + describe('getFlowPrefix', () => { + /** + * Test that appFlow type returns the correct prefix + * This is the most common flow type for standard application workflows + */ + it('should return "apps" for appFlow type', () => { + const result = getFlowPrefix(FlowType.appFlow) + expect(result).toBe('apps') + }) + + /** + * Test that ragPipeline type returns the correct prefix + * RAG pipelines have a different API structure with nested paths + */ + it('should return "rag/pipelines" for ragPipeline type', () => { + const result = getFlowPrefix(FlowType.ragPipeline) + expect(result).toBe('rag/pipelines') + }) + + /** + * Test fallback behavior when no flow type is provided + * Should default to 'apps' prefix for backward compatibility + */ + it('should return "apps" when flow type is undefined', () => { + const result = getFlowPrefix(undefined) + expect(result).toBe('apps') + }) + + /** + * Test fallback behavior for unknown flow types + * Any unrecognized flow type should default to 'apps' + */ + it('should return "apps" for unknown flow type', () => { + // Cast to FlowType to test the fallback behavior + const unknownType = 'unknown' as FlowType + const result = getFlowPrefix(unknownType) + expect(result).toBe('apps') + }) + + /** + * Test that the function handles null gracefully + * Null should be treated the same as undefined + */ + it('should return "apps" when flow type is null', () => { + const result = getFlowPrefix(null as any) + expect(result).toBe('apps') + }) + + /** + * Test consistency with flowPrefixMap + * The function should return the same values as direct map access + */ + it('should return values consistent with flowPrefixMap', () => { + expect(getFlowPrefix(FlowType.appFlow)).toBe(flowPrefixMap[FlowType.appFlow]) + expect(getFlowPrefix(FlowType.ragPipeline)).toBe(flowPrefixMap[FlowType.ragPipeline]) + }) + }) + + describe('Integration scenarios', () => { + /** + * Test typical usage pattern in API path construction + * This demonstrates how the function is used in real application code + */ + it('should construct correct API paths for different flow types', () => { + const appId = '123' + + // App flow path construction + const appFlowPath = `/${getFlowPrefix(FlowType.appFlow)}/${appId}` + expect(appFlowPath).toBe('/apps/123') + + // RAG pipeline path construction + const ragPipelinePath = `/${getFlowPrefix(FlowType.ragPipeline)}/${appId}` + expect(ragPipelinePath).toBe('/rag/pipelines/123') + }) + + /** + * Test that the function can be used in conditional logic + * Common pattern for determining which API endpoint to use + */ + it('should support conditional API routing logic', () => { + const determineEndpoint = (flowType?: FlowType, resourceId?: string) => { + const prefix = getFlowPrefix(flowType) + return `/${prefix}/${resourceId || 'default'}` + } + + expect(determineEndpoint(FlowType.appFlow, 'app-1')).toBe('/apps/app-1') + expect(determineEndpoint(FlowType.ragPipeline, 'pipeline-1')).toBe('/rag/pipelines/pipeline-1') + expect(determineEndpoint(undefined, 'fallback')).toBe('/apps/fallback') + }) + + /** + * Test behavior with empty string flow type + * Empty strings should fall back to default + */ + it('should handle empty string as flow type', () => { + const result = getFlowPrefix('' as any) + expect(result).toBe('apps') + }) + }) + + describe('Type safety', () => { + /** + * Test that all FlowType enum values are handled + * This ensures we don't miss any flow types in the mapping + */ + it('should handle all FlowType enum values', () => { + // Get all enum values + const flowTypes = Object.values(FlowType) + + // Each flow type should return a valid prefix + flowTypes.forEach((flowType) => { + const prefix = getFlowPrefix(flowType) + expect(prefix).toBeTruthy() + expect(typeof prefix).toBe('string') + expect(prefix.length).toBeGreaterThan(0) + }) + }) + + /** + * Test that returned prefixes are valid path segments + * Prefixes should not contain leading/trailing slashes or invalid characters + */ + it('should return valid path segments without leading/trailing slashes', () => { + const appFlowPrefix = getFlowPrefix(FlowType.appFlow) + const ragPipelinePrefix = getFlowPrefix(FlowType.ragPipeline) + + expect(appFlowPrefix).not.toMatch(/^\//) + expect(appFlowPrefix).not.toMatch(/\/$/) + expect(ragPipelinePrefix).not.toMatch(/^\//) + expect(ragPipelinePrefix).not.toMatch(/\/$/) + }) + }) +}) diff --git a/web/service/workflow-payload.ts b/web/service/workflow-payload.ts new file mode 100644 index 0000000000..b80c4a3731 --- /dev/null +++ b/web/service/workflow-payload.ts @@ -0,0 +1,152 @@ +import { produce } from 'immer' +import type { Edge, Node } from '@/app/components/workflow/types' +import { BlockEnum } from '@/app/components/workflow/types' +import type { PluginTriggerNodeType } from '@/app/components/workflow/nodes/trigger-plugin/types' +import type { FetchWorkflowDraftResponse } from '@/types/workflow' + +export type TriggerPluginNodePayload = { + title: string + desc: string + plugin_id: string + provider_id: string + event_name: string + subscription_id: string + plugin_unique_identifier: string + event_parameters: Record<string, unknown> +} + +export type WorkflowDraftSyncParams = Pick< + FetchWorkflowDraftResponse, + 'graph' | 'features' | 'environment_variables' | 'conversation_variables' +> + +const removeTempProperties = (data: Record<string, unknown>): void => { + Object.keys(data).forEach((key) => { + if (key.startsWith('_')) + delete data[key] + }) +} + +type TriggerParameterSchema = Record<string, unknown> + +type TriggerPluginHydratePayload = (PluginTriggerNodeType & { + paramSchemas?: TriggerParameterSchema[] + parameters_schema?: TriggerParameterSchema[] +}) + +const sanitizeTriggerPluginNode = (node: Node<TriggerPluginNodePayload>): Node<TriggerPluginNodePayload> => { + const data = node.data + + if (!data || data.type !== BlockEnum.TriggerPlugin) + return node + + const sanitizedData: TriggerPluginNodePayload & { type: BlockEnum.TriggerPlugin } = { + type: BlockEnum.TriggerPlugin, + title: data.title ?? '', + desc: data.desc ?? '', + plugin_id: data.plugin_id ?? '', + provider_id: data.provider_id ?? '', + event_name: data.event_name ?? '', + subscription_id: data.subscription_id ?? '', + plugin_unique_identifier: data.plugin_unique_identifier ?? '', + event_parameters: (typeof data.event_parameters === 'object' && data.event_parameters !== null) + ? data.event_parameters as Record<string, unknown> + : {}, + } + + return { + ...node, + data: sanitizedData, + } +} + +export const sanitizeWorkflowDraftPayload = (params: WorkflowDraftSyncParams): WorkflowDraftSyncParams => { + const { graph } = params + + if (!graph?.nodes?.length) + return params + + const sanitizedNodes = graph.nodes.map(node => sanitizeTriggerPluginNode(node as Node<TriggerPluginNodePayload>)) + + return { + ...params, + graph: { + ...graph, + nodes: sanitizedNodes, + }, + } +} + +const isTriggerPluginNode = (node: Node): node is Node<TriggerPluginHydratePayload> => { + const data = node.data as unknown + + if (!data || typeof data !== 'object') + return false + + const payload = data as Partial<TriggerPluginHydratePayload> & { type?: BlockEnum } + + if (payload.type !== BlockEnum.TriggerPlugin) + return false + + return 'event_parameters' in payload +} + +const hydrateTriggerPluginNode = (node: Node): Node => { + if (!isTriggerPluginNode(node)) + return node + + const typedNode = node as Node<TriggerPluginHydratePayload> + const data = typedNode.data + const eventParameters = data.event_parameters ?? {} + const parametersSchema = data.parameters_schema ?? data.paramSchemas ?? [] + const config = data.config ?? eventParameters ?? {} + + const nextData: typeof data = { + ...data, + config, + paramSchemas: data.paramSchemas ?? parametersSchema, + parameters_schema: parametersSchema, + } + + return { + ...typedNode, + data: nextData, + } +} + +export const hydrateWorkflowDraftResponse = (draft: FetchWorkflowDraftResponse): FetchWorkflowDraftResponse => { + return produce(draft, (mutableDraft) => { + if (!mutableDraft?.graph) + return + + if (mutableDraft.graph.nodes) { + mutableDraft.graph.nodes = mutableDraft.graph.nodes + .filter((node: Node) => !node.data?._isTempNode) + .map((node: Node) => { + if (node.data) + removeTempProperties(node.data as Record<string, unknown>) + + return hydrateTriggerPluginNode(node) + }) + } + + if (mutableDraft.graph.edges) { + mutableDraft.graph.edges = mutableDraft.graph.edges + .filter((edge: Edge) => !edge.data?._isTemp) + .map((edge: Edge) => { + if (edge.data) + removeTempProperties(edge.data as Record<string, unknown>) + + return edge + }) + } + + if (mutableDraft.environment_variables) { + mutableDraft.environment_variables = mutableDraft.environment_variables.map(env => + env.value_type === 'secret' + ? { ...env, value: '[__HIDDEN__]' } + : env, + ) + } + }) +} diff --git a/web/themes/dark.css b/web/themes/dark.css index cd1a016f75..dae2add2b1 100644 --- a/web/themes/dark.css +++ b/web/themes/dark.css @@ -435,7 +435,7 @@ html[data-theme="dark"] { --color-workflow-block-bg: #27272b; --color-workflow-block-bg-transparent: rgb(39 39 43 / 0.96); --color-workflow-block-border-highlight: rgb(200 206 218 / 0.2); - --color-workflow-block-wrapper-bg-1: #27272b; + --color-workflow-block-wrapper-bg-1: #323236; --color-workflow-block-wrapper-bg-2: rgb(39 39 43 / 0.2); --color-workflow-canvas-workflow-dot-color: rgb(133 133 173 / 0.11); diff --git a/web/tsconfig.json b/web/tsconfig.json index 3b022e4708..1d03daa576 100644 --- a/web/tsconfig.json +++ b/web/tsconfig.json @@ -40,6 +40,11 @@ "app/components/develop/Prose.jsx" ], "exclude": [ - "node_modules" + "node_modules", + "**/*.test.ts", + "**/*.test.tsx", + "**/*.spec.ts", + "**/*.spec.tsx", + "__tests__/**" ] } diff --git a/web/types/app.ts b/web/types/app.ts index 0a12e28729..73e11d396a 100644 --- a/web/types/app.ts +++ b/web/types/app.ts @@ -8,6 +8,7 @@ import type { } from '@/models/datasets' import type { UploadFileSetting } from '@/app/components/workflow/types' import type { AccessMode } from '@/models/access-control' +import type { ExternalDataTool } from '@/models/common' export enum Theme { light = 'light', @@ -59,8 +60,14 @@ export type VariableInput = { /** * App modes */ -export const AppModes = ['advanced-chat', 'agent-chat', 'chat', 'completion', 'workflow'] as const -export type AppMode = typeof AppModes[number] +export enum AppModeEnum { + COMPLETION = 'completion', + WORKFLOW = 'workflow', + CHAT = 'chat', + ADVANCED_CHAT = 'advanced-chat', + AGENT_CHAT = 'agent-chat', +} +export const AppModes = [AppModeEnum.COMPLETION, AppModeEnum.WORKFLOW, AppModeEnum.CHAT, AppModeEnum.ADVANCED_CHAT, AppModeEnum.AGENT_CHAT] as const /** * Variable type @@ -206,12 +213,12 @@ export type ModelConfig = { suggested_questions?: string[] pre_prompt: string prompt_type: PromptMode - chat_prompt_config: ChatPromptConfig | {} - completion_prompt_config: CompletionPromptConfig | {} + chat_prompt_config?: ChatPromptConfig | null + completion_prompt_config?: CompletionPromptConfig | null user_input_form: UserInputFormItem[] dataset_query_variable?: string more_like_this: { - enabled?: boolean + enabled: boolean } suggested_questions_after_answer: { enabled: boolean @@ -237,13 +244,20 @@ export type ModelConfig = { strategy?: AgentStrategy tools: ToolItem[] } + external_data_tools?: ExternalDataTool[] model: Model dataset_configs: DatasetConfigs file_upload?: { image: VisionSettings } & UploadFileSetting files?: VisionFile[] - external_data_tools: any[] + system_parameters: { + audio_file_size_limit: number + file_size_limit: number + image_file_size_limit: number + video_file_size_limit: number + workflow_file_upload_limit: number + } created_at?: number updated_at?: number } @@ -331,7 +345,7 @@ export type App = { use_icon_as_answer_icon: boolean /** Mode */ - mode: AppMode + mode: AppModeEnum /** Enable web app */ enable_site: boolean /** Enable web API */ @@ -361,9 +375,12 @@ export type App = { updated_at: number updated_by?: string } + deleted_tools?: Array<{ id: string; tool_name: string }> /** access control */ access_mode: AccessMode max_active_requests?: number | null + /** whether workflow trigger has un-published draft */ + has_draft_trigger?: boolean } export type AppSSO = { @@ -379,7 +396,7 @@ export type AppTemplate = { /** Description */ description: string /** Mode */ - mode: AppMode + mode: AppModeEnum /** Model */ model_config: ModelConfig } diff --git a/web/types/assets.d.ts b/web/types/assets.d.ts new file mode 100644 index 0000000000..d7711f7eb4 --- /dev/null +++ b/web/types/assets.d.ts @@ -0,0 +1,24 @@ +declare module '*.svg' { + const value: any + export default value +} + +declare module '*.png' { + const value: any + export default value +} + +declare module '*.jpg' { + const value: any + export default value +} + +declare module '*.jpeg' { + const value: any + export default value +} + +declare module '*.gif' { + const value: any + export default value +} diff --git a/web/types/feature.ts b/web/types/feature.ts index 432cd32ce5..fb0e3ada49 100644 --- a/web/types/feature.ts +++ b/web/types/feature.ts @@ -126,6 +126,7 @@ export enum DatasetAttr { DATA_PUBLIC_ENABLE_WEBSITE_JINAREADER = 'data-public-enable-website-jinareader', DATA_PUBLIC_ENABLE_WEBSITE_FIRECRAWL = 'data-public-enable-website-firecrawl', DATA_PUBLIC_ENABLE_WEBSITE_WATERCRAWL = 'data-public-enable-website-watercrawl', + DATA_PUBLIC_ENABLE_SINGLE_DOLLAR_LATEX = 'data-public-enable-single-dollar-latex', NEXT_PUBLIC_ZENDESK_WIDGET_KEY = 'next-public-zendesk-widget-key', NEXT_PUBLIC_ZENDESK_FIELD_ID_ENVIRONMENT = 'next-public-zendesk-field-id-environment', NEXT_PUBLIC_ZENDESK_FIELD_ID_VERSION = 'next-public-zendesk-field-id-version', diff --git a/web/types/i18n.d.ts b/web/types/i18n.d.ts index a6ed8f0a1e..826fcc1613 100644 --- a/web/types/i18n.d.ts +++ b/web/types/i18n.d.ts @@ -27,6 +27,7 @@ type LoginMessages = typeof import('../i18n/en-US/login').default type OauthMessages = typeof import('../i18n/en-US/oauth').default type PipelineMessages = typeof import('../i18n/en-US/pipeline').default type PluginTagsMessages = typeof import('../i18n/en-US/plugin-tags').default +type PluginTriggerMessages = typeof import('../i18n/en-US/plugin-trigger').default type PluginMessages = typeof import('../i18n/en-US/plugin').default type RegisterMessages = typeof import('../i18n/en-US/register').default type RunLogMessages = typeof import('../i18n/en-US/run-log').default @@ -59,6 +60,7 @@ export type Messages = { oauth: OauthMessages; pipeline: PipelineMessages; pluginTags: PluginTagsMessages; + pluginTrigger: PluginTriggerMessages; plugin: PluginMessages; register: RegisterMessages; runLog: RunLogMessages; diff --git a/web/utils/app-redirection.spec.ts b/web/utils/app-redirection.spec.ts new file mode 100644 index 0000000000..8a41d4d010 --- /dev/null +++ b/web/utils/app-redirection.spec.ts @@ -0,0 +1,106 @@ +/** + * Test suite for app redirection utility functions + * Tests navigation path generation based on user permissions and app modes + */ +import { getRedirection, getRedirectionPath } from './app-redirection' + +describe('app-redirection', () => { + /** + * Tests getRedirectionPath which determines the correct path based on: + * - User's editor permissions + * - App mode (workflow, advanced-chat, chat, completion, agent-chat) + */ + describe('getRedirectionPath', () => { + test('returns overview path when user is not editor', () => { + const app = { id: 'app-123', mode: 'chat' as const } + const result = getRedirectionPath(false, app) + expect(result).toBe('/app/app-123/overview') + }) + + test('returns workflow path for workflow mode when user is editor', () => { + const app = { id: 'app-123', mode: 'workflow' as const } + const result = getRedirectionPath(true, app) + expect(result).toBe('/app/app-123/workflow') + }) + + test('returns workflow path for advanced-chat mode when user is editor', () => { + const app = { id: 'app-123', mode: 'advanced-chat' as const } + const result = getRedirectionPath(true, app) + expect(result).toBe('/app/app-123/workflow') + }) + + test('returns configuration path for chat mode when user is editor', () => { + const app = { id: 'app-123', mode: 'chat' as const } + const result = getRedirectionPath(true, app) + expect(result).toBe('/app/app-123/configuration') + }) + + test('returns configuration path for completion mode when user is editor', () => { + const app = { id: 'app-123', mode: 'completion' as const } + const result = getRedirectionPath(true, app) + expect(result).toBe('/app/app-123/configuration') + }) + + test('returns configuration path for agent-chat mode when user is editor', () => { + const app = { id: 'app-456', mode: 'agent-chat' as const } + const result = getRedirectionPath(true, app) + expect(result).toBe('/app/app-456/configuration') + }) + + test('handles different app IDs', () => { + const app1 = { id: 'abc-123', mode: 'chat' as const } + const app2 = { id: 'xyz-789', mode: 'workflow' as const } + + expect(getRedirectionPath(false, app1)).toBe('/app/abc-123/overview') + expect(getRedirectionPath(true, app2)).toBe('/app/xyz-789/workflow') + }) + }) + + /** + * Tests getRedirection which combines path generation with a redirect callback + */ + describe('getRedirection', () => { + /** + * Tests that the redirection function is called with the correct path + */ + test('calls redirection function with correct path for non-editor', () => { + const app = { id: 'app-123', mode: 'chat' as const } + const mockRedirect = jest.fn() + + getRedirection(false, app, mockRedirect) + + expect(mockRedirect).toHaveBeenCalledWith('/app/app-123/overview') + expect(mockRedirect).toHaveBeenCalledTimes(1) + }) + + test('calls redirection function with workflow path for editor', () => { + const app = { id: 'app-123', mode: 'workflow' as const } + const mockRedirect = jest.fn() + + getRedirection(true, app, mockRedirect) + + expect(mockRedirect).toHaveBeenCalledWith('/app/app-123/workflow') + expect(mockRedirect).toHaveBeenCalledTimes(1) + }) + + test('calls redirection function with configuration path for chat mode editor', () => { + const app = { id: 'app-123', mode: 'chat' as const } + const mockRedirect = jest.fn() + + getRedirection(true, app, mockRedirect) + + expect(mockRedirect).toHaveBeenCalledWith('/app/app-123/configuration') + expect(mockRedirect).toHaveBeenCalledTimes(1) + }) + + test('works with different redirection functions', () => { + const app = { id: 'app-123', mode: 'workflow' as const } + const paths: string[] = [] + const customRedirect = (path: string) => paths.push(path) + + getRedirection(true, app, customRedirect) + + expect(paths).toEqual(['/app/app-123/workflow']) + }) + }) +}) diff --git a/web/utils/app-redirection.ts b/web/utils/app-redirection.ts index dfecbd17d4..5ed8419e05 100644 --- a/web/utils/app-redirection.ts +++ b/web/utils/app-redirection.ts @@ -1,14 +1,14 @@ -import type { AppMode } from '@/types/app' +import { AppModeEnum } from '@/types/app' export const getRedirectionPath = ( isCurrentWorkspaceEditor: boolean, - app: { id: string, mode: AppMode }, + app: { id: string, mode: AppModeEnum }, ) => { if (!isCurrentWorkspaceEditor) { return `/app/${app.id}/overview` } else { - if (app.mode === 'workflow' || app.mode === 'advanced-chat') + if (app.mode === AppModeEnum.WORKFLOW || app.mode === AppModeEnum.ADVANCED_CHAT) return `/app/${app.id}/workflow` else return `/app/${app.id}/configuration` @@ -17,7 +17,7 @@ export const getRedirectionPath = ( export const getRedirection = ( isCurrentWorkspaceEditor: boolean, - app: { id: string, mode: AppMode }, + app: { id: string, mode: AppModeEnum }, redirectionFunc: (href: string) => void, ) => { const redirectionPath = getRedirectionPath(isCurrentWorkspaceEditor, app) diff --git a/web/utils/classnames.spec.ts b/web/utils/classnames.spec.ts index a0b40684c9..55dc1cfd68 100644 --- a/web/utils/classnames.spec.ts +++ b/web/utils/classnames.spec.ts @@ -1,6 +1,18 @@ +/** + * Test suite for the classnames utility function + * This utility combines the classnames library with tailwind-merge + * to handle conditional CSS classes and merge conflicting Tailwind classes + */ import cn from './classnames' describe('classnames', () => { + /** + * Tests basic classnames library features: + * - String concatenation + * - Array handling + * - Falsy value filtering + * - Object-based conditional classes + */ test('classnames libs feature', () => { expect(cn('foo')).toBe('foo') expect(cn('foo', 'bar')).toBe('foo bar') @@ -17,6 +29,14 @@ describe('classnames', () => { })).toBe('foo baz') }) + /** + * Tests tailwind-merge functionality: + * - Conflicting class resolution (last one wins) + * - Modifier handling (hover, focus, etc.) + * - Important prefix (!) + * - Custom color classes + * - Arbitrary values + */ test('tailwind-merge', () => { /* eslint-disable tailwindcss/classnames-order */ expect(cn('p-0')).toBe('p-0') @@ -44,6 +64,10 @@ describe('classnames', () => { expect(cn('text-3.5xl text-black')).toBe('text-3.5xl text-black') }) + /** + * Tests the integration of classnames and tailwind-merge: + * - Object-based conditional classes with Tailwind conflict resolution + */ test('classnames combined with tailwind-merge', () => { expect(cn('text-right', { 'text-center': true, @@ -53,4 +77,81 @@ describe('classnames', () => { 'text-center': false, })).toBe('text-right') }) + + /** + * Tests handling of multiple mixed argument types: + * - Strings, arrays, and objects in a single call + * - Tailwind merge working across different argument types + */ + test('multiple mixed argument types', () => { + expect(cn('foo', ['bar', 'baz'], { qux: true, quux: false })).toBe('foo bar baz qux') + expect(cn('p-4', ['p-2', 'm-4'], { 'text-left': true, 'text-right': true })).toBe('p-2 m-4 text-right') + }) + + /** + * Tests nested array handling: + * - Deep array flattening + * - Tailwind merge with nested structures + */ + test('nested arrays', () => { + expect(cn(['foo', ['bar', 'baz']])).toBe('foo bar baz') + expect(cn(['p-4', ['p-2', 'text-center']])).toBe('p-2 text-center') + }) + + /** + * Tests empty input handling: + * - Empty strings, arrays, and objects + * - Mixed empty and non-empty values + */ + test('empty inputs', () => { + expect(cn('')).toBe('') + expect(cn([])).toBe('') + expect(cn({})).toBe('') + expect(cn('', [], {})).toBe('') + expect(cn('foo', '', 'bar')).toBe('foo bar') + }) + + /** + * Tests number input handling: + * - Truthy numbers converted to strings + * - Zero treated as falsy + */ + test('numbers as inputs', () => { + expect(cn(1)).toBe('1') + expect(cn(0)).toBe('') + expect(cn('foo', 1, 'bar')).toBe('foo 1 bar') + }) + + /** + * Tests multiple object arguments: + * - Object merging + * - Tailwind conflict resolution across objects + */ + test('multiple objects', () => { + expect(cn({ foo: true }, { bar: true })).toBe('foo bar') + expect(cn({ foo: true, bar: false }, { bar: true, baz: true })).toBe('foo bar baz') + expect(cn({ 'p-4': true }, { 'p-2': true })).toBe('p-2') + }) + + /** + * Tests complex edge cases: + * - Mixed falsy values + * - Nested arrays with falsy values + * - Multiple conflicting Tailwind classes + */ + test('complex edge cases', () => { + expect(cn('foo', null, undefined, false, 'bar', 0, 1, '')).toBe('foo bar 1') + expect(cn(['foo', null, ['bar', undefined, 'baz']])).toBe('foo bar baz') + expect(cn('text-sm', { 'text-lg': false, 'text-xl': true }, 'text-2xl')).toBe('text-2xl') + }) + + /** + * Tests important (!) modifier behavior: + * - Important modifiers in objects + * - Conflict resolution with important prefix + */ + test('important modifier with objects', () => { + expect(cn({ '!font-medium': true }, { '!font-bold': true })).toBe('!font-bold') + expect(cn('font-normal', { '!font-bold': true })).toBe('font-normal !font-bold') + }) }) diff --git a/web/utils/clipboard.spec.ts b/web/utils/clipboard.spec.ts new file mode 100644 index 0000000000..be64cbbe13 --- /dev/null +++ b/web/utils/clipboard.spec.ts @@ -0,0 +1,148 @@ +/** + * Test suite for clipboard utilities + * + * This module provides cross-browser clipboard functionality with automatic fallback: + * 1. Modern Clipboard API (navigator.clipboard.writeText) - preferred method + * 2. Legacy execCommand('copy') - fallback for older browsers + * + * The implementation ensures clipboard operations work across all supported browsers + * while gracefully handling permissions and API availability. + */ +import { writeTextToClipboard } from './clipboard' + +describe('Clipboard Utilities', () => { + describe('writeTextToClipboard', () => { + afterEach(() => { + jest.restoreAllMocks() + }) + + /** + * Test modern Clipboard API usage + * When navigator.clipboard is available, should use the modern API + */ + it('should use navigator.clipboard.writeText when available', async () => { + const mockWriteText = jest.fn().mockResolvedValue(undefined) + Object.defineProperty(navigator, 'clipboard', { + value: { writeText: mockWriteText }, + writable: true, + configurable: true, + }) + + await writeTextToClipboard('test text') + expect(mockWriteText).toHaveBeenCalledWith('test text') + }) + + /** + * Test fallback to legacy execCommand method + * When Clipboard API is unavailable, should use document.execCommand('copy') + * This involves creating a temporary textarea element + */ + it('should fallback to execCommand when clipboard API not available', async () => { + Object.defineProperty(navigator, 'clipboard', { + value: undefined, + writable: true, + configurable: true, + }) + + const mockExecCommand = jest.fn().mockReturnValue(true) + document.execCommand = mockExecCommand + + const appendChildSpy = jest.spyOn(document.body, 'appendChild') + const removeChildSpy = jest.spyOn(document.body, 'removeChild') + + await writeTextToClipboard('fallback text') + + expect(appendChildSpy).toHaveBeenCalled() + expect(mockExecCommand).toHaveBeenCalledWith('copy') + expect(removeChildSpy).toHaveBeenCalled() + }) + + /** + * Test error handling when execCommand returns false + * execCommand returns false when the operation fails + */ + it('should handle execCommand failure', async () => { + Object.defineProperty(navigator, 'clipboard', { + value: undefined, + writable: true, + configurable: true, + }) + + const mockExecCommand = jest.fn().mockReturnValue(false) + document.execCommand = mockExecCommand + + await expect(writeTextToClipboard('fail text')).rejects.toThrow() + }) + + /** + * Test error handling when execCommand throws an exception + * Should propagate the error to the caller + */ + it('should handle execCommand exception', async () => { + Object.defineProperty(navigator, 'clipboard', { + value: undefined, + writable: true, + configurable: true, + }) + + const mockExecCommand = jest.fn().mockImplementation(() => { + throw new Error('execCommand error') + }) + document.execCommand = mockExecCommand + + await expect(writeTextToClipboard('error text')).rejects.toThrow('execCommand error') + }) + + /** + * Test proper cleanup of temporary DOM elements + * The temporary textarea should be removed after copying + */ + it('should clean up textarea after fallback', async () => { + Object.defineProperty(navigator, 'clipboard', { + value: undefined, + writable: true, + configurable: true, + }) + + document.execCommand = jest.fn().mockReturnValue(true) + const removeChildSpy = jest.spyOn(document.body, 'removeChild') + + await writeTextToClipboard('cleanup test') + + expect(removeChildSpy).toHaveBeenCalled() + }) + + /** + * Test copying empty strings + * Should handle edge case of empty clipboard content + */ + it('should handle empty string', async () => { + const mockWriteText = jest.fn().mockResolvedValue(undefined) + Object.defineProperty(navigator, 'clipboard', { + value: { writeText: mockWriteText }, + writable: true, + configurable: true, + }) + + await writeTextToClipboard('') + expect(mockWriteText).toHaveBeenCalledWith('') + }) + + /** + * Test copying text with special characters + * Should preserve newlines, tabs, quotes, unicode, and emojis + */ + it('should handle special characters', async () => { + const mockWriteText = jest.fn().mockResolvedValue(undefined) + Object.defineProperty(navigator, 'clipboard', { + value: { writeText: mockWriteText }, + writable: true, + configurable: true, + }) + + const specialText = 'Test\n\t"quotes"\n中文\n😀' + await writeTextToClipboard(specialText) + expect(mockWriteText).toHaveBeenCalledWith(specialText) + }) + }) +}) diff --git a/web/utils/completion-params.spec.ts b/web/utils/completion-params.spec.ts new file mode 100644 index 0000000000..56aa1c0586 --- /dev/null +++ b/web/utils/completion-params.spec.ts @@ -0,0 +1,230 @@ +import { mergeValidCompletionParams } from './completion-params' +import type { FormValue, ModelParameterRule } from '@/app/components/header/account-setting/model-provider-page/declarations' + +describe('completion-params', () => { + describe('mergeValidCompletionParams', () => { + test('returns empty params and removedDetails for undefined oldParams', () => { + const rules: ModelParameterRule[] = [] + const result = mergeValidCompletionParams(undefined, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({}) + }) + + test('returns empty params and removedDetails for empty oldParams', () => { + const rules: ModelParameterRule[] = [] + const result = mergeValidCompletionParams({}, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({}) + }) + + test('validates int type parameter within range', () => { + const rules: ModelParameterRule[] = [ + { name: 'max_tokens', type: 'int', min: 1, max: 4096, label: { en_US: 'Max Tokens', zh_Hans: '最大标记' }, required: false }, + ] + const oldParams: FormValue = { max_tokens: 100 } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ max_tokens: 100 }) + expect(result.removedDetails).toEqual({}) + }) + + test('removes int parameter below minimum', () => { + const rules: ModelParameterRule[] = [ + { name: 'max_tokens', type: 'int', min: 1, max: 4096, label: { en_US: 'Max Tokens', zh_Hans: '最大标记' }, required: false }, + ] + const oldParams: FormValue = { max_tokens: 0 } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({ max_tokens: 'out of range (1-4096)' }) + }) + + test('removes int parameter above maximum', () => { + const rules: ModelParameterRule[] = [ + { name: 'max_tokens', type: 'int', min: 1, max: 4096, label: { en_US: 'Max Tokens', zh_Hans: '最大标记' }, required: false }, + ] + const oldParams: FormValue = { max_tokens: 5000 } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({ max_tokens: 'out of range (1-4096)' }) + }) + + test('removes int parameter with invalid type', () => { + const rules: ModelParameterRule[] = [ + { name: 'max_tokens', type: 'int', min: 1, max: 4096, label: { en_US: 'Max Tokens', zh_Hans: '最大标记' }, required: false }, + ] + const oldParams: FormValue = { max_tokens: 'not a number' as any } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({ max_tokens: 'invalid type' }) + }) + + test('validates float type parameter', () => { + const rules: ModelParameterRule[] = [ + { name: 'temperature', type: 'float', min: 0, max: 2, label: { en_US: 'Temperature', zh_Hans: '温度' }, required: false }, + ] + const oldParams: FormValue = { temperature: 0.7 } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ temperature: 0.7 }) + expect(result.removedDetails).toEqual({}) + }) + + test('validates float at boundary values', () => { + const rules: ModelParameterRule[] = [ + { name: 'temperature', type: 'float', min: 0, max: 2, label: { en_US: 'Temperature', zh_Hans: '温度' }, required: false }, + ] + + const result1 = mergeValidCompletionParams({ temperature: 0 }, rules) + expect(result1.params).toEqual({ temperature: 0 }) + + const result2 = mergeValidCompletionParams({ temperature: 2 }, rules) + expect(result2.params).toEqual({ temperature: 2 }) + }) + + test('validates boolean type parameter', () => { + const rules: ModelParameterRule[] = [ + { name: 'stream', type: 'boolean', label: { en_US: 'Stream', zh_Hans: '流' }, required: false }, + ] + const oldParams: FormValue = { stream: true } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ stream: true }) + expect(result.removedDetails).toEqual({}) + }) + + test('removes boolean parameter with invalid type', () => { + const rules: ModelParameterRule[] = [ + { name: 'stream', type: 'boolean', label: { en_US: 'Stream', zh_Hans: '流' }, required: false }, + ] + const oldParams: FormValue = { stream: 'yes' as any } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({ stream: 'invalid type' }) + }) + + test('validates string type parameter', () => { + const rules: ModelParameterRule[] = [ + { name: 'model', type: 'string', label: { en_US: 'Model', zh_Hans: '模型' }, required: false }, + ] + const oldParams: FormValue = { model: 'gpt-4' } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ model: 'gpt-4' }) + expect(result.removedDetails).toEqual({}) + }) + + test('validates string parameter with options', () => { + const rules: ModelParameterRule[] = [ + { name: 'model', type: 'string', options: ['gpt-3.5-turbo', 'gpt-4'], label: { en_US: 'Model', zh_Hans: '模型' }, required: false }, + ] + const oldParams: FormValue = { model: 'gpt-4' } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ model: 'gpt-4' }) + expect(result.removedDetails).toEqual({}) + }) + + test('removes string parameter with invalid option', () => { + const rules: ModelParameterRule[] = [ + { name: 'model', type: 'string', options: ['gpt-3.5-turbo', 'gpt-4'], label: { en_US: 'Model', zh_Hans: '模型' }, required: false }, + ] + const oldParams: FormValue = { model: 'invalid-model' } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({ model: 'unsupported option' }) + }) + + test('validates text type parameter', () => { + const rules: ModelParameterRule[] = [ + { name: 'prompt', type: 'text', label: { en_US: 'Prompt', zh_Hans: '提示' }, required: false }, + ] + const oldParams: FormValue = { prompt: 'Hello world' } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ prompt: 'Hello world' }) + expect(result.removedDetails).toEqual({}) + }) + + test('removes unsupported parameters', () => { + const rules: ModelParameterRule[] = [ + { name: 'temperature', type: 'float', min: 0, max: 2, label: { en_US: 'Temperature', zh_Hans: '温度' }, required: false }, + ] + const oldParams: FormValue = { temperature: 0.7, unsupported_param: 'value' } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ temperature: 0.7 }) + expect(result.removedDetails).toEqual({ unsupported_param: 'unsupported' }) + }) + + test('keeps stop parameter in advanced mode even without rule', () => { + const rules: ModelParameterRule[] = [] + const oldParams: FormValue = { stop: ['END'] } + const result = mergeValidCompletionParams(oldParams, rules, true) + + expect(result.params).toEqual({ stop: ['END'] }) + expect(result.removedDetails).toEqual({}) + }) + + test('removes stop parameter in normal mode without rule', () => { + const rules: ModelParameterRule[] = [] + const oldParams: FormValue = { stop: ['END'] } + const result = mergeValidCompletionParams(oldParams, rules, false) + + expect(result.params).toEqual({}) + expect(result.removedDetails).toEqual({ stop: 'unsupported' }) + }) + + test('handles multiple parameters with mixed validity', () => { + const rules: ModelParameterRule[] = [ + { name: 'temperature', type: 'float', min: 0, max: 2, label: { en_US: 'Temperature', zh_Hans: '温度' }, required: false }, + { name: 'max_tokens', type: 'int', min: 1, max: 4096, label: { en_US: 'Max Tokens', zh_Hans: '最大标记' }, required: false }, + { name: 'model', type: 'string', options: ['gpt-4'], label: { en_US: 'Model', zh_Hans: '模型' }, required: false }, + ] + const oldParams: FormValue = { + temperature: 0.7, + max_tokens: 5000, + model: 'gpt-4', + unsupported: 'value', + } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ + temperature: 0.7, + model: 'gpt-4', + }) + expect(result.removedDetails).toEqual({ + max_tokens: 'out of range (1-4096)', + unsupported: 'unsupported', + }) + }) + + test('handles parameters without min/max constraints', () => { + const rules: ModelParameterRule[] = [ + { name: 'value', type: 'int', label: { en_US: 'Value', zh_Hans: '值' }, required: false }, + ] + const oldParams: FormValue = { value: 999999 } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({ value: 999999 }) + expect(result.removedDetails).toEqual({}) + }) + + test('removes parameter with unsupported rule type', () => { + const rules: ModelParameterRule[] = [ + { name: 'custom', type: 'unknown_type', label: { en_US: 'Custom', zh_Hans: '自定义' }, required: false } as any, + ] + const oldParams: FormValue = { custom: 'value' } + const result = mergeValidCompletionParams(oldParams, rules) + + expect(result.params).toEqual({}) + expect(result.removedDetails.custom).toContain('unsupported rule type') + }) + }) +}) diff --git a/web/utils/context.spec.ts b/web/utils/context.spec.ts new file mode 100644 index 0000000000..fb72e4f4de --- /dev/null +++ b/web/utils/context.spec.ts @@ -0,0 +1,253 @@ +/** + * Test suite for React context creation utilities + * + * This module provides helper functions to create React contexts with better type safety + * and automatic error handling when context is used outside of its provider. + * + * Two variants are provided: + * - createCtx: Standard React context using useContext/createContext + * - createSelectorCtx: Context with selector support using use-context-selector library + */ +import React from 'react' +import { renderHook } from '@testing-library/react' +import { createCtx, createSelectorCtx } from './context' + +describe('Context Utilities', () => { + describe('createCtx', () => { + /** + * Test that createCtx creates a valid context with provider and hook + * The function should return a tuple with [Provider, useContextValue, Context] + * plus named properties for easier access + */ + it('should create context with provider and hook', () => { + type TestContextValue = { value: string } + const [Provider, useTestContext, Context] = createCtx<TestContextValue>({ + name: 'Test', + }) + + expect(Provider).toBeDefined() + expect(useTestContext).toBeDefined() + expect(Context).toBeDefined() + }) + + /** + * Test that the context hook returns the provided value correctly + * when used within the context provider + */ + it('should provide and consume context value', () => { + type TestContextValue = { value: string } + const [Provider, useTestContext] = createCtx<TestContextValue>({ + name: 'Test', + }) + + const testValue = { value: 'test-value' } + + const wrapper = ({ children }: { children: React.ReactNode }) => + React.createElement(Provider, { value: testValue }, children) + + const { result } = renderHook(() => useTestContext(), { wrapper }) + + expect(result.current).toEqual(testValue) + }) + + /** + * Test that accessing context outside of provider throws an error + * This ensures developers are notified when they forget to wrap components + */ + it('should throw error when used outside provider', () => { + type TestContextValue = { value: string } + const [, useTestContext] = createCtx<TestContextValue>({ + name: 'Test', + }) + + // Suppress console.error for this test + const consoleError = jest.spyOn(console, 'error').mockImplementation(() => { /* suppress error */ }) + + expect(() => { + renderHook(() => useTestContext()) + }).toThrow('No Test context found.') + + consoleError.mockRestore() + }) + + /** + * Test that context works with default values + * When a default value is provided, it should be accessible without a provider + */ + it('should use default value when provided', () => { + type TestContextValue = { value: string } + const defaultValue = { value: 'default' } + const [, useTestContext] = createCtx<TestContextValue>({ + name: 'Test', + defaultValue, + }) + + const { result } = renderHook(() => useTestContext()) + + expect(result.current).toEqual(defaultValue) + }) + + /** + * Test that the returned tuple has named properties for convenience + * This allows destructuring or property access based on preference + */ + it('should expose named properties', () => { + type TestContextValue = { value: string } + const result = createCtx<TestContextValue>({ name: 'Test' }) + + expect(result.provider).toBe(result[0]) + expect(result.useContextValue).toBe(result[1]) + expect(result.context).toBe(result[2]) + }) + + /** + * Test context with complex data types + * Ensures type safety is maintained with nested objects and arrays + */ + it('should handle complex context values', () => { + type ComplexContext = { + user: { id: string; name: string } + settings: { theme: string; locale: string } + actions: Array<() => void> + } + + const [Provider, useComplexContext] = createCtx<ComplexContext>({ + name: 'Complex', + }) + + const complexValue: ComplexContext = { + user: { id: '123', name: 'Test User' }, + settings: { theme: 'dark', locale: 'en-US' }, + actions: [ + () => { /* empty action 1 */ }, + () => { /* empty action 2 */ }, + ], + } + + const wrapper = ({ children }: { children: React.ReactNode }) => + React.createElement(Provider, { value: complexValue }, children) + + const { result } = renderHook(() => useComplexContext(), { wrapper }) + + expect(result.current).toEqual(complexValue) + expect(result.current.user.id).toBe('123') + expect(result.current.settings.theme).toBe('dark') + expect(result.current.actions).toHaveLength(2) + }) + + /** + * Test that context updates propagate to consumers + * When provider value changes, hooks should receive the new value + */ + it('should update when context value changes', () => { + type TestContextValue = { count: number } + const [Provider, useTestContext] = createCtx<TestContextValue>({ + name: 'Test', + }) + + let value = { count: 0 } + const wrapper = ({ children }: { children: React.ReactNode }) => + React.createElement(Provider, { value }, children) + + const { result, rerender } = renderHook(() => useTestContext(), { wrapper }) + + expect(result.current.count).toBe(0) + + value = { count: 5 } + rerender() + + expect(result.current.count).toBe(5) + }) + }) + + describe('createSelectorCtx', () => { + /** + * Test that createSelectorCtx creates a valid context with selector support + * This variant uses use-context-selector for optimized re-renders + */ + it('should create selector context with provider and hook', () => { + type TestContextValue = { value: string } + const [Provider, useTestContext, Context] = createSelectorCtx<TestContextValue>({ + name: 'SelectorTest', + }) + + expect(Provider).toBeDefined() + expect(useTestContext).toBeDefined() + expect(Context).toBeDefined() + }) + + /** + * Test that selector context provides and consumes values correctly + * The API should be identical to createCtx for basic usage + */ + it('should provide and consume context value with selector', () => { + type TestContextValue = { value: string } + const [Provider, useTestContext] = createSelectorCtx<TestContextValue>({ + name: 'SelectorTest', + }) + + const testValue = { value: 'selector-test' } + + const wrapper = ({ children }: { children: React.ReactNode }) => + React.createElement(Provider, { value: testValue }, children) + + const { result } = renderHook(() => useTestContext(), { wrapper }) + + expect(result.current).toEqual(testValue) + }) + + /** + * Test error handling for selector context + * Should throw error when used outside provider, same as createCtx + */ + it('should throw error when used outside provider', () => { + type TestContextValue = { value: string } + const [, useTestContext] = createSelectorCtx<TestContextValue>({ + name: 'SelectorTest', + }) + + const consoleError = jest.spyOn(console, 'error').mockImplementation(() => { /* suppress error */ }) + + expect(() => { + renderHook(() => useTestContext()) + }).toThrow('No SelectorTest context found.') + + consoleError.mockRestore() + }) + + /** + * Test that selector context works with default values + */ + it('should use default value when provided', () => { + type TestContextValue = { value: string } + const defaultValue = { value: 'selector-default' } + const [, useTestContext] = createSelectorCtx<TestContextValue>({ + name: 'SelectorTest', + defaultValue, + }) + + const { result } = renderHook(() => useTestContext()) + + expect(result.current).toEqual(defaultValue) + }) + }) + + describe('Context without name', () => { + /** + * Test that contexts can be created without a name + * The error message should use a generic fallback + */ + it('should create context without name and show generic error', () => { + type TestContextValue = { value: string } + const [, useTestContext] = createCtx<TestContextValue>() + + const consoleError = jest.spyOn(console, 'error').mockImplementation(() => { /* suppress error */ }) + + expect(() => { + renderHook(() => useTestContext()) + }).toThrow('No related context found.') + + consoleError.mockRestore() + }) + }) +}) diff --git a/web/utils/emoji.spec.ts b/web/utils/emoji.spec.ts new file mode 100644 index 0000000000..df9520234a --- /dev/null +++ b/web/utils/emoji.spec.ts @@ -0,0 +1,77 @@ +import { searchEmoji } from './emoji' +import { SearchIndex } from 'emoji-mart' + +jest.mock('emoji-mart', () => ({ + SearchIndex: { + search: jest.fn(), + }, +})) + +describe('Emoji Utilities', () => { + describe('searchEmoji', () => { + beforeEach(() => { + jest.clearAllMocks() + }) + + it('should return emoji natives for search results', async () => { + const mockEmojis = [ + { skins: [{ native: '😀' }] }, + { skins: [{ native: '😃' }] }, + { skins: [{ native: '😄' }] }, + ] + ;(SearchIndex.search as jest.Mock).mockResolvedValue(mockEmojis) + + const result = await searchEmoji('smile') + expect(result).toEqual(['😀', '😃', '😄']) + }) + + it('should return empty array when no results', async () => { + ;(SearchIndex.search as jest.Mock).mockResolvedValue([]) + + const result = await searchEmoji('nonexistent') + expect(result).toEqual([]) + }) + + it('should return empty array when search returns null', async () => { + ;(SearchIndex.search as jest.Mock).mockResolvedValue(null) + + const result = await searchEmoji('test') + expect(result).toEqual([]) + }) + + it('should handle search with empty string', async () => { + ;(SearchIndex.search as jest.Mock).mockResolvedValue([]) + + const result = await searchEmoji('') + expect(result).toEqual([]) + expect(SearchIndex.search).toHaveBeenCalledWith('') + }) + + it('should extract native from first skin', async () => { + const mockEmojis = [ + { + skins: [ + { native: '👍' }, + { native: '👍🏻' }, + { native: '👍🏼' }, + ], + }, + ] + ;(SearchIndex.search as jest.Mock).mockResolvedValue(mockEmojis) + + const result = await searchEmoji('thumbs') + expect(result).toEqual(['👍']) + }) + + it('should handle multiple search terms', async () => { + const mockEmojis = [ + { skins: [{ native: '❤️' }] }, + { skins: [{ native: '💙' }] }, + ] + ;(SearchIndex.search as jest.Mock).mockResolvedValue(mockEmojis) + + const result = await searchEmoji('heart love') + expect(result).toEqual(['❤️', '💙']) + }) + }) +}) diff --git a/web/utils/error-parser.ts b/web/utils/error-parser.ts new file mode 100644 index 0000000000..311505521f --- /dev/null +++ b/web/utils/error-parser.ts @@ -0,0 +1,52 @@ +/** + * Parse plugin error message from nested error structure + * Extracts the real error message from PluginInvokeError JSON string + * + * @example + * Input: { message: "req_id: xxx PluginInvokeError: {\"message\":\"Bad credentials\"}" } + * Output: "Bad credentials" + * + * @param error - Error object (can be Response object or error with message property) + * @returns Promise<string> or string - Parsed error message + */ +export const parsePluginErrorMessage = async (error: any): Promise<string> => { + let rawMessage = '' + + // Handle Response object from fetch/ky + if (error instanceof Response) { + try { + const body = await error.clone().json() + rawMessage = body?.message || error.statusText || 'Unknown error' + } + catch { + rawMessage = error.statusText || 'Unknown error' + } + } + else { + rawMessage = error?.message || error?.toString() || 'Unknown error' + } + + console.log('rawMessage', rawMessage) + + // Try to extract nested JSON from PluginInvokeError + // Use greedy match .+ to capture the complete JSON object with nested braces + const pluginErrorPattern = /PluginInvokeError:\s*(\{.+\})/ + const match = rawMessage.match(pluginErrorPattern) + + if (match) { + try { + const errorData = JSON.parse(match[1]) + // Return the inner message if exists + if (errorData.message) + return errorData.message + // Fallback to error_type if message not available + if (errorData.error_type) + return errorData.error_type + } + catch (parseError) { + console.warn('Failed to parse plugin error JSON:', parseError) + } + } + + return rawMessage +} diff --git a/web/utils/format.spec.ts b/web/utils/format.spec.ts index c94495d597..13c58bd7e5 100644 --- a/web/utils/format.spec.ts +++ b/web/utils/format.spec.ts @@ -1,4 +1,4 @@ -import { downloadFile, formatFileSize, formatNumber, formatTime } from './format' +import { downloadFile, formatFileSize, formatNumber, formatNumberAbbreviated, formatTime } from './format' describe('formatNumber', () => { test('should correctly format integers', () => { @@ -102,3 +102,95 @@ describe('downloadFile', () => { jest.restoreAllMocks() }) }) + +describe('formatNumberAbbreviated', () => { + it('should return number as string when less than 1000', () => { + expect(formatNumberAbbreviated(0)).toBe('0') + expect(formatNumberAbbreviated(1)).toBe('1') + expect(formatNumberAbbreviated(999)).toBe('999') + }) + + it('should format thousands with k suffix', () => { + expect(formatNumberAbbreviated(1000)).toBe('1k') + expect(formatNumberAbbreviated(1200)).toBe('1.2k') + expect(formatNumberAbbreviated(1500)).toBe('1.5k') + expect(formatNumberAbbreviated(9999)).toBe('10k') + }) + + it('should format millions with M suffix', () => { + expect(formatNumberAbbreviated(1000000)).toBe('1M') + expect(formatNumberAbbreviated(1500000)).toBe('1.5M') + expect(formatNumberAbbreviated(2300000)).toBe('2.3M') + expect(formatNumberAbbreviated(999999999)).toBe('1B') + }) + + it('should format billions with B suffix', () => { + expect(formatNumberAbbreviated(1000000000)).toBe('1B') + expect(formatNumberAbbreviated(1500000000)).toBe('1.5B') + expect(formatNumberAbbreviated(2300000000)).toBe('2.3B') + }) + + it('should remove .0 from whole numbers', () => { + expect(formatNumberAbbreviated(1000)).toBe('1k') + expect(formatNumberAbbreviated(2000000)).toBe('2M') + expect(formatNumberAbbreviated(3000000000)).toBe('3B') + }) + + it('should keep decimal for non-whole numbers', () => { + expect(formatNumberAbbreviated(1100)).toBe('1.1k') + expect(formatNumberAbbreviated(1500000)).toBe('1.5M') + expect(formatNumberAbbreviated(2700000000)).toBe('2.7B') + }) + + it('should handle edge cases', () => { + expect(formatNumberAbbreviated(950)).toBe('950') + expect(formatNumberAbbreviated(1001)).toBe('1k') + expect(formatNumberAbbreviated(999999)).toBe('1M') + }) +}) + +describe('formatNumber edge cases', () => { + it('should handle very large numbers', () => { + expect(formatNumber(1234567890123)).toBe('1,234,567,890,123') + }) + + it('should handle numbers with many decimal places', () => { + expect(formatNumber(1234.56789)).toBe('1,234.56789') + }) + + it('should handle negative decimals', () => { + expect(formatNumber(-1234.56)).toBe('-1,234.56') + }) + + it('should handle string with decimals', () => { + expect(formatNumber('9876543.21')).toBe('9,876,543.21') + }) +}) + +describe('formatFileSize edge cases', () => { + it('should handle exactly 1024 bytes', () => { + expect(formatFileSize(1024)).toBe('1.00 KB') + }) + + it('should handle fractional bytes', () => { + expect(formatFileSize(512.5)).toBe('512.50 bytes') + }) +}) + +describe('formatTime edge cases', () => { + it('should handle exactly 60 seconds', () => { + expect(formatTime(60)).toBe('1.00 min') + }) + + it('should handle exactly 3600 seconds', () => { + expect(formatTime(3600)).toBe('1.00 h') + }) + + it('should handle fractional seconds', () => { + expect(formatTime(45.5)).toBe('45.50 sec') + }) + + it('should handle very large durations', () => { + expect(formatTime(86400)).toBe('24.00 h') // 24 hours + }) +}) diff --git a/web/utils/format.ts b/web/utils/format.ts index 70238456c5..fe5b1deb7d 100644 --- a/web/utils/format.ts +++ b/web/utils/format.ts @@ -1,3 +1,50 @@ +import type { Locale } from '@/i18n-config' +import type { Dayjs } from 'dayjs' +import 'dayjs/locale/de' +import 'dayjs/locale/es' +import 'dayjs/locale/fa' +import 'dayjs/locale/fr' +import 'dayjs/locale/hi' +import 'dayjs/locale/id' +import 'dayjs/locale/it' +import 'dayjs/locale/ja' +import 'dayjs/locale/ko' +import 'dayjs/locale/pl' +import 'dayjs/locale/pt-br' +import 'dayjs/locale/ro' +import 'dayjs/locale/ru' +import 'dayjs/locale/sl' +import 'dayjs/locale/th' +import 'dayjs/locale/tr' +import 'dayjs/locale/uk' +import 'dayjs/locale/vi' +import 'dayjs/locale/zh-cn' +import 'dayjs/locale/zh-tw' + +const localeMap: Record<Locale, string> = { + 'en-US': 'en', + 'zh-Hans': 'zh-cn', + 'zh-Hant': 'zh-tw', + 'pt-BR': 'pt-br', + 'es-ES': 'es', + 'fr-FR': 'fr', + 'de-DE': 'de', + 'ja-JP': 'ja', + 'ko-KR': 'ko', + 'ru-RU': 'ru', + 'it-IT': 'it', + 'th-TH': 'th', + 'id-ID': 'id', + 'uk-UA': 'uk', + 'vi-VN': 'vi', + 'ro-RO': 'ro', + 'pl-PL': 'pl', + 'hi-IN': 'hi', + 'tr-TR': 'tr', + 'fa-IR': 'fa', + 'sl-SI': 'sl', +} + /** * Formats a number with comma separators. * @example formatNumber(1234567) will return '1,234,567' @@ -83,10 +130,24 @@ export const formatNumberAbbreviated = (num: number) => { for (let i = 0; i < units.length; i++) { if (num >= units[i].value) { - const formatted = (num / units[i].value).toFixed(1) + const value = num / units[i].value + let rounded = Math.round(value * 10) / 10 + let unitIndex = i + + // If rounded value >= 1000, promote to next unit + if (rounded >= 1000 && i > 0) { + rounded = rounded / 1000 + unitIndex = i - 1 + } + + const formatted = rounded.toFixed(1) return formatted.endsWith('.0') - ? `${Number.parseInt(formatted)}${units[i].symbol}` - : `${formatted}${units[i].symbol}` + ? `${Number.parseInt(formatted)}${units[unitIndex].symbol}` + : `${formatted}${units[unitIndex].symbol}` } } } + +export const formatToLocalTime = (time: Dayjs, local: string, format: string) => { + return time.locale(localeMap[local] ?? 'en').format(format) +} diff --git a/web/utils/get-icon.spec.ts b/web/utils/get-icon.spec.ts new file mode 100644 index 0000000000..98eb2288fd --- /dev/null +++ b/web/utils/get-icon.spec.ts @@ -0,0 +1,162 @@ +/** + * Test suite for icon utility functions + * Tests the generation of marketplace plugin icon URLs + */ +import { getIconFromMarketPlace } from './get-icon' +import { MARKETPLACE_API_PREFIX } from '@/config' + +describe('get-icon', () => { + describe('getIconFromMarketPlace', () => { + /** + * Tests basic URL generation for marketplace plugin icons + */ + test('returns correct marketplace icon URL', () => { + const pluginId = 'test-plugin-123' + const result = getIconFromMarketPlace(pluginId) + expect(result).toBe(`${MARKETPLACE_API_PREFIX}/plugins/${pluginId}/icon`) + }) + + /** + * Tests URL generation with plugin IDs containing special characters + * like dashes and underscores + */ + test('handles plugin ID with special characters', () => { + const pluginId = 'plugin-with-dashes_and_underscores' + const result = getIconFromMarketPlace(pluginId) + expect(result).toBe(`${MARKETPLACE_API_PREFIX}/plugins/${pluginId}/icon`) + }) + + /** + * Tests behavior with empty plugin ID + * Note: This creates a malformed URL but doesn't throw an error + */ + test('handles empty plugin ID', () => { + const pluginId = '' + const result = getIconFromMarketPlace(pluginId) + expect(result).toBe(`${MARKETPLACE_API_PREFIX}/plugins//icon`) + }) + + /** + * Tests URL generation with plugin IDs containing spaces + * Spaces will be URL-encoded when actually used + */ + test('handles plugin ID with spaces', () => { + const pluginId = 'plugin with spaces' + const result = getIconFromMarketPlace(pluginId) + expect(result).toBe(`${MARKETPLACE_API_PREFIX}/plugins/${pluginId}/icon`) + }) + + /** + * Security tests: Path traversal attempts + * These tests document current behavior and potential security concerns + * Note: Current implementation does not sanitize path traversal sequences + */ + test('handles path traversal attempts', () => { + const pluginId = '../../../etc/passwd' + const result = getIconFromMarketPlace(pluginId) + // Current implementation includes path traversal sequences in URL + // This is a potential security concern that should be addressed + expect(result).toContain('../') + expect(result).toContain(pluginId) + }) + + test('handles multiple path traversal attempts', () => { + const pluginId = '../../../../etc/passwd' + const result = getIconFromMarketPlace(pluginId) + // Current implementation includes path traversal sequences in URL + expect(result).toContain('../') + expect(result).toContain(pluginId) + }) + + test('passes through URL-encoded path traversal sequences', () => { + const pluginId = '..%2F..%2Fetc%2Fpasswd' + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + }) + + /** + * Security tests: Null and undefined handling + * These tests document current behavior with invalid input types + * Note: Current implementation converts null/undefined to strings instead of throwing + */ + test('handles null plugin ID', () => { + // Current implementation converts null to string "null" + const result = getIconFromMarketPlace(null as any) + expect(result).toContain('null') + // This is a potential issue - should validate input type + }) + + test('handles undefined plugin ID', () => { + // Current implementation converts undefined to string "undefined" + const result = getIconFromMarketPlace(undefined as any) + expect(result).toContain('undefined') + // This is a potential issue - should validate input type + }) + + /** + * Security tests: URL-sensitive characters + * These tests verify that URL-sensitive characters are handled appropriately + */ + test('does not encode URL-sensitive characters', () => { + const pluginId = 'plugin/with?special=chars#hash' + const result = getIconFromMarketPlace(pluginId) + // Note: Current implementation doesn't encode, but test documents the behavior + expect(result).toContain(pluginId) + expect(result).toContain('?') + expect(result).toContain('#') + expect(result).toContain('=') + }) + + test('handles URL characters like & and %', () => { + const pluginId = 'plugin&with%encoding' + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + }) + + /** + * Edge case tests: Extreme inputs + * These tests verify behavior with unusual but valid inputs + */ + test('handles very long plugin ID', () => { + const pluginId = 'a'.repeat(10000) + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + expect(result.length).toBeGreaterThan(10000) + }) + + test('handles Unicode characters', () => { + const pluginId = '插件-🚀-测试-日本語' + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + }) + + test('handles control characters', () => { + const pluginId = 'plugin\nwith\ttabs\r\nand\0null' + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + }) + + /** + * Security tests: XSS attempts + * These tests verify that XSS attempts are handled appropriately + */ + test('handles XSS attempts with script tags', () => { + const pluginId = '<script>alert("xss")</script>' + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + // Note: Current implementation doesn't sanitize, but test documents the behavior + }) + + test('handles XSS attempts with event handlers', () => { + const pluginId = 'plugin"onerror="alert(1)"' + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + }) + + test('handles XSS attempts with encoded script tags', () => { + const pluginId = '%3Cscript%3Ealert%28%22xss%22%29%3C%2Fscript%3E' + const result = getIconFromMarketPlace(pluginId) + expect(result).toContain(pluginId) + }) + }) +}) diff --git a/web/utils/index.spec.ts b/web/utils/index.spec.ts index 21a0d80dd0..beda974e5c 100644 --- a/web/utils/index.spec.ts +++ b/web/utils/index.spec.ts @@ -293,3 +293,308 @@ describe('removeSpecificQueryParam', () => { expect(replaceStateCall[2]).toMatch(/param3=value3/) }) }) + +describe('sleep', () => { + it('should resolve after specified milliseconds', async () => { + const start = Date.now() + await sleep(100) + const end = Date.now() + expect(end - start).toBeGreaterThanOrEqual(90) // Allow some tolerance + }) + + it('should handle zero milliseconds', async () => { + await expect(sleep(0)).resolves.toBeUndefined() + }) +}) + +describe('asyncRunSafe extended', () => { + it('should handle promise that resolves with null', async () => { + const [error, result] = await asyncRunSafe(Promise.resolve(null)) + expect(error).toBeNull() + expect(result).toBeNull() + }) + + it('should handle promise that resolves with undefined', async () => { + const [error, result] = await asyncRunSafe(Promise.resolve(undefined)) + expect(error).toBeNull() + expect(result).toBeUndefined() + }) + + it('should handle promise that resolves with false', async () => { + const [error, result] = await asyncRunSafe(Promise.resolve(false)) + expect(error).toBeNull() + expect(result).toBe(false) + }) + + it('should handle promise that resolves with 0', async () => { + const [error, result] = await asyncRunSafe(Promise.resolve(0)) + expect(error).toBeNull() + expect(result).toBe(0) + }) + + // TODO: pre-commit blocks this test case + // Error msg: "Expected the Promise rejection reason to be an Error" + + // it('should handle promise that rejects with null', async () => { + // const [error] = await asyncRunSafe(Promise.reject(null)) + // expect(error).toBeInstanceOf(Error) + // expect(error?.message).toBe('unknown error') + // }) +}) + +describe('getTextWidthWithCanvas', () => { + it('should return 0 when canvas context is not available', () => { + const mockGetContext = jest.fn().mockReturnValue(null) + jest.spyOn(document, 'createElement').mockReturnValue({ + getContext: mockGetContext, + } as any) + + const width = getTextWidthWithCanvas('test') + expect(width).toBe(0) + + jest.restoreAllMocks() + }) + + it('should measure text width with custom font', () => { + const mockMeasureText = jest.fn().mockReturnValue({ width: 123.456 }) + const mockContext = { + font: '', + measureText: mockMeasureText, + } + jest.spyOn(document, 'createElement').mockReturnValue({ + getContext: jest.fn().mockReturnValue(mockContext), + } as any) + + const width = getTextWidthWithCanvas('test', '16px Arial') + expect(mockContext.font).toBe('16px Arial') + expect(width).toBe(123.46) + + jest.restoreAllMocks() + }) + + it('should handle empty string', () => { + const mockMeasureText = jest.fn().mockReturnValue({ width: 0 }) + jest.spyOn(document, 'createElement').mockReturnValue({ + getContext: jest.fn().mockReturnValue({ + font: '', + measureText: mockMeasureText, + }), + } as any) + + const width = getTextWidthWithCanvas('') + expect(width).toBe(0) + + jest.restoreAllMocks() + }) +}) + +describe('randomString extended', () => { + it('should generate string of exact length', () => { + expect(randomString(10).length).toBe(10) + expect(randomString(50).length).toBe(50) + expect(randomString(100).length).toBe(100) + }) + + it('should generate different strings on multiple calls', () => { + const str1 = randomString(20) + const str2 = randomString(20) + const str3 = randomString(20) + expect(str1).not.toBe(str2) + expect(str2).not.toBe(str3) + expect(str1).not.toBe(str3) + }) + + it('should only contain valid characters', () => { + const validChars = /^[0-9a-zA-Z_-]+$/ + const str = randomString(100) + expect(validChars.test(str)).toBe(true) + }) + + it('should handle length of 1', () => { + const str = randomString(1) + expect(str.length).toBe(1) + }) + + it('should handle length of 0', () => { + const str = randomString(0) + expect(str).toBe('') + }) +}) + +describe('getPurifyHref extended', () => { + it('should escape HTML entities', () => { + expect(getPurifyHref('<script>alert(1)</script>')).not.toContain('<script>') + expect(getPurifyHref('test&test')).toContain('&') + expect(getPurifyHref('test"test')).toContain('"') + }) + + it('should handle URLs with query parameters', () => { + const url = 'https://example.com?param=<script>' + const purified = getPurifyHref(url) + expect(purified).not.toContain('<script>') + }) + + it('should handle empty string', () => { + expect(getPurifyHref('')).toBe('') + }) + + it('should handle null/undefined', () => { + expect(getPurifyHref(null as any)).toBe('') + expect(getPurifyHref(undefined as any)).toBe('') + }) +}) + +describe('fetchWithRetry extended', () => { + it('should succeed on first try', async () => { + const [error, result] = await fetchWithRetry(Promise.resolve('success')) + expect(error).toBeNull() + expect(result).toBe('success') + }) + + it('should retry specified number of times', async () => { + let attempts = 0 + const failingPromise = () => { + attempts++ + return Promise.reject(new Error('fail')) + } + + await fetchWithRetry(failingPromise(), 3) + // Initial attempt + 3 retries = 4 total attempts + // But the function structure means it will try once, then retry 3 times + }) + + it('should succeed after retries', async () => { + let attempts = 0 + const eventuallySucceed = new Promise((resolve, reject) => { + attempts++ + if (attempts < 2) + reject(new Error('not yet')) + else + resolve('success') + }) + + await fetchWithRetry(eventuallySucceed, 3) + // Note: This test may need adjustment based on actual retry logic + }) + + /* + TODO: Commented this case because of eslint + Error msg: Expected the Promise rejection reason to be an Error + */ + // it('should handle non-Error rejections', async () => { + // const [error] = await fetchWithRetry(Promise.reject('string error'), 0) + // expect(error).toBeInstanceOf(Error) + // }) +}) + +describe('correctModelProvider extended', () => { + it('should handle empty string', () => { + expect(correctModelProvider('')).toBe('') + }) + + it('should not modify provider with slash', () => { + expect(correctModelProvider('custom/provider/model')).toBe('custom/provider/model') + }) + + it('should handle google provider', () => { + expect(correctModelProvider('google')).toBe('langgenius/gemini/google') + }) + + it('should handle standard providers', () => { + expect(correctModelProvider('openai')).toBe('langgenius/openai/openai') + expect(correctModelProvider('anthropic')).toBe('langgenius/anthropic/anthropic') + }) + + it('should handle null/undefined', () => { + expect(correctModelProvider(null as any)).toBe('') + expect(correctModelProvider(undefined as any)).toBe('') + }) +}) + +describe('correctToolProvider extended', () => { + it('should return as-is when toolInCollectionList is true', () => { + expect(correctToolProvider('any-provider', true)).toBe('any-provider') + expect(correctToolProvider('', true)).toBe('') + }) + + it('should not modify provider with slash when not in collection', () => { + expect(correctToolProvider('custom/tool/provider', false)).toBe('custom/tool/provider') + }) + + it('should handle special tool providers', () => { + expect(correctToolProvider('stepfun', false)).toBe('langgenius/stepfun_tool/stepfun') + expect(correctToolProvider('jina', false)).toBe('langgenius/jina_tool/jina') + expect(correctToolProvider('siliconflow', false)).toBe('langgenius/siliconflow_tool/siliconflow') + expect(correctToolProvider('gitee_ai', false)).toBe('langgenius/gitee_ai_tool/gitee_ai') + }) + + it('should handle standard tool providers', () => { + expect(correctToolProvider('standard', false)).toBe('langgenius/standard/standard') + }) +}) + +describe('canFindTool extended', () => { + it('should match exact provider ID', () => { + expect(canFindTool('openai', 'openai')).toBe(true) + }) + + it('should match langgenius format', () => { + expect(canFindTool('langgenius/openai/openai', 'openai')).toBe(true) + }) + + it('should match tool format', () => { + expect(canFindTool('langgenius/jina_tool/jina', 'jina')).toBe(true) + }) + + it('should not match different providers', () => { + expect(canFindTool('openai', 'anthropic')).toBe(false) + }) + + it('should handle undefined oldToolId', () => { + expect(canFindTool('openai', undefined)).toBe(false) + }) +}) + +describe('removeSpecificQueryParam extended', () => { + beforeEach(() => { + // Reset window.location + delete (window as any).location + window.location = { + href: 'https://example.com?param1=value1¶m2=value2¶m3=value3', + } as any + }) + + it('should remove single query parameter', () => { + const mockReplaceState = jest.fn() + window.history.replaceState = mockReplaceState + + removeSpecificQueryParam('param1') + + expect(mockReplaceState).toHaveBeenCalled() + const newUrl = mockReplaceState.mock.calls[0][2] + expect(newUrl).not.toContain('param1') + }) + + it('should remove multiple query parameters', () => { + const mockReplaceState = jest.fn() + window.history.replaceState = mockReplaceState + + removeSpecificQueryParam(['param1', 'param2']) + + expect(mockReplaceState).toHaveBeenCalled() + const newUrl = mockReplaceState.mock.calls[0][2] + expect(newUrl).not.toContain('param1') + expect(newUrl).not.toContain('param2') + }) + + it('should preserve other parameters', () => { + const mockReplaceState = jest.fn() + window.history.replaceState = mockReplaceState + + removeSpecificQueryParam('param1') + + const newUrl = mockReplaceState.mock.calls[0][2] + expect(newUrl).toContain('param2') + expect(newUrl).toContain('param3') + }) +}) diff --git a/web/utils/mcp.spec.ts b/web/utils/mcp.spec.ts new file mode 100644 index 0000000000..d3c5ef1eab --- /dev/null +++ b/web/utils/mcp.spec.ts @@ -0,0 +1,88 @@ +/** + * Test suite for MCP (Model Context Protocol) utility functions + * Tests icon detection logic for MCP-related features + */ +import { shouldUseMcpIcon, shouldUseMcpIconForAppIcon } from './mcp' + +describe('mcp', () => { + /** + * Tests shouldUseMcpIcon function which determines if the MCP icon + * should be used based on the icon source format + */ + describe('shouldUseMcpIcon', () => { + /** + * The link emoji (🔗) is used as a special marker for MCP icons + */ + test('returns true for emoji object with 🔗 content', () => { + const src = { content: '🔗', background: '#fff' } + expect(shouldUseMcpIcon(src)).toBe(true) + }) + + test('returns false for emoji object with different content', () => { + const src = { content: '🎉', background: '#fff' } + expect(shouldUseMcpIcon(src)).toBe(false) + }) + + test('returns false for string URL', () => { + const src = 'https://example.com/icon.png' + expect(shouldUseMcpIcon(src)).toBe(false) + }) + + test('returns false for null', () => { + expect(shouldUseMcpIcon(null)).toBe(false) + }) + + test('returns false for undefined', () => { + expect(shouldUseMcpIcon(undefined)).toBe(false) + }) + + test('returns false for empty object', () => { + expect(shouldUseMcpIcon({})).toBe(false) + }) + + test('returns false for object without content property', () => { + const src = { background: '#fff' } + expect(shouldUseMcpIcon(src)).toBe(false) + }) + + test('returns false for object with null content', () => { + const src = { content: null, background: '#fff' } + expect(shouldUseMcpIcon(src)).toBe(false) + }) + }) + + /** + * Tests shouldUseMcpIconForAppIcon function which checks if an app icon + * should use the MCP icon based on icon type and content + */ + describe('shouldUseMcpIconForAppIcon', () => { + /** + * MCP icon should only be used when both conditions are met: + * - Icon type is 'emoji' + * - Icon content is the link emoji (🔗) + */ + test('returns true when iconType is emoji and icon is 🔗', () => { + expect(shouldUseMcpIconForAppIcon('emoji', '🔗')).toBe(true) + }) + + test('returns false when iconType is emoji but icon is different', () => { + expect(shouldUseMcpIconForAppIcon('emoji', '🎉')).toBe(false) + }) + + test('returns false when iconType is image', () => { + expect(shouldUseMcpIconForAppIcon('image', '🔗')).toBe(false) + }) + + test('returns false when iconType is image and icon is different', () => { + expect(shouldUseMcpIconForAppIcon('image', 'file-id-123')).toBe(false) + }) + + test('returns false for empty strings', () => { + expect(shouldUseMcpIconForAppIcon('', '')).toBe(false) + }) + + test('returns false when iconType is empty but icon is 🔗', () => { + expect(shouldUseMcpIconForAppIcon('', '🔗')).toBe(false) + }) + }) +}) diff --git a/web/utils/model-config.spec.ts b/web/utils/model-config.spec.ts new file mode 100644 index 0000000000..2cccaabc61 --- /dev/null +++ b/web/utils/model-config.spec.ts @@ -0,0 +1,819 @@ +/** + * Test suite for model configuration transformation utilities + * + * This module handles the conversion between two different representations of user input forms: + * 1. UserInputFormItem: The form structure used in the UI + * 2. PromptVariable: The variable structure used in prompts and model configuration + * + * Key functions: + * - userInputsFormToPromptVariables: Converts UI form items to prompt variables + * - promptVariablesToUserInputsForm: Converts prompt variables back to form items + * - formatBooleanInputs: Ensures boolean inputs are properly typed + */ +import { + formatBooleanInputs, + promptVariablesToUserInputsForm, + userInputsFormToPromptVariables, +} from './model-config' +import type { UserInputFormItem } from '@/types/app' +import type { PromptVariable } from '@/models/debug' + +describe('Model Config Utilities', () => { + describe('userInputsFormToPromptVariables', () => { + /** + * Test handling of null or undefined input + * Should return empty array when no inputs provided + */ + it('should return empty array for null input', () => { + const result = userInputsFormToPromptVariables(null) + expect(result).toEqual([]) + }) + + /** + * Test conversion of text-input (string) type + * Text inputs are the most common form field type + */ + it('should convert text-input to string prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + 'text-input': { + label: 'User Name', + variable: 'user_name', + required: true, + max_length: 100, + default: '', + hide: false, + }, + }, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result).toHaveLength(1) + expect(result[0]).toEqual({ + key: 'user_name', + name: 'User Name', + required: true, + type: 'string', + max_length: 100, + options: [], + is_context_var: false, + hide: false, + default: '', + }) + }) + + /** + * Test conversion of paragraph type + * Paragraphs are multi-line text inputs + */ + it('should convert paragraph to paragraph prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + paragraph: { + label: 'Description', + variable: 'description', + required: false, + max_length: 500, + default: '', + hide: false, + }, + }, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result[0]).toEqual({ + key: 'description', + name: 'Description', + required: false, + type: 'paragraph', + max_length: 500, + options: [], + is_context_var: false, + hide: false, + default: '', + }) + }) + + /** + * Test conversion of number type + * Number inputs should preserve numeric constraints + */ + it('should convert number input to number prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + number: { + label: 'Age', + variable: 'age', + required: true, + default: '', + hide: false, + }, + } as any, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result[0]).toEqual({ + key: 'age', + name: 'Age', + required: true, + type: 'number', + options: [], + hide: false, + default: '', + }) + }) + + /** + * Test conversion of checkbox (boolean) type + * Checkboxes are converted to 'checkbox' type in prompt variables + */ + it('should convert checkbox to checkbox prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + checkbox: { + label: 'Accept Terms', + variable: 'accept_terms', + required: true, + default: '', + hide: false, + }, + } as any, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result[0]).toEqual({ + key: 'accept_terms', + name: 'Accept Terms', + required: true, + type: 'checkbox', + options: [], + hide: false, + default: '', + }) + }) + + /** + * Test conversion of select (dropdown) type + * Select inputs include options array + */ + it('should convert select input to select prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + select: { + label: 'Country', + variable: 'country', + required: true, + options: ['USA', 'Canada', 'Mexico'], + default: 'USA', + hide: false, + }, + }, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result[0]).toEqual({ + key: 'country', + name: 'Country', + required: true, + type: 'select', + options: ['USA', 'Canada', 'Mexico'], + is_context_var: false, + hide: false, + default: 'USA', + }) + }) + + /** + * Test conversion of file upload type + * File inputs include configuration for allowed types and upload methods + */ + it('should convert file input to file prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + file: { + label: 'Profile Picture', + variable: 'profile_pic', + required: false, + allowed_file_types: ['image'], + allowed_file_extensions: ['.jpg', '.png'], + allowed_file_upload_methods: ['local_file', 'remote_url'], + default: '', + hide: false, + }, + } as any, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result[0]).toEqual({ + key: 'profile_pic', + name: 'Profile Picture', + required: false, + type: 'file', + config: { + allowed_file_types: ['image'], + allowed_file_extensions: ['.jpg', '.png'], + allowed_file_upload_methods: ['local_file', 'remote_url'], + number_limits: 1, + }, + hide: false, + default: '', + }) + }) + + /** + * Test conversion of file-list type + * File lists allow multiple file uploads with a max_length constraint + */ + it('should convert file-list input to file-list prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + 'file-list': { + label: 'Documents', + variable: 'documents', + required: true, + allowed_file_types: ['document'], + allowed_file_extensions: ['.pdf', '.docx'], + allowed_file_upload_methods: ['local_file'], + max_length: 5, + default: '', + hide: false, + }, + } as any, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result[0]).toEqual({ + key: 'documents', + name: 'Documents', + required: true, + type: 'file-list', + config: { + allowed_file_types: ['document'], + allowed_file_extensions: ['.pdf', '.docx'], + allowed_file_upload_methods: ['local_file'], + number_limits: 5, + }, + hide: false, + default: '', + }) + }) + + /** + * Test conversion of external_data_tool type + * External data tools have custom configuration and icons + */ + it('should convert external_data_tool to prompt variable', () => { + const userInputs: UserInputFormItem[] = [ + { + external_data_tool: { + label: 'API Data', + variable: 'api_data', + type: 'api', + enabled: true, + required: false, + config: { endpoint: 'https://api.example.com' }, + icon: 'api-icon', + icon_background: '#FF5733', + hide: false, + }, + } as any, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result[0]).toEqual({ + key: 'api_data', + name: 'API Data', + required: false, + type: 'api', + enabled: true, + config: { endpoint: 'https://api.example.com' }, + icon: 'api-icon', + icon_background: '#FF5733', + is_context_var: false, + hide: false, + }) + }) + + /** + * Test handling of dataset_query_variable + * When a variable matches the dataset_query_variable, is_context_var should be true + */ + it('should mark variable as context var when matching dataset_query_variable', () => { + const userInputs: UserInputFormItem[] = [ + { + 'text-input': { + label: 'Query', + variable: 'query', + required: true, + max_length: 200, + default: '', + hide: false, + }, + }, + ] + + const result = userInputsFormToPromptVariables(userInputs, 'query') + + expect(result[0].is_context_var).toBe(true) + }) + + /** + * Test conversion of multiple mixed input types + * Should handle an array with different input types correctly + */ + it('should convert multiple mixed input types', () => { + const userInputs: UserInputFormItem[] = [ + { + 'text-input': { + label: 'Name', + variable: 'name', + required: true, + max_length: 50, + default: '', + hide: false, + }, + }, + { + number: { + label: 'Age', + variable: 'age', + required: false, + default: '', + hide: false, + }, + } as any, + { + select: { + label: 'Gender', + variable: 'gender', + required: true, + options: ['Male', 'Female', 'Other'], + default: '', + hide: false, + }, + }, + ] + + const result = userInputsFormToPromptVariables(userInputs) + + expect(result).toHaveLength(3) + expect(result[0].type).toBe('string') + expect(result[1].type).toBe('number') + expect(result[2].type).toBe('select') + }) + }) + + describe('promptVariablesToUserInputsForm', () => { + /** + * Test conversion of string prompt variable back to text-input + */ + it('should convert string prompt variable to text-input', () => { + const promptVariables: PromptVariable[] = [ + { + key: 'user_name', + name: 'User Name', + required: true, + type: 'string', + max_length: 100, + options: [], + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect(result).toHaveLength(1) + expect(result[0]).toEqual({ + 'text-input': { + label: 'User Name', + variable: 'user_name', + required: true, + max_length: 100, + default: '', + hide: undefined, + }, + }) + }) + + /** + * Test conversion of paragraph prompt variable + */ + it('should convert paragraph prompt variable to paragraph input', () => { + const promptVariables: PromptVariable[] = [ + { + key: 'description', + name: 'Description', + required: false, + type: 'paragraph', + max_length: 500, + options: [], + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect(result[0]).toEqual({ + paragraph: { + label: 'Description', + variable: 'description', + required: false, + max_length: 500, + default: '', + hide: undefined, + }, + }) + }) + + /** + * Test conversion of number prompt variable + */ + it('should convert number prompt variable to number input', () => { + const promptVariables: PromptVariable[] = [ + { + key: 'age', + name: 'Age', + required: true, + type: 'number', + options: [], + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect(result[0]).toEqual({ + number: { + label: 'Age', + variable: 'age', + required: true, + default: '', + hide: undefined, + }, + }) + }) + + /** + * Test conversion of checkbox prompt variable + */ + it('should convert checkbox prompt variable to checkbox input', () => { + const promptVariables: PromptVariable[] = [ + { + key: 'accept_terms', + name: 'Accept Terms', + required: true, + type: 'checkbox', + options: [], + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect(result[0]).toEqual({ + checkbox: { + label: 'Accept Terms', + variable: 'accept_terms', + required: true, + default: '', + hide: undefined, + }, + }) + }) + + /** + * Test conversion of select prompt variable + */ + it('should convert select prompt variable to select input', () => { + const promptVariables: PromptVariable[] = [ + { + key: 'country', + name: 'Country', + required: true, + type: 'select', + options: ['USA', 'Canada', 'Mexico'], + default: 'USA', + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect(result[0]).toEqual({ + select: { + label: 'Country', + variable: 'country', + required: true, + options: ['USA', 'Canada', 'Mexico'], + default: 'USA', + hide: undefined, + }, + }) + }) + + /** + * Test filtering of invalid prompt variables + * Variables without key or name should be filtered out + */ + it('should filter out variables with empty key or name', () => { + const promptVariables: PromptVariable[] = [ + { + key: '', + name: 'Empty Key', + required: true, + type: 'string', + options: [], + }, + { + key: 'valid', + name: '', + required: true, + type: 'string', + options: [], + }, + { + key: ' ', + name: 'Whitespace Key', + required: true, + type: 'string', + options: [], + }, + { + key: 'valid_key', + name: 'Valid Name', + required: true, + type: 'string', + options: [], + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect(result).toHaveLength(1) + expect((result[0] as any)['text-input']?.variable).toBe('valid_key') + }) + + /** + * Test conversion of external data tool prompt variable + */ + it('should convert external data tool prompt variable', () => { + const promptVariables: PromptVariable[] = [ + { + key: 'api_data', + name: 'API Data', + required: false, + type: 'api', + enabled: true, + config: { endpoint: 'https://api.example.com' }, + icon: 'api-icon', + icon_background: '#FF5733', + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect(result[0]).toEqual({ + external_data_tool: { + label: 'API Data', + variable: 'api_data', + enabled: true, + type: 'api', + config: { endpoint: 'https://api.example.com' }, + required: false, + icon: 'api-icon', + icon_background: '#FF5733', + hide: undefined, + }, + }) + }) + + /** + * Test that required defaults to true when not explicitly set to false + */ + it('should default required to true when not false', () => { + const promptVariables: PromptVariable[] = [ + { + key: 'test1', + name: 'Test 1', + required: undefined, + type: 'string', + options: [], + }, + { + key: 'test2', + name: 'Test 2', + required: false, + type: 'string', + options: [], + }, + ] + + const result = promptVariablesToUserInputsForm(promptVariables) + + expect((result[0] as any)['text-input']?.required).toBe(true) + expect((result[1] as any)['text-input']?.required).toBe(false) + }) + }) + + describe('formatBooleanInputs', () => { + /** + * Test that null or undefined inputs are handled gracefully + */ + it('should return inputs unchanged when useInputs is null', () => { + const inputs = { key1: 'value1', key2: 'value2' } + const result = formatBooleanInputs(null, inputs) + expect(result).toEqual(inputs) + }) + + it('should return inputs unchanged when useInputs is undefined', () => { + const inputs = { key1: 'value1', key2: 'value2' } + const result = formatBooleanInputs(undefined, inputs) + expect(result).toEqual(inputs) + }) + + /** + * Test conversion of boolean input values to actual boolean type + * This is important for proper type handling in the backend + * Note: checkbox inputs are converted to type 'checkbox' by userInputsFormToPromptVariables + */ + it('should convert boolean inputs to boolean type', () => { + const useInputs: PromptVariable[] = [ + { + key: 'accept_terms', + name: 'Accept Terms', + required: true, + type: 'checkbox', + options: [], + }, + { + key: 'subscribe', + name: 'Subscribe', + required: false, + type: 'checkbox', + options: [], + }, + ] + + const inputs = { + accept_terms: 'true', + subscribe: '', + other_field: 'value', + } + + const result = formatBooleanInputs(useInputs, inputs) + + expect(result).toEqual({ + accept_terms: true, + subscribe: false, + other_field: 'value', + }) + }) + + /** + * Test that non-boolean inputs are not affected + */ + it('should not modify non-boolean inputs', () => { + const useInputs: PromptVariable[] = [ + { + key: 'name', + name: 'Name', + required: true, + type: 'string', + options: [], + }, + { + key: 'age', + name: 'Age', + required: true, + type: 'number', + options: [], + }, + ] + + const inputs = { + name: 'John Doe', + age: 30, + } + + const result = formatBooleanInputs(useInputs, inputs) + + expect(result).toEqual(inputs) + }) + + /** + * Test handling of truthy and falsy values for boolean conversion + * Note: checkbox inputs are converted to type 'checkbox' by userInputsFormToPromptVariables + */ + it('should handle various truthy and falsy values', () => { + const useInputs: PromptVariable[] = [ + { + key: 'bool1', + name: 'Bool 1', + required: true, + type: 'checkbox', + options: [], + }, + { + key: 'bool2', + name: 'Bool 2', + required: true, + type: 'checkbox', + options: [], + }, + { + key: 'bool3', + name: 'Bool 3', + required: true, + type: 'checkbox', + options: [], + }, + { + key: 'bool4', + name: 'Bool 4', + required: true, + type: 'checkbox', + options: [], + }, + ] + + const inputs = { + bool1: 1, + bool2: 0, + bool3: 'yes', + bool4: null as any, + } + + const result = formatBooleanInputs(useInputs, inputs) + + expect(result?.bool1).toBe(true) + expect(result?.bool2).toBe(false) + expect(result?.bool3).toBe(true) + expect(result?.bool4).toBe(false) + }) + + /** + * Test that the function creates a new object and doesn't mutate the original + * Note: checkbox inputs are converted to type 'checkbox' by userInputsFormToPromptVariables + */ + it('should not mutate original inputs object', () => { + const useInputs: PromptVariable[] = [ + { + key: 'flag', + name: 'Flag', + required: true, + type: 'checkbox', + options: [], + }, + ] + + const inputs = { flag: 'true', other: 'value' } + const originalInputs = { ...inputs } + + formatBooleanInputs(useInputs, inputs) + + expect(inputs).toEqual(originalInputs) + }) + }) + + describe('Round-trip conversion', () => { + /** + * Test that converting from UserInputForm to PromptVariable and back + * preserves the essential data (though some fields may have defaults applied) + */ + it('should preserve data through round-trip conversion', () => { + const originalUserInputs: UserInputFormItem[] = [ + { + 'text-input': { + label: 'Name', + variable: 'name', + required: true, + max_length: 50, + default: '', + hide: false, + }, + }, + { + select: { + label: 'Type', + variable: 'type', + required: false, + options: ['A', 'B', 'C'], + default: 'A', + hide: false, + }, + }, + ] + + const promptVars = userInputsFormToPromptVariables(originalUserInputs) + const backToUserInputs = promptVariablesToUserInputsForm(promptVars) + + expect(backToUserInputs).toHaveLength(2) + expect((backToUserInputs[0] as any)['text-input']?.variable).toBe('name') + expect((backToUserInputs[1] as any).select?.variable).toBe('type') + expect((backToUserInputs[1] as any).select?.options).toEqual(['A', 'B', 'C']) + }) + }) +}) diff --git a/web/utils/model-config.ts b/web/utils/model-config.ts index 3f655ce036..707a3685b9 100644 --- a/web/utils/model-config.ts +++ b/web/utils/model-config.ts @@ -200,7 +200,7 @@ export const formatBooleanInputs = (useInputs?: PromptVariable[] | null, inputs? return inputs const res = { ...inputs } useInputs.forEach((item) => { - const isBooleanInput = item.type === 'boolean' + const isBooleanInput = item.type === 'checkbox' if (isBooleanInput) { // Convert boolean inputs to boolean type res[item.key] = !!res[item.key] diff --git a/web/utils/navigation.spec.ts b/web/utils/navigation.spec.ts new file mode 100644 index 0000000000..bbd8f36767 --- /dev/null +++ b/web/utils/navigation.spec.ts @@ -0,0 +1,297 @@ +/** + * Test suite for navigation utility functions + * Tests URL and query parameter manipulation for consistent navigation behavior + * Includes helpers for preserving state during navigation (pagination, filters, etc.) + */ +import { + createBackNavigation, + createNavigationPath, + createNavigationPathWithParams, + datasetNavigation, + extractQueryParams, + mergeQueryParams, +} from './navigation' + +describe('navigation', () => { + const originalWindow = globalThis.window + + beforeEach(() => { + // Mock window.location with sample query parameters + delete (globalThis as any).window + globalThis.window = { + location: { + search: '?page=3&limit=10&keyword=test', + }, + } as any + }) + + afterEach(() => { + globalThis.window = originalWindow + }) + + /** + * Tests createNavigationPath which builds URLs with optional query parameter preservation + */ + describe('createNavigationPath', () => { + test('preserves query parameters by default', () => { + const result = createNavigationPath('/datasets/123/documents') + expect(result).toBe('/datasets/123/documents?page=3&limit=10&keyword=test') + }) + + test('returns clean path when preserveParams is false', () => { + const result = createNavigationPath('/datasets/123/documents', false) + expect(result).toBe('/datasets/123/documents') + }) + + test('handles empty query string', () => { + globalThis.window.location.search = '' + const result = createNavigationPath('/datasets/123/documents') + expect(result).toBe('/datasets/123/documents') + }) + + test('handles path with trailing slash', () => { + const result = createNavigationPath('/datasets/123/documents/') + expect(result).toBe('/datasets/123/documents/?page=3&limit=10&keyword=test') + }) + + test('handles root path', () => { + const result = createNavigationPath('/') + expect(result).toBe('/?page=3&limit=10&keyword=test') + }) + }) + + /** + * Tests createBackNavigation which creates a navigation callback function + */ + describe('createBackNavigation', () => { + /** + * Tests that the returned function properly navigates with preserved params + */ + test('returns function that calls router.push with correct path', () => { + const mockRouter = { push: jest.fn() } + const backNav = createBackNavigation(mockRouter, '/datasets/123/documents') + + backNav() + + expect(mockRouter.push).toHaveBeenCalledWith('/datasets/123/documents?page=3&limit=10&keyword=test') + }) + + test('returns function that navigates without params when preserveParams is false', () => { + const mockRouter = { push: jest.fn() } + const backNav = createBackNavigation(mockRouter, '/datasets/123/documents', false) + + backNav() + + expect(mockRouter.push).toHaveBeenCalledWith('/datasets/123/documents') + }) + + test('can be called multiple times', () => { + const mockRouter = { push: jest.fn() } + const backNav = createBackNavigation(mockRouter, '/datasets/123/documents') + + backNav() + backNav() + + expect(mockRouter.push).toHaveBeenCalledTimes(2) + }) + }) + + /** + * Tests extractQueryParams which extracts specific parameters from current URL + */ + describe('extractQueryParams', () => { + /** + * Tests selective parameter extraction + */ + test('extracts specified parameters', () => { + const result = extractQueryParams(['page', 'limit']) + expect(result).toEqual({ page: '3', limit: '10' }) + }) + + test('extracts all specified parameters including keyword', () => { + const result = extractQueryParams(['page', 'limit', 'keyword']) + expect(result).toEqual({ page: '3', limit: '10', keyword: 'test' }) + }) + + test('ignores non-existent parameters', () => { + const result = extractQueryParams(['page', 'nonexistent']) + expect(result).toEqual({ page: '3' }) + }) + + test('returns empty object when no parameters match', () => { + const result = extractQueryParams(['foo', 'bar']) + expect(result).toEqual({}) + }) + + test('returns empty object for empty array', () => { + const result = extractQueryParams([]) + expect(result).toEqual({}) + }) + + test('handles empty query string', () => { + globalThis.window.location.search = '' + const result = extractQueryParams(['page', 'limit']) + expect(result).toEqual({}) + }) + }) + + /** + * Tests createNavigationPathWithParams which builds URLs with specific parameters + */ + describe('createNavigationPathWithParams', () => { + /** + * Tests URL construction with custom parameters + */ + test('creates path with specified parameters', () => { + const result = createNavigationPathWithParams('/datasets/123/documents', { + page: '1', + limit: '25', + }) + expect(result).toBe('/datasets/123/documents?page=1&limit=25') + }) + + test('handles string and number values', () => { + const result = createNavigationPathWithParams('/datasets/123/documents', { + page: 1, + limit: 25, + keyword: 'search', + }) + expect(result).toBe('/datasets/123/documents?page=1&limit=25&keyword=search') + }) + + test('filters out empty string values', () => { + const result = createNavigationPathWithParams('/datasets/123/documents', { + page: '1', + keyword: '', + }) + expect(result).toBe('/datasets/123/documents?page=1') + }) + + test('filters out null and undefined values', () => { + const result = createNavigationPathWithParams('/datasets/123/documents', { + page: '1', + keyword: null as any, + filter: undefined as any, + }) + expect(result).toBe('/datasets/123/documents?page=1') + }) + + test('returns base path when params are empty', () => { + const result = createNavigationPathWithParams('/datasets/123/documents', {}) + expect(result).toBe('/datasets/123/documents') + }) + + test('encodes special characters in values', () => { + const result = createNavigationPathWithParams('/datasets/123/documents', { + keyword: 'search term', + }) + expect(result).toBe('/datasets/123/documents?keyword=search+term') + }) + }) + + /** + * Tests mergeQueryParams which combines new parameters with existing URL params + */ + describe('mergeQueryParams', () => { + /** + * Tests parameter merging and overriding + */ + test('merges new params with existing ones', () => { + const result = mergeQueryParams({ keyword: 'new', page: '1' }) + expect(result.get('page')).toBe('1') + expect(result.get('limit')).toBe('10') + expect(result.get('keyword')).toBe('new') + }) + + test('overrides existing parameters', () => { + const result = mergeQueryParams({ page: '5' }) + expect(result.get('page')).toBe('5') + expect(result.get('limit')).toBe('10') + }) + + test('adds new parameters', () => { + const result = mergeQueryParams({ filter: 'active' }) + expect(result.get('filter')).toBe('active') + expect(result.get('page')).toBe('3') + }) + + test('removes parameters with null value', () => { + const result = mergeQueryParams({ page: null }) + expect(result.get('page')).toBeNull() + expect(result.get('limit')).toBe('10') + }) + + test('removes parameters with undefined value', () => { + const result = mergeQueryParams({ page: undefined }) + expect(result.get('page')).toBeNull() + expect(result.get('limit')).toBe('10') + }) + + test('does not preserve existing when preserveExisting is false', () => { + const result = mergeQueryParams({ filter: 'active' }, false) + expect(result.get('filter')).toBe('active') + expect(result.get('page')).toBeNull() + expect(result.get('limit')).toBeNull() + }) + + test('handles number values', () => { + const result = mergeQueryParams({ page: 5, limit: 20 }) + expect(result.get('page')).toBe('5') + expect(result.get('limit')).toBe('20') + }) + + test('does not add empty string values', () => { + const result = mergeQueryParams({ newParam: '' }) + expect(result.get('newParam')).toBeNull() + // Existing params are preserved + expect(result.get('keyword')).toBe('test') + }) + }) + + /** + * Tests datasetNavigation helper object with common dataset navigation patterns + */ + describe('datasetNavigation', () => { + /** + * Tests navigation back to dataset documents list + */ + describe('backToDocuments', () => { + test('creates navigation function with preserved params', () => { + const mockRouter = { push: jest.fn() } + const backNav = datasetNavigation.backToDocuments(mockRouter, 'dataset-123') + + backNav() + + expect(mockRouter.push).toHaveBeenCalledWith('/datasets/dataset-123/documents?page=3&limit=10&keyword=test') + }) + }) + + /** + * Tests navigation to document detail page + */ + describe('toDocumentDetail', () => { + test('creates navigation function to document detail', () => { + const mockRouter = { push: jest.fn() } + const navFunc = datasetNavigation.toDocumentDetail(mockRouter, 'dataset-123', 'doc-456') + + navFunc() + + expect(mockRouter.push).toHaveBeenCalledWith('/datasets/dataset-123/documents/doc-456') + }) + }) + + /** + * Tests navigation to document settings page + */ + describe('toDocumentSettings', () => { + test('creates navigation function to document settings', () => { + const mockRouter = { push: jest.fn() } + const navFunc = datasetNavigation.toDocumentSettings(mockRouter, 'dataset-123', 'doc-456') + + navFunc() + + expect(mockRouter.push).toHaveBeenCalledWith('/datasets/dataset-123/documents/doc-456/settings') + }) + }) + }) +}) diff --git a/web/utils/permission.spec.ts b/web/utils/permission.spec.ts new file mode 100644 index 0000000000..758c38037e --- /dev/null +++ b/web/utils/permission.spec.ts @@ -0,0 +1,95 @@ +/** + * Test suite for permission utility functions + * Tests dataset edit permission logic based on user roles and dataset settings + */ +import { hasEditPermissionForDataset } from './permission' +import { DatasetPermission } from '@/models/datasets' + +describe('permission', () => { + /** + * Tests hasEditPermissionForDataset which checks if a user can edit a dataset + * Based on three permission levels: + * - onlyMe: Only the creator can edit + * - allTeamMembers: All team members can edit + * - partialMembers: Only specified members can edit + */ + describe('hasEditPermissionForDataset', () => { + const userId = 'user-123' + const creatorId = 'creator-456' + const otherUserId = 'user-789' + + test('returns true when permission is onlyMe and user is creator', () => { + const config = { + createdBy: userId, + partialMemberList: [], + permission: DatasetPermission.onlyMe, + } + expect(hasEditPermissionForDataset(userId, config)).toBe(true) + }) + + test('returns false when permission is onlyMe and user is not creator', () => { + const config = { + createdBy: creatorId, + partialMemberList: [], + permission: DatasetPermission.onlyMe, + } + expect(hasEditPermissionForDataset(userId, config)).toBe(false) + }) + + test('returns true when permission is allTeamMembers for any user', () => { + const config = { + createdBy: creatorId, + partialMemberList: [], + permission: DatasetPermission.allTeamMembers, + } + expect(hasEditPermissionForDataset(userId, config)).toBe(true) + expect(hasEditPermissionForDataset(otherUserId, config)).toBe(true) + expect(hasEditPermissionForDataset(creatorId, config)).toBe(true) + }) + + test('returns true when permission is partialMembers and user is in list', () => { + const config = { + createdBy: creatorId, + partialMemberList: [userId, otherUserId], + permission: DatasetPermission.partialMembers, + } + expect(hasEditPermissionForDataset(userId, config)).toBe(true) + }) + + test('returns false when permission is partialMembers and user is not in list', () => { + const config = { + createdBy: creatorId, + partialMemberList: [otherUserId], + permission: DatasetPermission.partialMembers, + } + expect(hasEditPermissionForDataset(userId, config)).toBe(false) + }) + + test('returns false when permission is partialMembers with empty list', () => { + const config = { + createdBy: creatorId, + partialMemberList: [], + permission: DatasetPermission.partialMembers, + } + expect(hasEditPermissionForDataset(userId, config)).toBe(false) + }) + + test('creator is not automatically granted access with partialMembers permission', () => { + const config = { + createdBy: creatorId, + partialMemberList: [userId], + permission: DatasetPermission.partialMembers, + } + expect(hasEditPermissionForDataset(creatorId, config)).toBe(false) + }) + + test('creator has access when included in partialMemberList', () => { + const config = { + createdBy: creatorId, + partialMemberList: [creatorId, userId], + permission: DatasetPermission.partialMembers, + } + expect(hasEditPermissionForDataset(creatorId, config)).toBe(true) + }) + }) +}) diff --git a/web/utils/time.spec.ts b/web/utils/time.spec.ts new file mode 100644 index 0000000000..fc57390fad --- /dev/null +++ b/web/utils/time.spec.ts @@ -0,0 +1,100 @@ +/** + * Test suite for time utility functions + * Tests date comparison and formatting using dayjs + */ +import { formatTime, isAfter } from './time' + +describe('time', () => { + /** + * Tests isAfter function which compares two dates + * Returns true if the first date is after the second + */ + describe('isAfter', () => { + test('returns true when first date is after second date', () => { + const date1 = '2024-01-02' + const date2 = '2024-01-01' + expect(isAfter(date1, date2)).toBe(true) + }) + + test('returns false when first date is before second date', () => { + const date1 = '2024-01-01' + const date2 = '2024-01-02' + expect(isAfter(date1, date2)).toBe(false) + }) + + test('returns false when dates are equal', () => { + const date = '2024-01-01' + expect(isAfter(date, date)).toBe(false) + }) + + test('works with Date objects', () => { + const date1 = new Date('2024-01-02') + const date2 = new Date('2024-01-01') + expect(isAfter(date1, date2)).toBe(true) + }) + + test('works with timestamps', () => { + const date1 = 1704240000000 // 2024-01-03 + const date2 = 1704153600000 // 2024-01-02 + expect(isAfter(date1, date2)).toBe(true) + }) + + test('handles time differences within same day', () => { + const date1 = '2024-01-01 12:00:00' + const date2 = '2024-01-01 11:00:00' + expect(isAfter(date1, date2)).toBe(true) + }) + }) + + /** + * Tests formatTime function which formats dates using dayjs + * Supports various date formats and input types + */ + describe('formatTime', () => { + /** + * Tests basic date formatting with standard format + */ + test('formats date with YYYY-MM-DD format', () => { + const date = '2024-01-15' + const result = formatTime({ date, dateFormat: 'YYYY-MM-DD' }) + expect(result).toBe('2024-01-15') + }) + + test('formats date with custom format', () => { + const date = '2024-01-15 14:30:00' + const result = formatTime({ date, dateFormat: 'MMM DD, YYYY HH:mm' }) + expect(result).toBe('Jan 15, 2024 14:30') + }) + + test('formats date with full month name', () => { + const date = '2024-01-15' + const result = formatTime({ date, dateFormat: 'MMMM DD, YYYY' }) + expect(result).toBe('January 15, 2024') + }) + + test('formats date with time only', () => { + const date = '2024-01-15 14:30:45' + const result = formatTime({ date, dateFormat: 'HH:mm:ss' }) + expect(result).toBe('14:30:45') + }) + + test('works with Date objects', () => { + const date = new Date(2024, 0, 15) // Month is 0-indexed + const result = formatTime({ date, dateFormat: 'YYYY-MM-DD' }) + expect(result).toBe('2024-01-15') + }) + + test('works with timestamps', () => { + const date = 1705276800000 // 2024-01-15 00:00:00 UTC + const result = formatTime({ date, dateFormat: 'YYYY-MM-DD' }) + // Account for timezone differences: UTC-5 to UTC+8 can result in 2024-01-14 or 2024-01-15 + expect(result).toMatch(/^2024-01-(14|15)$/) + }) + + test('handles ISO 8601 format', () => { + const date = '2024-01-15T14:30:00Z' + const result = formatTime({ date, dateFormat: 'YYYY-MM-DD HH:mm' }) + expect(result).toContain('2024-01-15') + }) + }) +}) diff --git a/web/utils/tool-call.spec.ts b/web/utils/tool-call.spec.ts new file mode 100644 index 0000000000..ccfb06f0cc --- /dev/null +++ b/web/utils/tool-call.spec.ts @@ -0,0 +1,79 @@ +/** + * Test suite for tool call utility functions + * Tests detection of function/tool call support in AI models + */ +import { supportFunctionCall } from './tool-call' +import { ModelFeatureEnum } from '@/app/components/header/account-setting/model-provider-page/declarations' + +describe('tool-call', () => { + /** + * Tests supportFunctionCall which checks if a model supports any form of + * function calling (toolCall, multiToolCall, or streamToolCall) + */ + describe('supportFunctionCall', () => { + /** + * Tests detection of basic tool call support + */ + test('returns true when features include toolCall', () => { + const features = [ModelFeatureEnum.toolCall] + expect(supportFunctionCall(features)).toBe(true) + }) + + /** + * Tests detection of multi-tool call support (calling multiple tools in one request) + */ + test('returns true when features include multiToolCall', () => { + const features = [ModelFeatureEnum.multiToolCall] + expect(supportFunctionCall(features)).toBe(true) + }) + + /** + * Tests detection of streaming tool call support + */ + test('returns true when features include streamToolCall', () => { + const features = [ModelFeatureEnum.streamToolCall] + expect(supportFunctionCall(features)).toBe(true) + }) + + test('returns true when features include multiple tool call types', () => { + const features = [ + ModelFeatureEnum.toolCall, + ModelFeatureEnum.multiToolCall, + ModelFeatureEnum.streamToolCall, + ] + expect(supportFunctionCall(features)).toBe(true) + }) + + /** + * Tests that tool call support is detected even when mixed with other features + */ + test('returns true when features include tool call among other features', () => { + const features = [ + ModelFeatureEnum.agentThought, + ModelFeatureEnum.toolCall, + ModelFeatureEnum.vision, + ] + expect(supportFunctionCall(features)).toBe(true) + }) + + /** + * Tests that false is returned when no tool call features are present + */ + test('returns false when features do not include any tool call type', () => { + const features = [ModelFeatureEnum.agentThought, ModelFeatureEnum.vision] + expect(supportFunctionCall(features)).toBe(false) + }) + + test('returns false for empty array', () => { + expect(supportFunctionCall([])).toBe(false) + }) + + test('returns false for undefined', () => { + expect(supportFunctionCall(undefined)).toBe(false) + }) + + test('returns false for null', () => { + expect(supportFunctionCall(null as any)).toBe(false) + }) + }) +}) diff --git a/web/utils/urlValidation.spec.ts b/web/utils/urlValidation.spec.ts new file mode 100644 index 0000000000..af06b24fa6 --- /dev/null +++ b/web/utils/urlValidation.spec.ts @@ -0,0 +1,49 @@ +import { validateRedirectUrl } from './urlValidation' + +describe('URL Validation', () => { + describe('validateRedirectUrl', () => { + it('should reject data: protocol', () => { + expect(() => validateRedirectUrl('data:text/html,<script>alert(1)</script>')).toThrow('Authorization URL must be HTTP or HTTPS') + }) + + it('should reject file: protocol', () => { + expect(() => validateRedirectUrl('file:///etc/passwd')).toThrow('Authorization URL must be HTTP or HTTPS') + }) + + it('should reject ftp: protocol', () => { + expect(() => validateRedirectUrl('ftp://example.com')).toThrow('Authorization URL must be HTTP or HTTPS') + }) + + it('should reject vbscript: protocol', () => { + expect(() => validateRedirectUrl('vbscript:msgbox(1)')).toThrow('Authorization URL must be HTTP or HTTPS') + }) + + it('should reject malformed URLs', () => { + expect(() => validateRedirectUrl('not a url')).toThrow('Invalid URL') + expect(() => validateRedirectUrl('://example.com')).toThrow('Invalid URL') + expect(() => validateRedirectUrl('')).toThrow('Invalid URL') + }) + + it('should handle URLs with query parameters', () => { + expect(() => validateRedirectUrl('https://example.com?param=value')).not.toThrow() + expect(() => validateRedirectUrl('https://example.com?redirect=http://evil.com')).not.toThrow() + }) + + it('should handle URLs with fragments', () => { + expect(() => validateRedirectUrl('https://example.com#section')).not.toThrow() + expect(() => validateRedirectUrl('https://example.com/path#fragment')).not.toThrow() + }) + + it('should handle URLs with authentication', () => { + expect(() => validateRedirectUrl('https://user:pass@example.com')).not.toThrow() + }) + + it('should handle international domain names', () => { + expect(() => validateRedirectUrl('https://例え.jp')).not.toThrow() + }) + + it('should reject protocol-relative URLs', () => { + expect(() => validateRedirectUrl('//example.com')).toThrow('Invalid URL') + }) + }) +}) diff --git a/web/utils/urlValidation.ts b/web/utils/urlValidation.ts index abc15a1365..db6de5275a 100644 --- a/web/utils/urlValidation.ts +++ b/web/utils/urlValidation.ts @@ -21,3 +21,44 @@ export function validateRedirectUrl(url: string): void { throw new Error(`Invalid URL: ${url}`) } } + +/** + * Check if URL is a private/local network address or cloud debug URL + * @param url - The URL string to check + * @returns true if the URL is a private/local address or cloud debug URL + */ +export function isPrivateOrLocalAddress(url: string): boolean { + try { + const urlObj = new URL(url) + const hostname = urlObj.hostname.toLowerCase() + + // Check for localhost + if (hostname === 'localhost' || hostname === '127.0.0.1' || hostname === '::1') + return true + + // Check for private IP ranges + const ipv4Regex = /^(\d+)\.(\d+)\.(\d+)\.(\d+)$/ + const ipv4Match = hostname.match(ipv4Regex) + if (ipv4Match) { + const [, a, b] = ipv4Match.map(Number) + // 10.0.0.0/8 + if (a === 10) + return true + // 172.16.0.0/12 + if (a === 172 && b >= 16 && b <= 31) + return true + // 192.168.0.0/16 + if (a === 192 && b === 168) + return true + // 169.254.0.0/16 (link-local) + if (a === 169 && b === 254) + return true + } + + // Check for .local domains + return hostname.endsWith('.local') + } + catch { + return false + } +} diff --git a/web/utils/validators.spec.ts b/web/utils/validators.spec.ts new file mode 100644 index 0000000000..b09955d12e --- /dev/null +++ b/web/utils/validators.spec.ts @@ -0,0 +1,139 @@ +import { draft07Validator, forbidBooleanProperties } from './validators' + +describe('Validators', () => { + describe('draft07Validator', () => { + it('should validate a valid JSON schema', () => { + const validSchema = { + type: 'object', + properties: { + name: { type: 'string' }, + age: { type: 'number' }, + }, + } + const result = draft07Validator(validSchema) + expect(result.valid).toBe(true) + expect(result.errors).toHaveLength(0) + }) + + it('should invalidate schema with unknown type', () => { + const invalidSchema = { + type: 'invalid_type', + } + const result = draft07Validator(invalidSchema) + expect(result.valid).toBe(false) + expect(result.errors.length).toBeGreaterThan(0) + }) + + it('should validate nested schemas', () => { + const nestedSchema = { + type: 'object', + properties: { + user: { + type: 'object', + properties: { + name: { type: 'string' }, + address: { + type: 'object', + properties: { + street: { type: 'string' }, + city: { type: 'string' }, + }, + }, + }, + }, + }, + } + const result = draft07Validator(nestedSchema) + expect(result.valid).toBe(true) + }) + + it('should validate array schemas', () => { + const arraySchema = { + type: 'array', + items: { type: 'string' }, + } + const result = draft07Validator(arraySchema) + expect(result.valid).toBe(true) + }) + }) + + describe('forbidBooleanProperties', () => { + it('should return empty array for schema without boolean properties', () => { + const schema = { + properties: { + name: { type: 'string' }, + age: { type: 'number' }, + }, + } + const errors = forbidBooleanProperties(schema) + expect(errors).toHaveLength(0) + }) + + it('should detect boolean property at root level', () => { + const schema = { + properties: { + name: true, + age: { type: 'number' }, + }, + } + const errors = forbidBooleanProperties(schema) + expect(errors).toHaveLength(1) + expect(errors[0]).toContain('name') + }) + + it('should detect boolean properties in nested objects', () => { + const schema = { + properties: { + user: { + properties: { + name: true, + profile: { + properties: { + bio: false, + }, + }, + }, + }, + }, + } + const errors = forbidBooleanProperties(schema) + expect(errors).toHaveLength(2) + expect(errors.some(e => e.includes('user.name'))).toBe(true) + expect(errors.some(e => e.includes('user.profile.bio'))).toBe(true) + }) + + it('should handle schema without properties', () => { + const schema = { type: 'string' } + const errors = forbidBooleanProperties(schema) + expect(errors).toHaveLength(0) + }) + + it('should handle null schema', () => { + const errors = forbidBooleanProperties(null) + expect(errors).toHaveLength(0) + }) + + it('should handle empty schema', () => { + const errors = forbidBooleanProperties({}) + expect(errors).toHaveLength(0) + }) + + it('should provide correct path in error messages', () => { + const schema = { + properties: { + level1: { + properties: { + level2: { + properties: { + level3: true, + }, + }, + }, + }, + }, + } + const errors = forbidBooleanProperties(schema) + expect(errors[0]).toContain('level1.level2.level3') + }) + }) +}) diff --git a/web/utils/var.spec.ts b/web/utils/var.spec.ts new file mode 100644 index 0000000000..6f55df0d34 --- /dev/null +++ b/web/utils/var.spec.ts @@ -0,0 +1,236 @@ +import { + checkKey, + checkKeys, + getMarketplaceUrl, + getNewVar, + getNewVarInWorkflow, + getVars, + hasDuplicateStr, + replaceSpaceWithUnderscoreInVarNameInput, +} from './var' +import { InputVarType } from '@/app/components/workflow/types' + +describe('Variable Utilities', () => { + describe('checkKey', () => { + it('should return error for empty key when canBeEmpty is false', () => { + expect(checkKey('', false)).toBe('canNoBeEmpty') + }) + + it('should return true for empty key when canBeEmpty is true', () => { + expect(checkKey('', true)).toBe(true) + }) + + it('should return error for key that is too long', () => { + const longKey = 'a'.repeat(101) // Assuming MAX_VAR_KEY_LENGTH is 100 + expect(checkKey(longKey)).toBe('tooLong') + }) + + it('should return error for key starting with number', () => { + expect(checkKey('1variable')).toBe('notStartWithNumber') + }) + + it('should return true for valid key', () => { + expect(checkKey('valid_variable_name')).toBe(true) + expect(checkKey('validVariableName')).toBe(true) + expect(checkKey('valid123')).toBe(true) + }) + + it('should return error for invalid characters', () => { + expect(checkKey('invalid-key')).toBe('notValid') + expect(checkKey('invalid key')).toBe('notValid') + expect(checkKey('invalid.key')).toBe('notValid') + expect(checkKey('invalid@key')).toBe('notValid') + }) + + it('should handle underscore correctly', () => { + expect(checkKey('_valid')).toBe(true) + expect(checkKey('valid_name')).toBe(true) + expect(checkKey('valid_name_123')).toBe(true) + }) + }) + + describe('checkKeys', () => { + it('should return valid for all valid keys', () => { + const result = checkKeys(['key1', 'key2', 'validKey']) + expect(result.isValid).toBe(true) + expect(result.errorKey).toBe('') + expect(result.errorMessageKey).toBe('') + }) + + it('should return error for first invalid key', () => { + const result = checkKeys(['validKey', '1invalid', 'anotherValid']) + expect(result.isValid).toBe(false) + expect(result.errorKey).toBe('1invalid') + expect(result.errorMessageKey).toBe('notStartWithNumber') + }) + + it('should handle empty array', () => { + const result = checkKeys([]) + expect(result.isValid).toBe(true) + }) + + it('should stop checking after first error', () => { + const result = checkKeys(['valid', 'invalid-key', '1invalid']) + expect(result.isValid).toBe(false) + expect(result.errorKey).toBe('invalid-key') + expect(result.errorMessageKey).toBe('notValid') + }) + }) + + describe('hasDuplicateStr', () => { + it('should return false for unique strings', () => { + expect(hasDuplicateStr(['a', 'b', 'c'])).toBe(false) + }) + + it('should return true for duplicate strings', () => { + expect(hasDuplicateStr(['a', 'b', 'a'])).toBe(true) + expect(hasDuplicateStr(['test', 'test'])).toBe(true) + }) + + it('should handle empty array', () => { + expect(hasDuplicateStr([])).toBe(false) + }) + + it('should handle single element', () => { + expect(hasDuplicateStr(['single'])).toBe(false) + }) + + it('should handle multiple duplicates', () => { + expect(hasDuplicateStr(['a', 'b', 'a', 'b', 'c'])).toBe(true) + }) + }) + + describe('getVars', () => { + it('should extract variables from template string', () => { + const result = getVars('Hello {{name}}, your age is {{age}}') + expect(result).toEqual(['name', 'age']) + }) + + it('should handle empty string', () => { + expect(getVars('')).toEqual([]) + }) + + it('should handle string without variables', () => { + expect(getVars('Hello world')).toEqual([]) + }) + + it('should remove duplicate variables', () => { + const result = getVars('{{name}} and {{name}} again') + expect(result).toEqual(['name']) + }) + + it('should filter out placeholder variables', () => { + const result = getVars('{{#context#}} {{name}} {{#histories#}}') + expect(result).toEqual(['name']) + }) + + it('should handle variables with underscores', () => { + const result = getVars('{{user_name}} {{user_age}}') + expect(result).toEqual(['user_name', 'user_age']) + }) + + it('should handle variables with numbers', () => { + const result = getVars('{{var1}} {{var2}} {{var123}}') + expect(result).toEqual(['var1', 'var2', 'var123']) + }) + + it('should ignore invalid variable names', () => { + const result = getVars('{{1invalid}} {{valid}} {{-invalid}}') + expect(result).toEqual(['valid']) + }) + + it('should filter out variables that are too long', () => { + const longVar = 'a'.repeat(101) + const result = getVars(`{{${longVar}}} {{valid}}`) + expect(result).toEqual(['valid']) + }) + }) + + describe('getNewVar', () => { + it('should create new string variable', () => { + const result = getNewVar('testKey', 'string') + expect(result.key).toBe('testKey') + expect(result.type).toBe('string') + expect(result.name).toBe('testKey') + }) + + it('should create new number variable', () => { + const result = getNewVar('numKey', 'number') + expect(result.key).toBe('numKey') + expect(result.type).toBe('number') + }) + + it('should truncate long names', () => { + const longKey = 'a'.repeat(100) + const result = getNewVar(longKey, 'string') + expect(result.name.length).toBeLessThanOrEqual(result.key.length) + }) + }) + + describe('getNewVarInWorkflow', () => { + it('should create text input variable by default', () => { + const result = getNewVarInWorkflow('testVar') + expect(result.variable).toBe('testVar') + expect(result.type).toBe(InputVarType.textInput) + expect(result.label).toBe('testVar') + }) + + it('should create select variable', () => { + const result = getNewVarInWorkflow('selectVar', InputVarType.select) + expect(result.variable).toBe('selectVar') + expect(result.type).toBe(InputVarType.select) + }) + + it('should create number variable', () => { + const result = getNewVarInWorkflow('numVar', InputVarType.number) + expect(result.variable).toBe('numVar') + expect(result.type).toBe(InputVarType.number) + }) + }) + + describe('getMarketplaceUrl', () => { + beforeEach(() => { + Object.defineProperty(window, 'location', { + value: { origin: 'https://example.com' }, + writable: true, + }) + }) + + it('should add additional parameters', () => { + const url = getMarketplaceUrl('/plugins', { category: 'ai', version: '1.0' }) + expect(url).toContain('category=ai') + expect(url).toContain('version=1.0') + }) + + it('should skip undefined parameters', () => { + const url = getMarketplaceUrl('/plugins', { category: 'ai', version: undefined }) + expect(url).toContain('category=ai') + expect(url).not.toContain('version=') + }) + }) + + describe('replaceSpaceWithUnderscoreInVarNameInput', () => { + it('should replace spaces with underscores', () => { + const input = document.createElement('input') + input.value = 'test variable name' + replaceSpaceWithUnderscoreInVarNameInput(input) + expect(input.value).toBe('test_variable_name') + }) + + it('should preserve cursor position', () => { + const input = document.createElement('input') + input.value = 'test name' + input.setSelectionRange(5, 5) + replaceSpaceWithUnderscoreInVarNameInput(input) + expect(input.selectionStart).toBe(5) + expect(input.selectionEnd).toBe(5) + }) + + it('should handle multiple spaces', () => { + const input = document.createElement('input') + input.value = 'test multiple spaces' + replaceSpaceWithUnderscoreInVarNameInput(input) + expect(input.value).toBe('test__multiple___spaces') + }) + }) +})