diff options
| -rw-r--r-- | Config/__init__.py (renamed from pb/Config/__init__.py) | 0 | ||||
| -rw-r--r-- | Db/__init__.py (renamed from pb/lib/Db/__init__.py) | 12 | ||||
| -rw-r--r-- | Db/sqlalchemy_example_code.py (renamed from pb/lib/Db/sqlalchemy_example_code.py) | 0 | ||||
| -rwxr-xr-x | Pb/Breaker/__init__.py (renamed from pb/Breaker/__init__.py) | 26 | ||||
| -rwxr-xr-x | Pb/Generate/__init__.py (renamed from pb/Generate/__init__.py) | 25 | ||||
| -rwxr-xr-x | Pb/Gradient/__init__.py (renamed from pb/Gradient/__init__.py) | 61 | ||||
| -rwxr-xr-x | Pb/Imgrid/__init__.py (renamed from pb/Imgrid/__init__.py) | 28 | ||||
| -rwxr-xr-x | Pb/Imlandscape/__init__.py (renamed from pb/Imlandscape/__init__.py) | 31 | ||||
| -rwxr-xr-x | Pb/Imlandscape/landscape (renamed from pb/Imlandscape/landscape) | 0 | ||||
| -rwxr-xr-x | Pb/Pattern/__init__.py (renamed from pb/Pattern/__init__.py) | 24 | ||||
| -rw-r--r-- | Pb/__init__.py | 143 | ||||
| -rw-r--r-- | pb/__init__.py | 0 | ||||
| -rw-r--r-- | pb/lib/Utils/__init__.py | 123 | ||||
| -rw-r--r-- | pb/lib/__init__.py | 0 | ||||
| -rw-r--r-- | pb/test/base64img | 1 | ||||
| -rw-r--r-- | pb/test/photo_data | 1 | ||||
| -rwxr-xr-x | pbserver.py | 22 |
17 files changed, 254 insertions, 243 deletions
diff --git a/pb/Config/__init__.py b/Config/__init__.py index b849b9e..b849b9e 100644 --- a/pb/Config/__init__.py +++ b/Config/__init__.py diff --git a/pb/lib/Db/__init__.py b/Db/__init__.py index ac7ca17..c42a7c5 100644 --- a/pb/lib/Db/__init__.py +++ b/Db/__init__.py @@ -5,7 +5,6 @@ USER = "asdfus" PASSWORD = "gTYgT&M6q" DATABASE = "asdfus" - from sqlalchemy import Column, Integer, LargeBinary, String, create_engine, sql from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.orm import sessionmaker @@ -51,6 +50,17 @@ class Db(object): cmd=_NULL, dataobj=_NULL, tag=_NULL): + +# sys.stderr.write("date={}".format(str(date)) + "\n") +# sys.stderr.write("remote_addr={}".format(str(remote_addr)) + "\n") +# sys.stderr.write("name={}".format(str(username)) + "\n") +# sys.stderr.write("url={}".format(str(url)) + "\n") +# sys.stderr.write("dir={}".format(str(directory)) + "\n") +# sys.stderr.write("oldfile={}".format(str(oldfile)) + "\n") +# sys.stderr.write("newfile={}".format(str(newfile)) + "\n") +# sys.stderr.write("cmd={}".format(str(cmd)) + "\n") +# sys.stderr.write("dataobj={}".format(str(dataobj)) + "\n") +# sys.stderr.write("tag={}".format(str(tag)) + "\n") try: entry = ImCmd( date=date, diff --git a/pb/lib/Db/sqlalchemy_example_code.py b/Db/sqlalchemy_example_code.py index fc38b38..fc38b38 100644 --- a/pb/lib/Db/sqlalchemy_example_code.py +++ b/Db/sqlalchemy_example_code.py diff --git a/pb/Breaker/__init__.py b/Pb/Breaker/__init__.py index 9f5d31b..262ac4f 100755 --- a/pb/Breaker/__init__.py +++ b/Pb/Breaker/__init__.py @@ -3,9 +3,9 @@ import os import sys import random import re -import pb.lib.Utils as utils import urllib -from pb.Config import * +from Config import * +from Pb import Pb DEFAULT_FINALFORMAT = "png"; SUBTLE_BREAK_MARK = 'pron' @@ -27,7 +27,7 @@ HEADER_OFFSET = 2000 # 'STRIPES':'exr', # 'PHOTOCOPY':'art', -class Breaker(): +class Breaker(Pb): def __init__(self, **kwargs): self.params = {} self.tag = "imBreak" @@ -41,7 +41,7 @@ class Breaker(): "username", "expanded" ] - self.now = utils.now() + self._now = self.now() self.files_created = [] for k in self._required_keys: if k in kwargs: @@ -50,25 +50,25 @@ class Breaker(): elif k == 'url': self.params[k] = kwargs[k] else: - self.params[k] = utils.bool_correct(utils.sanitize(kwargs[k])) + self.params[k] = self.bool_correct(self.sanitize(kwargs[k])) else: self.params[k] = False; - self.params = utils.dotdict(self.params) + self.params = self.make_dotdict(self.params) self.basename, self._first_format = self._get_filename(); self._downloaded_file = os.path.join(WORKING_DIR, "IMBREAKTMP{}.{}".format(self.basename, self._first_format)) # same here try: - utils.download(self.params.url, self._downloaded_file) + self.download(self.params.url, self._downloaded_file) self.files_created.append(self._downloaded_file) except Exception as e: sys.stderr.write(str(e)) raise; - self._gif_frames = utils.gif_frames(self._downloaded_file) + self._gif_frames = self.gif_frames(self._downloaded_file) self._gif_frames = self._gif_frames if len(self._gif_frames) > 1 else False - self.width, self.height = utils.dimensions(self._downloaded_file) # same here + self.width, self.height = self.dimensions(self._downloaded_file) # same here if not self.params.finalformat: self.params.finalformat = DEFAULT_FINALFORMAT @@ -85,7 +85,7 @@ class Breaker(): def _call_cmd(self, cmd, error=""): try: - utils.call_cmd(cmd) + self.call_cmd(cmd) self.commands.append(" ".join(cmd)); except Exception: raise Exception("Unable to call cmd {}".format(str(cmd))) @@ -121,8 +121,8 @@ class Breaker(): name_part = url.split("/")[-1] try: parts = name_part.split(".") - name_part = utils.sanitize(parts[-2]) - file_format = utils.sanitize(parts[-1]) + name_part = self.sanitize(parts[-2]) + file_format = self.sanitize(parts[-1]) if not name_part or not file_format: sys.stderr.write( "Incompatible input file type") raise; @@ -134,7 +134,7 @@ class Breaker(): raise; if (len(name_part) > 20): name_part = name_part[:-20] - return "{}{}_{}_{}".format(self.tag, name_part, self.now, self.params.username or ""), file_format + return "{}{}_{}_{}".format(self.tag, name_part, self._now, self.params.username or ""), file_format #{{{#########rotatefunctions####################################### def _rotate(self): diff --git a/pb/Generate/__init__.py b/Pb/Generate/__init__.py index fbb74e3..385591e 100755 --- a/pb/Generate/__init__.py +++ b/Pb/Generate/__init__.py @@ -2,9 +2,8 @@ import sys import os -from pb.Config import * -import pb.lib.Utils as utils - +from Config import * +from Pb import Pb #FIXME these guys can do stuff wider than 1000 LIKE_A_BOSS = "ryz pepper seamonkey JAMES".split(" ") DEFAULT_FINALFORMAT = "gif" @@ -23,10 +22,10 @@ DISPOSE_DEFAULT = "None" def debuglog(s): sys.stderr.write(str(s) + "\n"); -class Generate(): +class Generate(Pb): def __init__(self, **kwargs): self.params = {} - self.now = utils.now() + self._now = self.now() self.files_created = [] self.commands = []; self._required_keys = [ @@ -79,22 +78,22 @@ class Generate(): 'path' : os.path.join(WORKING_DIR, self._make_tempname(k)) , } try: - utils.download(self.params[k]['url'], self.params[k]['path']) + self.download(self.params[k]['url'], self.params[k]['path']) self.files_created.append(self.params[k]['path']) - self.params[k]['mimetype'] = utils.get_mimetype(self.params[k]['path']) + self.params[k]['mimetype'] = self.get_mimetype(self.params[k]['path']) except Exception as e: sys.stderr.write(str(e)) raise Exception ("BAD PARAMS"); elif k in [ 'black', 'white', 'subtract' ]: try: - self.params[k] = utils.is_color(kwargs[k]) + self.params[k] = self.is_color(kwargs[k]) except Exception: raise Exception("Unable to process color for:\n{}".format(k)) elif k in [ "coalesce", "dispose", "nearest", "merge_early", "flip", "flop", "tile", "transparent", ]: - self.params[k] = utils.bool_correct(utils.sanitize(kwargs[k])) + self.params[k] = self.bool_correct(self.sanitize(kwargs[k])) elif k == 'gravity' and self._test_enum(kwargs[k], GRAVITY_PARAMS): self.params[k] = kwargs[k] elif k == 'format' and self._test_enum(kwargs[k], FORMAT_PARAMS): @@ -114,7 +113,7 @@ class Generate(): else: - self.params[k] = utils.sanitize(kwargs[k]) + self.params[k] = self.sanitize(kwargs[k]) if self.params.get('background'): self.tag = self.params.get('compose') @@ -126,7 +125,7 @@ class Generate(): self.filepath = os.path.join(WORKING_DIR, self.filename) def _make_tempname(self, s): - return "PBTMP{}{}".format(self.now, s); + return "PBTMP{}{}".format(self._now, s); def _test_enum(self, e, arr): if e in arr: return True @@ -135,13 +134,13 @@ class Generate(): def _get_filename(self): return "{}_{}_{}".format( self.tag, - self.now, + self._now, self.params.get('username',"") ); def _call_cmd(self, cmd): try: - utils.call_cmd(cmd) + self.call_cmd(cmd) self.commands.append(" ".join(cmd)); except Exception: raise Exception("Unable to call cmd {}".format(str(cmd))) diff --git a/pb/Gradient/__init__.py b/Pb/Gradient/__init__.py index aeb4d21..5120c7d 100755 --- a/pb/Gradient/__init__.py +++ b/Pb/Gradient/__init__.py @@ -6,9 +6,8 @@ import simplejson as json import sys
import os
import sha
-import pb.lib.Utils as utils
-from pb.Config import *
-
+from Config import *
+from Pb import Pb
PARAM_LIST = [
"width", "height",
@@ -45,31 +44,31 @@ HALFTONEVALUES = { }
-class Gradient:
+class Gradient(Pb):
def __init__(self, **kwargs):
self.tag = "imGradient"
self.directory = WORKING_DIR
self.commands = []
self.filename = ""
self.filepath = ""
- self.now = utils.now()
+ self._now = self.now()
params = {}
for key in PARAM_LIST:
if key in kwargs:
if key in ['color1', 'color2']:
- params[key] = utils.is_color(kwargs[key])
+ params[key] = self.is_color(kwargs[key])
else:
- params[key] = utils.sanitize(kwargs[key])
+ params[key] = self.sanitize(kwargs[key])
if key in ['rotate','tilt','blurriness','stripenumber','stripeintensity']:
- params[key] = params[key] if utils.is_number(params[key]) else ""
+ params[key] = params[key] if self.is_number(params[key]) else ""
elif key in ['brightness', 'contrast', 'hue']:
- if not utils.is_number(params[key]) or params[key] == "100": params[key] = ""
+ if not self.is_number(params[key]) or params[key] == "100": params[key] = ""
else:
params[key] = ""
- params['width'] = params['width'] if utils.is_number(params['width']) else DEFAULT_WIDTH
- params['height'] = params['height'] if utils.is_number(params['height']) else DEFAULT_HEIGHT
+ params['width'] = params['width'] if self.is_number(params['width']) else DEFAULT_WIDTH
+ params['height'] = params['height'] if self.is_number(params['height']) else DEFAULT_HEIGHT
params["color1"] = params["color1"] or DEFAULT_COLORS["color1"];
params["color2"] = params["color2"] or DEFAULT_COLORS["color2"];
self.params = params
@@ -84,31 +83,31 @@ class Gradient: self.tag,
self.params['color1'].replace('#','').replace('(','-').replace(')','-'),
self.params['color2'].replace('#','').replace('(','-').replace(')','-'),
- self.now,
+ self._now,
self.params['username'],
self.params['filetype'] or DEFAULT_FORMAT,
)
def _call_cmd(self, cmd):
try:
- utils.call_cmd(cmd)
+ self.call_cmd(cmd)
self.commands.append(" ".join(cmd));
except Exception:
raise Exception("Unable to call cmd {}".format(str(cmd)))
def _build_cmd(self):
- cmd = [BIN_CONVERT]
- cmd.extend([
+ self.cmd = [BIN_CONVERT]
+ self.cmd.extend([
'-size',
"{}x{}".format(self.params["width"],self.params["height"])
])
- if self.params['rotate']: cmd.extend(["-rotate", self.params["rotate"]])
- if self.params['tilt']: cmd.extend(["-distort","SRT",self.params['tilt']])
- if self.params['flip'] == "true": cmd.append("-flip")
- if self.params['flop'] == "true": cmd.append("-flop")
- if self.params['contrast']: cmd.extend(["-contrast-stretch", self.params['contrast']])
+ if self.params['rotate']: self.cmd.extend(["-rotate", self.params["rotate"]])
+ if self.params['tilt']: self.cmd.extend(["-distort","SRT",self.params['tilt']])
+ if self.params['flip'] == "true": self.cmd.append("-flip")
+ if self.params['flop'] == "true": self.cmd.append("-flop")
+ if self.params['contrast']: self.cmd.extend(["-contrast-stretch", self.params['contrast']])
gradients = {
"canvas" : ["canvas:{}".format(self.params['color1'])],
"radial" : [
@@ -135,33 +134,33 @@ class Gradient: "noise" : ["xc:","+noise","Random","-virtual-pixel","tile"]
}
if self.params["gradienttype"] in gradients:
- cmd.extend(gradients[self.params['gradienttype']])
+ self.cmd.extend(gradients[self.params['gradienttype']])
else:
- cmd.append("gradient:{}-{}".format(self.params['color1'], self.params['color2']))
+ self.cmd.append("gradient:{}-{}".format(self.params['color1'], self.params['color2']))
if self.params['blurriness']:
- cmd.extend(["-blur","0x{}".format(self.params["blurriness"]),"-auto-level"])
+ self.cmd.extend(["-blur","0x{}".format(self.params["blurriness"]),"-auto-level"])
if self.params['stripes'] == "true" and len(self.params['stripenumber']):
- cmd.extend(["-function","Sinusoid"])
+ self.cmd.extend(["-function","Sinusoid"])
if self.params['stripeintensity']:
- cmd.append("{},{}".format(self.params['stripenumber'],self.params["stripeintensity"]))
+ self.cmd.append("{},{}".format(self.params['stripenumber'],self.params["stripeintensity"]))
else:
- cmd.append(self.params['stripenumber'])
+ self.cmd.append(self.params['stripenumber'])
if self.params["halftone"] in HALFTONEVALUES:
- cmd.extend([
+ self.cmd.extend([
"-ordered-dither",
HALFTONEVALUES[self.params["halftone"]]
])
- cmd += [
+ self.cmd += [
'-modulate',
"{},{},{}".format(
self.params['brightness'] or "100",
self.params['saturation'] or "100",
self.params['hue'] or "100")
]
- cmd.append(os.path.join(self.directory,self.filename));
- self._call_cmd(cmd)
+ self.cmd.append(os.path.join(self.directory,self.filename));
+ self._call_cmd(self.cmd)
def _get_bevelvalue(self):
w, h = map(int, (self.params['width'], self.params['height']))
@@ -189,6 +188,8 @@ class Gradient: self.filename = self.newfilename()
self.filepath = os.path.join(self.directory, self.filename)
self._build_cmd()
+# sys.stderr.write(str(self.cmd))
+
if self.params['bevel'] in self._bevelvalues:
self._make_bevel()
diff --git a/pb/Imgrid/__init__.py b/Pb/Imgrid/__init__.py index 70da965..52cb748 100755 --- a/pb/Imgrid/__init__.py +++ b/Pb/Imgrid/__init__.py @@ -4,15 +4,15 @@ import re import os import simplejson as json import random -import pb.lib.Utils as utils -from pb.Config import * +from Config import * import tempfile +from Pb import Pb DEFAULT_HEIGHT = 400 DEFAULT_WIDTH = 600 DEFAULT_LINE_COLOR = "silver" -class Imgrid(): +class Imgrid(Pb): def __init__(self, **kwargs): self.tag = "imGrid" self.files_created = [] @@ -44,7 +44,7 @@ class Imgrid(): "username" #}}} ] - self.now = utils.now() + self._now = self.now() #Work out params ... #note, tmpfile lib is pretty much useless here, given imagemagick's behavior with gifs (it splits them) etc... @@ -52,17 +52,17 @@ class Imgrid(): self.params = {} for k in self._required_keys: if k in kwargs: - if k in [ 'bgimage', 'planebgimage', 'imageinstead' ] and utils.bool_correct(kwargs[k]): + if k in [ 'bgimage', 'planebgimage', 'imageinstead' ] and self.bool_correct(kwargs[k]): self.params[k] = { 'url' : kwargs[k], 'filename' : self._make_tempname(k), 'path' : os.path.join(WORKING_DIR, self._make_tempname(k)) , } try: - utils.download(self.params[k]['url'], self.params[k]['path']) + self.download(self.params[k]['url'], self.params[k]['path']) self.files_created.append(self.params[k]['path']) - self.params[k]['mimetype'] = utils.get_mimetype(self.params[k]['path']) - frames = utils.gif_frames(self.params[k]['path']) + self.params[k]['mimetype'] = self.get_mimetype(self.params[k]['path']) + frames = self.gif_frames(self.params[k]['path']) if len(frames) > 1: self.params[k]['path'] = random.choice(frames) @@ -71,7 +71,7 @@ class Imgrid(): raise Exception ("BAD PARAMS"); elif k in [ 'skycolor', 'bgcolor', 'planebgcolor','linecolor' ]: try: - self.params[k] = utils.is_color(kwargs[k]) + self.params[k] = self.is_color(kwargs[k]) except Exception: sys.stderr.write(kwargs[k] + "\n") raise Exception("Unable to process color for:\n{}".format(k)) @@ -80,11 +80,11 @@ class Imgrid(): elif k == 'zoom': self.params[k] = int(float(kwargs[k])) else: - self.params[k] = utils.bool_correct(utils.sanitize(kwargs[k])) + self.params[k] = self.bool_correct(self.sanitize(kwargs[k])) else: self.params[k] = None; - self.params = utils.dotdict(self.params) + self.params = self.make_dotdict(self.params) self.basename = self._get_filename(); @@ -97,19 +97,19 @@ class Imgrid(): def _get_filename(self): return "{}_{}_{}".format( self.tag, - self.now, + self._now, self.params.username or "" ); def _call_cmd(self, cmd): try: - utils.call_cmd(cmd) + self.call_cmd(cmd) self.commands.append(" ".join(cmd)); except Exception: raise Exception("Unable to call cmd {}".format(str(cmd))) def _make_tempname(self, s): - return "IMGRIDTMP{}{}".format(self.now, s); + return "IMGRIDTMP{}{}".format(self._now, s); #makes a canvas file...step 1 (if not bgimage) diff --git a/pb/Imlandscape/__init__.py b/Pb/Imlandscape/__init__.py index 156dac1..59212a6 100755 --- a/pb/Imlandscape/__init__.py +++ b/Pb/Imlandscape/__init__.py @@ -1,24 +1,22 @@ #!/usr/bin/python2.7 -import os import sys import random import re -import pb.lib.Utils as utils import urllib import urlparse -from pb.Config import * - +from Config import * +import Pb import base64 import time import string -import urllib from subprocess import Popen, PIPE import sha import simplejson as json +from Pb import Pb import mimetypes -class Imlandscape(object): +class Imlandscape(Pb): def __init__(self, **kwargs): sys.stderr.write("inside init"); sys.stderr.write(str(kwargs)) @@ -32,13 +30,13 @@ class Imlandscape(object): "texture", "name", ] - self.now = utils.now() + self._now = self.now() self.files_created = [] for k in self._required_keys: if k in kwargs: self.params[k] = kwargs[k] else: - self.params[k] = utils.bool_correct(utils.sanitize(kwargs[k])) + self.params[k] = self.bool_correct(self.sanitize(kwargs[k])) def _filename_from_url (self, url, name=""): if "?" in url: @@ -52,7 +50,7 @@ class Imlandscape(object): filename = "" if name != "": name = name+"_" - return "{}_{}{}_{}.{}".format("imlandscape", name, filename,self.now, "png") + return "{}_{}{}_{}.{}".format("imlandscape", name, filename,self._now, "png") def _saveImgData(self, imgdata, filename): try: @@ -87,18 +85,3 @@ class Imlandscape(object): self.filepath = self._filename_from_url(self.params.get('texture',""), self.params.get('name',"")); self._saveImgData(self.params.get('imgdata'), self.filepath); -if __name__ == "__main__": - f = open("/tmp/base64img") - data = f.read() - f.close() - TEST_PARAMS = { - "heightmap" : "https%3A%2F%2Fwww.google.com%2Fimages%2Fsrpr%2Flogo11w.png" , - "imgdata" : data, - "texture" : "https%3A%2F%2Fwww.google.com%2Fimages%2Fsrpr%2Flogo11w.png", - "name" : "pepper", - } - b = Imlandscape(**TEST_PARAMS) - b.create(); - print b.filepath - - diff --git a/pb/Imlandscape/landscape b/Pb/Imlandscape/landscape index 10e8ede..10e8ede 100755 --- a/pb/Imlandscape/landscape +++ b/Pb/Imlandscape/landscape diff --git a/pb/Pattern/__init__.py b/Pb/Pattern/__init__.py index b1fb9af..905ddf9 100755 --- a/pb/Pattern/__init__.py +++ b/Pb/Pattern/__init__.py @@ -4,8 +4,8 @@ import sys import random import re import urllib -from pb.Config import * -import pb.lib.Utils as utils +from Config import * +from Pb import Pb import simplejson as json from PIL import Image @@ -13,13 +13,13 @@ import uuid FUSE_MODE="Pin_Light" -class Pattern: +class Pattern(Pb): def __init__(self, **kwargs): self.params = {} self.tag = "imPattern"; self._pid = str(os.getpid()) self.commands = []; - self.now = utils.now() + self._now = self.now() self.height = "" self.width = "" self._required_keys = [ @@ -38,14 +38,14 @@ class Pattern: elif k == 'pattern_data': self.params[k] = kwargs[k] #FIXME add conversion data else: - self.params[k] = utils.sanitize(kwargs[k]) + self.params[k] = self.sanitize(kwargs[k]) else: self.params[k] = False; if not self.params['image_url']: sys.stderr.write('no image url'); raise ValueError - self.params = utils.dotdict(self.params) + self.params = self.make_dotdict(self.params) self.basename, self._format = self._get_filename(); #FIXME omit file extension for downloaded files @@ -55,7 +55,7 @@ class Pattern: self._download(self.params.image_url, self._downloaded_file) - self.width, self.height = utils.dimensions(self._downloaded_file) # same here + self.width, self.height = self.dimensions(self._downloaded_file) # same here self.filename = "{}.{}".format(self.basename, self._format) self.filepath = os.path.join(WORKING_DIR, self.filename) @@ -70,7 +70,7 @@ class Pattern: def _download(self, url, dest): try: - utils.download(url, dest) + self.download(url, dest) self.files_created.append(dest) except Exception as e: sys.stderr.write(str(e)) @@ -78,7 +78,7 @@ class Pattern: def _call_cmd(self, cmd): try: - utils.call_cmd(cmd) + self.call_cmd(cmd) self.commands.append(" ".join(cmd)); except Exception: raise Exception("Unable to call cmd {}".format(str(cmd))) @@ -113,8 +113,8 @@ class Pattern: name_part = url.split("/")[-1] try: parts = name_part.split(".") - name_part = utils.sanitize(parts[-2]) - file_format = utils.sanitize(parts[-1]) + name_part = self.sanitize(parts[-2]) + file_format = self.sanitize(parts[-1]) if not name_part or not file_format: sys.stderr.write( "Incompatible input file type") raise; @@ -126,7 +126,7 @@ class Pattern: raise; if (len(name_part) > 20): name_part = name_part[:-20] - return "{}{}_{}_{}".format(self.tag, name_part, self.now, self.params.username or ""), file_format + return "{}{}_{}_{}".format(self.tag, name_part, self._now, self.params.username or ""), file_format def _cleanup(self): cmd = ["rm"]+self.files_created diff --git a/Pb/__init__.py b/Pb/__init__.py new file mode 100644 index 0000000..f584f00 --- /dev/null +++ b/Pb/__init__.py @@ -0,0 +1,143 @@ +import re +from Config import * +import time +import urllib +import urllib2 +import sys +import os +from subprocess import Popen,PIPE,call +Request = urllib2.Request +urlencode = urllib.urlencode +urlopen = urllib2.urlopen + + +class Pb(object): + def __init__(self): + pass; + + @staticmethod + def call_cmd(cmd, error=""): + try: + call(cmd) + except Exception as e: + raise (str(e)) + + @staticmethod + def is_color(s): + if s == "": + return "transparent" + if re.match('(rgba?\([0-9]+,[0-9]+,[0-9]+\))|([a-zA-Z]+)|(\#[A-Ha-h0-9]+)', s): + return s.replace(' ', ''); + else: + sys.stderr.write("Not a color: {}\n".format(s)) + raise ValueError + + @staticmethod + def dimensions (filepath): + #works in lieu of a mimetype check (it reads the header as well) + ident = (Popen([BIN_IDENTIFY, filepath], stdout=PIPE).communicate()[0]).split(" ") + return ident[2].split("x") + + @staticmethod + def is_number(s): + try: + return int(s) + except (ValueError, TypeError): + return False + + @staticmethod + def bool_correct(s): + if re.match(r'^false$', s, re.IGNORECASE): + return False + elif re.match(r'^true$', s, re.IGNORECASE): + return True + else: + return s + + @staticmethod + def get_mimetype(f): + try: + mimetype = Popen( + [BIN_IDENTIFY, f], stdout=PIPE + ).communicate()[0].split(" ")[1].lower() + return mimetype + except Exception as e: + sys.stderr.write("couldn't determine mimetype") + sys.stderr.write(str(e)) + raise; + + @staticmethod + def sanitize (s): + return re.sub(r'\W+', '', s) + + @staticmethod + def now(): + return int(time.time()) + + @staticmethod + def browser_request (url, data=None): + headers = { + 'User-Agent': 'Mozilla/4.0 (compatible; MSIE 5.5; Windows NT)', + 'Accept': '*/*', + } + try: + req = Request(url, data, headers) + response = urlopen(req) + except IOError, e: + if hasattr(e, 'code'): + sys.stderr.write( '%s - ERROR %s' % (url, e.code) ) + raise; + return None + else: + return response + + def download(self, url, destination, max_size=MAX_SIZE): + response = self.browser_request(url, None) + rawimg = response.read() + if len(rawimg) == 0: + sys.stderr.write("got zero-length file") + raise; + if len(rawimg) > max_size: + sys.stderr.write("file too big: max size {} KB / {} is {} KB".format( + str(MAX_SIZE/1024), + destination, + str(len(rawimg)/1024) + )) + raise; + f = open(destination, "w") + f.write(rawimg) + f.close() + + @staticmethod + def file_size (filepath): + try: + return os.stat(filepath)[6] + except Exception as e: + sys.stderr.write("Couldn't determine filesize\n") + sys.stderr.write(str(e)+"\n") + raise; + + def gif_frames(self, filepath): + try: + info = Popen([BIN_IDENTIFY,filepath], stdout=PIPE).communicate()[0] + frames = filter((lambda x: x), map( + (lambda x: x.split(" ")[0]), + (info).split('\n') + )) + return frames + except Exception as e: + sys.stderr.write("{} couldn't get gif frames".format(self.__class__.__name__)) + sys.stderr.write(str(e)) + raise; + + @staticmethod + def make_dotdict(d): + return dotdict(d) + +class dotdict(dict): + """dot.notation access to dictionary attributes""" + def __getattr__(self, attr): + return self.get(attr) + __setattr__= dict.__setitem__ + __delattr__= dict.__delitem__ + diff --git a/pb/__init__.py b/pb/__init__.py deleted file mode 100644 index e69de29..0000000 --- a/pb/__init__.py +++ /dev/null diff --git a/pb/lib/Utils/__init__.py b/pb/lib/Utils/__init__.py deleted file mode 100644 index 1d5a708..0000000 --- a/pb/lib/Utils/__init__.py +++ /dev/null @@ -1,123 +0,0 @@ -import re -from pb.Config import * -import time -import urllib -import urllib2 -import sys -import os -from subprocess import Popen,PIPE,call -Request = urllib2.Request -urlencode = urllib.urlencode -urlopen = urllib2.urlopen - -def call_cmd(cmd, error=""): - try: - call(cmd) - except Exception as e: - raise (str(e)) - -def is_color(s): - if s == "": - return "transparent" - if re.match('(rgba?\([0-9]+,[0-9]+,[0-9]+\))|([a-zA-Z]+)|(\#[A-Ha-h0-9]+)', s): - return s.replace(' ', ''); - else: - sys.stderr.write("Not a color: {}\n".format(s)) - raise ValueError - -def dimensions (filepath): - #works in lieu of a mimetype check (it reads the header as well) - ident = (Popen([BIN_IDENTIFY, filepath], stdout=PIPE).communicate()[0]).split(" ") - return ident[2].split("x") - -def is_number(s): - try: - return int(s) - except (ValueError, TypeError): - return False - -def bool_correct(s): - if re.match(r'^false$', s, re.IGNORECASE): - return False - elif re.match(r'^true$', s, re.IGNORECASE): - return True - else: - return s - -class dotdict(dict): - """dot.notation access to dictionary attributes""" - def __getattr__(self, attr): - return self.get(attr) - __setattr__= dict.__setitem__ - __delattr__= dict.__delitem__ - -def get_mimetype(f): - try: - mimetype = Popen( - [BIN_IDENTIFY, f], stdout=PIPE - ).communicate()[0].split(" ")[1].lower() - return mimetype - except Exception as e: - sys.stderr.write("couldn't determine mimetype") - sys.stderr.write(str(e)) - raise; - -def sanitize (str): - return re.sub(r'\W+', '', str) - -def now(): - return int(time.time()) - -def browser_request (url, data=None): - headers = { - 'User-Agent': 'Mozilla/4.0 (compatible; MSIE 5.5; Windows NT)', - 'Accept': '*/*', - } - try: - req = Request(url, data, headers) - response = urlopen(req) - except IOError, e: - if hasattr(e, 'code'): - sys.stderr.write( '%s - ERROR %s' % (url, e.code) ) - raise; - return None - else: - return response - -def download(url, destination, max_size=MAX_SIZE): - response = browser_request(url, None) - rawimg = response.read() - if len(rawimg) == 0: - sys.stderr.write("got zero-length file") - raise; - if len(rawimg) > max_size: - sys.stderr.write("file too big: max size {} KB / {} is {} KB".format( - str(MAX_SIZE/1024), - destination, - str(len(rawimg)/1024) - )) - raise; - f = open(destination, "w") - f.write(rawimg) - f.close() - -def file_size (filepath): - try: - return os.stat(filepath)[6] - except Exception as e: - sys.stderr.write("Couldn't determine filesize\n") - sys.stderr.write(str(e)+"\n") - raise; - -def gif_frames(filepath): - try: - info = Popen([BIN_IDENTIFY,filepath], stdout=PIPE).communicate()[0] - frames = filter((lambda x: x), map( - (lambda x: x.split(" ")[0]), - (info).split('\n') - )) - return frames - except Exception as e: - sys.stderr.write("IMGRID couldn't get gif frames") - sys.stderr.write(str(e)) - raise; diff --git a/pb/lib/__init__.py b/pb/lib/__init__.py deleted file mode 100644 index e69de29..0000000 --- a/pb/lib/__init__.py +++ /dev/null diff --git a/pb/test/base64img b/pb/test/base64img deleted file mode 100644 index 81bb894..0000000 --- a/pb/test/base64img +++ /dev/null @@ -1 +0,0 @@ -
\ No newline at end of file diff --git a/pb/test/photo_data b/pb/test/photo_data deleted file mode 100644 index 461554b..0000000 --- a/pb/test/photo_data +++ /dev/null @@ -1 +0,0 @@ -data%3Aimage%2Fpng%3Bbase64%2CiVBORw0KGgoAAAANSUhEUgAAAb0AAAHVCAYAAABliSVzAAAgAElEQVR4Xuy9CYwkWXrf915EHpV131VdVX1fc%2BzMHrMUuSINLSAJkg3BhmGsYQg2TRCEAEuGBUGiIcgGaFKyDosmZS5lSTQpyLzh4bFLcjni7iw53Htnt2dmZ7Znunt6uuuuzMrK%2B86MiOf%2FF5FZldXT3VXV3VWV1fHPRnRkZuUR7%2Fdexj%2B%2B733f97TijQRIgARIgARCQkCHpJ1sJgmQAAmQAAkoih4HAQmQAAmQQGgIUPRC09VsKAmQAAmQAEWPY4AESIAESCA0BCh6oelqNpQESIAESICixzFAAiRAAiQQGgIUvdB0NRtKAiRAAiRA0eMYIAESIAESCA0Bil5oupoNJQESIAESoOhxDJAACZAACYSGAEUvNF3NhpIACZAACVD0OAZIgARIgARCQ4CiF5quZkNJgARIgAQoehwDJEACJEACoSFA0QtNV7OhJEACJEACFD2OARIgARIggdAQoOiFpqvZUBIgARIgAYoexwAJkAAJkEBoCFD0QtPVbCgJkAAJkABFj2OABEiABEggNAQoeqHpajaUBEiABEiAoscxQAIkQAIkEBoCFL3QdDUbSgIkQAIkQNHjGCABEiABEggNAYpeaLqaDSUBEiABEqDocQyQAAmQAAmEhgBFLzRdzYaSAAmQAAlQ9DgGSIAESIAEQkOAohearmZDSYAESIAEKHocAyRAAiRAAqEhQNELTVezoSRAAiRAAhQ9jgESIAESIIHQEKDohaar2VASIAESIAGKHscACZAACZBAaAhQ9ELT1WwoCZAACZAARY9jgARIgARIIDQEKHqh6Wo2lARIgARIgKLHMUACJEACJBAaAhS90HQ1G0oCJEACJEDR4xggARIgARIIDQGKXmi6mg0lARIgARKg6HEMkAAJkAAJhIYARS80Xc2GkgAJkAAJUPQ4BkiABEiABEJDgKIXmq5mQ0mABEiABCh6HAMkQAIkQAKhIUDRC01Xs6EkQAIkQAIUPY4BEiABEiCB0BCg6IWmq9lQEiABEiABih7HAAmQAAmQQGgIUPRC09VsKAmQAAmQAEWPY4AESIAESCA0BCh6oelqNpQESIAESICixzFAAiRAAiQQGgIUvdB0NRtKAiRAAiRA0eMYIAESIAESCA0Bil5oupoNJQESIAESoOhxDJAACZAACYSGAEUvNF3NhpIACZAACVD0OAZIgARIgARCQ4CiF5quZkNJgARIgAQoehwDJEACJEACoSFA0QtNV7OhJEACJEACFD2OARIgARIggdAQoOiFpqvZUBIgARIgAYoexwAJkAAJkEBoCFD0QtPVbCgJkAAJkABFj2OABEiABEggNAQoeqHpajaUBEiABEiAoscxQAIkQAIkEBoCFL3QdDUbSgIkQAIkQNHjGCABEiABEggNAYpeaLqaDSUBEiABEqDocQyQAAmQAAmEhgBFLzRdzYaSAAmQAAlQ9DgGSIAESIAEQkOAohearmZDSYAESIAEKHocAyRAAiRAAqEhQNELTVezoSRAAiRAAhQ9jgESIAESIIHQEKDohaar2VASIAESIAGKHscACZAACZBAaAhQ9ELT1WwoCZAACZAARY9jgARIgARIIDQEKHqh6Wo2lARIgARIgKLHMUACJEACJBAaAhS90HQ1G0oCJEACJEDR4xggARIgARIIDQGKXmi6mg0lARIgARKg6HEMkAAJkAAJhIYARS80Xc2GkgAJkAAJUPQ4BkiABEiABEJDgKIXmq5mQ0mABEiABCh6HAMkQAIkQAKhIUDRC01Xs6EkQAIkQAIUPY4BEiABEiCB0BCg6IWmq9lQEiABEiABih7HAAmQAAmQQGgIUPRC09VsKAmQAAmQAEWPY4AESIAESCA0BCh6oelqNpQESIAESICixzFAAiRAAiQQGgIUvdB0NRtKAiRAAiRA0eMYIAESIAESCA0Bil5oupoNJQESIAESoOhxDJAACZAACYSGAEUvNF3NhpIACZAACVD0OAZIgARIgARCQ4CiF5quZkNJgARIgAQoehwDJEACJEACoSFA0QtNV7OhJEACJEACFD2OARIgARIggdAQoOiFpqvZUBIgARIgAYoexwAJkAAJkEBoCFD0QtPVbCgJkAAJkABFj2OABEiABEggNAQoeqHpajaUBEiABEiAoscxQAIkQAIkEBoCFL3QdDUbSgIkQAIkQNHjGCABEiABEggNAYpeaLqaDSUBEiABEqDocQyQAAmQAAmEhgBFLzRdzYaSAAmQAAlQ9DgGSIAESIAEQkOAohearmZDSYAESIAEKHocAyRAAiRAAqEhQNELTVezoSRAAiRAAhQ9jgESIAESIIHQEKDohaar2VASIAESIAGKHscACZAACZBAaAhQ9ELT1WwoCZAACZAARY9jgARIgARIIDQEKHqh6Wo2lARIgARIgKLHMUACJEACJBAaAhS90HQ1G0oCJEACJEDR4xggARIgARIIDQGKXmi6mg0lARIgARKg6HEMkAAJkAAJhIYARS80Xc2GkgAJkAAJUPQ4BkiABEiABEJDgKIXmq5mQ0mABEiABCh6HAMkQAIkQAKhIUDRC01Xs6EkQAIkQAIUPY4BEiABEiCB0BCg6IWmq9lQEiABEiABih7HAAmQAAmQQGgIUPRC09VsKAmQAAmQAEWPY4AESIAESCA0BCh6oelqNpQESIAESICixzFAAiRAAiQQGgIUvdB0NRtKAiRAAiRA0eMYIAESIAESCA0Bil5oupoNJQESIAESoOhxDJAACZAACYSGAEUvNF3NhpIACZAACVD0OAZIgARIgARCQ4CiF5quZkNJgARIgAQoehwDJEACJEACoSFA0QtNV7OhJEACJEACFD2OARIgARIggdAQoOiFpqvZUBIgARIgAYoexwAJkAAJkEBoCFD0QtPVbCgJkAAJkABFj2OABEiABEggNAQoeqHpajaUBEiABEiAoscxQAIkQAIkEBoCFL3QdDUbSgIkQAIkQNHjGCABEiABEggNAYpeaLqaDSUBEiABEqDocQyQAAmQAAmEhgBFLzRdzYaSAAmQAAlQ9DgGSIAESIAEQkOAohearmZDSYAESIAEKHocAyRAAiRAAqEhQNELTVezoSRAAiRAAhQ9jgESIAESIIHQEKDohaar2VASIAESIAGKHscACZAACZBAaAhQ9ELT1WwoCZAACZAARY9jgARIgARIIDQEKHqh6Wo2lARIgARIgKLHMUACJEACJBAaAhS90HQ1G0oCJEACJEDR4xggARIgARIIDQGKXmi6mg0lARIgARKg6HEMkAAJkAAJhIYARS80Xc2GkgAJkAAJUPQ4BkiABEiABEJDgKIXmq5mQ0mABEiABCh6HAMkQAIkQAKhIUDRC01Xs6EkQAIkQAIUPY4BEiABEiCB0BCg6IWmq9lQEiABEiABih7HAAmQAAmQQGgIUPRC09VsKAmQAAmQAEWPY4AESIAESCA0BCh6oelqNpQESIAESICixzFAAiRAAiQQGgIUvdB0NRtKAiRAAiRA0eMYIAESIAESCA0Bil5oupoNJQESIAESoOhxDJAACZAACYSGAEUvNF3NhpIACZAACVD0OAZIgARIgARCQ4CiF5quZkNJgARIgAQoehwDJEACJEACoSFA0QtNV7OhJEACJEACFD2OARIgARIggdAQoOiFpqvZUBIgARIgAYoexwAJkAAJkEBoCFD0QtPVbCgJkAAJkABFj2OABEiABEggNAQoeqHpajaUBEiABEiAoscxQAIkQAIkEBoCFL3QdDUbSgIkQAIkQNHjGCABEiABEggNAYpeaLqaDSUBEiABEqDocQyQAAmQAAmEhgBFLzRdzYaSAAmQAAlQ9DgGSIAESIAEQkOAohearmZDSYAESIAEKHocAyRAAiRAAqEhQNELTVezoSRAAiRAAhQ9jgESIAESIIHQEKDohaar2VASIAESIAGKHscACZAACZBAaAhQ9ELT1WwoCZAACZAARY9jgARIgARIIDQEKHqh6Wo2lARIgARIgKLHMUACJEACJBAaAhS90HQ1G0oCJEACJEDR4xggARIgARIIDQGKXmi6mg0lARIgARKg6HEMkAAJkAAJhIYARS80Xc2GkgAJkAAJUPQ4BkiABEiABEJDgKIXmq5mQ0mABEiABCh6HAMkQAIkQAKhIUDRC01Xs6EkQAIkQAIUPY4BEiABEiCB0BCg6IWmq9lQEiABEiABih7HAAmQAAmQQGgIUPRC09VsKAmQAAmQAEWPY4AESIAESCA0BCh6oelqNpQESIAESICixzFAAiRAAiQQGgIUvdB0NRtKAiRAAiRA0eMYIAESIAESCA0Bil5oupoNJQESIAESoOhxDJAACZAACYSGAEUvNF3NhpIACZAACVD0OAZIgARIgARCQ4CiF5quZkNJgARIgAQoehwDJEACJEACoSFA0QtNV7OhJEACJEACFD2OARIgARIggdAQoOiFpqvZUBIgARIgAYoexwAJkAAJkEBoCFD0QtPVbCgJkAAJkABFj2OABEiABEggNAQoeqHpajaUBEiABEiAoscxQAIkQAIkEBoCFL3QdDUbSgIkQAIkQNHjGCABEiABEggNAYpeaLqaDSUBEiABEqDocQyQAAmQAAmEhgBFLzRdzYaSAAmQAAlQ9DgGSIAESIAEQkOAohearmZDSYAESOD4CPzjHxu%2BfP19p7KUr9bTDeXU66q1uqocHFHrKI%2BKoneUtPldJEACJBBCAr%2F9j0Y%2BMTxgzYwMmsmpUWt8eswesLTacFyVenfZKb11x6tcv1OvLKasyjvv1ar4m6lGldPXp5wZCOM1pVxgM13ouu8fiChF70C4%2BGISIAESIIH9EjCvTm2Lk8E9DTHz%2F8nev%2BmWMqapLN2QPV7TrDZMrlY3uWoTW0Plak2TsZRaU55ZM7a9ql21fitlFVYLafebq74Yui%2B%2FrLx7RPGBh0jR22%2Fv8XUkQAIkQAIHItAtegd6YyCITRh3FdwpQyIrEEq5v4UtiedTeC6pLJVc31SVpc1W5c6mqWwkvfJizilVsnYjoypOKa%2Bc7IZqvZ1SDbxPXKn4GN5IgARIgARI4AkSeDyxO9CBiMUIK8%2FA4tNi7blQtSYEcQWW4woer8C8XK63vPXvL7n59z5w8hS9A%2FHli0mABEiABB5GIBC8R55yexy4Hddp29Xpi2AZWwHHgw17o4oUvcdBzPeSAAmQAAlsE9ix8I5F9PbVExS9fWHii0iABEiABPYi8FDRk0iWHrhR9HqgE3gIJEACJPA0ENgRPfEs3nPrDc1jIMvTMNDYBhIgARLoBQKB6N1H8HpIAGnp9cJI4TGQAAmQwFNAwLw68RB77gF%2FOmILkKL3FAw0NoEESIAEeoHAg0XvAMp2gJc%2BSpspeo9C7fjfI%2F3W6Tv%2F%2FvPPK2uhT01cOhcbv3rGGn%2F2dHT88rw9cXZGTyrLzOIl2NQpeB5O4Z1yf%2BRImtE1gPWncxxvRwKdX0ICR09gx7W5D%2Ffmfg%2FvEASQJ6H9wj%2F%2B11nnzqnYpQkVf%2FbCYOz8hBu7NG9ik5PW6GDcPtPfp8%2F0x%2FXZ%2FoQ6k4ips7EoxE6pGAQuhtyUGEQvjmRNeWwfWVMOYcA%2B5Njl29qJqp29lnyhIp7fUEZLFYckElXXZV8s6Uwq7%2BaTeZPfzJn8etotZJO1%2FHWlmmN3lLk1pMxrr%2B36vCPDxi8igZNIYMfKe4Ki1w3iCZ1PKHq9M7psWGt2X05F%2Bs4reyqq7OkhFZmb6hueHbHGZ8b1%2BPSkHp%2BbsMdmx9R4X8wXtSmcwLHX2Pv3sem%2BY23SExqYB2qD8asw1PCemr83pg6Rq7lGoZ67Wqo1vcVawyw1WtZSqaIWS04sdSud9jYWlXe9orx0WnkQOPmlHtKv9UCt4YtJ4EQSOHTRu5fKI55rKHrHN7yEvf3DP6wSL0wPJp494%2FZdmYsknjnvJc5Nx4aUds%2FgxH0GrzkN6wR7cxp2x1mc1FF71bfWLP%2B%2Bwf1gf7R9%2BYgD7hBw1%2FGZsN70mtJmHYKHTa83XbNxJ2kKd1ZN8YM1U7iVdIvvr9YLw99UhZd3ykV0WtE7rTkEQPxIEjgKAkcueo9oBR7tifIoyPfWd1if%2FrSyrpTgW5xW1vlBZV1%2BXmkr0jc1MGhNjwxGpof7DfYKez0djahxnLxH0QTZZM4tuK%2FVAPbBPJ6Im%2Fzr3Jf2HkUvHq8sSKFYETdYcH7h2DoEru64upwtemvZklnLls1arqjWMkV3rVQwaxs53UhnrGYatdo3C6q51lCtxUXU5KM111u%2FEB7NU0Hg0QXvkE4sD%2FnYozhdPhWdut9GXLqk4pcx7%2FaxM4Px58%2B78fmz0b4XT3vx8SE9pixrDpbIPCw37NU8tjnYaKeCOTfMtymNbdd9ser2dztoTx7SWNvfwR7oVVW8Oru9adz3VK7hqOzNJW%2FzvRVv88aSs%2FneMu4vtzbfvq1SeO2RLkp5oNbwxSTwFBJ4QqInqykUcBGPi1wTD%2BIQFPaqr%2B3hOhi5B5zjDnqqPNiXhuTVU1Nq8IUFRE4uRCeeOW1NXj1tTVw9Y09cnLfG0Vnj6LidvVJjwAILTg%2B2bTS4Jo3lOyu9tqvyaGy33ukdabmUihW3o9yXR1qKxZocnltruy7XxG2pjLfmuNb6zVWv%2Bu6iV1nccCo37tiVN9LV6ltv%2BUuPUPB6p2d5JCEh8Giit0uV5EEWv%2FwcvDj5tvgVcS7ow1p7I3jc8XqN4XyZeBysFL296UUuYQ6tNKMi54aUbcVVZDSm7JGRvoGpUXd6ZsKaOTWhpxFgMj07oWZOT%2BupiRFrAp3U728am4J70mAL3JTHG2iyd3uP8hUiclhE0l%2FnqrNBtLTjeSbVcsyi46nFlquXHM%2Fc9RreUqQ%2Fvvj2Xdu7TyDKvSsrH2U7%2BF0kEFoCu3PzDhIL1i16CD5TOuOLnhLhw7p5RmFiSJUcx5QzBc%2BzbTUZsfV0JKKmoraejUfVNF4r3jCJaejEOAT7h9woeveBg9SAvlMDqv%2FsnMLkWzwxP2X65qci8ekJ0zcU1%2BOYe5uJxfR0QrY%2BNd0ft2YG%2B8w00gSG0XERWCjYVATwo4HrEvsg%2BIS3HQIyNyfLfpTADHuDvTw2pWxRpZFGkEzmvWQqozZwP7WexuNkPRkrKfcaXjiElIIppBS0g1I66QrkSwIkcMQEnoDoyQWvnAsy%2BP1jE4tPb2KPpYBMKVdR5bWMU72TVHmIXxORezA41KnJEevU1Ig%2BNTWq5wb6EAsh6VkeXKJapop8t6hscu7ddQuj6HUSu%2FVngCL9aaVLuJ544bQamO3vG58ed8dPjdsjM2MamzUyN6GGRwdgXmtxS2JeTkmwib9NwAU3gb2kDoSR4%2BP8tCR3Lu1fzXnYtAx2bEZl1zNe7va6yb%2B%2F5m%2B5xWU3d2e5mfvOoj%2BvJ4EovJEACfQQgQOKXmfNO7TA%2BOdiLL6QxVaAjZi3MaWhtcq5nl63LXnOFFZSbnUtbarfve2urGd1o1BymumiapVrujkYN4MX5%2FX81QV7%2FsqChc2enx6VFC5MHxkziHOLbOJpk%2Bkk2UJxsvbBShTlVFpZs2eUlijKAURRXhGzeDg%2B25ewZvpjarYvbst%2BZiChpnHlMB2LiOUG%2F7HW8CFjbzRclQAp7kq6KR%2F1ZycCt9FOEsccnZKE8Q10ESw6p3B33au%2Bv6arH6w71aV1u3JjrVZJX1eV28qP2uSNBEigxwjsQ%2FS6%2FZiYhjAya99RH%2B0Ztey4Bg5MlYf45RstswplKkRtq1CpmfL6llN%2B%2FaY3%2F9Yd9z%2BupXU1l29UIHrNzZpqZbNKLqDlFv34JTX67Pn46OVT1tizZ6zRZ%2Bb16DPn7NE43KGQ1ilYgTP43Kmn1kL5NNyLjU%2Bp6IVRFb08pyLPTQ%2Fas9Ne5MIZY8%2BPR%2FuU7c0r11oI8t%2BQB6cNNuTEGSW5cUEeXPfWyYfrsQF3og5Hw7ozagnHvATW2OtF8F9U2lvMFL3s7aTl3LxjORuZknN9UTlbd5Tzym1%2Frk%2Fm63gjARLoQQI7onffWYZ7YigNvDX%2BiubB5J9RK03HK7dcVXBcVXQ9UyhX1ZLWpgiLr5TKeZVU1lR%2B68vujWxe11eL9dpGUVVRUEKiuh82gegX%2B%2FjhCyrykXMqMjsxHLl6wYu8OOtFnjbRsz92Tg2dPaOGLkzFhi%2FP2UPn5rzhSwvW0IW5yBDM5UlADqqXaN8tKffbzxkJMNnJgbs3J64HB9uTP6THHg4IQjEOLLcWhmMQmKINAlW0U2uqZLluVnDltlKpquVy01vJF9VKxdErW61qYQPVUTIpZRqbymMJsCffs%2FxEEjgsAvsWvSBCGzm2chFr%2FMAziNytRksVm01VQkGJYqupsvmKWsHz5VoLASx5r%2Fbdm97wF1%2BvfzlTVo2co%2BqplF996SARM9J0OblJ3jS09OTf7GfPqumLM9Hps%2FNm%2BsKsNXNp3p4%2BP6dmLpyyxE0poa4dlyT2kirguyhluzeS8mngcRw9igGsceVlsGFvsLd0xd9rXU1mvBR88snVLZPcyKjkaqqVXMnbyXSmkRw8gzQDRKOwSspxdBu%2FkwQej8A%2BXZuB3Ek6khILDRfGQelAD67MN2sNVaq3sDVNKY8iE8WqmzGehftePV106%2F%2F35xtvYXKvCWFsrq760xwHFbxdjezlk7xfbguRlPZwTdkJpAtMJpAyMKHsQVvZpyajs9PD1uz0mA62ETU7NWrNTuH%2BQJ%2Bf09FJbEQ0j190OYjm0f6etydHQCqlSF5NEVZdASMbmx91VXCMVbi74WburKkM8um2bq17maUNK3M71cx875YfvCLv5Y0ESOCEEthD9AJ%2F546DE54fP0LbT0sSQ69cM9%2Bo1FVZ5u5g8ZWWUu4HjYZXabiYuyt6jffWTOQ3v9L4OtyZkn8r22MJnmDuNdGLvDSnYglUNDk9rWIXphKxiXEvvjBtYjPjkfhgVE%2F39alTsSiCTqLqVFyCT2La38dsRFPKXJyfo6GxGQQCtetUBnkbvdbWEzrMdx12Ho8yEDtEXlpBBKaEHCMas1ZXucUN1L7EhuCUIsSvsJSyC4ub9eK1WxBFJpE%2FDf3PNoScwJ6iJ5InxRODG8oImgq8P5jXk7k9zNsXzJ9VaqqMqY9KvuTmFlPeUrVlVetVt56v6Oa%2F%2F9Xam4vBnL5sjy14PSF6SPyOTz2vBs5OqIHz87GBuVmEoJ6yBy7OWgPnsMVi%2FpybJCHK%2FNu0fz%2BYj%2Bs8F%2FJhdwzNF9eEQrkvC5tRmxjEwV55KfRNKlcxueWkVVlcc6t3Nt3qyoZVublcr97Kqert237VFAamHEO38StJ4EkTeIjodVITAsEL3JtSYkymPySYpVGtm1fzFVPNl0y1VDPV1bS3tpZxNyt1XYcQtrIIbfk3X2i91Ra7J1Y48Visn5kZNXBmSo1fmIqOn50y4%2BdPWeNnsZ2b1WPnpq2JRB%2Fy4pSSdAHkxyFtQJthaHxwX%2BH%2BU5vofSzdcb%2FfQXvAIoA4GKydEmGm1dSVUt2sl2tqo1z1NopVtYEBul6sq%2FU6ntssu8U8CjxvZaxGqlJvJnPwxZdUI%2BWoJgs%2BP%2BlTDj%2BPBI6XwEOCWLpFrxPWKZ4gBKFoTGuYRqZovpApqWqx5FQLVav6%2FQ%2Bad1MFq9hoOU3k4Dm%2F9MfNdw6jdUd2lh0dVaMXp9Xs%2BbnI7LkZa%2FbsrD51bgaPZ%2B1ZrO59arBfo8YGIiiDObe%2B9hzczuOnVugOo1sf%2BTNlcIoLoasGZrsuJmph5kre1kbWrCazajWZ81ZRLWU1mTErGzmztpZqrdYcVWvWlFuOKg8J%2Fy6sdA9CJ5%2FXcU08sau1R24h30gCJPDECDxU9DrnksDSk8AVKQYP0YPwGd1YSju%2FCyuvlivq2kbGyX%2FrhrNYq%2Bl6C68t1bX7u19pvP%2FEDrTrgx5X9CSfzV5YQC3KhrJrCDaZQbCJtlXExiKos%2F2xsakxb2Z23DqFCiczqE05i7Ixs3g8g%2BCTUwg4wbpxKNdl%2FFIxUrZr537wmKW7DqPX72%2FbBX7z7XBiHdw3vui5d1NmZTllVpdS3srihlpZTDqrd9N6ZXHNWV1MqpW2sB3V0fJ7SIAEeoDAPkVPLnYlnWkDYif5dRLBWb%2Bx4vzWZt7UcyVdv7XqJN%2B%2BbZLlhnKqjvY8t%2B6%2Bek0tH0YTDyx6MgensHzOqSEVH4yp%2BMRovG9q0MQnh018YsTEhwbtscE%2BPYUaldMDA2pKKpsM9qupQdSoxDYVj4mLsivxW4JMdlYX6CSE793WAx%2F53h8Z0leI2EnRZ5lYbkHoZN%2BE1S0RVs1a01SWkmZjadOsLyW9dew3MNm8vrppbSyvtzYW06iowhsJkEAoCTxA9Doene4i8CJ2sshzBWf8SipnfmNp06mkkXC%2BVfAaX3%2FXWfxguZmXBLxIU3lffQ9Vmw7pth%2Fp0M9PoS4l5uFmZvsGTs94A%2FPjZuDctN2%2FMGMNnp6x%2Bvuj3pil1YTRehKZfxNy37LUBDK9Jy1tsIdFJ7fg23a%2BU%2B496orf%2BznyQ4L21HysLMiq%2FUTPYIHW7c33udczBZNfSXmby5sqtZg2qeUNk5LHHySdzXRKpZYLfkV03kiABEJIYHcQy65qLJ0HwVSJv8kyYVJu0C8sXbmx1Pp%2FUlndTELwbq15ha%2B%2BVVtZqqimCEUFGb43kOZ0WEgfJB2RK3NqdOGUGjk7GR05O2dGT09YI2dmrNFzM3pkYUqP9sV9iw3rHElgiRRkbu87ASiohH1YB83kg8ckG%2FjX5corSCD3E8v9x8Fz%2BNvapsmubHlbKymTXkqZzMqmt7W4qbeW11pbaxW1hbwZGby8kQAJhJTAPiI3O5aeCN86hE%2FKEPrLBV27af7N2pbbWs24jTduept%2F8EbD9xiNwwi6uSUrrxzerVv0%2Bl6cV5Mzc9HJc9Nm8vSsNXlmUk%2FI%2BnBnpvXEmWlrMhZDJZOg6DI2FGH214trF2LW8twRrxX3VBSUObzOvfm39C0AACAASURBVP8n%2B9FTWKEYbgbZawPx2nmMNexKq2mVXUmbLObwsnBpZhc3vezKppVdTDZzN1f81Q6YVH7U3cbvI4EeI7CP%2BbxgcWh%2FJXR1F5ssEisFoot%2Fes35xbWs21xJOs3Pv95ce%2F2Gn7trncO2eMjnF%2F3RC%2BrK%2FGxk9syEmV2YxCIEM3oWAje7gEjL09hjpQFZYUACTbo2jXXi8JysF3ef9Yp6rG%2F2PpxedJU%2B%2BThHuBo0rrJMERcuuJKS9ev8vQxCubIqIv2guLoJl%2BaW79bML6W9%2FPqWzq8lW%2Fm7WMvq1joqrwSLvfJGAiQQcgJ7iF53xLYko98ALjl%2FSNWm93%2F3q83fXkm3Wt%2B9a5d%2B45WaWHlyXgkmvA75pv%2Fffxj7GZTumplFCa%2FpMTU7izJeWKFW0gb8yExR3wfc70WpODiuXmnF4Xa1WGZBiTC%2FVJiUCcMVF9awCsqGmQIqIxSX0qYIsSsub%2Bni%2BqZbXExaxbV0s7i%2Brgp3cv4V2uEe5cF7j%2B8gARI4JgIPCWKR84REfgepT8G0yXWce3J4VPjKm94%2F%2FmCj6S5ntPONt2qZL77tF6w4spvOfK7%2Fe1g%2FbgpLr4u7Ekuti8%2FQdC%2FB7lenDv728GXYj%2Byon%2BQXHZfoHZ18yNWVBJxg8yeTpU6m3M%2Bjr3NYyyq3kfHKy5vY0rqylnbLyymrvLTRrGAJj%2FKdlKpw%2Fu5JDjh%2BFgk8HQQeKHo7Ls2dpHRj3sO5JwsNyf32q7V%2FvrxpNa8vVRvXkqp0%2FfrRLg6tzZ%2F2I1EQh7ITWSn37rc9HT0VnlZINXKJgMIcHAabQU1MS1YnN3iMOpnaZJstU1jd1LXltFffzOraUsqppTK6upFr1lZQMuzGqqrtY92q8BBlS0mABLYJPFT0%2FGXF2oUulEEer76JC20RvexnP9f4ueVVu3lrs1z7g6%2F7Vt4Tqam5364R0bvfFx6X%2FbPf4%2BbrHk5ArLuttugFRaCNhQHnF4POKs9kG47Or2a8xmYaCzNuOY3VrK5vbOl6Kt2or5VVPb2k6m%2Bn%2FIAV1snkaCMBEvgQgT0sPVkRQSw9WUbofVxwS3WVjPK87E%2F%2FhvOLd1Yqra98oCqo2HTkQXEiekfnaOPAOQoCYtVtwXW5hcEmIgfxw2Nl4TkRP7XltLzcak43N3JOYz2tm6mkbiyXrWZqudZMZ1Qjie3aur9uFQXvKHqM30ECJ4zAbsGTg9%2BWkaCUofEXkxaDSnKBv4fHi3Ih%2Ftuv1n%2FxGx9EKreXy41Xvu3P9R15YBxF74QNtj0OV9yXEDeFfBiInm%2Ft4TGEDisRYo%2F7rslsZHRzMe20UA2huZbVLeTfNTfStSYWeG3cSPkLNUpVFgre0zU22BoSeGIEHiJ6InSyiZgFQSxafQP7NTn%2F%2FOS%2Fdn5xvVlu3XzTv7CWwhhHbnRp8%2BXEkX%2FpEyPPD%2BoiICInFh0ET%2Bl0IHwQOwsJoS6et%2B0UcvAyyxtOK1PUzaWcbiU3deuDrVorXVTN9UXVWm2oFpb%2BeSILNbJrSIAEnl4Ce7g25YJZ3JoSLSJxBX%2BOc5MsQbb53%2F%2FT3C%2FX8sp5t65qRx3A0ukNit5TMS411rKTde2MrGsH0ZN17jDIbCuFay15frNUcQsIUmkhHcGBa9PZzNRa6xnl3FhWTt5SreY3lXMtuDrjRdBTMSbYCBI4PAL7mM%2FrVGNZhvB9DeejJLxPm%2F%2Fl%2F5779TtZ1Xr77eOLF6DoHd64OPxPlrqZ%2FuKt%2FpIdEDqIn0YBaBE8C3sXeyzsmm9a1eV17WZzlnM3V3FTuNKCZecubirnvYJyYd0FV2a8kQAJkMA%2BCDwkR09cmlLA3sM5CLl6kqqgvgkrbyOVVl%2F4W5%2FNLb2LKRSccyRm4FhuFL1jwf74X%2Bq4Ktt0TKrlqpTrqmTL0alG06Qw4pJIRUi6rpdqNiKprYblLKcqXr6u3FRFuemk8tJrWK8KcVSlCeVeu7a91t3jHxQ%2FgQRIIBQE9hA9iQkIgliMeQOC9x1cjK%2F%2FZ%2F8o96trWG8TVp4I3rFdZFP0TtgQxVrmTqWuktWaSmHtqWS5ZpJVPC7XsW%2BYVLXiJQuOnaxmsTCxq7xMS3kuluoooQbL7ZLyRtPK2xxRHsSuM%2BFMd%2BYJGwM8XBI4bgIPEL1OrU0RNdxHIXvtfQX7txBbsPbX%2Fl7%2B5TtV5Rx33ABF77hHzwG%2Bv9YwpVxZJfMltZGrqGSuZDawpfIVs1EomWSuoDeKbiuZQiXzPsRqpuPKJFaVWca%2BD2vRQ%2Bg6FRK69wc4Ar6UBEiABDDx%2F%2BqELBfURtHei0szSEiXNAURwDwqPr3iuuY9lPRa%2F09%2BovCH31z1A%2BWOzcqTA6bonZARvFVUqVTO20jn1UYqa5JbBbWRxD5d8DY2IHbZvLNxJ6OSqJMproV7RqPfyG6LjtbdCel3HiYJ9BqBB6YrBKIn8QGBe1Or5XrT%2BwOnpZYs21r76N%2FNfem4rTyKXq%2BNpvscT71p6sifg8B5G4i23NjImo1kxqRkv57D44y7sQ4BRLkwrmB%2BAvqTh0gCJ53AA0QvSEoPrLggkEWr67mS9znUuNz4m%2F88%2F%2Fl2Mnr3aurHggKWXl8PXPWz6tn9ej9bVNlkzksmsxC9nEmuY4%2Fi0CmIHh4jGmrLSa5C8ApcwfxYfjz8UhIII4E9RE%2FclyJ6LQjLN1M59z%2FCoZj8G%2F914fNIieosN3Ss2HpE9I6VQc99OaIx3WROJVNZldzIiuhB5ODKXM%2FpVCrjbWxm3SRWQEjeXlEbOHgp5cMbCZAACRwJgYfk6Il1J65NEb4qosj%2FZCNnvuZ6XvLif1t6pW0JHruRRdE7kmGy%2Fy8p11RZrDpYdyJ2wQZ3ZjJvkhsQwi0I3hpcmWv4e9uVsP8P5ytJgARI4DEJPDBys5OmELg48%2Fmy%2Bb1U3rvmeTr13I8Xvoznjl3wpOkUvcccAE%2Fy7QhSSfsiB5fmZl6lcJWUxHwerDsJWNGYx3OS6U2IYdmvusIbCZAACRwpAfPlSRQX646T%2B1DkJhaqxloKrlqHa%2FN3Noveu7%2FyRfXVz75clhKJFL0j7a3j%2BbIgNUD%2B19vpAnIkOykDuIck83oqpzY3shC4vNmEyIllt4loTVh5Go%2Fd1FpBbearKpXN%2BiuY80YCJEACR05gR%2FTum64gq6VLwXvPcc27i0nvc5mSWvyh%2F7GIeb2jXTPvYWBo6R3OsOlEMgUJ4JKzIlkrQQ4LNllyw%2FiJnMixy2%2FmTBpzeOnNPPYZs5mExbeZNelUTqfX8s5msqg219Z86%2B7YSvccDiZ%2BKgmQwEki8EDRkyAVg00jzgCRm5WG962bK%2BYL%2BbK38Zd%2FsvxnvWLlCWuK3pMdcSJ20vmdfBVEK2lsRqKWdjb8HarX2sypLczfbUHggn3ebKULWPpnS22lCnornXO21ku4n%2FIXhD3S1YWfLBZ%2BGgmQwNNAYM%2FITaNWcRKsYz7vtbfvul8sV1Xmb%2FyvJVlaqGduFL0n1xWdos2YxNVYVsMXOiyk6E%2Fq7tpQNqwKN2Ymldvespt5nUFkZmaziH3OycLVmUmXVCaXU4Und4j8JBIgARJ4dAIPET0534knatH1TCWVUa98b7H1jcWks%2FS3%2F6%2FGzUf%2Fxif%2FTore4zOVObsWxA1huhp7g2qXvtgF%2BSr%2BhueV5%2B8zRVOAZZdN5nVus2CymzkvB%2FdmdhO5dpi%2Fy25B8LZKKndrVWXxXllkkTcSIAESOHYCHw5ikUPy5%2FaCaRujcjjvrWI%2BL%2F9B0vv89bvem%2F%2FV%2F1b%2BWvvvx378nQOg6D1eV4gFh7wUETUj%2BSmdHJX2Xp4TIUSsSkvVMW%2BXgwszn86bXDLv5TFnl9sqeLnkls5v5p0c8vJyyarKJ5MYPMdcn%2B7xsPDdJEACTxuBB4reznQOVkdXqVpTLb275PzxuyvOjR%2F9Z7Vv9xoHit6j9UiwZpSGqBkRPIhdYO3tFru2CBarpgyxy0HkfMGDdZffKuocSovltwp4LgPBgxtTKqssF1S%2Bc%2Fn0aIfGd5EACZDAkyewZ%2BSmUrfwrRmUHrv%2B5m3vSx%2BseXf%2F1r%2BqvPXkj%2BTxPpGid2B%2Bpm3dicjpHbHzfBGU8jvb1h3SWZoQuQLcmOLCzCPXLr8FS2%2Bz4BUyOZXfLMtjJ49ozTwSzwt57A98OHwDCZAACRwBgYdGbgaBerKOSxF1gb%2Fz9eut15ZSrY1%2F8O%2Ba7x7BoR3oKyB68Z5IGDzQUR%2FPiwPrLpijCyw6sewMhC54vm3lBZZfo2Xq6TxErQiRg%2FCl4c7cymvZF7BiAp7TuUzBKaxlVL7QVPmtLVU6nmbxW0mABEhgbwL3Fb0d12YdUzzfwuLWlcWk%2B9U%2Ff8v52k%2F8fA3lNn0x7KkbRW9%2F3bFj3ckcne%2FK9Ofw7glWCYSvUjdVWHhwXcr8nS6iKkE%2BAwHchOBlYM35Fh8EL1WASxOChxUSyvs7DL6KBEiABI6HwAMiN%2BXcKKlZiDI3rzdbqnR90Xn1le82Xv9ffqX5%2FeM50od%2FK0Xv4Xw6kZlNuDIDsVNty247YrPL8sPf8mVVgnVXQL5dPp3TxQwED%2FN3EDqvgMeFLbgxt6pOfiOp8q0tVVhnwehe%2FF3wmEiABLoIPCSIBedG35pL4dx4rVw3m9%2B56X75S99rvfPPfrUhc3w9d6PoPbhLpCO7Ug52iZ10dFsE%2FchNP2UB83Xiuiz483Z5U8RCr%2BLeLGRg0SH%2FLp8vOQUUkvbn8%2B5u%2BvN3rLDScz8JHhAJkMC9BB6SrgC3pp%2BitYT999I57%2B7Xrjt%2F%2FpU33ff%2F1efqi71IkqJ3%2F17pclu2c%2FB8K%2B%2FeTQJXdMvxVAOuTF%2FwfAsPgpdpW3cQPHFxwpXp5JFsns9uqMIa9oFLgDcSIAES6H0CD7H0qm0D4AamfK4vbarrX%2Fpu4xv%2F9udq1zChJ%2BfMnrtR9HZ3SVei%2BX1ErjsJ3SDZHC7PelNVEZkJwcOGuTrM24k70w9YScOiy1U0ktGdfCYDV2ZNFVhhped%2BAzwgEiCBPQh8OIgFbzCoH2ypCqZ9pALVNdfVN99bdd74wuved%2F7hL1Xe6VWoFL2dngncmb7bcsdl6T%2FXsfJ80duprpIrqWKmZAooJwbR0whSQcBKScOF6RWyeJxCOkKuogqcv%2BvV4c%2FjIgES2A%2BB%2B4qe8tO3RPQkTeurDcfcef1m61u%2F%2F1XnzZ%2F%2F3fqd%2FXzucbyGohdQl85rR2WiA7fn6%2FZwZ4rrUoJWutyZW3lVyJZh6WUcWH3IWYHooWC01M%2Fk%2FN1xjHB%2BJwmQwGMR8AVPbv46el1LConYaT%2FyHJt5FQtgr%2F%2Fp262v%2FP6X1Dv%2F4bWKLHLdk7ewi1537t1OZOa9c3di%2BemgdmalriqScB4EqEDcEJUpVt5WzitmEbDi3y8jOhNRmlVsqLAigscVEnpy%2BPOgSIAE9iLwQNHDagqB4MmyZ%2BaVVF6tvvp645u%2F%2FMXajdfe6t1CG2EWvS7rTiIzO%2Fl391p3O65O5NgVt%2BDO9INWIHgZqawCd2YnHSFXdCRi08%2B%2FyzcggEw43%2Bv3xL%2BTAAn0OIEHBLGIyVeB2GGzVnD%2Fj9%2Ff8BZf%2Bbrznb%2F7b8s32t6znmxZGEVvx7rzc%2B%2F80mE7qyLcJ1ilgWLRgdBhDk8CVmDpZeDC3Mp7mNOTiE2Zz3P8%2B1tISVhM%2B9YdV0joySHPgyIBEjgIgfuKngSxaFUMhE%2B97xn1J2%2Fddm7%2B%2Fjdab%2F2TX6vfPcjnH%2FVrwyZ6H7buOvN3HeG7J1ilWFEVEbwsLDypriJWnczZ%2Be7MMkQOll4u6xRTiMzMlFVxfd0XvJ4M1T3qwcXv61kC1ksvKXu6oKzBKcTfxZWN%2BRjLio7YsaZnNYwX7bO8SCtuolgbLRq%2FpxmOMVGjTcQy0aiH%2B3u1UlvKRlh0BGEPEYS125ZRlrGVxl5jmsjf9vqMXX%2B3%2FYWaETGoXbzR8SzlWVobWcUSTxiNrfv1WLHZ0xHtIOAaf9ZOxFYeVj0xlrzW0lJIy7XwbrzZtftkb7lOTHvNhvaGm0U3XVfuYkU5i4t%2BmpGcQ0JVuvEBlp4YCjmQqGF7q%2BGaP%2FvG9dY7v%2FnH6vu%2F%2FOUKEtV79xYO0Qvqw7Utuge4MretPUk01y1cx0ix6GIW1husO4gdhK%2BoiojKhAsTbk1YfZsVV4pEF9MQu3bBaBG8UP0gendo88geQiDymU%2Bp6IWFsdhLl1qxxFQ8NqhdCFwk5npuzNNWImL0AK7eB5XtDUBOYt2fBcGLQiSCvys1gCF%2Fry7u%2BmoNPYXoJYxWCShcH4Qnjt%2BjjSkFG4JnQ3usg%2FRW%2BwdWw%2BfUPG3qkC3cVy6ex5pu2sX8%2B4dzYI2uQ3xxgjZ1Y3QNIon0Wg9vx%2BstBW03DRxnDU%2FVLdvUXNdqWhqb5TZty27eTrr177zXqi8ulWpffNufywpFnu395%2FOkt%2FwyjFlf9LT6Oi6avv7Kd%2Btv%2FIs%2Fqn0fCXrVg%2FTnUb82DKLXtu6Cqin4UQSuzF1lxHAN2uXWrNRUTSw7EbsMhC5TwL7tzhTh28L9LObvYOkVc5i%2Fy9YhfKyfedRjN8zfJyJhPY%2BtsKDs8ZayGqPYGsoecJTVQvaUm1BWzFG2i1O6i8fRGOTL68N9Y02PmcgnL0Xjn3yuL%2F5Dz1uxKwuxOH4XECIvht8BBEz3wf4aUp4ZxgXgEL5LxO3eG57H3wxeY%2FDavWTL4DOV6t%2FejIrhsyF8In6wAPFhB%2BtQP1xeTq7YNDZTx7GLELWtMbNblEQpLS3h9fIesU4kqRrnAoieB5GUfXACl%2BfldTI9gc9EcXltJPK6cWPFqX3rvVb1zfca1ddvO6ivq1u2pT3LqnutpnJtWJw2rMgmLEPZR%2FG4FlNutKi8OB4XE8odWFTe9c6iqycgwO3BoifCbzJt9l%2FESjLf%2Fv3XnGs%2F%2B0r5g9u3eztSHaIXu8cyOeDYO9hIPbpX%2B9Zd93ydv%2FrB7vk7KR%2B2beH5i722ckVTEqHLiuAV4M4sihsT7kzsfasOll665BRzuF9COsIHKd%2BvLVd%2BvJHAURGI%2FuAllbhyYbDvzOlIYizhJqAfCaXdhOVZIlgJuO36YPvgOQ0Ly%2FTBwZfAScp%2FDI9e38K0bc%2BN2xHZL0zhFA33o1hevgAZcT761h3EUKw4COK9omRUFL%2BX9mvwOt9ye%2BgNhhneo7B5IngidLC9DKw8C4%2Fg8nzou%2B%2FvP%2BkqFeivaQnxkkhpia3XMnd%2F77tgcPqenE6xeBHO4HX%2BHJX%2FnnsuiMWdKYIIEbWUgzJbrbUtt7WWDvaVhgiiqdmwHuG0rRlPixjAAsXeeDVtw3rE3nKtuo64tULNrt1cadVXU9XaalnVel0gHmLpycXDFkjntdafv7PWev3Xvui89Ye%2FUUv1aiWWzvi6j%2Bgd1e%2F2EL5nZ4jvWHO%2BDz4QNF%2Fg%2FPm2jtXXfgx3puOglJhv2YnYQfTKcG364ucVN0s6uI%2BAFax%2FVyyUVCmTVUWUExN3pnwmbyRwZAQuXVLxFy4Mjnz6GWvkR16IDT93IT4CE27Es60RjT3O4Nj0CCRlGEozYjx5rEZwOTuCEzueMyM4UdkWZAbPaX%2B%2Fc60b3JP%2F4Qf078ss2f1uMhe38%2Bq92%2B9%2F5i6Lbn9X2HtPGOy8ojOf97Akoc5r5F33zP%2FtNMJPSgtumC%2FsTjrCHwysZxiIgkhuvvVYQNOK%2BLgCsBUwYVmA67SAC46iPIa6Fzx%2F7xWqTavwje%2FXCl99q1589VqtcO2Ofx7pydsDRE8QSADLJrYkxsfvvPlB89rP%2Fnr5nd%2F8mm8E9LTr92SL3od%2FDDLUA2ETkQvSEHYeG7hEdhLP2xGbyL2rmqpYdyJ6UmUFgSv%2B3F2u6hVRSgzPYbWEglPKwZ0p6QiZqiqhnJh07t4%2Fx54cyjyoYyZgfQaDM%2F1ppackkCStYgl7KB6x3HhL%2B56wuKOjcZwg45ikwl5hb8Vx7vXv9yesvqtn7MSV%2BUj%2F1TOR%2FgvzkQROyv0Yjf0Y32LRBXt5rNp73b6%2F7WLU4lLc%2B9ZLI7yXjqVDbidXWxK1xc0KtyjcrXLf4L7svbbbNHDH%2Bs%2FBoVq9teZUbi27FVh%2BlVurTqXRVA0IZ8OyvQZOZA0YwNhbjYhpNWBmNloVq5HOVBvJjGpcT%2FsuxEMPmHtAjp64giVdAQEr%2Bg7ClH7z1W833%2Fn7v1W99fbbfjRnT99Onug9eOB3BK0tcm3BC1ycMjjaVl47PQEJ5%2FioFtyYcGcGYoeSYpJnV8qWEaFZ0KUtiFxeAlfKcGfi%2BXRGFetbEL4gTJc3EngkAhA8W31G2Z%2FCfNzHL8G3GDFDlquGPQcWmFLDsA78PT58GGPU32NCasTfGzUCZ%2BTg8ICOjAxa9nA%2F9gNW2z0p0ZFwUXpwVfrzZeKy9ANGsN91vzOXtr%2Fj72Wx2V8LDvdVO8IHrZKYAX8%2BUTxM2Lbv4zmZh%2Bx6DkE0pZrXKFa9ZqFsGgXs4XGSZG9cUMv6dKoIGxsWpCrC1CzigqdYbqrC9241i28tucW7d2qFb9%2F2L74P5bZb8OQrthsqRkUZlu4mImDfLtXNr%2F%2Fuq%2FW3f%2FZL9fXr1%2F01RXv6djJE7%2BE%2FOlwMt4WsE6ji%2B%2BXbVl23W3Nn%2Fs6pN00NeXWw3mDVwZ0JK6%2BEtfB8dyaqqpQggkhHcEvi1oRlh7k8VUom%2FQHGcmI9PaSP%2FeDEZWdLkEnlnLKGa8quDik70VR2raki8T5l%2FxACSD5%2BPhr%2F%2BEcisY9d6Iv3D0gwiB7FSWU02GMzcl%2BN4uQiYhfct2SP5%2BGyPNJW9qLodQD0yrE93nF0AnCCfeAulZVYgk0b7DW2YO95pvDGbSd37WYz98bNVu6tO16uVGg0XFs50YhyGwikScSU06wqp9KnnJkZ5SCisuNy3JHoPQbRQ0QP50BdcFyTidjm6wj6e%2FkXfs9966d%2FrZzDRx669fm4Y7%2B3RW%2FvgbTbuvOFzg9YCaIxg3WeOksCBX%2FzlJOvmnJerDoIWa5kfGHz3ZklJJtjEdhsCaKH6Mx0RVybqlTB%2FB3KiZWCqzfeSODBBCB2scFLqm8OQSZXRty%2B8clIwmtZg9p2B%2BCqGoxqb3B%2Byh5GAMnQ6enI8MKkNRSJwAUp7khP9hJwoiTSEe5JBJ7I3n%2BM5wOXJfYITDlobtvT2ml7nyOOruWPdyzybpmekbQLyV6U6RlxhSKNAoFyWtym7YjS9vNbebe8sumVV9JuaQ1bvgIr0TMVbXllbayya5uycewS3KXFSMsrVmMjxczKuvuNTeVduKa8lztBPA8g9KAgFhwognRMFlaoLBz7p4vr3st%2F798Xb37uNf8c2dPzedLU3hO9%2FQ2cHeuuE6gSVPzujs7ccWf6z2vH8bxmviyrmUPo2u5MBK2UchWpm6nluZLvziw5pWJg4ZUkHYHlxI7uvHHSv%2BnFGTXwiZcGhj75jDX8qWfjQ%2FOT9qjreZM4eU1ijmYCv7jJSMSaikXUZBxbLKYnYRpKVCNiHSSCEeEnQd6abBIoEjyv25GOwfN7JQicdIwHO%2F79nTMO9pmP%2BurHP5adT5CFzvxoVIkqRXSpL4id54LnXde0mo6pNVum3nRUreX4Uy8ZjLMt%2FH0Lr8pgAKUQkpTCqEq5LZNaXlal15N9zjdXV92XX95O8%2FhQix8YuQlLFHmNW8i%2BlDm9z%2F3Jd9z%2F73%2F%2BreIG5vNEmHu%2BznDviN7%2BB0u3dSc%2Bc7ki2j1fB2uubeG1X6udSt3UJBBFBE%2Fm72QuT%2Fa5sgerTpd8C6%2FiSl5eKZtTpQJei9XN5cqlpxMtH%2FW3yfc9EoHI3JyKTcdUrH9YxWKJRCze8uCwNHHjmj5jmb6rZ%2BODHzkXGXrmjD30kYvRwYkhmYvzXZKB%2B1JclR13pW7fD3LVgtv%2BfweP1ICn9k29xO3oj6WrcL4%2FpyYX%2FzInuO0Sxf0sBlcG11P%2BvtEyue%2FfadXfWzW163ca9XeQg%2BjUdd2KWnXVrDWK%2Far%2BtZ%2BZDOYLd62uICPUVFquSiMPcRmXX7%2F92c%2FV%2F%2Bjf%2FWltE%2FN54lU7%2BtYfcFAfn%2BgdHM2DrLsPW3ji0hTh8yeVvVauDFGTOTtYcvA%2FI1BFkso13AFeCfN5CFhBSkIQnVnqSkeQCWXO3x1wQD3NL0ek5eDHFvqHL512hy4uxIenhpEOYKlxXIOP4yJrHE6p8elxq39yxOqfGbP6p7H1xfxEb7gjt92W4roMNt3e3y9H7eC%2Fj6cZ%2Fd5t6zVex3M8nblBEUG5LwIoOcSSuB8k2%2FvRpVY7ylRVJYYBuciFdBbrgmLh62LdzcOuzP7YX499YedCTBrTbhAiamBsQvRMKhbRNzCuf%2B1%2F%2BqXmn332ZX8%2B70RM%2F2jz6r3J6XuPr2N4hcDsclduW3f3RGT6c3mdVASn0fIacGfKvJ1vweUqiNSEyOVh3SHhPAhcKehytoLcu6IqJ5FwLi5NJKGfCN%2F0MfRDmL9S%2F8UX1dQPPj849Refi0x96pnoFERtDvlY88i%2FOg0X0jzONHOoqRVHYrgkc%2Feh1BWSu2HFdefAdeeqfThvbTff4zlxntw%2B7jVex388u12lvtWGTXITO3vfkDMVGHNpjN80nk%2FjT5tR2%2F2JHSuv80b%2FxSjqoyqWpd%2FHxdzNt286P%2FNPfqWUfPmaH3Xa8%2FN50pJeF717rLtOuoEvgDuCt7NKwnYgS6lqKnBflhGd6QeqQPjKflpCxSsXUGnF%2F1tFks4duDVVWdIR4Awvc%2F7u5J7zHuPI9blzKo4k7fjsoIpPjSbiluclYlEv4XkR5L%2BZRF%2BfNXD1tB69vBAdu3LGGru8YI8OJOwJ%2FG0C39vZxHUpaQKYo9PYpBjzAxK793Owx3%2FS3M9R9sZrepFVLx3TXscSVLAKUiUM9tp94T6iJ65NTFGrKlIV3sPf3%2FjDb9X%2B5U9%2Frp5CdKhYlD0%2Fn9frohcIWxB96cCo3onM3GX1dVuAqJCO0neIxAzETgJT2vuOO3MLYleUiiplF%2FN5EDm4NNN4jNXNxbrjckC9cQo76qOIvXRFDV9aiI88f94evjQXG0aQyTiqESPww2jQWAAAIABJREFURE3A3TMRj1uj40N6cHzIGhwfsQYmcB%2B1FoPkbz8h3N%2FH8bMXyy4INgmCUPZXdeR%2BLd7rRHXUlB70fb14nL14TN38evb42rVGjSs1V%2B%2Bdz5NnRNwwX2jex1j%2F6j%2F9ncZnf%2FPr1cxJmc%2FrVdELrLtuwZPITH%2Ftu7b7ctfftpPNJfeuLvNyEqSC8F2ZxxPrzndnFuDOhODBxQm3Zt4pIypTAllKxS1VRjkxmbDt%2BfySXjnHneDjuJ8AiZU3%2FCNX45M%2F8mxs6i88F5%2F6%2BGV7GsWJF3DlO4%2BLLez1PH7kY7gKFrdl93b4KI7y5Nhxe%2B2nVd2lvPbz%2BoO85lEvFB71fQc5tif12qPs10c65ransjuIJVitBoaBgTWobmXK3jd%2F6le9%2F%2FNf%2F35J8glPhGuzF0Wv47bcnpfzLbz7ilxHAINVE%2FIVUxGrzg9WKbfFTiw6uDNRRkyiMSF%2BsACRjlCQ6ipwabbLiYmFdyLM8kcau%2BF%2Bk0beXFRNqdjwtIqNTQ3EhpXbh7rCA462B7CwWz9CvgcuzEVGLi3YY1fn7ZGLc%2FbY3JQ9BhcPglP0OPBhw33JkwtSC6S6ibgt91fG60nwf%2FInSKkOIvlgSB30k6F3Cjfv5Lq2X4PfhuWvQNCpKIJqHFLlyM%2BHffCRPf4xywUKUjb8%2Bp%2FtgtTtgtjdxbGlT%2Fzi16g4Y%2BBSDopa477s%2FX7aqxD2k%2BihR%2FuMx2f0aN%2B757u6BU9evH2gUtBbyqthdQXzfjJjvvjjP1f4pVe%2B7c%2FnnZhzaK%2FM6e1Yd9sC1ynZ0%2B3ivNcCRO6dg9w7WG%2BIyiwjMEUqrMgqCeWCJJhXvXJeUhDwOFdzS0WJ1IR7U4SxloZLk%2BXE9hz%2BJ%2FwF0Y99TA08P5oYPH8hMnBp3hrsh9sSJ8gp%2FI6nkPo2iUCUyckha2BsWA%2BI%2B3Ji2B5MxP06lp1E8WB1guDkKaW9Om7Lo8%2BVe3InybbnpONB8ddGk5qRQTK0XzNSN3amFPzfohRqlyV5sJoAXucvHuq%2F7%2BG3Rz0VCt1OfqJwR7Y1TqvtlR1kGaT26g%2FST37%2FtN3MEiXr1xyVFSXabudABHvz1pPH1e607ZrbnYP0DZCg5qbRN%2F6Pzzf%2Bh899rpr75qof5f6oPX3k%2FdILordj3fmCJ27MtnX3ITfndgCL%2F7pKzdSkCoEsB9RxZ0L4IH6BO1PEDlGa5WIlSEeAAJa3WqqE1c3lyoTLAR35cDvaL3weKQZ%2F4eOJsR941hr75PPRsU9ciY2jaPMZRFuegflwBiFsZ6FgUzgqFHOWZXTa%2Bw%2FPxT36vNyTbvKTOUnKb64BUfDXicPpSooli8dD3PwSzCD3y76oifj54oYldIJ15koQvWLX%2Fskc0f04iR2qvGCpIxsVR12pKepfjATFtY03gMfYW1j3D2v6KSWb1CrFOn%2FYy1qAst6f5T8OLlIO72gfvad77pjuI3pBoIv8oQSGq%2FCm3fqF33P%2B9h%2Bh9BiWEpLx1HOteFCHHJ%2FoBYt1dAleu1ZmkGh%2B79YpGC3J6P7fJPdOIjJF5MS688VO3JklRGdC6MSdmYUFWCo65XRNlZFFUsb6dyWWE3v032YPvVNEyHrpJWVNF1DbCysVVG3VZxusP2Cwtpys0o0HF0%2FHxq6etseuzNtjsPLG5qdtcVeKlTeNcQSxg%2BAZrPztr%2B0mbjLfHSbusd6%2BdU4vUq1D1nmT9d6COZUgN0v7a8uJFdbZOqX4RLxazZZuoiiDbI18EYWOK6ZRR0UPRORVg1XFDSpu6BrKWSHfFSuM4wITXsYWlpxr2UDtYn04GL5VbXs12%2FPdnId2Q9K%2F7UXFCWsQN2RQl9tEUEsbAUMeLlQs3%2BKDD7QPFUL6sVySlGjzLT2E4QeWutF43l9fMDE2pKNjA1Z0dFDHxgataCLejrCVftd%2BvweWpCxw668l6K%2F7t2vV%2BMNr6KF98gE%2FWAZXe4DttvQCd3cQxHIX3rR3%2FsEv5P%2F%2Bfzghpce6IRyX6LVFrW3RbZcS286zC%2FLtusXPr0WnHFl%2Bw5%2Bbg9AVYOWJW1OiMAslVFapat%2BdWaiK9eeWt%2FD3CvLu5HWZGlybCFo5SVckBxytYXp55NPnVGQK2%2FPnRiNnTjlRnPgm8KucsJEo7iHiMhFV4xMj9tjkmB6bGrHGkFM3Go%2F6VsAAtkGczAYwxrAiOObm%2FFw6uM9kfxJKfO1cU0spKlnEFGKGqDuxyCzfGpNxXm5bb7DYrEp7LkaqC1UcLGvzwZrbvJtyGosbXuMOtkzBa0LgMF%2FnYh12CJynWrCCHZz3PBCCzmkEs0JULadlXLzOxmta2CLWoYpezEX9QEib1YLgYUYpKuu7K2PjwCIRbAjXRs1jVC%2F1IFyWiroiYBY2uEK1ZfsXMihlGcOboqNDVvz8bKT%2F3KwekP35U9bAQB%2BsRlluKbj4EbeoPA5Wizey1xgr%2FnOHf%2BsZW0kyE7ZdmkG7gzlducCSAtg3sVL66%2F%2FdT%2Bd%2F6osnpPTY8Ymeb911C11H2NrWXSc9oVvsOq5OPFeq6woKRcNlCZGTYBXskVyH%2BTzM3QWCV8b8Xlmqq5QayL3D%2FF1N5u9SqrzOcmKH%2F6M9mm%2FQELz4Cz%2BsYp%2B6PBR75nQ8dvaUJ6sQIMJSLeAEv4A8ogWcCScjth6zI2pU9hFLjcGFKRZdp75lt8C1F0t9jPSCo2l75wTU%2BTa58g7WaJP5OGXkvrgmEWigs7BygtJTcqKyvTzkC1fpVh7CVXj3tld%2F45bTfHOx1fjWu7XGu8uqCT6yXKxBvqK%2Fly%2BR%2B7KHc3H7cWTVjyQxtwOT4KhP1X5fXZLgFtz8skuIr0WfaxFF2eN8vb2X%2B93bR88kBl64aI1%2B4qI18okrkZGPXo6O2rYZwZkJK1r4q1d0ysZJDqYEMEm9VPEQ4ALpCG5HTfNDTbqvlSdPBqvHSzkzbb7%2FE%2F8y%2F9%2F8%2BQeqgpXfxZtw7Ed9kJ45Sktv2zUZjNV2mbDO%2FW7Lzhe%2FHXHElWlLXJaoqOK7MyF4MkcnLkyInFf28%2FLgwvRfU3L8PLwsHrfTEVhO7CAjordeK%2FMwESyNEj2VUFE7piKXZxOJS2ftwWfOWf72kbORgXhUT%2BMUOIOf3iwkbdbfy1W69k9UnbkesfAefOu1n61cWe9aIaRTTB0eD7gc5feDQ26WKqZarpsKPBqVUt2r1htGRK%2Fgr8OG%2BovY48pcFV3HLoNNCauolzBDVkoVmvV3b0VaH6znW2%2BsqdZJyrN63CF69aoaujrZP%2FTieXvo2fPW4IVpmfvTQ45WQ7bWg8bzhrBEz%2FBAQo8N9lljQ%2F3Y4z6eE%2B9Ax%2BUZuEHlQsqPHu1E9vr7x7sd61i8r%2BiJsQLRE0%2BAycJz9s6P%2FXzub2JVhU4VlmM94oPC1uZL0cM%2B4Hvm7u4jeB3h0%2FK33ZZgvalrWPKnXIAFhzkIiJ6ubLszJUgFrkwkoZdLVbeMEmPlslRXwSauTZQTk045EfXgDtpxYXj9woJKzA2q%2FpnJxMDZGbf%2F1ESkf24yOjE7ZibmJu3xU5PWOFyXUg1lBCeenWLOOHnhuZ0al8H9%2FaUYHPavYb8dZyTQREuknFhvvlvS38taa5aBCxP1E42prqVNeWXLraylvfLapltJ5k0ZKRk17eqasXQV83I119G1KAoJY7UtVOK3Gq4XqTetaDOfzrortyCui8p5LZgP7JXW75fSo77OQuHwvrNjqm9qtL9vftbrm%2BiPxB0UD8f8Hza7DysIxIcTemB%2BSg9hKShZLQObNYTI3q7gGATQKFkLUVyj4gL1A2zabtJHPbT2%2B46tJ7pdm3Is8tgPYhGjWlzp2ULVvPmj%2FyL%2Fo3%2FwdX9MnpiozU6PHLboSQ7dPUEpXYEq29bd%2FVyeWPeuoivFthsToCXRvC1yu92ZhYrjuzrh3qzk4OIseaqSzZ6s3JHH%2FIk8jW%2BPokrKyAtX4mMfOWOPPns2Onr2lDWB4INT%2FXGFzTqFE9BcNOIvugpRQ1HnYN05uS9X253VwTv7%2FUdgHtsJZ1c3ygllZwFRIxXz%2FYVEJTE4uG95%2BVsrbun6Had8Y9mUvr%2FUKF%2B%2FbZVchH1E3YaDABDHaSIxURYUxcKi%2FTHl1vqVO15UbnpAeVPXlQex67goj8NVeZzjVrwINgoT2DUs9DvUXug3hjUzGpixjMaUDXepdW4q3vfRC5HEM%2BfsxLNnrcQLF6IJhJBOgn0QCKURGOXB0yAraPgL%2FvoLAY%2FhvswfP97tyMfhfQQvaEHg2pQcPbjOYel95z%2F9O7kfR6qCRMAf%2BVE%2BHlT02iFZentbd%2F6VQ1sQg%2BTY7cdwZyJYBa5MWHft6MyKBKMUxNKTFIWSriASs1yoYo6vDHcmxC6NObxyQVWkfmY67V%2BBnLjOeNzOPEHv7xYgPxLz0iVll0oq0sIJZ9BV9oXzfSPPnrMmPnZZT7xwPjb5scvWOCpinsJr59Czc%2F5eG%2By1RNdJ%2Fhy2Ti7dY5I43JETiMv2umgSMSCxGb47U1Jx%2FBOM66ky1kjLthyTQbRlVlapxkk4h2WMZK4uh3fkXMvK3V1rld677ZXfWHXK33u7Wvn2bV7sPWbvf%2Bjtzz%2BvYs%2BhwMEnnh2J%2FsBVJ%2FbcXCyuPWvGtfUMovhnEPE6g76bRqDUeNTWE7gQG4%2F6930PhKRZdC68gvQLhJniS4KEe1kxsT0%2Fed%2FjPtyxeM9XPlD0EFjoz%2BdVKg21%2BZd%2BKveSuqZaSFU41CCmJ92Pnc87DNG7v3W3S9j89IRu0du%2BX6lp5N55CFDRlWx7HwSseEg5EJGD2FVg0dVcv6C0zGXkIHabED2bASuHNU6e5Of6IvfpTyt9pYS6Xij41YgMDVl2a9BrekOOZQ%2FGbG%2F47Gxk%2FNyMPX52xho%2FM2uNw70kSeUSWNCploK9XFlLYEp70VX%2FBCKLsD7G7dBPMhL2jUhL%2F4ThrxYiealYGduPupQoS0Rb4uSCMnlbeZPfKniFraKXxyrZhbpjFRGVUYa3qWJ5dtnDPl2xq2trrfr76Wp9445qvJ3yr75PTEmox%2Bipo3yrRnpMJIY55WenVOTM%2BKg9P9eKW441ijiZEQSYjiKJcASCN4KI4ZHpETU8MWoNoz7ryMiAuEDbkaDBHLNsmBvsSpMI7j%2B8cszhj8uuiM0O2u0vDc7PxpSqTbX2%2FN%2FJfWpx0Q9gOXGuTWnZkxS93dbdtsj5ZcLuE8SyXV0lmJSHqxMBKhURuxxETaIz8xC0oghezUPUJlydmL%2BDpYe9WylJlRX8XQJWCg1VQcK5WHcsGH2Up4KDf5f%2BzGeUNXZHWXMvKPvUOWUNRQeG%2Blp6CsWmpmDgTSNHbApuy4mxYWsM%2BVRjyKcaGx3SY1iXDvN0QQ4Whq1U4Oj3S08FV8wid52r5%2F27Mfc6%2FsM50cjvoRNtGaxzhkATWG%2By2jUW%2BcQeEZfIn8vfSbmlD1ad0p2UKb2%2FDi9H3qkiVQAOS920YlZDOVbTqVZay65yWqvKcVMISAk8JifyZLRXdxzz3z%2BUGxpDGkTVHuzD6S1hYy7QiXl9I32RxPk5a%2BAC0iEuz0f6LyA9AvPOsPjEFSq5oXCHqrY7VKrIiDvezyn0q8c8PAjmcMbjDtbtNIVdoifmX7sijyqivvHiJ38y95dOcuDTkxK97nm7LlflrlJi7auFtitT5vOsQPgcRzXhzqz4c3hVD65MP7lcXJiw6PAYwSuIyJR5OxE8cXlWSu2%2FIyG9goLRjNA85jPCfr7%2BM%2BKG%2FIyyP7Wg7I9fmrI%2FNteKIGzgjPLsMxC700iOPoPf3Rnb1qN4jFBypBto3EfaAU4akkvXrsHoC5y4hoLbky40fLgnF7lCDvLodjYInVpDO9bQJmx6LZPzMm%2Fe8apv3vCqby%2B1qu%2B9X61egyXnt%2Fb%2B83Ddz%2B%2BnO%2FiaRyNwr2u%2B454MxiRUDJWAIi%2B9OBR%2F%2Fqwbf%2F5ytO8T5yPxU1IYwXhnMIZP42WyP4OexFxgx%2FpDPqB%2FX8qrPeR2WGPz%2FoInByIFEPwKWHCx5%2FFTu%2Fmpv5b96yfVtSkNelzR67bupChtIGxaglW2S4kFcxXd83ddj%2BsNXc8HQgdxwx7zdDJ%2FV6p4VSSZIzAFEZo1JYJYLpedClZHEOtP3JyVdoSmWHgn0rf8aL%2B5E%2FGu4KoYW%2BGSskZgn5y%2BomLnFvoTLy5EE0gK7n%2FhvJWYGMHKBZaNlQyQY2cwP6cl185M4r1BUnBwQuhEYh5twx%2Ft5CLyG8zXyfuDcY4wb7iCkOwt%2B6ajqtWaKVTqplCqecVqXRVajs7CVbkJY3UT5xh%2Fj1zU%2FI2NZv3G7Ujj1p1SvXRdNU7yieZoO68nvs2GCz96YVhFP3JxOHpx0o1OwXuB6b1ZY3mnUOluFmN9FquPTw70qWGkRowgSGtkMKGGYzGNMCNUhOlUhwmqwgTRx482Lh8O5EOC1%2FVFQW41xq5ulaueM%2FSf5%2BX3KUdxYr0JjyN6HSHbFYQCHpK1%2FwDLb1fAiovamVXfXSlzdSgOXYRFJ48LEEERNXneFzy4Nst1x09VkAjNWgtzePh7Lue7NJmS0BO%2F8e2D0BC7iHpJRbEWT%2FTqmaHI1IgXnZ9GZZRxPXZqLDJ2ekqNnhqzIHiIggvKgckPCXsRPPzgd0diSlTm483TPSqfg51gOicCET5oGOqXaBRoVrrboisVK6aYzJlcKuNkN3JeLplVuVLZSJRm0fOsIqICi1HjQRDtSiprt9LlYiu9pBxEynVK8T1qa%2Fi%2BoydgIwjGfm5Y2VNxRIhOjdlDifqgp21EHLujKKg2amM%2BEA7SsVOTamxm3Bo7NR7BpseGBpAoL%2FVDO7VEZS9u0P3cDjZuH%2FCJ7Q%2FpyhctVLzG6H%2BRnznJgieNfRTR61h3gdjJ1aw%2Ffydzdx7u%2B3UAu56%2F536QVLtr%2Fg5XEL6lF8zX%2BdaezN1VC3BdliRCs%2BaUC5KOACsPKzdV2yucn8gckf2M2ZP8GghedOh5FT9zBe6deTf%2B7Ew0PjmJH3a%2Fnh2Km1kk%2FM4OJ6zZeFyJwEkgivywRzAS5UcupZ%2FkihYbil9picjcZ37dYULb30lEfhcSpCLjX1xCsvaYFHDGXB3m6aQ6ijHZdMFs3V51MjdXvcytVXfrgzWTSW65Rc%2FSjRg2pNA1nLpq9MMqLGWVl%2B5THqpeBJ95gq%2BuD7N7evyzt70e9eeR41BRVmJGxQo11RfxMHsd8fpQ5K1vZsIeefaMnrg0b09enrMmLp22sfqHXAz6c4GT%2Fpygf4G4j1SI%2FY3XPbDtCmJpQvDUSL9e1381d7nHee95eAcVvUDQOuHV2%2Ffb7sxO8Mq2ELbn7by2hYc5vM78Hay4KgJUJC1B3JlVicgsibUngSxlVUWSOcQO83f4WydCs4oIzSKWqmdKwp79elwvsD%2F9vEp88spg%2F%2FNXnf5PPjeQ%2BMh5JOxqcxa1Ec%2F4cxkG97HCATZxXQYrj%2B9sJ3Pts8ClGaw5F7g0g71RWzhpJaFWSc8zSTyfRDJ56rs3WunvLTqbb7xv0t%2B%2BUU%2BjJqxUUeEt5AQujKmRlz6KRYwvWFMfuRCZ%2FuTV6NTUmAVXqJqHqyMos4cpAETwojSaD%2Bve%2BcUn5P7sVk0%2FwliKk7so2L2u%2F3Lu2ZPeTfsVPaHQ7cbsDlbpdmc%2BQBSDOT2Zv4PrEnN1cFtWggAVmbcrI1hFglT8eb0qBE%2BiNMuuL4S4X8Fir5U8rLxksl2V4qRTfzqOX%2BbsbH%2FObgSum7qyLsz1D1%2B6YI1enNdjF%2Bf06AWsbjA2gKtVjTwmgzJhymCT%2B7I4qz9hL1tQ0T7Y9%2B5t5zwQuDG1L3JifYnAyX3xPCDtANVSLEuW5anAfZ8t1UwGLs2tUk1lyhVvaz2jcrfXW4X3l9384loz%2F0FBFRB5LEm%2FvJGAmsJyWC%2FMJUYuzrqj5xfs0SsL0REI3%2BRwv5keSFgzI%2F1qGiXRZvrj%2FgLHqADjX1R2imbLBWRHCLsDbA5Itkv0UMQchoeLVSzqWG9yVf%2BV3IsH%2FLCee%2FneohcsER9cvd7rzlRe263ZcXF2vS6Ya9sWx%2Fbad5VSDcJWguDVdFVSEnzXprgx8bwfnYk5vFIgeGWJ0JS5uwa2Vbg38XkSvcbb8ROwkUweWYir6Nw5bOOD0fkJp39iKDZzaszMYm5iZmZMz0yP2pisl%2FJgqFKhpHKKXypMXJqdsmDBkj6BC%2FN45u32w3K3u0iErrNUTxCo4o%2FLINXA3xsPRXl1JpnzCqmMV9jYMoXVjFPcyJpCOmuV1zNOdb1gVTcKtQrynUTw9l6MdT%2FHydc8FQTO4ffRN68GpscS%2FTNTXv%2FsuN1%2FetwanpnQ47NjanxhCnN%2FE9Y4LC9EhKIgNlYXwbjDBgvQyP12dPPO%2FoC%2Fre0Bb2CQNB1XO0gfKum%2FkpXiECf%2B9nDR63ZjdgTMf67Lndntzul2fbZdnQgMamG9rqrvzoQFJ5aeuDDFioN7s9oJWJEKKzm8hgErPT%2BmNEo3xX%2FgkoqfPTUYf27KjZ%2B%2FGI2PxK05FOY9g1mKM%2F196swA7sds%2FwcouUiSX9fJRUJekp9uIPX7j28V8oNg3i16InTBAqxSHzO4EJNglXU8XsfzwR4pCHc3nMqNVa96e8WtvrfkVj%2FYsKrJbK0Bh1Fro6qcwS3VWgxEk8nkB%2BmPcLw2gtqz0b6qikZQbH1kTEWmx%2Fv6r87bQ88sWEMXT%2Bmhy6cjQ%2FNjlhRtWAASREArpENgr2W1ke1lsjqrifjpFPtDtx3E4qGYedNB5PHogC5C9OR7TvztQaInre621D7szuwI4ofm99pW373zd5JoXkW0pp9r51dYCfLyasoXw2LJrWTxGn%2F%2BDsnm5Tw2uIw4f3fsY2zXvAF%2BiPFPYd7uo%2BcTiY9eiPZ%2F7IKdWJi1J7Co50VcDF2EW%2B8Srj5x35zHkUvJJVmUprtMWGcZn2Nv2L4OIEg9CFIQgpuInBRBaC%2Fpo2r4Yw4phHcRvbKIdef8LWI5d6%2FdjjSv3Wq0vv9uqfnOe6r52gmuYrEvVnzRoROQQLHpH1SJj18YSLxwNZr4yLydeGbBjGpbX0Sx7IsYrBfxA7uEQme4708ZdILB5HcoJdG63Z%2F3Od6dgY4yeBKD0cLJvzmUUHm4Ns8degOP4AvuJ3r3m5Bvz9u13Zk7bsv7zfP5z90zfyerJPgBK4HYSeFoXRX3pbg0Kw3Xj9yUCisQwAorrBxBz%2B%2F9FTZGeHQAibYWrjSHRlV0ajQRvzDlDp%2Bbt0cuLUSHL83pEcw7DEctdQo%2FJamDibqYHibcsQ9y7UTs5AqzbdF1lQh7IhFmezfikV%2FhC52%2FnEpnzk7GtWwIOvGLPhfQZqxPhzw7V29hXCdLVbNRrXkb5Ya1Uaq3kjfet5yVzYjz7p1c605dOSe5isUjc%2BQbD4NA5MUXVfwjp1XsytRA%2FPKEGz81HxkcSVgLiJI%2BPdhnLwz1q9ODcX0avzhJeG%2FP%2FbUXyTWoVyuelqCEH8QQrwoe47bzw5Ql3VyIXjyOMnkofg7REyE98bd7RW%2FHuttlycGdaSB4luzvmdvrjtRsiyEm7iVYRVZEQB6eF%2BTbtfPvSmLJSd4d%2FoZ5PQgh0hHE2msnnDexXy74V9Gcvzu%2B4YVJOZVYGMGG9eumB73E7HgkMT0peXZqYmrSnpwdVROz43pieADuFYWamFpqYiqZXJcNKXpBdYr2FhTW7a6c0uuiF6QfYK4Nq5EbzLnp9srkQbmwNP6GzUh05la5ZraSWTeXzqnseraVXcl6uUK%2BL7%2BczbvpNeW%2BV1BuO%2B0gTMv3HN%2FoDcc3b8%2Brj%2BLC9PQU5tUTbmJqQk%2BfmoxMybz69Lienh3R01hXHqs%2BYE49WH5LatfKgrmSDyupQe05dX9lku3ltyB4IoBNjTQalFhrQvAkl%2FapuHVErxOFtiNq29bcdrJ5x6rbidCUMmJShUWE8H7uzBpKi5UC9yXSESCAvtj583kQxark31UgcAVEZmItvEp7SSARPCacH%2BPwguANPHsGC23OxIfOzNlDlzGHgITZ8YG4NTswgHw75NrhSnIGUWQSqILIMakd6NfDbNfG9H8890aP7bhJe1%2FwhL6MwaBGphSEDu5XIXjBnJ1sFjbPrGM5q9yNdVNdTrqVG2tO9dZGrbJeUbVVrDCOyFbv2jU%2FyvNEV7E4xuHIr34wAd9aQwK8Jfl%2FWSTBXxhX0XOjfaNn5%2BzRiwsaUaCR0Qun7NG%2BCHL8sBIEXj%2FtR1Jr1P80qIgUVHtpR09r7E0Mguf%2FVnGVKms6NrCyfAOpCvLep%2BImotedd9fJLwqCVcS6C6qrPNCN2T45uJW6riOx3HdbSjkxP0qz7c6UtAR5XlyYUk%2BzUHEryDDHKgp%2BeoJUlK8w%2F643xhOCVPquTqixKxdiYx%2B9FB974bwae%2FGcPRaJ6AWctU9bllrALwJ7LZPaEom5u%2BDzfibLT4ToaRSDRvqBrH4ggSpBSoK4Mxfx21jC4yUYg4tILF6%2BseqW39mIOEvXc87ylnLeeUW5r%2B0OTjkRLe6NEcijeEQC3XPv1g9fVf0vPDeY%2BOhZp%2F9jlyOJF8%2Fa%2Ff391gwqBUmQy2k%2F2EXy%2FhQE0OCC1cJ6lBJshg2ih4V0fd9Mw7awagc8HfqvZqUSy1NxE9GTSfnueTypqA2RQ%2B3M7mTbToHoIAlXKq74QthVXQXJ5iJocGdC4MSdiXkOeU7m86pwa%2Fp%2Fz0PwyhLMAgtQBLBdMLoTFPBUQD0hjZCrRIkQs0caWDRzQEVQAyX6zIXYxNUFe%2FLKXGTy8llr4uIpa3Js2K8Kjw2WnX%2BVKDl3%2FlXjwW69eerfHbTVHvP1linVUBcTsZYFFEUo1Juq0GggFUHpDWPB2nP1RtRy16sW5u9yicb111Meyq%2B4r6WxMOtrrJ5ysIHBVx8CgQ9FWT%2BzEI0PDKgJEb8BVEeKR%2FRsog9bDNVepOQZFr5F1MoI1HMY2xCEL4IV5CUdp%2FG0pCsIZ21ejWJZk%2B0KEoHYBaWUPjy%2Fd89zWGG4geRbicKsSoK5iJ0EpoiYiTuzWPdLi23ik8b0AAALnklEQVSXEytWHDynKiiy60dxSg5eHmXF8F3MUzqEUf%2BQj7QkCnPMUn0TMyo%2BPtQXnxk0fdOo%2FYcamdOnxv%2F%2F9s4lNq6zDMPnn5udxo5jexwn40vTgiiW0nTDAokNEgs2lViygg0rJFgisUIVElsQ26pStyxYsGHBZQUSG4QoSqKogqZpXIyc2OPGnpnj25nD%2B%2F1nTuI4adRFHI%2Fne0YajdNK4%2Fmf77dff%2FfqfGu2cmlhLlxSbsBGH1m%2BruyzU74u5gSePw3%2B6DcfTrErPmExQd68uhjK0X%2FQ12FPO%2Bw2Nz%2Frbz7YzjcVet%2FUXrtN5e7a%2Bou4reloeubtatZv915J292NJLvV1ciwQvDKMOYwn%2Frl3ja%2B22kRCAp91rWtpG5hT8v7LU72682mxoBeCNMzk8nM%2FFQyPT1VmZmdUG6%2Bks8eHCZzcvE0UCKfrYYwq4pQ5f7C7qi0KxSi96e6mmkHvXfHi1WeU6UZm83lvdlsTPPoynBmOV0lVmfGXjzL6cXqzDhOTPvvejY%2Fc5C%2Fs5ARPUov%2BUei1Upe%2BZLydgtT4%2BcX5%2FsTi83q%2BYVmZeriherlC5P5lemJcFnNqHqtXK7Z9uckTn0o8nVBX3%2FRwbefd65hkoMQc3bFNJUQ4iJXhXe2P1nP7ttzVWPD7q3n66v3s%2FudNLPVKt08r3TH9Kz00253LunduZPkk5NJPhA8O135fMmW5dtB4CkCFva0opeqbTsZn0yqrSXlABvnz11dPDy%2FPF%2BbaM1UJpYuhYlWszaZH%2FZbVXmAasG5Uq3llrNvqYjlrVHiaqKnmYCDYpXjvXeDAhUFa2KhiomjNSqqNFveXGw2t40Ien0czrTqTAtnWigzVmgqnLkjz04C2LNxYlsSvWQ96a4XfU7D9OtvlOz6uWe5qmkP89eSqevLY1Mrr1cvfGUhXFxZrl8YH%2Bu36o2woO3PC2P1pGU5vHpVQ6GL3h61Hij4aeXNRYXXF2xyfcbHGD6L21Do2HoQWxDUimCjwz78JFu7vZqv3b6brf179WDt1urhfw90d%2Fcb2hiqxvLueHKoIpVMbQhl0dUTAwtdXCYOeZYIHC0ss00oobGY1N78qlZ%2BLU00Lrey8evLjcYbC7XGeD1frH%2B7%2FedR%2Ff2s8GZt9dFmhKc9uyfCnLv7YS%2Fm6RS2jK%2FF7rvo0ZXhTLUepFacUk5XsXYEjRmL4cxdvWrhq%2F1VbdVwPE6WwKOllvo2lS%2Frkj%2BcS%2BpvLZ%2Bbufb64czKa2PTK0vJzMqr1dnmVAxhWp%2Bdxgxpr12ivrv47xe44eD0xa78BEc9MRMs5en0DNqGYJsQkv7meju%2Ff%2FPjbO3GR9n%2F%2FvXx4dqNewdr%2F7gVp6xQVXyyd5Z3f%2FkEgvb%2BVd88l1Rnl5LqO%2B%2FGwQv2OP2f2BNiYZ6egjNHcnoDj06%2FBB4Xsuj%2F669fK06JYcztjoZGa3amFadYONPCnB3bbl702%2FUe7mXRC7R2BPPsbP%2Fdugkf%2BbsTMuNTbxsby8eaSX1mPqlrTU19YuZco3Uxm16%2BVGm%2BeqXWXJpP5q7qdbGp%2BH0uj04xfImeTW8v5%2FfZFoQX9zj9H6FyPU%2FZZG7LMRVt0OLWoGeeP3r9dL3%2F4IM7%2FY0bHx1s3L7b37h5d3%2Fjn%2F%2BxjQmj%2B4vgxRmadzqDBI5Hbk7%2Fp%2FUEIZrofaj3f7JwpazmHPTeaXZmarm7YgOCClbSkJbVmRbO7OyqTcFCmVaRuZv1ynYEC2lqunzPxolpfYqFM8nfnaAxeWsIQAACEHg%2BAavevBk9vWcIn%2FXeyZOL3p0mq8irk7gNwpmd3X6qvF4sUIk5O8vrpYfy%2BIp2hIeHhZdX0TogxYXMZR7pvx64aBCAAAQgMPwEQv7H%2BgexSOVIsUrsvdMoMRO56K3ZDjx9HUOWthlBXl23aE%2Fo9SxnFwtZHrcjtCV6lr%2FTOiDz7sjfDf894BNCAAIQcEHAwpt%2F10mLnjx5fHtZsr%2FTzbXYtfDqzNOL1ZnWXB69Pb3uqS%2FPvDkLXSqcWbYjqIAltXFi8vJS5e%2BsYMXWpvCAAAQgAAEIDAUBE72%2FlaFN9d7txiWulrOTdxd78CRk1nqwI%2FEzwfusI8Gzf%2B%2FpqWWvZTtCmb9T%2F12P%2FN1Q2JYPAQEIQAACxwiY6P3Feu862nUnsYseXadoS5DwFeFMeXypPD8rZLECll5PhSuWvzNPL7WB0gpzblnf3VrSU%2F7OQpo8IAABCEAAAkNHIKS%2Fb%2FwhhjC7IbVKy9h4nkrcinYEq75MbQWQxC7tWjizm8WwZ1dPK1Tp7CUp64CGzq58IAhAAAIQeAaB8Olvar%2BNBSsWziw8vafCmdqCEP%2Bb5e9Undmz7Qg7mTy8tsRRr1tb0bsjf8cVgwAEIACBoSYQbr5Xf1%2F5O01UKTy9MpxpLQjm2SmEqZBm6KUKZ1rzubaap%2Bbp2Uimh1mSKn9nBSu2L4wHBCAAAQhAYKgJhL%2F%2Bsvbr4%2BFMq9aMnp28PpuuYqHMWKkpAdwatCNonJj13pG%2FG2rz8uEgAAEIQOAogfC7n9d%2BUVZnxtDmYLyYqjFTm66inJ3yeNaukOz29ov%2Bu3saL6Y30RoWHhCAAAQgAIGzQyC8%2F5PKT82Di%2BFMTVZRCDN%2BbdWZajzX1JVYnRlDn5a%2Fa7ej4DF49%2BzYmE8KAQhAAAIDAuFXP6z8eMfaE1SgEkVP7Qid3mFq%2BTtrR7BxYvvWqqD%2BOy3JNMEjf8f1gQAEIACBM0kg%2FOx71R9Ye4IET4OjM23FVAN6LGhR%2Fk5CuKdVQLnyd%2FTfnUn78qEhAAEIQOAIgfCj71S%2F21NF5raWvZb5O5u8Qv8d9wQCEIAABEaNQPj%2Bt6pvy8OzXXhpVxNYynFi9N%2BNmqk5DwQgAAEIhLe%2FnnwzVWhzW5WZNmJsf7D%2FjvwdlwMCEIAABEaNQPjGteRr6V7Sjfk77cJbU%2FGKDkn%2F3ahZmvNAAAIQgEASrr%2BWvNFVL94DW%2F7K%2FjuuBAQgAAEIjDCBsNRMWp1Mnh7zM0fYzBwNAhCAAASMQGg2k0n233EZIAABCEDAA4Hg4ZCcEQIQgAAEIBA9PTBAAAIQgAAEvBBA9LxYmnNCAAIQgACeHncAAhCAAAT8EMDT82NrTgoBCEDAPQFEz%2F0VAAAEIAABPwQQPT%2B25qQQgAAE3BNA9NxfAQBAAAIQ8EMA0fNja04KAQhAwD0BRM%2F9FQAABCAAAT8EED0%2FtuakEIAABNwTQPTcXwEAQAACEPBDANHzY2tOCgEIQMA9AUTP%2FRUAAAQgAAE%2FBBA9P7bmpBCAAATcE0D03F8BAEAAAhDwQwDR82NrTgoBCEDAPQFEz%2F0VAAAEIAABPwQQPT%2B25qQQgAAE3BNA9NxfAQBAAAIQ8EMA0fNja04KAQhAwD0BRM%2F9FQAABCAAAT8EED0%2FtuakEIAABNwTQPTcXwEAQAACEPBDANHzY2tOCgEIQMA9AUTP%2FRUAAAQgAAE%2FBBA9P7bmpBCAAATcE0D03F8BAEAAAhDwQwDR82NrTgoBCEDAPQFEz%2F0VAAAEIAABPwQQPT%2B25qQQgAAE3BNA9NxfAQBAAAIQ8EMA0fNja04KAQhAwD0BRM%2F9FQAABCAAAT8EED0%2FtuakEIAABNwTQPTcXwEAQAACEPBDANHzY2tOCgEIQMA9AUTP%2FRUAAAQgAAE%2FBBA9P7bmpBCAAATcE0D03F8BAEAAAhDwQwDR82NrTgoBCEDAPQFEz%2F0VAAAEIAABPwQQPT%2B25qQQgAAE3BNA9NxfAQBAAAIQ8EMA0fNja04KAQhAwD0BRM%2F9FQAABCAAAT8EED0%2FtuakEIAABNwTQPTcXwEAQAACEPBDANHzY2tOCgEIQMA9AUTP%2FRUAAAQgAAE%2FBBA9P7bmpBCAAATcE0D03F8BAEAAAhDwQwDR82NrTgoBCEDAPQFEz%2F0VAAAEIAABPwQQPT%2B25qQQgAAE3BNA9NxfAQBAAAIQ8EMA0fNja04KAQhAwD0BRM%2F9FQAABCAAAT8E%2Fg99EShaIUEKdgAAAABJRU5ErkJggg%3D%3D diff --git a/pbserver.py b/pbserver.py index 4185aa0..d1fcc32 100755 --- a/pbserver.py +++ b/pbserver.py @@ -1,19 +1,19 @@ #!/usr/bin/python2.7 from bottle import route, run, post, request, static_file -from pb.Gradient import Gradient -from pb.Imgrid import Imgrid -from pb.Breaker import Breaker -from pb.Pattern import Pattern -from pb.Generate import Generate -from pb.Imlandscape import Imlandscape +from Pb.Gradient import Gradient +from Pb.Imgrid import Imgrid +from Pb.Breaker import Breaker +from Pb.Pattern import Pattern +from Pb.Generate import Generate +from Pb.Imlandscape import Imlandscape -from pb.Config import AWS_SECRET_ACCESS_KEY, AWS_ACCESS_KEY_ID, BUCKET_NAME, BIN_IDENTIFY -import pb.lib.Utils as utils +from Config import AWS_SECRET_ACCESS_KEY, AWS_ACCESS_KEY_ID, BUCKET_NAME, BIN_IDENTIFY +from Pb import Pb +from Db import Db import os import sys -from pb.lib.Db import Db import sha from subprocess import call, Popen, PIPE @@ -61,13 +61,13 @@ def s3move(filename,objectname): def format_im_data(im, insert_url="NULL"): directory = hashdir(im.filename) dimensions = bin_identify(im.filepath) - size = utils.file_size(im.filepath) + size = Pb.file_size(im.filepath) objectname = "im/{}/{}".format(directory, im.filename) try: s3move(im.filepath, objectname) cleanup(im.filepath) db.insert_cmd( - date=im.now, + date=im._now, remote_addr=request.environ.get('REMOTE_ADDR', "NULL"), username=im.params.get('username', "NULL"), url=insert_url, |
