mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-03 20:44:36 +00:00
The /api/v1/system/debug/exec endpoint passes user-supplied HTTP body directly to Python exec(), enabling arbitrary code execution for any authenticated user when debug_mode is enabled. This is a critical security risk (CWE-94): a single misconfiguration or compromised JWT grants full server-side code execution. Remove the endpoint entirely. The /debug/plugin/action endpoint (which does not use exec()) is left intact as it serves a different, scoped purpose. Co-authored-by: Junyan Chin <rockchinq@gmail.com>
190 lines
7.6 KiB
Python
190 lines
7.6 KiB
Python
import json
|
|
|
|
import quart
|
|
import sqlalchemy
|
|
|
|
from .. import group
|
|
from .....utils import constants
|
|
from .....entity.persistence.metadata import Metadata
|
|
|
|
|
|
@group.group_class('system', '/api/v1/system')
|
|
class SystemRouterGroup(group.RouterGroup):
|
|
async def initialize(self) -> None:
|
|
@self.route('/info', methods=['GET'], auth_type=group.AuthType.NONE)
|
|
async def _() -> str:
|
|
# Read wizard_status and wizard_progress from metadata table
|
|
wizard_status = 'none'
|
|
wizard_progress = None
|
|
try:
|
|
result = await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.select(Metadata).where(Metadata.key.in_(['wizard_status', 'wizard_progress']))
|
|
)
|
|
for row in result:
|
|
if row.key == 'wizard_status':
|
|
wizard_status = row.value
|
|
elif row.key == 'wizard_progress':
|
|
try:
|
|
wizard_progress = json.loads(row.value)
|
|
except (json.JSONDecodeError, TypeError):
|
|
wizard_progress = None
|
|
except Exception:
|
|
pass
|
|
|
|
return self.success(
|
|
data={
|
|
'version': constants.semantic_version,
|
|
'debug': constants.debug_mode,
|
|
'edition': constants.edition,
|
|
'enable_marketplace': self.ap.instance_config.data.get('plugin', {}).get(
|
|
'enable_marketplace', True
|
|
),
|
|
'cloud_service_url': (
|
|
self.ap.instance_config.data.get('space', {}).get('url', 'https://space.langbot.app')
|
|
),
|
|
'allow_modify_login_info': self.ap.instance_config.data.get('system', {}).get(
|
|
'allow_modify_login_info', True
|
|
),
|
|
'disable_models_service': self.ap.instance_config.data.get('space', {}).get(
|
|
'disable_models_service', False
|
|
),
|
|
'limitation': self.ap.instance_config.data.get('system', {}).get('limitation', {}),
|
|
'wizard_status': wizard_status,
|
|
'wizard_progress': wizard_progress,
|
|
}
|
|
)
|
|
|
|
@self.route('/wizard/completed', methods=['POST'], auth_type=group.AuthType.USER_TOKEN)
|
|
async def _() -> str:
|
|
"""Mark wizard status in metadata table and clear progress.
|
|
|
|
Accepts JSON body: { "status": "skipped" | "completed" }
|
|
"""
|
|
data = await quart.request.get_json(silent=True) or {}
|
|
status = data.get('status', 'completed')
|
|
if status not in ('skipped', 'completed'):
|
|
return self.http_status(400, 400, f'Invalid wizard status: {status}')
|
|
|
|
try:
|
|
result = await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.select(Metadata).where(Metadata.key == 'wizard_status')
|
|
)
|
|
if result.first():
|
|
await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.update(Metadata).where(Metadata.key == 'wizard_status').values(value=status)
|
|
)
|
|
else:
|
|
await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.insert(Metadata).values(key='wizard_status', value=status)
|
|
)
|
|
|
|
# Clear wizard progress when wizard is completed/skipped
|
|
await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.delete(Metadata).where(Metadata.key == 'wizard_progress')
|
|
)
|
|
except Exception as e:
|
|
return self.http_status(500, 500, f'Failed to update wizard status: {e}')
|
|
|
|
return self.success(data={})
|
|
|
|
@self.route('/wizard/progress', methods=['PUT'], auth_type=group.AuthType.USER_TOKEN)
|
|
async def _() -> str:
|
|
"""Save wizard progress to metadata table.
|
|
|
|
Accepts JSON body with wizard state fields:
|
|
{ "step": int, "selected_adapter": str|null, "created_bot_uuid": str|null,
|
|
"bot_saved": bool, "selected_runner": str|null }
|
|
"""
|
|
data = await quart.request.get_json(silent=True) or {}
|
|
progress_json = json.dumps(data, ensure_ascii=False)
|
|
|
|
try:
|
|
result = await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.select(Metadata).where(Metadata.key == 'wizard_progress')
|
|
)
|
|
if result.first():
|
|
await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.update(Metadata).where(Metadata.key == 'wizard_progress').values(value=progress_json)
|
|
)
|
|
else:
|
|
await self.ap.persistence_mgr.execute_async(
|
|
sqlalchemy.insert(Metadata).values(key='wizard_progress', value=progress_json)
|
|
)
|
|
except Exception as e:
|
|
return self.http_status(500, 500, f'Failed to save wizard progress: {e}')
|
|
|
|
return self.success(data={})
|
|
|
|
@self.route('/tasks', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
|
async def _() -> str:
|
|
task_type = quart.request.args.get('type')
|
|
task_kind = quart.request.args.get('kind')
|
|
|
|
if task_type == '':
|
|
task_type = None
|
|
if task_kind == '':
|
|
task_kind = None
|
|
|
|
return self.success(data=self.ap.task_mgr.get_tasks_dict(task_type, task_kind))
|
|
|
|
@self.route('/tasks/<task_id>', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
|
async def _(task_id: str) -> str:
|
|
task = self.ap.task_mgr.get_task_by_id(int(task_id))
|
|
|
|
if task is None:
|
|
return self.http_status(404, 404, 'Task not found')
|
|
|
|
return self.success(data=task.to_dict())
|
|
|
|
@self.route('/storage-analysis', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
|
async def _() -> str:
|
|
return self.success(data=await self.ap.maintenance_service.get_storage_analysis())
|
|
|
|
@self.route(
|
|
'/debug/plugin/action',
|
|
methods=['POST'],
|
|
auth_type=group.AuthType.USER_TOKEN,
|
|
)
|
|
async def _() -> str:
|
|
if not constants.debug_mode:
|
|
return self.http_status(403, 403, 'Forbidden')
|
|
|
|
data = await quart.request.json
|
|
|
|
class AnoymousAction:
|
|
value = 'anonymous_action'
|
|
|
|
def __init__(self, value: str):
|
|
self.value = value
|
|
|
|
resp = await self.ap.plugin_connector.handler.call_action(
|
|
AnoymousAction(data['action']),
|
|
data['data'],
|
|
timeout=data.get('timeout', 10),
|
|
)
|
|
|
|
return self.success(data=resp)
|
|
|
|
@self.route(
|
|
'/status/plugin-system',
|
|
methods=['GET'],
|
|
auth_type=group.AuthType.USER_TOKEN,
|
|
)
|
|
async def _() -> str:
|
|
plugin_connector_error = 'ok'
|
|
is_connected = True
|
|
|
|
try:
|
|
await self.ap.plugin_connector.ping_plugin_runtime()
|
|
except Exception as e:
|
|
plugin_connector_error = str(e)
|
|
is_connected = False
|
|
|
|
return self.success(
|
|
data={
|
|
'is_enable': self.ap.plugin_connector.is_enable_plugin,
|
|
'is_connected': is_connected,
|
|
'plugin_connector_error': plugin_connector_error,
|
|
}
|
|
)
|