-
Notifications
You must be signed in to change notification settings - Fork 494
/
Copy pathmsgClassify.py
100 lines (78 loc) · 3.24 KB
/
msgClassify.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
import json
import re
# convenience function: check if any topic term is contained within a sentence
def contains(a_list, a_sentence):
for item in a_list:
# handle patterned case (prefaced by asterisk)
# eg. "*dd:ddam" will catch the pattern "10:30am"
if item[0] == '*':
if re.search(item.replace('*', '').replace('d', '\d'), a_sentence):
return (item)
else:
if item.lower() in a_sentence.lower():
return (item)
# convenience function: split text into one or more sentences
def split(text, rows_split_delim=['.', '!', '?']):
# sentences after the split
rows_split = []
# pointer to position within text
pointer = 0
# don't try to split messages with links
if 'http' in text.lower():
return [text]
# loop through each character in the message text
for char in text:
# if character is a sentence delimeter
if char in rows_split_delim:
# split out the text from the previous pointer to this delimeter
sentence = text[pointer:text.index(char, pointer) + 1]
# remove extra spaces
sentence = sentence.lstrip().strip()
rows_split.append(sentence)
# update the pointer
pointer = text.index(char, pointer) + 1
# finish by splitting out the remaining text
# from the previous pointer to this delimeter
# this handles the case of text with no split sentences
sentence = text[pointer:].lstrip().strip()
rows_split.append(sentence)
return rows_split
# Classify class definition
class Classifier(object):
"""A classification object, for topics defined in a json definiton
Attributes:
topics_file: a json structure containing words/patterns for a list of topics
"""
def __init__(self, topics_file):
"""Return a Classify object given a topics definition file
topics.json
{
"checkin" : ["daily check","daily note","check ins","check-in","notes"],
"stipend" : ["stipend","paid", "pay", "money", "payment"],
"scheduling" : ["*d:dd","*dam","*dpm","*d am","*d pm"]
}
c = Classifier('../topics.json')
"""
# load topics and their words
try:
self.topics = json.load(open(topics_file))
except:
print ('error opening file', topics_file)
def classify(self, text):
"""Returns a json structure with classification for sentences within the text
c.classify("The nurse will be there. We'll be there around 10:30")
{'medpro': ['The nurse will be there.'], 'scheduling': ["We'll be there around 10:30"]}
"""
topics_data = {}
# split out sentences from the text
sentences = split(text)
for sentence in sentences:
# loop through the topics
for key in self.topics.keys():
# if the sentence contains any of the words for this topic, add to results
if contains(self.topics[key], sentence):
if key not in topics_data:
topics_data[key] = [sentence]
else:
topics_data[key].append(sentence)
return topics_data