Project

General

Profile

1 5079 aaronmk
#!/usr/bin/env python
2 5640 aaronmk
# Scrubs the taxonlabels in VegBIEN using TNRS.
3 5079 aaronmk
# Runs continuously until no new rows are added after max_pause.
4
5
import os.path
6
import StringIO
7
import sys
8
import time
9
10
sys.path.append(os.path.dirname(__file__)+"/../lib")
11
12
import csvs
13
import opts
14 5098 aaronmk
import profiling
15 5079 aaronmk
import sql
16
import sql_gen
17
import sql_io
18
import streams
19
import strings
20
import tnrs
21
22
# Config
23 5156 aaronmk
pause = 2*60*60 # sec; = 2 hr
24
max_pause = 9*60*60 # sec; = 9 hr; must be >= max partition import time (1.5 hr)
25 5079 aaronmk
assert pause <= max_pause
26
27
tnrs_data = sql_gen.Table('tnrs')
28
29
def main():
30
    # Input
31
    env_names = []
32
    db_config = opts.get_env_vars(sql.db_config_names, None, env_names)
33
    verbosity = float(opts.get_env_var('verbosity', 3, env_names))
34 5214 aaronmk
    wait = opts.env_flag('wait', False, env_names)
35 5079 aaronmk
    if not 'engine' in db_config: raise SystemExit('Usage: '
36
        +opts.env_usage(env_names)+' '+sys.argv[0]+' 2>>log')
37
38
    def log(msg, level=1):
39
        '''Higher level -> more verbose'''
40
        if level <= verbosity:
41
            sys.stderr.write(strings.to_raw_str(msg.rstrip('\n')+'\n'))
42
43
    # Connect to DB
44
    db = sql.connect(db_config, log_debug=log)
45
46 5124 aaronmk
    tnrs_profiler = profiling.ItersProfiler(iter_text='name')
47 5123 aaronmk
48 5640 aaronmk
    # Iterate over unscrubbed verbatim taxonlabels
49 5123 aaronmk
    total_pause = 0
50 5640 aaronmk
    tables = ['taxonlabel', sql_gen.Join('tnrs',
51 5661 aaronmk
        {'Name_submitted': 'taxonomicname'}, sql_gen.filter_out)]
52
    cols = ['taxonomicname']
53 5159 aaronmk
    # Has a concatenated name and not already linked to an accepted name
54 5661 aaronmk
    conds = [('taxonomicname', sql_gen.CompareCond(None, '!=')),
55 5640 aaronmk
        ('matched_label_id', None)]
56 5123 aaronmk
    while True:
57
        # Fetch next set
58 5643 aaronmk
        cur = sql.select(db, tables, cols, conds, cols, limit=tnrs.max_names,
59
            cacheable=False)
60 5123 aaronmk
        this_ct = cur.rowcount
61 5640 aaronmk
        log('Processing '+str(this_ct)+' taxonlabels')
62 5123 aaronmk
        if this_ct == 0:
63 5213 aaronmk
            if not wait: break
64 5212 aaronmk
            log('Waited '+str(total_pause)+' sec total')
65 5123 aaronmk
            total_pause += pause
66
            if total_pause > max_pause: break
67 5212 aaronmk
            log('Waiting '+str(pause)+' sec...')
68 5123 aaronmk
            time.sleep(pause) # wait for more rows
69
            continue # try again
70
        # otherwise, rows found
71 5079 aaronmk
        total_pause = 0
72 5123 aaronmk
        names = list(sql.values(cur))
73
74
        # Run TNRS
75
        log('Making TNRS request')
76
        tnrs_profiler.start()
77
        try:
78
            try: stream = tnrs.repeated_tnrs_request(names)
79
            finally:
80
                tnrs_profiler.stop(iter_ct=this_ct)
81
                log('Cumulatively: '+tnrs_profiler.msg())
82
        except tnrs.InvalidResponse: pass # skip set in case it caused error
83
        else:
84
            log('Storing TNRS response data')
85 5590 aaronmk
            sql_io.append_csv(db, tnrs_data, *csvs.reader_and_header(stream))
86 5079 aaronmk
87
main()