Compare commits
	
		
			44 Commits
		
	
	
		
			v4.3.0
			...
			90d0eb150c
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 90d0eb150c | |||
| 150c06072a | |||
| 37d2d4e215 | |||
| 4601a7afa2 | |||
| 2bcc1f94f0 | |||
| f0532f9d39 | |||
| 97ac53b597 | |||
| e94baabcfe | |||
| c3b8eca06f | |||
| d207c564dc | |||
| e4807e2999 | |||
| 540c94c0ed | |||
| a10c4a8f51 | |||
| 9e40f89d67 | |||
| f5db3cb529 | |||
| eca59e4673 | |||
| 923bad856b | |||
| 902ccb5947 | |||
| 1fea7ea5f8 | |||
| 37928baa6f | |||
| 1f28b1771a | |||
| 5b677f6741 | |||
| 883cd50fb3 | |||
| 403b52042e | |||
| 5ca813b095 | |||
| b8cd7f2316 | |||
| 454c80ad6b | |||
| 1bbab5c154 | |||
| 6cb7f9b2f3 | |||
| a4c797079e | |||
| 9a9edbc1a8 | |||
| f991d86d4a | |||
| cea1b087d3 | |||
| 19399fe0ac | |||
| 6da6812d1d | |||
| ec3dc3a42a | |||
| 01dc9f5d87 | |||
| aa38fc5f0f | |||
| 097581bac7 | |||
| b8bbba66ec | |||
| ef7380ae45 | |||
| 727d531d63 | |||
| 7d95b1efee | |||
| 03115b4059 | 
| @@ -1,2 +1,2 @@ | |||||||
| aiofiles>=23.0.0 | aiofiles>=23.0.0 | ||||||
| typing-extensions~=4.14.0 | typing-extensions~=4.15.0 | ||||||
| @@ -1,2 +1,2 @@ | |||||||
| pymemcache~=4.0.0 | pymemcache~=4.0.0 | ||||||
| redis~=6.2.0 | redis~=6.4.0 | ||||||
| @@ -1,12 +1,12 @@ | |||||||
| black==25.1.0 | black==25.1.0 | ||||||
| build==1.2.2.post1 | build==1.3.0 | ||||||
| isort==5.13.2 | isort==5.13.2 | ||||||
| mypy==1.16.1 | mypy==1.17.1 | ||||||
| pylint==3.3.7 | pylint==3.3.8 | ||||||
| pytest-asyncio==1.0.0 | pytest-asyncio==1.1.0 | ||||||
| pytest-cov==6.2.1 | pytest-cov==6.2.1 | ||||||
| pytest==8.4.1 | pytest==8.4.1 | ||||||
| tox==4.27.0 | tox==4.30.1 | ||||||
| twine==6.1.0 | twine==6.1.0 | ||||||
| types-aiofiles==24.1.0.20250606 | types-aiofiles==24.1.0.20250822 | ||||||
| types-ujson==5.10.0.20250326 | types-ujson==5.10.0.20250822 | ||||||
| @@ -1 +1 @@ | |||||||
| ujson~=5.10.0 | ujson~=5.11.0 | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| __version__ = "4.3.0" | __version__ = "4.4.0" | ||||||
| __license__ = "GPL3" | __license__ = "GPL3" | ||||||
| __author__ = "Profitroll" | __author__ = "Profitroll" | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								src/libbot/cache/classes/cache_memcached.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								src/libbot/cache/classes/cache_memcached.py
									
									
									
									
										vendored
									
									
								
							| @@ -23,13 +23,13 @@ class CacheMemcached(Cache): | |||||||
|         logger.info("Initialized Memcached for caching") |         logger.info("Initialized Memcached for caching") | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def from_config(cls, engine_config: Dict[str, Any], prefix: Optional[str] = None) -> "CacheMemcached": |     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: |         if "uri" not in engine_config: | ||||||
|             raise KeyError( |             raise KeyError( | ||||||
|                 "Cache configuration is invalid. Please check if all keys are set (engine: memcached)" |                 "Cache configuration is invalid. Please check if all keys are set (engine: memcached)" | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         return cls(Client(engine_config["uri"], default_noreply=True), prefix=prefix) |         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: |     def _get_prefixed_key(self, key: str) -> str: | ||||||
|         return key if self.prefix is None else f"{self.prefix}_{key}" |         return key if self.prefix is None else f"{self.prefix}_{key}" | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								src/libbot/cache/classes/cache_redis.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								src/libbot/cache/classes/cache_redis.py
									
									
									
									
										vendored
									
									
								
							| @@ -23,13 +23,13 @@ class CacheRedis(Cache): | |||||||
|         logger.info("Initialized Redis for caching") |         logger.info("Initialized Redis for caching") | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def from_config(cls, engine_config: Dict[str, Any], prefix: Optional[str] = None) -> Any: |     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: |         if "uri" not in engine_config: | ||||||
|             raise KeyError( |             raise KeyError( | ||||||
|                 "Cache configuration is invalid. Please check if all keys are set (engine: memcached)" |                 "Cache configuration is invalid. Please check if all keys are set (engine: memcached)" | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         return cls(Redis.from_url(engine_config["uri"]), prefix=prefix) |         return cls(Redis.from_url(engine_config["uri"]), prefix=prefix, default_ttl_seconds=default_ttl_seconds) | ||||||
|  |  | ||||||
|     def _get_prefixed_key(self, key: str) -> str: |     def _get_prefixed_key(self, key: str) -> str: | ||||||
|         return key if self.prefix is None else f"{self.prefix}_{key}" |         return key if self.prefix is None else f"{self.prefix}_{key}" | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								src/libbot/cache/manager/manager.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								src/libbot/cache/manager/manager.py
									
									
									
									
										vendored
									
									
								
							| @@ -7,7 +7,19 @@ def create_cache_client( | |||||||
|     config: Dict[str, Any], |     config: Dict[str, Any], | ||||||
|     engine: Literal["memcached", "redis"] | None = None, |     engine: Literal["memcached", "redis"] | None = None, | ||||||
|     prefix: Optional[str] = None, |     prefix: Optional[str] = None, | ||||||
|  |     default_ttl_seconds: Optional[int] = None, | ||||||
| ) -> CacheMemcached | CacheRedis: | ) -> 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: |     if engine not in ["memcached", "redis"] or engine is None: | ||||||
|         raise KeyError(f"Incorrect cache engine provided. Expected 'memcached' or 'redis', got '{engine}'") |         raise KeyError(f"Incorrect cache engine provided. Expected 'memcached' or 'redis', got '{engine}'") | ||||||
|  |  | ||||||
| @@ -18,8 +30,8 @@ def create_cache_client( | |||||||
|  |  | ||||||
|     match engine: |     match engine: | ||||||
|         case "memcached": |         case "memcached": | ||||||
|             return CacheMemcached.from_config(config["cache"][engine], prefix=prefix) |             return CacheMemcached.from_config(config["cache"][engine], prefix=prefix, default_ttl_seconds=default_ttl_seconds) | ||||||
|         case "redis": |         case "redis": | ||||||
|             return CacheRedis.from_config(config["cache"][engine], prefix=prefix) |             return CacheRedis.from_config(config["cache"][engine], prefix=prefix, default_ttl_seconds=default_ttl_seconds) | ||||||
|         case _: |         case _: | ||||||
|             raise KeyError(f"Cache implementation for the engine '{engine}' is not present.") |             raise KeyError(f"Cache implementation for the engine '{engine}' is not present.") | ||||||
|   | |||||||
| @@ -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,6 +14,11 @@ 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): | ||||||
| @@ -30,15 +35,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 | ||||||
| @@ -64,14 +69,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] = [] | ||||||
|  |  | ||||||
| @@ -94,14 +99,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] = {} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,34 +15,24 @@ 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 (pass *[] or don't pass anything at all to get on the top/root level) | ||||||
|         * 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: |         Get the "salary" of "Pete" from this JSON structure: `{"users": {"Pete": {"salary": 10.0}}}` | ||||||
|     ```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 = libbot.sync.config_get("salary", "users", "Pete") |         salary: float = 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) | ||||||
|  |  | ||||||
| @@ -54,34 +44,24 @@ 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 (pass *[] or don't pass anything at all to get on the top/root level) | ||||||
|         * 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: |         Get the "salary" of "Pete" from this JSON structure: `{"users": {"Pete": {"salary": 10.0}}}` | ||||||
|     ```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 = await libbot.config_get("salary", "users", "Pete") |         salary: float = libbot.sync.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) | ||||||
|  |  | ||||||
| @@ -93,16 +73,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 (pass *[] or don't pass anything at all to set on the top/root level). | ||||||
|         * 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 |         KeyError: Key was not found under the provided path. | ||||||
|     """ |     """ | ||||||
|     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) | ||||||
|  |  | ||||||
| @@ -111,16 +91,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 (pass *[] or don't pass anything at all to set on the top/root level). | ||||||
|         * 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 |         KeyError: Key was not found under the provided path. | ||||||
|     """ |     """ | ||||||
|     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) | ||||||
|  |  | ||||||
| @@ -132,16 +112,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: | ||||||
|     """Set config's key by its path |     """Delete 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 (pass *[] or don't pass anything at all to delete on the top/root level) | ||||||
|         * 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) | ||||||
|  |  | ||||||
| @@ -161,16 +141,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: | ||||||
|     """Set config's key by its path |     """Delete 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 (pass *[] or don't pass anything at all to delete on the top/root level) | ||||||
|         * 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 |     """Read contents of a JSON file and return it. | ||||||
|  |  | ||||||
|     ### Args: |     Args: | ||||||
|         * path (`str | Path`): Path-like object or path as a string |         path (str | Path): Path-like object or path to the file 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 |     """Read contents of a JSON file and return it. | ||||||
|  |  | ||||||
|     ### Args: |     Args: | ||||||
|         * path (`str | Path`): Path-like object or path as a string |         path (str | Path): Path-like object or path to the file 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 |         data (Any): Contents to write. Must be a JSON-serializable object. | ||||||
|         * path (`str | Path`): Path-like object or path as a string of a destination |         path (str | Path): Path-like object or path to the file as a string. | ||||||
|     """ |     """ | ||||||
|     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 |         data (Any): Contents to write. Must be a JSON-serializable object. | ||||||
|         * path (`str | Path`): Path-like object or path as a string of a destination |         path (str | Path): Path-like object or path to the file as a string. | ||||||
|     """ |     """ | ||||||
|     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( | ||||||
|   | |||||||
| @@ -4,14 +4,14 @@ from typing import Callable | |||||||
|  |  | ||||||
|  |  | ||||||
| def supports_argument(func: Callable[..., Any], arg_name: str) -> bool: | def supports_argument(func: Callable[..., Any], 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[..., Any]): 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 | ||||||
| @@ -29,17 +29,17 @@ def nested_set( | |||||||
| ) -> Dict[str, Any]: | ) -> 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 modifications on |         target (Dict[str, Any]): Dictionary to perform the modification on. | ||||||
|         * value (`Any`): Any data |         value (Any): New value. | ||||||
|         * *path (`str`): Path to the key of the target |         *path (str): Path to the key. | ||||||
|         * create_missing (`bool`, *optional*): Create keys on the way if they're missing. Defaults to `True` |         create_missing (:obj:`bool`, optional): Create keys on the way if they're missing. Defaults to True. | ||||||
|  |  | ||||||
|     ### Raises: |     Raises: | ||||||
|         * `KeyError`: Key is not found under path provided |         KeyError: Key is not found under the provided path. | ||||||
|  |  | ||||||
|     ### Returns: |     Returns: | ||||||
|         * `Dict[str, Any]`: Changed dictionary |         Dict[str, Any]: Modified dictionary. | ||||||
|     """ |     """ | ||||||
|     target_copy: Dict[str, Any] = target |     target_copy: Dict[str, Any] = target | ||||||
|  |  | ||||||
| @@ -60,16 +60,16 @@ def nested_set( | |||||||
|  |  | ||||||
|  |  | ||||||
| def nested_delete(target: Dict[str, Any], *path: str) -> Dict[str, Any]: | def nested_delete(target: Dict[str, Any], *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 modifications on |         target (Dict[str, Any]): Dictionary to perform the modification on. | ||||||
|  |  | ||||||
|     ### Raises: |     Raises: | ||||||
|         * `KeyError`: Key is not found under path provided |         KeyError: Key is not found under the provided path. | ||||||
|  |  | ||||||
|     ### Returns: |     Returns: | ||||||
|         `Dict[str, Any]`: Changed dictionary |         Dict[str, Any]: Modified dictionary. | ||||||
|     """ |     """ | ||||||
|     target_copy: Dict[str, Any] = target |     target_copy: Dict[str, Any] = target | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user