From 22424519386b55b47f4dc26f6eff54359f283b3d Mon Sep 17 00:00:00 2001 From: Gabryel Reyes Date: Wed, 19 Jun 2024 13:36:28 +0200 Subject: [PATCH 1/6] Added dev folder to gitignore --- .gitignore | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.gitignore b/.gitignore index 3ec7adf..a68c88a 100644 --- a/.gitignore +++ b/.gitignore @@ -165,3 +165,6 @@ cython_debug/ # and can be added to the global gitignore or merged into this file. For a more nuclear # option (not recommended) you can uncomment the following to ignore the entire idea folder. #.idea/ + +# Development Scripts +dev/ \ No newline at end of file From 65a31628471915acd06f468219405d7879741e08 Mon Sep 17 00:00:00 2001 From: Gabryel Reyes Date: Wed, 19 Jun 2024 16:48:41 +0200 Subject: [PATCH 2/6] Added invalid arguments error --- src/pyPolarionCli/ret.py | 1 + 1 file changed, 1 insertion(+) diff --git a/src/pyPolarionCli/ret.py b/src/pyPolarionCli/ret.py index 92b1eb9..c7cf579 100644 --- a/src/pyPolarionCli/ret.py +++ b/src/pyPolarionCli/ret.py @@ -48,6 +48,7 @@ class Ret(IntEnum): OK = 0 ERROR_LOGIN = 1 ERROR_ARGPARSE = 2 # Must be 2 to match the argparse error code. + ERROR_INVALID_ARGUMENTS = 3 ################################################################################ # Functions From 98e1e6fdc8b338d6db3588343bdfbe9548283cfb Mon Sep 17 00:00:00 2001 From: Gabryel Reyes Date: Wed, 19 Jun 2024 15:31:34 +0200 Subject: [PATCH 3/6] Added search command --- src/pyPolarionCli/cmd_search.py | 149 ++++++++++++++++++++++++++++++++ 1 file changed, 149 insertions(+) create mode 100644 src/pyPolarionCli/cmd_search.py diff --git a/src/pyPolarionCli/cmd_search.py b/src/pyPolarionCli/cmd_search.py new file mode 100644 index 0000000..fe2f42d --- /dev/null +++ b/src/pyPolarionCli/cmd_search.py @@ -0,0 +1,149 @@ +"""Search command module of the pyPolarionCli""" + +# BSD 3-Clause License +# +# Copyright (c) 2024, NewTec GmbH +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# 1. Redistributions of source code must retain the above copyright notice, this +# list of conditions and the following disclaimer. +# +# 2. Redistributions in binary form must reproduce the above copyright notice, +# this list of conditions and the following disclaimer in the documentation +# and/or other materials provided with the distribution. +# +# 3. Neither the name of the copyright holder nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICU5LAR PURPOSE ARE +# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +################################################################################ +# Imports +################################################################################ + +import json +import argparse +import logging +from polarion.polarion import Polarion +from polarion.workitem import Workitem +from .ret import Ret + +################################################################################ +# Variables +################################################################################ + +_CMD_NAME = "search" +_OUTPUT_FILE_NAME = "search_results.json" + +################################################################################ +# Classes +################################################################################ + +################################################################################ +# Functions +################################################################################ + + +def register(subparser) -> dict: + """ Register subparser commands for the login module. + + Args: + subparser (obj): the command subparser provided via __main__.py + + Returns: + obj: the command parser of this module + """ + cmd_dict = { + "name": _CMD_NAME, + "handler": _execute + } + + sub_parser_search: argparse.ArgumentParser = \ + subparser.add_parser(_CMD_NAME, + help="Search for Polarion work items.") + + sub_parser_search.add_argument('-p', + '--project', + type=str, + metavar='', + required=True, + help="The ID of the Polarion project to search in.") + + sub_parser_search.add_argument('-q', + '--query', + type=str, + metavar='', + required=True, + help="The query string to search for work items.") + + sub_parser_search.add_argument('-o', + '--output', + type=str, + metavar='', + required=False, + help="The path to output folder to store the search results.") + + return cmd_dict + + +def _execute(args, polarion_client: Polarion) -> Ret: + """ This function servers as entry point for the command 'search'. + It will be stored as callback for this modules subparser command. + + Args: + args (obj): The command line arguments. + polarion_client (obj): The Polarion client object. + + Returns: + bool: The status of the command execution. + """ + ret_status = Ret.ERROR_INVALID_ARGUMENTS + + if ("" != args.project) and ("" != args.query) and (None is not polarion_client): + project_id: str = args.project + query: str = args.query + output_folder = "." + output_dict: dict = { + "project": project_id, + "query": query, + "number_of_results": 0, + "results": [], + } + + if args.output is not None: + output_folder = args.output + + file_path = f"{output_folder}/{project_id}_{_OUTPUT_FILE_NAME}" + + search_result: list[Workitem] = polarion_client.getProject( + project_id).searchWorkitem(query) + + output_dict["number_of_results"] = len(search_result) + + for item in search_result: + item_dict = vars(item).get("__values__") + output_dict["results"].append(item_dict) + + with open(file_path, 'w', encoding="UTF-8") as file: + file.write(json.dumps(output_dict, indent=2)) + + logging.info("Search results stored in %s", file_path) + ret_status = Ret.OK + + return ret_status + +################################################################################ +# Main +################################################################################ From bdcbe3b0473c64557cf272082a397ee483d1019a Mon Sep 17 00:00:00 2001 From: Gabryel Reyes Date: Wed, 19 Jun 2024 16:49:03 +0200 Subject: [PATCH 4/6] Registered search command. made login creds required --- src/pyPolarionCli/__main__.py | 53 ++++++++++++++++++++++++----------- 1 file changed, 37 insertions(+), 16 deletions(-) diff --git a/src/pyPolarionCli/__main__.py b/src/pyPolarionCli/__main__.py index 4b3f29c..583feee 100644 --- a/src/pyPolarionCli/__main__.py +++ b/src/pyPolarionCli/__main__.py @@ -40,14 +40,16 @@ from .version import __version__, __author__, __email__, __repository__, __license__ from .ret import Ret +from .cmd_search import register as cmd_search_register ################################################################################ # Variables ################################################################################ -# Add command modules here -_CMD_MODULES = [ +# Register a command here! +_COMMAND_REG_LIST = [ + cmd_search_register ] PROG_NAME = "pyPolarionCli" @@ -84,18 +86,21 @@ def add_parser() -> argparse.ArgumentParser: '--user', type=str, metavar='', + required=True, help="The user to authenticate with the Polarion server") parser.add_argument('-p', '--password', type=str, metavar='', + required=True, help="The password to authenticate with the Polarion server") parser.add_argument('-s', '--server', type=str, metavar='', + required=True, help="The Polarion server URL to connect to.") parser.add_argument("--version", @@ -108,40 +113,45 @@ def add_parser() -> argparse.ArgumentParser: help="Print full command details before executing the command.\ Enables logs of type INFO and WARNING.") - # to do: Register subparsers once the command files are generated. - return parser -def main() -> int: +def main() -> Ret: """ The program entry point function. Returns: int: System exit status. """ ret_status = Ret.OK + commands = [] - # Parse the command line arguments. + # Create the main parser and add the subparsers. parser = add_parser() + subparser = parser.add_subparsers(required='True', dest="cmd") + + # Register all commands. + for cmd_register in _COMMAND_REG_LIST: + cmd_par_dict = cmd_register(subparser) + commands.append(cmd_par_dict) + + # Parse the command line arguments. args = parser.parse_args() + # Check if the command line arguments are valid. if args is None: ret_status = Ret.ERROR_ARGPARSE + parser.print_help() else: # If the verbose flag is set, change the default logging level. if args.verbose: logging.basicConfig(level=logging.INFO) - - logging.info("Program arguments: ") - - for arg in vars(args): - logging.info("* %s = %s", arg, vars(args)[arg]) + logging.info("Program arguments: ") + for arg in vars(args): + logging.info("* %s = %s", arg, vars(args)[arg]) # Create a Polarion client which communicates to the Polarion server. # A broad exception has to be caught since the specific Exception Type can't be accessed. try: - # to do: remove the "pylint: disable" once the client is used. - # pylint: disable=unused-variable client = Polarion(polarion_url=args.server, user=args.user, password=args.password, @@ -152,9 +162,20 @@ def main() -> int: ret_status = Ret.ERROR_LOGIN if Ret.OK == ret_status: - pass - # to do: Call the respective Function and pass the - # Polarion Client once the Commands are implemented + handler = None + + # Find the command handler. + for command in commands: + if command["name"] == args.cmd: + handler = command["handler"] + break + + # Execute the command. + if handler is not None: + ret_status = handler(args, client) + else: + logging.error("Command '%s' not found!", args.cmd) + ret_status = Ret.ERROR_INVALID_ARGUMENTS return ret_status From cf77d005ca85dc68bf59b808819dc3d9c0908ca6 Mon Sep 17 00:00:00 2001 From: Gabryel Reyes Date: Thu, 20 Jun 2024 09:50:12 +0200 Subject: [PATCH 5/6] Fixed recursive handling of WorkItems --- src/pyPolarionCli/cmd_search.py | 90 +++++++++++++++++++++++++++++++-- 1 file changed, 86 insertions(+), 4 deletions(-) diff --git a/src/pyPolarionCli/cmd_search.py b/src/pyPolarionCli/cmd_search.py index fe2f42d..20b6216 100644 --- a/src/pyPolarionCli/cmd_search.py +++ b/src/pyPolarionCli/cmd_search.py @@ -36,6 +36,7 @@ import json import argparse import logging +from datetime import date, datetime from polarion.polarion import Polarion from polarion.workitem import Workitem from .ret import Ret @@ -56,6 +57,75 @@ ################################################################################ +def _handle_object_with_dict(obj_with_dict) -> dict: + """ + Handle an object with a __dict__ attribute. + + Args: + obj_with_dict (obj): The object to handle. + + Returns: + dict: The dictionary representation of the object. + """ + parsed_dict = {} + for _, subvalue in obj_with_dict.__dict__.items(): + for subkey in subvalue: + _parse_attributes_recursively( + parsed_dict, + subvalue[subkey], + subkey) + return parsed_dict + + +def _parse_attributes_recursively(output_dict: dict, value, key): + """ + Parse the attributes of Python objects recursively and store them in a dictionary. + + Args: + output_dict (dict): The dictionary to store the parsed attributes. + value (obj): The value to parse. + key (str): The key of the value in the dictionary. + + Returns: + None + """ + attribute_value = None + + # Check if the value is a datetime or date object + if isinstance(value, (datetime, date)): + attribute_value = value.isoformat() + + # Check if the value is a list + elif isinstance(value, list): + sublist = [] + for element in value: + # Check if the element is an object with a __dict__ attribute + if hasattr(element, "__dict__"): + sublist.append(_handle_object_with_dict(element)) + + # Check if the element is a list + elif isinstance(element, list): + raise RuntimeWarning("List in List") + + # element is a simple value + else: + sublist.append(element) + + # Store the list in the attribute value + attribute_value = sublist + + # Check if the value is an object with a __dict__ attribute + elif hasattr(value, "__dict__"): + attribute_value = _handle_object_with_dict(value) + + # value is a simple value + else: + attribute_value = value + + # Store the attribute value in the output dictionary + output_dict[key] = attribute_value + + def register(subparser) -> dict: """ Register subparser commands for the login module. @@ -128,14 +198,26 @@ def _execute(args, polarion_client: Polarion) -> Ret: file_path = f"{output_folder}/{project_id}_{_OUTPUT_FILE_NAME}" search_result: list[Workitem] = polarion_client.getProject( - project_id).searchWorkitem(query) + project_id).searchWorkitemFullItem(query) output_dict["number_of_results"] = len(search_result) - for item in search_result: - item_dict = vars(item).get("__values__") - output_dict["results"].append(item_dict) + # Iterate over the search results and store them in the output dictionary. + for workitem in search_result: + workitem_dict = {} + + # Parse the attributes of the work item recursively. + # Internal _polariom_item attribute is used to access the work item attributes. + # pylint: disable=protected-access + for _, value in workitem._polarion_item.__dict__.items(): + for key in value: + _parse_attributes_recursively( + workitem_dict, value[key], key) + + # Append the work item dictionary to the results list. + output_dict["results"].append(workitem_dict) + # Store the search results in a JSON file. with open(file_path, 'w', encoding="UTF-8") as file: file.write(json.dumps(output_dict, indent=2)) From 903b2ee6ecf73ed7c8b3bf5e0b6e3539f9d64a04 Mon Sep 17 00:00:00 2001 From: Gabryel Reyes Date: Thu, 20 Jun 2024 10:34:13 +0200 Subject: [PATCH 6/6] Fixed review findings --- src/pyPolarionCli/__main__.py | 2 +- src/pyPolarionCli/cmd_search.py | 89 ++++++++++++++++++--------------- src/pyPolarionCli/ret.py | 1 + 3 files changed, 51 insertions(+), 41 deletions(-) diff --git a/src/pyPolarionCli/__main__.py b/src/pyPolarionCli/__main__.py index 583feee..67cf562 100644 --- a/src/pyPolarionCli/__main__.py +++ b/src/pyPolarionCli/__main__.py @@ -127,7 +127,7 @@ def main() -> Ret: # Create the main parser and add the subparsers. parser = add_parser() - subparser = parser.add_subparsers(required='True', dest="cmd") + subparser = parser.add_subparsers(required=True, dest="cmd") # Register all commands. for cmd_register in _COMMAND_REG_LIST: diff --git a/src/pyPolarionCli/cmd_search.py b/src/pyPolarionCli/cmd_search.py index 20b6216..19a3aed 100644 --- a/src/pyPolarionCli/cmd_search.py +++ b/src/pyPolarionCli/cmd_search.py @@ -38,6 +38,7 @@ import logging from datetime import date, datetime from polarion.polarion import Polarion +from polarion.project import Project from polarion.workitem import Workitem from .ret import Ret @@ -57,7 +58,7 @@ ################################################################################ -def _handle_object_with_dict(obj_with_dict) -> dict: +def _handle_object_with_dict(obj_with_dict: object) -> dict: """ Handle an object with a __dict__ attribute. @@ -67,7 +68,7 @@ def _handle_object_with_dict(obj_with_dict) -> dict: Returns: dict: The dictionary representation of the object. """ - parsed_dict = {} + parsed_dict: dict = {} for _, subvalue in obj_with_dict.__dict__.items(): for subkey in subvalue: _parse_attributes_recursively( @@ -77,7 +78,7 @@ def _handle_object_with_dict(obj_with_dict) -> dict: return parsed_dict -def _parse_attributes_recursively(output_dict: dict, value, key): +def _parse_attributes_recursively(output_dict: dict, value: object, key: str) -> None: """ Parse the attributes of Python objects recursively and store them in a dictionary. @@ -97,7 +98,7 @@ def _parse_attributes_recursively(output_dict: dict, value, key): # Check if the value is a list elif isinstance(value, list): - sublist = [] + sublist: list = [] for element in value: # Check if the element is an object with a __dict__ attribute if hasattr(element, "__dict__"): @@ -135,7 +136,7 @@ def register(subparser) -> dict: Returns: obj: the command parser of this module """ - cmd_dict = { + cmd_dict: dict = { "name": _CMD_NAME, "handler": _execute } @@ -170,7 +171,7 @@ def register(subparser) -> dict: def _execute(args, polarion_client: Polarion) -> Ret: """ This function servers as entry point for the command 'search'. - It will be stored as callback for this modules subparser command. + It will be stored as callback for this module's subparser command. Args: args (obj): The command line arguments. @@ -179,15 +180,13 @@ def _execute(args, polarion_client: Polarion) -> Ret: Returns: bool: The status of the command execution. """ - ret_status = Ret.ERROR_INVALID_ARGUMENTS + ret_status: Ret = Ret.ERROR_INVALID_ARGUMENTS if ("" != args.project) and ("" != args.query) and (None is not polarion_client): - project_id: str = args.project - query: str = args.query - output_folder = "." + output_folder: str = "." output_dict: dict = { - "project": project_id, - "query": query, + "project": args.project, + "query": args.query, "number_of_results": 0, "results": [], } @@ -195,34 +194,44 @@ def _execute(args, polarion_client: Polarion) -> Ret: if args.output is not None: output_folder = args.output - file_path = f"{output_folder}/{project_id}_{_OUTPUT_FILE_NAME}" - - search_result: list[Workitem] = polarion_client.getProject( - project_id).searchWorkitemFullItem(query) - - output_dict["number_of_results"] = len(search_result) - - # Iterate over the search results and store them in the output dictionary. - for workitem in search_result: - workitem_dict = {} - - # Parse the attributes of the work item recursively. - # Internal _polariom_item attribute is used to access the work item attributes. - # pylint: disable=protected-access - for _, value in workitem._polarion_item.__dict__.items(): - for key in value: - _parse_attributes_recursively( - workitem_dict, value[key], key) - - # Append the work item dictionary to the results list. - output_dict["results"].append(workitem_dict) - - # Store the search results in a JSON file. - with open(file_path, 'w', encoding="UTF-8") as file: - file.write(json.dumps(output_dict, indent=2)) - - logging.info("Search results stored in %s", file_path) - ret_status = Ret.OK + file_path: str = f"{output_folder}/{output_dict['project']}_{_OUTPUT_FILE_NAME}" + + try: + # Get the project object from the Polarion client. + project: Project = polarion_client.getProject( + output_dict['project']) + # Exception of type Exception is raised when the project does not exist. + except Exception as ex: # pylint: disable=broad-except + logging.error("%s", ex) + ret_status = Ret.ERROR_SEARCH_FAILED + else: + # Search for work items in the project. + search_result: list[Workitem] = project.searchWorkitemFullItem( + output_dict['query']) + + output_dict["number_of_results"] = len(search_result) + + # Iterate over the search results and store them in the output dictionary. + for workitem in search_result: + workitem_dict: dict = {} + + # Parse the attributes of the work item recursively. + # Internal _polarion_item attribute is used to access the work item attributes. + # pylint: disable=protected-access + for _, value in workitem._polarion_item.__dict__.items(): + for key in value: + _parse_attributes_recursively( + workitem_dict, value[key], key) + + # Append the work item dictionary to the results list. + output_dict["results"].append(workitem_dict) + + # Store the search results in a JSON file. + with open(file_path, 'w', encoding="UTF-8") as file: + file.write(json.dumps(output_dict, indent=2)) + + logging.info("Search results stored in %s", file_path) + ret_status = Ret.OK return ret_status diff --git a/src/pyPolarionCli/ret.py b/src/pyPolarionCli/ret.py index c7cf579..ef0c8da 100644 --- a/src/pyPolarionCli/ret.py +++ b/src/pyPolarionCli/ret.py @@ -49,6 +49,7 @@ class Ret(IntEnum): ERROR_LOGIN = 1 ERROR_ARGPARSE = 2 # Must be 2 to match the argparse error code. ERROR_INVALID_ARGUMENTS = 3 + ERROR_SEARCH_FAILED = 4 ################################################################################ # Functions