pyegeria 5.3.9.9.3__py3-none-any.whl → 5.5.3.3__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of pyegeria might be problematic. Click here for more details.

Files changed (272) hide show
  1. commands/__init__.py +24 -0
  2. commands/cat/Dr-Egeria_md-orig.py +2 -2
  3. commands/cat/__init__.py +1 -17
  4. commands/cat/collection_actions.py +197 -0
  5. commands/cat/dr_egeria_command_help.py +372 -0
  6. commands/cat/dr_egeria_jupyter.py +7 -7
  7. commands/cat/dr_egeria_md.py +27 -182
  8. commands/cat/exp_list_glossaries.py +11 -14
  9. commands/cat/get_asset_graph.py +37 -267
  10. commands/cat/{get_collection.py → get_collection_tree.py} +10 -18
  11. commands/cat/get_project_dependencies.py +14 -14
  12. commands/cat/get_project_structure.py +15 -14
  13. commands/cat/get_tech_type_elements.py +16 -116
  14. commands/cat/glossary_actions.py +145 -298
  15. commands/cat/list_assets.py +3 -11
  16. commands/cat/list_cert_types.py +17 -63
  17. commands/cat/list_collections.py +46 -138
  18. commands/cat/list_deployed_catalogs.py +15 -27
  19. commands/cat/list_deployed_database_schemas.py +27 -43
  20. commands/cat/list_deployed_databases.py +16 -31
  21. commands/cat/list_deployed_servers.py +35 -54
  22. commands/cat/list_glossaries.py +18 -17
  23. commands/cat/list_projects.py +10 -12
  24. commands/cat/list_tech_type_elements.py +21 -37
  25. commands/cat/list_tech_types.py +13 -25
  26. commands/cat/list_terms.py +38 -79
  27. commands/cat/list_todos.py +4 -11
  28. commands/cat/list_user_ids.py +3 -10
  29. commands/cat/my_reports.py +559 -0
  30. commands/cat/run_report.py +394 -0
  31. commands/cat/run_report_orig.py +528 -0
  32. commands/cli/egeria.py +222 -247
  33. commands/cli/egeria_cat.py +68 -81
  34. commands/cli/egeria_my.py +13 -0
  35. commands/cli/egeria_ops.py +69 -74
  36. commands/cli/egeria_tech.py +17 -93
  37. commands/cli/ops_config.py +3 -6
  38. commands/{cat/list_categories.py → deprecated/list_data_designer.py} +53 -64
  39. commands/{cat/list_data_structures.py → deprecated/list_data_structures_full.py} +3 -6
  40. commands/deprecated/old_get_asset_graph.py +315 -0
  41. commands/my/__init__.py +0 -2
  42. commands/my/list_my_profile.py +27 -34
  43. commands/my/list_my_roles.py +1 -7
  44. commands/my/monitor_my_todos.py +1 -7
  45. commands/my/monitor_open_todos.py +6 -7
  46. commands/my/todo_actions.py +4 -5
  47. commands/ops/__init__.py +0 -2
  48. commands/ops/gov_server_actions.py +17 -21
  49. commands/ops/list_archives.py +17 -38
  50. commands/ops/list_catalog_targets.py +33 -40
  51. commands/ops/load_archive.py +35 -26
  52. commands/ops/{monitor_engine_activity_c.py → monitor_active_engine_activity.py} +51 -82
  53. commands/ops/{monitor_integ_daemon_status.py → monitor_daemon_status.py} +35 -55
  54. commands/ops/monitor_engine_activity.py +79 -77
  55. commands/ops/{monitor_gov_eng_status.py → monitor_engine_status.py} +10 -7
  56. commands/ops/monitor_platform_status.py +38 -50
  57. commands/ops/monitor_server_startup.py +6 -11
  58. commands/ops/monitor_server_status.py +7 -11
  59. commands/ops/orig_monitor_server_list.py +8 -8
  60. commands/ops/orig_monitor_server_status.py +1 -5
  61. commands/ops/refresh_integration_daemon.py +5 -5
  62. commands/ops/restart_integration_daemon.py +5 -5
  63. commands/ops/table_integ_daemon_status.py +6 -6
  64. commands/ops/x_engine_actions.py +7 -7
  65. commands/tech/__init__.py +0 -2
  66. commands/tech/{generic_actions.py → element_actions.py} +6 -11
  67. commands/tech/get_element_info.py +20 -29
  68. commands/tech/get_guid_info.py +23 -42
  69. commands/tech/get_tech_details.py +20 -35
  70. commands/tech/get_tech_type_template.py +28 -39
  71. commands/tech/list_all_om_type_elements.py +24 -30
  72. commands/tech/list_all_om_type_elements_x.py +22 -28
  73. commands/tech/list_all_related_elements.py +19 -28
  74. commands/tech/list_anchored_elements.py +22 -30
  75. commands/tech/list_asset_types.py +19 -24
  76. commands/tech/list_elements_by_classification_by_property_value.py +26 -32
  77. commands/tech/list_elements_by_property_value.py +19 -25
  78. commands/tech/list_elements_by_property_value_x.py +20 -28
  79. commands/tech/list_elements_for_classification.py +28 -41
  80. commands/tech/list_gov_action_processes.py +16 -27
  81. commands/tech/list_information_supply_chains.py +22 -30
  82. commands/tech/list_registered_services.py +14 -26
  83. commands/tech/list_related_elements_with_prop_value.py +15 -25
  84. commands/tech/list_related_specification.py +1 -4
  85. commands/tech/list_relationship_types.py +15 -25
  86. commands/tech/list_relationships.py +20 -36
  87. commands/tech/list_solution_blueprints.py +28 -33
  88. commands/tech/list_solution_components.py +23 -29
  89. commands/tech/list_solution_roles.py +21 -32
  90. commands/tech/list_tech_templates.py +51 -54
  91. commands/tech/list_valid_metadata_values.py +5 -9
  92. commands/tech/table_tech_templates.py +2 -6
  93. commands/tech/x_list_related_elements.py +1 -4
  94. examples/GeoSpatial Products Example.py +524 -0
  95. examples/Jupyter Notebooks/P-egeria-server-config.ipynb +2137 -0
  96. examples/Jupyter Notebooks/README.md +2 -0
  97. examples/Jupyter Notebooks/common/P-environment-check.ipynb +115 -0
  98. examples/Jupyter Notebooks/common/__init__.py +14 -0
  99. examples/Jupyter Notebooks/common/common-functions.ipynb +4694 -0
  100. examples/Jupyter Notebooks/common/environment-check.ipynb +52 -0
  101. examples/Jupyter Notebooks/common/globals.ipynb +184 -0
  102. examples/Jupyter Notebooks/common/globals.py +154 -0
  103. examples/Jupyter Notebooks/common/orig_globals.py +152 -0
  104. examples/format_sets/all_format_sets.json +910 -0
  105. examples/format_sets/custom_format_sets.json +268 -0
  106. examples/format_sets/subset_format_sets.json +187 -0
  107. examples/format_sets_save_load_example.py +291 -0
  108. examples/jacquard_data_sets.py +129 -0
  109. examples/output_formats_example.py +193 -0
  110. examples/test_jacquard_data_sets.py +54 -0
  111. examples/test_jacquard_data_sets_scenarios.py +94 -0
  112. md_processing/__init__.py +90 -0
  113. md_processing/command_dispatcher.py +33 -0
  114. md_processing/command_mapping.py +221 -0
  115. md_processing/data/commands/commands_data_designer.json +537 -0
  116. md_processing/data/commands/commands_external_reference.json +733 -0
  117. md_processing/data/commands/commands_feedback.json +155 -0
  118. md_processing/data/commands/commands_general.json +204 -0
  119. md_processing/data/commands/commands_glossary.json +218 -0
  120. md_processing/data/commands/commands_governance.json +3678 -0
  121. md_processing/data/commands/commands_product_manager.json +865 -0
  122. md_processing/data/commands/commands_project.json +642 -0
  123. md_processing/data/commands/commands_solution_architect.json +366 -0
  124. md_processing/data/commands.json +17568 -0
  125. md_processing/data/commands_working.json +30641 -0
  126. md_processing/data/gened_report_specs.py +6584 -0
  127. md_processing/data/generated_format_sets.json +6533 -0
  128. md_processing/data/generated_format_sets_old.json +4137 -0
  129. md_processing/data/generated_format_sets_old.py +45 -0
  130. md_processing/dr_egeria.py +182 -0
  131. md_processing/md_commands/__init__.py +3 -0
  132. md_processing/md_commands/data_designer_commands.py +1276 -0
  133. md_processing/md_commands/ext_ref_commands.py +530 -0
  134. md_processing/md_commands/feedback_commands.py +726 -0
  135. md_processing/md_commands/glossary_commands.py +684 -0
  136. md_processing/md_commands/governance_officer_commands.py +600 -0
  137. md_processing/md_commands/product_manager_commands.py +1266 -0
  138. md_processing/md_commands/project_commands.py +383 -0
  139. md_processing/md_commands/solution_architect_commands.py +1184 -0
  140. md_processing/md_commands/view_commands.py +295 -0
  141. md_processing/md_processing_utils/__init__.py +4 -0
  142. md_processing/md_processing_utils/common_md_proc_utils.py +1249 -0
  143. md_processing/md_processing_utils/common_md_utils.py +578 -0
  144. md_processing/md_processing_utils/determine_width.py +103 -0
  145. md_processing/md_processing_utils/extraction_utils.py +547 -0
  146. md_processing/md_processing_utils/gen_report_specs.py +643 -0
  147. md_processing/md_processing_utils/generate_dr_help.py +193 -0
  148. md_processing/md_processing_utils/generate_md_cmd_templates.py +144 -0
  149. md_processing/md_processing_utils/generate_md_templates.py +83 -0
  150. md_processing/md_processing_utils/md_processing_constants.py +1228 -0
  151. md_processing/md_processing_utils/message_constants.py +19 -0
  152. pyegeria/__init__.py +201 -443
  153. pyegeria/core/__init__.py +40 -0
  154. pyegeria/core/_base_platform_client.py +574 -0
  155. pyegeria/core/_base_server_client.py +573 -0
  156. pyegeria/core/_exceptions.py +457 -0
  157. pyegeria/core/_globals.py +60 -0
  158. pyegeria/core/_server_client.py +6073 -0
  159. pyegeria/core/_validators.py +257 -0
  160. pyegeria/core/config.py +654 -0
  161. pyegeria/{create_tech_guid_lists.py → core/create_tech_guid_lists.py} +0 -1
  162. pyegeria/core/load_config.py +37 -0
  163. pyegeria/core/logging_configuration.py +207 -0
  164. pyegeria/core/mcp_adapter.py +144 -0
  165. pyegeria/core/mcp_server.py +212 -0
  166. pyegeria/core/utils.py +405 -0
  167. pyegeria/deprecated/__init__.py +0 -0
  168. pyegeria/{_client.py → deprecated/_client.py} +62 -24
  169. pyegeria/{_deprecated_gov_engine.py → deprecated/_deprecated_gov_engine.py} +16 -16
  170. pyegeria/{classification_manager_omvs.py → deprecated/classification_manager_omvs.py} +1988 -1878
  171. pyegeria/deprecated/output_formatter_with_machine_keys.py +1127 -0
  172. pyegeria/{runtime_manager_omvs.py → deprecated/runtime_manager_omvs.py} +216 -229
  173. pyegeria/{valid_metadata_omvs.py → deprecated/valid_metadata_omvs.py} +93 -93
  174. pyegeria/{x_action_author_omvs.py → deprecated/x_action_author_omvs.py} +2 -3
  175. pyegeria/egeria_cat_client.py +25 -51
  176. pyegeria/egeria_client.py +140 -98
  177. pyegeria/egeria_config_client.py +48 -24
  178. pyegeria/egeria_tech_client.py +170 -83
  179. pyegeria/models/__init__.py +150 -0
  180. pyegeria/models/collection_models.py +168 -0
  181. pyegeria/models/models.py +654 -0
  182. pyegeria/omvs/__init__.py +84 -0
  183. pyegeria/omvs/action_author.py +342 -0
  184. pyegeria/omvs/actor_manager.py +5980 -0
  185. pyegeria/omvs/asset_catalog.py +842 -0
  186. pyegeria/omvs/asset_maker.py +2736 -0
  187. pyegeria/omvs/automated_curation.py +4403 -0
  188. pyegeria/omvs/classification_manager.py +11213 -0
  189. pyegeria/omvs/collection_manager.py +5780 -0
  190. pyegeria/omvs/community_matters_omvs.py +468 -0
  191. pyegeria/{core_omag_server_config.py → omvs/core_omag_server_config.py} +157 -157
  192. pyegeria/{data_designer_omvs.py → omvs/data_designer.py} +1991 -1691
  193. pyegeria/omvs/data_discovery.py +869 -0
  194. pyegeria/omvs/data_engineer.py +372 -0
  195. pyegeria/omvs/digital_business.py +1133 -0
  196. pyegeria/omvs/external_links.py +1752 -0
  197. pyegeria/omvs/feedback_manager.py +834 -0
  198. pyegeria/{full_omag_server_config.py → omvs/full_omag_server_config.py} +73 -69
  199. pyegeria/omvs/glossary_manager.py +3231 -0
  200. pyegeria/omvs/governance_officer.py +3009 -0
  201. pyegeria/omvs/lineage_linker.py +314 -0
  202. pyegeria/omvs/location_arena.py +1525 -0
  203. pyegeria/omvs/metadata_expert.py +668 -0
  204. pyegeria/omvs/metadata_explorer_omvs.py +2943 -0
  205. pyegeria/omvs/my_profile.py +1042 -0
  206. pyegeria/omvs/notification_manager.py +358 -0
  207. pyegeria/omvs/people_organizer.py +394 -0
  208. pyegeria/{platform_services.py → omvs/platform_services.py} +113 -193
  209. pyegeria/omvs/product_manager.py +1825 -0
  210. pyegeria/omvs/project_manager.py +1907 -0
  211. pyegeria/omvs/reference_data.py +1140 -0
  212. pyegeria/omvs/registered_info.py +334 -0
  213. pyegeria/omvs/runtime_manager.py +2817 -0
  214. pyegeria/omvs/schema_maker.py +446 -0
  215. pyegeria/{server_operations.py → omvs/server_operations.py} +27 -26
  216. pyegeria/omvs/solution_architect.py +6490 -0
  217. pyegeria/omvs/specification_properties.py +37 -0
  218. pyegeria/omvs/subject_area.py +1042 -0
  219. pyegeria/omvs/template_manager_omvs.py +236 -0
  220. pyegeria/omvs/time_keeper.py +1761 -0
  221. pyegeria/omvs/valid_metadata.py +3221 -0
  222. pyegeria/omvs/valid_metadata_lists.py +37 -0
  223. pyegeria/omvs/valid_type_lists.py +37 -0
  224. pyegeria/view/__init__.py +28 -0
  225. pyegeria/view/_output_format_models.py +514 -0
  226. pyegeria/view/_output_formats.py +14 -0
  227. pyegeria/view/base_report_formats.py +2719 -0
  228. pyegeria/view/dr_egeria_reports.py +56 -0
  229. pyegeria/view/format_set_executor.py +397 -0
  230. pyegeria/{md_processing_utils.py → view/md_processing_utils.py} +5 -5
  231. pyegeria/{mermaid_utilities.py → view/mermaid_utilities.py} +2 -154
  232. pyegeria/view/output_formatter.py +1297 -0
  233. pyegeria-5.5.3.3.dist-info/METADATA +218 -0
  234. pyegeria-5.5.3.3.dist-info/RECORD +241 -0
  235. {pyegeria-5.3.9.9.3.dist-info → pyegeria-5.5.3.3.dist-info}/WHEEL +2 -1
  236. pyegeria-5.5.3.3.dist-info/entry_points.txt +103 -0
  237. pyegeria-5.5.3.3.dist-info/top_level.txt +4 -0
  238. commands/cat/.DS_Store +0 -0
  239. commands/cat/README.md +0 -16
  240. commands/cli/txt_custom_v2.tcss +0 -19
  241. commands/my/README.md +0 -17
  242. commands/ops/README.md +0 -24
  243. commands/ops/monitor_asset_events.py +0 -108
  244. commands/tech/README.md +0 -24
  245. pyegeria/.DS_Store +0 -0
  246. pyegeria/README.md +0 -35
  247. pyegeria/_globals.py +0 -47
  248. pyegeria/_validators.py +0 -385
  249. pyegeria/asset_catalog_omvs.py +0 -864
  250. pyegeria/automated_curation_omvs.py +0 -3765
  251. pyegeria/collection_manager_omvs.py +0 -2744
  252. pyegeria/dr.egeria spec.md +0 -9
  253. pyegeria/egeria_my_client.py +0 -56
  254. pyegeria/feedback_manager_omvs.py +0 -4573
  255. pyegeria/glossary_browser_omvs.py +0 -3728
  256. pyegeria/glossary_manager_omvs.py +0 -2440
  257. pyegeria/m_test.py +0 -118
  258. pyegeria/md_processing_helpers.py +0 -58
  259. pyegeria/md_processing_utils_orig.py +0 -1103
  260. pyegeria/metadata_explorer_omvs.py +0 -2326
  261. pyegeria/my_profile_omvs.py +0 -1022
  262. pyegeria/output_formatter.py +0 -389
  263. pyegeria/project_manager_omvs.py +0 -1933
  264. pyegeria/registered_info.py +0 -167
  265. pyegeria/solution_architect_omvs.py +0 -2156
  266. pyegeria/template_manager_omvs.py +0 -1414
  267. pyegeria/utils.py +0 -197
  268. pyegeria-5.3.9.9.3.dist-info/METADATA +0 -72
  269. pyegeria-5.3.9.9.3.dist-info/RECORD +0 -143
  270. pyegeria-5.3.9.9.3.dist-info/entry_points.txt +0 -99
  271. /pyegeria/{_exceptions.py → deprecated/_exceptions.py} +0 -0
  272. {pyegeria-5.3.9.9.3.dist-info → pyegeria-5.5.3.3.dist-info/licenses}/LICENSE +0 -0
@@ -0,0 +1,394 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ SPDX-License-Identifier: Apache-2.0
4
+ Copyright Contributors to the ODPi Egeria project.
5
+
6
+ Generic list command that wraps exec_report_spec and adds TABLE output.
7
+
8
+ - TABLE: renders results as a paged Rich table; table title comes from get_report_spec_heading().
9
+ - MD, REPORT, FORM, HTML, MERMAID, LIST: writes the output to a timestamped file in the outbox.
10
+
11
+ Environment variables used (with defaults aligned to other commands):
12
+ - EGERIA_VIEW_SERVER, EGERIA_VIEW_SERVER_URL
13
+ - EGERIA_USER, EGERIA_USER_PASSWORD
14
+ - EGERIA_WIDTH, EGERIA_JUPYTER
15
+ - EGERIA_ROOT_PATH, EGERIA_OUTBOX_PATH
16
+ """
17
+ from __future__ import annotations
18
+
19
+ import argparse
20
+ import json
21
+ import os
22
+ import sys
23
+ import time
24
+ from typing import Any, Dict, List, Mapping, Sequence
25
+
26
+ from rich import box
27
+ from rich.console import Console
28
+ from rich.table import Table
29
+ from rich.prompt import Prompt
30
+
31
+ from pyegeria.core.config import settings
32
+ from pyegeria.core._exceptions import PyegeriaException, print_exception_response
33
+ from pyegeria.view.base_report_formats import get_report_spec_heading, select_report_spec
34
+ from pyegeria.view.format_set_executor import exec_report_spec
35
+ EGERIA_USER = os.environ.get("EGERIA_USER", "erinoverview")
36
+ EGERIA_USER_PASSWORD = os.environ.get("EGERIA_USER_PASSWORD", "secret")
37
+
38
+
39
+ app_config = settings.Environment
40
+ # config_logging()
41
+
42
+ TEXT_FILE_FORMATS = {"MD", "REPORT", "FORM", "HTML", "MERMAID", "LIST", "DICT", "JSON"}
43
+
44
+
45
+ def list_generic(
46
+ report_spec: str,
47
+ *,
48
+ output_format: str = "TABLE",
49
+ params: Dict[str, Any] | None = None,
50
+ view_server: str = app_config.egeria_view_server,
51
+ view_url: str = app_config.egeria_view_server_url,
52
+ user: str = EGERIA_USER,
53
+ user_pass: str = EGERIA_USER_PASSWORD,
54
+ jupyter: bool = app_config.egeria_jupyter,
55
+ width: int = app_config.console_width,
56
+ prompt_missing: bool = False,
57
+ write_file: bool = False,
58
+ render_table: bool = False,
59
+ table_caption: str | None = None,
60
+ use_pager: bool = True,
61
+ ) -> Dict[str, Any]:
62
+ """
63
+ Execute a report spec and return results, usable both as a library function and CLI.
64
+
65
+ Required vs Optional parameters are derived from the report spec's action:
66
+ - Required: fmt.action.required_params (or action.user_params) not supplied by spec_params.
67
+ - Optional: fmt.action.optional_params.
68
+
69
+ Behavior:
70
+ - If prompt_missing is True, will interactively Prompt.ask for any missing required params.
71
+ - If prompt_missing is False and required params are missing, raises ValueError.
72
+ - For output_format == "TABLE": returns a JSON payload for table rendering; no file I/O by default.
73
+ - If render_table=True, this function will also render a Rich table immediately (useful for remote/programmatic calls).
74
+ You can customize rendering via table_caption and use_pager.
75
+ - For narrative formats (MD, REPORT, FORM, MERMAID, LIST, HTML):
76
+ - If write_file=True, writes to outbox and returns the file path in the result.
77
+ - Otherwise returns the text content in the result.
78
+
79
+ Parameters (additional rendering options)
80
+ - render_table: bool (default False) — when True and output_format==TABLE, render the Rich table directly.
81
+ - table_caption: Optional[str] — custom caption; defaults to "View Server '{view_server}' @ Platform - {view_url}".
82
+ - use_pager: bool (default True) — wrap the table output in console.pager().
83
+
84
+ Returns a dict with keys like:
85
+ {"kind": "table"|"text"|"json"|"empty", "data"|"content", "file_path"?, "heading", "required", "optional", "rendered"?}
86
+ """
87
+ params = dict(params or {})
88
+ ofmt = (output_format or "TABLE").upper()
89
+
90
+ # Load format spec to introspect required/optional params
91
+ fmt = select_report_spec(report_spec, ofmt if ofmt not in {"TABLE", "MD", "FORM", "LIST"} else "DICT")
92
+ if not fmt:
93
+ raise ValueError(f"Report spec '{report_spec}' not found for format '{ofmt}'.")
94
+ action = fmt.get("action", {}) or {}
95
+ required_params = action.get("required_params", action.get("user_params", [])) or []
96
+ optional_params = action.get("optional_params", []) or []
97
+ spec_params = action.get("spec_params", {}) or {}
98
+
99
+ # Identify missing required params (not provided by caller and not satisfied by spec_params)
100
+ missing = [p for p in required_params if params.get(p) in (None, "") and p not in spec_params]
101
+
102
+ if missing:
103
+ if prompt_missing:
104
+ # Prompt user for each missing required param
105
+ for p in missing:
106
+ val = Prompt.ask(f"Enter value for required parameter '{p}'")
107
+ params[p] = val
108
+ else:
109
+ raise ValueError(f"Missing required parameter(s): {', '.join(missing)}")
110
+
111
+ # Map only TABLE to DICT for data retrieval; pass through other formats
112
+ # mapped_format = "DICT" if ofmt == "TABLE" else ofmt
113
+ mapped_format = ofmt
114
+ try:
115
+ # Execute
116
+ result = exec_report_spec(
117
+ report_spec,
118
+ output_format=mapped_format,
119
+ params=params,
120
+ view_server=view_server,
121
+ view_url=view_url,
122
+ user=user,
123
+ user_pass=user_pass,
124
+ )
125
+ except ValueError as e:
126
+ console = Console(
127
+ style="bold bright_white on black",
128
+ )
129
+ console.print(f"Error executing report: {e}")
130
+ sys.exit(1)
131
+
132
+ # Normalize/augment response
133
+ if result is None:
134
+ return {
135
+ "kind": "empty",
136
+ "heading": get_report_spec_heading(report_spec) or f"Report: {report_spec}",
137
+ "required": required_params,
138
+ "optional": optional_params,
139
+ }
140
+ if result.get("kind") == "empty":
141
+ return {
142
+ "kind": "empty",
143
+ "heading": get_report_spec_heading(report_spec) or f"Report: {report_spec}",
144
+ "required": required_params,
145
+ "optional": optional_params,
146
+ }
147
+
148
+ # For TABLE, optionally render Rich table and return json data for caller
149
+ if ofmt == "TABLE":
150
+ if result.get("kind") == "json":
151
+ heading = get_report_spec_heading(report_spec) or f"Report: {report_spec}"
152
+ data = result.get("data")
153
+ rendered = False
154
+ if render_table:
155
+ console = Console(
156
+ style="bold bright_white on black",
157
+ width=width,
158
+ force_terminal=not jupyter,
159
+ )
160
+ caption = table_caption or f"View Server '{view_server}' @ Platform - {view_url}"
161
+ _render_table(console, heading, caption, data, use_pager=use_pager)
162
+ rendered = True
163
+ return {
164
+ "kind": "table",
165
+ "heading": heading,
166
+ "data": data,
167
+ "required": required_params,
168
+ "optional": optional_params,
169
+ "rendered": rendered,
170
+ }
171
+ # Fallback: unexpected shape
172
+ return {
173
+ "kind": "unknown",
174
+ "raw": result,
175
+ "required": required_params,
176
+ "optional": optional_params,
177
+ }
178
+
179
+ # Narrative formats (all non-TABLE should generate files)
180
+ content: str = ""
181
+ if result.get("kind") == "text":
182
+ # Already-rendered text like Markdown or HTML
183
+ content = str(result.get("content", ""))
184
+ elif result.get("kind") == "json":
185
+ data = result.get("data")
186
+ # For DICT/JSON explicitly produce raw JSON text; otherwise fallback to fenced JSON
187
+ if ofmt in {"DICT", "JSON"}:
188
+ content = json.dumps(data, indent=2)
189
+ else:
190
+ content = "```json\n" + json.dumps(data, indent=2) + "\n```"
191
+ else:
192
+ content = json.dumps(result, indent=2, default=str)
193
+
194
+ if write_file or ofmt in TEXT_FILE_FORMATS:
195
+ # Determine suffix for file naming
196
+ suffix = ofmt
197
+ path = write_output_file(content, report_spec, suffix)
198
+ return {
199
+ "kind": "text" if ofmt not in {"DICT", "JSON"} else "json",
200
+ "content": content,
201
+ "file_path": path,
202
+ "heading": get_report_spec_heading(report_spec) or f"Report: {report_spec}",
203
+ "required": required_params,
204
+ "optional": optional_params,
205
+ }
206
+
207
+ return {
208
+ "kind": "text",
209
+ "content": content,
210
+ "heading": get_report_spec_heading(report_spec) or f"Report: {report_spec}",
211
+ "required": required_params,
212
+ "optional": optional_params,
213
+ }
214
+
215
+
216
+ def _flatten(item: Any, parent_key: str = "", sep: str = ".") -> Dict[str, Any]:
217
+ """Flatten nested dict-like objects into a single level using dot-notation keys."""
218
+ flat: Dict[str, Any] = {}
219
+ if isinstance(item, Mapping):
220
+ for k, v in item.items():
221
+ new_key = f"{parent_key}{sep}{k}" if parent_key else str(k)
222
+ flat.update(_flatten(v, new_key, sep))
223
+ elif isinstance(item, (list, tuple)):
224
+ # Represent short sequences inline; longer ones as counts
225
+ if len(item) <= 3:
226
+ flat[parent_key or "list"] = ", ".join([str(x) for x in item])
227
+ else:
228
+ flat[parent_key or "list"] = f"{len(item)} items"
229
+ else:
230
+ flat[parent_key or "value"] = item
231
+ return flat
232
+
233
+
234
+ def _collect_columns(rows: Sequence[Mapping[str, Any]], max_cols: int = 12) -> List[str]:
235
+ """Collect a union of keys across rows, up to max_cols, keeping a stable ordering."""
236
+ seen: List[str] = []
237
+ for row in rows:
238
+ for k in row.keys():
239
+ if k not in seen:
240
+ seen.append(k)
241
+ if len(seen) >= max_cols:
242
+ return seen
243
+ return seen
244
+
245
+
246
+ def _ensure_rows(data: Any) -> List[Mapping[str, Any]]:
247
+ """Convert result data into a list of row mappings suitable for table rendering."""
248
+ if data is None:
249
+ return []
250
+ if isinstance(data, list):
251
+ rows: List[Mapping[str, Any]] = []
252
+ for item in data:
253
+ if isinstance(item, Mapping):
254
+ rows.append(_flatten(item))
255
+ else:
256
+ rows.append({"value": item})
257
+ return rows
258
+ if isinstance(data, Mapping):
259
+ # If a single object contains a list-like 'elements' or similar, prefer that
260
+ for key in ("elements", "items", "results"):
261
+ if isinstance(data.get(key), list):
262
+ return _ensure_rows(data[key])
263
+ return [_flatten(data)]
264
+ # Fallback: scalar
265
+ return [{"value": data}]
266
+
267
+
268
+ def _render_table(console: Console, title: str, caption: str, data: Any, use_pager: bool = True) -> None:
269
+ rows = _ensure_rows(data)
270
+ if not rows:
271
+ console.print("No results found.")
272
+ return
273
+
274
+ columns = _collect_columns(rows)
275
+
276
+ table = Table(
277
+ title=title,
278
+ style="bright_white on black",
279
+ header_style="bright_white on dark_blue",
280
+ title_style="bold white on black",
281
+ caption_style="white on black",
282
+ show_lines=True,
283
+ box=box.ROUNDED,
284
+ caption=caption,
285
+ expand=True,
286
+ )
287
+ for col in columns:
288
+ table.add_column(str(col))
289
+
290
+ for row in rows:
291
+ table.add_row(*[str(row.get(col, "")) for col in columns])
292
+
293
+ # Page the table if it's large
294
+ if use_pager:
295
+ with console.pager():
296
+ console.print(table)
297
+ else:
298
+ console.print(table)
299
+
300
+
301
+ def write_output_file(content: str, report_spec: str, suffix: str) -> str:
302
+ # Base directory is outbox/report_spec specific folder
303
+ file_path = os.path.join(app_config.pyegeria_root, app_config.dr_egeria_outbox, f"{report_spec}")
304
+ ts = time.strftime("%Y-%m-%d-%H-%M-%S")
305
+ safe_spec = "".join(c for c in report_spec if c.isalnum() or c in ("-", "_", "+", ".", " ")).strip().replace(" ", "_")
306
+ # Determine extension by content type rules
307
+ if suffix in {"MD", "REPORT", "FORM", "MERMAID", "LIST"}:
308
+ ext = ".md"
309
+ elif suffix in {"HTML"}:
310
+ ext = ".html"
311
+ elif suffix in {"DICT", "JSON"}:
312
+ ext = ".json"
313
+ else:
314
+ ext = ".txt"
315
+ file_name = f"{safe_spec}-{ts}-{suffix}{ext}"
316
+ full_file_path = os.path.join(file_path, file_name)
317
+ os.makedirs(os.path.dirname(full_file_path), exist_ok=True)
318
+ with open(full_file_path, "w", encoding="utf-8") as f:
319
+ f.write(content)
320
+ return full_file_path
321
+
322
+
323
+ def main():
324
+ parser = argparse.ArgumentParser(description="Generic list command using exec_report_spec")
325
+ parser.add_argument("--report", dest="report_spec", help="Report spec name to execute (see report specs)")
326
+ parser.add_argument("--output-format", dest="output_format", default="TABLE",
327
+ choices=["TABLE", "MD", "REPORT", "FORM", "HTML", "MERMAID", "LIST", "JSON", "DICT"],
328
+ help="Output rendering format")
329
+ parser.add_argument("--server", dest="server", help="Egeria view server name", default=app_config.egeria_view_server)
330
+ parser.add_argument("--url", dest="url", help="Egeria platform URL", default=app_config.egeria_view_server_url)
331
+ parser.add_argument("--userid", dest="user", help="User Id", default=EGERIA_USER)
332
+ parser.add_argument("--password", dest="password", help="User Password", default=EGERIA_USER_PASSWORD)
333
+ # Allow arbitrary parameters as JSON or repeated key=value
334
+ parser.add_argument("--params-json", dest="params_json", help="JSON string of parameters for the report spec")
335
+ parser.add_argument("--param", dest="params_kv", action="append", default=[], help="Repeated key=value parameters")
336
+
337
+ args = parser.parse_args()
338
+
339
+ # Prompt for report spec if not provided
340
+ report_spec = args.report_spec or Prompt.ask("Enter the report spec to execute").strip()
341
+ output_format = (args.output_format or "TABLE").upper()
342
+
343
+ # Build params dict
344
+ params: Dict[str, Any] = {}
345
+ if args.params_json:
346
+ try:
347
+ parsed = json.loads(args.params_json)
348
+ if isinstance(parsed, dict):
349
+ params.update(parsed)
350
+ except Exception:
351
+ pass
352
+ for kv in (args.params_kv or []):
353
+ if "=" in kv:
354
+ k, v = kv.split("=", 1)
355
+ params[k.strip()] = v.strip()
356
+
357
+ console = Console(
358
+ style="bold bright_white on black",
359
+ width=app_config.console_width,
360
+ force_terminal=not app_config.egeria_jupyter,
361
+ )
362
+
363
+ try:
364
+ # Use shared function with prompting enabled for missing required params
365
+ write_file = output_format in TEXT_FILE_FORMATS
366
+ result = list_generic(report_spec, output_format=output_format, params=params, view_server=args.server,
367
+ view_url=args.url, user=args.user, user_pass=args.password, jupyter=app_config.egeria_jupyter,
368
+ width=app_config.console_width, prompt_missing=True, write_file=write_file)
369
+
370
+ if result.get("kind") == "empty":
371
+ console.print("No results found.")
372
+ return
373
+
374
+ if output_format == "TABLE":
375
+ heading = result.get("heading") or (get_report_spec_heading(report_spec) or f"Report: {report_spec}")
376
+ caption = f"View Server '{args.server}' @ Platform - {args.url}"
377
+ _render_table(console, heading, caption, result.get("data"))
378
+ return
379
+
380
+ # For narrative formats, either a file was written or content is returned
381
+ fpath = result.get("file_path")
382
+ if fpath:
383
+ print(f"\n==> Output written to {fpath}")
384
+ else:
385
+ print(result.get("content", ""))
386
+
387
+ except PyegeriaException as e:
388
+ print_exception_response(e)
389
+ except KeyboardInterrupt:
390
+ pass
391
+
392
+
393
+ if __name__ == "__main__":
394
+ main()