Compare commits
	
		
			4 Commits
		
	
	
		
			1.1.6
			...
			77d723b9da
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 77d723b9da | |||
| 75c316f2d2 | |||
| de78cec96c | |||
| 9de66d4fd4 | 
| @@ -1,2 +1,14 @@ | ||||
| # 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", | ||||
|       "stats": "src/modules/stats/stats.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", | ||||
|       "post-build": "tools/post_build/post-build.json", | ||||
|       "set-version": "tools/set_version/set-version.json" | ||||
|     }, | ||||
|     "Scripts": { | ||||
|       "test": "export $(cat test/ui_tests/.env); export PYTHONPATH=$PWD/src:$PYTHONPATH; cpl run ui-tests", | ||||
|       "sv": "cpl set-version $ARGS", | ||||
|       "set-version": "cpl run set-version $ARGS; echo '';", | ||||
|  | ||||
|       "gv": "cpl get-version", | ||||
|       "get-version": "export VERSION=$(cpl run get-version); echo $VERSION;", | ||||
|  | ||||
|       "pre-build": "cpl set-version $ARGS", | ||||
|       "post-build": "cpl run post-build", | ||||
|  | ||||
|       "pre-prod": "cpl build", | ||||
|       "prod": "export KDB_ENVIRONMENT=production; export KDB_NAME=KDB-Prod; cpl start;", | ||||
|  | ||||
|       "pre-stage": "cpl build", | ||||
|       "stage": "export KDB_ENVIRONMENT=staging; export KDB_NAME=KDB-Stage; cpl start;", | ||||
|  | ||||
|       "pre-dev": "cpl build", | ||||
|       "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) .;", | ||||
|       "dc-up": "docker-compose up -d", | ||||
|       "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