dify/api/controllers/console/red_blue_challenges.py
Joey Yakimowich-Payne 6038fc25f5
Add multi-site Caddy helpers and document usage
- add startup/shutdown scripts that render a Caddyfile from JSON config and run health checks

- add Python utilities and a sample sites.json for declarative multi-site configuration

- document the workflow and ignore generated Caddy state artifacts

- normalize double-quote style across challenge workflow controllers, nodes, and tests
2025-10-15 22:03:56 -06:00

144 lines
5 KiB
Python

from __future__ import annotations
from flask_restx import Resource, reqparse
from controllers.console import console_ns as api
from controllers.console.wraps import (
account_initialization_required,
setup_required,
)
from extensions.ext_database import db
from libs.login import current_user, login_required
from models.red_blue import RedBlueChallenge, TeamPairing
@api.route("/red-blue-challenges")
class RedBlueListCreateApi(Resource):
@api.doc("list_red_blue_challenges")
@setup_required
@login_required
@account_initialization_required
def get(self):
tenant_id = current_user.current_tenant_id
if not tenant_id:
return {"result": "success", "data": []}
rows = (
db.session.query(RedBlueChallenge)
.filter(RedBlueChallenge.tenant_id == tenant_id)
.order_by(RedBlueChallenge.created_at.desc())
.all()
)
return {
"result": "success",
"data": [
{
"id": r.id,
"name": r.name,
"description": r.description,
"is_active": r.is_active,
}
for r in rows
],
}
@api.doc("create_red_blue_challenge")
@setup_required
@login_required
@account_initialization_required
def post(self):
parser = reqparse.RequestParser()
parser.add_argument("tenant_id", type=str, required=True, location="json")
parser.add_argument("app_id", type=str, required=True, location="json")
parser.add_argument("name", type=str, required=True, location="json")
parser.add_argument("description", type=str, required=False, location="json")
parser.add_argument("judge_suite", type=dict, required=True, location="json")
args = parser.parse_args()
c = RedBlueChallenge()
c.tenant_id = args.get("tenant_id") or current_user.current_tenant_id
c.app_id = args["app_id"]
c.name = args["name"]
c.description = args.get("description")
c.judge_suite = args["judge_suite"]
db.session.add(c)
db.session.commit()
return {"result": "success", "data": {"id": c.id}}, 201
@api.route("/red-blue-challenges/<uuid:challenge_id>")
class RedBlueDetailApi(Resource):
@api.doc("get_red_blue_challenge")
@setup_required
@login_required
@account_initialization_required
def get(self, challenge_id):
c = db.session.get(RedBlueChallenge, str(challenge_id))
if not c:
return {"result": "not_found"}, 404
return {
"result": "success",
"data": {"id": c.id, "name": c.name, "description": c.description, "is_active": c.is_active},
}
@api.doc("update_red_blue_challenge")
@setup_required
@login_required
@account_initialization_required
def patch(self, challenge_id):
c = db.session.get(RedBlueChallenge, str(challenge_id))
if not c:
return {"result": "not_found"}, 404
parser = reqparse.RequestParser()
parser.add_argument("name", type=str, required=False, location="json")
parser.add_argument("description", type=str, required=False, location="json")
parser.add_argument("is_active", type=bool, required=False, location="json")
args = parser.parse_args()
if args.get("name"):
c.name = args["name"]
if args.get("description") is not None:
c.description = args["description"]
if args.get("is_active") is not None:
c.is_active = bool(args["is_active"])
db.session.commit()
return {"result": "success"}
@api.doc("delete_red_blue_challenge")
@setup_required
@login_required
@account_initialization_required
def delete(self, challenge_id):
c = db.session.get(RedBlueChallenge, str(challenge_id))
if not c:
return {"result": "not_found"}, 404
db.session.delete(c)
db.session.commit()
return {"result": "success"}, 204
@api.route("/red-blue-challenges/<uuid:challenge_id>/pairings")
class RedBluePairingsApi(Resource):
@api.doc("list_red_blue_pairings")
@setup_required
@login_required
@account_initialization_required
def get(self, challenge_id):
rows = (
db.session.query(TeamPairing)
.filter(TeamPairing.red_blue_challenge_id == str(challenge_id))
.order_by(TeamPairing.created_at.desc())
.limit(100)
.all()
)
return {
"result": "success",
"data": [
{
"id": r.id,
"red_points": r.red_points,
"blue_points": r.blue_points,
"judge_rating": r.judge_rating,
"created_at": r.created_at.isoformat() if hasattr(r.created_at, "isoformat") else None,
}
for r in rows
],
}