-
Notifications
You must be signed in to change notification settings - Fork 46
/
Copy pathbuild_locmatch.py
executable file
·396 lines (364 loc) · 16.5 KB
/
build_locmatch.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
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
#Non_US http://earth-info.nga.mil/gns/html/gis_countryfiles.htm
#US http://geonames.usgs.gov/domestic/download_data.htm
##NEED TO DO...
## CREATE INDEX IF NOT EXISTS idx_ctc0 ON gnsloc (SORT_NAME, CC1);
# TODO: Take advantage of SQLite
import datetime, csv, os, re, sqlite3, sys
sys.path.append("lib")
import SQLite
from fwork import *
from config import *
def sep_wrd(word, seq):
if seq==-1:
return word
else:
p = re.compile(" *?[,|] *")
ln = p.split(word)
if len(ln)> seq:
return ln[seq]
else:
return ""
s = SQLite.SQLite(loc["db"])
s.conn.create_function("blk_split", 1, lambda x: re.sub(" ", "", x))
s.conn.create_function("sep_cnt", 1, lambda x: len(re.findall("[,|]", x)))
s.conn.create_function("jarow", 2, jarow)
s.conn.create_function("cityctry", 3, cityctry)
s.conn.create_function("sep_wrd", 2, sep_wrd)
s.conn.create_function("rev_wrd", 2, lambda x,y:x.upper()[::-1][:y])
s.optimize()
s.attach("{dir}/assignee.sqlite3".format(dir=loc["s3_dir"]), name="asg")
s.attach("{dir}/inventor.sqlite3".format(dir=loc["s3_dir"]), name="inv")
s.c.executescript("""
/* DROP TABLE IF EXISTS loc; */
CREATE TABLE IF NOT EXISTS loc (
Cnt INTEGER,
City VARCHAR(10), State VARCHAR(2),
Country VARCHAR(2), Zipcode VARCHAR(5),
City3 VARCHAR,
NCity VARCHAR(10), NState VARCHAR(2),
NCountry VARCHAR(2),
UNIQUE(City,State,Country,Zipcode));
""") #"""
if not s.tables(lookup="locMerge"):
print datetime.datetime.now()
s.chgTbl("loc")
s.c.executescript("""
CREATE TEMPORARY TABLE temp AS
SELECT Upper(City) as CityX, Upper(State) as StateX,
Upper(Country) as CountryX, count(*) as Cnt
FROM asg.assignee
WHERE City!=""
GROUP BY CityX, StateX, CountryX;
CREATE TEMPORARY TABLE temp2 AS
SELECT sum(Cnt) as Cnt,
cityctry(CityX, CountryX, 'city') as CityY, StateX as StateY,
cityctry(CityX, CountryX, 'ctry') as CtryY, '' as ZipcodeY
FROM temp
WHERE CityY!=""
GROUP BY CityY, StateY, CtryY;
INSERT OR REPLACE INTO loc
SELECT a.*, SUBSTR(CityY,1,3), b.NewCity, b.NewState, b.NewCountry
FROM temp2 AS a
LEFT JOIN typos AS b
ON a.CityY=b.City AND a.StateY=b.State AND a.CtryY=b.Country;
DROP TABLE temp;
DROP TABLE temp2;
CREATE TEMPORARY TABLE temp AS
SELECT Upper(City) as CityX, Upper(State) as StateX,
Upper(Country) as CountryX, Zipcode, count(*) as Cnt
FROM inv.inventor
WHERE City!="" OR (City="" AND Zipcode!="")
GROUP BY CityX, StateX, CountryX, Zipcode;
CREATE TEMPORARY TABLE temp2 AS
SELECT sum(Cnt) as Cnt,
cityctry(CityX, CountryX, 'city') as CityY, StateX as StateY,
cityctry(CityX, CountryX, 'ctry') as CtryY, Zipcode as ZipcodeY
FROM temp
WHERE CityY!=""
GROUP BY CityY, StateY, CtryY, ZipcodeY;
INSERT OR REPLACE INTO loc
SELECT a.*, SUBSTR(CityY,1,3), b.NewCity, b.NewState, b.NewCountry
FROM temp2 AS a
LEFT JOIN typos AS b
ON a.CityY=b.City AND a.StateY=b.State AND a.CtryY=b.Country;
DROP TABLE temp;
DROP TABLE temp2;
""") #"""
s.index(["City3", "State", "Country"])
s.index(["City", "State"])
s.index(["City", "Country"])
s.index(["NCity", "NCountry"])
s.index(["NCity", "NState"])
s.index(["NCity", "NState", "NCountry"])
print datetime.datetime.now()
s.c.executescript("""
CREATE TABLE IF NOT EXISTS uslocM AS
SELECT Zipcode, Latitude AS Lat, Longitude AS Long, Upper(City) as City,
BLK_SPLIT(Upper(City)) as BlkCity,
SUBSTR(UPPER(BLK_SPLIT(City)),1,3) as City3,
REV_WRD(BLK_SPLIT(City), 4) as City4R,
Upper(State) as State, "US" as Country
FROM usloc
GROUP BY City, State;
""") #"""
s.chgTbl("uslocM")
s.index(["Zipcode"])
s.index(["City", "State"])
s.index(["BlkCity", "State"])
s.index(["City3", "State"])
s.index(["City4R", "State"])
s.c.executescript("""
CREATE TABLE IF NOT EXISTS locMerge (
Mtch INTEGER,
Val FLOAT, Cnt INTEGER,
City VARCHAR, State VARCHAR,
Country VARCHAR, Zipcode VARCHAR,
NCity VARCHAR, NState VARCHAR,
NCountry VARCHAR, NZipcode VARCHAR,
NLat FLOAT, NLong FLOAT,
City3 VARCHAR,
UNIQUE(City, State, Country, Zipcode));
""") #"""
s.chgTbl("locMerge")
s.index(["City3", "State", "Country"])
s.index(["City", "State"])
s.index(["City", "Country"])
s.index(["City", "State", "Country", "Zipcode"])
def replace_loc(script):
#ALLOWS US TO REPLACE THE PREV LOC DATASET
s.c.executescript("""
DROP TABLE IF EXISTS temp1;
CREATE TEMPORARY TABLE temp1 AS %s;
CREATE INDEX IF NOT EXISTS tmp1_idx ON temp1 (CityA, StateA, CountryA, ZipcodeA);;
""" % script) #"""
field = ["[%s]" % x[1] for x in s.c.execute("PRAGMA TABLE_INFO(temp1)")][2:6]
var_f = ",".join(field)
if s.c.execute("SELECT count(*) FROM temp1").fetchone()[0]>0:
s.c.executescript("""
CREATE TEMPORARY TABLE temp2 AS
SELECT count(*) as cnt, CityA, StateA, CountryA, ZipcodeA
FROM temp1
GROUP BY CityA, StateA, CountryA, ZipcodeA;
CREATE INDEX IF NOT EXISTS t1_idx ON temp1 (CityA, StateA, CountryA, ZipcodeA);
CREATE INDEX IF NOT EXISTS t2_idx ON temp2 (CityA, StateA, CountryA, ZipcodeA);
INSERT OR REPLACE INTO locMerge
SELECT b.cnt, a.*, SUBSTR(a.CityA,1,3)
FROM temp1 AS a
INNER JOIN temp2 AS b
ON a.CityA=b.CityA AND a.StateA=b.StateA AND a.CountryA=b.CountryA AND a.ZipcodeA=b.ZipcodeA;
CREATE TEMPORARY TABLE temp3 AS
SELECT a.*
FROM LOC AS a LEFT JOIN locMerge AS b
ON a.City=b.City AND a.State=b.State AND a.Country=b.Country AND a.Zipcode=b.Zipcode
WHERE b.Zipcode IS NULL;
DROP TABLE IF EXISTS loc;
CREATE TABLE loc AS SELECT * FROM temp3;
CREATE INDEX IF NOT EXISTS loc_idxCC ON loc (City, Country);
CREATE INDEX IF NOT EXISTS loc_idx ON loc (City, State, Country, Zipcode);
CREATE INDEX IF NOT EXISTS loc_idxCS ON loc (City, State);
DROP TABLE IF EXISTS temp2;
DROP TABLE IF EXISTS temp3;
""") #"""
VarX = s.c.execute("select count(*) from loc").fetchone()[0]
VarY = s.c.execute("select count(*) from locMerge").fetchone()[0]
print " - Loc =", VarX, " OkM =", VarY, " Total =", VarX+VarY, " ", datetime.datetime.now()
s.commit()
print "Loc =", s.c.execute("select count(*) from loc").fetchone()[0]
for scnt in range(-1, s.c.execute("select max(sep_cnt(city)) from loc").fetchone()[0]+1):
sep = scnt
print "------", scnt, "------"
##DOMESTIC
replace_loc("""
SELECT 11,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.city, b.state, 'US', b.zipcode, b.lat, b.long
FROM loc AS a INNER JOIN uslocM AS b
ON SEP_WRD(CityA, %d)=b.city AND StateA=b.state AND CountryA='US'
WHERE SEP_CNT(CityA)>=%d AND CityA!="";
""" % (sep, scnt)) #"""
##DOMESTIC (Blk Remove)
replace_loc("""
SELECT 11,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.city, b.state, 'US', b.zipcode, b.lat, b.long
FROM loc AS a INNER JOIN uslocM AS b
ON BLK_SPLIT(SEP_WRD(a.City, %d))=b.blkcity AND a.state=b.state AND a.country='US'
WHERE SEP_CNT(a.City)>=%d AND a.City!=""
""" % (sep, scnt)) #"""
##DOMESTIC FIRST3 (JARO WINKLER)
replace_loc("""
SELECT (10+jarow(BLK_SPLIT(SEP_WRD(a.City, %d)), b.BlkCity)) AS Jaro,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.city, b.state, 'US', b.zipcode, b.lat, b.long
FROM loc AS a INNER JOIN uslocM AS b
ON SUBSTR(BLK_SPLIT(SEP_WRD(a.City, %d)),1,3)=b.City3 AND a.state=b.state AND a.country='US'
WHERE jaro>%s AND SEP_CNT(a.City)>=%d AND a.City!=""
ORDER BY a.City, a.State, jaro
""" % (sep, sep, "10.92", scnt)) #"""
##DOMESTIC LAST4 (JARO WINKLER)
replace_loc("""
SELECT (10+jarow(BLK_SPLIT(SEP_WRD(a.City, %d)), b.BlkCity)) AS Jaro,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.city, b.state, 'US', b.zipcode, b.lat, b.long
FROM loc AS a INNER JOIN uslocM AS b
ON REV_WRD(BLK_SPLIT(SEP_WRD(a.City, %d)),4)=b.City4R AND a.state=b.state AND a.country='US'
WHERE jaro>%s AND SEP_CNT(a.City)>=%d AND a.City!=""
ORDER BY a.City, a.State, jaro
""" % (sep, sep, "10.90", scnt)) #"""
#------------------------------------------#
##FOREIGN COUNTRY (Full Name 1)
replace_loc("""
SELECT 21,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, "", b.cc1, "", b.lat, b.long
FROM loc AS a INNER JOIN gnsloc AS b
ON SEP_WRD(a.City, %d)=b.full_name_ro AND a.country=b.cc1
WHERE SEP_CNT(a.City)>=%d AND a.City!=""
""" % (sep, scnt)) #"""
##FOREIGN COUNTRY (Full Name 2)
replace_loc("""
SELECT 21,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, "", b.cc1, "", b.lat, b.long
FROM loc AS a INNER JOIN gnsloc AS b
ON SEP_WRD(a.City, %d)=b.full_name_nd_ro AND a.country=b.cc1
WHERE SEP_CNT(a.City)>=%d AND a.City!="";
""" % (sep, scnt)) #"""
##FOREIGN COUNTRY (Short Form)
replace_loc("""
SELECT 21,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, "", b.cc1, "", b.lat, b.long
FROM loc AS a INNER JOIN gnsloc AS b
ON SEP_WRD(a.City, %d)=b.short_form AND a.country=b.cc1
WHERE SEP_CNT(a.City)>=%d AND a.City!="";
""" % (sep, scnt)) #"""
##FOREIGN COUNTRY (Blk Split)
replace_loc("""
SELECT 21,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, "", b.cc1, "", b.lat, b.long
FROM loc AS a INNER JOIN gnsloc AS b
ON BLK_SPLIT(SEP_WRD(a.City, %d))=b.sort_name_ro AND a.country=b.cc1
WHERE SEP_CNT(a.City)>=%d AND a.City!="";
""" % (sep, scnt)) #"""
##FOREIGN COUNTRY FIRST3 (JARO WINKLER)
replace_loc("""
SELECT (20+jarow(BLK_SPLIT(SEP_WRD(a.City, %d)), b.sort_name_ro)) AS Jaro,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, "", b.cc1, "", b.lat, b.long
FROM loc AS a INNER JOIN gnsloc AS b
ON SUBSTR(BLK_SPLIT(SEP_WRD(a.City, %d)),1,3)=SUBSTR(b.sort_name_ro,1,3) AND a.country=b.cc1
WHERE jaro>%s AND SEP_CNT(a.City)>=%d AND a.City!=""
ORDER BY a.City, a.Country, jaro;
""" % (sep, sep, "20.92", scnt)) #"""
##FOREIGN COUNTRY LAST4 (JARO WINKLER)
replace_loc("""
SELECT (20+jarow(BLK_SPLIT(SEP_WRD(a.City, %d)), b.sort_name_ro)) AS Jaro,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, "", b.cc1, "", b.lat, b.long
FROM loc AS a INNER JOIN gnsloc AS b
ON REV_WRD(BLK_SPLIT(SEP_WRD(a.City, %d)),4)=REV_WRD(b.sort_name_ro,4) AND a.country=b.cc1
WHERE jaro>%s AND SEP_CNT(a.City)>=%d AND a.City!=""
ORDER BY a.City, a.Country, jaro;
""" % (sep, sep, "20.90", scnt)) #"""
print "------ F ------"
##DOMESTIC (2nd LAYER)
replace_loc("""
SELECT 15,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.city, b.state, 'US', b.zipcode, b.lat, b.long
FROM (SELECT *
FROM loc
WHERE NCity IS NOT NULL) AS a
INNER JOIN uslocM AS b
ON a.NCity=b.city AND a.NState=b.state AND a.NCountry='US';
""") #"""
##DOMESTIC FIRST3 (2nd, JARO WINKLER)
replace_loc("""
SELECT 14+jarow(BLK_SPLIT(a.NCity), b.BlkCity) AS Jaro,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.city, b.state, 'US', b.zipcode, b.lat, b.long
FROM (SELECT *
FROM loc
WHERE NCity IS NOT NULL) AS a
INNER JOIN uslocM AS b
ON SUBSTR(BLK_SPLIT(a.NCity),1,3)=b.City3 AND a.Nstate=b.state AND a.Ncountry='US'
WHERE jaro>%s
ORDER BY a.NCity, a.NState, jaro
""" % "14.95") #"""
##FOREIGN FULL NAME (2nd LAYER)
replace_loc("""
SELECT 25,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, '' as state, b.cc1, '' as zip, b.lat, b.long
FROM (SELECT *
FROM loc
WHERE NCity IS NOT NULL) AS a
INNER JOIN gnsloc AS b
ON a.NCity=b.full_name_ro AND a.NCountry=b.cc1;
""") #"""
##FOREIGN FULL ND (2nd LAYER)
replace_loc("""
SELECT 25,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, '' as state, b.cc1, '' as zip, b.lat, b.long
FROM (SELECT *
FROM loc
WHERE NCity IS NOT NULL) AS a
INNER JOIN gnsloc AS b
ON a.NCity=b.full_name_nd_ro AND a.NCountry=b.cc1;
""") #"""
##FOREIGN NO SPACE (2nd LAYER)
replace_loc("""
SELECT 25,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, '' as state, b.cc1, '' as zip, b.lat, b.long
FROM (SELECT *
FROM loc
WHERE NCity IS NOT NULL) AS a
INNER JOIN gnsloc AS b
ON BLK_SPLIT(a.NCity)=b.sort_name_ro AND a.NCountry=b.cc1;
""") #"""
##FOREIGN COUNTRY FIRST3 (2nd, JARO WINKLER)
replace_loc("""
SELECT 24+jarow(BLK_SPLIT(a.NCity), b.sort_name_ro) AS Jaro,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.full_name_nd_ro, '' as state, b.cc1, '' as zip, b.lat, b.long
FROM (SELECT *
FROM loc
WHERE NCity IS NOT NULL) AS a
INNER JOIN gnsloc AS b
ON SUBSTR(BLK_SPLIT(a.NCity),1,3)=SUBSTR(b.sort_name_ro,1,3) AND a.Ncountry=b.cc1
WHERE jaro>%s
ORDER BY a.NCity, a.NCountry, jaro;
""" % "24.95") #"""
##DOMESTIC ZIPCODE
replace_loc("""
SELECT 31,
a.cnt as cnt,
a.city as CityA, a.state as StateA, a.country as CountryA, a.zipcode as ZipcodeA,
b.City, b.State, 'US', b.zipcode, b.lat, b.long
FROM (SELECT *, (SEP_WRD(zipcode,0)+0) as Zip2
FROM loc
WHERE Zipcode!='' AND Country='US') AS a
INNER JOIN uslocM AS b
ON a.Zip2=b.Zipcode;
""") #"""
s.close()