mirror of
https://github.com/langgenius/dify.git
synced 2025-11-12 17:34:48 +00:00
Signed-off-by: -LAN- <laipz8200@outlook.com> Co-authored-by: twwu <twwu@dify.ai> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: jyong <718720800@qq.com> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: QuantumGhost <obelisk.reg+git@gmail.com> Co-authored-by: lyzno1 <yuanyouhuilyz@gmail.com> Co-authored-by: quicksand <quicksandzn@gmail.com> Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com> Co-authored-by: lyzno1 <92089059+lyzno1@users.noreply.github.com> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: Yongtao Huang <yongtaoh2022@gmail.com> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: nite-knite <nkCoding@gmail.com> Co-authored-by: Hanqing Zhao <sherry9277@gmail.com> Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com> Co-authored-by: Harry <xh001x@hotmail.com>
140 lines
4.2 KiB
Python
140 lines
4.2 KiB
Python
from collections.abc import Callable
|
|
from functools import wraps
|
|
from typing import ParamSpec, TypeVar, cast
|
|
|
|
from flask import current_app, request
|
|
from flask_login import user_logged_in
|
|
from flask_restx import reqparse
|
|
from pydantic import BaseModel
|
|
from sqlalchemy.orm import Session
|
|
|
|
from extensions.ext_database import db
|
|
from libs.login import current_user
|
|
from models.account import Tenant
|
|
from models.model import DefaultEndUserSessionID, EndUser
|
|
|
|
P = ParamSpec("P")
|
|
R = TypeVar("R")
|
|
|
|
|
|
def get_user(tenant_id: str, user_id: str | None) -> EndUser:
|
|
"""
|
|
Get current user
|
|
|
|
NOTE: user_id is not trusted, it could be maliciously set to any value.
|
|
As a result, it could only be considered as an end user id.
|
|
"""
|
|
try:
|
|
with Session(db.engine) as session:
|
|
if not user_id:
|
|
user_id = DefaultEndUserSessionID.DEFAULT_SESSION_ID.value
|
|
|
|
user_model = (
|
|
session.query(EndUser)
|
|
.where(
|
|
EndUser.id == user_id,
|
|
EndUser.tenant_id == tenant_id,
|
|
)
|
|
.first()
|
|
)
|
|
if not user_model:
|
|
user_model = (
|
|
session.query(EndUser)
|
|
.where(
|
|
EndUser.session_id == user_id,
|
|
EndUser.tenant_id == tenant_id,
|
|
)
|
|
.first()
|
|
)
|
|
if not user_model:
|
|
user_model = EndUser(
|
|
tenant_id=tenant_id,
|
|
type="service_api",
|
|
is_anonymous=user_id == DefaultEndUserSessionID.DEFAULT_SESSION_ID.value,
|
|
session_id=user_id,
|
|
)
|
|
session.add(user_model)
|
|
session.commit()
|
|
session.refresh(user_model)
|
|
|
|
except Exception:
|
|
raise ValueError("user not found")
|
|
|
|
return user_model
|
|
|
|
|
|
def get_user_tenant(view: Callable[P, R] | None = None):
|
|
def decorator(view_func: Callable[P, R]):
|
|
@wraps(view_func)
|
|
def decorated_view(*args: P.args, **kwargs: P.kwargs):
|
|
# fetch json body
|
|
parser = reqparse.RequestParser()
|
|
parser.add_argument("tenant_id", type=str, required=True, location="json")
|
|
parser.add_argument("user_id", type=str, required=True, location="json")
|
|
|
|
p = parser.parse_args()
|
|
|
|
user_id = cast(str, p.get("user_id"))
|
|
tenant_id = cast(str, p.get("tenant_id"))
|
|
|
|
if not tenant_id:
|
|
raise ValueError("tenant_id is required")
|
|
|
|
if not user_id:
|
|
user_id = DefaultEndUserSessionID.DEFAULT_SESSION_ID.value
|
|
|
|
try:
|
|
tenant_model = (
|
|
db.session.query(Tenant)
|
|
.where(
|
|
Tenant.id == tenant_id,
|
|
)
|
|
.first()
|
|
)
|
|
except Exception:
|
|
raise ValueError("tenant not found")
|
|
|
|
if not tenant_model:
|
|
raise ValueError("tenant not found")
|
|
|
|
kwargs["tenant_model"] = tenant_model
|
|
|
|
user = get_user(tenant_id, user_id)
|
|
kwargs["user_model"] = user
|
|
|
|
current_app.login_manager._update_request_context_with_user(user) # type: ignore
|
|
user_logged_in.send(current_app._get_current_object(), user=current_user) # type: ignore
|
|
|
|
return view_func(*args, **kwargs)
|
|
|
|
return decorated_view
|
|
|
|
if view is None:
|
|
return decorator
|
|
else:
|
|
return decorator(view)
|
|
|
|
|
|
def plugin_data(view: Callable[P, R] | None = None, *, payload_type: type[BaseModel]):
|
|
def decorator(view_func: Callable[P, R]):
|
|
def decorated_view(*args: P.args, **kwargs: P.kwargs):
|
|
try:
|
|
data = request.get_json()
|
|
except Exception:
|
|
raise ValueError("invalid json")
|
|
|
|
try:
|
|
payload = payload_type(**data)
|
|
except Exception as e:
|
|
raise ValueError(f"invalid payload: {str(e)}")
|
|
|
|
kwargs["payload"] = payload
|
|
return view_func(*args, **kwargs)
|
|
|
|
return decorated_view
|
|
|
|
if view is None:
|
|
return decorator
|
|
else:
|
|
return decorator(view)
|