Compare commits
	
		
			4 Commits
		
	
	
		
			dev
			...
			profitroll
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| a458788841 | |||
| b76f727263 | |||
|  | 5a244f603d | ||
|  | 9bc4d0348d | 
| @@ -6,18 +6,19 @@ on: | |||||||
|       - main |       - main | ||||||
|       - dev |       - dev | ||||||
|   pull_request: |   pull_request: | ||||||
|     types: [ opened, synchronize, reopened ] |     types: [opened, synchronize, reopened] | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   sonarcloud: |   sonarcloud: | ||||||
|     name: SonarCloud |     name: SonarCloud | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|  |     container: catthehacker/ubuntu:act-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|         with: |         with: | ||||||
|           fetch-depth: 0 |           fetch-depth: 0 | ||||||
|       - name: SonarQube Scan |       - name: SonarCloud Scan | ||||||
|         uses: SonarSource/sonarqube-scan-action@v4.2.1 |         uses: SonarSource/sonarcloud-github-action@master | ||||||
|         env: |         env: | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} |           SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} | ||||||
| @@ -9,58 +9,81 @@ permissions: | |||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   release-build: |   release-build: | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|  |     container: catthehacker/ubuntu:act-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|       - uses: actions/setup-python@v5 |       - uses: actions/setup-python@v5 | ||||||
|         with: |         with: | ||||||
|           python-version: "3.x" |           python-version: "3.x" | ||||||
|  |  | ||||||
|       - name: Build release distributions |       - name: Build release distributions | ||||||
|         run: | |         run: | | ||||||
|           python -m pip install build |           python -m pip install build | ||||||
|           python -m build |           python -m build | ||||||
|  |  | ||||||
|       - name: Upload distributions |       - name: Upload distributions | ||||||
|         uses: christopherhx/gitea-upload-artifact@v4 |         uses: christopherhx/gitea-upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: release-dists |           name: release-dists | ||||||
|           path: dist/ |           path: dist/ | ||||||
|  |  | ||||||
|   gitea-publish: |   gitea-publish: | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|     needs: release-build |     container: catthehacker/ubuntu:act-latest | ||||||
|  |  | ||||||
|  |     needs: | ||||||
|  |       - release-build | ||||||
|  |  | ||||||
|     permissions: |     permissions: | ||||||
|       id-token: write |       id-token: write | ||||||
|  |  | ||||||
|     environment: |     environment: | ||||||
|       name: gitea |       name: gitea | ||||||
|       url: https://git.end-play.xyz/profitroll/-/packages/pypi/libbot |       url: https://git.end-play.xyz/profitroll/-/packages/pypi/libbot | ||||||
|  |  | ||||||
|     env: |     env: | ||||||
|       GITHUB_WORKFLOW_REF: ${{ gitea.workflow_ref }} |       GITHUB_WORKFLOW_REF: ${{ gitea.workflow_ref }} | ||||||
|       INPUT_REPOSITORY_URL: https://git.end-play.xyz/api/packages/profitroll/pypi |       INPUT_REPOSITORY_URL: https://git.end-play.xyz/api/packages/profitroll/pypi | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Retrieve release distributions |       - name: Retrieve release distributions | ||||||
|         uses: christopherhx/gitea-download-artifact@v4 |         uses: christopherhx/gitea-download-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: release-dists |           name: release-dists | ||||||
|           path: dist/ |           path: dist/ | ||||||
|  |  | ||||||
|       - name: Publish package distributions to TestPyPI |       - name: Publish package distributions to TestPyPI | ||||||
|         uses: pypa/gh-action-pypi-publish@release/v1 |         uses: pypa/gh-action-pypi-publish@release/v1 | ||||||
|         with: |         with: | ||||||
|           password: ${{ secrets.PYPI_GITEA_API_TOKEN }} |           password: ${{ secrets.PYPI_GITEA_API_TOKEN }} | ||||||
|           repository-url: https://git.end-play.xyz/api/packages/profitroll/pypi |           repository-url: https://git.end-play.xyz/api/packages/profitroll/pypi | ||||||
|  |  | ||||||
|   pypi-publish: |   pypi-publish: | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|     needs: release-build |     container: catthehacker/ubuntu:act-latest | ||||||
|  |  | ||||||
|  |     needs: | ||||||
|  |       - release-build | ||||||
|  |  | ||||||
|     permissions: |     permissions: | ||||||
|       id-token: write |       id-token: write | ||||||
|  |  | ||||||
|     environment: |     environment: | ||||||
|       name: pypi |       name: pypi | ||||||
|  |  | ||||||
|     env: |     env: | ||||||
|       GITHUB_WORKFLOW_REF: ${{ gitea.workflow_ref }} |       GITHUB_WORKFLOW_REF: ${{ gitea.workflow_ref }} | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Retrieve release distributions |       - name: Retrieve release distributions | ||||||
|         uses: christopherhx/gitea-download-artifact@v4 |         uses: christopherhx/gitea-download-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: release-dists |           name: release-dists | ||||||
|           path: dist/ |           path: dist/ | ||||||
|  |  | ||||||
|       - name: Publish package distributions to TestPyPI |       - name: Publish package distributions to TestPyPI | ||||||
|         uses: pypa/gh-action-pypi-publish@release/v1 |         uses: pypa/gh-action-pypi-publish@release/v1 | ||||||
|         with: |         with: | ||||||
|   | |||||||
| @@ -11,18 +11,18 @@ on: | |||||||
| jobs: | jobs: | ||||||
|   test: |   test: | ||||||
|     name: Build and Test |     name: Build and Test | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|  |     container: catthehacker/ubuntu:act-latest | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         python-version: [ "3.11", "3.12", "3.13" ] |         python-version: [ "3.11", "3.12", "3.13" ] | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v3 | ||||||
|       - name: Set up Python ${{ matrix.python-version }} |       - name: Set up Python ${{ matrix.python-version }} | ||||||
|         uses: actions/setup-python@v3 |         uses: actions/setup-python@v3 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ matrix.python-version }} |           python-version: ${{ matrix.python-version }} | ||||||
|           cache: 'pip' |  | ||||||
|           cache-dependency-path: './requirements/*' |  | ||||||
|         env: |         env: | ||||||
|           AGENT_TOOLSDIRECTORY: /opt/hostedtoolcache |           AGENT_TOOLSDIRECTORY: /opt/hostedtoolcache | ||||||
|       - name: Install dependencies |       - name: Install dependencies | ||||||
|   | |||||||
| @@ -16,7 +16,6 @@ There are different sub-packages available: | |||||||
| * pyrogram - Telegram bots with Pyrogram's fork "Pyrofork" | * pyrogram - Telegram bots with Pyrogram's fork "Pyrofork" | ||||||
| * pycord - Discord bots with Pycord | * pycord - Discord bots with Pycord | ||||||
| * speed - Performance improvements | * speed - Performance improvements | ||||||
| * cache - Support for Redis and Memcached |  | ||||||
| * dev - Dependencies for package development purposes | * dev - Dependencies for package development purposes | ||||||
|  |  | ||||||
| You can freely choose any sub-package you want, as well as add multiple (comma-separated) or none at all. | You can freely choose any sub-package you want, as well as add multiple (comma-separated) or none at all. | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| [build-system] | [build-system] | ||||||
| requires = ["setuptools>=77.0.3", "wheel"] | requires = ["setuptools>=62.6", "wheel"] | ||||||
| build-backend = "setuptools.build_meta" | build-backend = "setuptools.build_meta" | ||||||
|  |  | ||||||
| [project] | [project] | ||||||
| @@ -9,11 +9,11 @@ authors = [{ name = "Profitroll" }] | |||||||
| description = "Universal bot library with functions needed for basic Discord/Telegram bot development." | description = "Universal bot library with functions needed for basic Discord/Telegram bot development." | ||||||
| readme = "README.md" | readme = "README.md" | ||||||
| requires-python = ">=3.11" | requires-python = ">=3.11" | ||||||
| license = "GPL-3.0" | license = { text = "GPLv3" } | ||||||
| license-files = ["LICENSE"] |  | ||||||
| classifiers = [ | classifiers = [ | ||||||
|     "Development Status :: 3 - Alpha", |     "Development Status :: 3 - Alpha", | ||||||
|     "Intended Audience :: Developers", |     "Intended Audience :: Developers", | ||||||
|  |     "License :: OSI Approved :: GNU General Public License v3 (GPLv3)", | ||||||
|     "Operating System :: OS Independent", |     "Operating System :: OS Independent", | ||||||
|     "Programming Language :: Python :: 3.11", |     "Programming Language :: Python :: 3.11", | ||||||
|     "Programming Language :: Python :: 3.12", |     "Programming Language :: Python :: 3.12", | ||||||
| @@ -28,10 +28,10 @@ dependencies = { file = "requirements/_.txt" } | |||||||
|  |  | ||||||
| [tool.setuptools.dynamic.optional-dependencies] | [tool.setuptools.dynamic.optional-dependencies] | ||||||
| dev = { file = "requirements/dev.txt" } | dev = { file = "requirements/dev.txt" } | ||||||
|  | matrix = { file = "requirements/matrix.txt" } | ||||||
| pycord = { file = "requirements/pycord.txt" } | pycord = { file = "requirements/pycord.txt" } | ||||||
| pyrogram = { file = "requirements/pyrogram.txt" } | pyrogram = { file = "requirements/pyrogram.txt" } | ||||||
| speed = { file = "requirements/speed.txt" } | speed = { file = "requirements/speed.txt" } | ||||||
| cache = { file = "requirements/cache.txt" } |  | ||||||
|  |  | ||||||
| [project.urls] | [project.urls] | ||||||
| Source = "https://git.end-play.xyz/profitroll/LibBotUniversal" | Source = "https://git.end-play.xyz/profitroll/LibBotUniversal" | ||||||
|   | |||||||
| @@ -1,2 +1,2 @@ | |||||||
| aiofiles>=23.0.0 | aiofiles>=23.0.0 | ||||||
| typing-extensions~=4.15.0 | typing-extensions~=4.12.2 | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| pymemcache~=4.0.0 |  | ||||||
| redis~=6.4.0 |  | ||||||
| @@ -1,12 +1,12 @@ | |||||||
| black==25.9.0 | black==24.10.0 | ||||||
| build==1.3.0 | build==1.2.2.post1 | ||||||
| isort==5.13.2 | isort==5.13.2 | ||||||
| mypy==1.18.2 | mypy==1.14.1 | ||||||
| pylint==4.0.1 | pylint==3.3.3 | ||||||
| pytest-asyncio==1.2.0 | pytest-asyncio==0.25.0 | ||||||
| pytest-cov==7.0.0 | pytest-cov==6.0.0 | ||||||
| pytest==8.4.2 | pytest==8.3.4 | ||||||
| tox==4.31.0 | tox==4.23.2 | ||||||
| twine==6.2.0 | twine==6.0.1 | ||||||
| types-aiofiles==25.1.0.20251011 | types-aiofiles==24.1.0.20241221 | ||||||
| types-ujson==5.10.0.20250822 | types-ujson==5.10.0.20240515 | ||||||
							
								
								
									
										1
									
								
								requirements/matrix.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								requirements/matrix.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | simplematrixbotlib~=2.12.1 | ||||||
| @@ -1 +1 @@ | |||||||
| ujson~=5.11.0 | ujson~=5.10.0 | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| __version__ = "4.4.0" | __version__ = "4.0.2" | ||||||
| __license__ = "GPL3" | __license__ = "GPL3" | ||||||
| __author__ = "Profitroll" | __author__ = "Profitroll" | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								src/libbot/cache/__init__.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/libbot/cache/__init__.py
									
									
									
									
										vendored
									
									
								
							| @@ -1,2 +0,0 @@ | |||||||
| # This file is left empty on purpose |  | ||||||
| # Adding imports here will cause import errors when libbot[pycord] is not installed |  | ||||||
							
								
								
									
										3
									
								
								src/libbot/cache/classes/__init__.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								src/libbot/cache/classes/__init__.py
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +0,0 @@ | |||||||
| from .cache import Cache |  | ||||||
| from .cache_memcached import CacheMemcached |  | ||||||
| from .cache_redis import CacheRedis |  | ||||||
							
								
								
									
										44
									
								
								src/libbot/cache/classes/cache.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										44
									
								
								src/libbot/cache/classes/cache.py
									
									
									
									
										vendored
									
									
								
							| @@ -1,44 +0,0 @@ | |||||||
| from abc import ABC, abstractmethod |  | ||||||
| from typing import Any, Dict, Optional |  | ||||||
|  |  | ||||||
| import pymemcache |  | ||||||
| import redis |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Cache(ABC): |  | ||||||
|     client: pymemcache.Client | redis.Redis |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     @abstractmethod |  | ||||||
|     def from_config(cls, engine_config: Dict[str, Any]) -> Any: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     @abstractmethod |  | ||||||
|     def get_json(self, key: str) -> Any | None: |  | ||||||
|         # TODO This method must also carry out ObjectId conversion! |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     @abstractmethod |  | ||||||
|     def get_string(self, key: str) -> str | None: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     @abstractmethod |  | ||||||
|     def get_object(self, key: str) -> Any | None: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     @abstractmethod |  | ||||||
|     def set_json(self, key: str, value: Any, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         # TODO This method must also carry out ObjectId conversion! |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     @abstractmethod |  | ||||||
|     def set_string(self, key: str, value: str, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     @abstractmethod |  | ||||||
|     def set_object(self, key: str, value: Any, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     @abstractmethod |  | ||||||
|     def delete(self, key: str) -> None: |  | ||||||
|         pass |  | ||||||
							
								
								
									
										112
									
								
								src/libbot/cache/classes/cache_memcached.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										112
									
								
								src/libbot/cache/classes/cache_memcached.py
									
									
									
									
										vendored
									
									
								
							| @@ -1,112 +0,0 @@ | |||||||
| import logging |  | ||||||
| from logging import Logger |  | ||||||
| from typing import Dict, Any, Optional |  | ||||||
|  |  | ||||||
| from pymemcache import Client |  | ||||||
|  |  | ||||||
| from .cache import Cache |  | ||||||
| from ..utils._objects import _json_to_string, _string_to_json |  | ||||||
|  |  | ||||||
| logger: Logger = logging.getLogger(__name__) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class CacheMemcached(Cache): |  | ||||||
|     client: Client |  | ||||||
|  |  | ||||||
|     def __init__( |  | ||||||
|         self, client: Client, prefix: Optional[str] = None, default_ttl_seconds: Optional[int] = None |  | ||||||
|     ) -> None: |  | ||||||
|         self.client: Client = client |  | ||||||
|         self.prefix: str | None = prefix |  | ||||||
|         self.default_ttl_seconds: int = default_ttl_seconds if default_ttl_seconds is not None else 0 |  | ||||||
|  |  | ||||||
|         logger.info("Initialized Memcached for caching") |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def from_config(cls, engine_config: Dict[str, Any], prefix: Optional[str] = None, default_ttl_seconds: Optional[int] = None) -> "CacheMemcached": |  | ||||||
|         if "uri" not in engine_config: |  | ||||||
|             raise KeyError( |  | ||||||
|                 "Cache configuration is invalid. Please check if all keys are set (engine: memcached)" |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|         return cls(Client(engine_config["uri"], default_noreply=True), prefix=prefix, default_ttl_seconds=default_ttl_seconds) |  | ||||||
|  |  | ||||||
|     def _get_prefixed_key(self, key: str) -> str: |  | ||||||
|         return key if self.prefix is None else f"{self.prefix}_{key}" |  | ||||||
|  |  | ||||||
|     def get_json(self, key: str) -> Any | None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             result: Any | None = self.client.get(key, None) |  | ||||||
|  |  | ||||||
|             logger.debug( |  | ||||||
|                 "Got json cache key '%s'%s", |  | ||||||
|                 key, |  | ||||||
|                 "" if result is not None else " (not found)", |  | ||||||
|             ) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not get json cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         return None if result is None else _string_to_json(result) |  | ||||||
|  |  | ||||||
|     def get_string(self, key: str) -> str | None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             result: str | None = self.client.get(key, None) |  | ||||||
|  |  | ||||||
|             logger.debug( |  | ||||||
|                 "Got string cache key '%s'%s", |  | ||||||
|                 key, |  | ||||||
|                 "" if result is not None else " (not found)", |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|             return result |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not get string cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|     # TODO Implement binary deserialization |  | ||||||
|     def get_object(self, key: str) -> Any | None: |  | ||||||
|         raise NotImplementedError() |  | ||||||
|  |  | ||||||
|     def set_json(self, key: str, value: Any, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             self.client.set( |  | ||||||
|                 key, |  | ||||||
|                 _json_to_string(value), |  | ||||||
|                 expire=self.default_ttl_seconds if ttl_seconds is None else ttl_seconds, |  | ||||||
|             ) |  | ||||||
|             logger.debug("Set json cache key '%s'", key) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not set json cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|     def set_string(self, key: str, value: str, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             self.client.set( |  | ||||||
|                 key, value, expire=self.default_ttl_seconds if ttl_seconds is None else ttl_seconds |  | ||||||
|             ) |  | ||||||
|             logger.debug("Set string cache key '%s'", key) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not set string cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|     # TODO Implement binary serialization |  | ||||||
|     def set_object(self, key: str, value: Any, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         raise NotImplementedError() |  | ||||||
|  |  | ||||||
|     def delete(self, key: str) -> None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             self.client.delete(key) |  | ||||||
|             logger.debug("Deleted cache key '%s'", key) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not delete cache key '%s' due to: %s", key, exc) |  | ||||||
							
								
								
									
										110
									
								
								src/libbot/cache/classes/cache_redis.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										110
									
								
								src/libbot/cache/classes/cache_redis.py
									
									
									
									
										vendored
									
									
								
							| @@ -1,110 +0,0 @@ | |||||||
| import logging |  | ||||||
| from logging import Logger |  | ||||||
| from typing import Dict, Any, Optional |  | ||||||
|  |  | ||||||
| from redis import Redis |  | ||||||
|  |  | ||||||
| from .cache import Cache |  | ||||||
| from ..utils._objects import _json_to_string, _string_to_json |  | ||||||
|  |  | ||||||
| logger: Logger = logging.getLogger(__name__) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class CacheRedis(Cache): |  | ||||||
|     client: Redis |  | ||||||
|  |  | ||||||
|     def __init__( |  | ||||||
|         self, client: Redis, prefix: Optional[str] = None, default_ttl_seconds: Optional[int] = None |  | ||||||
|     ) -> None: |  | ||||||
|         self.client: Redis = client |  | ||||||
|         self.prefix: str | None = prefix |  | ||||||
|         self.default_ttl_seconds: int | None = default_ttl_seconds |  | ||||||
|  |  | ||||||
|         logger.info("Initialized Redis for caching") |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def from_config(cls, engine_config: Dict[str, Any], prefix: Optional[str] = None, default_ttl_seconds: Optional[int] = None) -> Any: |  | ||||||
|         if "uri" not in engine_config: |  | ||||||
|             raise KeyError( |  | ||||||
|                 "Cache configuration is invalid. Please check if all keys are set (engine: memcached)" |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|         return cls(Redis.from_url(engine_config["uri"]), prefix=prefix, default_ttl_seconds=default_ttl_seconds) |  | ||||||
|  |  | ||||||
|     def _get_prefixed_key(self, key: str) -> str: |  | ||||||
|         return key if self.prefix is None else f"{self.prefix}_{key}" |  | ||||||
|  |  | ||||||
|     def get_json(self, key: str) -> Any | None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             result: Any | None = self.client.get(key) |  | ||||||
|  |  | ||||||
|             logger.debug( |  | ||||||
|                 "Got json cache key '%s'%s", |  | ||||||
|                 key, |  | ||||||
|                 "" if result is not None else " (not found)", |  | ||||||
|             ) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not get json cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         return None if result is None else _string_to_json(result) |  | ||||||
|  |  | ||||||
|     def get_string(self, key: str) -> str | None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             result: str | None = self.client.get(key) |  | ||||||
|  |  | ||||||
|             logger.debug( |  | ||||||
|                 "Got string cache key '%s'%s", |  | ||||||
|                 key, |  | ||||||
|                 "" if result is not None else " (not found)", |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|             return result |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not get string cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|     # TODO Implement binary deserialization |  | ||||||
|     def get_object(self, key: str) -> Any | None: |  | ||||||
|         raise NotImplementedError() |  | ||||||
|  |  | ||||||
|     def set_json(self, key: str, value: Any, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             self.client.set( |  | ||||||
|                 key, |  | ||||||
|                 _json_to_string(value), |  | ||||||
|                 ex=self.default_ttl_seconds if ttl_seconds is None else ttl_seconds, |  | ||||||
|             ) |  | ||||||
|             logger.debug("Set json cache key '%s'", key) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not set json cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|     def set_string(self, key: str, value: str, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             self.client.set(key, value, ex=self.default_ttl_seconds if ttl_seconds is None else ttl_seconds) |  | ||||||
|             logger.debug("Set string cache key '%s'", key) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not set string cache key '%s' due to: %s", key, exc) |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|     # TODO Implement binary serialization |  | ||||||
|     def set_object(self, key: str, value: Any, ttl_seconds: Optional[int] = None) -> None: |  | ||||||
|         raise NotImplementedError() |  | ||||||
|  |  | ||||||
|     def delete(self, key: str) -> None: |  | ||||||
|         key = self._get_prefixed_key(key) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             self.client.delete(key) |  | ||||||
|             logger.debug("Deleted cache key '%s'", key) |  | ||||||
|         except Exception as exc: |  | ||||||
|             logger.error("Could not delete cache key '%s' due to: %s", key, exc) |  | ||||||
							
								
								
									
										1
									
								
								src/libbot/cache/manager/__init__.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								src/libbot/cache/manager/__init__.py
									
									
									
									
										vendored
									
									
								
							| @@ -1 +0,0 @@ | |||||||
| from .manager import create_cache_client |  | ||||||
							
								
								
									
										37
									
								
								src/libbot/cache/manager/manager.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								src/libbot/cache/manager/manager.py
									
									
									
									
										vendored
									
									
								
							| @@ -1,37 +0,0 @@ | |||||||
| from typing import Dict, Any, Literal, Optional |  | ||||||
|  |  | ||||||
| from ..classes import CacheMemcached, CacheRedis |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def create_cache_client( |  | ||||||
|     config: Dict[str, Any], |  | ||||||
|     engine: Literal["memcached", "redis"] | None = None, |  | ||||||
|     prefix: Optional[str] = None, |  | ||||||
|     default_ttl_seconds: Optional[int] = None, |  | ||||||
| ) -> CacheMemcached | CacheRedis: |  | ||||||
|     """Create a cache client of a provided type. |  | ||||||
|  |  | ||||||
|     Args: |  | ||||||
|         config (Dict[str, Any]): Cache client configuration. |  | ||||||
|         engine (Literal["memcached", "redis"] | None): Cache engine to use. Defaults to None. |  | ||||||
|         prefix (:obj:`str`, optional): Prefix used for each key-value pair. Defaults to None (no prefix). |  | ||||||
|         default_ttl_seconds (:obj:`int`, optional): Default TTL for values (in seconds). Defaults to None (does not expire). |  | ||||||
|  |  | ||||||
|     Returns: |  | ||||||
|         CacheMemcached | CacheRedis: Cache client. |  | ||||||
|     """ |  | ||||||
|     if engine not in ["memcached", "redis"] or engine is None: |  | ||||||
|         raise KeyError(f"Incorrect cache engine provided. Expected 'memcached' or 'redis', got '{engine}'") |  | ||||||
|  |  | ||||||
|     if "cache" not in config or engine not in config["cache"]: |  | ||||||
|         raise KeyError( |  | ||||||
|             f"Cache configuration is invalid. Please check if all keys are set (engine: '{engine}')" |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     match engine: |  | ||||||
|         case "memcached": |  | ||||||
|             return CacheMemcached.from_config(config["cache"][engine], prefix=prefix, default_ttl_seconds=default_ttl_seconds) |  | ||||||
|         case "redis": |  | ||||||
|             return CacheRedis.from_config(config["cache"][engine], prefix=prefix, default_ttl_seconds=default_ttl_seconds) |  | ||||||
|         case _: |  | ||||||
|             raise KeyError(f"Cache implementation for the engine '{engine}' is not present.") |  | ||||||
							
								
								
									
										0
									
								
								src/libbot/cache/utils/__init__.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										0
									
								
								src/libbot/cache/utils/__init__.py
									
									
									
									
										vendored
									
									
								
							
							
								
								
									
										42
									
								
								src/libbot/cache/utils/_objects.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										42
									
								
								src/libbot/cache/utils/_objects.py
									
									
									
									
										vendored
									
									
								
							| @@ -1,42 +0,0 @@ | |||||||
| import logging |  | ||||||
| from copy import deepcopy |  | ||||||
| from logging import Logger |  | ||||||
| from typing import Any |  | ||||||
|  |  | ||||||
| try: |  | ||||||
|     from ujson import dumps, loads |  | ||||||
| except ImportError: |  | ||||||
|     from json import dumps, loads |  | ||||||
|  |  | ||||||
| logger: Logger = logging.getLogger(__name__) |  | ||||||
|  |  | ||||||
| try: |  | ||||||
|     from bson import ObjectId |  | ||||||
| except ImportError: |  | ||||||
|     logger.warning( |  | ||||||
|         "Could not import bson.ObjectId. PyMongo conversions will not be supported by the cache. It's safe to ignore this message if you do not use MongoDB." |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def _json_to_string(json_object: Any) -> str: |  | ||||||
|     json_object_copy: Any = deepcopy(json_object) |  | ||||||
|  |  | ||||||
|     if isinstance(json_object_copy, dict) and "_id" in json_object_copy: |  | ||||||
|         json_object_copy["_id"] = str(json_object_copy["_id"]) |  | ||||||
|  |  | ||||||
|     return dumps(json_object_copy, ensure_ascii=False, indent=0, escape_forward_slashes=False) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def _string_to_json(json_string: str) -> Any: |  | ||||||
|     json_object: Any = loads(json_string) |  | ||||||
|  |  | ||||||
|     if "_id" in json_object: |  | ||||||
|         try: |  | ||||||
|             json_object["_id"] = ObjectId(json_object["_id"]) |  | ||||||
|         except NameError: |  | ||||||
|             logger.debug( |  | ||||||
|                 "Tried to convert attribute '_id' with value '%s' but bson.ObjectId is not present, skipping the conversion.", |  | ||||||
|                 json_object["_id"], |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|     return json_object |  | ||||||
| @@ -18,16 +18,16 @@ def _( | |||||||
|     locale: str | None = "en", |     locale: str | None = "en", | ||||||
|     locales_root: str | Path = Path("locale"), |     locales_root: str | Path = Path("locale"), | ||||||
| ) -> Any: | ) -> Any: | ||||||
|     """Get value of locale string. |     """Get value of locale string | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): The last key of the locale's keys path. |         * key (`str`): The last key of the locale's keys path. | ||||||
|         *args (str): Path to key like: `dict[args][key]`. |         * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|         locale (str | None): Locale to looked up in. Defaults to "en". |         * locale (`str | None`): Locale to looked up in. Defaults to `"en"`. | ||||||
|         locales_root (str | Path, optional): Folder where locales are located. Defaults to Path("locale"). |         * locales_root (`str | Path`, *optional*): Folder where locales are located. Defaults to `Path("locale")`. | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Any: Value of provided locale key. Is usually `str`, `Dict[str, Any]` or `List[Any]`. |         * `Any`: Value of provided locale key. Is usually `str`, `Dict[str, Any]` or `List[Any]` | ||||||
|     """ |     """ | ||||||
|     if locale is None: |     if locale is None: | ||||||
|         locale: str = config_get("locale") |         locale: str = config_get("locale") | ||||||
| @@ -58,16 +58,16 @@ async def _( | |||||||
|     locale: str | None = "en", |     locale: str | None = "en", | ||||||
|     locales_root: str | Path = Path("locale"), |     locales_root: str | Path = Path("locale"), | ||||||
| ) -> Any: | ) -> Any: | ||||||
|     """Get value of locale string. |     """Get value of locale string | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): The last key of the locale's keys path. |         * key (`str`): The last key of the locale's keys path. | ||||||
|         *args (str): Path to key like: `dict[args][key]`. |         * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|         locale (str | None): Locale to looked up in. Defaults to "en". |         * locale (`str | None`): Locale to looked up in. Defaults to `"en"`. | ||||||
|         locales_root (str | Path, optional): Folder where locales are located. Defaults to Path("locale"). |         * locales_root (`str | Path`, *optional*): Folder where locales are located. Defaults to `Path("locale")`. | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Any: Value of provided locale key. Is usually `str`, `Dict[str, Any]` or `List[Any]`. |         * `Any`: Value of provided locale key. Is usually `str`, `Dict[str, Any]` or `List[Any]` | ||||||
|     """ |     """ | ||||||
|     locale: str = config_get("locale") if locale is None else locale |     locale: str = config_get("locale") if locale is None else locale | ||||||
|  |  | ||||||
| @@ -94,15 +94,15 @@ async def _( | |||||||
|  |  | ||||||
| @asyncable | @asyncable | ||||||
| def in_all_locales(key: str, *args: str, locales_root: str | Path = Path("locale")) -> List[Any]: | def in_all_locales(key: str, *args: str, locales_root: str | Path = Path("locale")) -> List[Any]: | ||||||
|     """Get value of the provided key and path in all available locales. |     """Get value of the provided key and path in all available locales | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): The last key of the locale's keys path. |         * key (`str`): The last key of the locale's keys path. | ||||||
|         *args (str): Path to key like: `dict[args][key]`. |         * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|         locales_root (str | Path, optional): Folder where locales are located. Defaults to `Path("locale")`. |         * locales_root (`str | Path`, *optional*): Folder where locales are located. Defaults to `Path("locale")`. | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         List[Any]: List of values in all locales. |         * `List[Any]`: List of values in all locales | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     output: List[Any] = [] |     output: List[Any] = [] | ||||||
| @@ -128,15 +128,15 @@ def in_all_locales(key: str, *args: str, locales_root: str | Path = Path("locale | |||||||
|  |  | ||||||
| @in_all_locales.asynchronous | @in_all_locales.asynchronous | ||||||
| async def in_all_locales(key: str, *args: str, locales_root: str | Path = Path("locale")) -> List[Any]: | async def in_all_locales(key: str, *args: str, locales_root: str | Path = Path("locale")) -> List[Any]: | ||||||
|     """Get value of the provided key and path in all available locales. |     """Get value of the provided key and path in all available locales | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): The last key of the locale's keys path. |         * key (`str`): The last key of the locale's keys path. | ||||||
|         *args (str): Path to key like: `dict[args][key]`. |         * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|         locales_root (str | Path, optional): Folder where locales are located. Defaults to Path("locale"). |         * locales_root (`str | Path`, *optional*): Folder where locales are located. Defaults to `Path("locale")`. | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         List[Any]: List of values in all locales. |         * `List[Any]`: List of values in all locales | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     output: List[Any] = [] |     output: List[Any] = [] | ||||||
| @@ -164,15 +164,15 @@ async def in_all_locales(key: str, *args: str, locales_root: str | Path = Path(" | |||||||
| def in_every_locale( | def in_every_locale( | ||||||
|     key: str, *args: str, locales_root: str | Path = Path("locale") |     key: str, *args: str, locales_root: str | Path = Path("locale") | ||||||
| ) -> Dict[str, Any]: | ) -> Dict[str, Any]: | ||||||
|     """Get value of the provided key and path in every available locale with locale tag. |     """Get value of the provided key and path in every available locale with locale tag | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): The last key of the locale's keys path. |         * key (`str`): The last key of the locale's keys path. | ||||||
|         *args (str): Path to key like: `dict[args][key]`. |         * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|         locales_root (str | Path, optional): Folder where locales are located. Defaults to Path("locale"). |         * locales_root (`str | Path`, *optional*): Folder where locales are located. Defaults to `Path("locale")`. | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Dict[str, Any]: Locale is a key, and it's value from locale file is a value. |         * `Dict[str, Any]`: Locale is a key, and it's value from locale file is a value | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     output: Dict[str, Any] = {} |     output: Dict[str, Any] = {} | ||||||
| @@ -200,15 +200,15 @@ def in_every_locale( | |||||||
| async def in_every_locale( | async def in_every_locale( | ||||||
|     key: str, *args: str, locales_root: str | Path = Path("locale") |     key: str, *args: str, locales_root: str | Path = Path("locale") | ||||||
| ) -> Dict[str, Any]: | ) -> Dict[str, Any]: | ||||||
|     """Get value of the provided key and path in every available locale with locale tag. |     """Get value of the provided key and path in every available locale with locale tag | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): The last key of the locale's keys path. |         * key (`str`): The last key of the locale's keys path. | ||||||
|         *args (str): Path to key like: `dict[args][key]`. |         * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|         locales_root (str | Path, optional): Folder where locales are located. Defaults to Path("locale"). |         * locales_root (`str | Path`, *optional*): Folder where locales are located. Defaults to `Path("locale")`. | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Dict[str, Any]: Locale is a key, and it's value from locale file is a value. |         * `Dict[str, Any]`: Locale is a key, and it's value from locale file is a value | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     output: Dict[str, Any] = {} |     output: Dict[str, Any] = {} | ||||||
|   | |||||||
| @@ -14,11 +14,6 @@ class BotLocale: | |||||||
|         default_locale: str | None = "en", |         default_locale: str | None = "en", | ||||||
|         locales_root: str | Path = Path("locale"), |         locales_root: str | Path = Path("locale"), | ||||||
|     ) -> None: |     ) -> None: | ||||||
|         """ |  | ||||||
|         Args: |  | ||||||
|             default_locale (str | None, optional): Default locale. Defaults to "en". |  | ||||||
|             locales_root (str | Path, optional): Path to a directory with locale files. Defaults to Path("locale"). |  | ||||||
|         """ |  | ||||||
|         if isinstance(locales_root, str): |         if isinstance(locales_root, str): | ||||||
|             locales_root = Path(locales_root) |             locales_root = Path(locales_root) | ||||||
|         elif not isinstance(locales_root, Path): |         elif not isinstance(locales_root, Path): | ||||||
| @@ -35,15 +30,15 @@ class BotLocale: | |||||||
|             self.locales[locale] = json_read(Path(f"{locales_root}/{locale}.json")) |             self.locales[locale] = json_read(Path(f"{locales_root}/{locale}.json")) | ||||||
|  |  | ||||||
|     def _(self, key: str, *args: str, locale: str | None = None) -> Any: |     def _(self, key: str, *args: str, locale: str | None = None) -> Any: | ||||||
|         """Get value of locale string. |         """Get value of locale string | ||||||
|  |  | ||||||
|         Args: |         ### Args: | ||||||
|             key (str): The last key of the locale's keys path. |             * key (`str`): The last key of the locale's keys path | ||||||
|             *args (str): Path to key like: `dict[args][key]`. |             * *args (`str`): Path to key like: `dict[args][key]` | ||||||
|             locale (str | None, optional): Locale to looked up in. Defaults to config's `"locale"` value. |             * locale (`str | None`, *optional*): Locale to looked up in. Defaults to config's `"locale"` value | ||||||
|  |  | ||||||
|         Returns: |         ### Returns: | ||||||
|             Any: Value of provided locale key. Is usually `str`, `Dict[str, Any]` or `List[Any]`. |             * `Any`: Value of provided locale key. Is usually `str`, `Dict[str, Any]` or `List[Any]` | ||||||
|         """ |         """ | ||||||
|         if locale is None: |         if locale is None: | ||||||
|             locale: str = self.default |             locale: str = self.default | ||||||
| @@ -69,14 +64,14 @@ class BotLocale: | |||||||
|             return f'⚠️ Locale in config is invalid: could not get "{key}" in {args} from locale "{locale}"' |             return f'⚠️ Locale in config is invalid: could not get "{key}" in {args} from locale "{locale}"' | ||||||
|  |  | ||||||
|     def in_all_locales(self, key: str, *args: str) -> List[Any]: |     def in_all_locales(self, key: str, *args: str) -> List[Any]: | ||||||
|         """Get value of the provided key and path in all available locales. |         """Get value of the provided key and path in all available locales | ||||||
|  |  | ||||||
|         Args: |         ### Args: | ||||||
|             key (str): The last key of the locale's keys path. |             * key (`str`): The last key of the locale's keys path. | ||||||
|             *args (str): Path to key like: `dict[args][key]`. |             * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|  |  | ||||||
|         Returns: |         ### Returns: | ||||||
|             List[Any]: List of values in all locales. |             * `List[Any]`: List of values in all locales | ||||||
|         """ |         """ | ||||||
|         output: List[Any] = [] |         output: List[Any] = [] | ||||||
|  |  | ||||||
| @@ -99,14 +94,14 @@ class BotLocale: | |||||||
|         return output |         return output | ||||||
|  |  | ||||||
|     def in_every_locale(self, key: str, *args: str) -> Dict[str, Any]: |     def in_every_locale(self, key: str, *args: str) -> Dict[str, Any]: | ||||||
|         """Get value of the provided key and path in every available locale with locale tag. |         """Get value of the provided key and path in every available locale with locale tag | ||||||
|  |  | ||||||
|         Args: |         ### Args: | ||||||
|             key (str): The last key of the locale's keys path. |             * key (`str`): The last key of the locale's keys path. | ||||||
|             *args (str): Path to key like: `dict[args][key]`. |             * *args (`str`): Path to key like: `dict[args][key]`. | ||||||
|  |  | ||||||
|         Returns: |         ### Returns: | ||||||
|             Dict[str, Any]: Locale is a key, and it's value from locale file is a value. |             * `Dict[str, Any]`: Locale is a key, and it's value from locale file is a value | ||||||
|         """ |         """ | ||||||
|         output: Dict[str, Any] = {} |         output: Dict[str, Any] = {} | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								src/libbot/matrix/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/libbot/matrix/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | |||||||
|  | # This file is left empty on purpose | ||||||
|  | # Adding imports here will cause import errors when libbot[matrix] is not installed | ||||||
							
								
								
									
										1
									
								
								src/libbot/matrix/classes/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/libbot/matrix/classes/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | from .bot import MatrixBot | ||||||
							
								
								
									
										80
									
								
								src/libbot/matrix/classes/bot.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								src/libbot/matrix/classes/bot.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | |||||||
|  | import logging | ||||||
|  | from logging import Logger | ||||||
|  | from pathlib import Path | ||||||
|  | from typing import Dict, Any | ||||||
|  |  | ||||||
|  | from typing_extensions import override | ||||||
|  |  | ||||||
|  | from ... import __version__ as __libbot_version__ | ||||||
|  | from ...i18n.classes import BotLocale | ||||||
|  | from ...utils import json_read | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     from apscheduler.schedulers.asyncio import AsyncIOScheduler | ||||||
|  |     from apscheduler.schedulers.background import BackgroundScheduler | ||||||
|  |     from simplematrixbotlib import Bot, Creds, Config | ||||||
|  | except ImportError as exc: | ||||||
|  |     raise ImportError("You need to install libbot[matrix] in order to use this class.") from exc | ||||||
|  |  | ||||||
|  | logger: Logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MatrixBot(Bot): | ||||||
|  |     @override | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         config: Dict[str, Any] | None = None, | ||||||
|  |         config_path: str | Path = Path("config.json"), | ||||||
|  |         locales_root: str | Path | None = None, | ||||||
|  |         scheduler: AsyncIOScheduler | BackgroundScheduler | None = None, | ||||||
|  |         smbl_creds: Creds = None, | ||||||
|  |         smbl_config: Config = None, | ||||||
|  |     ): | ||||||
|  |         self.bot_config: Dict[str, Any] = config if config is not None else json_read(config_path) | ||||||
|  |  | ||||||
|  |         super().__init__( | ||||||
|  |             creds=( | ||||||
|  |                 smbl_creds | ||||||
|  |                 if smbl_creds is not None | ||||||
|  |                 else Creds( | ||||||
|  |                     homeserver=self.bot_config["bot"]["homeserver"], | ||||||
|  |                     username=self.bot_config["bot"]["username"], | ||||||
|  |                     password=self.bot_config["bot"]["password"], | ||||||
|  |                     device_name=( | ||||||
|  |                         f"LibBotUniversal v{__libbot_version__}" | ||||||
|  |                         if "device_name" not in self.bot_config["bot"] | ||||||
|  |                         else self.bot_config["bot"]["device_name"] | ||||||
|  |                     ), | ||||||
|  |                 ) | ||||||
|  |             ), | ||||||
|  |             config=smbl_config, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         self.bot_prefix: str = ( | ||||||
|  |             "!" if "prefix" not in self.bot_config["bot"] else self.bot_config["bot"]["prefix"] | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         self.bot_locale: BotLocale = BotLocale( | ||||||
|  |             default_locale=self.bot_config["locale"], | ||||||
|  |             locales_root=(Path("locale") if locales_root is None else locales_root), | ||||||
|  |         ) | ||||||
|  |         self.default_locale: str = self.bot_locale.default | ||||||
|  |         self.locales: Dict[str, Any] = self.bot_locale.locales | ||||||
|  |  | ||||||
|  |         self._ = self.bot_locale._ | ||||||
|  |         self.in_all_locales = self.bot_locale.in_all_locales | ||||||
|  |         self.in_every_locale = self.bot_locale.in_every_locale | ||||||
|  |  | ||||||
|  |         self.scheduler: AsyncIOScheduler | BackgroundScheduler | None = scheduler | ||||||
|  |  | ||||||
|  |     @override | ||||||
|  |     def run( | ||||||
|  |         self, scheduler_start: bool = True, scheduler_shutdown: bool = True, scheduler_wait: bool = True | ||||||
|  |     ) -> None: | ||||||
|  |         if self.scheduler is not None and scheduler_start: | ||||||
|  |             self.scheduler.start() | ||||||
|  |  | ||||||
|  |         super().run() | ||||||
|  |  | ||||||
|  |         if self.scheduler is not None and scheduler_shutdown: | ||||||
|  |             self.scheduler.shutdown(scheduler_wait) | ||||||
| @@ -1,13 +1,9 @@ | |||||||
| import logging | import logging | ||||||
| from logging import Logger |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from typing import Any, Dict | from typing import Any, Dict | ||||||
|  |  | ||||||
| from typing_extensions import override | from typing_extensions import override | ||||||
|  |  | ||||||
| from ...i18n.classes import BotLocale |  | ||||||
| from ...utils import json_read |  | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     from apscheduler.schedulers.asyncio import AsyncIOScheduler |     from apscheduler.schedulers.asyncio import AsyncIOScheduler | ||||||
|     from apscheduler.schedulers.background import BackgroundScheduler |     from apscheduler.schedulers.background import BackgroundScheduler | ||||||
| @@ -15,21 +11,32 @@ try: | |||||||
| except ImportError as exc: | except ImportError as exc: | ||||||
|     raise ImportError("You need to install libbot[pycord] in order to use this class.") from exc |     raise ImportError("You need to install libbot[pycord] in order to use this class.") from exc | ||||||
|  |  | ||||||
| logger: Logger = logging.getLogger(__name__) | try: | ||||||
|  |     from ujson import loads | ||||||
|  | except ImportError: | ||||||
|  |     from json import loads | ||||||
|  |  | ||||||
|  | from ...i18n.classes import BotLocale | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PycordBot(Bot): | class PycordBot(Bot): | ||||||
|     @override |     @override | ||||||
|     def __init__( |     def __init__( | ||||||
|             self, |         self, | ||||||
|             *args, |         *args, | ||||||
|             config: Dict[str, Any] | None = None, |         config: Dict[str, Any] | None = None, | ||||||
|             config_path: str | Path = Path("config.json"), |         config_path: str | Path = Path("config.json"), | ||||||
|             locales_root: str | Path | None = None, |         locales_root: str | Path | None = None, | ||||||
|             scheduler: AsyncIOScheduler | BackgroundScheduler | None = None, |         scheduler: AsyncIOScheduler | BackgroundScheduler | None = None, | ||||||
|             **kwargs, |         **kwargs, | ||||||
|     ): |     ): | ||||||
|         self.config: Dict[str, Any] = config if config is not None else json_read(config_path) |         if config is None: | ||||||
|  |             with open(config_path, "r", encoding="utf-8") as f: | ||||||
|  |                 self.config: dict = loads(f.read()) | ||||||
|  |         else: | ||||||
|  |             self.config = config | ||||||
|  |  | ||||||
|         super().__init__( |         super().__init__( | ||||||
|             debug_guilds=(self.config["bot"]["debug_guilds"] if self.config["debug"] else None), |             debug_guilds=(self.config["bot"]["debug_guilds"] if self.config["debug"] else None), | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ import asyncio | |||||||
| import logging | import logging | ||||||
| import sys | import sys | ||||||
| from datetime import datetime, timedelta | from datetime import datetime, timedelta | ||||||
| from logging import Logger |  | ||||||
| from os import cpu_count, getpid | from os import cpu_count, getpid | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from time import time | from time import time | ||||||
| @@ -10,12 +9,6 @@ from typing import Any, Dict, List | |||||||
|  |  | ||||||
| from typing_extensions import override | from typing_extensions import override | ||||||
|  |  | ||||||
| from .command import PyroCommand |  | ||||||
| from .commandset import CommandSet |  | ||||||
| from ...i18n import _ |  | ||||||
| from ...i18n.classes import BotLocale |  | ||||||
| from ...utils import json_read |  | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     import pyrogram |     import pyrogram | ||||||
|     from apscheduler.schedulers.asyncio import AsyncIOScheduler |     from apscheduler.schedulers.asyncio import AsyncIOScheduler | ||||||
| @@ -42,33 +35,42 @@ try: | |||||||
| except ImportError: | except ImportError: | ||||||
|     from json import dumps, loads |     from json import dumps, loads | ||||||
|  |  | ||||||
| logger: Logger = logging.getLogger(__name__) | from ...i18n.classes import BotLocale | ||||||
|  | from ...i18n import _ | ||||||
|  | from .command import PyroCommand | ||||||
|  | from .commandset import CommandSet | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PyroClient(Client): | class PyroClient(Client): | ||||||
|     @override |     @override | ||||||
|     def __init__( |     def __init__( | ||||||
|             self, |         self, | ||||||
|             name: str = "bot_client", |         name: str = "bot_client", | ||||||
|             owner: int | None = None, |         owner: int | None = None, | ||||||
|             config: Dict[str, Any] | None = None, |         config: Dict[str, Any] | None = None, | ||||||
|             config_path: str | Path = Path("config.json"), |         config_path: str | Path = Path("config.json"), | ||||||
|             api_id: int | None = None, |         api_id: int | None = None, | ||||||
|             api_hash: str | None = None, |         api_hash: str | None = None, | ||||||
|             bot_token: str | None = None, |         bot_token: str | None = None, | ||||||
|             workers: int = min(32, cpu_count() + 4), |         workers: int = min(32, cpu_count() + 4), | ||||||
|             locales_root: str | Path | None = None, |         locales_root: str | Path | None = None, | ||||||
|             plugins_root: str = "plugins", |         plugins_root: str = "plugins", | ||||||
|             plugins_exclude: List[str] | None = None, |         plugins_exclude: List[str] | None = None, | ||||||
|             sleep_threshold: int = 120, |         sleep_threshold: int = 120, | ||||||
|             max_concurrent_transmissions: int = 1, |         max_concurrent_transmissions: int = 1, | ||||||
|             commands_source: Dict[str, dict] | None = None, |         commands_source: Dict[str, dict] | None = None, | ||||||
|             scoped_commands: bool | None = None, |         scoped_commands: bool | None = None, | ||||||
|             i18n_bot_info: bool = False, |         i18n_bot_info: bool = False, | ||||||
|             scheduler: AsyncIOScheduler | BackgroundScheduler | None = None, |         scheduler: AsyncIOScheduler | BackgroundScheduler | None = None, | ||||||
|             **kwargs, |         **kwargs, | ||||||
|     ): |     ): | ||||||
|         self.config: Dict[str, Any] = config if config is not None else json_read(config_path) |         if config is None: | ||||||
|  |             with open(config_path, "r", encoding="utf-8") as f: | ||||||
|  |                 self.config: dict = loads(f.read()) | ||||||
|  |         else: | ||||||
|  |             self.config = config | ||||||
|  |  | ||||||
|         super().__init__( |         super().__init__( | ||||||
|             name=name, |             name=name, | ||||||
| @@ -209,7 +211,7 @@ class PyroClient(Client): | |||||||
|  |  | ||||||
|     @override |     @override | ||||||
|     async def stop( |     async def stop( | ||||||
|             self, exit_completely: bool = True, scheduler_shutdown: bool = True, scheduler_wait: bool = True |         self, exit_completely: bool = True, scheduler_shutdown: bool = True, scheduler_wait: bool = True | ||||||
|     ) -> None: |     ) -> None: | ||||||
|         try: |         try: | ||||||
|             await self.send_message( |             await self.send_message( | ||||||
| @@ -308,7 +310,7 @@ class PyroClient(Client): | |||||||
|             # in it, if there are any. Then adds them to self.commands |             # in it, if there are any. Then adds them to self.commands | ||||||
|             for handler in self.dispatcher.groups[0]: |             for handler in self.dispatcher.groups[0]: | ||||||
|                 if isinstance(handler, MessageHandler) and ( |                 if isinstance(handler, MessageHandler) and ( | ||||||
|                         hasattr(handler.filters, "base") or hasattr(handler.filters, "other") |                     hasattr(handler.filters, "base") or hasattr(handler.filters, "other") | ||||||
|                 ): |                 ): | ||||||
|                     for entry in [handler.filters.base, handler.filters.other]: |                     for entry in [handler.filters.base, handler.filters.other]: | ||||||
|                         if hasattr(entry, "commands"): |                         if hasattr(entry, "commands"): | ||||||
| @@ -319,8 +321,8 @@ class PyroClient(Client): | |||||||
|         return command_sets |         return command_sets | ||||||
|  |  | ||||||
|     def add_command( |     def add_command( | ||||||
|             self, |         self, | ||||||
|             command: str, |         command: str, | ||||||
|     ) -> None: |     ) -> None: | ||||||
|         """Add command to the bot's internal commands list |         """Add command to the bot's internal commands list | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,24 +15,34 @@ DEFAULT_CONFIG_LOCATION: str = "config.json" | |||||||
|  |  | ||||||
| @asyncable | @asyncable | ||||||
| def config_get(key: str, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION) -> Any: | def config_get(key: str, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION) -> Any: | ||||||
|     """Get a value of the config key by its path provided. |     """Get a value of the config key by its path provided | ||||||
|     For example, `foo.bar.key` has a path of `"foo", "bar"` and the key `"key"`. |     For example, `foo.bar.key` has a path of `"foo", "bar"` and the key `"key"` | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): Key that contains the value |         * key (`str`): Key that contains the value | ||||||
|         *path (str): Path to the key that contains the value (pass *[] or don't pass anything at all to get on the top/root level) |         * *path (`str`): Path to the key that contains the value | ||||||
|         config_file (str | Path, optional): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` |         * config_file (`str | Path`, *optional*): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Any: Key's value |         * `Any`: Key's value | ||||||
|  |  | ||||||
|     Example: |     ### Example: | ||||||
|         Get the "salary" of "Pete" from this JSON structure: `{"users": {"Pete": {"salary": 10.0}}}` |     Get the "salary" of "Pete" from this JSON structure: | ||||||
|  |     ```json | ||||||
|  |     { | ||||||
|  |         "users": { | ||||||
|  |             "Pete": { | ||||||
|  |                 "salary": 10.0 | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     ``` | ||||||
|  |  | ||||||
|         This can be easily done with the following code: |     This can be easily done with the following code: | ||||||
|  |     ```python | ||||||
|         >>> import libbot |     import libbot | ||||||
|         salary: float = libbot.sync.config_get("salary", "users", "Pete") |     salary = libbot.sync.config_get("salary", "users", "Pete") | ||||||
|  |     ``` | ||||||
|     """ |     """ | ||||||
|     this_key: Dict[str, Any] = json_read(config_file) |     this_key: Dict[str, Any] = json_read(config_file) | ||||||
|  |  | ||||||
| @@ -44,24 +54,34 @@ def config_get(key: str, *path: str, config_file: str | Path = DEFAULT_CONFIG_LO | |||||||
|  |  | ||||||
| @config_get.asynchronous | @config_get.asynchronous | ||||||
| async def config_get(key: str, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION) -> Any: | async def config_get(key: str, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION) -> Any: | ||||||
|     """Get a value of the config key by its path provided. |     """Get a value of the config key by its path provided | ||||||
|     For example, `foo.bar.key` has a path of `"foo", "bar"` and the key `"key"`. |     For example, `foo.bar.key` has a path of `"foo", "bar"` and the key `"key"` | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): Key that contains the value |         * key (`str`): Key that contains the value | ||||||
|         *path (str): Path to the key that contains the value (pass *[] or don't pass anything at all to get on the top/root level) |         * *path (`str`): Path to the key that contains the value | ||||||
|         config_file (str | Path, optional): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` |         * config_file (`str | Path`, *optional*): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Any: Key's value |         * `Any`: Key's value | ||||||
|  |  | ||||||
|     Example: |     ### Example: | ||||||
|         Get the "salary" of "Pete" from this JSON structure: `{"users": {"Pete": {"salary": 10.0}}}` |     Get the "salary" of "Pete" from this JSON structure: | ||||||
|  |     ```json | ||||||
|  |     { | ||||||
|  |         "users": { | ||||||
|  |             "Pete": { | ||||||
|  |                 "salary": 10.0 | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     ``` | ||||||
|  |  | ||||||
|         This can be easily done with the following code: |     This can be easily done with the following code: | ||||||
|  |     ```python | ||||||
|         >>> import libbot |     import libbot | ||||||
|         salary: float = libbot.sync.config_get("salary", "users", "Pete") |     salary = await libbot.config_get("salary", "users", "Pete") | ||||||
|  |     ``` | ||||||
|     """ |     """ | ||||||
|     this_key: Dict[str, Any] = await json_read(config_file) |     this_key: Dict[str, Any] = await json_read(config_file) | ||||||
|  |  | ||||||
| @@ -73,16 +93,16 @@ async def config_get(key: str, *path: str, config_file: str | Path = DEFAULT_CON | |||||||
|  |  | ||||||
| @asyncable | @asyncable | ||||||
| def config_set(key: str, value: Any, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION) -> None: | def config_set(key: str, value: Any, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION) -> None: | ||||||
|     """Set config's key by its path to the value. |     """Set config's key by its path to the value | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): Key that leads to the value. |         * key (`str`): Key that leads to the value | ||||||
|         value (Any): Any JSON-serializable data. |         * value (`Any`): Any JSON serializable data | ||||||
|         *path (str): Path to the key of the target (pass *[] or don't pass anything at all to set on the top/root level). |         * *path (`str`): Path to the key of the target | ||||||
|         config_file (str | Path, optional): Path-like object or path as a string of a location of the config file. Defaults to "config.json". |         * config_file (`str | Path`, *optional*): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` | ||||||
|  |  | ||||||
|     Raises: |     ### Raises: | ||||||
|         KeyError: Key was not found under the provided path. |         * `KeyError`: Key is not found under path provided | ||||||
|     """ |     """ | ||||||
|     json_write(nested_set(json_read(config_file), value, *(*path, key)), config_file) |     json_write(nested_set(json_read(config_file), value, *(*path, key)), config_file) | ||||||
|  |  | ||||||
| @@ -91,16 +111,16 @@ def config_set(key: str, value: Any, *path: str, config_file: str | Path = DEFAU | |||||||
| async def config_set( | async def config_set( | ||||||
|     key: str, value: Any, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION |     key: str, value: Any, *path: str, config_file: str | Path = DEFAULT_CONFIG_LOCATION | ||||||
| ) -> None: | ) -> None: | ||||||
|     """Set config's key by its path to the value. |     """Set config's key by its path to the value | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): Key that leads to the value. |         * key (`str`): Key that leads to the value | ||||||
|         value (Any): Any JSON-serializable data. |         * value (`Any`): Any JSON serializable data | ||||||
|         *path (str): Path to the key of the target (pass *[] or don't pass anything at all to set on the top/root level). |         * *path (`str`): Path to the key of the target | ||||||
|         config_file (str | Path, optional): Path-like object or path as a string of a location of the config file. Defaults to "config.json". |         * config_file (`str | Path`, *optional*): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` | ||||||
|  |  | ||||||
|     Raises: |     ### Raises: | ||||||
|         KeyError: Key was not found under the provided path. |         * `KeyError`: Key is not found under path provided | ||||||
|     """ |     """ | ||||||
|     await json_write(nested_set(await json_read(config_file), value, *(*path, key)), config_file) |     await json_write(nested_set(await json_read(config_file), value, *(*path, key)), config_file) | ||||||
|  |  | ||||||
| @@ -112,16 +132,16 @@ def config_delete( | |||||||
|     missing_ok: bool = False, |     missing_ok: bool = False, | ||||||
|     config_file: str | Path = DEFAULT_CONFIG_LOCATION, |     config_file: str | Path = DEFAULT_CONFIG_LOCATION, | ||||||
| ) -> None: | ) -> None: | ||||||
|     """Delete config's key by its path. |     """Set config's key by its path | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): Key to delete. |         * key (`str`): Key to delete | ||||||
|         *path (str): Path to the key of the target (pass *[] or don't pass anything at all to delete on the top/root level) |         * *path (`str`): Path to the key of the target | ||||||
|         missing_ok (bool): Do not raise an exception if the key is missing. Defaults to False. |         * missing_ok (`bool`): Do not raise an exception if the key is missing. Defaults to `False` | ||||||
|         config_file (str | Path, optional): Path-like object or path as a string of a location of the config file. Defaults to "config.json". |         * config_file (`str | Path`, *optional*): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` | ||||||
|  |  | ||||||
|     Raises: |     ### Raises: | ||||||
|         KeyError: Key is not found under path provided and `missing_ok` is False. |         * `KeyError`: Key is not found under path provided and `missing_ok` is `False` | ||||||
|     """ |     """ | ||||||
|     config_data: Dict[str, Any] = json_read(config_file) |     config_data: Dict[str, Any] = json_read(config_file) | ||||||
|  |  | ||||||
| @@ -141,16 +161,16 @@ async def config_delete( | |||||||
|     missing_ok: bool = False, |     missing_ok: bool = False, | ||||||
|     config_file: str | Path = DEFAULT_CONFIG_LOCATION, |     config_file: str | Path = DEFAULT_CONFIG_LOCATION, | ||||||
| ) -> None: | ) -> None: | ||||||
|     """Delete config's key by its path. |     """Set config's key by its path | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         key (str): Key to delete. |         * key (`str`): Key to delete | ||||||
|         *path (str): Path to the key of the target (pass *[] or don't pass anything at all to delete on the top/root level) |         * *path (`str`): Path to the key of the target | ||||||
|         missing_ok (bool): Do not raise an exception if the key is missing. Defaults to False. |         * missing_ok (`bool`): Do not raise an exception if the key is missing. Defaults to `False` | ||||||
|         config_file (str | Path, optional): Path-like object or path as a string of a location of the config file. Defaults to "config.json". |         * config_file (`str | Path`, *optional*): Path-like object or path as a string of a location of the config file. Defaults to `"config.json"` | ||||||
|  |  | ||||||
|     Raises: |     ### Raises: | ||||||
|         KeyError: Key is not found under path provided and `missing_ok` is False. |         * `KeyError`: Key is not found under path provided and `missing_ok` is `False` | ||||||
|     """ |     """ | ||||||
|     config_data: Dict[str, Any] = await json_read(config_file) |     config_data: Dict[str, Any] = await json_read(config_file) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,13 +14,13 @@ except ImportError: | |||||||
|  |  | ||||||
| @asyncable | @asyncable | ||||||
| def json_read(path: str | Path) -> Any: | def json_read(path: str | Path) -> Any: | ||||||
|     """Read contents of a JSON file and return it. |     """Read contents of a JSON file | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         path (str | Path): Path-like object or path to the file as a string. |         * path (`str | Path`): Path-like object or path as a string | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Any: File contents. |         * `Any`: File contents | ||||||
|     """ |     """ | ||||||
|     with open(str(path), mode="r", encoding="utf-8") as f: |     with open(str(path), mode="r", encoding="utf-8") as f: | ||||||
|         data = f.read() |         data = f.read() | ||||||
| @@ -30,13 +30,13 @@ def json_read(path: str | Path) -> Any: | |||||||
|  |  | ||||||
| @json_read.asynchronous | @json_read.asynchronous | ||||||
| async def json_read(path: str | Path) -> Any: | async def json_read(path: str | Path) -> Any: | ||||||
|     """Read contents of a JSON file and return it. |     """Read contents of a JSON file | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         path (str | Path): Path-like object or path to the file as a string. |         * path (`str | Path`): Path-like object or path as a string | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Any: File contents. |         * `Any`: File contents | ||||||
|     """ |     """ | ||||||
|     async with aiofiles.open(str(path), mode="r", encoding="utf-8") as f: |     async with aiofiles.open(str(path), mode="r", encoding="utf-8") as f: | ||||||
|         data = await f.read() |         data = await f.read() | ||||||
| @@ -46,11 +46,11 @@ async def json_read(path: str | Path) -> Any: | |||||||
|  |  | ||||||
| @asyncable | @asyncable | ||||||
| def json_write(data: Any, path: str | Path) -> None: | def json_write(data: Any, path: str | Path) -> None: | ||||||
|     """Write contents to a JSON file. |     """Write contents to a JSON file | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         data (Any): Contents to write. Must be a JSON-serializable object. |         * data (`Any`): Contents to write. Must be a JSON serializable | ||||||
|         path (str | Path): Path-like object or path to the file as a string. |         * path (`str | Path`): Path-like object or path as a string of a destination | ||||||
|     """ |     """ | ||||||
|     with open(str(path), mode="w", encoding="utf-8") as f: |     with open(str(path), mode="w", encoding="utf-8") as f: | ||||||
|         f.write( |         f.write( | ||||||
| @@ -62,11 +62,11 @@ def json_write(data: Any, path: str | Path) -> None: | |||||||
|  |  | ||||||
| @json_write.asynchronous | @json_write.asynchronous | ||||||
| async def json_write(data: Any, path: str | Path) -> None: | async def json_write(data: Any, path: str | Path) -> None: | ||||||
|     """Write contents to a JSON file. |     """Write contents to a JSON file | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         data (Any): Contents to write. Must be a JSON-serializable object. |         * data (`Any`): Contents to write. Must be a JSON serializable | ||||||
|         path (str | Path): Path-like object or path to the file as a string. |         * path (`str | Path`): Path-like object or path as a string of a destination | ||||||
|     """ |     """ | ||||||
|     async with aiofiles.open(str(path), mode="w", encoding="utf-8") as f: |     async with aiofiles.open(str(path), mode="w", encoding="utf-8") as f: | ||||||
|         await f.write( |         await f.write( | ||||||
|   | |||||||
| @@ -3,15 +3,15 @@ from typing import Any, Dict | |||||||
| from typing import Callable | from typing import Callable | ||||||
|  |  | ||||||
|  |  | ||||||
| def supports_argument(func: Callable[..., Any], arg_name: str) -> bool: | def supports_argument(func: Callable, arg_name: str) -> bool: | ||||||
|     """Check whether a function has a specific argument. |     """Check whether a function has a specific argument | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         func (Callable[..., Any]): Function to be inspected. |         * func (`Callable`): Function to be inspected | ||||||
|         arg_name (str): Argument to be checked. |         * arg_name (`str`): Argument to be checked | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         bool: True if argument is supported and False if not. |         * `bool`: `True` if argument is supported and `False` if not | ||||||
|     """ |     """ | ||||||
|     if hasattr(func, "__code__"): |     if hasattr(func, "__code__"): | ||||||
|         return arg_name in inspect.signature(func).parameters |         return arg_name in inspect.signature(func).parameters | ||||||
| @@ -24,63 +24,61 @@ def supports_argument(func: Callable[..., Any], arg_name: str) -> bool: | |||||||
|     return False |     return False | ||||||
|  |  | ||||||
|  |  | ||||||
| def nested_set( | def nested_set(target: dict, value: Any, *path: str, create_missing=True) -> Dict[str, Any]: | ||||||
|     target: Dict[str, Any], value: Any, *path: str, create_missing: bool = True |  | ||||||
| ) -> Dict[str, Any]: |  | ||||||
|     """Set the key by its path to the value |     """Set the key by its path to the value | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         target (Dict[str, Any]): Dictionary to perform the modification on. |         * target (`dict`): Dictionary to perform modifications on | ||||||
|         value (Any): New value. |         * value (`Any`): Any data | ||||||
|         *path (str): Path to the key. |         * *path (`str`): Path to the key of the target | ||||||
|         create_missing (:obj:`bool`, optional): Create keys on the way if they're missing. Defaults to True. |         * create_missing (`bool`, *optional*): Create keys on the way if they're missing. Defaults to `True` | ||||||
|  |  | ||||||
|     Raises: |     ### Raises: | ||||||
|         KeyError: Key is not found under the provided path. |         * `KeyError`: Key is not found under path provided | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Dict[str, Any]: Modified dictionary. |         * `Dict[str, Any]`: Changed dictionary | ||||||
|     """ |     """ | ||||||
|     target_copy: Dict[str, Any] = target |     d = target | ||||||
|  |  | ||||||
|     for key in path[:-1]: |     for key in path[:-1]: | ||||||
|         if key in target_copy: |         if key in d: | ||||||
|             target_copy = target_copy[key] |             d = d[key] | ||||||
|         elif create_missing: |         elif create_missing: | ||||||
|             target_copy = target_copy.setdefault(key, {}) |             d = d.setdefault(key, {}) | ||||||
|         else: |         else: | ||||||
|             raise KeyError( |             raise KeyError( | ||||||
|                 f"Key '{key}' is not found under path provided ({path}) and create_missing is False" |                 f"Key '{key}' is not found under path provided ({path}) and create_missing is False" | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|     if path[-1] in target_copy or create_missing: |     if path[-1] in d or create_missing: | ||||||
|         target_copy[path[-1]] = value |         d[path[-1]] = value | ||||||
|  |  | ||||||
|     return target |     return target | ||||||
|  |  | ||||||
|  |  | ||||||
| def nested_delete(target: Dict[str, Any], *path: str) -> Dict[str, Any]: | def nested_delete(target: dict, *path: str) -> Dict[str, Any]: | ||||||
|     """Delete the key by its path. |     """Delete the key by its path | ||||||
|  |  | ||||||
|     Args: |     ### Args: | ||||||
|         target (Dict[str, Any]): Dictionary to perform the modification on. |         * target (`dict`): Dictionary to perform modifications on | ||||||
|  |  | ||||||
|     Raises: |     ### Raises: | ||||||
|         KeyError: Key is not found under the provided path. |         * `KeyError`: Key is not found under path provided | ||||||
|  |  | ||||||
|     Returns: |     ### Returns: | ||||||
|         Dict[str, Any]: Modified dictionary. |         `Dict[str, Any]`: Changed dictionary | ||||||
|     """ |     """ | ||||||
|     target_copy: Dict[str, Any] = target |     d = target | ||||||
|  |  | ||||||
|     for key in path[:-1]: |     for key in path[:-1]: | ||||||
|         if key in target_copy: |         if key in d: | ||||||
|             target_copy = target_copy[key] |             d = d[key] | ||||||
|         else: |         else: | ||||||
|             raise KeyError(f"Key '{key}' is not found under path provided ({path})") |             raise KeyError(f"Key '{key}' is not found under path provided ({path})") | ||||||
|  |  | ||||||
|     if path[-1] in target_copy: |     if path[-1] in d: | ||||||
|         del target_copy[path[-1]] |         del d[path[-1]] | ||||||
|     else: |     else: | ||||||
|         raise KeyError(f"Key '{path[-1]}' is not found under path provided ({path})") |         raise KeyError(f"Key '{path[-1]}' is not found under path provided ({path})") | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,14 +2,5 @@ | |||||||
|     "locale": "en", |     "locale": "en", | ||||||
|     "bot": { |     "bot": { | ||||||
|         "bot_token": "sample_token" |         "bot_token": "sample_token" | ||||||
|     }, |  | ||||||
|     "cache": { |  | ||||||
|         "type": "memcached", |  | ||||||
|         "memcached": { |  | ||||||
|             "uri": "127.0.0.1:11211" |  | ||||||
|         }, |  | ||||||
|         "redis": { |  | ||||||
|             "uri": "redis://127.0.0.1:6379/0" |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| from pathlib import Path |  | ||||||
|  |  | ||||||
| from libbot.cache.classes import Cache |  | ||||||
| from libbot.cache.manager import create_cache_client |  | ||||||
|  |  | ||||||
| try: |  | ||||||
|     from ujson import JSONDecodeError, dumps, loads |  | ||||||
| except ImportError: |  | ||||||
|     from json import JSONDecodeError, dumps, loads |  | ||||||
|  |  | ||||||
| from typing import Any, Dict |  | ||||||
|  |  | ||||||
| import pytest |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.parametrize( |  | ||||||
|     "engine", |  | ||||||
|     [ |  | ||||||
|         "memcached", |  | ||||||
|         "redis", |  | ||||||
|     ], |  | ||||||
| ) |  | ||||||
| def test_cache_creation(engine: str, location_config: Path): |  | ||||||
|     with open(location_config, "r", encoding="utf-8") as file: |  | ||||||
|         config: Dict[str, Any] = loads(file.read()) |  | ||||||
|  |  | ||||||
|     cache: Cache = create_cache_client(config, engine) |  | ||||||
|     assert isinstance(cache, Cache) |  | ||||||
							
								
								
									
										5
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -10,14 +10,13 @@ python = | |||||||
|     3.13: py313 |     3.13: py313 | ||||||
|  |  | ||||||
| [testenv] | [testenv] | ||||||
| setenv = | setenv =  | ||||||
|     PYTHONPATH = {toxinidir} |     PYTHONPATH = {toxinidir} | ||||||
| deps = | deps =  | ||||||
|     -r{toxinidir}/requirements/_.txt |     -r{toxinidir}/requirements/_.txt | ||||||
|     -r{toxinidir}/requirements/dev.txt |     -r{toxinidir}/requirements/dev.txt | ||||||
|     -r{toxinidir}/requirements/pycord.txt |     -r{toxinidir}/requirements/pycord.txt | ||||||
|     -r{toxinidir}/requirements/pyrogram.txt |     -r{toxinidir}/requirements/pyrogram.txt | ||||||
|     -r{toxinidir}/requirements/speed.txt |     -r{toxinidir}/requirements/speed.txt | ||||||
|     -r{toxinidir}/requirements/cache.txt |  | ||||||
| commands = | commands = | ||||||
|     pytest --basetemp={envtmpdir} --cov=libbot |     pytest --basetemp={envtmpdir} --cov=libbot | ||||||
		Reference in New Issue
	
	Block a user