mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-08 06:46:02 +00:00
* Initial plan * Add package structure and resource path utilities - Created langbot/ package with __init__.py and __main__.py entry point - Added paths utility to find frontend and resource files from package installation - Updated config loading to use resource paths - Updated frontend serving to use resource paths - Added MANIFEST.in for package data inclusion - Updated pyproject.toml with build system and entry points Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Add PyPI publishing workflow and update license - Created GitHub Actions workflow to build frontend and publish to PyPI - Added license field to pyproject.toml to fix deprecation warning - Updated .gitignore to exclude build artifacts - Tested package building successfully Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Add PyPI installation documentation - Created PYPI_INSTALLATION.md with detailed installation and usage instructions - Updated README.md to feature uvx/pip installation as recommended method - Updated README_EN.md with same changes for English documentation Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Address code review feedback - Made package-data configuration more specific to langbot package only - Improved path detection with caching to avoid repeated file I/O - Removed sys.path searching which was incorrect for package data - Removed interactive input() call for non-interactive environment compatibility - Simplified error messages for version check Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Fix code review issues - Use specific exception types instead of bare except - Fix misleading comments about directory levels - Remove redundant existence check before makedirs with exist_ok=True - Use context manager for file opening to ensure proper cleanup Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Simplify package configuration and document behavioral differences - Removed redundant package-data configuration, relying on MANIFEST.in - Added documentation about behavioral differences between package and source installation - Clarified that include-package-data=true uses MANIFEST.in for data files Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * chore: update pyproject.toml * chore: try pack templates in langbot/ * chore: update * chore: update * chore: update * chore: update * chore: update * chore: adjust dir structure * chore: fix imports * fix: read default-pipeline-config.json * fix: read default-pipeline-config.json * fix: tests * ci: publish pypi * chore: bump version 4.6.0-beta.1 for testing * chore: add templates/** * fix: send adapters and requesters icons * chore: bump version 4.6.0b2 for testing * chore: add platform field for docker-compose.yaml --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> Co-authored-by: Junyan Qin <rockchinq@gmail.com>
107 lines
4.0 KiB
Python
107 lines
4.0 KiB
Python
from __future__ import annotations
|
|
|
|
import asyncio
|
|
import logging
|
|
import aiohttp
|
|
import uuid
|
|
from typing import TYPE_CHECKING
|
|
|
|
if TYPE_CHECKING:
|
|
from ..core import app
|
|
|
|
import langbot_plugin.api.entities.builtin.platform.events as platform_events
|
|
|
|
|
|
class WebhookPusher:
|
|
"""Push bot events to configured webhooks"""
|
|
|
|
ap: app.Application
|
|
logger: logging.Logger
|
|
|
|
def __init__(self, ap: app.Application):
|
|
self.ap = ap
|
|
self.logger = self.ap.logger
|
|
|
|
async def push_person_message(self, event: platform_events.FriendMessage, bot_uuid: str, adapter_name: str) -> None:
|
|
"""Push person message event to webhooks"""
|
|
try:
|
|
webhooks = await self.ap.webhook_service.get_enabled_webhooks()
|
|
if not webhooks:
|
|
return
|
|
|
|
# Build payload
|
|
payload = {
|
|
'uuid': str(uuid.uuid4()), # unique id for the event
|
|
'event_type': 'bot.person_message',
|
|
'data': {
|
|
'bot_uuid': bot_uuid,
|
|
'adapter_name': adapter_name,
|
|
'sender': {
|
|
'id': str(event.sender.id),
|
|
'name': getattr(event.sender, 'name', ''),
|
|
},
|
|
'message': event.message_chain.model_dump(),
|
|
'timestamp': event.time if hasattr(event, 'time') else None,
|
|
},
|
|
}
|
|
|
|
# Push to all webhooks asynchronously
|
|
tasks = [self._push_to_webhook(webhook['url'], payload) for webhook in webhooks]
|
|
await asyncio.gather(*tasks, return_exceptions=True)
|
|
|
|
except Exception as e:
|
|
self.logger.error(f'Failed to push person message to webhooks: {e}')
|
|
|
|
async def push_group_message(self, event: platform_events.GroupMessage, bot_uuid: str, adapter_name: str) -> None:
|
|
"""Push group message event to webhooks"""
|
|
try:
|
|
webhooks = await self.ap.webhook_service.get_enabled_webhooks()
|
|
if not webhooks:
|
|
return
|
|
|
|
# Build payload
|
|
payload = {
|
|
'uuid': str(uuid.uuid4()), # unique id for the event
|
|
'event_type': 'bot.group_message',
|
|
'data': {
|
|
'bot_uuid': bot_uuid,
|
|
'adapter_name': adapter_name,
|
|
'group': {
|
|
'id': str(event.group.id),
|
|
'name': getattr(event.group, 'name', ''),
|
|
},
|
|
'sender': {
|
|
'id': str(event.sender.id),
|
|
'name': getattr(event.sender, 'name', ''),
|
|
},
|
|
'message': event.message_chain.model_dump(),
|
|
'timestamp': event.time if hasattr(event, 'time') else None,
|
|
},
|
|
}
|
|
|
|
# Push to all webhooks asynchronously
|
|
tasks = [self._push_to_webhook(webhook['url'], payload) for webhook in webhooks]
|
|
await asyncio.gather(*tasks, return_exceptions=True)
|
|
|
|
except Exception as e:
|
|
self.logger.error(f'Failed to push group message to webhooks: {e}')
|
|
|
|
async def _push_to_webhook(self, url: str, payload: dict) -> None:
|
|
"""Push payload to a single webhook URL"""
|
|
try:
|
|
async with aiohttp.ClientSession() as session:
|
|
async with session.post(
|
|
url,
|
|
json=payload,
|
|
headers={'Content-Type': 'application/json'},
|
|
timeout=aiohttp.ClientTimeout(total=15),
|
|
) as response:
|
|
if response.status >= 400:
|
|
self.logger.warning(f'Webhook {url} returned status {response.status}')
|
|
else:
|
|
self.logger.debug(f'Successfully pushed to webhook {url}')
|
|
except asyncio.TimeoutError:
|
|
self.logger.warning(f'Timeout pushing to webhook {url}')
|
|
except Exception as e:
|
|
self.logger.warning(f'Error pushing to webhook {url}: {e}')
|