pvw-cli 1.2.0__tar.gz → 1.2.2__tar.gz

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 pvw-cli might be problematic. Click here for more details.

Files changed (66) hide show
  1. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/PKG-INFO +6 -6
  2. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/README.md +4 -4
  3. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/__init__.py +1 -1
  4. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/search.py +235 -33
  5. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pvw_cli.egg-info/PKG-INFO +6 -6
  6. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pvw_cli.egg-info/requires.txt +1 -1
  7. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pyproject.toml +2 -6
  8. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/__main__.py +0 -0
  9. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/__init__.py +0 -0
  10. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/account.py +0 -0
  11. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/cli.py +0 -0
  12. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/collections.py +0 -0
  13. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/domain.py +0 -0
  14. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/entity.py +0 -0
  15. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/glossary.py +0 -0
  16. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/health.py +0 -0
  17. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/insight.py +0 -0
  18. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/lineage.py +0 -0
  19. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/management.py +0 -0
  20. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/policystore.py +0 -0
  21. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/relationship.py +0 -0
  22. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/scan.py +0 -0
  23. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/share.py +0 -0
  24. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/types.py +0 -0
  25. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/unified_catalog.py +0 -0
  26. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/cli/workflow.py +0 -0
  27. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/__init__.py +0 -0
  28. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_account.py +0 -0
  29. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_collections.py +0 -0
  30. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_domain.py +0 -0
  31. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_entity.py +0 -0
  32. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_glossary.py +0 -0
  33. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_health.py +0 -0
  34. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_insight.py +0 -0
  35. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_lineage.py +0 -0
  36. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_management.py +0 -0
  37. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_policystore.py +0 -0
  38. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_relationship.py +0 -0
  39. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_scan.py +0 -0
  40. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_search.py +0 -0
  41. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_share.py +0 -0
  42. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_types.py +0 -0
  43. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_unified_catalog.py +0 -0
  44. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/_workflow.py +0 -0
  45. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/api_client.py +0 -0
  46. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/business_rules.py +0 -0
  47. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/config.py +0 -0
  48. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/data_quality.py +0 -0
  49. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/endpoint.py +0 -0
  50. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/endpoints.py +0 -0
  51. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/exceptions.py +0 -0
  52. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/lineage_visualization.py +0 -0
  53. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/monitoring_dashboard.py +0 -0
  54. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/rate_limiter.py +0 -0
  55. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/retry_handler.py +0 -0
  56. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/scanning_operations.py +0 -0
  57. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/settings.py +0 -0
  58. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/client/sync_client.py +0 -0
  59. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/plugins/__init__.py +0 -0
  60. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/purviewcli/plugins/plugin_system.py +0 -0
  61. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pvw_cli.egg-info/SOURCES.txt +0 -0
  62. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pvw_cli.egg-info/dependency_links.txt +0 -0
  63. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pvw_cli.egg-info/entry_points.txt +0 -0
  64. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pvw_cli.egg-info/not-zip-safe +0 -0
  65. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/pvw_cli.egg-info/top_level.txt +0 -0
  66. {pvw_cli-1.2.0 → pvw_cli-1.2.2}/setup.cfg +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: pvw-cli
3
- Version: 1.2.0
3
+ Version: 1.2.2
4
4
  Summary: Microsoft Purview CLI with comprehensive automation capabilities
5
5
  Author-email: AYOUB KEBAILI <keayoub@msn.com>
6
6
  Maintainer-email: AYOUB KEBAILI <keayoub@msn.com>
@@ -34,7 +34,7 @@ Requires-Dist: rich>=12.0.0
34
34
  Requires-Dist: requests>=2.28.0
35
35
  Requires-Dist: pandas>=1.5.0
36
36
  Requires-Dist: aiohttp>=3.8.0
37
- Requires-Dist: pydantic<3.0.0,>=1.10.0
37
+ Requires-Dist: pydantic<2.12,>=1.10.0
38
38
  Requires-Dist: PyYAML>=6.0
39
39
  Requires-Dist: cryptography<46.0.0,>=41.0.5
40
40
  Provides-Extra: dev
@@ -56,7 +56,7 @@ Requires-Dist: pytest-asyncio>=0.20.0; extra == "test"
56
56
  Requires-Dist: pytest-cov>=4.0.0; extra == "test"
57
57
  Requires-Dist: requests-mock>=1.9.0; extra == "test"
58
58
 
59
- # PURVIEW CLI v1.2.0 - Microsoft Purview Automation & Data Governance
59
+ # PURVIEW CLI v1.2.1 - Microsoft Purview Automation & Data Governance
60
60
 
61
61
  > **LATEST UPDATE (October 2025):**
62
62
  > - **� NEW: Bulk Term Import/Export** - Import multiple terms from CSV/JSON with dry-run support
@@ -72,7 +72,7 @@ Requires-Dist: requests-mock>=1.9.0; extra == "test"
72
72
 
73
73
  ## What is PVW CLI?
74
74
 
75
- **PVW CLI v1.2.0** is a modern, full-featured command-line interface and Python library for Microsoft Purview. It enables automation and management of *all major Purview APIs* including:
75
+ **PVW CLI v1.2.1** is a modern, full-featured command-line interface and Python library for Microsoft Purview. It enables automation and management of *all major Purview APIs* including:
76
76
 
77
77
  - **Unified Catalog (UC) Management** - Complete governance domains, glossary terms, data products, OKRs, CDEs
78
78
  - **Bulk Operations** - Import/export terms from CSV/JSON, bulk delete scripts with progress tracking
@@ -164,7 +164,7 @@ For more advanced usage, see the documentation in `doc/` or the project docs: ht
164
164
 
165
165
  ## Overview
166
166
 
167
- **PVW CLI v1.2.0** is a modern command-line interface and Python library for Microsoft Purview, enabling:
167
+ **PVW CLI v1.2.1** is a modern command-line interface and Python library for Microsoft Purview, enabling:
168
168
 
169
169
  - Advanced data catalog search and discovery
170
170
  - Bulk import/export of entities, glossary terms, and lineage
@@ -1203,6 +1203,6 @@ See [LICENSE](LICENSE) file for details.
1203
1203
 
1204
1204
  ---
1205
1205
 
1206
- **PVW CLI v1.2.0 empowers data engineers, stewards, and architects to automate, scale, and enhance their Microsoft Purview experience with powerful command-line and programmatic capabilities.**
1206
+ **PVW CLI v1.2.1 empowers data engineers, stewards, and architects to automate, scale, and enhance their Microsoft Purview experience with powerful command-line and programmatic capabilities.**
1207
1207
 
1208
1208
  **Latest Features:** Bulk term import/export, PowerShell integration, multiple output formats, and comprehensive bulk delete scripts with beautiful progress tracking.
@@ -1,4 +1,4 @@
1
- # PURVIEW CLI v1.2.0 - Microsoft Purview Automation & Data Governance
1
+ # PURVIEW CLI v1.2.1 - Microsoft Purview Automation & Data Governance
2
2
 
3
3
  > **LATEST UPDATE (October 2025):**
4
4
  > - **� NEW: Bulk Term Import/Export** - Import multiple terms from CSV/JSON with dry-run support
@@ -14,7 +14,7 @@
14
14
 
15
15
  ## What is PVW CLI?
16
16
 
17
- **PVW CLI v1.2.0** is a modern, full-featured command-line interface and Python library for Microsoft Purview. It enables automation and management of *all major Purview APIs* including:
17
+ **PVW CLI v1.2.1** is a modern, full-featured command-line interface and Python library for Microsoft Purview. It enables automation and management of *all major Purview APIs* including:
18
18
 
19
19
  - **Unified Catalog (UC) Management** - Complete governance domains, glossary terms, data products, OKRs, CDEs
20
20
  - **Bulk Operations** - Import/export terms from CSV/JSON, bulk delete scripts with progress tracking
@@ -106,7 +106,7 @@ For more advanced usage, see the documentation in `doc/` or the project docs: ht
106
106
 
107
107
  ## Overview
108
108
 
109
- **PVW CLI v1.2.0** is a modern command-line interface and Python library for Microsoft Purview, enabling:
109
+ **PVW CLI v1.2.1** is a modern command-line interface and Python library for Microsoft Purview, enabling:
110
110
 
111
111
  - Advanced data catalog search and discovery
112
112
  - Bulk import/export of entities, glossary terms, and lineage
@@ -1145,6 +1145,6 @@ See [LICENSE](LICENSE) file for details.
1145
1145
 
1146
1146
  ---
1147
1147
 
1148
- **PVW CLI v1.2.0 empowers data engineers, stewards, and architects to automate, scale, and enhance their Microsoft Purview experience with powerful command-line and programmatic capabilities.**
1148
+ **PVW CLI v1.2.1 empowers data engineers, stewards, and architects to automate, scale, and enhance their Microsoft Purview experience with powerful command-line and programmatic capabilities.**
1149
1149
 
1150
1150
  **Latest Features:** Bulk term import/export, PowerShell integration, multiple output formats, and comprehensive bulk delete scripts with beautiful progress tracking.
@@ -1,4 +1,4 @@
1
- __version__ = "1.2.0"
1
+ __version__ = "1.2.2"
2
2
 
3
3
  # Import main client modules
4
4
  from .client import *
@@ -139,12 +139,8 @@ def _format_search_results(data, show_ids=False):
139
139
  table = Table(title=f"Search Results ({len(items)} of {count} total)")
140
140
  table.add_column("Name", style="cyan", min_width=15, max_width=25)
141
141
  table.add_column("Type", style="green", min_width=15, max_width=20)
142
+ table.add_column("ID", style="yellow", min_width=36, max_width=36)
142
143
  table.add_column("Collection", style="blue", min_width=12, max_width=20)
143
- table.add_column("Classifications", style="magenta", min_width=15, max_width=30)
144
-
145
- if show_ids:
146
- table.add_column("ID", style="yellow", min_width=36, max_width=36)
147
-
148
144
  table.add_column("Qualified Name", style="white", min_width=30)
149
145
 
150
146
  for item in items:
@@ -158,34 +154,17 @@ def _format_search_results(data, show_ids=False):
158
154
  if len(qualified_name) > 60:
159
155
  qualified_name = qualified_name[:57] + "..."
160
156
 
161
- # Handle collection
157
+ # Handle collection - try multiple sources
162
158
  collection = 'N/A'
163
159
  if 'collection' in item and item['collection']:
164
160
  collection = item['collection'].get('name', 'N/A')
161
+ elif 'collectionId' in item:
162
+ collection = item.get('collectionId', 'N/A')
163
+ elif 'assetName' in item:
164
+ collection = item.get('assetName', 'N/A')
165
165
 
166
- # Handle classifications - truncate long classification lists
167
- classifications = []
168
- if 'classification' in item and item['classification']:
169
- for cls in item['classification']:
170
- if isinstance(cls, dict):
171
- cls_name = cls.get('typeName', str(cls))
172
- # Simplify Microsoft classifications for display
173
- if cls_name.startswith('MICROSOFT.'):
174
- cls_name = cls_name.replace('MICROSOFT.', 'MS.')
175
- classifications.append(cls_name)
176
- else:
177
- classifications.append(str(cls))
178
-
179
- # Truncate classifications if too long
180
- cls_display = ", ".join(classifications) if classifications else ""
181
- if len(cls_display) > 40:
182
- cls_display = cls_display[:37] + "..."
183
-
184
- # Build row data
185
- row_data = [name, entity_type, collection, cls_display]
186
- if show_ids:
187
- row_data.append(entity_id)
188
- row_data.append(qualified_name)
166
+ # Build row data with ID always shown
167
+ row_data = [name, entity_type, entity_id, collection, qualified_name]
189
168
 
190
169
  # Add row to table
191
170
  table.add_row(*row_data)
@@ -214,9 +193,9 @@ def _invoke_search_method(method_name, **kwargs):
214
193
  # Choose output format
215
194
  if output_json:
216
195
  _format_json_output(result)
217
- elif detailed and method_name in ['searchQuery', 'searchBrowse', 'searchSuggest', 'searchAutoComplete']:
196
+ elif detailed and method_name in ['searchQuery', 'searchBrowse', 'searchSuggest', 'searchAutocomplete', 'searchFaceted']:
218
197
  _format_detailed_output(result)
219
- elif method_name in ['searchQuery', 'searchBrowse', 'searchSuggest', 'searchAutoComplete']:
198
+ elif method_name in ['searchQuery', 'searchBrowse', 'searchSuggest', 'searchAutocomplete', 'searchFaceted']:
220
199
  _format_search_results(result, show_ids=show_ids)
221
200
  else:
222
201
  _format_json_output(result)
@@ -230,7 +209,7 @@ def _invoke_search_method(method_name, **kwargs):
230
209
  @click.option('--json', 'output_json', is_flag=True, help='Show full JSON details instead of table')
231
210
  def autocomplete(keywords, limit, filterfile, output_json):
232
211
  """Autocomplete search suggestions"""
233
- _invoke_search_method('searchAutoComplete', keywords=keywords, limit=limit, filterFile=filterfile, output_json=output_json)
212
+ _invoke_search_method('searchAutocomplete', keywords=keywords, limit=limit, filterFile=filterfile, output_json=output_json)
234
213
 
235
214
  @search.command()
236
215
  @click.option('--entityType', required=False)
@@ -305,7 +284,7 @@ def advanced(keywords, limit, offset, filterfile, facets_file, businessmetadata,
305
284
  with open(businessmetadata, 'r', encoding='utf-8') as f:
306
285
  business_metadata_content = json.load(f)
307
286
  _invoke_search_method(
308
- 'searchAdvancedQuery',
287
+ 'searchAdvanced',
309
288
  keywords=keywords,
310
289
  limit=limit,
311
290
  offset=offset,
@@ -316,4 +295,227 @@ def advanced(keywords, limit, offset, filterfile, facets_file, businessmetadata,
316
295
  termAssignments=termassignments
317
296
  )
318
297
 
298
+ @search.command('find-table')
299
+ @click.option('--name', required=False, help='Table name (e.g., Address)')
300
+ @click.option('--schema', required=False, help='Schema name (e.g., SalesLT, dbo)')
301
+ @click.option('--database', required=False, help='Database name (e.g., Adventureworks)')
302
+ @click.option('--server', required=False, help='Server name (e.g., fabricdemos001.database.windows.net)')
303
+ @click.option('--qualified-name', required=False, help='Full qualified name from Purview (e.g., mssql://server/database/schema/table)')
304
+ @click.option('--entity-type', required=False, help='Entity type to search for (e.g., azure_sql_table, mssql_table)')
305
+ @click.option('--limit', required=False, type=int, default=25, help='Maximum number of results to return')
306
+ @click.option('--show-ids', is_flag=True, help='Show entity IDs in the results')
307
+ @click.option('--json', 'output_json', is_flag=True, help='Show full JSON details')
308
+ @click.option('--detailed', is_flag=True, help='Show detailed information')
309
+ @click.option('--id-only', is_flag=True, help='Output only the GUID (useful for scripts and automation)')
310
+ def find_table(name, schema, database, server, qualified_name, entity_type, limit, show_ids, output_json, detailed, id_only):
311
+ """Find a table by name, schema, database, or get all tables in a schema/database.
312
+
313
+ Perfect for getting the GUID of a data asset before updating it.
314
+ You can search for ONE specific table or ALL tables in a schema/database.
315
+
316
+ \b
317
+ SEARCH ONE SPECIFIC TABLE:
318
+ pvw search find-table --name Address --schema SalesLT --database Adventureworks
319
+ pvw search find-table --qualified-name "mssql://server/database/schema/table"
320
+
321
+ \b
322
+ SEARCH MULTIPLE TABLES:
323
+ pvw search find-table --schema SalesLT --database Adventureworks
324
+ pvw search find-table --database Adventureworks
325
+ pvw search find-table --schema SalesLT
326
+
327
+ \b
328
+ GET GUIDS FOR AUTOMATION:
329
+ pvw search find-table --name Address --schema SalesLT --database Adventureworks --id-only
330
+ pvw search find-table --schema SalesLT --database Adventureworks --id-only
331
+
332
+ \b
333
+ USE IN SCRIPTS (PowerShell):
334
+ $guid = pvw search find-table --name Address --schema SalesLT --database Adventureworks --id-only
335
+ pvw entity update --guid $guid --payload update.json
336
+
337
+ $guids = pvw search find-table --schema SalesLT --database Adventureworks --id-only
338
+ foreach ($guid in $guids) { pvw entity update --guid $guid --payload update.json }
339
+ """
340
+ search_client = Search()
341
+
342
+ # Validate that at least some search criteria is provided
343
+ if not name and not qualified_name and not schema and not database:
344
+ console.print("[red]ERROR:[/red] You must provide at least --name, --qualified-name, --schema, or --database")
345
+ return
346
+
347
+ # Build search pattern
348
+ search_pattern = qualified_name
349
+ if not search_pattern:
350
+ # Build pattern from components
351
+ # Try to build a full qualified name pattern that matches Purview's format
352
+ if server and database and schema and name:
353
+ # Full path with server: mssql://server/database/schema/table
354
+ search_pattern = f"mssql://{server}/{database}/{schema}/{name}"
355
+ elif database and schema and name:
356
+ # Database.schema.table format
357
+ search_pattern = f"{database}/{schema}/{name}"
358
+ elif database and schema:
359
+ # Database.schema format (all tables in schema)
360
+ search_pattern = f"{database}/{schema}"
361
+ elif schema and name:
362
+ # Schema.table format
363
+ search_pattern = f"{schema}/{name}"
364
+ elif database:
365
+ # Just database (all tables in database)
366
+ search_pattern = database
367
+ elif schema:
368
+ # Just schema (all tables in schema)
369
+ search_pattern = schema
370
+ elif name:
371
+ # Just the table name
372
+ search_pattern = name
373
+ else:
374
+ console.print("[red]ERROR:[/red] You must provide at least one search criterion")
375
+ return
376
+
377
+ # For keyword search, use different strategies based on what we have
378
+ if name:
379
+ search_keywords = name
380
+ elif schema:
381
+ search_keywords = schema
382
+ elif database:
383
+ search_keywords = database
384
+ else:
385
+ search_keywords = search_pattern.split('/')[-1]
386
+
387
+ # Build search arguments - use keywords that will match
388
+ args = {
389
+ '--keywords': search_keywords,
390
+ '--limit': limit,
391
+ '--offset': 0
392
+ }
393
+
394
+ # Create filter for entity type if specified
395
+ import tempfile
396
+ import json
397
+ import os
398
+
399
+ temp_filter_file = None
400
+
401
+ if entity_type:
402
+ filter_obj = {
403
+ 'entityType': entity_type
404
+ }
405
+
406
+ # Write filter to temp file
407
+ with tempfile.NamedTemporaryFile(mode='w', suffix='.json', delete=False, encoding='utf-8') as f:
408
+ json.dump(filter_obj, f)
409
+ temp_filter_file = f.name
410
+
411
+ args['--filterFile'] = temp_filter_file
412
+
413
+ try:
414
+ # Execute search
415
+ result = search_client.searchQuery(args)
416
+
417
+ if not result:
418
+ console.print("[yellow]No results returned from search[/yellow]")
419
+ if temp_filter_file:
420
+ os.unlink(temp_filter_file)
421
+ return
422
+
423
+ # Filter results by qualified name match if provided
424
+ if result and 'value' in result and result['value']:
425
+ filtered_results = []
426
+ search_lower = search_pattern.lower()
427
+
428
+ for item in result.get('value', []):
429
+ item_qn = item.get('qualifiedName', '').lower()
430
+ item_name = item.get('name', '').lower()
431
+
432
+ # Build matching criteria
433
+ matches = False
434
+
435
+ # If we have all components, do strict matching
436
+ if name and schema and database:
437
+ # Exact name match (not substring) - critical for precision
438
+ name_match = name.lower() == item_name
439
+ schema_match = schema.lower() in item_qn
440
+ database_match = database.lower() in item_qn
441
+ server_match = not server or server.lower() in item_qn
442
+ matches = name_match and schema_match and database_match and server_match
443
+
444
+ # If we have database and schema (all tables in this schema)
445
+ elif database and schema and not name:
446
+ schema_match = schema.lower() in item_qn
447
+ database_match = database.lower() in item_qn
448
+ server_match = not server or server.lower() in item_qn
449
+ matches = schema_match and database_match and server_match
450
+
451
+ # If we have schema and name
452
+ elif name and schema:
453
+ # Exact name match
454
+ name_match = name.lower() == item_name
455
+ schema_match = schema.lower() in item_qn
456
+ matches = name_match and schema_match
457
+
458
+ # If we have just database (all tables in this database)
459
+ elif database and not name and not schema:
460
+ database_match = database.lower() in item_qn
461
+ server_match = not server or server.lower() in item_qn
462
+ matches = database_match and server_match
463
+
464
+ # If we have just schema (all tables in this schema)
465
+ elif schema and not name and not database:
466
+ schema_match = schema.lower() in item_qn
467
+ matches = schema_match
468
+
469
+ # If we have just name or a qualified name pattern
470
+ elif name or qualified_name:
471
+ # If qualified_name was provided, do exact match
472
+ if qualified_name:
473
+ # Check for exact match of the qualified name
474
+ matches = search_lower == item_qn or item_qn.endswith('/' + search_keywords.lower())
475
+ else:
476
+ # Just name provided, match by name
477
+ matches = search_keywords.lower() == item_name
478
+
479
+ if matches:
480
+ filtered_results.append(item)
481
+
482
+ if filtered_results:
483
+ result['value'] = filtered_results
484
+ result['@search.count'] = len(filtered_results)
485
+ else:
486
+ console.print(f"[yellow]No results found matching '{search_pattern}'[/yellow]")
487
+ if temp_filter_file:
488
+ os.unlink(temp_filter_file)
489
+ return
490
+
491
+ # Display results
492
+ if id_only:
493
+ # Output only the ID(s) for scripting purposes
494
+ if result and 'value' in result and result['value']:
495
+ for item in result['value']:
496
+ print(item.get('id', ''))
497
+ else:
498
+ console.print("[yellow]No results found[/yellow]")
499
+ elif output_json:
500
+ _format_json_output(result)
501
+ elif detailed:
502
+ _format_detailed_output(result)
503
+ else:
504
+ _format_search_results(result, show_ids=show_ids)
505
+
506
+ # Clean up temp file
507
+ if temp_filter_file:
508
+ import os
509
+ os.unlink(temp_filter_file)
510
+
511
+ except Exception as e:
512
+ console.print(f"[red]ERROR:[/red] {str(e)}")
513
+ # Clean up temp file on error
514
+ if temp_filter_file:
515
+ import os
516
+ try:
517
+ os.unlink(temp_filter_file)
518
+ except:
519
+ pass
520
+
319
521
  __all__ = ['search']
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: pvw-cli
3
- Version: 1.2.0
3
+ Version: 1.2.2
4
4
  Summary: Microsoft Purview CLI with comprehensive automation capabilities
5
5
  Author-email: AYOUB KEBAILI <keayoub@msn.com>
6
6
  Maintainer-email: AYOUB KEBAILI <keayoub@msn.com>
@@ -34,7 +34,7 @@ Requires-Dist: rich>=12.0.0
34
34
  Requires-Dist: requests>=2.28.0
35
35
  Requires-Dist: pandas>=1.5.0
36
36
  Requires-Dist: aiohttp>=3.8.0
37
- Requires-Dist: pydantic<3.0.0,>=1.10.0
37
+ Requires-Dist: pydantic<2.12,>=1.10.0
38
38
  Requires-Dist: PyYAML>=6.0
39
39
  Requires-Dist: cryptography<46.0.0,>=41.0.5
40
40
  Provides-Extra: dev
@@ -56,7 +56,7 @@ Requires-Dist: pytest-asyncio>=0.20.0; extra == "test"
56
56
  Requires-Dist: pytest-cov>=4.0.0; extra == "test"
57
57
  Requires-Dist: requests-mock>=1.9.0; extra == "test"
58
58
 
59
- # PURVIEW CLI v1.2.0 - Microsoft Purview Automation & Data Governance
59
+ # PURVIEW CLI v1.2.1 - Microsoft Purview Automation & Data Governance
60
60
 
61
61
  > **LATEST UPDATE (October 2025):**
62
62
  > - **� NEW: Bulk Term Import/Export** - Import multiple terms from CSV/JSON with dry-run support
@@ -72,7 +72,7 @@ Requires-Dist: requests-mock>=1.9.0; extra == "test"
72
72
 
73
73
  ## What is PVW CLI?
74
74
 
75
- **PVW CLI v1.2.0** is a modern, full-featured command-line interface and Python library for Microsoft Purview. It enables automation and management of *all major Purview APIs* including:
75
+ **PVW CLI v1.2.1** is a modern, full-featured command-line interface and Python library for Microsoft Purview. It enables automation and management of *all major Purview APIs* including:
76
76
 
77
77
  - **Unified Catalog (UC) Management** - Complete governance domains, glossary terms, data products, OKRs, CDEs
78
78
  - **Bulk Operations** - Import/export terms from CSV/JSON, bulk delete scripts with progress tracking
@@ -164,7 +164,7 @@ For more advanced usage, see the documentation in `doc/` or the project docs: ht
164
164
 
165
165
  ## Overview
166
166
 
167
- **PVW CLI v1.2.0** is a modern command-line interface and Python library for Microsoft Purview, enabling:
167
+ **PVW CLI v1.2.1** is a modern command-line interface and Python library for Microsoft Purview, enabling:
168
168
 
169
169
  - Advanced data catalog search and discovery
170
170
  - Bulk import/export of entities, glossary terms, and lineage
@@ -1203,6 +1203,6 @@ See [LICENSE](LICENSE) file for details.
1203
1203
 
1204
1204
  ---
1205
1205
 
1206
- **PVW CLI v1.2.0 empowers data engineers, stewards, and architects to automate, scale, and enhance their Microsoft Purview experience with powerful command-line and programmatic capabilities.**
1206
+ **PVW CLI v1.2.1 empowers data engineers, stewards, and architects to automate, scale, and enhance their Microsoft Purview experience with powerful command-line and programmatic capabilities.**
1207
1207
 
1208
1208
  **Latest Features:** Bulk term import/export, PowerShell integration, multiple output formats, and comprehensive bulk delete scripts with beautiful progress tracking.
@@ -5,7 +5,7 @@ rich>=12.0.0
5
5
  requests>=2.28.0
6
6
  pandas>=1.5.0
7
7
  aiohttp>=3.8.0
8
- pydantic<3.0.0,>=1.10.0
8
+ pydantic<2.12,>=1.10.0
9
9
  PyYAML>=6.0
10
10
  cryptography<46.0.0,>=41.0.5
11
11
 
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "pvw-cli"
7
- version = "1.2.0"
7
+ version = "1.2.2"
8
8
  description = "Microsoft Purview CLI with comprehensive automation capabilities"
9
9
  readme = "README.md"
10
10
  license = "MIT"
@@ -41,7 +41,7 @@ dependencies = [
41
41
  "requests>=2.28.0",
42
42
  "pandas>=1.5.0",
43
43
  "aiohttp>=3.8.0",
44
- "pydantic>=1.10.0,<3.0.0",
44
+ "pydantic>=1.10.0,<2.12",
45
45
  "PyYAML>=6.0",
46
46
  "cryptography>=41.0.5,<46.0.0",
47
47
  ]
@@ -177,7 +177,3 @@ exclude_lines = [
177
177
  "class .*\\bProtocol\\):",
178
178
  "@(abc\\.)?abstractmethod",
179
179
  ]
180
-
181
-
182
-
183
-
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes