datacontract-cli 0.10.23__py3-none-any.whl → 0.10.40__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.
Files changed (84) hide show
  1. datacontract/__init__.py +13 -0
  2. datacontract/api.py +12 -5
  3. datacontract/catalog/catalog.py +5 -3
  4. datacontract/cli.py +119 -13
  5. datacontract/data_contract.py +145 -67
  6. datacontract/engines/data_contract_checks.py +366 -60
  7. datacontract/engines/data_contract_test.py +50 -4
  8. datacontract/engines/fastjsonschema/check_jsonschema.py +37 -19
  9. datacontract/engines/fastjsonschema/s3/s3_read_files.py +3 -2
  10. datacontract/engines/soda/check_soda_execute.py +27 -3
  11. datacontract/engines/soda/connections/athena.py +79 -0
  12. datacontract/engines/soda/connections/duckdb_connection.py +65 -6
  13. datacontract/engines/soda/connections/kafka.py +4 -2
  14. datacontract/engines/soda/connections/oracle.py +50 -0
  15. datacontract/export/avro_converter.py +20 -3
  16. datacontract/export/bigquery_converter.py +1 -1
  17. datacontract/export/dbt_converter.py +36 -7
  18. datacontract/export/dqx_converter.py +126 -0
  19. datacontract/export/duckdb_type_converter.py +57 -0
  20. datacontract/export/excel_exporter.py +923 -0
  21. datacontract/export/exporter.py +3 -0
  22. datacontract/export/exporter_factory.py +17 -1
  23. datacontract/export/great_expectations_converter.py +55 -5
  24. datacontract/export/{html_export.py → html_exporter.py} +31 -20
  25. datacontract/export/markdown_converter.py +134 -5
  26. datacontract/export/mermaid_exporter.py +110 -0
  27. datacontract/export/odcs_v3_exporter.py +193 -149
  28. datacontract/export/protobuf_converter.py +163 -69
  29. datacontract/export/rdf_converter.py +2 -2
  30. datacontract/export/sodacl_converter.py +9 -1
  31. datacontract/export/spark_converter.py +31 -4
  32. datacontract/export/sql_converter.py +6 -2
  33. datacontract/export/sql_type_converter.py +124 -8
  34. datacontract/imports/avro_importer.py +63 -12
  35. datacontract/imports/csv_importer.py +111 -57
  36. datacontract/imports/excel_importer.py +1112 -0
  37. datacontract/imports/importer.py +16 -3
  38. datacontract/imports/importer_factory.py +17 -0
  39. datacontract/imports/json_importer.py +325 -0
  40. datacontract/imports/odcs_importer.py +2 -2
  41. datacontract/imports/odcs_v3_importer.py +367 -151
  42. datacontract/imports/protobuf_importer.py +264 -0
  43. datacontract/imports/spark_importer.py +117 -13
  44. datacontract/imports/sql_importer.py +32 -16
  45. datacontract/imports/unity_importer.py +84 -38
  46. datacontract/init/init_template.py +1 -1
  47. datacontract/integration/entropy_data.py +126 -0
  48. datacontract/lint/resolve.py +112 -23
  49. datacontract/lint/schema.py +24 -15
  50. datacontract/lint/urls.py +17 -3
  51. datacontract/model/data_contract_specification/__init__.py +1 -0
  52. datacontract/model/odcs.py +13 -0
  53. datacontract/model/run.py +3 -0
  54. datacontract/output/junit_test_results.py +3 -3
  55. datacontract/schemas/datacontract-1.1.0.init.yaml +1 -1
  56. datacontract/schemas/datacontract-1.2.0.init.yaml +91 -0
  57. datacontract/schemas/datacontract-1.2.0.schema.json +2029 -0
  58. datacontract/schemas/datacontract-1.2.1.init.yaml +91 -0
  59. datacontract/schemas/datacontract-1.2.1.schema.json +2058 -0
  60. datacontract/schemas/odcs-3.0.2.schema.json +2382 -0
  61. datacontract/schemas/odcs-3.1.0.schema.json +2809 -0
  62. datacontract/templates/datacontract.html +54 -3
  63. datacontract/templates/datacontract_odcs.html +685 -0
  64. datacontract/templates/index.html +5 -2
  65. datacontract/templates/partials/server.html +2 -0
  66. datacontract/templates/style/output.css +319 -145
  67. {datacontract_cli-0.10.23.dist-info → datacontract_cli-0.10.40.dist-info}/METADATA +711 -433
  68. datacontract_cli-0.10.40.dist-info/RECORD +121 -0
  69. {datacontract_cli-0.10.23.dist-info → datacontract_cli-0.10.40.dist-info}/WHEEL +1 -1
  70. {datacontract_cli-0.10.23.dist-info → datacontract_cli-0.10.40.dist-info/licenses}/LICENSE +1 -1
  71. datacontract/export/csv_type_converter.py +0 -36
  72. datacontract/integration/datamesh_manager.py +0 -72
  73. datacontract/lint/lint.py +0 -142
  74. datacontract/lint/linters/description_linter.py +0 -35
  75. datacontract/lint/linters/field_pattern_linter.py +0 -34
  76. datacontract/lint/linters/field_reference_linter.py +0 -48
  77. datacontract/lint/linters/notice_period_linter.py +0 -55
  78. datacontract/lint/linters/quality_schema_linter.py +0 -52
  79. datacontract/lint/linters/valid_constraints_linter.py +0 -100
  80. datacontract/model/data_contract_specification.py +0 -327
  81. datacontract_cli-0.10.23.dist-info/RECORD +0 -113
  82. /datacontract/{lint/linters → output}/__init__.py +0 -0
  83. {datacontract_cli-0.10.23.dist-info → datacontract_cli-0.10.40.dist-info}/entry_points.txt +0 -0
  84. {datacontract_cli-0.10.23.dist-info → datacontract_cli-0.10.40.dist-info}/top_level.txt +0 -0
@@ -1,7 +1,8 @@
1
1
  from abc import ABC, abstractmethod
2
2
  from enum import Enum
3
3
 
4
- from datacontract.model.data_contract_specification import DataContractSpecification
4
+ from datacontract_specification.model import DataContractSpecification
5
+ from open_data_contract_standard.model import OpenDataContractStandard
5
6
 
6
7
 
7
8
  class Importer(ABC):
@@ -11,10 +12,10 @@ class Importer(ABC):
11
12
  @abstractmethod
12
13
  def import_source(
13
14
  self,
14
- data_contract_specification: DataContractSpecification,
15
+ data_contract_specification: DataContractSpecification | OpenDataContractStandard,
15
16
  source: str,
16
17
  import_args: dict,
17
- ) -> DataContractSpecification:
18
+ ) -> DataContractSpecification | OpenDataContractStandard:
18
19
  pass
19
20
 
20
21
 
@@ -25,6 +26,7 @@ class ImportFormat(str, Enum):
25
26
  dbml = "dbml"
26
27
  glue = "glue"
27
28
  jsonschema = "jsonschema"
29
+ json = "json"
28
30
  bigquery = "bigquery"
29
31
  odcs = "odcs"
30
32
  unity = "unity"
@@ -32,7 +34,18 @@ class ImportFormat(str, Enum):
32
34
  iceberg = "iceberg"
33
35
  parquet = "parquet"
34
36
  csv = "csv"
37
+ protobuf = "protobuf"
38
+ excel = "excel"
35
39
 
36
40
  @classmethod
37
41
  def get_supported_formats(cls):
38
42
  return list(map(lambda c: c.value, cls))
43
+
44
+
45
+ class Spec(str, Enum):
46
+ datacontract_specification = "datacontract_specification"
47
+ odcs = "odcs"
48
+
49
+ @classmethod
50
+ def get_supported_types(cls):
51
+ return list(map(lambda c: c.value, cls))
@@ -109,3 +109,20 @@ importer_factory.register_lazy_importer(
109
109
  module_path="datacontract.imports.csv_importer",
110
110
  class_name="CsvImporter",
111
111
  )
112
+ importer_factory.register_lazy_importer(
113
+ name=ImportFormat.protobuf,
114
+ module_path="datacontract.imports.protobuf_importer",
115
+ class_name="ProtoBufImporter",
116
+ )
117
+ importer_factory.register_lazy_importer(
118
+ name=ImportFormat.excel,
119
+ module_path="datacontract.imports.excel_importer",
120
+ class_name="ExcelImporter",
121
+ )
122
+
123
+
124
+ importer_factory.register_lazy_importer(
125
+ name=ImportFormat.json,
126
+ module_path="datacontract.imports.json_importer",
127
+ class_name="JsonImporter",
128
+ )
@@ -0,0 +1,325 @@
1
+ import json
2
+ import os
3
+ import re
4
+ from typing import Any, Dict, List, Optional, Tuple
5
+
6
+ from datacontract.imports.importer import Importer
7
+ from datacontract.model.data_contract_specification import DataContractSpecification, Model, Server
8
+
9
+
10
+ class JsonImporter(Importer):
11
+ def import_source(
12
+ self, data_contract_specification: DataContractSpecification, source: str, import_args: dict
13
+ ) -> DataContractSpecification:
14
+ return import_json(data_contract_specification, source)
15
+
16
+
17
+ def is_ndjson(file_path: str) -> bool:
18
+ """Check if a file contains newline-delimited JSON."""
19
+ with open(file_path, "r", encoding="utf-8") as file:
20
+ for _ in range(5):
21
+ line = file.readline().strip()
22
+ if not line:
23
+ continue
24
+ try:
25
+ json.loads(line)
26
+ return True
27
+ except json.JSONDecodeError:
28
+ break
29
+ return False
30
+
31
+
32
+ def import_json(
33
+ data_contract_specification: DataContractSpecification, source: str, include_examples: bool = False
34
+ ) -> DataContractSpecification:
35
+ # use the file name as base model name
36
+ base_model_name = os.path.splitext(os.path.basename(source))[0]
37
+
38
+ # check if file is newline-delimited JSON
39
+ if is_ndjson(source):
40
+ # load NDJSON data
41
+ json_data = []
42
+ with open(source, "r", encoding="utf-8") as file:
43
+ for line in file:
44
+ line = line.strip()
45
+ if line:
46
+ try:
47
+ json_data.append(json.loads(line))
48
+ except json.JSONDecodeError:
49
+ continue
50
+ else:
51
+ # load regular JSON data
52
+ with open(source, "r", encoding="utf-8") as file:
53
+ json_data = json.load(file)
54
+
55
+ if data_contract_specification.servers is None:
56
+ data_contract_specification.servers = {}
57
+
58
+ data_contract_specification.servers["production"] = Server(type="local", path=source, format="json")
59
+
60
+ # initialisation
61
+ models = {}
62
+
63
+ if isinstance(json_data, list) and json_data:
64
+ # Array of items
65
+ if all(isinstance(item, dict) for item in json_data[:5]):
66
+ # Array of objects, as table
67
+ fields = {}
68
+ for item in json_data[:20]:
69
+ for key, value in item.items():
70
+ field_def = generate_field_definition(value, key, base_model_name, models)
71
+ if key in fields:
72
+ fields[key] = merge_field_definitions(fields[key], field_def)
73
+ else:
74
+ fields[key] = field_def
75
+
76
+ models[base_model_name] = {
77
+ "type": "table",
78
+ "description": f"Generated from JSON array in {source}",
79
+ "fields": fields,
80
+ "examples": json_data[:3] if include_examples else None,
81
+ }
82
+ else:
83
+ # Simple array
84
+ item_type, item_format = infer_array_type(json_data[:20])
85
+ models[base_model_name] = {
86
+ "type": "array",
87
+ "description": f"Generated from JSON array in {source}",
88
+ "items": {"type": item_type, "format": item_format} if item_format else {"type": item_type},
89
+ "examples": [json_data[:5]] if include_examples else None,
90
+ }
91
+ elif isinstance(json_data, dict):
92
+ # Single object
93
+ fields = {}
94
+ for key, value in json_data.items():
95
+ fields[key] = generate_field_definition(value, key, base_model_name, models)
96
+
97
+ models[base_model_name] = {
98
+ "type": "object",
99
+ "description": f"Generated from JSON object in {source}",
100
+ "fields": fields,
101
+ "examples": [json_data] if include_examples else None,
102
+ }
103
+ else:
104
+ # Primitive value
105
+ field_type, field_format = determine_type_and_format(json_data)
106
+ models[base_model_name] = {
107
+ "type": field_type,
108
+ "description": f"Generated from JSON primitive in {source}",
109
+ "format": field_format,
110
+ "examples": [json_data] if include_examples and field_type != "boolean" else None,
111
+ }
112
+
113
+ for model_name, model_def in models.items():
114
+ model_type = model_def.pop("type")
115
+ data_contract_specification.models[model_name] = Model(type=model_type, **model_def)
116
+
117
+ return data_contract_specification
118
+
119
+
120
+ def generate_field_definition(
121
+ value: Any, field_name: str, parent_model: str, models: Dict[str, Dict[str, Any]]
122
+ ) -> Dict[str, Any]:
123
+ """Generate a field definition for a JSON value, creating nested models."""
124
+
125
+ if isinstance(value, dict):
126
+ # Handle object fields
127
+ fields = {}
128
+ for key, nested_value in value.items():
129
+ fields[key] = generate_field_definition(nested_value, key, parent_model, models)
130
+
131
+ return {"type": "object", "fields": fields}
132
+
133
+ elif isinstance(value, list):
134
+ # Handle array fields
135
+ if not value:
136
+ return {"type": "array", "items": {"type": "string"}}
137
+
138
+ if all(isinstance(item, dict) for item in value):
139
+ # Array of objects
140
+ fields = {}
141
+ for item in value:
142
+ for key, nested_value in item.items():
143
+ field_def = generate_field_definition(nested_value, key, parent_model, models)
144
+ if key in fields:
145
+ fields[key] = merge_field_definitions(fields[key], field_def)
146
+ else:
147
+ fields[key] = field_def
148
+
149
+ return {"type": "array", "items": {"type": "object", "fields": fields}}
150
+
151
+ elif all(isinstance(item, list) for item in value):
152
+ # Array of arrays
153
+ inner_type, inner_format = infer_array_type(value[0])
154
+ return {
155
+ "type": "array",
156
+ "items": {
157
+ "type": "array",
158
+ "items": {"type": inner_type, "format": inner_format} if inner_format else {"type": inner_type},
159
+ },
160
+ "examples": value[:5], # Include examples for nested arrays
161
+ }
162
+
163
+ else:
164
+ # Array of simple or mixed types
165
+ item_type, item_format = infer_array_type(value)
166
+ items_def = {"type": item_type}
167
+ if item_format:
168
+ items_def["format"] = item_format
169
+
170
+ field_def = {"type": "array", "items": items_def}
171
+
172
+ # Add examples if appropriate
173
+ sample_values = [item for item in value[:5] if item is not None]
174
+ if sample_values:
175
+ field_def["examples"] = sample_values
176
+
177
+ return field_def
178
+
179
+ else:
180
+ # Handle primitive types
181
+ field_type, field_format = determine_type_and_format(value)
182
+ field_def = {"type": field_type}
183
+ if field_format:
184
+ field_def["format"] = field_format
185
+
186
+ # Add examples
187
+ if value is not None and field_type != "boolean":
188
+ field_def["examples"] = [value]
189
+
190
+ return field_def
191
+
192
+
193
+ def infer_array_type(array: List) -> Tuple[str, Optional[str]]:
194
+ """Infer the common type of items in an array."""
195
+ if not array:
196
+ return "string", None
197
+
198
+ # if all items are dictionaries with the same structure
199
+ if all(isinstance(item, dict) for item in array):
200
+ return "object", None
201
+
202
+ # if all items are of the same primitive type
203
+ non_null_items = [item for item in array if item is not None]
204
+ if not non_null_items:
205
+ return "null", None
206
+
207
+ types_and_formats = [determine_type_and_format(item) for item in non_null_items]
208
+ types = {t for t, _ in types_and_formats}
209
+ formats = {f for _, f in types_and_formats if f is not None}
210
+
211
+ # simplify type combinations
212
+ if types == {"integer", "number"}:
213
+ return "number", None
214
+ if len(types) == 1:
215
+ type_name = next(iter(types))
216
+ format_name = next(iter(formats)) if len(formats) == 1 else None
217
+ return type_name, format_name
218
+ if all(t in {"string", "integer", "number", "boolean", "null"} for t in types):
219
+ # If all string values have the same format, keep it
220
+ if len(formats) == 1 and "string" in types:
221
+ return "string", next(iter(formats))
222
+ return "string", None
223
+
224
+ # Mixed types
225
+ return "string", None
226
+
227
+
228
+ def determine_type_and_format(value: Any) -> Tuple[str, Optional[str]]:
229
+ """determine the datacontract type and format for a JSON value."""
230
+ if value is None:
231
+ return "null", None
232
+ elif isinstance(value, bool):
233
+ return "boolean", None
234
+ elif isinstance(value, int):
235
+ return "integer", None
236
+ elif isinstance(value, float):
237
+ return "number", None
238
+ elif isinstance(value, str):
239
+ try:
240
+ if re.match(r"^\d{4}-\d{2}-\d{2}$", value):
241
+ return "string", "date"
242
+ elif re.match(r"^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d+)?(Z|[+-]\d{2}:\d{2})?$", value):
243
+ return "string", "date-time"
244
+ elif re.match(r"^[\w\.-]+@([\w-]+\.)+[\w-]{2,4}$", value):
245
+ return "string", "email"
246
+ elif re.match(r"^[a-f0-9]{8}-?[a-f0-9]{4}-?[a-f0-9]{4}-?[a-f0-9]{4}-?[a-f0-9]{12}$", value.lower()):
247
+ return "string", "uuid"
248
+ else:
249
+ return "string", None
250
+ except re.error:
251
+ return "string", None
252
+ elif isinstance(value, dict):
253
+ return "object", None
254
+ elif isinstance(value, list):
255
+ return "array", None
256
+ else:
257
+ return "string", None
258
+
259
+
260
+ def merge_field_definitions(field1: Dict[str, Any], field2: Dict[str, Any]) -> Dict[str, Any]:
261
+ """Merge two field definitions."""
262
+ result = field1.copy()
263
+ if field1.get("type") == "object" and field2.get("type") != "object":
264
+ return field1
265
+ if field2.get("type") == "object" and field1.get("type") != "object":
266
+ return field2
267
+ # Handle type differences
268
+ if field1.get("type") != field2.get("type"):
269
+ type1, _ = field1.get("type", "string"), field1.get("format")
270
+ type2, _ = field2.get("type", "string"), field2.get("format")
271
+
272
+ if type1 == "integer" and type2 == "number" or type1 == "number" and type2 == "integer":
273
+ common_type = "number"
274
+ common_format = None
275
+ elif "string" in [type1, type2]:
276
+ common_type = "string"
277
+ common_format = None
278
+ elif all(t in ["string", "integer", "number", "boolean", "null"] for t in [type1, type2]):
279
+ common_type = "string"
280
+ common_format = None
281
+ elif type1 == "array" and type2 == "array":
282
+ # Handle mixed array types
283
+ items1 = field1.get("items", {})
284
+ items2 = field2.get("items", {})
285
+ if items1.get("type") == "object" or items2.get("type") == "object":
286
+ if items1.get("type") == "object" and items2.get("type") == "object":
287
+ merged_items = merge_field_definitions(items1, items2)
288
+ else:
289
+ merged_items = items1 if items1.get("type") == "object" else items2
290
+ return {"type": "array", "items": merged_items}
291
+ else:
292
+ merged_items = merge_field_definitions(items1, items2)
293
+ return {"type": "array", "items": merged_items}
294
+ else:
295
+ common_type = "array" if "array" in [type1, type2] else "object"
296
+ common_format = None
297
+
298
+ result["type"] = common_type
299
+ if common_format:
300
+ result["format"] = common_format
301
+ elif "format" in result:
302
+ del result["format"]
303
+
304
+ # Merge examples
305
+ if "examples" in field2:
306
+ if "examples" in result:
307
+ combined = result["examples"] + [ex for ex in field2["examples"] if ex not in result["examples"]]
308
+ result["examples"] = combined[:5] # Limit to 5 examples
309
+ else:
310
+ result["examples"] = field2["examples"]
311
+
312
+ # Handle nested structures
313
+ if result.get("type") == "array" and "items" in field1 and "items" in field2:
314
+ result["items"] = merge_field_definitions(field1["items"], field2["items"])
315
+ elif result.get("type") == "object" and "fields" in field1 and "fields" in field2:
316
+ # Merge fields from both objects
317
+ merged_fields = field1["fields"].copy()
318
+ for key, field_def in field2["fields"].items():
319
+ if key in merged_fields:
320
+ merged_fields[key] = merge_field_definitions(merged_fields[key], field_def)
321
+ else:
322
+ merged_fields[key] = field_def
323
+ result["fields"] = merged_fields
324
+
325
+ return result
@@ -48,9 +48,9 @@ def import_odcs(data_contract_specification: DataContractSpecification, source:
48
48
  engine="datacontract",
49
49
  )
50
50
  elif odcs_api_version.startswith("v3."):
51
- from datacontract.imports.odcs_v3_importer import import_odcs_v3
51
+ from datacontract.imports.odcs_v3_importer import import_odcs_v3_as_dcs
52
52
 
53
- return import_odcs_v3(data_contract_specification, source)
53
+ return import_odcs_v3_as_dcs(data_contract_specification, source)
54
54
  else:
55
55
  raise DataContractException(
56
56
  type="schema",