mirror of
				https://github.com/open-metadata/OpenMetadata.git
				synced 2025-10-25 07:42:40 +00:00 
			
		
		
		
	* fix: profiler refactor * fix: catch division by zero error * fix: instantiated new column object from name to delegate type handler back to dbapi * fix: reverted columns instantiation and updated NUMBER type to NUMERIC * fix: updated doc string for process_pii_sensitive method
This commit is contained in:
		
							parent
							
								
									bbce9c5aa4
								
							
						
					
					
						commit
						3d8e7e6d41
					
				| @ -362,7 +362,8 @@ class ProfilerWorkflow(WorkflowStatusMixin): | ||||
|             self.create_profiler(entity, profiler_interface) | ||||
|             self.profiler = cast(Profiler, self.profiler)  # satisfy type checker | ||||
|             profile: ProfilerResponse = self.profiler.process( | ||||
|                 self.source_config.generateSampleData | ||||
|                 self.source_config.generateSampleData, | ||||
|                 self.source_config.processPiiSensitive, | ||||
|             ) | ||||
|         except Exception as exc:  # pylint: disable=broad-except | ||||
| 
 | ||||
|  | ||||
| @ -53,8 +53,10 @@ class NonParametricSkew(ComposedMetric): | ||||
|         res_median = res.get(Median.name()) | ||||
| 
 | ||||
|         if res_mean is not None and res_stddev is not None and res_median is not None: | ||||
|             return (float(res_mean) - float(res_median)) / float( | ||||
|                 res_stddev | ||||
|             )  # convert from decimal | ||||
| 
 | ||||
|             try: | ||||
|                 return (float(res_mean) - float(res_median)) / float( | ||||
|                     res_stddev | ||||
|                 )  # convert from decimal | ||||
|             except ZeroDivisionError: | ||||
|                 return None | ||||
|         return None | ||||
|  | ||||
| @ -30,7 +30,7 @@ from metadata.profiler.orm.registry import CustomTypes | ||||
| Base = declarative_base() | ||||
| 
 | ||||
| _TYPE_MAP = { | ||||
|     DataType.NUMBER: sqlalchemy.INTEGER, | ||||
|     DataType.NUMBER: sqlalchemy.NUMERIC, | ||||
|     DataType.TINYINT: sqlalchemy.SMALLINT, | ||||
|     DataType.SMALLINT: sqlalchemy.SMALLINT, | ||||
|     DataType.INT: sqlalchemy.INT, | ||||
|  | ||||
| @ -30,6 +30,7 @@ from metadata.generated.schema.entity.data.table import ( | ||||
|     ColumnProfile, | ||||
|     ColumnProfilerConfig, | ||||
|     SystemProfile, | ||||
|     TableData, | ||||
|     TableProfile, | ||||
| ) | ||||
| from metadata.ingestion.processor.pii import NERScanner | ||||
| @ -442,7 +443,11 @@ class Profiler(Generic[TMetric]): | ||||
| 
 | ||||
|         return self | ||||
| 
 | ||||
|     def process(self, generate_sample_data: Optional[bool]) -> ProfilerResponse: | ||||
|     def process( | ||||
|         self, | ||||
|         generate_sample_data: Optional[bool], | ||||
|         process_pii_sensitive: Optional[bool], | ||||
|     ) -> ProfilerResponse: | ||||
|         """ | ||||
|         Given a table, we will prepare the profiler for | ||||
|         all its columns and return all the run profilers | ||||
| @ -454,40 +459,13 @@ class Profiler(Generic[TMetric]): | ||||
| 
 | ||||
|         self.compute_metrics() | ||||
|         if generate_sample_data: | ||||
|             try: | ||||
|                 logger.info( | ||||
|                     f"Fetching sample data for {self.profiler_interface.table_entity.fullyQualifiedName.__root__}..." | ||||
|                 ) | ||||
|                 sample_data = self.profiler_interface.fetch_sample_data(self.table) | ||||
| 
 | ||||
|                 if self.profiler_interface.source_config.processPiiSensitive: | ||||
|                     try: | ||||
|                         entity_scanner = NERScanner( | ||||
|                             metadata=self.profiler_interface.ometa_client | ||||
|                         ) | ||||
|                         entity_scanner.process( | ||||
|                             sample_data, | ||||
|                             self.profiler_interface.table_entity, | ||||
|                             self.profiler_interface.ometa_client, | ||||
|                         ) | ||||
|                     except Exception as exc: | ||||
|                         logger.warning( | ||||
|                             f"Unexpected error while processing sample data for auto pii tagging - {exc}" | ||||
|                         ) | ||||
|                         logger.debug(traceback.format_exc()) | ||||
| 
 | ||||
|                 logger.info( | ||||
|                     "Successfully fetched sample data for " | ||||
|                     f"{self.profiler_interface.table_entity.fullyQualifiedName.__root__}..." | ||||
|                 ) | ||||
|             except Exception as err: | ||||
|                 logger.debug(traceback.format_exc()) | ||||
|                 logger.warning(f"Error fetching sample data: {err}") | ||||
|                 sample_data = None | ||||
| 
 | ||||
|             sample_data = self.generate_sample_data() | ||||
|         else: | ||||
|             sample_data = None | ||||
| 
 | ||||
|         if process_pii_sensitive and sample_data: | ||||
|             self.process_pii_sensitive(sample_data) | ||||
| 
 | ||||
|         profile = self._check_profile_and_handle(self.get_profile()) | ||||
| 
 | ||||
|         table_profile = ProfilerResponse( | ||||
| @ -498,6 +476,45 @@ class Profiler(Generic[TMetric]): | ||||
| 
 | ||||
|         return table_profile | ||||
| 
 | ||||
|     def generate_sample_data(self) -> TableData: | ||||
|         """Fetch and ingest sample data | ||||
| 
 | ||||
|         Returns: | ||||
|             TableData: sample data | ||||
|         """ | ||||
|         try: | ||||
|             logger.info( | ||||
|                 "Fetching sample data for " | ||||
|                 f"{self.profiler_interface.table_entity.fullyQualifiedName.__root__}..."  # type: ignore | ||||
|             ) | ||||
|             return self.profiler_interface.fetch_sample_data(self.table) | ||||
|         except Exception as err: | ||||
|             logger.debug(traceback.format_exc()) | ||||
|             logger.warning(f"Error fetching sample data: {err}") | ||||
|             return None | ||||
| 
 | ||||
|     def process_pii_sensitive(self, sample_data: TableData) -> None: | ||||
|         """Read sample data to find pii sensitive columns and tag them | ||||
|         as PII sensitive data | ||||
| 
 | ||||
|         Args: | ||||
|             sample_data (TableData): sample data | ||||
|         """ | ||||
|         try: | ||||
|             entity_scanner = NERScanner( | ||||
|                 metadata=self.profiler_interface.ometa_client  # type: ignore | ||||
|             ) | ||||
|             entity_scanner.process( | ||||
|                 sample_data, | ||||
|                 self.profiler_interface.table_entity,  # type: ignore | ||||
|                 self.profiler_interface.ometa_client,  # type: ignore | ||||
|             ) | ||||
|         except Exception as exc: | ||||
|             logger.warning( | ||||
|                 f"Unexpected error while processing sample data for auto pii tagging - {exc}" | ||||
|             ) | ||||
|             logger.debug(traceback.format_exc()) | ||||
| 
 | ||||
|     def get_profile(self) -> CreateTableProfileRequest: | ||||
|         """ | ||||
|         After executing the profiler, get all results | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Teddy
						Teddy