Compare commits
	
		
			4 Commits
		
	
	
		
			1.2.5
			...
			77d723b9da
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 77d723b9da | |||
| 75c316f2d2 | |||
| de78cec96c | |||
| 9de66d4fd4 | 
| @@ -1,2 +1,14 @@ | |||||||
| # kd_discord_bot | # kd_discord_bot | ||||||
|  |  | ||||||
|  | ## Test Bot | ||||||
|  |  | ||||||
|  | To test the bot run unittests or call ```cpl test```. | ||||||
|  | Configure test instance by creating the file ./test/ui_tests/.env and set following environment variables: | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | KDB_TEST_DB_PASSWORD= | ||||||
|  | KDB_TEST_NAME= | ||||||
|  | KDB_TEST_TOKEN= | ||||||
|  | KDB_TEST_DISCORD_MAIL= | ||||||
|  | KDB_TEST_DISCORD_PASSWORD= | ||||||
|  | ``` | ||||||
| @@ -14,29 +14,27 @@ | |||||||
|       "permission": "src/modules/permission/permission.json", |       "permission": "src/modules/permission/permission.json", | ||||||
|       "stats": "src/modules/stats/stats.json", |       "stats": "src/modules/stats/stats.json", | ||||||
|       "technician": "src/modules/technician/technician.json", |       "technician": "src/modules/technician/technician.json", | ||||||
|  |       "ui-tests": "test/ui_tests/ui-tests.json", | ||||||
|  |       "ui-tests-shared": "test/ui_tests_shared/ui-tests-shared.json", | ||||||
|  |       "ui-tests-tests": "test/ui_tests_tests/ui-tests-tests.json", | ||||||
|       "get-version": "tools/get_version/get-version.json", |       "get-version": "tools/get_version/get-version.json", | ||||||
|       "post-build": "tools/post_build/post-build.json", |       "post-build": "tools/post_build/post-build.json", | ||||||
|       "set-version": "tools/set_version/set-version.json" |       "set-version": "tools/set_version/set-version.json" | ||||||
|     }, |     }, | ||||||
|     "Scripts": { |     "Scripts": { | ||||||
|  |       "test": "export $(cat test/ui_tests/.env); export PYTHONPATH=$PWD/src:$PYTHONPATH; cpl run ui-tests", | ||||||
|       "sv": "cpl set-version $ARGS", |       "sv": "cpl set-version $ARGS", | ||||||
|       "set-version": "cpl run set-version $ARGS; echo '';", |       "set-version": "cpl run set-version $ARGS; echo '';", | ||||||
|  |  | ||||||
|       "gv": "cpl get-version", |       "gv": "cpl get-version", | ||||||
|       "get-version": "export VERSION=$(cpl run get-version); echo $VERSION;", |       "get-version": "export VERSION=$(cpl run get-version); echo $VERSION;", | ||||||
|  |  | ||||||
|       "pre-build": "cpl set-version $ARGS", |       "pre-build": "cpl set-version $ARGS", | ||||||
|       "post-build": "cpl run post-build", |       "post-build": "cpl run post-build", | ||||||
|  |  | ||||||
|       "pre-prod": "cpl build", |       "pre-prod": "cpl build", | ||||||
|       "prod": "export KDB_ENVIRONMENT=production; export KDB_NAME=KDB-Prod; cpl start;", |       "prod": "export KDB_ENVIRONMENT=production; export KDB_NAME=KDB-Prod; cpl start;", | ||||||
|  |  | ||||||
|       "pre-stage": "cpl build", |       "pre-stage": "cpl build", | ||||||
|       "stage": "export KDB_ENVIRONMENT=staging; export KDB_NAME=KDB-Stage; cpl start;", |       "stage": "export KDB_ENVIRONMENT=staging; export KDB_NAME=KDB-Stage; cpl start;", | ||||||
|  |  | ||||||
|       "pre-dev": "cpl build", |       "pre-dev": "cpl build", | ||||||
|       "dev": "export KDB_ENVIRONMENT=development; export KDB_NAME=KDB-Dev; cpl start;", |       "dev": "export KDB_ENVIRONMENT=development; export KDB_NAME=KDB-Dev; cpl start;", | ||||||
|  |  | ||||||
|       "docker-build": "cpl build $ARGS; docker build -t kdb-bot/kdb-bot:$(cpl gv) .;", |       "docker-build": "cpl build $ARGS; docker build -t kdb-bot/kdb-bot:$(cpl gv) .;", | ||||||
|       "dc-up": "docker-compose up -d", |       "dc-up": "docker-compose up -d", | ||||||
|       "dc-down": "docker-compose down", |       "dc-down": "docker-compose down", | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								kdb-bot/test/ui_tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								kdb-bot/test/ui_tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | # imports: | ||||||
							
								
								
									
										16
									
								
								kdb-bot/test/ui_tests/config/appsettings.edrafts-pc.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								kdb-bot/test/ui_tests/config/appsettings.edrafts-pc.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | { | ||||||
|  |   "DatabaseSettings": { | ||||||
|  |     "Host": "localhost", | ||||||
|  |     "User": "kd_kdb", | ||||||
|  |     "Password": "", | ||||||
|  |     "Database": "keksdose_bot_dev", | ||||||
|  |     "Charset": "utf8mb4", | ||||||
|  |     "UseUnicode": "true", | ||||||
|  |     "Buffered": "true", | ||||||
|  |     "AuthPlugin": "mysql_native_password" | ||||||
|  |   }, | ||||||
|  |   "DiscordBot": { | ||||||
|  |     "Token": "", | ||||||
|  |     "Prefix": "!kab-e " | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										25
									
								
								kdb-bot/test/ui_tests/config/appsettings.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								kdb-bot/test/ui_tests/config/appsettings.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | { | ||||||
|  |   "TimeFormatSettings": { | ||||||
|  |     "DateFormat": "%Y-%m-%d", | ||||||
|  |     "TimeFormat": "%H:%M:%S", | ||||||
|  |     "DateTimeFormat": "%Y-%m-%d %H:%M:%S.%f", | ||||||
|  |     "DateTimeLogFormat": "%Y-%m-%d_%H-%M-%S" | ||||||
|  |   }, | ||||||
|  |   "LoggingSettings": { | ||||||
|  |     "Path": "logs/$date_now/", | ||||||
|  |     "Filename": "bot.log", | ||||||
|  |     "ConsoleLogLevel": "ERROR", | ||||||
|  |     "FileLogLevel": "WARN" | ||||||
|  |   }, | ||||||
|  |   "Translation": { | ||||||
|  |     "DefaultLanguage": "de", | ||||||
|  |     "Languages": [ | ||||||
|  |       "de" | ||||||
|  |     ] | ||||||
|  |   }, | ||||||
|  |   "TestSettings": { | ||||||
|  |     "LoginUrl": "https://discord.com/login", | ||||||
|  |     "MePageUrl": "https://discord.com/channels/@me", | ||||||
|  |     "CmdURL": "https://discord.com/channels/910199451145076828/911578636899987526" | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										34
									
								
								kdb-bot/test/ui_tests/main.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								kdb-bot/test/ui_tests/main.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | import asyncio | ||||||
|  |  | ||||||
|  | from cpl_core.application import ApplicationABC | ||||||
|  | from cpl_core.application import ApplicationBuilder | ||||||
|  |  | ||||||
|  | from startup_test_extension import StartupTestExtension | ||||||
|  | from ui_tests.startup import Startup | ||||||
|  | from ui_tests_shared.declarations import Declarations | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     from test_application import TestApplication | ||||||
|  | except ImportError: | ||||||
|  |     from .test_application import TestApplication | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_app() -> ApplicationABC: | ||||||
|  |     app_builder = ApplicationBuilder(TestApplication) \ | ||||||
|  |         .use_extension(StartupTestExtension) \ | ||||||
|  |         .use_startup(Startup) | ||||||
|  |     app = app_builder.build() | ||||||
|  |     Declarations.app = app | ||||||
|  |     return app | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def main(): | ||||||
|  |     app = get_app() | ||||||
|  |     await app.run_async() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     import nest_asyncio | ||||||
|  |  | ||||||
|  |     nest_asyncio.apply() | ||||||
|  |     asyncio.run(main()) | ||||||
							
								
								
									
										58
									
								
								kdb-bot/test/ui_tests/startup.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								kdb-bot/test/ui_tests/startup.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | |||||||
|  | import os | ||||||
|  | from typing import Optional | ||||||
|  |  | ||||||
|  | from cpl_core.application import StartupABC | ||||||
|  | from cpl_core.configuration import ConfigurationABC | ||||||
|  | from cpl_core.database import DatabaseSettings | ||||||
|  | from cpl_core.database.context import DatabaseContext | ||||||
|  | from cpl_core.dependency_injection import ServiceProviderABC, ServiceCollectionABC | ||||||
|  | from cpl_core.environment import ApplicationEnvironment | ||||||
|  | from cpl_discord import get_discord_collection | ||||||
|  | from cpl_discord.discord_event_types_enum import DiscordEventTypesEnum | ||||||
|  | from dotenv import load_dotenv | ||||||
|  |  | ||||||
|  | from ui_tests.test_on_ready_event import TestOnReadyEvent | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Startup(StartupABC): | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         StartupABC.__init__(self) | ||||||
|  |  | ||||||
|  |         self._config: Optional[ConfigurationABC] = None | ||||||
|  |  | ||||||
|  |     def configure_configuration(self, configuration: ConfigurationABC, environment: ApplicationEnvironment) -> ConfigurationABC: | ||||||
|  |         load_dotenv() | ||||||
|  |         configuration.add_environment_variables('KDB_TEST_') | ||||||
|  |         configuration.add_environment_variables('DISCORD_') | ||||||
|  |  | ||||||
|  |         cwd = os.path.dirname(os.path.realpath(__file__)) | ||||||
|  |         configuration.add_json_file(f'{cwd}/config/appsettings.json', optional=False) | ||||||
|  |         configuration.add_json_file(f'{cwd}/config/appsettings.{environment.host_name}.json', optional=True) | ||||||
|  |  | ||||||
|  |         self._config = configuration | ||||||
|  |         return configuration | ||||||
|  |  | ||||||
|  |     def configure_services(self, services: ServiceCollectionABC, environment: ApplicationEnvironment) -> ServiceProviderABC: | ||||||
|  |         services.add_logging() | ||||||
|  |         services.add_translation() | ||||||
|  |         db_settings: DatabaseSettings = self._config.get_configuration(DatabaseSettings) | ||||||
|  |         db_settings_with_pw = DatabaseSettings() | ||||||
|  |         pw = self._config.get_configuration('DB_PASSWORD') | ||||||
|  |         db_settings_with_pw.from_dict({ | ||||||
|  |             "Host": db_settings.host, | ||||||
|  |             "User": db_settings.user, | ||||||
|  |             "Password": '' if pw is None else pw, | ||||||
|  |             "Database": db_settings.database, | ||||||
|  |             "Charset": db_settings.charset, | ||||||
|  |             "UseUnicode": db_settings.use_unicode, | ||||||
|  |             "Buffered": db_settings.buffered, | ||||||
|  |             "AuthPlugin": db_settings.auth_plugin | ||||||
|  |         }) | ||||||
|  |         services.add_db_context(DatabaseContext, db_settings_with_pw) | ||||||
|  |  | ||||||
|  |         services.add_discord() | ||||||
|  |         dc = get_discord_collection(services) | ||||||
|  |         dc.add_event(DiscordEventTypesEnum.on_ready.value, TestOnReadyEvent) | ||||||
|  |  | ||||||
|  |         return services.build_service_provider() | ||||||
							
								
								
									
										20
									
								
								kdb-bot/test/ui_tests/startup_test_extension.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								kdb-bot/test/ui_tests/startup_test_extension.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | import os | ||||||
|  |  | ||||||
|  | from cpl_core.application import StartupExtensionABC | ||||||
|  | from cpl_core.configuration import ConfigurationABC | ||||||
|  | from cpl_core.dependency_injection import ServiceCollectionABC | ||||||
|  | from cpl_core.environment import ApplicationEnvironmentABC | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class StartupTestExtension(StartupExtensionABC): | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         StartupExtensionABC.__init__(self) | ||||||
|  |  | ||||||
|  |     def configure_configuration(self, configuration: ConfigurationABC, environment: ApplicationEnvironmentABC): | ||||||
|  |         # this shit has to be done here because we need settings in subsequent startup extensions | ||||||
|  |         environment.set_working_directory(os.path.dirname(os.path.realpath(__file__))) | ||||||
|  |         environment.set_working_directory('../../src/bot/') | ||||||
|  |  | ||||||
|  |     def configure_services(self, services: ServiceCollectionABC, env: ApplicationEnvironmentABC): | ||||||
|  |         pass | ||||||
							
								
								
									
										64
									
								
								kdb-bot/test/ui_tests/test_application.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								kdb-bot/test/ui_tests/test_application.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | |||||||
|  | from cpl_core.configuration import ConfigurationABC | ||||||
|  | from cpl_core.dependency_injection import ServiceProviderABC | ||||||
|  | from cpl_discord.application import DiscordBotApplicationABC | ||||||
|  | from cpl_discord.configuration import DiscordBotSettings | ||||||
|  | from cpl_discord.service import DiscordBotServiceABC | ||||||
|  | from cpl_translation import TranslationSettings, TranslationServiceABC | ||||||
|  |  | ||||||
|  | from bot_core.configuration.bot_settings import BotSettings | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestApplication(DiscordBotApplicationABC): | ||||||
|  |  | ||||||
|  |     def __init__(self, config: ConfigurationABC, services: ServiceProviderABC): | ||||||
|  |         DiscordBotApplicationABC.__init__(self, config, services) | ||||||
|  |  | ||||||
|  |         self._config = config | ||||||
|  |         self._services = services | ||||||
|  |  | ||||||
|  |         self._bot: DiscordBotServiceABC = services.get_service(DiscordBotServiceABC) | ||||||
|  |         self._translation: TranslationServiceABC = services.get_service(TranslationServiceABC) | ||||||
|  |  | ||||||
|  |         discord_bot_settings: DiscordBotSettings = config.get_configuration(DiscordBotSettings) | ||||||
|  |         self._discord_settings = self._get_settings(discord_bot_settings) | ||||||
|  |  | ||||||
|  |     def _get_settings(self, settings_from_config: DiscordBotSettings) -> DiscordBotSettings: | ||||||
|  |         new_settings = DiscordBotSettings() | ||||||
|  |         token = None if settings_from_config is None else settings_from_config.token | ||||||
|  |         prefix = None if settings_from_config is None else settings_from_config.prefix | ||||||
|  |         env_token = self._config.get_configuration('TOKEN') | ||||||
|  |         env_prefix = self._config.get_configuration('PREFIX') | ||||||
|  |  | ||||||
|  |         new_settings.from_dict({ | ||||||
|  |             'Token': env_token if token is None or token == '' else token, | ||||||
|  |             'Prefix': ('! ' if self._is_string_invalid(env_prefix) else env_prefix) if self._is_string_invalid(prefix) else prefix | ||||||
|  |         }) | ||||||
|  |         if new_settings.token is None or new_settings.token == '': | ||||||
|  |             raise Exception('You have to configure discord token by appsettings or environment variables') | ||||||
|  |         return new_settings | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def _is_string_invalid(x): | ||||||
|  |         return x is None or x == '' | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def config(self) -> ConfigurationABC: | ||||||
|  |         return self._config | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def services(self) -> ServiceProviderABC: | ||||||
|  |         return self._services | ||||||
|  |  | ||||||
|  |     async def configure(self): | ||||||
|  |         self._translation.load_by_settings(self._configuration.get_configuration(TranslationSettings)) | ||||||
|  |  | ||||||
|  |     async def main(self): | ||||||
|  |         if self._config.get_configuration('IS_UNITTEST'): | ||||||
|  |             await self._bot.login(self._discord_settings.token) | ||||||
|  |             self._bot.loop.create_task(self._bot.connect()) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         await self._bot.start_async() | ||||||
|  |  | ||||||
|  |     async def stop_async(self): | ||||||
|  |         await self._bot.close() | ||||||
							
								
								
									
										47
									
								
								kdb-bot/test/ui_tests/test_on_ready_event.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								kdb-bot/test/ui_tests/test_on_ready_event.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | |||||||
|  | import os | ||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from cpl_core.configuration import ConfigurationABC | ||||||
|  | from cpl_core.console import Console, ForegroundColorEnum | ||||||
|  | from cpl_core.dependency_injection import ServiceProviderABC | ||||||
|  | from cpl_core.logging import LoggerABC | ||||||
|  | from cpl_discord.events import OnReadyABC | ||||||
|  | from cpl_discord.service import DiscordBotServiceABC | ||||||
|  | from cpl_translation import TranslatePipe | ||||||
|  |  | ||||||
|  | from bot_core.abc.client_utils_service_abc import ClientUtilsServiceABC | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestOnReadyEvent(OnReadyABC): | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |             self, | ||||||
|  |             config: ConfigurationABC, | ||||||
|  |             bot: DiscordBotServiceABC, | ||||||
|  |             services: ServiceProviderABC, | ||||||
|  |             client_utils: ClientUtilsServiceABC, | ||||||
|  |             t: TranslatePipe | ||||||
|  |     ): | ||||||
|  |         OnReadyABC.__init__(self) | ||||||
|  |  | ||||||
|  |         self._config = config | ||||||
|  |         self._bot = bot | ||||||
|  |         self._services = services | ||||||
|  |         self._client_utils = client_utils | ||||||
|  |         self._t = t | ||||||
|  |  | ||||||
|  |     async def on_ready(self): | ||||||
|  |         if self._config.get_configuration('IS_UNITTEST'): | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         Console.write_line('\nStarting tests:\n') | ||||||
|  |         loader = unittest.TestLoader() | ||||||
|  |         path = f'{os.path.dirname(os.path.realpath(__file__))}/../' | ||||||
|  |         tests = loader.discover(path, pattern='*_test_case.py') | ||||||
|  |         runner = unittest.TextTestRunner() | ||||||
|  |         runner.run(tests) | ||||||
|  |         # for cls in CommandTestABC.__subclasses__(): | ||||||
|  |         #     service: CommandTestABC = self._services.get_service(cls) | ||||||
|  |         #     await service.run(self._tests) | ||||||
|  |  | ||||||
|  |         await self._bot.close() | ||||||
							
								
								
									
										44
									
								
								kdb-bot/test/ui_tests/ui-tests.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								kdb-bot/test/ui_tests/ui-tests.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | |||||||
|  | { | ||||||
|  |   "ProjectSettings": { | ||||||
|  |     "Name": "ui-tests", | ||||||
|  |     "Version": { | ||||||
|  |       "Major": "0", | ||||||
|  |       "Minor": "0", | ||||||
|  |       "Micro": "0" | ||||||
|  |     }, | ||||||
|  |     "Author": "", | ||||||
|  |     "AuthorEmail": "", | ||||||
|  |     "Description": "", | ||||||
|  |     "LongDescription": "", | ||||||
|  |     "URL": "", | ||||||
|  |     "CopyrightDate": "", | ||||||
|  |     "CopyrightName": "", | ||||||
|  |     "LicenseName": "", | ||||||
|  |     "LicenseDescription": "", | ||||||
|  |     "Dependencies": [ | ||||||
|  |       "cpl-core>=2022.10.0.post9" | ||||||
|  |     ], | ||||||
|  |     "DevDependencies": [ | ||||||
|  |       "cpl-cli>=2022.10.0" | ||||||
|  |     ], | ||||||
|  |     "PythonVersion": ">=3.10.4", | ||||||
|  |     "PythonPath": {}, | ||||||
|  |     "Classifiers": [] | ||||||
|  |   }, | ||||||
|  |   "BuildSettings": { | ||||||
|  |     "ProjectType": "console", | ||||||
|  |     "SourcePath": "", | ||||||
|  |     "OutputPath": "../../dist", | ||||||
|  |     "Main": "ui_tests_shared.main", | ||||||
|  |     "EntryPoint": "ui-tests-shared", | ||||||
|  |     "IncludePackageData": false, | ||||||
|  |     "Included": [], | ||||||
|  |     "Excluded": [ | ||||||
|  |       "*/__pycache__", | ||||||
|  |       "*/logs", | ||||||
|  |       "*/tests" | ||||||
|  |     ], | ||||||
|  |     "PackageData": {}, | ||||||
|  |     "ProjectReferences": [] | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								kdb-bot/test/ui_tests_shared/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								kdb-bot/test/ui_tests_shared/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | # imports:  | ||||||
							
								
								
									
										48
									
								
								kdb-bot/test/ui_tests_shared/command_test_case_with_app.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								kdb-bot/test/ui_tests_shared/command_test_case_with_app.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | |||||||
|  | import time | ||||||
|  | from typing import Optional | ||||||
|  |  | ||||||
|  | from cpl_discord.service import DiscordBotServiceABC | ||||||
|  | from cpl_translation import TranslatePipe | ||||||
|  | from selenium.common import NoSuchElementException, StaleElementReferenceException | ||||||
|  | from selenium.webdriver import Keys | ||||||
|  | from selenium.webdriver.common.by import By | ||||||
|  | from selenium.webdriver.remote.webelement import WebElement | ||||||
|  | from selenium.webdriver.support import expected_conditions | ||||||
|  | from selenium.webdriver.support.wait import WebDriverWait | ||||||
|  |  | ||||||
|  | from ui_tests_shared.test_case_with_app import TestCaseWithApp | ||||||
|  | from ui_tests_shared.ui import UI | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CommandTestCaseWithApp(TestCaseWithApp): | ||||||
|  |     _cmd: WebElement | ||||||
|  |  | ||||||
|  |     _bot: Optional[DiscordBotServiceABC] = None | ||||||
|  |     _t: Optional[TranslatePipe] = None | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def setUpClass(cls): | ||||||
|  |         TestCaseWithApp.setUpClass() | ||||||
|  |         cls._bot = cls._services.get_service(DiscordBotServiceABC) | ||||||
|  |         cls._t = cls._services.get_service(TranslatePipe) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def send_command(cls, cmd: str): | ||||||
|  |         UI.driver.get(cls._test_settings.cmd_url) | ||||||
|  |         time.sleep(2) | ||||||
|  |  | ||||||
|  |         cmd_element_ident = (By.XPATH, '/html/body/div[1]/div[2]/div/div[1]/div/div[2]/div/div[1]/div/div/div[3]/div[2]/main/form/div/div[1]/div/div[3]/div/div[2]/div') | ||||||
|  |         cmd_element = UI.driver.find_element(*cmd_element_ident) | ||||||
|  |         cmd_element.send_keys(f'/{cmd}') | ||||||
|  |         time.sleep(2) | ||||||
|  |  | ||||||
|  |         ignored_exceptions = (NoSuchElementException, StaleElementReferenceException,) | ||||||
|  |         WebDriverWait(UI.driver, 20, ignored_exceptions=ignored_exceptions).until( | ||||||
|  |             expected_conditions.presence_of_element_located(( | ||||||
|  |                 By.XPATH, | ||||||
|  |                 '/html/body/div[1]/div[2]/div/div[1]/div/div[2]/div/div[1]/div/div/div[3]/div[2]/main/form/div/div[2]/div/div/div[5]' | ||||||
|  |             )) | ||||||
|  |         ).click() | ||||||
|  |         time.sleep(2) | ||||||
|  |         UI.driver.find_element(By.XPATH, '/html/body/div[1]/div[2]/div/div[1]/div/div[2]/div/div[1]/div/div/div[3]/div/main/form/div/div[2]/div/div[2]/div/div').send_keys( | ||||||
|  |             Keys.ENTER) | ||||||
							
								
								
									
										1
									
								
								kdb-bot/test/ui_tests_shared/configuration/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								kdb-bot/test/ui_tests_shared/configuration/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | # imports | ||||||
							
								
								
									
										35
									
								
								kdb-bot/test/ui_tests_shared/configuration/test_settings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								kdb-bot/test/ui_tests_shared/configuration/test_settings.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | |||||||
|  | import traceback | ||||||
|  |  | ||||||
|  | from cpl_core.configuration.configuration_model_abc import ConfigurationModelABC | ||||||
|  | from cpl_core.console import Console | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestSettings(ConfigurationModelABC): | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         ConfigurationModelABC.__init__(self) | ||||||
|  |  | ||||||
|  |         self._login_url = '' | ||||||
|  |         self._me_page_url = '' | ||||||
|  |         self._cmd_url = '' | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def login_url(self) -> str: | ||||||
|  |         return self._login_url | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def me_page_url(self) -> str: | ||||||
|  |         return self._me_page_url | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def cmd_url(self) -> str: | ||||||
|  |         return self._cmd_url | ||||||
|  |  | ||||||
|  |     def from_dict(self, settings: dict): | ||||||
|  |         try: | ||||||
|  |             self._login_url = settings['LoginUrl'] | ||||||
|  |             self._me_page_url = settings['MePageUrl'] | ||||||
|  |             self._cmd_url = settings['CmdURL'] | ||||||
|  |         except Exception as e: | ||||||
|  |             Console.error(f'[ ERROR ] [ {__name__} ]: Reading error in {type(self).__name__} settings') | ||||||
|  |             Console.error(f'[ EXCEPTION ] [ {__name__} ]: {e} -> {traceback.format_exc()}') | ||||||
							
								
								
									
										12
									
								
								kdb-bot/test/ui_tests_shared/declarations.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								kdb-bot/test/ui_tests_shared/declarations.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | from typing import Optional | ||||||
|  |  | ||||||
|  | from cpl_core.environment import ApplicationEnvironmentABC | ||||||
|  |  | ||||||
|  | from ui_tests.test_application import TestApplication | ||||||
|  | from ui_tests_shared.configuration.test_settings import TestSettings | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Declarations: | ||||||
|  |     app: Optional[TestApplication] = None | ||||||
|  |     env: Optional[ApplicationEnvironmentABC] = None | ||||||
|  |     test_settings: Optional[TestSettings] = None | ||||||
							
								
								
									
										14
									
								
								kdb-bot/test/ui_tests_shared/decorators.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								kdb-bot/test/ui_tests_shared/decorators.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | import asyncio | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Async: | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def test(cls, coro): | ||||||
|  |         def wrapper(*args, **kwargs): | ||||||
|  |             try: | ||||||
|  |                 return asyncio.run(coro(*args, **kwargs)) | ||||||
|  |             except Exception as e: | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |         return wrapper | ||||||
							
								
								
									
										39
									
								
								kdb-bot/test/ui_tests_shared/test_case_with_app.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								kdb-bot/test/ui_tests_shared/test_case_with_app.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | import asyncio | ||||||
|  | import time | ||||||
|  | import unittest | ||||||
|  | from typing import Optional | ||||||
|  |  | ||||||
|  | from cpl_core.configuration import ConfigurationABC | ||||||
|  | from cpl_core.dependency_injection import ServiceProviderABC | ||||||
|  |  | ||||||
|  | from ui_tests.main import get_app | ||||||
|  | from ui_tests_shared.configuration.test_settings import TestSettings | ||||||
|  | from ui_tests_shared.declarations import Declarations | ||||||
|  | from ui_tests_shared.ui import UI | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestCaseWithApp(unittest.TestCase): | ||||||
|  |     _config: Optional[ConfigurationABC] = None | ||||||
|  |     _services: Optional[ServiceProviderABC] = None | ||||||
|  |     _test_settings: Optional[TestSettings] = None | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def setUpClass(cls): | ||||||
|  |         if Declarations.app is None: | ||||||
|  |             app = get_app() | ||||||
|  |             import nest_asyncio | ||||||
|  |  | ||||||
|  |             nest_asyncio.apply() | ||||||
|  |             asyncio.run(app.run_async()) | ||||||
|  |  | ||||||
|  |         cls._config = Declarations.app.config | ||||||
|  |         cls._services = Declarations.app.services | ||||||
|  |         cls._test_settings: TestSettings = cls._config.get_configuration(TestSettings) | ||||||
|  |         UI.set_settings(cls._test_settings) | ||||||
|  |         UI.login(cls._config.get_configuration('DISCORD_MAIL'), cls._config.get_configuration('DISCORD_PASSWORD')) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def tearDownClass(cls): | ||||||
|  |         if cls._config.get_configuration('IS_UNITTEST'): | ||||||
|  |             loop = asyncio.get_event_loop() | ||||||
|  |             loop.run_until_complete(Declarations.app.stop_async()) | ||||||
							
								
								
									
										44
									
								
								kdb-bot/test/ui_tests_shared/ui-tests-shared.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								kdb-bot/test/ui_tests_shared/ui-tests-shared.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | |||||||
|  | { | ||||||
|  |   "ProjectSettings": { | ||||||
|  |     "Name": "ui-tests-shared", | ||||||
|  |     "Version": { | ||||||
|  |       "Major": "0", | ||||||
|  |       "Minor": "0", | ||||||
|  |       "Micro": "0" | ||||||
|  |     }, | ||||||
|  |     "Author": "", | ||||||
|  |     "AuthorEmail": "", | ||||||
|  |     "Description": "", | ||||||
|  |     "LongDescription": "", | ||||||
|  |     "URL": "", | ||||||
|  |     "CopyrightDate": "", | ||||||
|  |     "CopyrightName": "", | ||||||
|  |     "LicenseName": "", | ||||||
|  |     "LicenseDescription": "", | ||||||
|  |     "Dependencies": [ | ||||||
|  |       "cpl-core>=2022.10.0.post9" | ||||||
|  |     ], | ||||||
|  |     "DevDependencies": [ | ||||||
|  |       "cpl-cli>=2022.10.0" | ||||||
|  |     ], | ||||||
|  |     "PythonVersion": ">=3.10.4", | ||||||
|  |     "PythonPath": {}, | ||||||
|  |     "Classifiers": [] | ||||||
|  |   }, | ||||||
|  |   "BuildSettings": { | ||||||
|  |     "ProjectType": "console", | ||||||
|  |     "SourcePath": "", | ||||||
|  |     "OutputPath": "../../dist", | ||||||
|  |     "Main": "ui_tests_shared.main", | ||||||
|  |     "EntryPoint": "ui-tests-shared", | ||||||
|  |     "IncludePackageData": false, | ||||||
|  |     "Included": [], | ||||||
|  |     "Excluded": [ | ||||||
|  |       "*/__pycache__", | ||||||
|  |       "*/logs", | ||||||
|  |       "*/tests" | ||||||
|  |     ], | ||||||
|  |     "PackageData": {}, | ||||||
|  |     "ProjectReferences": [] | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										57
									
								
								kdb-bot/test/ui_tests_shared/ui.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								kdb-bot/test/ui_tests_shared/ui.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | |||||||
|  | import time | ||||||
|  | import unittest | ||||||
|  | from typing import Optional | ||||||
|  |  | ||||||
|  | from cpl_core.configuration import ConfigurationABC | ||||||
|  | from cpl_core.dependency_injection import ServiceProviderABC | ||||||
|  | from selenium import webdriver | ||||||
|  | from selenium.webdriver import Keys | ||||||
|  | from selenium.webdriver.common.by import By | ||||||
|  | from selenium.webdriver.support import expected_conditions | ||||||
|  | from selenium.webdriver.support.wait import WebDriverWait | ||||||
|  |  | ||||||
|  | from ui_tests.main import get_app | ||||||
|  | from ui_tests_shared.configuration.test_settings import TestSettings | ||||||
|  | from ui_tests_shared.declarations import Declarations | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class UI: | ||||||
|  |     _test_settings: Optional[TestSettings] = None | ||||||
|  |  | ||||||
|  |     options = webdriver.ChromeOptions() | ||||||
|  |     options.add_experimental_option('useAutomationExtension', False) | ||||||
|  |     options.add_experimental_option("excludeSwitches", ["enable-automation"]) | ||||||
|  |     driver = webdriver.Chrome(options=options) | ||||||
|  |  | ||||||
|  |     _is_logged_in = False | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def set_settings(cls, test_settings: TestSettings): | ||||||
|  |         cls._test_settings = test_settings | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def login(cls, mail: str, password: str): | ||||||
|  |         if cls._is_logged_in: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         # use full xpath: https://stackoverflow.com/questions/71179006/how-can-selenium-python-chrome-find-web-elements-visible-in-dev-tools-but-no | ||||||
|  |  | ||||||
|  |         cls.driver.get(cls._test_settings.login_url) | ||||||
|  |  | ||||||
|  |         WebDriverWait(cls.driver, 20).until(expected_conditions.presence_of_element_located((By.NAME, 'email'))) | ||||||
|  |  | ||||||
|  |         mail_element = cls.driver.find_element(By.NAME, 'email') | ||||||
|  |         mail_element.clear() | ||||||
|  |         mail_element.send_keys(mail) | ||||||
|  |         mail_element.send_keys(Keys.RETURN) | ||||||
|  |  | ||||||
|  |         pw_element = cls.driver.find_element(By.NAME, 'password') | ||||||
|  |         pw_element.clear() | ||||||
|  |         pw_element.send_keys(password) | ||||||
|  |         pw_element.send_keys(Keys.RETURN) | ||||||
|  |         time.sleep(1) | ||||||
|  |         pw_element.send_keys(Keys.RETURN) | ||||||
|  |  | ||||||
|  |         WebDriverWait(cls.driver, 20).until(expected_conditions.url_matches(cls._test_settings.me_page_url)) | ||||||
|  |         time.sleep(4) | ||||||
|  |         cls._is_logged_in = True | ||||||
							
								
								
									
										1
									
								
								kdb-bot/test/ui_tests_tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								kdb-bot/test/ui_tests_tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | # imports:  | ||||||
							
								
								
									
										15
									
								
								kdb-bot/test/ui_tests_tests/appsettings.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								kdb-bot/test/ui_tests_tests/appsettings.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | { | ||||||
|  |   "TimeFormatSettings": { | ||||||
|  |     "DateFormat": "%Y-%m-%d", | ||||||
|  |     "TimeFormat": "%H:%M:%S", | ||||||
|  |     "DateTimeFormat": "%Y-%m-%d %H:%M:%S.%f", | ||||||
|  |     "DateTimeLogFormat": "%Y-%m-%d_%H-%M-%S" | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   "LoggingSettings": { | ||||||
|  |     "Path": "logs/", | ||||||
|  |     "Filename": "log_$start_time.log", | ||||||
|  |     "ConsoleLogLevel": "ERROR", | ||||||
|  |     "FileLogLevel": "WARN" | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										0
									
								
								kdb-bot/test/ui_tests_tests/cases/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								kdb-bot/test/ui_tests_tests/cases/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										19
									
								
								kdb-bot/test/ui_tests_tests/cases/help_test_case.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								kdb-bot/test/ui_tests_tests/cases/help_test_case.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | import discord | ||||||
|  |  | ||||||
|  | from ui_tests_shared.command_test_case_with_app import CommandTestCaseWithApp | ||||||
|  | from ui_tests_shared.decorators import Async | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class HelpTestCase(CommandTestCaseWithApp): | ||||||
|  |  | ||||||
|  |     @Async.test | ||||||
|  |     async def test_help(self): | ||||||
|  |         correct_response = 'https://git.sh-edraft.de/sh-edraft.de/kd_discord_bot/wiki/Befehle' | ||||||
|  |         self.send_command('help') | ||||||
|  |  | ||||||
|  |         def check(m: discord.Message): | ||||||
|  |             return m.content == correct_response and m.author.id == 998159802393964594 | ||||||
|  |  | ||||||
|  |         response = await self._bot.wait_for('message', check=check) | ||||||
|  |         self.assertEqual(response.content, correct_response) | ||||||
|  |         await self._bot.close() | ||||||
							
								
								
									
										19
									
								
								kdb-bot/test/ui_tests_tests/cases/ping_test_case.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								kdb-bot/test/ui_tests_tests/cases/ping_test_case.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | import discord | ||||||
|  |  | ||||||
|  | from ui_tests_shared.command_test_case_with_app import CommandTestCaseWithApp | ||||||
|  | from ui_tests_shared.decorators import Async | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PingTestCase(CommandTestCaseWithApp): | ||||||
|  |  | ||||||
|  |     @Async.test | ||||||
|  |     async def test_ping(self): | ||||||
|  |         correct_response = self._t.transform('modules.base.pong') | ||||||
|  |         self.assertIsNotNone(correct_response) | ||||||
|  |         self.send_command('ping') | ||||||
|  |  | ||||||
|  |         def check(m: discord.Message): | ||||||
|  |             return m.content == correct_response and m.author.id == 998159802393964594 | ||||||
|  |  | ||||||
|  |         response = await self._bot.wait_for('message', check=check) | ||||||
|  |         self.assertEqual(response.content, correct_response) | ||||||
							
								
								
									
										44
									
								
								kdb-bot/test/ui_tests_tests/ui-tests-tests.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								kdb-bot/test/ui_tests_tests/ui-tests-tests.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | |||||||
|  | { | ||||||
|  |   "ProjectSettings": { | ||||||
|  |     "Name": "ui-tests-tests", | ||||||
|  |     "Version": { | ||||||
|  |       "Major": "0", | ||||||
|  |       "Minor": "0", | ||||||
|  |       "Micro": "0" | ||||||
|  |     }, | ||||||
|  |     "Author": "", | ||||||
|  |     "AuthorEmail": "", | ||||||
|  |     "Description": "", | ||||||
|  |     "LongDescription": "", | ||||||
|  |     "URL": "", | ||||||
|  |     "CopyrightDate": "", | ||||||
|  |     "CopyrightName": "", | ||||||
|  |     "LicenseName": "", | ||||||
|  |     "LicenseDescription": "", | ||||||
|  |     "Dependencies": [ | ||||||
|  |       "cpl-core>=2022.10.0.post9" | ||||||
|  |     ], | ||||||
|  |     "DevDependencies": [ | ||||||
|  |       "cpl-cli>=2022.10.0" | ||||||
|  |     ], | ||||||
|  |     "PythonVersion": ">=3.10.4", | ||||||
|  |     "PythonPath": {}, | ||||||
|  |     "Classifiers": [] | ||||||
|  |   }, | ||||||
|  |   "BuildSettings": { | ||||||
|  |     "ProjectType": "console", | ||||||
|  |     "SourcePath": "", | ||||||
|  |     "OutputPath": "../../dist", | ||||||
|  |     "Main": "ui_tests__tests.main", | ||||||
|  |     "EntryPoint": "ui-tests_tests", | ||||||
|  |     "IncludePackageData": false, | ||||||
|  |     "Included": [], | ||||||
|  |     "Excluded": [ | ||||||
|  |       "*/__pycache__", | ||||||
|  |       "*/logs", | ||||||
|  |       "*/tests" | ||||||
|  |     ], | ||||||
|  |     "PackageData": {}, | ||||||
|  |     "ProjectReferences": [] | ||||||
|  |   } | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user