1
|
#!/usr/bin/env python
|
2
|
# A DiGIR client
|
3
|
|
4
|
import os
|
5
|
import os.path
|
6
|
import sys
|
7
|
import urllib
|
8
|
import urllib2
|
9
|
import xml.dom.minidom as minidom
|
10
|
|
11
|
sys.path.append(os.path.dirname(__file__)+"/../lib")
|
12
|
|
13
|
import dates
|
14
|
import opts
|
15
|
import streams
|
16
|
import util
|
17
|
import xml_dom
|
18
|
import xpath
|
19
|
|
20
|
# Config
|
21
|
timeout = 20 # sec
|
22
|
chunk_size = 10000 # records
|
23
|
default_schema = 'http://digir.net/schema/conceptual/darwin/full/2003/1.0/darwin2full.xsd'
|
24
|
|
25
|
request_xml_template = '''\
|
26
|
<?xml version="1.0" encoding="UTF-8"?>
|
27
|
<request
|
28
|
xmlns="http://digir.net/schema/protocol/2003/1.0"
|
29
|
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
|
30
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
31
|
xmlns:digir="http://digir.net/schema/protocol/2003/1.0"
|
32
|
xmlns:darwin="http://digir.net/schema/conceptual/darwin/2003/1.0"
|
33
|
xmlns:dwc="http://digir.net/schema/conceptual/darwin/2003/1.0"
|
34
|
xsi:schemaLocation="http://digir.net/schema/protocol/2003/1.0
|
35
|
http://digir.sourceforge.net/schema/protocol/2003/1.0/digir.xsd
|
36
|
http://digir.net/schema/conceptual/darwin/2003/1.0
|
37
|
http://digir.sourceforge.net/schema/conceptual/darwin/2003/1.0/darwin2.xsd">
|
38
|
<header>
|
39
|
<version>1.0</version>
|
40
|
<sendTime>[time]</sendTime>
|
41
|
<source>[source]</source>
|
42
|
<destination resource="[resource]">[url]</destination>
|
43
|
<type>search</type>
|
44
|
</header>
|
45
|
<search>
|
46
|
<filter>
|
47
|
<equals>
|
48
|
<darwin:Kingdom>plantae</darwin:Kingdom>
|
49
|
</equals>
|
50
|
</filter>
|
51
|
<records limit="[count]" start="[start]">
|
52
|
<structure schemaLocation="[schema]"/>
|
53
|
</records>
|
54
|
<count>true</count>
|
55
|
</search>
|
56
|
</request>
|
57
|
'''
|
58
|
|
59
|
diags_start = '<diagnostics>'
|
60
|
diags_end = '</diagnostics>'
|
61
|
|
62
|
class InputError(Exception): pass
|
63
|
|
64
|
def main():
|
65
|
# Usage
|
66
|
env_names = []
|
67
|
def usage_err():
|
68
|
raise SystemExit('Usage: '+opts.env_usage(env_names, True)+' '
|
69
|
+sys.argv[0]+' 2>>log')
|
70
|
|
71
|
# Get config from env vars
|
72
|
url = opts.get_env_var('url', None, env_names)
|
73
|
resource = opts.get_env_var('resource', None, env_names)
|
74
|
schema = opts.get_env_var('schema', default_schema, env_names)
|
75
|
start = util.cast(int, opts.get_env_var('start', 0, env_names))
|
76
|
count = util.cast(int, opts.get_env_var('n', None, env_names))
|
77
|
debug = opts.env_flag('debug', False, env_names)
|
78
|
if url == None or resource == None: usage_err()
|
79
|
|
80
|
# Logging
|
81
|
def clear_line(): sys.stderr.write('\n')
|
82
|
log_indent = 0
|
83
|
def log(msg, line_ending='\n'): sys.stderr.write(msg+line_ending)
|
84
|
def debug_log(str_, label=None):
|
85
|
if debug:
|
86
|
if label != None: sys.stderr.write(label+':\n')
|
87
|
sys.stderr.write(str_+'\n')
|
88
|
|
89
|
# Request XML
|
90
|
self_dir = os.path.dirname(__file__)
|
91
|
source = os.popen(self_dir+"/local_ip").read().strip()
|
92
|
this_request_xml_template = (request_xml_template
|
93
|
.replace('[source]', source)
|
94
|
.replace('[url]', url)
|
95
|
.replace('[resource]', resource)
|
96
|
.replace('[schema]', schema)
|
97
|
)
|
98
|
|
99
|
# Stats
|
100
|
total = 0
|
101
|
def print_status(line_ending='\n'):
|
102
|
log('Processed '+str(total)+' record(s)', line_ending)
|
103
|
match_ct = None
|
104
|
|
105
|
# Retrieve data
|
106
|
while count == None or total < count:
|
107
|
# Adjust chunk size if last chunk
|
108
|
remaining_ct = count - total
|
109
|
this_chunk_size = min(chunk_size, remaining_ct)
|
110
|
|
111
|
# Request XML
|
112
|
time = dates.strftime('%Y-%m-%d %H:%M:%S %Z', dates.now())
|
113
|
request_xml = (this_request_xml_template
|
114
|
.replace('[count]', str(this_chunk_size))
|
115
|
.replace('[start]', str(start))
|
116
|
.replace('[time]', time)
|
117
|
)
|
118
|
debug_log(request_xml, 'request')
|
119
|
|
120
|
# Send request
|
121
|
this_url = url+'?'+urllib.urlencode({'request': request_xml})
|
122
|
stream = streams.CaptureStream(streams.TimeoutInputStream(
|
123
|
urllib2.urlopen(this_url), timeout), diags_start, diags_end)
|
124
|
|
125
|
# Retrieve response
|
126
|
streams.copy(stream, sys.stdout)
|
127
|
stream.close()
|
128
|
|
129
|
# Parse diagnostics
|
130
|
diags_str = stream.match
|
131
|
debug_log(diags_str, 'diagnostics')
|
132
|
diags = xml_dom.parse_str(diags_str)
|
133
|
def get_diag(name):
|
134
|
return xpath.get_value(diags, 'diagnostic[@code='+name+']')
|
135
|
|
136
|
# Process match count
|
137
|
this_match_ct = util.cast(int, get_diag('MATCH_COUNT'))
|
138
|
if this_match_ct != match_ct: # first or updated match count
|
139
|
match_ct = this_match_ct
|
140
|
log('Found '+str(match_ct)+' record(s)')
|
141
|
|
142
|
# Process record count
|
143
|
this_ct = util.cast(int, get_diag('RECORD_COUNT'))
|
144
|
if this_ct == None: raise InputError('Missing RECORD_COUNT diagnostic')
|
145
|
total += this_ct
|
146
|
start += this_ct # advance start to fetch next set
|
147
|
print_status('\r') # CR at end so next print overwrites msg
|
148
|
if this_ct == 0 or get_diag('END_OF_RECORDS') == 'true': break
|
149
|
|
150
|
print_status()
|
151
|
|
152
|
main()
|