-
Notifications
You must be signed in to change notification settings - Fork 46
/
Copy pathbenchmark.py
executable file
·233 lines (192 loc) · 10.3 KB
/
benchmark.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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
#!/usr/bin/env python
# TODO
# Why are records over & underclumped? ie. Charles Linder 7475467, Nirmala Ramanujam 7570988, Edwin L. Thomas 7799416
# Investigate "WEIRDNESS" output : Jaro-Winkler exception thrown on comparison between and ROBERT BRUCE
# Underclumping: unique records vs. total records
import sqlite3
import csv
import datetime;
# http://docs.python.org/howto/argparse.html
import argparse
import sys
sys.path.append( './lib/' )
sys.path.append("lib")
from fwork import *;
def bmVerify(results, filepath="", outdir = ""):
"""
Analysis function on disambiguation results, assuming that all benchmark data
are in the large results dataset.
Creates analysis detail csv file and prints summary information on
over- and underclumping statistics.
Running from the command line (make sure to set correct file paths in file)
python bmVerify_v3.py "input filepath" "output directory" databases
example:
python bmVerify_v3.py /home/ysun/disambig/newcode/all/ /home/ayu/results_v2/ invpatC_NBNA.good.Jan2011 invpatC_NBYA.good.Jan2011 invpatC_YBNA.good.Jan2011
Running interactively:
import bmVerify_v3
bmVerify(['final_r7', 'final_r8'], filepath="/home/ysun/disambig/newcode/all/", outdir = "/home/ayu/results_v2/")
"""
for result in results:
#uqB = "Unique_Inventor_ID2"
#uqB = "final_1"
#uqB = "final_2"
#uqB = "final_3"
#uqB = "final_4"
uqB = "final_5"
#uqB = "final_6"
tblB = "invpat"
#fileS = "/home/doolin/src/patentbenchmarks/combined.csv"
#fileS = "/home/doolin/src/patentbenchmarks/DefTruth5.csv"
#fileS = "/home/doolin/src/patentbenchmarks/berkeley.csv"
fileS = "/home/doolin/src/patentbenchmarks/combined.csv"
#fileS = "/home/doolin/src/patentbenchmarks/pister.csv"
#fileS = "/home/doolin/src/patentbenchmarks/paulrgray.csv"
#fileS = "/home/doolin/src/patentbenchmarks/allbritton.csv"
#fileS = "/home/doolin/src/patentbenchmarks/siddhu.csv"
#fileS = "/var/share/patentdata/disambiguation/experiments/earth/berkeley/benchmark.csv"
fileB = filepath + "{result}.sqlite3".format(result=result)
output = outdir + "{result}_DT5.csv".format(result=result)
t=datetime.datetime.now()
print "Start time: " + str(datetime.datetime.now())
class freqUQ:
def __init__(self):
self.list=[]
def step(self, value):
self.list.append(value)
def finalize(self):
return sorted([(self.list.count(x), x) for x in set(self.list)], reverse=True)[0][1]
#MAKE THIS SO IT CAN ATTACH SQLITE3 FOR BENCHMARK
dataS = uniVert([x for x in csv.reader(open(fileS, "rb"))])
#print dataS
#1 = Variables, 2 = Type, 3 = Format (If necessary), 4 = Matching Type
tList = ["%s %s" % (dataS[0][i], x) for i,x in enumerate(dataS[1]) if x != ""]
dataS2 = [dataS[0]]
dataS2.extend(dataS[4:])
#print dataS[2]
#Format if its necessary --> Basically for Patents..
for i,x in enumerate(dataS[2]):
if x!="":
for j in xrange(1,len(dataS2)):
if dataS2[j][i].isdigit():
dataS2[j][i] = x % int(dataS2[j][i])
conn = sqlite3.connect(":memory:")
#conn = sqlite3.connect("benchmark.sqlite3")
conn.create_function("jarow", 2, jarow)
conn.create_function("errD", 2, lambda x,y: (x!=y) and 1 or None)
conn.create_aggregate("freqUQ", 1, freqUQ)
c = conn.cursor()
#FIGURE OUT WHICH ONES HAVE EXACT/FUZZY
exact = [dataS[0][i] for i,x in enumerate(dataS[3]) if x.upper()[0]=="E"]
print "Exact: ", exact
fuzzy = [dataS[0][i] for i,x in enumerate(dataS[3]) if x.upper()[0]=="F"]
print "Fuzzy: ", fuzzy
uqS = [dataS[0][i] for i,x in enumerate(dataS[3]) if x.upper()[0]=="U"][0]
#CREATE INDEX, MERGE DATA BASED ON EXACTS
print "Creating indices... " + str(datetime.datetime.now())
exAnd = " AND ".join(["a.%s=b.%s" % (x, x) for x in exact])
exCom = ", ".join(exact)
if fileB.split(".")[-1].lower()=="csv":
dataB = uniVert([x for x in csv.reader(open("%s" % fileB, "rb"))])
quickSQL(c, data=dataB, table="dataB", header=True, typeList=["Patent VARCHAR"])
c.execute("CREATE INDEX IF NOT EXISTS dB_E ON dataB (%s)" % exCom)
c.execute("CREATE INDEX IF NOT EXISTS dB_U ON dataB (%s)" % uqB)
fBnme = "dataB"
else:
c.execute("ATTACH DATABASE '%s' AS db" % fileB)
if tblB=="":
fBnme = "db.%s" % fileB.split(".")[-2].split("/")[-1]
else:
fBnme = "db.%s" % tblB
quickSQL(c, data=dataS2, table="dataS", header=True, typeList=tList)
if fuzzy:
c.execute("CREATE INDEX IF NOT EXISTS dS_E ON dataS (%s);" % (exCom))
c.executescript("""
CREATE INDEX IF NOT EXISTS dS_E ON dataS (%s);
/* RETAIN ONLY JARO>0.9 FUZZY AND EXACT MATCHES */
CREATE TABLE dataM AS
SELECT a.*, %s AS uqB, %s AS uqS, %s AS jaro
FROM %s AS a
INNER JOIN dataS AS b
ON %s
WHERE jaro>0.90;
/* DETERMINE MAXIMUM JARO FOR EACH UQ AND EXACT COMBO */
CREATE TABLE dataT AS
SELECT uqS, %s, MAX(jaro) AS jaro, count(*) as cnt
FROM dataM
GROUP BY uqS, %s;
/* RETAIN ONLY MAXIMUM JARO */
CREATE TABLE dataM2 AS
SELECT a.*
FROM dataM AS a
INNER JOIN dataT AS b
ON a.uqS=b.uqS AND a.jaro=b.jaro AND %s;
""" % (exCom, uqB, uqS,
"*".join(["jarow(a.%s, b.%s)" % (x,x) for x in fuzzy]),
fBnme, exAnd, exCom, exCom, exAnd))
else:
c.executescript("""
CREATE INDEX IF NOT EXISTS dS_E ON dataS (%s);
CREATE TABLE dataM2 AS
SELECT *, %s AS uqB, %s AS uqS
FROM %s AS a
INNER JOIN dataS AS b
ON %s;
""" % (exCom, uqB, uqS, fBnme, exAnd))
c.executescript("""
/* EXPAND UNIQUE BASE AND INDICATE ACTIVE MATCHES */
CREATE TABLE dataM3 AS
SELECT uqS, a.*
FROM (SELECT uqS AS uqSUB, a.*
FROM (SELECT uqB, b.*
FROM (SELECT DISTINCT(uqB) FROM dataM2 WHERE uqB!="") AS a
INNER JOIN %s AS b
ON a.uqB=b.%s) AS a
LEFT JOIN (SELECT %s, uqB, uqS FROM dataM2) AS b
ON a.uqB=b.uqB AND %s) AS a
INNER JOIN (SELECT DISTINCT uqB, uqS FROM dataM2) AS b
ON a.%s=b.uqB;
/* INDICATE INVENTORS WHO DO NOT MATCH */
CREATE TABLE dataM4 AS
SELECT errD(a.ErrUQ, uqB) AS ErrUQ, b.*
FROM (SELECT uqS, freqUQ(uqB) as ErrUQ FROM dataM3 GROUP BY uqS) AS a
INNER JOIN dataM3 AS b
ON a.uqS=b.uqS AND b.AppYear <= '2012' /*AND a.uqS not in (83, 85, 93)*/
ORDER BY uqS, %s;
""" % (fBnme, uqB, exCom, exAnd, uqB, exCom))
print "Indices Done ... " + str(datetime.datetime.now())
#EXPORT THE RESULTS
writer = csv.writer(open(output, "wb"), lineterminator="\n")
writer.writerows([[x[1] for x in c.execute("PRAGMA TABLE_INFO(dataM4)")]])
writer.writerows(c.execute("SELECT * FROM dataM4").fetchall())
print "Printing results ..." + str(datetime.datetime.now())
rep = [list(x) for x in c.execute("SELECT ErrUQ, uqSUB FROM dataM4")]
#print "Rep: ", rep
orig = len([x for x in rep if x[1]!=None])
errm = sum([int(x[0]) for x in rep if x[0]!=None])
#print errm
u = 1.0*errm/orig
o = 1-(float(orig)/len(rep))
recall = 1.0 - u
# overclumping is lumping
# underclumping is splitting
print """
RESULTS ==================
Original: {original}
New Records: {new}
Total: {total}
Overclump: {overclump} ({o:.2%})
Underclump: {underclump} ({u:.2%})
Precision: {precision:.2%}
Recall: {recall:.2%}
File Detail: {filename}
Time: {time}
""".format(original = orig, new = len(rep)-orig, total = len(rep), overclump = len(rep)-orig, o = o,
underclump = errm, u = u, recall = recall, precision = recall/(recall+o), filename = output, time = datetime.datetime.now()-t)
c.close()
conn.close()
if __name__ == "__main__":
import sys
if(sys.argv[1] == 'help' or sys.argv[1] == '?'):
print bmVerify.__doc__
else:
bmVerify(sys.argv[3:], sys.argv[1], sys.argv[2])