-
Notifications
You must be signed in to change notification settings - Fork 586
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Get patterns from regex match in ITIN recognizer #959
Closed
aperezfals
wants to merge
15
commits into
microsoft:main
from
aperezfals:get_patterns_from_regex_match
+468
−44
Closed
Changes from 11 commits
Commits
Show all changes
15 commits
Select commit
Hold shift + click to select a range
77c772e
Get patterns from regex match in itin recognizer
aperezfals fcc7465
Change the improve pattern function to the pattern object
aperezfals 9980e89
update get_improved_pattern doc
aperezfals 9c93650
Merge branch 'main' into get_patterns_from_regex_match
aperezfals 988fdd9
Merge branch 'main' into get_patterns_from_regex_match
omri374 74f8711
Merge branch 'main' into get_patterns_from_regex_match
omri374 a6ce31e
Rollback changes to pattern recognizer
aperezfals b46cd7c
Add improvable pattern recognizer
aperezfals 66c280e
Add string sanitaizer classes
aperezfals a5989b1
Implement itin and ssn recognizer as improvable
aperezfals 93a7efc
Rollback pattern
aperezfals f35c52b
Merge branch 'main' into get_patterns_from_regex_match
omri374 0233fa1
Merge branch 'main' into get_patterns_from_regex_match
aperezfals 808b2d2
Merge branch 'main' into get_patterns_from_regex_match
omri374 cca167f
Merge branch 'main' into get_patterns_from_regex_match
omri374 File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
from presidio_analyzer import Pattern | ||
from typing import Optional, Callable | ||
from regex import Match | ||
from .analysis_explanation import AnalysisExplanation | ||
|
||
|
||
class ImprovablePattern(Pattern): | ||
""" | ||
A class that represents a regex pattern. | ||
|
||
:param name: the name of the pattern | ||
:param regex: the regex pattern to detect | ||
:param score: the pattern's strength (values varies 0-1) | ||
:param get_improved_pattern_func: a function that improve the score of the analysis explanation | ||
based on the regex match info. | ||
Can be used when is needed to improve the score based on detected groups in the regex match | ||
or any logic applied only to this pattern. | ||
""" | ||
|
||
def __init__( | ||
self, | ||
name: str, | ||
regex: str, | ||
score: float, | ||
improve_score_fn: Optional[ | ||
Callable[["ImprovablePattern", str, Match, AnalysisExplanation], None] | ||
] = None, | ||
) -> None: | ||
|
||
super().__init__(name, regex, score) | ||
self.improve_score_fn = improve_score_fn | ||
|
||
def improve_score( | ||
self, matched_text: str, match: Match, analysis_explanation: AnalysisExplanation | ||
) -> None: | ||
if self.improve_score_fn: | ||
self.improve_score_fn(self, matched_text, match, analysis_explanation) |
248 changes: 248 additions & 0 deletions
248
presidio-analyzer/presidio_analyzer/improvable_pattern_recognizer.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,248 @@ | ||
import datetime | ||
import logging | ||
from typing import List, Dict | ||
from regex import Match | ||
|
||
import regex as re | ||
|
||
from presidio_analyzer import ( | ||
LocalRecognizer, | ||
Pattern, | ||
RecognizerResult, | ||
EntityRecognizer, | ||
AnalysisExplanation, | ||
ImprovablePattern, | ||
) | ||
|
||
from presidio_analyzer.nlp_engine import NlpArtifacts | ||
|
||
logger = logging.getLogger("presidio-analyzer") | ||
|
||
|
||
class ImprovablePatternRecognizer(LocalRecognizer): | ||
""" | ||
PII entity recognizer using regular expressions or deny-lists. | ||
Analysis explanations can be improved by a pattern or by the recognizer. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It is not just the explanations that could be improved, but also the score. Am I correct? |
||
|
||
:param patterns: A list of patterns to detect | ||
:param deny_list: A list of words to detect, | ||
in case our recognizer uses a predefined list of words (deny list) | ||
:param context: list of context words | ||
:param deny_list_score: confidence score for a term | ||
identified using a deny-list | ||
""" | ||
|
||
def __init__( | ||
self, | ||
supported_entity: str, | ||
name: str = None, | ||
supported_language: str = "en", | ||
patterns: List[ImprovablePattern] = None, | ||
deny_list: List[str] = None, | ||
context: List[str] = None, | ||
deny_list_score: float = 1.0, | ||
version: str = "0.0.1", | ||
): | ||
|
||
if not supported_entity: | ||
raise ValueError("Pattern recognizer should be initialized with entity") | ||
|
||
if not patterns and not deny_list: | ||
raise ValueError( | ||
"Pattern recognizer should be initialized with patterns" | ||
" or with deny list" | ||
) | ||
|
||
super().__init__( | ||
supported_entities=[supported_entity], | ||
supported_language=supported_language, | ||
name=name, | ||
version=version, | ||
) | ||
if patterns is None: | ||
self.patterns = [] | ||
else: | ||
self.patterns = patterns | ||
self.context = context | ||
self.deny_list_score = deny_list_score | ||
|
||
if deny_list: | ||
deny_list_pattern = self._deny_list_to_regex(deny_list) | ||
self.patterns.append(deny_list_pattern) | ||
self.deny_list = deny_list | ||
else: | ||
self.deny_list = [] | ||
|
||
def load(self): # noqa D102 | ||
pass | ||
|
||
def analyze( | ||
self, | ||
text: str, | ||
entities: List[str], | ||
nlp_artifacts: NlpArtifacts = None, | ||
regex_flags: int = None, | ||
) -> List[RecognizerResult]: | ||
""" | ||
Analyzes text to detect PII using regular expressions or deny-lists. | ||
|
||
:param text: Text to be analyzed | ||
:param entities: Entities this recognizer can detect | ||
:param nlp_artifacts: Output values from the NLP engine | ||
:param regex_flags: | ||
:return: | ||
""" | ||
results = [] | ||
|
||
if self.patterns: | ||
pattern_result = self.__analyze_patterns(text, regex_flags) | ||
results.extend(pattern_result) | ||
|
||
return results | ||
|
||
def _deny_list_to_regex(self, deny_list: List[str]) -> Pattern: | ||
""" | ||
Convert a list of words to a matching regex. | ||
|
||
To be analyzed by the analyze method as any other regex patterns. | ||
|
||
:param deny_list: the list of words to detect | ||
:return:the regex of the words for detection | ||
""" | ||
# Escape deny list elements as preparation for regex | ||
escaped_deny_list = [re.escape(element) for element in deny_list] | ||
regex = r"(?:^|(?<=\W))(" + "|".join(escaped_deny_list) + r")(?:(?=\W)|$)" | ||
return Pattern(name="deny_list", regex=regex, score=self.deny_list_score) | ||
|
||
@staticmethod | ||
def build_regex_explanation( | ||
recognizer_name: str, | ||
pattern_name: str, | ||
pattern: str, | ||
original_score: float, | ||
validation_result: bool, | ||
) -> AnalysisExplanation: | ||
""" | ||
Construct an explanation for why this entity was detected. | ||
|
||
:param recognizer_name: Name of recognizer detecting the entity | ||
:param pattern_name: Regex pattern name which detected the entity | ||
:param pattern: Regex pattern logic | ||
:param original_score: Score given by the recognizer | ||
:param validation_result: Whether validation was used and its result | ||
:return: Analysis explanation | ||
""" | ||
explanation = AnalysisExplanation( | ||
recognizer=recognizer_name, | ||
original_score=original_score, | ||
pattern_name=pattern_name, | ||
pattern=pattern, | ||
validation_result=validation_result, | ||
) | ||
return explanation | ||
|
||
def __analyze_patterns( | ||
self, text: str, flags: int = None | ||
) -> List[RecognizerResult]: | ||
""" | ||
Evaluate all patterns in the provided text. | ||
|
||
Including words in the provided deny-list | ||
|
||
:param text: text to analyze | ||
:param flags: regex flags | ||
:return: A list of RecognizerResult | ||
""" | ||
flags = flags if flags else re.DOTALL | re.MULTILINE | ||
results = [] | ||
for pattern in self.patterns: | ||
match_start_time = datetime.datetime.now() | ||
matches = re.finditer(pattern.regex, text, flags=flags) | ||
match_time = datetime.datetime.now() - match_start_time | ||
logger.debug( | ||
"--- match_time[%s]: %s.%s seconds", | ||
pattern.name, | ||
match_time.seconds, | ||
match_time.microseconds, | ||
) | ||
|
||
for match in matches: | ||
start, end = match.span() | ||
|
||
current_match = text[start:end] | ||
|
||
# Skip empty results | ||
if current_match == "": | ||
continue | ||
|
||
description = self.build_regex_explanation( | ||
self.name, pattern.name, pattern.regex, pattern.score, False | ||
) | ||
|
||
pattern.improve_score(current_match, match, description) | ||
self.improve_score(pattern, current_match, match, description) | ||
|
||
pattern_result = RecognizerResult( | ||
entity_type=self.supported_entities[0], | ||
start=start, | ||
end=end, | ||
score=description.score, | ||
analysis_explanation=description, | ||
recognition_metadata={ | ||
RecognizerResult.RECOGNIZER_NAME_KEY: self.name, | ||
RecognizerResult.RECOGNIZER_IDENTIFIER_KEY: self.id, | ||
}, | ||
) | ||
|
||
if pattern_result.score > EntityRecognizer.MIN_SCORE: | ||
results.append(pattern_result) | ||
|
||
# Update analysis explanation score following validation or invalidation | ||
description.score = pattern_result.score | ||
|
||
results = EntityRecognizer.remove_duplicates(results) | ||
return results | ||
|
||
def to_dict(self) -> Dict: | ||
"""Serialize instance into a dictionary.""" | ||
return_dict = super().to_dict() | ||
|
||
return_dict["patterns"] = [pat.to_dict() for pat in self.patterns] | ||
return_dict["deny_list"] = self.deny_list | ||
return_dict["context"] = self.context | ||
return_dict["supported_entity"] = return_dict["supported_entities"][0] | ||
del return_dict["supported_entities"] | ||
|
||
return return_dict | ||
|
||
@classmethod | ||
def from_dict(cls, entity_recognizer_dict: Dict) -> "ImprovablePatternRecognizer": | ||
"""Create instance from a serialized dict.""" | ||
patterns = entity_recognizer_dict.get("patterns") | ||
if patterns: | ||
patterns_list = [Pattern.from_dict(pat) for pat in patterns] | ||
entity_recognizer_dict["patterns"] = patterns_list | ||
|
||
return cls(**entity_recognizer_dict) | ||
|
||
def improve_score( | ||
self, | ||
pattern: ImprovablePattern, | ||
matched_text: str, | ||
match: Match, | ||
analysis_explanation: AnalysisExplanation, | ||
): | ||
""" | ||
Logic to improve the recognizer score. | ||
|
||
Override this method if is needed an improvement | ||
of the score at recognizer level. It will be applied | ||
to all patterns. | ||
|
||
:param pattern: the pattern that matched. | ||
:param pattern_text: the text matched. | ||
:param match: the regex match info. | ||
:param analysis_explanation: the analysis explanation that should improved. | ||
Modify the score and explanation. | ||
""" | ||
pass |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.