
Signed-off-by: yihong0618 <zouzou0208@gmail.com> Signed-off-by: -LAN- <laipz8200@outlook.com> Co-authored-by: kurokobo <kuro664@gmail.com> Co-authored-by: Hiroshi Fujita <fujita-h@users.noreply.github.com> Co-authored-by: NFish <douxc512@gmail.com> Co-authored-by: Gen Sato <52241300+halogen22@users.noreply.github.com> Co-authored-by: eux <euxuuu@gmail.com> Co-authored-by: huangzhuo1949 <167434202+huangzhuo1949@users.noreply.github.com> Co-authored-by: huangzhuo <huangzhuo1@xiaomi.com> Co-authored-by: lotsik <lotsik@mail.ru> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: nite-knite <nkCoding@gmail.com> Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: gakkiyomi <gakkiyomi@aliyun.com> Co-authored-by: CN-P5 <heibai2006@gmail.com> Co-authored-by: CN-P5 <heibai2006@qq.com> Co-authored-by: Chuehnone <1897025+chuehnone@users.noreply.github.com> Co-authored-by: yihong <zouzou0208@gmail.com> Co-authored-by: Kevin9703 <51311316+Kevin9703@users.noreply.github.com> Co-authored-by: -LAN- <laipz8200@outlook.com> Co-authored-by: Boris Feld <lothiraldan@gmail.com> Co-authored-by: mbo <himabo@gmail.com> Co-authored-by: mabo <mabo@aeyes.ai> Co-authored-by: Warren Chen <warren.chen830@gmail.com> Co-authored-by: KVOJJJin <jzongcode@gmail.com> Co-authored-by: JzoNgKVO <27049666+JzoNgKVO@users.noreply.github.com> Co-authored-by: jiandanfeng <chenjh3@wangsu.com> Co-authored-by: zhu-an <70234959+xhdd123321@users.noreply.github.com> Co-authored-by: zhaoqingyu.1075 <zhaoqingyu.1075@bytedance.com> Co-authored-by: 海狸大師 <86974027+yenslife@users.noreply.github.com> Co-authored-by: Xu Song <xusong.vip@gmail.com> Co-authored-by: rayshaw001 <396301947@163.com> Co-authored-by: Ding Jiatong <dingjiatong@gmail.com> Co-authored-by: Bowen Liang <liangbowen@gf.com.cn> Co-authored-by: JasonVV <jasonwangiii@outlook.com> Co-authored-by: le0zh <newlight@qq.com> Co-authored-by: zhuxinliang <zhuxinliang@didiglobal.com> Co-authored-by: k-zaku <zaku99@outlook.jp> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: luckylhb90 <luckylhb90@gmail.com> Co-authored-by: hobo.l <hobo.l@binance.com> Co-authored-by: jiangbo721 <365065261@qq.com> Co-authored-by: 刘江波 <jiangbo721@163.com> Co-authored-by: Shun Miyazawa <34241526+miya@users.noreply.github.com> Co-authored-by: EricPan <30651140+Egfly@users.noreply.github.com> Co-authored-by: crazywoola <427733928@qq.com> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: sino <sino2322@gmail.com> Co-authored-by: Jhvcc <37662342+Jhvcc@users.noreply.github.com> Co-authored-by: lowell <lowell.hu@zkteco.in>
200 lines
7.7 KiB
Python
200 lines
7.7 KiB
Python
from typing import Optional
|
|
|
|
from core.ops.ops_trace_manager import OpsTraceManager, provider_config_map
|
|
from extensions.ext_database import db
|
|
from models.model import App, TraceAppConfig
|
|
|
|
|
|
class OpsService:
|
|
@classmethod
|
|
def get_tracing_app_config(cls, app_id: str, tracing_provider: str):
|
|
"""
|
|
Get tracing app config
|
|
:param app_id: app id
|
|
:param tracing_provider: tracing provider
|
|
:return:
|
|
"""
|
|
trace_config_data: Optional[TraceAppConfig] = (
|
|
db.session.query(TraceAppConfig)
|
|
.filter(TraceAppConfig.app_id == app_id, TraceAppConfig.tracing_provider == tracing_provider)
|
|
.first()
|
|
)
|
|
|
|
if not trace_config_data:
|
|
return None
|
|
|
|
# decrypt_token and obfuscated_token
|
|
tenant = db.session.query(App).filter(App.id == app_id).first()
|
|
if not tenant:
|
|
return None
|
|
tenant_id = tenant.tenant_id
|
|
decrypt_tracing_config = OpsTraceManager.decrypt_tracing_config(
|
|
tenant_id, tracing_provider, trace_config_data.tracing_config
|
|
)
|
|
new_decrypt_tracing_config = OpsTraceManager.obfuscated_decrypt_token(tracing_provider, decrypt_tracing_config)
|
|
|
|
if tracing_provider == "langfuse" and (
|
|
"project_key" not in decrypt_tracing_config or not decrypt_tracing_config.get("project_key")
|
|
):
|
|
try:
|
|
project_key = OpsTraceManager.get_trace_config_project_key(decrypt_tracing_config, tracing_provider)
|
|
new_decrypt_tracing_config.update(
|
|
{
|
|
"project_url": "{host}/project/{key}".format(
|
|
host=decrypt_tracing_config.get("host"), key=project_key
|
|
)
|
|
}
|
|
)
|
|
except Exception:
|
|
new_decrypt_tracing_config.update(
|
|
{"project_url": "{host}/".format(host=decrypt_tracing_config.get("host"))}
|
|
)
|
|
|
|
if tracing_provider == "langsmith" and (
|
|
"project_url" not in decrypt_tracing_config or not decrypt_tracing_config.get("project_url")
|
|
):
|
|
try:
|
|
project_url = OpsTraceManager.get_trace_config_project_url(decrypt_tracing_config, tracing_provider)
|
|
new_decrypt_tracing_config.update({"project_url": project_url})
|
|
except Exception:
|
|
new_decrypt_tracing_config.update({"project_url": "https://smith.langchain.com/"})
|
|
|
|
if tracing_provider == "opik" and (
|
|
"project_url" not in decrypt_tracing_config or not decrypt_tracing_config.get("project_url")
|
|
):
|
|
try:
|
|
project_url = OpsTraceManager.get_trace_config_project_url(decrypt_tracing_config, tracing_provider)
|
|
new_decrypt_tracing_config.update({"project_url": project_url})
|
|
except Exception:
|
|
new_decrypt_tracing_config.update({"project_url": "https://www.comet.com/opik/"})
|
|
|
|
trace_config_data.tracing_config = new_decrypt_tracing_config
|
|
return trace_config_data.to_dict()
|
|
|
|
@classmethod
|
|
def create_tracing_app_config(cls, app_id: str, tracing_provider: str, tracing_config: dict):
|
|
"""
|
|
Create tracing app config
|
|
:param app_id: app id
|
|
:param tracing_provider: tracing provider
|
|
:param tracing_config: tracing config
|
|
:return:
|
|
"""
|
|
if tracing_provider not in provider_config_map and tracing_provider:
|
|
return {"error": f"Invalid tracing provider: {tracing_provider}"}
|
|
|
|
config_class, other_keys = (
|
|
provider_config_map[tracing_provider]["config_class"],
|
|
provider_config_map[tracing_provider]["other_keys"],
|
|
)
|
|
# FIXME: ignore type error
|
|
default_config_instance = config_class(**tracing_config) # type: ignore
|
|
for key in other_keys: # type: ignore
|
|
if key in tracing_config and tracing_config[key] == "":
|
|
tracing_config[key] = getattr(default_config_instance, key, None)
|
|
|
|
# api check
|
|
if not OpsTraceManager.check_trace_config_is_effective(tracing_config, tracing_provider):
|
|
return {"error": "Invalid Credentials"}
|
|
|
|
# get project url
|
|
if tracing_provider == "langfuse":
|
|
project_key = OpsTraceManager.get_trace_config_project_key(tracing_config, tracing_provider)
|
|
project_url = "{host}/project/{key}".format(host=tracing_config.get("host"), key=project_key)
|
|
elif tracing_provider in ("langsmith", "opik"):
|
|
project_url = OpsTraceManager.get_trace_config_project_url(tracing_config, tracing_provider)
|
|
else:
|
|
project_url = None
|
|
|
|
# check if trace config already exists
|
|
trace_config_data: Optional[TraceAppConfig] = (
|
|
db.session.query(TraceAppConfig)
|
|
.filter(TraceAppConfig.app_id == app_id, TraceAppConfig.tracing_provider == tracing_provider)
|
|
.first()
|
|
)
|
|
|
|
if trace_config_data:
|
|
return None
|
|
|
|
# get tenant id
|
|
tenant = db.session.query(App).filter(App.id == app_id).first()
|
|
if not tenant:
|
|
return None
|
|
tenant_id = tenant.tenant_id
|
|
tracing_config = OpsTraceManager.encrypt_tracing_config(tenant_id, tracing_provider, tracing_config)
|
|
if project_url:
|
|
tracing_config["project_url"] = project_url
|
|
trace_config_data = TraceAppConfig(
|
|
app_id=app_id,
|
|
tracing_provider=tracing_provider,
|
|
tracing_config=tracing_config,
|
|
)
|
|
db.session.add(trace_config_data)
|
|
db.session.commit()
|
|
|
|
return {"result": "success"}
|
|
|
|
@classmethod
|
|
def update_tracing_app_config(cls, app_id: str, tracing_provider: str, tracing_config: dict):
|
|
"""
|
|
Update tracing app config
|
|
:param app_id: app id
|
|
:param tracing_provider: tracing provider
|
|
:param tracing_config: tracing config
|
|
:return:
|
|
"""
|
|
if tracing_provider not in provider_config_map:
|
|
raise ValueError(f"Invalid tracing provider: {tracing_provider}")
|
|
|
|
# check if trace config already exists
|
|
current_trace_config = (
|
|
db.session.query(TraceAppConfig)
|
|
.filter(TraceAppConfig.app_id == app_id, TraceAppConfig.tracing_provider == tracing_provider)
|
|
.first()
|
|
)
|
|
|
|
if not current_trace_config:
|
|
return None
|
|
|
|
# get tenant id
|
|
tenant = db.session.query(App).filter(App.id == app_id).first()
|
|
if not tenant:
|
|
return None
|
|
tenant_id = tenant.tenant_id
|
|
tracing_config = OpsTraceManager.encrypt_tracing_config(
|
|
tenant_id, tracing_provider, tracing_config, current_trace_config.tracing_config
|
|
)
|
|
|
|
# api check
|
|
# decrypt_token
|
|
decrypt_tracing_config = OpsTraceManager.decrypt_tracing_config(tenant_id, tracing_provider, tracing_config)
|
|
if not OpsTraceManager.check_trace_config_is_effective(decrypt_tracing_config, tracing_provider):
|
|
raise ValueError("Invalid Credentials")
|
|
|
|
current_trace_config.tracing_config = tracing_config
|
|
db.session.commit()
|
|
|
|
return current_trace_config.to_dict()
|
|
|
|
@classmethod
|
|
def delete_tracing_app_config(cls, app_id: str, tracing_provider: str):
|
|
"""
|
|
Delete tracing app config
|
|
:param app_id: app id
|
|
:param tracing_provider: tracing provider
|
|
:return:
|
|
"""
|
|
trace_config = (
|
|
db.session.query(TraceAppConfig)
|
|
.filter(TraceAppConfig.app_id == app_id, TraceAppConfig.tracing_provider == tracing_provider)
|
|
.first()
|
|
)
|
|
|
|
if not trace_config:
|
|
return None
|
|
|
|
db.session.delete(trace_config)
|
|
db.session.commit()
|
|
|
|
return True
|