2011-07-11 22:24:26 +00:00
|
|
|
import ConfigParser
|
|
|
|
import boto.exception
|
|
|
|
import boto.s3.connection
|
|
|
|
import bunch
|
|
|
|
import itertools
|
|
|
|
import os
|
|
|
|
import random
|
|
|
|
import string
|
|
|
|
|
|
|
|
s3 = bunch.Bunch()
|
|
|
|
config = bunch.Bunch()
|
2013-07-25 21:13:34 +00:00
|
|
|
targets = bunch.Bunch()
|
2011-07-11 22:24:26 +00:00
|
|
|
|
|
|
|
# this will be assigned by setup()
|
|
|
|
prefix = None
|
|
|
|
|
2013-07-25 21:13:34 +00:00
|
|
|
calling_formats = dict(
|
|
|
|
ordinary=boto.s3.connection.OrdinaryCallingFormat(),
|
|
|
|
subdomain=boto.s3.connection.SubdomainCallingFormat(),
|
|
|
|
vhost=boto.s3.connection.VHostCallingFormat(),
|
|
|
|
)
|
|
|
|
|
2011-07-13 21:49:07 +00:00
|
|
|
def get_prefix():
|
|
|
|
assert prefix is not None
|
|
|
|
return prefix
|
2011-07-11 22:24:26 +00:00
|
|
|
|
|
|
|
def choose_bucket_prefix(template, max_len=30):
|
|
|
|
"""
|
|
|
|
Choose a prefix for our test buckets, so they're easy to identify.
|
|
|
|
|
|
|
|
Use template and feed it more and more random filler, until it's
|
|
|
|
as long as possible but still below max_len.
|
|
|
|
"""
|
|
|
|
rand = ''.join(
|
|
|
|
random.choice(string.ascii_lowercase + string.digits)
|
|
|
|
for c in range(255)
|
|
|
|
)
|
|
|
|
|
|
|
|
while rand:
|
|
|
|
s = template.format(random=rand)
|
|
|
|
if len(s) <= max_len:
|
|
|
|
return s
|
|
|
|
rand = rand[:-1]
|
|
|
|
|
|
|
|
raise RuntimeError(
|
|
|
|
'Bucket prefix template is impossible to fulfill: {template!r}'.format(
|
|
|
|
template=template,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def nuke_prefixed_buckets(prefix):
|
|
|
|
for name, conn in s3.items():
|
|
|
|
print 'Cleaning buckets from connection {name} prefix {prefix!r}.'.format(
|
|
|
|
name=name,
|
|
|
|
prefix=prefix,
|
|
|
|
)
|
|
|
|
for bucket in conn.get_all_buckets():
|
|
|
|
if bucket.name.startswith(prefix):
|
|
|
|
print 'Cleaning bucket {bucket}'.format(bucket=bucket)
|
|
|
|
try:
|
|
|
|
bucket.set_canned_acl('private')
|
|
|
|
for key in bucket.list():
|
|
|
|
print 'Cleaning bucket {bucket} key {key}'.format(
|
|
|
|
bucket=bucket,
|
|
|
|
key=key,
|
|
|
|
)
|
|
|
|
key.set_canned_acl('private')
|
|
|
|
key.delete()
|
|
|
|
bucket.delete()
|
|
|
|
except boto.exception.S3ResponseError as e:
|
|
|
|
if e.error_code != 'AccessDenied':
|
|
|
|
print 'GOT UNWANTED ERROR', e.error_code
|
|
|
|
raise
|
|
|
|
# seems like we're not the owner of the bucket; ignore
|
|
|
|
pass
|
|
|
|
|
|
|
|
print 'Done with cleanup of test buckets.'
|
|
|
|
|
|
|
|
|
2013-07-25 21:13:34 +00:00
|
|
|
class TargetConfig:
|
|
|
|
def __init__(self, cfg, section):
|
|
|
|
self.port = None
|
|
|
|
self.api_name = ''
|
|
|
|
self.is_master = False
|
|
|
|
self.is_secure = False
|
|
|
|
try:
|
|
|
|
self.api_name = cfg.get(section, 'api_name')
|
|
|
|
except (ConfigParser.NoSectionError, ConfigParser.NoOptionError):
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
self.port = cfg.getint(section, 'port')
|
|
|
|
except ConfigParser.NoOptionError:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
self.host=cfg.get(section, 'host')
|
|
|
|
except ConfigParser.NoOptionError:
|
|
|
|
raise RuntimeError(
|
|
|
|
'host not specified for section {s}'.format(s=section)
|
|
|
|
)
|
|
|
|
try:
|
|
|
|
self.is_secure=cfg.getboolean(section, 'is_secure')
|
|
|
|
except ConfigParser.NoOptionError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
raw_calling_format = cfg.get(section, 'calling_format')
|
|
|
|
except ConfigParser.NoOptionError:
|
|
|
|
raw_calling_format = 'ordinary'
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.calling_format = calling_formats[raw_calling_format]
|
|
|
|
except KeyError:
|
|
|
|
raise RuntimeError(
|
|
|
|
'calling_format unknown: %r' % raw_calling_format
|
|
|
|
)
|
|
|
|
|
|
|
|
class TargetConnection:
|
|
|
|
def __init__(self, conf, conn):
|
|
|
|
self.conf = conf
|
|
|
|
self.connection = conn
|
|
|
|
|
2013-07-25 23:43:19 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RegionsInfo:
|
|
|
|
def __init__(self):
|
|
|
|
self.m = bunch.Bunch()
|
|
|
|
self.master = None
|
|
|
|
self.secondaries = []
|
|
|
|
|
|
|
|
def add(self, name, region_config):
|
|
|
|
self.m[name] = region_config
|
|
|
|
if (region_config.is_master):
|
|
|
|
if not self.master is None:
|
|
|
|
raise RuntimeError(
|
|
|
|
'multiple regions defined as master'
|
|
|
|
)
|
|
|
|
self.master = region_config
|
|
|
|
else:
|
|
|
|
self.secondaries.append(region_config)
|
|
|
|
def get(self, name):
|
|
|
|
return self.m[name];
|
|
|
|
|
|
|
|
regions = RegionsInfo()
|
|
|
|
|
2011-10-13 20:35:56 +00:00
|
|
|
# nosetests --processes=N with N>1 is safe
|
|
|
|
_multiprocess_can_split_ = True
|
|
|
|
|
2011-07-11 22:24:26 +00:00
|
|
|
def setup():
|
|
|
|
|
|
|
|
cfg = ConfigParser.RawConfigParser()
|
|
|
|
try:
|
|
|
|
path = os.environ['S3TEST_CONF']
|
|
|
|
except KeyError:
|
|
|
|
raise RuntimeError(
|
|
|
|
'To run tests, point environment '
|
|
|
|
+ 'variable S3TEST_CONF to a config file.',
|
|
|
|
)
|
|
|
|
with file(path) as f:
|
|
|
|
cfg.readfp(f)
|
|
|
|
|
|
|
|
global prefix
|
2013-07-25 21:13:34 +00:00
|
|
|
global targets
|
|
|
|
|
2011-07-11 22:24:26 +00:00
|
|
|
try:
|
|
|
|
template = cfg.get('fixtures', 'bucket prefix')
|
|
|
|
except (ConfigParser.NoSectionError, ConfigParser.NoOptionError):
|
|
|
|
template = 'test-{random}-'
|
|
|
|
prefix = choose_bucket_prefix(template=template)
|
|
|
|
|
|
|
|
s3.clear()
|
|
|
|
config.clear()
|
2013-07-25 21:13:34 +00:00
|
|
|
|
2011-07-11 22:24:26 +00:00
|
|
|
for section in cfg.sections():
|
|
|
|
try:
|
|
|
|
(type_, name) = section.split(None, 1)
|
|
|
|
except ValueError:
|
|
|
|
continue
|
2013-07-25 21:13:34 +00:00
|
|
|
if type_ != 'region':
|
2011-07-11 22:24:26 +00:00
|
|
|
continue
|
2013-07-25 23:43:19 +00:00
|
|
|
regions.add(name, TargetConfig(cfg, section))
|
2011-07-11 22:24:26 +00:00
|
|
|
|
2013-07-25 21:13:34 +00:00
|
|
|
for section in cfg.sections():
|
2011-11-03 17:22:23 +00:00
|
|
|
try:
|
2013-07-25 21:13:34 +00:00
|
|
|
(type_, name) = section.split(None, 1)
|
|
|
|
except ValueError:
|
|
|
|
continue
|
|
|
|
if type_ != 's3':
|
|
|
|
continue
|
2011-11-03 17:22:23 +00:00
|
|
|
|
|
|
|
try:
|
2013-07-25 21:13:34 +00:00
|
|
|
region_name = cfg.get(section, 'region')
|
2013-07-25 23:43:19 +00:00
|
|
|
region_config = regions.get(region_name)
|
2013-07-25 21:13:34 +00:00
|
|
|
except ConfigParser.NoOptionError:
|
|
|
|
region_config = TargetConfig(cfg, section)
|
2011-11-03 17:22:23 +00:00
|
|
|
|
2011-07-11 22:24:26 +00:00
|
|
|
config[name] = bunch.Bunch()
|
|
|
|
for var in [
|
|
|
|
'user_id',
|
|
|
|
'display_name',
|
|
|
|
'email',
|
|
|
|
]:
|
|
|
|
try:
|
|
|
|
config[name][var] = cfg.get(section, var)
|
|
|
|
except ConfigParser.NoOptionError:
|
|
|
|
pass
|
|
|
|
conn = boto.s3.connection.S3Connection(
|
|
|
|
aws_access_key_id=cfg.get(section, 'access_key'),
|
|
|
|
aws_secret_access_key=cfg.get(section, 'secret_key'),
|
2013-07-25 21:13:34 +00:00
|
|
|
is_secure=region_config.is_secure,
|
|
|
|
port=region_config.port,
|
|
|
|
host=region_config.host,
|
2011-11-03 17:22:23 +00:00
|
|
|
# TODO test vhost calling format
|
2013-07-25 21:13:34 +00:00
|
|
|
calling_format=region_config.calling_format,
|
2011-07-11 22:24:26 +00:00
|
|
|
)
|
|
|
|
s3[name] = conn
|
2013-07-25 21:13:34 +00:00
|
|
|
targets[name] = TargetConnection(region_config, conn)
|
2011-07-11 22:24:26 +00:00
|
|
|
|
|
|
|
# WARNING! we actively delete all buckets we see with the prefix
|
|
|
|
# we've chosen! Choose your prefix with care, and don't reuse
|
|
|
|
# credentials!
|
|
|
|
|
|
|
|
# We also assume nobody else is going to use buckets with that
|
|
|
|
# prefix. This is racy but given enough randomness, should not
|
|
|
|
# really fail.
|
|
|
|
nuke_prefixed_buckets(prefix=prefix)
|
|
|
|
|
|
|
|
|
|
|
|
def teardown():
|
|
|
|
# remove our buckets here also, to avoid littering
|
|
|
|
nuke_prefixed_buckets(prefix=prefix)
|
|
|
|
|
|
|
|
|
|
|
|
bucket_counter = itertools.count(1)
|
|
|
|
|
|
|
|
|
|
|
|
def get_new_bucket_name():
|
|
|
|
"""
|
|
|
|
Get a bucket name that probably does not exist.
|
|
|
|
|
|
|
|
We make every attempt to use a unique random prefix, so if a
|
|
|
|
bucket by this name happens to exist, it's ok if tests give
|
|
|
|
false negatives.
|
|
|
|
"""
|
|
|
|
name = '{prefix}{num}'.format(
|
|
|
|
prefix=prefix,
|
|
|
|
num=next(bucket_counter),
|
|
|
|
)
|
|
|
|
return name
|
|
|
|
|
|
|
|
|
2013-07-25 21:13:34 +00:00
|
|
|
def get_new_bucket(target=None, name=None, headers=None):
|
2011-07-11 22:24:26 +00:00
|
|
|
"""
|
|
|
|
Get a bucket that exists and is empty.
|
|
|
|
|
|
|
|
Always recreates a bucket from scratch. This is useful to also
|
|
|
|
reset ACLs and such.
|
|
|
|
"""
|
2013-07-25 21:13:34 +00:00
|
|
|
if target is None:
|
|
|
|
target = targets.main
|
|
|
|
connection = target.connection
|
2013-07-24 20:23:24 +00:00
|
|
|
if name is None:
|
|
|
|
name = get_new_bucket_name()
|
2011-07-11 22:24:26 +00:00
|
|
|
# the only way for this to fail with a pre-existing bucket is if
|
|
|
|
# someone raced us between setup nuke_prefixed_buckets and here;
|
|
|
|
# ignore that as astronomically unlikely
|
2013-07-25 21:13:34 +00:00
|
|
|
bucket = connection.create_bucket(name, location=target.conf.api_name, headers=headers)
|
2011-07-11 22:24:26 +00:00
|
|
|
return bucket
|