import uuid from typing import cast from flask_login import current_user # type: ignore from flask_restful import Resource, inputs, marshal, marshal_with, reqparse # type: ignore from sqlalchemy import select from sqlalchemy.orm import Session from werkzeug.exceptions import BadRequest, Forbidden, abort from controllers.console import api from controllers.console.app.wraps import get_app_model from controllers.console.wraps import ( account_initialization_required, cloud_edition_billing_resource_check, enterprise_license_required, setup_required, ) from core.ops.ops_trace_manager import OpsTraceManager 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 login_required from models import Account, App from services.app_dsl_service import AppDslService, ImportMode from services.app_service import AppService ALLOW_CREATE_APP_MODES = ["chat", "agent-chat", "advanced-chat", "workflow", "completion"] class AppListApi(Resource): @setup_required @login_required @account_initialization_required @enterprise_license_required def get(self): """Get app list""" def uuid_list(value): try: return [str(uuid.UUID(v)) for v in value.split(",")] except ValueError: abort(400, message="Invalid UUID format in tag_ids.") parser = reqparse.RequestParser() parser.add_argument("page", type=inputs.int_range(1, 99999), required=False, default=1, location="args") parser.add_argument("limit", type=inputs.int_range(1, 100), required=False, default=20, location="args") parser.add_argument( "mode", type=str, choices=["chat", "workflow", "agent-chat", "channel", "all"], default="all", location="args", required=False, ) parser.add_argument("name", type=str, location="args", required=False) parser.add_argument("tag_ids", type=uuid_list, location="args", required=False) parser.add_argument("is_created_by_me", type=inputs.boolean, location="args", required=False) args = parser.parse_args() # get app list app_service = AppService() app_pagination = app_service.get_paginate_apps(current_user.id, current_user.current_tenant_id, args) if not app_pagination: return {"data": [], "total": 0, "page": 1, "limit": 20, "has_more": False} return marshal(app_pagination, app_pagination_fields) @setup_required @login_required @account_initialization_required @marshal_with(app_detail_fields) @cloud_edition_billing_resource_check("apps") def post(self): """Create app""" parser = reqparse.RequestParser() parser.add_argument("name", type=str, required=True, location="json") parser.add_argument("description", type=str, location="json") parser.add_argument("mode", type=str, choices=ALLOW_CREATE_APP_MODES, location="json") parser.add_argument("icon_type", type=str, location="json") parser.add_argument("icon", type=str, location="json") parser.add_argument("icon_background", type=str, location="json") args = parser.parse_args() # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() if "mode" not in args or args["mode"] is None: raise BadRequest("mode is required") app_service = AppService() app = app_service.create_app(current_user.current_tenant_id, args, current_user) return app, 201 class AppApi(Resource): @setup_required @login_required @account_initialization_required @enterprise_license_required @get_app_model @marshal_with(app_detail_fields_with_site) def get(self, app_model): """Get app detail""" app_service = AppService() app_model = app_service.get_app(app_model) return app_model @setup_required @login_required @account_initialization_required @get_app_model @marshal_with(app_detail_fields_with_site) def put(self, app_model): """Update app""" # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() parser = reqparse.RequestParser() parser.add_argument("name", type=str, required=True, nullable=False, location="json") parser.add_argument("description", type=str, location="json") parser.add_argument("icon_type", type=str, location="json") parser.add_argument("icon", type=str, location="json") parser.add_argument("icon_background", type=str, location="json") parser.add_argument("max_active_requests", type=int, location="json") parser.add_argument("use_icon_as_answer_icon", type=bool, location="json") args = parser.parse_args() app_service = AppService() app_model = app_service.update_app(app_model, args) return app_model @setup_required @login_required @account_initialization_required @get_app_model def delete(self, app_model): """Delete app""" # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() app_service = AppService() app_service.delete_app(app_model) return {"result": "success"}, 204 class AppCopyApi(Resource): @setup_required @login_required @account_initialization_required @get_app_model @marshal_with(app_detail_fields_with_site) def post(self, app_model): """Copy app""" # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() parser = reqparse.RequestParser() parser.add_argument("name", type=str, location="json") parser.add_argument("description", type=str, location="json") parser.add_argument("icon_type", type=str, location="json") parser.add_argument("icon", type=str, location="json") parser.add_argument("icon_background", type=str, location="json") args = parser.parse_args() with Session(db.engine) as session: import_service = AppDslService(session) yaml_content = import_service.export_dsl(app_model=app_model, include_secret=True) account = cast(Account, current_user) result = import_service.import_app( account=account, import_mode=ImportMode.YAML_CONTENT.value, yaml_content=yaml_content, name=args.get("name"), description=args.get("description"), icon_type=args.get("icon_type"), icon=args.get("icon"), icon_background=args.get("icon_background"), ) session.commit() stmt = select(App).where(App.id == result.app_id) app = session.scalar(stmt) return app, 201 class AppExportApi(Resource): @setup_required @login_required @account_initialization_required @get_app_model def get(self, app_model): """Export app""" # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() # Add include_secret params parser = reqparse.RequestParser() parser.add_argument("include_secret", type=inputs.boolean, default=False, location="args") args = parser.parse_args() return {"data": AppDslService.export_dsl(app_model=app_model, include_secret=args["include_secret"])} class AppNameApi(Resource): @setup_required @login_required @account_initialization_required @get_app_model @marshal_with(app_detail_fields) def post(self, app_model): # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() parser = reqparse.RequestParser() parser.add_argument("name", type=str, required=True, location="json") args = parser.parse_args() app_service = AppService() app_model = app_service.update_app_name(app_model, args.get("name")) return app_model class AppIconApi(Resource): @setup_required @login_required @account_initialization_required @get_app_model @marshal_with(app_detail_fields) def post(self, app_model): # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() parser = reqparse.RequestParser() parser.add_argument("icon", type=str, location="json") parser.add_argument("icon_background", type=str, location="json") args = parser.parse_args() app_service = AppService() app_model = app_service.update_app_icon(app_model, args.get("icon"), args.get("icon_background")) return app_model class AppSiteStatus(Resource): @setup_required @login_required @account_initialization_required @get_app_model @marshal_with(app_detail_fields) def post(self, app_model): # The role of the current user in the ta table must be admin, owner, or editor if not current_user.is_editor: raise Forbidden() parser = reqparse.RequestParser() parser.add_argument("enable_site", type=bool, required=True, location="json") args = parser.parse_args() app_service = AppService() app_model = app_service.update_app_site_status(app_model, args.get("enable_site")) return app_model class AppApiStatus(Resource): @setup_required @login_required @account_initialization_required @get_app_model @marshal_with(app_detail_fields) def post(self, app_model): # The role of the current user in the ta table must be admin or owner if not current_user.is_admin_or_owner: raise Forbidden() parser = reqparse.RequestParser() parser.add_argument("enable_api", type=bool, required=True, location="json") args = parser.parse_args() app_service = AppService() app_model = app_service.update_app_api_status(app_model, args.get("enable_api")) return app_model class AppTraceApi(Resource): @setup_required @login_required @account_initialization_required def get(self, app_id): """Get app trace""" app_trace_config = OpsTraceManager.get_app_tracing_config(app_id=app_id) return app_trace_config @setup_required @login_required @account_initialization_required def post(self, app_id): # add app trace if not current_user.is_admin_or_owner: raise Forbidden() parser = reqparse.RequestParser() parser.add_argument("enabled", type=bool, required=True, location="json") parser.add_argument("tracing_provider", type=str, required=True, location="json") args = parser.parse_args() OpsTraceManager.update_app_tracing_config( app_id=app_id, enabled=args["enabled"], tracing_provider=args["tracing_provider"], ) return {"result": "success"} api.add_resource(AppListApi, "/apps") api.add_resource(AppApi, "/apps/") api.add_resource(AppCopyApi, "/apps//copy") api.add_resource(AppExportApi, "/apps//export") api.add_resource(AppNameApi, "/apps//name") api.add_resource(AppIconApi, "/apps//icon") api.add_resource(AppSiteStatus, "/apps//site-enable") api.add_resource(AppApiStatus, "/apps//api-enable") api.add_resource(AppTraceApi, "/apps//trace")