mirror of
https://github.com/lnbits/lnbits-legend.git
synced 2025-02-25 07:07:48 +01:00
* feat: add `max_lnbits_version` * fix: min/max lables * chore: `make bundle` * refactor: extract `is_version_compatible_with_lnbits` * refactor: rename method
179 lines
5.7 KiB
Python
179 lines
5.7 KiB
Python
import asyncio
|
|
import importlib
|
|
from typing import Optional
|
|
|
|
from loguru import logger
|
|
|
|
from lnbits.core import core_app_extra
|
|
from lnbits.core.crud import (
|
|
create_installed_extension,
|
|
delete_installed_extension,
|
|
get_db_version,
|
|
get_installed_extension,
|
|
update_installed_extension_state,
|
|
)
|
|
from lnbits.core.crud.extensions import (
|
|
get_installed_extensions,
|
|
update_installed_extension,
|
|
)
|
|
from lnbits.core.helpers import migrate_extension_database
|
|
from lnbits.settings import settings
|
|
|
|
from ..models.extensions import Extension, ExtensionMeta, InstallableExtension
|
|
|
|
|
|
async def install_extension(ext_info: InstallableExtension) -> Extension:
|
|
|
|
ext_info.meta = ext_info.meta or ExtensionMeta()
|
|
|
|
if ext_info.meta.installed_release:
|
|
assert (
|
|
ext_info.meta.installed_release.is_version_compatible
|
|
), "Incompatible extension version"
|
|
|
|
installed_ext = await get_installed_extension(ext_info.id)
|
|
if installed_ext and installed_ext.meta:
|
|
ext_info.meta.payments = installed_ext.meta.payments
|
|
|
|
await ext_info.download_archive()
|
|
|
|
ext_info.extract_archive()
|
|
|
|
db_version = await get_db_version(ext_info.id)
|
|
await migrate_extension_database(ext_info, db_version)
|
|
|
|
# if the extensions does not exist in the installed extensions table, create it
|
|
# if it does exist, it will be activated later in the code
|
|
if not installed_ext:
|
|
await create_installed_extension(ext_info)
|
|
else:
|
|
await update_installed_extension(ext_info)
|
|
|
|
extension = Extension.from_installable_ext(ext_info)
|
|
if extension.is_upgrade_extension:
|
|
# call stop while the old routes are still active
|
|
await stop_extension_background_work(ext_info.id)
|
|
|
|
await start_extension_background_work(ext_info.id)
|
|
|
|
return extension
|
|
|
|
|
|
async def uninstall_extension(ext_id: str):
|
|
await stop_extension_background_work(ext_id)
|
|
|
|
settings.deactivate_extension_paths(ext_id)
|
|
|
|
extension = await get_installed_extension(ext_id)
|
|
if extension:
|
|
extension.clean_extension_files()
|
|
await delete_installed_extension(ext_id=ext_id)
|
|
|
|
|
|
async def activate_extension(ext: Extension):
|
|
core_app_extra.register_new_ext_routes(ext)
|
|
await update_installed_extension_state(ext_id=ext.code, active=True)
|
|
|
|
|
|
async def deactivate_extension(ext_id: str):
|
|
settings.deactivate_extension_paths(ext_id)
|
|
await update_installed_extension_state(ext_id=ext_id, active=False)
|
|
|
|
|
|
async def stop_extension_background_work(ext_id: str) -> bool:
|
|
"""
|
|
Stop background work for extension (like asyncio.Tasks, WebSockets, etc).
|
|
Extension must expose a `myextension_stop()` function if it is starting tasks.
|
|
"""
|
|
upgrade_hash = settings.extension_upgrade_hash(ext_id)
|
|
ext = Extension(code=ext_id, is_valid=True, upgrade_hash=upgrade_hash)
|
|
|
|
try:
|
|
logger.info(f"Stopping background work for extension '{ext.module_name}'.")
|
|
old_module = importlib.import_module(ext.module_name)
|
|
|
|
stop_fn_name = f"{ext_id}_stop"
|
|
assert hasattr(
|
|
old_module, stop_fn_name
|
|
), f"No stop function found for '{ext.module_name}'."
|
|
|
|
stop_fn = getattr(old_module, stop_fn_name)
|
|
if stop_fn:
|
|
if asyncio.iscoroutinefunction(stop_fn):
|
|
await stop_fn()
|
|
else:
|
|
stop_fn()
|
|
logger.info(f"Stopped background work for extension '{ext.module_name}'.")
|
|
except Exception as ex:
|
|
logger.warning(f"Failed to stop background work for '{ext.module_name}'.")
|
|
logger.warning(ex)
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
async def start_extension_background_work(ext_id: str) -> bool:
|
|
"""
|
|
Start background work for extension (like asyncio.Tasks, WebSockets, etc).
|
|
Extension CAN expose a `myextension_start()` function if it is starting tasks.
|
|
Extension MUST expose a `myextension_stop()` in that case.
|
|
"""
|
|
upgrade_hash = settings.extension_upgrade_hash(ext_id)
|
|
ext = Extension(code=ext_id, is_valid=True, upgrade_hash=upgrade_hash)
|
|
|
|
try:
|
|
logger.info(f"Starting background work for extension '{ext.module_name}'.")
|
|
new_module = importlib.import_module(ext.module_name)
|
|
start_fn_name = f"{ext_id}_start"
|
|
|
|
# start function is optional, return False if not found
|
|
if not hasattr(new_module, start_fn_name):
|
|
return False
|
|
|
|
start_fn = getattr(new_module, start_fn_name)
|
|
if start_fn:
|
|
if asyncio.iscoroutinefunction(start_fn):
|
|
await start_fn()
|
|
else:
|
|
start_fn()
|
|
logger.info(f"Started background work for extension '{ext.module_name}'.")
|
|
return True
|
|
except Exception as ex:
|
|
logger.warning(f"Failed to start background work for '{ext.module_name}'.")
|
|
logger.warning(ex)
|
|
return False
|
|
|
|
|
|
async def get_valid_extensions(
|
|
include_deactivated: Optional[bool] = True,
|
|
) -> list[Extension]:
|
|
installed_extensions = await get_installed_extensions()
|
|
valid_extensions = [Extension.from_installable_ext(e) for e in installed_extensions]
|
|
|
|
if include_deactivated:
|
|
return valid_extensions
|
|
|
|
if settings.lnbits_extensions_deactivate_all:
|
|
return []
|
|
|
|
return [
|
|
e
|
|
for e in valid_extensions
|
|
if e.code not in settings.lnbits_deactivated_extensions
|
|
]
|
|
|
|
|
|
async def get_valid_extension(
|
|
ext_id: str, include_deactivated: Optional[bool] = True
|
|
) -> Optional[Extension]:
|
|
ext = await get_installed_extension(ext_id)
|
|
if not ext:
|
|
return None
|
|
|
|
if include_deactivated:
|
|
return Extension.from_installable_ext(ext)
|
|
|
|
if settings.lnbits_extensions_deactivate_all:
|
|
return None
|
|
|
|
return Extension.from_installable_ext(ext)
|