2019-03-24 14:14:00 +01:00
|
|
|
from __future__ import print_function
|
2008-11-18 19:11:13 -07:00
|
|
|
|
|
|
|
import os, time
|
2010-02-10 18:43:18 -08:00
|
|
|
from allmydata.scripts.common import get_alias, DEFAULT_ALIAS, escape_path, \
|
|
|
|
UnknownAliasError
|
2010-06-06 18:02:15 -07:00
|
|
|
from allmydata.scripts.common_http import do_http, format_http_error
|
2008-11-18 19:11:13 -07:00
|
|
|
from allmydata.util import base32
|
2010-07-11 17:30:15 -07:00
|
|
|
from allmydata.util.encodingutil import quote_output, is_printable_ascii
|
2008-11-18 19:11:13 -07:00
|
|
|
import urllib
|
2017-01-19 15:39:53 -07:00
|
|
|
import json
|
2008-11-18 19:11:13 -07:00
|
|
|
|
2019-05-15 08:17:44 +02:00
|
|
|
class SlowOperationRunner(object):
|
2008-11-18 19:11:13 -07:00
|
|
|
|
|
|
|
def run(self, options):
|
|
|
|
stderr = options.stderr
|
|
|
|
self.options = options
|
|
|
|
self.ophandle = ophandle = base32.b2a(os.urandom(16))
|
|
|
|
nodeurl = options['node-url']
|
|
|
|
if not nodeurl.endswith("/"):
|
|
|
|
nodeurl += "/"
|
|
|
|
self.nodeurl = nodeurl
|
|
|
|
where = options.where
|
2010-02-10 18:43:18 -08:00
|
|
|
try:
|
|
|
|
rootcap, path = get_alias(options.aliases, where, DEFAULT_ALIAS)
|
2019-03-28 12:45:28 +01:00
|
|
|
except UnknownAliasError as e:
|
2010-06-06 18:02:15 -07:00
|
|
|
e.display(stderr)
|
2010-02-10 18:43:18 -08:00
|
|
|
return 1
|
2008-11-18 19:11:13 -07:00
|
|
|
if path == '/':
|
|
|
|
path = ''
|
|
|
|
url = nodeurl + "uri/%s" % urllib.quote(rootcap)
|
|
|
|
if path:
|
|
|
|
url += "/" + escape_path(path)
|
|
|
|
# todo: should it end with a slash?
|
|
|
|
url = self.make_url(url, ophandle)
|
|
|
|
resp = do_http("POST", url)
|
|
|
|
if resp.status not in (200, 302):
|
2019-03-24 14:14:00 +01:00
|
|
|
print(format_http_error("ERROR", resp), file=stderr)
|
2008-11-18 19:11:13 -07:00
|
|
|
return 1
|
|
|
|
# now we poll for results. We nominally poll at t=1, 5, 10, 30, 60,
|
|
|
|
# 90, k*120 seconds, but if the poll takes non-zero time, that will
|
|
|
|
# be slightly longer. I'm not worried about trying to make up for
|
|
|
|
# that time.
|
|
|
|
|
|
|
|
return self.wait_for_results()
|
|
|
|
|
|
|
|
def poll_times(self):
|
|
|
|
for i in (1,5,10,30,60,90):
|
|
|
|
yield i
|
|
|
|
i = 120
|
|
|
|
while True:
|
|
|
|
yield i
|
|
|
|
i += 120
|
|
|
|
|
|
|
|
def wait_for_results(self):
|
|
|
|
last = 0
|
|
|
|
for next in self.poll_times():
|
|
|
|
delay = next - last
|
|
|
|
time.sleep(delay)
|
|
|
|
last = next
|
|
|
|
if self.poll():
|
|
|
|
return 0
|
|
|
|
|
|
|
|
def poll(self):
|
|
|
|
url = self.nodeurl + "operations/" + self.ophandle
|
|
|
|
url += "?t=status&output=JSON&release-after-complete=true"
|
|
|
|
stdout = self.options.stdout
|
|
|
|
stderr = self.options.stderr
|
|
|
|
resp = do_http("GET", url)
|
|
|
|
if resp.status != 200:
|
2019-03-24 14:14:00 +01:00
|
|
|
print(format_http_error("ERROR", resp), file=stderr)
|
2008-11-18 19:11:13 -07:00
|
|
|
return True
|
|
|
|
jdata = resp.read()
|
2017-01-19 15:39:53 -07:00
|
|
|
data = json.loads(jdata)
|
2008-11-18 19:11:13 -07:00
|
|
|
if not data["finished"]:
|
|
|
|
return False
|
|
|
|
if self.options.get("raw"):
|
2010-06-06 18:02:15 -07:00
|
|
|
if is_printable_ascii(jdata):
|
2019-03-24 14:14:00 +01:00
|
|
|
print(jdata, file=stdout)
|
2010-06-06 18:02:15 -07:00
|
|
|
else:
|
2019-03-24 14:14:00 +01:00
|
|
|
print("The JSON response contained unprintable characters:\n%s" % quote_output(jdata), file=stderr)
|
2008-11-18 19:11:13 -07:00
|
|
|
return True
|
|
|
|
self.write_results(data)
|
|
|
|
return True
|
|
|
|
|