summaryrefslogtreecommitdiff
path: root/src/leap/bitmask/bonafide/_http.py
diff options
context:
space:
mode:
authorKali Kaneko <kali@leap.se>2017-09-18 15:35:13 +0200
committerKali Kaneko <kali@leap.se>2017-09-20 19:06:28 +0200
commit0d0459ac31f40d90270de0a9da8e1eac0b57784d (patch)
tree2292077ece65c26fd303b5f05203767c4a587242 /src/leap/bitmask/bonafide/_http.py
parent99d59d5942f691f65bdf7a387a62acbb972e5dee (diff)
[feat] check if there are newest configuration files
here we port the if-modified-since conditional mechanism, so that we only write the config if it is newer than whan we have. we also add a line with the status code to the logs, so that it's easier to debug. note that the 'configs.json' file is never returning 304. - Resolves: #8773
Diffstat (limited to 'src/leap/bitmask/bonafide/_http.py')
-rw-r--r--src/leap/bitmask/bonafide/_http.py48
1 files changed, 44 insertions, 4 deletions
diff --git a/src/leap/bitmask/bonafide/_http.py b/src/leap/bitmask/bonafide/_http.py
index 8f05b421..94d0bcb9 100644
--- a/src/leap/bitmask/bonafide/_http.py
+++ b/src/leap/bitmask/bonafide/_http.py
@@ -1,6 +1,6 @@
# -*- coding: utf-8 -*-
# _http.py
-# Copyright (C) 2015 LEAP
+# Copyright (C) 2015-2017 LEAP
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
@@ -18,10 +18,14 @@
"""
twisted.web utils for bonafide.
"""
-import base64
+
import cookielib
+import os
import urllib
+from leap.common.files import get_mtime
+
+from twisted.logger import Logger
from twisted.internet import defer, protocol, reactor
from twisted.internet.ssl import Certificate
from twisted.python.filepath import FilePath
@@ -32,6 +36,9 @@ from twisted.web.iweb import IBodyProducer
from zope.interface import implements
+log = Logger()
+
+
def cookieAgentFactory(verify_path, connectTimeout=30):
customPolicy = BrowserLikePolicyForHTTPS(
Certificate.loadPEM(FilePath(verify_path).getContent()))
@@ -40,19 +47,36 @@ def cookieAgentFactory(verify_path, connectTimeout=30):
return CookieAgent(agent, cookiejar)
-def httpRequest(agent, url, values={}, headers={}, method='POST', token=None):
+class Unchanged(Exception):
+ pass
+
+
+# TODO this should be ported to use treq client.
+
+def httpRequest(agent, url, values={}, headers={}, method='POST', token=None,
+ saveto=None):
data = ''
if values:
data = urllib.urlencode(values)
headers['Content-Type'] = ['application/x-www-form-urlencoded']
+ if saveto is not None:
+ # TODO - I think we need a force parameter, because we might have a
+ # malformed file. Or maybe just remove the file if sanity check does
+ # not pass.
+ mtime = get_mtime(saveto)
+ if mtime is not None:
+ headers['if-modified-since'] = [mtime]
if token:
headers['Authorization'] = ['Token token="%s"' % (bytes(token))]
def handle_response(response):
- # print "RESPONSE CODE", response.code
+ log.debug("RESPONSE %s %s %s" % (method, response.code, url))
if response.code == 204:
d = defer.succeed('')
+ if response.code == 304:
+ log.debug('304 (Not modified): %s' % url)
+ raise Unchanged()
else:
class SimpleReceiver(protocol.Protocol):
def __init__(s, d):
@@ -70,12 +94,28 @@ def httpRequest(agent, url, values={}, headers={}, method='POST', token=None):
response.deliverBody(SimpleReceiver(d))
return d
+ def passthru(failure):
+ failure.trap(Unchanged)
+
d = agent.request(method, url, Headers(headers),
StringProducer(data) if data else None)
d.addCallback(handle_response)
+ if saveto:
+ d.addCallback(lambda body: _write_to_file(body, saveto))
+ d.addErrback(passthru)
return d
+def _write_to_file(content, path):
+ folder = os.path.split(path)[0]
+ if not os.path.isdir(folder):
+ os.makedirs(folder)
+ with open(path, 'w') as f:
+ f.write(content)
+ # touch it to update its utime
+ os.utime(path, None)
+
+
class StringProducer(object):
implements(IBodyProducer)