feat: add llm catalog (type) management module
- Add CRUD API for llmcatelog (list/create/update/delete) - Add llmcatelog.ui as main entry for catalog management - Add llmcatelog_list.ui as DataViewer interface - Add index.ui as module navigation page - Update json/llmcatelog.json with editable section
This commit is contained in:
parent
4280459fc4
commit
1b74b5bf10
@ -1,15 +1,18 @@
|
|||||||
{
|
{
|
||||||
"tblname": "llmcatelog",
|
"tblname": "llmcatelog",
|
||||||
"title":"模型类目",
|
"alias": "llmcatelog_list",
|
||||||
|
"title": "模型类型管理",
|
||||||
"params": {
|
"params": {
|
||||||
"sortby":"name",
|
"sortby": ["name"],
|
||||||
|
"editable": {
|
||||||
|
"new_data_url": "{{entire_url('../api/llmcatelog_create.dspy')}}",
|
||||||
|
"update_data_url": "{{entire_url('../api/llmcatelog_update.dspy')}}",
|
||||||
|
"delete_data_url": "{{entire_url('../api/llmcatelog_delete.dspy')}}"
|
||||||
|
},
|
||||||
"browserfields": {
|
"browserfields": {
|
||||||
"exclouded": ["id"],
|
"exclouded": ["id"],
|
||||||
"alters": {}
|
"alters": {}
|
||||||
},
|
},
|
||||||
"editexclouded": [
|
"editexclouded": ["id"]
|
||||||
"id"
|
|
||||||
],
|
|
||||||
"record_toolbar": null
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
29
wwwroot/api/llmcatelog_create.dspy
Normal file
29
wwwroot/api/llmcatelog_create.dspy
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
import json
|
||||||
|
|
||||||
|
result = {'widgettype': 'Message', 'options': {'title': 'Error', 'message': 'Invalid', 'type': 'error'}}
|
||||||
|
|
||||||
|
try:
|
||||||
|
dbname = get_module_dbname('llmage')
|
||||||
|
name = params_kw.get('name', '').strip()
|
||||||
|
description = params_kw.get('description', '').strip()
|
||||||
|
|
||||||
|
if not name:
|
||||||
|
result['options'] = {'title': '错误', 'message': '类型名不能为空', 'type': 'error'}
|
||||||
|
else:
|
||||||
|
async with DBPools().sqlorContext(dbname) as sor:
|
||||||
|
# 检查名称是否已存在
|
||||||
|
check_sql = "select id from llmcatelog where name=${name}$"
|
||||||
|
exists = await sor.sqlExe(check_sql, {'name': name})
|
||||||
|
if exists:
|
||||||
|
result['options'] = {'title': '提示', 'message': f'类型名「{name}」已存在', 'type': 'warning'}
|
||||||
|
else:
|
||||||
|
new_id = getID()
|
||||||
|
insert_sql = "insert into llmcatelog (id, name, description) values (${id}$, ${name}$, ${description}$)"
|
||||||
|
await sor.sqlExe(insert_sql, {'id': new_id, 'name': name, 'description': description})
|
||||||
|
result = {'widgettype': 'Message', 'options': {'title': '成功', 'message': f'类型「{name}」已创建', 'type': 'info'}}
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
result['options'] = {'title': '错误', 'message': str(e), 'type': 'error'}
|
||||||
|
|
||||||
|
return json.dumps(result, ensure_ascii=False)
|
||||||
33
wwwroot/api/llmcatelog_delete.dspy
Normal file
33
wwwroot/api/llmcatelog_delete.dspy
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
import json
|
||||||
|
|
||||||
|
result = {'widgettype': 'Message', 'options': {'title': 'Error', 'message': 'Invalid', 'type': 'error'}}
|
||||||
|
|
||||||
|
try:
|
||||||
|
dbname = get_module_dbname('llmage')
|
||||||
|
id = params_kw.get('id', '').strip()
|
||||||
|
|
||||||
|
if not id:
|
||||||
|
result['options'] = {'title': '错误', 'message': '记录ID不能为空', 'type': 'error'}
|
||||||
|
else:
|
||||||
|
async with DBPools().sqlorContext(dbname) as sor:
|
||||||
|
# 检查是否有模型关联此类型
|
||||||
|
check_sql = "select count(*) as cnt from llm_catalog_rel where llmcatelogid=${id}$"
|
||||||
|
rel_count = await sor.sqlExe(check_sql, {'id': id})
|
||||||
|
cnt = rel_count[0]['cnt'] if rel_count else 0
|
||||||
|
if cnt > 0:
|
||||||
|
result['options'] = {'title': '提示', 'message': f'该类型下还有 {cnt} 个模型关联,无法删除', 'type': 'warning'}
|
||||||
|
else:
|
||||||
|
# 获取类型名用于提示
|
||||||
|
name_sql = "select name from llmcatelog where id=${id}$"
|
||||||
|
name_rows = await sor.sqlExe(name_sql, {'id': id})
|
||||||
|
type_name = name_rows[0]['name'] if name_rows else id
|
||||||
|
|
||||||
|
delete_sql = "delete from llmcatelog where id=${id}$"
|
||||||
|
await sor.sqlExe(delete_sql, {'id': id})
|
||||||
|
result = {'widgettype': 'Message', 'options': {'title': '成功', 'message': f'类型「{type_name}」已删除', 'type': 'info'}}
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
result['options'] = {'title': '错误', 'message': str(e), 'type': 'error'}
|
||||||
|
|
||||||
|
return json.dumps(result, ensure_ascii=False)
|
||||||
28
wwwroot/api/llmcatelog_list.dspy
Normal file
28
wwwroot/api/llmcatelog_list.dspy
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
import json
|
||||||
|
|
||||||
|
result = {'success': False, 'rows': [], 'total': 0}
|
||||||
|
|
||||||
|
try:
|
||||||
|
dbname = get_module_dbname('llmage')
|
||||||
|
page = int(params_kw.get('page', 1))
|
||||||
|
rows_per_page = int(params_kw.get('rows', 20))
|
||||||
|
offset = (page - 1) * rows_per_page
|
||||||
|
|
||||||
|
async with DBPools().sqlorContext(dbname) as sor:
|
||||||
|
# 获取总数
|
||||||
|
count_sql = "select count(*) as cnt from llmcatelog"
|
||||||
|
count_rows = await sor.sqlExe(count_sql, {})
|
||||||
|
total = count_rows[0]['cnt'] if count_rows else 0
|
||||||
|
|
||||||
|
# 获取分页数据
|
||||||
|
data_sql = "select id, name, description from llmcatelog order by name limit ${limit}$ offset ${offset}$"
|
||||||
|
rows = await sor.sqlExe(data_sql, {'limit': rows_per_page, 'offset': offset})
|
||||||
|
result['rows'] = [dict(r) for r in (rows or [])]
|
||||||
|
result['total'] = total
|
||||||
|
result['success'] = True
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
result['error'] = str(e)
|
||||||
|
|
||||||
|
return json.dumps(result, ensure_ascii=False, default=str)
|
||||||
31
wwwroot/api/llmcatelog_update.dspy
Normal file
31
wwwroot/api/llmcatelog_update.dspy
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
import json
|
||||||
|
|
||||||
|
result = {'widgettype': 'Message', 'options': {'title': 'Error', 'message': 'Invalid', 'type': 'error'}}
|
||||||
|
|
||||||
|
try:
|
||||||
|
dbname = get_module_dbname('llmage')
|
||||||
|
id = params_kw.get('id', '').strip()
|
||||||
|
name = params_kw.get('name', '').strip()
|
||||||
|
description = params_kw.get('description', '').strip()
|
||||||
|
|
||||||
|
if not id:
|
||||||
|
result['options'] = {'title': '错误', 'message': '记录ID不能为空', 'type': 'error'}
|
||||||
|
elif not name:
|
||||||
|
result['options'] = {'title': '错误', 'message': '类型名不能为空', 'type': 'error'}
|
||||||
|
else:
|
||||||
|
async with DBPools().sqlorContext(dbname) as sor:
|
||||||
|
# 检查名称是否已被其他记录使用
|
||||||
|
check_sql = "select id from llmcatelog where name=${name}$ and id != ${id}$"
|
||||||
|
exists = await sor.sqlExe(check_sql, {'name': name, 'id': id})
|
||||||
|
if exists:
|
||||||
|
result['options'] = {'title': '提示', 'message': f'类型名「{name}」已被其他记录使用', 'type': 'warning'}
|
||||||
|
else:
|
||||||
|
update_sql = "update llmcatelog set name=${name}$, description=${description}$ where id=${id}$"
|
||||||
|
await sor.sqlExe(update_sql, {'name': name, 'description': description, 'id': id})
|
||||||
|
result = {'widgettype': 'Message', 'options': {'title': '成功', 'message': f'类型「{name}」已更新', 'type': 'info'}}
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
result['options'] = {'title': '错误', 'message': str(e), 'type': 'error'}
|
||||||
|
|
||||||
|
return json.dumps(result, ensure_ascii=False)
|
||||||
177
wwwroot/index.ui
Normal file
177
wwwroot/index.ui
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
{
|
||||||
|
"widgettype": "VBox",
|
||||||
|
"options": {
|
||||||
|
"width": "100%",
|
||||||
|
"height": "100%",
|
||||||
|
"padding": "20px",
|
||||||
|
"spacing": 16
|
||||||
|
},
|
||||||
|
"subwidgets": [
|
||||||
|
{
|
||||||
|
"widgettype": "Title2",
|
||||||
|
"options": {
|
||||||
|
"text": "LLM 模型管理",
|
||||||
|
"halign": "left"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "ResponsableBox",
|
||||||
|
"options": {
|
||||||
|
"gap": "16px",
|
||||||
|
"minWidth": "250px"
|
||||||
|
},
|
||||||
|
"subwidgets": [
|
||||||
|
{
|
||||||
|
"widgettype": "VBox",
|
||||||
|
"options": {
|
||||||
|
"backgroundColor": "#1e3a5f",
|
||||||
|
"padding": "24px",
|
||||||
|
"cursor": "pointer",
|
||||||
|
"borderRadius": "8px"
|
||||||
|
},
|
||||||
|
"binds": [
|
||||||
|
{
|
||||||
|
"wid": "self",
|
||||||
|
"event": "click",
|
||||||
|
"actiontype": "urlwidget",
|
||||||
|
"target": "app.llmage_content",
|
||||||
|
"options": {
|
||||||
|
"url": "{{entire_url('/llmage/llmcatelog_list.ui')}}"
|
||||||
|
},
|
||||||
|
"mode": "replace"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"subwidgets": [
|
||||||
|
{
|
||||||
|
"widgettype": "Svg",
|
||||||
|
"options": {
|
||||||
|
"svg": "<svg width=\"40\" height=\"40\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"#90caf9\" stroke-width=\"2\"><path d=\"M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z\"/></svg>",
|
||||||
|
"width": "40px",
|
||||||
|
"height": "40px"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "Title4",
|
||||||
|
"options": {
|
||||||
|
"text": "模型类型管理",
|
||||||
|
"color": "#ffffff",
|
||||||
|
"marginTop": "12px"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "Text",
|
||||||
|
"options": {
|
||||||
|
"text": "管理模型的分类和类型",
|
||||||
|
"color": "#90caf9",
|
||||||
|
"fontSize": "14px"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "VBox",
|
||||||
|
"options": {
|
||||||
|
"backgroundColor": "#1e3a5f",
|
||||||
|
"padding": "24px",
|
||||||
|
"cursor": "pointer",
|
||||||
|
"borderRadius": "8px"
|
||||||
|
},
|
||||||
|
"binds": [
|
||||||
|
{
|
||||||
|
"wid": "self",
|
||||||
|
"event": "click",
|
||||||
|
"actiontype": "urlwidget",
|
||||||
|
"target": "app.llmage_content",
|
||||||
|
"options": {
|
||||||
|
"url": "{{entire_url('/llmage/llm_catalog_rel_manage.ui')}}"
|
||||||
|
},
|
||||||
|
"mode": "replace"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"subwidgets": [
|
||||||
|
{
|
||||||
|
"widgettype": "Svg",
|
||||||
|
"options": {
|
||||||
|
"svg": "<svg width=\"40\" height=\"40\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"#ce93d8\" stroke-width=\"2\"><path d=\"M13.5 6H5.25A2.25 2.25 0 003 8.25v10.5A2.25 2.25 0 005.25 21h10.5A2.25 2.25 0 0018 18.75V10.5m-10.5 6L21 3m0 0h-5.25M21 3v5.25\"/></svg>",
|
||||||
|
"width": "40px",
|
||||||
|
"height": "40px"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "Title4",
|
||||||
|
"options": {
|
||||||
|
"text": "模型-类型关联",
|
||||||
|
"color": "#ffffff",
|
||||||
|
"marginTop": "12px"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "Text",
|
||||||
|
"options": {
|
||||||
|
"text": "管理模型与类型的多对多关系",
|
||||||
|
"color": "#ce93d8",
|
||||||
|
"fontSize": "14px"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "VBox",
|
||||||
|
"options": {
|
||||||
|
"backgroundColor": "#1e3a5f",
|
||||||
|
"padding": "24px",
|
||||||
|
"cursor": "pointer",
|
||||||
|
"borderRadius": "8px"
|
||||||
|
},
|
||||||
|
"binds": [
|
||||||
|
{
|
||||||
|
"wid": "self",
|
||||||
|
"event": "click",
|
||||||
|
"actiontype": "urlwidget",
|
||||||
|
"target": "app.llmage_content",
|
||||||
|
"options": {
|
||||||
|
"url": "{{entire_url('/llmage/llm')}}"
|
||||||
|
},
|
||||||
|
"mode": "replace"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"subwidgets": [
|
||||||
|
{
|
||||||
|
"widgettype": "Svg",
|
||||||
|
"options": {
|
||||||
|
"svg": "<svg width=\"40\" height=\"40\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"#4caf50\" stroke-width=\"2\"><path d=\"M9.75 3.104v5.714a2.25 2.25 0 01-.659 1.591L5 14.5M9.75 3.104c-.251.023-.501.05-.75.082m.75-.082a24.301 24.301 0 014.5 0m0 0v5.714c0 .597.237 1.17.659 1.591L19.8 15.3M14.25 3.104c.251.023.501.05.75.082M19.8 15.3l-1.57.393A9.065 9.065 0 0112 15.75c-2.062 0-4.024-.614-5.67-1.757l-1.57-.393m15.04 0L12 21 5.25 13.893\"/></svg>",
|
||||||
|
"width": "40px",
|
||||||
|
"height": "40px"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "Title4",
|
||||||
|
"options": {
|
||||||
|
"text": "模型管理",
|
||||||
|
"color": "#ffffff",
|
||||||
|
"marginTop": "12px"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "Text",
|
||||||
|
"options": {
|
||||||
|
"text": "管理 LLM 模型配置",
|
||||||
|
"color": "#4caf50",
|
||||||
|
"fontSize": "14px"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"widgettype": "VBox",
|
||||||
|
"id": "llmage_content",
|
||||||
|
"options": {
|
||||||
|
"width": "100%",
|
||||||
|
"flex": "1",
|
||||||
|
"marginTop": "20px"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
16
wwwroot/llmcatelog.ui
Normal file
16
wwwroot/llmcatelog.ui
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
{
|
||||||
|
"widgettype": "DataViewer",
|
||||||
|
"options": {
|
||||||
|
"url": "{{entire_url('/llmage/api/llmcatelog_list.dspy')}}",
|
||||||
|
"title": "模型类型管理",
|
||||||
|
"pageSize": 20,
|
||||||
|
"new_data_url": "{{entire_url('/llmage/api/llmcatelog_create.dspy')}}",
|
||||||
|
"update_data_url": "{{entire_url('/llmage/api/llmcatelog_update.dspy')}}",
|
||||||
|
"delete_data_url": "{{entire_url('/llmage/api/llmcatelog_delete.dspy')}}",
|
||||||
|
"fields": [
|
||||||
|
{"name": "id", "title": "ID", "hidden": true},
|
||||||
|
{"name": "name", "title": "类型名", "width": "200px", "editable": true},
|
||||||
|
{"name": "description", "title": "类型说明", "width": "50%", "editable": true}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
16
wwwroot/llmcatelog_list.ui
Normal file
16
wwwroot/llmcatelog_list.ui
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
{
|
||||||
|
"widgettype": "DataViewer",
|
||||||
|
"options": {
|
||||||
|
"url": "{{entire_url('/llmage/api/llmcatelog_list.dspy')}}",
|
||||||
|
"title": "模型类型管理",
|
||||||
|
"pageSize": 20,
|
||||||
|
"new_data_url": "{{entire_url('/llmage/api/llmcatelog_create.dspy')}}",
|
||||||
|
"update_data_url": "{{entire_url('/llmage/api/llmcatelog_update.dspy')}}",
|
||||||
|
"delete_data_url": "{{entire_url('/llmage/api/llmcatelog_delete.dspy')}}",
|
||||||
|
"fields": [
|
||||||
|
{"name": "id", "title": "ID", "hidden": true},
|
||||||
|
{"name": "name", "title": "类型名", "width": "200px", "editable": true},
|
||||||
|
{"name": "description", "title": "类型说明", "width": "50%", "editable": true}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user