dynamodb v2 no indexes
This commit is contained in:
		
							parent
							
								
									ac1d2f5ef4
								
							
						
					
					
						commit
						48cfd19fe6
					
				@ -3,6 +3,7 @@ logging.getLogger('boto').setLevel(logging.CRITICAL)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from .autoscaling import mock_autoscaling
 | 
					from .autoscaling import mock_autoscaling
 | 
				
			||||||
from .dynamodb import mock_dynamodb
 | 
					from .dynamodb import mock_dynamodb
 | 
				
			||||||
 | 
					from .dynamodb2 import mock_dynamodb2
 | 
				
			||||||
from .ec2 import mock_ec2
 | 
					from .ec2 import mock_ec2
 | 
				
			||||||
from .elb import mock_elb
 | 
					from .elb import mock_elb
 | 
				
			||||||
from .emr import mock_emr
 | 
					from .emr import mock_emr
 | 
				
			||||||
 | 
				
			|||||||
@ -1,5 +1,6 @@
 | 
				
			|||||||
from moto.autoscaling import autoscaling_backend
 | 
					from moto.autoscaling import autoscaling_backend
 | 
				
			||||||
from moto.dynamodb import dynamodb_backend
 | 
					from moto.dynamodb import dynamodb_backend
 | 
				
			||||||
 | 
					from moto.dynamodb2 import dynamodb_backend2
 | 
				
			||||||
from moto.ec2 import ec2_backend
 | 
					from moto.ec2 import ec2_backend
 | 
				
			||||||
from moto.elb import elb_backend
 | 
					from moto.elb import elb_backend
 | 
				
			||||||
from moto.emr import emr_backend
 | 
					from moto.emr import emr_backend
 | 
				
			||||||
@ -13,6 +14,7 @@ from moto.route53 import route53_backend
 | 
				
			|||||||
BACKENDS = {
 | 
					BACKENDS = {
 | 
				
			||||||
    'autoscaling': autoscaling_backend,
 | 
					    'autoscaling': autoscaling_backend,
 | 
				
			||||||
    'dynamodb': dynamodb_backend,
 | 
					    'dynamodb': dynamodb_backend,
 | 
				
			||||||
 | 
					    'dynamodb2': dynamodb_backend2,
 | 
				
			||||||
    'ec2': ec2_backend,
 | 
					    'ec2': ec2_backend,
 | 
				
			||||||
    'elb': elb_backend,
 | 
					    'elb': elb_backend,
 | 
				
			||||||
    'emr': emr_backend,
 | 
					    'emr': emr_backend,
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										2
									
								
								moto/dynamodb2/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								moto/dynamodb2/__init__.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					from .models import dynamodb_backend2
 | 
				
			||||||
 | 
					mock_dynamodb2 = dynamodb_backend2.decorator
 | 
				
			||||||
							
								
								
									
										20
									
								
								moto/dynamodb2/comparisons.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								moto/dynamodb2/comparisons.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,20 @@
 | 
				
			|||||||
 | 
					# TODO add tests for all of these
 | 
				
			||||||
 | 
					COMPARISON_FUNCS = {
 | 
				
			||||||
 | 
					    'EQ': lambda item_value, test_value: item_value == test_value,
 | 
				
			||||||
 | 
					    'NE': lambda item_value, test_value: item_value != test_value,
 | 
				
			||||||
 | 
					    'LE': lambda item_value, test_value: item_value <= test_value,
 | 
				
			||||||
 | 
					    'LT': lambda item_value, test_value: item_value < test_value,
 | 
				
			||||||
 | 
					    'GE': lambda item_value, test_value: item_value >= test_value,
 | 
				
			||||||
 | 
					    'GT': lambda item_value, test_value: item_value > test_value,
 | 
				
			||||||
 | 
					    'NULL': lambda item_value: item_value is None,
 | 
				
			||||||
 | 
					    'NOT_NULL': lambda item_value: item_value is not None,
 | 
				
			||||||
 | 
					    'CONTAINS': lambda item_value, test_value: test_value in item_value,
 | 
				
			||||||
 | 
					    'NOT_CONTAINS': lambda item_value, test_value: test_value not in item_value,
 | 
				
			||||||
 | 
					    'BEGINS_WITH': lambda item_value, test_value: item_value.startswith(test_value),
 | 
				
			||||||
 | 
					    'IN': lambda item_value, test_value: item_value in test_value,
 | 
				
			||||||
 | 
					    'BETWEEN': lambda item_value, lower_test_value, upper_test_value: lower_test_value <= item_value <= upper_test_value,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def get_comparison_func(range_comparison):
 | 
				
			||||||
 | 
					    return COMPARISON_FUNCS.get(range_comparison)
 | 
				
			||||||
							
								
								
									
										313
									
								
								moto/dynamodb2/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										313
									
								
								moto/dynamodb2/models.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,313 @@
 | 
				
			|||||||
 | 
					from collections import defaultdict
 | 
				
			||||||
 | 
					import datetime
 | 
				
			||||||
 | 
					import json
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					try:
 | 
				
			||||||
 | 
					        from collections import OrderedDict
 | 
				
			||||||
 | 
					except ImportError:
 | 
				
			||||||
 | 
					        # python 2.6 or earlier, use backport
 | 
				
			||||||
 | 
					        from ordereddict import OrderedDict
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from moto.core import BaseBackend
 | 
				
			||||||
 | 
					from .comparisons import get_comparison_func
 | 
				
			||||||
 | 
					from .utils import unix_time
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class DynamoJsonEncoder(json.JSONEncoder):
 | 
				
			||||||
 | 
					    def default(self, obj):
 | 
				
			||||||
 | 
					        if hasattr(obj, 'to_json'):
 | 
				
			||||||
 | 
					            return obj.to_json()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def dynamo_json_dump(dynamo_object):
 | 
				
			||||||
 | 
					    return json.dumps(dynamo_object, cls=DynamoJsonEncoder)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class DynamoType(object):
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					    http://docs.aws.amazon.com/amazondynamodb/latest/developerguide/DataModel.html#DataModelDataTypes
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self, type_as_dict):
 | 
				
			||||||
 | 
					        self.type = type_as_dict.keys()[0]
 | 
				
			||||||
 | 
					        self.value = type_as_dict.values()[0]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __hash__(self):
 | 
				
			||||||
 | 
					        return hash((self.type, self.value))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __eq__(self, other):
 | 
				
			||||||
 | 
					        return (
 | 
				
			||||||
 | 
					            self.type == other.type and
 | 
				
			||||||
 | 
					            self.value == other.value
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __repr__(self):
 | 
				
			||||||
 | 
					        return "DynamoType: {0}".format(self.to_json())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def to_json(self):
 | 
				
			||||||
 | 
					        return {self.type: self.value}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def compare(self, range_comparison, range_objs):
 | 
				
			||||||
 | 
					        """
 | 
				
			||||||
 | 
					        Compares this type against comparison filters
 | 
				
			||||||
 | 
					        """
 | 
				
			||||||
 | 
					        range_values = [obj.value for obj in range_objs]
 | 
				
			||||||
 | 
					        comparison_func = get_comparison_func(range_comparison)
 | 
				
			||||||
 | 
					        return comparison_func(self.value, *range_values)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Item(object):
 | 
				
			||||||
 | 
					    def __init__(self, hash_key, hash_key_type, range_key, range_key_type, attrs):
 | 
				
			||||||
 | 
					        self.hash_key = hash_key
 | 
				
			||||||
 | 
					        self.hash_key_type = hash_key_type
 | 
				
			||||||
 | 
					        self.range_key = range_key
 | 
				
			||||||
 | 
					        self.range_key_type = range_key_type
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.attrs = {}
 | 
				
			||||||
 | 
					        for key, value in attrs.iteritems():
 | 
				
			||||||
 | 
					            self.attrs[key] = DynamoType(value)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __repr__(self):
 | 
				
			||||||
 | 
					        return "Item: {0}".format(self.to_json())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def to_json(self):
 | 
				
			||||||
 | 
					        attributes = {}
 | 
				
			||||||
 | 
					        for attribute_key, attribute in self.attrs.iteritems():
 | 
				
			||||||
 | 
					            attributes[attribute_key] = attribute.value
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            "Attributes": attributes
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def describe_attrs(self, attributes):
 | 
				
			||||||
 | 
					        if attributes:
 | 
				
			||||||
 | 
					            included = {}
 | 
				
			||||||
 | 
					            for key, value in self.attrs.iteritems():
 | 
				
			||||||
 | 
					                if key in attributes:
 | 
				
			||||||
 | 
					                    included[key] = value
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            included = self.attrs
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            "Item": included
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Table(object):
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self, table_name, schema=None, attr = None, throughput=None, indexes=None):
 | 
				
			||||||
 | 
					        self.name = table_name
 | 
				
			||||||
 | 
					        self.attr = attr
 | 
				
			||||||
 | 
					        self.schema = schema
 | 
				
			||||||
 | 
					        self.range_key_attr = None
 | 
				
			||||||
 | 
					        self.hash_key_attr = None
 | 
				
			||||||
 | 
					        self.range_key_type = None
 | 
				
			||||||
 | 
					        self.hash_key_type = None
 | 
				
			||||||
 | 
					        for elem in schema:
 | 
				
			||||||
 | 
					            if elem["KeyType"] == "HASH":
 | 
				
			||||||
 | 
					                self.hash_key_attr = elem["AttributeName"]
 | 
				
			||||||
 | 
					                self.hash_key_type = elem["KeyType"]
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                self.range_key_attr = elem["AttributeName"]
 | 
				
			||||||
 | 
					                self.range_key_type = elem["KeyType"]
 | 
				
			||||||
 | 
					        if throughput is None:
 | 
				
			||||||
 | 
					             self.throughput = {u'WriteCapacityUnits': 10, u'ReadCapacityUnits': 10}
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            self.throughput = throughput
 | 
				
			||||||
 | 
					        self.throughput["NumberOfDecreasesToday"] = 0
 | 
				
			||||||
 | 
					        self.indexes = indexes
 | 
				
			||||||
 | 
					        self.created_at = datetime.datetime.now()
 | 
				
			||||||
 | 
					        self.items = defaultdict(dict)
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def describe(self):
 | 
				
			||||||
 | 
					        results = {
 | 
				
			||||||
 | 
					        'Table': {
 | 
				
			||||||
 | 
					            'AttributeDefinitions': self.attr,
 | 
				
			||||||
 | 
					            'ProvisionedThroughput': self.throughput, 
 | 
				
			||||||
 | 
					            'TableSizeBytes': 0, 
 | 
				
			||||||
 | 
					            'TableName': self.name, 
 | 
				
			||||||
 | 
					            'TableStatus': 'ACTIVE', 
 | 
				
			||||||
 | 
					            'KeySchema': self.schema, 
 | 
				
			||||||
 | 
					            'ItemCount': len(self), 
 | 
				
			||||||
 | 
					            'CreationDateTime': unix_time(self.created_at)
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return results
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    def __len__(self):
 | 
				
			||||||
 | 
					        count = 0
 | 
				
			||||||
 | 
					        for key, value in self.items.iteritems():
 | 
				
			||||||
 | 
					            if self.has_range_key:
 | 
				
			||||||
 | 
					                count += len(value)
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                count += 1
 | 
				
			||||||
 | 
					        return count
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    def put_item(self, item_attrs):
 | 
				
			||||||
 | 
					        hash_value = DynamoType(item_attrs.get(self.hash_key_attr))
 | 
				
			||||||
 | 
					        if self.has_range_key:
 | 
				
			||||||
 | 
					            range_value = DynamoType(item_attrs.get(self.range_key_attr))
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            range_value = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        item = Item(hash_value, self.hash_key_type, range_value, self.range_key_type, item_attrs)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if range_value:
 | 
				
			||||||
 | 
					            self.items[hash_value][range_value] = item
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            self.items[hash_value] = item
 | 
				
			||||||
 | 
					        return item
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    def __nonzero__(self):
 | 
				
			||||||
 | 
					        return True
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def has_range_key(self):
 | 
				
			||||||
 | 
					        return self.range_key_attr is not None
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    def get_item(self, hash_key, range_key):
 | 
				
			||||||
 | 
					        if self.has_range_key and not range_key:
 | 
				
			||||||
 | 
					            raise ValueError("Table has a range key, but no range key was passed into get_item")
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            if range_key:
 | 
				
			||||||
 | 
					                return self.items[hash_key][range_key]
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                return self.items[hash_key]
 | 
				
			||||||
 | 
					        except KeyError:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					    def delete_item(self, hash_key, range_key):
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            if range_key:
 | 
				
			||||||
 | 
					                return self.items[hash_key].pop(range_key)
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                return self.items.pop(hash_key)
 | 
				
			||||||
 | 
					        except KeyError:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					    def query(self, hash_key, range_comparison, range_objs):
 | 
				
			||||||
 | 
					        results = []
 | 
				
			||||||
 | 
					        last_page = True  # Once pagination is implemented, change this
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        possible_results =  [ item for item in list(self.all_items()) if item.hash_key == hash_key] 
 | 
				
			||||||
 | 
					        if range_comparison:
 | 
				
			||||||
 | 
					            for result in possible_results:
 | 
				
			||||||
 | 
					                if result.range_key.compare(range_comparison, range_objs):
 | 
				
			||||||
 | 
					                    results.append(result)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            # If we're not filtering on range key, return all values
 | 
				
			||||||
 | 
					            results = possible_results
 | 
				
			||||||
 | 
					        return results, last_page
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def all_items(self):
 | 
				
			||||||
 | 
					        for hash_set in self.items.values():
 | 
				
			||||||
 | 
					            if self.range_key_attr:
 | 
				
			||||||
 | 
					                for item in hash_set.values():
 | 
				
			||||||
 | 
					                    yield item
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                yield hash_set
 | 
				
			||||||
 | 
					                
 | 
				
			||||||
 | 
					    def scan(self, filters):
 | 
				
			||||||
 | 
					        results = []
 | 
				
			||||||
 | 
					        scanned_count = 0
 | 
				
			||||||
 | 
					        last_page = True  # Once pagination is implemented, change this
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for result in self.all_items():
 | 
				
			||||||
 | 
					            scanned_count += 1
 | 
				
			||||||
 | 
					            passes_all_conditions = True
 | 
				
			||||||
 | 
					            for attribute_name, (comparison_operator, comparison_objs) in filters.iteritems():
 | 
				
			||||||
 | 
					                attribute = result.attrs.get(attribute_name)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if attribute:
 | 
				
			||||||
 | 
					                    # Attribute found
 | 
				
			||||||
 | 
					                    if not attribute.compare(comparison_operator, comparison_objs):
 | 
				
			||||||
 | 
					                        passes_all_conditions = False
 | 
				
			||||||
 | 
					                        break
 | 
				
			||||||
 | 
					                elif comparison_operator == 'NULL':
 | 
				
			||||||
 | 
					                    # Comparison is NULL and we don't have the attribute
 | 
				
			||||||
 | 
					                    continue
 | 
				
			||||||
 | 
					                else:
 | 
				
			||||||
 | 
					                    # No attribute found and comparison is no NULL. This item fails
 | 
				
			||||||
 | 
					                    passes_all_conditions = False
 | 
				
			||||||
 | 
					                    break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if passes_all_conditions:
 | 
				
			||||||
 | 
					                results.append(result)
 | 
				
			||||||
 | 
					        return results, scanned_count, last_page
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class DynamoDBBackend(BaseBackend):
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self):
 | 
				
			||||||
 | 
					        self.tables = OrderedDict()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def create_table(self, name, **params):
 | 
				
			||||||
 | 
					        table = Table(name, **params)
 | 
				
			||||||
 | 
					        self.tables[name] = table
 | 
				
			||||||
 | 
					        return table
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def delete_table(self, name):
 | 
				
			||||||
 | 
					        return self.tables.pop(name, None)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def update_table_throughput(self, name, throughput):
 | 
				
			||||||
 | 
					        table = self.tables[name]
 | 
				
			||||||
 | 
					        table.throughput = throughput
 | 
				
			||||||
 | 
					        return table
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def put_item(self, table_name, item_attrs):
 | 
				
			||||||
 | 
					        table = self.tables.get(table_name)
 | 
				
			||||||
 | 
					        if not table:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        return table.put_item(item_attrs)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    def get_table_keys_name(self, table_name):
 | 
				
			||||||
 | 
					        table = self.tables.get(table_name)
 | 
				
			||||||
 | 
					        if not table:
 | 
				
			||||||
 | 
					            return None, None
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            return table.hash_key_attr, table.range_key_attr
 | 
				
			||||||
 | 
					         
 | 
				
			||||||
 | 
					    def get_keys_value(self, table, keys):
 | 
				
			||||||
 | 
					        if not table.hash_key_attr in keys or (table.has_range_key and not table.range_key_attr in keys):
 | 
				
			||||||
 | 
					            raise ValueError("Table has a range key, but no range key was passed into get_item")        
 | 
				
			||||||
 | 
					        hash_key = DynamoType(keys[table.hash_key_attr])    
 | 
				
			||||||
 | 
					        range_key = DynamoType(keys[table.range_key_attr]) if table.has_range_key else None
 | 
				
			||||||
 | 
					        return hash_key,range_key
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_item(self, table_name, keys):
 | 
				
			||||||
 | 
					        table = self.tables.get(table_name)
 | 
				
			||||||
 | 
					        if not table:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        hash_key,range_key = self.get_keys_value(table,keys)
 | 
				
			||||||
 | 
					        return table.get_item(hash_key, range_key)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def query(self, table_name, hash_key_dict, range_comparison, range_value_dicts):
 | 
				
			||||||
 | 
					        table = self.tables.get(table_name)
 | 
				
			||||||
 | 
					        if not table:
 | 
				
			||||||
 | 
					            return None, None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        hash_key = DynamoType(hash_key_dict)
 | 
				
			||||||
 | 
					        range_values = [DynamoType(range_value) for range_value in range_value_dicts]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return table.query(hash_key, range_comparison, range_values)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    def scan(self, table_name, filters):
 | 
				
			||||||
 | 
					        table = self.tables.get(table_name)
 | 
				
			||||||
 | 
					        if not table:
 | 
				
			||||||
 | 
					            return None, None, None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        scan_filters = {}
 | 
				
			||||||
 | 
					        for key, (comparison_operator, comparison_values) in filters.iteritems():
 | 
				
			||||||
 | 
					            dynamo_types = [DynamoType(value) for value in comparison_values]
 | 
				
			||||||
 | 
					            scan_filters[key] = (comparison_operator, dynamo_types)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return table.scan(scan_filters)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    def delete_item(self, table_name, keys):
 | 
				
			||||||
 | 
					        table = self.tables.get(table_name)
 | 
				
			||||||
 | 
					        if not table:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        hash_key, range_key = self.get_keys_value(table, keys)
 | 
				
			||||||
 | 
					        return table.delete_item(hash_key, range_key)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					dynamodb_backend2 = DynamoDBBackend()
 | 
				
			||||||
							
								
								
									
										302
									
								
								moto/dynamodb2/responses.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										302
									
								
								moto/dynamodb2/responses.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,302 @@
 | 
				
			|||||||
 | 
					import json
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from moto.core.responses import BaseResponse
 | 
				
			||||||
 | 
					from moto.core.utils import camelcase_to_underscores
 | 
				
			||||||
 | 
					from .models import dynamodb_backend2, dynamo_json_dump
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					GET_SESSION_TOKEN_RESULT = """
 | 
				
			||||||
 | 
					<GetSessionTokenResponse xmlns="https://sts.amazonaws.com/doc/2011-06-15/">
 | 
				
			||||||
 | 
					 <GetSessionTokenResult>
 | 
				
			||||||
 | 
					 <Credentials>
 | 
				
			||||||
 | 
					 <SessionToken>
 | 
				
			||||||
 | 
					 AQoEXAMPLEH4aoAH0gNCAPyJxz4BlCFFxWNE1OPTgk5TthT+FvwqnKwRcOIfrRh3c/L
 | 
				
			||||||
 | 
					 To6UDdyJwOOvEVPvLXCrrrUtdnniCEXAMPLE/IvU1dYUg2RVAJBanLiHb4IgRmpRV3z
 | 
				
			||||||
 | 
					 rkuWJOgQs8IZZaIv2BXIa2R4OlgkBN9bkUDNCJiBeb/AXlzBBko7b15fjrBs2+cTQtp
 | 
				
			||||||
 | 
					 Z3CYWFXG8C5zqx37wnOE49mRl/+OtkIKGO7fAE
 | 
				
			||||||
 | 
					 </SessionToken>
 | 
				
			||||||
 | 
					 <SecretAccessKey>
 | 
				
			||||||
 | 
					 wJalrXUtnFEMI/K7MDENG/bPxRfiCYzEXAMPLEKEY
 | 
				
			||||||
 | 
					 </SecretAccessKey>
 | 
				
			||||||
 | 
					 <Expiration>2011-07-11T19:55:29.611Z</Expiration>
 | 
				
			||||||
 | 
					 <AccessKeyId>AKIAIOSFODNN7EXAMPLE</AccessKeyId>
 | 
				
			||||||
 | 
					 </Credentials>
 | 
				
			||||||
 | 
					 </GetSessionTokenResult>
 | 
				
			||||||
 | 
					 <ResponseMetadata>
 | 
				
			||||||
 | 
					 <RequestId>58c5dbae-abef-11e0-8cfe-09039844ac7d</RequestId>
 | 
				
			||||||
 | 
					 </ResponseMetadata>
 | 
				
			||||||
 | 
					</GetSessionTokenResponse>"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def sts_handler():
 | 
				
			||||||
 | 
					    return GET_SESSION_TOKEN_RESULT
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class DynamoHandler(BaseResponse):
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_endpoint_name(self, headers):
 | 
				
			||||||
 | 
					        """Parses request headers and extracts part od the X-Amz-Target
 | 
				
			||||||
 | 
					        that corresponds to a method of DynamoHandler
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ie: X-Amz-Target: DynamoDB_20111205.ListTables -> ListTables
 | 
				
			||||||
 | 
					        """
 | 
				
			||||||
 | 
					        # Headers are case-insensitive. Probably a better way to do this.
 | 
				
			||||||
 | 
					        match = headers.get('x-amz-target') or headers.get('X-Amz-Target')
 | 
				
			||||||
 | 
					        if match:
 | 
				
			||||||
 | 
					            return match.split(".")[1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def error(self, type_, status=400):
 | 
				
			||||||
 | 
					        return status, self.response_headers, dynamo_json_dump({'__type': type_})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def call_action(self):
 | 
				
			||||||
 | 
					        if 'GetSessionToken' in self.body:
 | 
				
			||||||
 | 
					            return 200, self.response_headers, sts_handler()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.body = json.loads(self.body or '{}')
 | 
				
			||||||
 | 
					        endpoint = self.get_endpoint_name(self.headers)
 | 
				
			||||||
 | 
					        if endpoint:
 | 
				
			||||||
 | 
					            endpoint = camelcase_to_underscores(endpoint)
 | 
				
			||||||
 | 
					            response = getattr(self, endpoint)()
 | 
				
			||||||
 | 
					            if isinstance(response, basestring):
 | 
				
			||||||
 | 
					                return 200, self.response_headers, response
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                status_code, new_headers, response_content = response
 | 
				
			||||||
 | 
					                self.response_headers.update(new_headers)
 | 
				
			||||||
 | 
					                return status_code, self.response_headers, response_content
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            return 404, self.response_headers, ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def list_tables(self):
 | 
				
			||||||
 | 
					        body = self.body
 | 
				
			||||||
 | 
					        limit = body.get('Limit')
 | 
				
			||||||
 | 
					        if body.get("ExclusiveStartTableName"):
 | 
				
			||||||
 | 
					            last = body.get("ExclusiveStartTableName")
 | 
				
			||||||
 | 
					            start = dynamodb_backend2.tables.keys().index(last) + 1
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            start = 0
 | 
				
			||||||
 | 
					        all_tables = dynamodb_backend2.tables.keys()
 | 
				
			||||||
 | 
					        if limit:
 | 
				
			||||||
 | 
					            tables = all_tables[start:start + limit]
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            tables = all_tables[start:]
 | 
				
			||||||
 | 
					        response = {"TableNames": tables}
 | 
				
			||||||
 | 
					        if limit and len(all_tables) > start + limit:
 | 
				
			||||||
 | 
					            response["LastEvaluatedTableName"] = tables[-1]
 | 
				
			||||||
 | 
					        return dynamo_json_dump(response)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def create_table(self):
 | 
				
			||||||
 | 
					        body = self.body
 | 
				
			||||||
 | 
					        #get the table name
 | 
				
			||||||
 | 
					        table_name = body['TableName']
 | 
				
			||||||
 | 
					        #get the throughput
 | 
				
			||||||
 | 
					        throughput = body["ProvisionedThroughput"]        
 | 
				
			||||||
 | 
					        #getting the schema
 | 
				
			||||||
 | 
					        key_schema = body['KeySchema']
 | 
				
			||||||
 | 
					        #getting attribute definition
 | 
				
			||||||
 | 
					        attr = body["AttributeDefinitions"]
 | 
				
			||||||
 | 
					        #getting the indexes
 | 
				
			||||||
 | 
					        table = dynamodb_backend2.create_table(table_name, 
 | 
				
			||||||
 | 
					                   schema = key_schema,           
 | 
				
			||||||
 | 
					                   throughput = throughput, 
 | 
				
			||||||
 | 
					                   attr = attr)
 | 
				
			||||||
 | 
					        return dynamo_json_dump(table.describe)        
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def delete_table(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					        table = dynamodb_backend2.delete_table(name)
 | 
				
			||||||
 | 
					        if table is not None:
 | 
				
			||||||
 | 
					            return dynamo_json_dump(table.describe)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            er = 'com.amazonaws.dynamodb.v20111205#ResourceNotFoundException'
 | 
				
			||||||
 | 
					            return self.error(er)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def update_table(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					        throughput = self.body["ProvisionedThroughput"]
 | 
				
			||||||
 | 
					        table = dynamodb_backend2.update_table_throughput(name, throughput)
 | 
				
			||||||
 | 
					        return dynamo_json_dump(table.describe)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def describe_table(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            table = dynamodb_backend2.tables[name]
 | 
				
			||||||
 | 
					        except KeyError:
 | 
				
			||||||
 | 
					            er = 'com.amazonaws.dynamodb.v20111205#ResourceNotFoundException'
 | 
				
			||||||
 | 
					            return self.error(er)
 | 
				
			||||||
 | 
					        return dynamo_json_dump(table.describe)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def put_item(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					        item = self.body['Item']
 | 
				
			||||||
 | 
					        result = dynamodb_backend2.put_item(name, item)
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					        if result:
 | 
				
			||||||
 | 
					            item_dict = result.to_json()
 | 
				
			||||||
 | 
					            item_dict['ConsumedCapacityUnits'] = 1
 | 
				
			||||||
 | 
					            return dynamo_json_dump(item_dict)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            er = 'com.amazonaws.dynamodb.v20111205#ResourceNotFoundException'
 | 
				
			||||||
 | 
					            return self.error(er)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def batch_write_item(self):
 | 
				
			||||||
 | 
					        table_batches = self.body['RequestItems']
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for table_name, table_requests in table_batches.iteritems():
 | 
				
			||||||
 | 
					            for table_request in table_requests:
 | 
				
			||||||
 | 
					                request_type = table_request.keys()[0]
 | 
				
			||||||
 | 
					                request = table_request.values()[0]
 | 
				
			||||||
 | 
					                if request_type == 'PutRequest':
 | 
				
			||||||
 | 
					                    item = request['Item']
 | 
				
			||||||
 | 
					                    dynamodb_backend2.put_item(table_name, item)
 | 
				
			||||||
 | 
					                elif request_type == 'DeleteRequest':
 | 
				
			||||||
 | 
					                    keys = request['Key']
 | 
				
			||||||
 | 
					                    item = dynamodb_backend2.delete_item(table_name, keys)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        response = {
 | 
				
			||||||
 | 
					            "Responses": {
 | 
				
			||||||
 | 
					                "Thread": {
 | 
				
			||||||
 | 
					                    "ConsumedCapacityUnits": 1.0
 | 
				
			||||||
 | 
					                },
 | 
				
			||||||
 | 
					                "Reply": {
 | 
				
			||||||
 | 
					                    "ConsumedCapacityUnits": 1.0
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            "UnprocessedItems": {}
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return dynamo_json_dump(response)
 | 
				
			||||||
 | 
					    def get_item(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					        key = self.body['Key']
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            item = dynamodb_backend2.get_item(name, key)
 | 
				
			||||||
 | 
					        except ValueError:
 | 
				
			||||||
 | 
					            er = 'com.amazon.coral.validate#ValidationException'
 | 
				
			||||||
 | 
					            return self.error(er, status=400)
 | 
				
			||||||
 | 
					        if item:
 | 
				
			||||||
 | 
					            item_dict = item.describe_attrs(attributes = None)
 | 
				
			||||||
 | 
					            item_dict['ConsumedCapacityUnits'] = 0.5
 | 
				
			||||||
 | 
					            return dynamo_json_dump(item_dict)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            # Item not found
 | 
				
			||||||
 | 
					            er = 'com.amazonaws.dynamodb.v20111205#ResourceNotFoundException'
 | 
				
			||||||
 | 
					            return self.error(er, status=404)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def batch_get_item(self):
 | 
				
			||||||
 | 
					        table_batches = self.body['RequestItems']
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        results = { 
 | 
				
			||||||
 | 
					            "ConsumedCapacity":[],
 | 
				
			||||||
 | 
					            "Responses": {                
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            "UnprocessedKeys": {
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for table_name, table_request in table_batches.iteritems():
 | 
				
			||||||
 | 
					            items = []
 | 
				
			||||||
 | 
					            keys = table_request['Keys']
 | 
				
			||||||
 | 
					            attributes_to_get = table_request.get('AttributesToGet')
 | 
				
			||||||
 | 
					            results["Responses"][table_name]=[]
 | 
				
			||||||
 | 
					            for key in keys:
 | 
				
			||||||
 | 
					                item = dynamodb_backend2.get_item(table_name, key)
 | 
				
			||||||
 | 
					                if item:
 | 
				
			||||||
 | 
					                    item_describe = item.describe_attrs(attributes_to_get)
 | 
				
			||||||
 | 
					                    results["Responses"][table_name].append(item_describe["Item"])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            results["ConsumedCapacity"].append({
 | 
				
			||||||
 | 
					                "CapacityUnits": len(keys),
 | 
				
			||||||
 | 
					                "TableName": table_name
 | 
				
			||||||
 | 
					            })
 | 
				
			||||||
 | 
					        return dynamo_json_dump(results)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def query(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					        keys = self.body['KeyConditions']
 | 
				
			||||||
 | 
					        hash_key_name, range_key_name = dynamodb_backend2.get_table_keys_name(name)
 | 
				
			||||||
 | 
					        if hash_key_name is None:
 | 
				
			||||||
 | 
					            er = "'com.amazonaws.dynamodb.v20120810#ResourceNotFoundException"  
 | 
				
			||||||
 | 
					            return self.error(er)
 | 
				
			||||||
 | 
					        hash_key = keys[hash_key_name]['AttributeValueList'][0]
 | 
				
			||||||
 | 
					        if len(keys) == 1:
 | 
				
			||||||
 | 
					            range_comparison = None
 | 
				
			||||||
 | 
					            range_values = []
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            if range_key_name == None:
 | 
				
			||||||
 | 
					                er = "com.amazon.coral.validate#ValidationException"  
 | 
				
			||||||
 | 
					                return self.error(er)
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                range_condition = keys[range_key_name]
 | 
				
			||||||
 | 
					                if range_condition:
 | 
				
			||||||
 | 
					                    range_comparison = range_condition['ComparisonOperator']
 | 
				
			||||||
 | 
					                    range_values = range_condition['AttributeValueList']
 | 
				
			||||||
 | 
					                else:
 | 
				
			||||||
 | 
					                    range_comparison = None
 | 
				
			||||||
 | 
					                    range_values = []
 | 
				
			||||||
 | 
					        items, last_page = dynamodb_backend2.query(name, hash_key, range_comparison, range_values)
 | 
				
			||||||
 | 
					        if items is None:
 | 
				
			||||||
 | 
					            er = 'com.amazonaws.dynamodb.v20111205#ResourceNotFoundException'
 | 
				
			||||||
 | 
					            return self.error(er) 
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					        result = {
 | 
				
			||||||
 | 
					            "Count": len(items),
 | 
				
			||||||
 | 
					            "Items": [item.attrs for item in items],
 | 
				
			||||||
 | 
					            "ConsumedCapacityUnits": 1,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # Implement this when we do pagination
 | 
				
			||||||
 | 
					        # if not last_page:
 | 
				
			||||||
 | 
					        #     result["LastEvaluatedKey"] = {
 | 
				
			||||||
 | 
					        #         "HashKeyElement": items[-1].hash_key,
 | 
				
			||||||
 | 
					        #         "RangeKeyElement": items[-1].range_key,
 | 
				
			||||||
 | 
					        #     }
 | 
				
			||||||
 | 
					        return dynamo_json_dump(result)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def scan(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        filters = {}
 | 
				
			||||||
 | 
					        scan_filters = self.body.get('ScanFilter', {})
 | 
				
			||||||
 | 
					        for attribute_name, scan_filter in scan_filters.iteritems():
 | 
				
			||||||
 | 
					            # Keys are attribute names. Values are tuples of (comparison, comparison_value)
 | 
				
			||||||
 | 
					            comparison_operator = scan_filter["ComparisonOperator"]
 | 
				
			||||||
 | 
					            comparison_values = scan_filter.get("AttributeValueList", [])
 | 
				
			||||||
 | 
					            filters[attribute_name] = (comparison_operator, comparison_values)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        items, scanned_count, last_page = dynamodb_backend2.scan(name, filters)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if items is None:
 | 
				
			||||||
 | 
					            er = 'com.amazonaws.dynamodb.v20111205#ResourceNotFoundException'
 | 
				
			||||||
 | 
					            return self.error(er)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        result = {
 | 
				
			||||||
 | 
					            "Count": len(items),
 | 
				
			||||||
 | 
					            "Items": [item.attrs for item in items],
 | 
				
			||||||
 | 
					            "ConsumedCapacityUnits": 1,
 | 
				
			||||||
 | 
					            "ScannedCount": scanned_count
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # Implement this when we do pagination
 | 
				
			||||||
 | 
					        # if not last_page:
 | 
				
			||||||
 | 
					        #     result["LastEvaluatedKey"] = {
 | 
				
			||||||
 | 
					        #         "HashKeyElement": items[-1].hash_key,
 | 
				
			||||||
 | 
					        #         "RangeKeyElement": items[-1].range_key,
 | 
				
			||||||
 | 
					        #     }
 | 
				
			||||||
 | 
					        return dynamo_json_dump(result)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def delete_item(self):
 | 
				
			||||||
 | 
					        name = self.body['TableName']
 | 
				
			||||||
 | 
					        keys = self.body['Key']
 | 
				
			||||||
 | 
					        return_values = self.body.get('ReturnValues', '')
 | 
				
			||||||
 | 
					        item = dynamodb_backend2.delete_item(name, keys)
 | 
				
			||||||
 | 
					        if item:
 | 
				
			||||||
 | 
					            if return_values == 'ALL_OLD':
 | 
				
			||||||
 | 
					                item_dict = item.to_json()
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                item_dict = {'Attributes': []}
 | 
				
			||||||
 | 
					            item_dict['ConsumedCapacityUnits'] = 0.5
 | 
				
			||||||
 | 
					            return dynamo_json_dump(item_dict)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            er = 'com.amazonaws.dynamodb.v20120810#ConditionalCheckFailedException'
 | 
				
			||||||
 | 
					            return self.error(er)
 | 
				
			||||||
							
								
								
									
										10
									
								
								moto/dynamodb2/urls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								moto/dynamodb2/urls.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,10 @@
 | 
				
			|||||||
 | 
					from .responses import DynamoHandler
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					url_bases = [
 | 
				
			||||||
 | 
					    "https?://dynamodb.(.+).amazonaws.com",
 | 
				
			||||||
 | 
					    "https?://sts.amazonaws.com",
 | 
				
			||||||
 | 
					]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					url_paths = {
 | 
				
			||||||
 | 
					    "{0}/": DynamoHandler().dispatch,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										5
									
								
								moto/dynamodb2/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								moto/dynamodb2/utils.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,5 @@
 | 
				
			|||||||
 | 
					import calendar
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def unix_time(dt):
 | 
				
			||||||
 | 
					    return calendar.timegm(dt.timetuple())
 | 
				
			||||||
							
								
								
									
										60
									
								
								tests/test_dynamodb2/test_dynamodb.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								tests/test_dynamodb2/test_dynamodb.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,60 @@
 | 
				
			|||||||
 | 
					import boto
 | 
				
			||||||
 | 
					import sure  # noqa
 | 
				
			||||||
 | 
					import requests
 | 
				
			||||||
 | 
					import boto.dynamodb2
 | 
				
			||||||
 | 
					from moto import mock_dynamodb2
 | 
				
			||||||
 | 
					from moto.dynamodb2 import dynamodb_backend2
 | 
				
			||||||
 | 
					from boto.exception import JSONResponseError
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_list_tables():
 | 
				
			||||||
 | 
					    name = 'TestTable'    
 | 
				
			||||||
 | 
					    #{'schema': }    
 | 
				
			||||||
 | 
					    dynamodb_backend2.create_table(name,schema=[
 | 
				
			||||||
 | 
					        {u'KeyType': u'HASH', u'AttributeName': u'forum_name'}, 
 | 
				
			||||||
 | 
					        {u'KeyType': u'RANGE', u'AttributeName': u'subject'}
 | 
				
			||||||
 | 
					    ])
 | 
				
			||||||
 | 
					    conn =  boto.dynamodb2.connect_to_region(
 | 
				
			||||||
 | 
					            'us-west-2',
 | 
				
			||||||
 | 
					        aws_access_key_id="ak",
 | 
				
			||||||
 | 
					        aws_secret_access_key="sk")
 | 
				
			||||||
 | 
					    assert conn.list_tables()["TableNames"] == [name]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_list_tables_layer_1():
 | 
				
			||||||
 | 
					    dynamodb_backend2.create_table("test_1",schema=[
 | 
				
			||||||
 | 
					        {u'KeyType': u'HASH', u'AttributeName': u'name'}
 | 
				
			||||||
 | 
					    ])
 | 
				
			||||||
 | 
					    dynamodb_backend2.create_table("test_2",schema=[
 | 
				
			||||||
 | 
					        {u'KeyType': u'HASH', u'AttributeName': u'name'}
 | 
				
			||||||
 | 
					    ])
 | 
				
			||||||
 | 
					    conn =  boto.dynamodb2.connect_to_region(
 | 
				
			||||||
 | 
					        'us-west-2',
 | 
				
			||||||
 | 
					        aws_access_key_id="ak",
 | 
				
			||||||
 | 
					        aws_secret_access_key="sk")
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    res = conn.list_tables(limit=1)
 | 
				
			||||||
 | 
					    expected = {"TableNames": ["test_1"], "LastEvaluatedTableName": "test_1"}
 | 
				
			||||||
 | 
					    res.should.equal(expected)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    res = conn.list_tables(limit=1, exclusive_start_table_name="test_1")
 | 
				
			||||||
 | 
					    expected = {"TableNames": ["test_2"]}
 | 
				
			||||||
 | 
					    res.should.equal(expected)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_describe_missing_table():
 | 
				
			||||||
 | 
					    conn =  boto.dynamodb2.connect_to_region(
 | 
				
			||||||
 | 
					        'us-west-2',
 | 
				
			||||||
 | 
					        aws_access_key_id="ak",
 | 
				
			||||||
 | 
					        aws_secret_access_key="sk")
 | 
				
			||||||
 | 
					    conn.describe_table.when.called_with('messages').should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_sts_handler():
 | 
				
			||||||
 | 
					    res = requests.post("https://sts.amazonaws.com/", data={"GetSessionToken": ""})
 | 
				
			||||||
 | 
					    res.ok.should.be.ok
 | 
				
			||||||
 | 
					    res.text.should.contain("SecretAccessKey")
 | 
				
			||||||
							
								
								
									
										415
									
								
								tests/test_dynamodb2/test_dynamodb_table_with_range_key.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										415
									
								
								tests/test_dynamodb2/test_dynamodb_table_with_range_key.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,415 @@
 | 
				
			|||||||
 | 
					import boto
 | 
				
			||||||
 | 
					import sure  # noqa
 | 
				
			||||||
 | 
					from freezegun import freeze_time
 | 
				
			||||||
 | 
					from moto import mock_dynamodb2
 | 
				
			||||||
 | 
					from boto.dynamodb2.fields import HashKey
 | 
				
			||||||
 | 
					from boto.dynamodb2.fields import RangeKey
 | 
				
			||||||
 | 
					from boto.dynamodb2.table import Table
 | 
				
			||||||
 | 
					from boto.dynamodb2.table import Item
 | 
				
			||||||
 | 
					from boto.dynamodb.exceptions import DynamoDBKeyNotFoundError
 | 
				
			||||||
 | 
					from boto.dynamodb2.exceptions import ValidationException
 | 
				
			||||||
 | 
					from boto.dynamodb2.exceptions import ConditionalCheckFailedException
 | 
				
			||||||
 | 
					from boto.exception import JSONResponseError
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def create_table():
 | 
				
			||||||
 | 
					    table = Table.create('messages', schema=[
 | 
				
			||||||
 | 
					        HashKey('forum_name'),
 | 
				
			||||||
 | 
					        RangeKey('subject'),
 | 
				
			||||||
 | 
					    ], throughput={
 | 
				
			||||||
 | 
					        'read': 10,
 | 
				
			||||||
 | 
					        'write': 10,
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					    return table
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def iterate_results(res):
 | 
				
			||||||
 | 
					    for i in res:
 | 
				
			||||||
 | 
					        print i
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@freeze_time("2012-01-14")
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_create_table():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    expected = {
 | 
				
			||||||
 | 
					        'Table': {
 | 
				
			||||||
 | 
					            'AttributeDefinitions': [
 | 
				
			||||||
 | 
					                {'AttributeName': 'forum_name', 'AttributeType': 'S'}, 
 | 
				
			||||||
 | 
					                {'AttributeName': 'subject', 'AttributeType': 'S'}
 | 
				
			||||||
 | 
					            ], 
 | 
				
			||||||
 | 
					            'ProvisionedThroughput': {
 | 
				
			||||||
 | 
					                'NumberOfDecreasesToday': 0, 'WriteCapacityUnits': 10, 'ReadCapacityUnits': 10
 | 
				
			||||||
 | 
					                }, 
 | 
				
			||||||
 | 
					            'TableSizeBytes': 0, 
 | 
				
			||||||
 | 
					            'TableName': 'messages', 
 | 
				
			||||||
 | 
					            'TableStatus': 'ACTIVE', 
 | 
				
			||||||
 | 
					            'KeySchema': [
 | 
				
			||||||
 | 
					                {'KeyType': 'HASH', 'AttributeName': 'forum_name'}, 
 | 
				
			||||||
 | 
					                {'KeyType': 'RANGE', 'AttributeName': 'subject'}
 | 
				
			||||||
 | 
					            ], 
 | 
				
			||||||
 | 
					            'ItemCount': 0, 'CreationDateTime': 1326499200.0
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    table.describe().should.equal(expected)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_delete_table():
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    conn.list_tables()["TableNames"].should.have.length_of(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.delete()
 | 
				
			||||||
 | 
					    conn.list_tables()["TableNames"].should.have.length_of(0)
 | 
				
			||||||
 | 
					    conn.delete_table.when.called_with('messages').should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_update_table_throughput():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    table.throughput["read"].should.equal(10)
 | 
				
			||||||
 | 
					    table.throughput["write"].should.equal(10)    
 | 
				
			||||||
 | 
					    table.update(throughput={
 | 
				
			||||||
 | 
					        'read': 5,
 | 
				
			||||||
 | 
					        'write': 15,
 | 
				
			||||||
 | 
					     })
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    table.throughput["read"].should.equal(5)
 | 
				
			||||||
 | 
					    table.throughput["write"].should.equal(15)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.update(throughput={
 | 
				
			||||||
 | 
					        'read': 5,
 | 
				
			||||||
 | 
					        'write': 6,
 | 
				
			||||||
 | 
					     })
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    table.describe()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.throughput["read"].should.equal(5)
 | 
				
			||||||
 | 
					    table.throughput["write"].should.equal(6)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_item_add_and_describe_and_update():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    ok = table.put_item(data={
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'subject': 'Check this out!',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					     })
 | 
				
			||||||
 | 
					    ok.should.equal(True)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    table.get_item(forum_name="LOLCat Forum",subject='Check this out!').should_not.be.none
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    returned_item = table.get_item(
 | 
				
			||||||
 | 
					        forum_name='LOLCat Forum',
 | 
				
			||||||
 | 
					        subject='Check this out!'
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    dict(returned_item).should.equal({
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'subject': 'Check this out!',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    returned_item['SentBy'] = 'User B'
 | 
				
			||||||
 | 
					    returned_item.save(overwrite=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    returned_item = table.get_item(
 | 
				
			||||||
 | 
					        forum_name='LOLCat Forum',
 | 
				
			||||||
 | 
					        subject='Check this out!'
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    dict(returned_item).should.equal({
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'subject': 'Check this out!',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User B',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_item_put_without_table():
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table = Table('undeclared-table')
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item =Item(table,item_data)   
 | 
				
			||||||
 | 
					    item.save.when.called_with().should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_missing_item():
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.get_item.when.called_with(
 | 
				
			||||||
 | 
					        hash_key='tester',
 | 
				
			||||||
 | 
					        range_key='other',
 | 
				
			||||||
 | 
					    ).should.throw(ValidationException)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_item_with_undeclared_table():
 | 
				
			||||||
 | 
					    table = Table('undeclared-table')
 | 
				
			||||||
 | 
					    table.get_item.when.called_with(test_hash=3241526475).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_item_without_range_key():
 | 
				
			||||||
 | 
					    table = Table.create('messages', schema=[
 | 
				
			||||||
 | 
					        HashKey('test_hash'),
 | 
				
			||||||
 | 
					        RangeKey('test_range'),
 | 
				
			||||||
 | 
					    ], throughput={
 | 
				
			||||||
 | 
					        'read': 10,
 | 
				
			||||||
 | 
					        'write': 10,
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    hash_key = 3241526475
 | 
				
			||||||
 | 
					    range_key = 1234567890987
 | 
				
			||||||
 | 
					    table.put_item( data = {'test_hash':hash_key, 'test_range':range_key})
 | 
				
			||||||
 | 
					    table.get_item.when.called_with(test_hash=hash_key).should.throw(ValidationException)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_delete_item():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item =Item(table,item_data)
 | 
				
			||||||
 | 
					    item['subject'] = 'Check this out!'        
 | 
				
			||||||
 | 
					    item.save()
 | 
				
			||||||
 | 
					    table.count().should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    response = item.delete()
 | 
				
			||||||
 | 
					    response.should.equal(True)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    table.count().should.equal(0)
 | 
				
			||||||
 | 
					    item.delete.when.called_with().should.throw(ConditionalCheckFailedException)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_delete_item_with_undeclared_table():
 | 
				
			||||||
 | 
					    conn = boto.connect_dynamodb()
 | 
				
			||||||
 | 
					    table = Table("undeclared-table")
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item =Item(table,item_data)
 | 
				
			||||||
 | 
					    item.delete.when.called_with().should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_query():
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        'subject': 'Check this out!' 
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item =Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save(overwrite=True)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item['subject'] = '456'
 | 
				
			||||||
 | 
					    item.save(overwrite=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item['subject'] = '123'
 | 
				
			||||||
 | 
					    item.save(overwrite=True)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item['subject'] = '789'
 | 
				
			||||||
 | 
					    item.save(overwrite=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.count().should.equal(4)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.query(forum_name__eq='the-key', subject__gt='1',consistent=True)
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.query(forum_name__eq='the-key', subject__gt='234',consistent=True)
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(2)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    results = table.query(forum_name__eq='the-key', subject__gt='9999')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(0)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    results = table.query(forum_name__eq='the-key', subject__beginswith='12')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    results = table.query(forum_name__eq='the-key', subject__beginswith='7')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.query(forum_name__eq='the-key', subject__between=['567', '890'])
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_query_with_undeclared_table():
 | 
				
			||||||
 | 
					    table = Table('undeclared')
 | 
				
			||||||
 | 
					    results = table.query(
 | 
				
			||||||
 | 
					        forum_name__eq='Amazon DynamoDB',
 | 
				
			||||||
 | 
					        subject__beginswith='DynamoDB',
 | 
				
			||||||
 | 
					        limit=1
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    iterate_results.when.called_with(results).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_scan():
 | 
				
			||||||
 | 
					    table = create_table()    
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item_data['subject'] = '456'
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item = Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save()    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item['subject'] = '123'
 | 
				
			||||||
 | 
					    item.save()
 | 
				
			||||||
 | 
					   
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User B',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:09 PM',
 | 
				
			||||||
 | 
					        'Ids': set([1, 2, 3]),
 | 
				
			||||||
 | 
					        'PK': 7,
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item_data['subject'] = '789'
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item = Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save()    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan()
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(SentBy__eq='User B')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(Body__beginswith='http')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(Ids__null=False)
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    results = table.scan(Ids__null=True)
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(2)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    results = table.scan(PK__between=[8, 9])
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(PK__between=[5, 8])
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_scan_with_undeclared_table():
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					    conn.scan.when.called_with(
 | 
				
			||||||
 | 
					        table_name='undeclared-table',
 | 
				
			||||||
 | 
					        scan_filter={
 | 
				
			||||||
 | 
					            "SentBy": {
 | 
				
			||||||
 | 
					                "AttributeValueList": [{
 | 
				
			||||||
 | 
					                    "S": "User B"}
 | 
				
			||||||
 | 
					                ],
 | 
				
			||||||
 | 
					                "ComparisonOperator": "EQ"
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    ).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_write_batch():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    with table.batch_write() as batch:
 | 
				
			||||||
 | 
					        batch.put_item(data={
 | 
				
			||||||
 | 
					            'forum_name': 'the-key',
 | 
				
			||||||
 | 
					            'subject': '123',
 | 
				
			||||||
 | 
					            'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					            'SentBy': 'User A',
 | 
				
			||||||
 | 
					            'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        })  
 | 
				
			||||||
 | 
					        batch.put_item(data={
 | 
				
			||||||
 | 
					            'forum_name': 'the-key',
 | 
				
			||||||
 | 
					            'subject': '789',
 | 
				
			||||||
 | 
					            'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					            'SentBy': 'User B',
 | 
				
			||||||
 | 
					            'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        }) 
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					    table.count().should.equal(2)
 | 
				
			||||||
 | 
					    with table.batch_write() as batch:
 | 
				
			||||||
 | 
					        batch.delete_item(
 | 
				
			||||||
 | 
					            forum_name='the-key',
 | 
				
			||||||
 | 
					            subject='789'
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.count().should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_batch_read():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item_data['subject'] = '456'
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item = Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save()    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item = Item(table,item_data) 
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    item_data['subject'] = '123'
 | 
				
			||||||
 | 
					    item.save() 
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User B',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        'Ids': set([1, 2, 3]),
 | 
				
			||||||
 | 
					        'PK': 7,
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item = Item(table,item_data) 
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'another-key'
 | 
				
			||||||
 | 
					    item_data['subject'] = '789'
 | 
				
			||||||
 | 
					    item.save() 
 | 
				
			||||||
 | 
					    results = table.batch_get(keys=[
 | 
				
			||||||
 | 
					                {'forum_name': 'the-key', 'subject': '123'},
 | 
				
			||||||
 | 
					                {'forum_name': 'another-key', 'subject': '789'}])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # Iterate through so that batch_item gets called
 | 
				
			||||||
 | 
					    count = len([x for x in results])
 | 
				
			||||||
 | 
					    count.should.equal(2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_key_fields():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    kf = table.get_key_fields()
 | 
				
			||||||
 | 
					    kf.should.equal(['forum_name','subject'])
 | 
				
			||||||
							
								
								
									
										361
									
								
								tests/test_dynamodb2/test_dynamodb_table_without_range_key.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										361
									
								
								tests/test_dynamodb2/test_dynamodb_table_without_range_key.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,361 @@
 | 
				
			|||||||
 | 
					import boto
 | 
				
			||||||
 | 
					import sure  # noqa
 | 
				
			||||||
 | 
					from freezegun import freeze_time
 | 
				
			||||||
 | 
					from boto.exception import JSONResponseError
 | 
				
			||||||
 | 
					from moto import mock_dynamodb2
 | 
				
			||||||
 | 
					from boto.dynamodb2.fields import HashKey
 | 
				
			||||||
 | 
					from boto.dynamodb2.fields import RangeKey
 | 
				
			||||||
 | 
					from boto.dynamodb2.table import Table
 | 
				
			||||||
 | 
					from boto.dynamodb2.table import Item
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def create_table():
 | 
				
			||||||
 | 
					    table = Table.create('messages', schema=[
 | 
				
			||||||
 | 
					        HashKey('forum_name')
 | 
				
			||||||
 | 
					    ], throughput={
 | 
				
			||||||
 | 
					        'read': 10,
 | 
				
			||||||
 | 
					        'write': 10,
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					    return table
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@freeze_time("2012-01-14")
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_create_table():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    expected = {
 | 
				
			||||||
 | 
					        'Table': {
 | 
				
			||||||
 | 
					            'AttributeDefinitions': [
 | 
				
			||||||
 | 
					                {'AttributeName': 'forum_name', 'AttributeType': 'S'}    
 | 
				
			||||||
 | 
					            ], 
 | 
				
			||||||
 | 
					            'ProvisionedThroughput': {
 | 
				
			||||||
 | 
					                'NumberOfDecreasesToday': 0, 'WriteCapacityUnits': 10, 'ReadCapacityUnits': 10
 | 
				
			||||||
 | 
					                }, 
 | 
				
			||||||
 | 
					            'TableSizeBytes': 0, 
 | 
				
			||||||
 | 
					            'TableName': 'messages', 
 | 
				
			||||||
 | 
					            'TableStatus': 'ACTIVE', 
 | 
				
			||||||
 | 
					            'KeySchema': [
 | 
				
			||||||
 | 
					                {'KeyType': 'HASH', 'AttributeName': 'forum_name'} 
 | 
				
			||||||
 | 
					            ], 
 | 
				
			||||||
 | 
					            'ItemCount': 0, 'CreationDateTime': 1326499200.0
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    conn =  boto.dynamodb2.connect_to_region(
 | 
				
			||||||
 | 
					            'us-west-2',
 | 
				
			||||||
 | 
					        aws_access_key_id="ak",
 | 
				
			||||||
 | 
					        aws_secret_access_key="sk")
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    conn.describe_table('messages').should.equal(expected)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_delete_table():
 | 
				
			||||||
 | 
					    create_table()
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					    conn.list_tables()["TableNames"].should.have.length_of(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    conn.delete_table('messages')
 | 
				
			||||||
 | 
					    conn.list_tables()["TableNames"].should.have.length_of(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    conn.delete_table.when.called_with('messages').should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_update_table_throughput():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    table.throughput["read"].should.equal(10)
 | 
				
			||||||
 | 
					    table.throughput["write"].should.equal(10)    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.update(throughput={
 | 
				
			||||||
 | 
					        'read': 5,
 | 
				
			||||||
 | 
					        'write': 6,
 | 
				
			||||||
 | 
					     })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.throughput["read"].should.equal(5)
 | 
				
			||||||
 | 
					    table.throughput["write"].should.equal(6)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_item_add_and_describe_and_update():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    data={
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					     }
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    table.put_item(data = data)
 | 
				
			||||||
 | 
					    returned_item = table.get_item(forum_name="LOLCat Forum")
 | 
				
			||||||
 | 
					    returned_item.should_not.be.none
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    dict(returned_item).should.equal({
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    returned_item['SentBy'] = 'User B'
 | 
				
			||||||
 | 
					    returned_item.save(overwrite=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    returned_item = table.get_item(
 | 
				
			||||||
 | 
					          forum_name='LOLCat Forum'
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    dict(returned_item).should.equal({
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User B',
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_item_put_without_table():
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    conn.put_item.when.called_with(
 | 
				
			||||||
 | 
					        table_name='undeclared-table',
 | 
				
			||||||
 | 
					        item={
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    ).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_missing_item():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.get_item.when.called_with(test_hash=3241526475).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_item_with_undeclared_table():
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    conn.get_item.when.called_with(
 | 
				
			||||||
 | 
					        table_name='undeclared-table',
 | 
				
			||||||
 | 
					        key={"forum_name": {"S": "LOLCat Forum"}},
 | 
				
			||||||
 | 
					    ).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_delete_item():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'forum_name': 'LOLCat Forum',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item =Item(table,item_data)
 | 
				
			||||||
 | 
					    item.save()
 | 
				
			||||||
 | 
					    table.count().should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    response = item.delete()
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    response.should.equal(True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.count().should.equal(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item.delete.when.called_with().should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_delete_item_with_undeclared_table():
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    conn.delete_item.when.called_with(
 | 
				
			||||||
 | 
					        table_name='undeclared-table',
 | 
				
			||||||
 | 
					        key={"forum_name": {"S": "LOLCat Forum"}},
 | 
				
			||||||
 | 
					    ).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_query():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'forum_name': 'the-key',
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item =Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save(overwrite = True)
 | 
				
			||||||
 | 
					    table.count().should.equal(1)
 | 
				
			||||||
 | 
					    table = Table("messages")
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    results = table.query(forum_name__eq='the-key')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_query_with_undeclared_table():
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    conn.query.when.called_with(
 | 
				
			||||||
 | 
					        table_name='undeclared-table',
 | 
				
			||||||
 | 
					         key_conditions= {"forum_name": {"ComparisonOperator": "EQ", "AttributeValueList": [{"S": "the-key"}]}}
 | 
				
			||||||
 | 
					    ).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_scan():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key'
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    item = Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save()    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item['forum_name'] = 'the-key2'
 | 
				
			||||||
 | 
					    item.save(overwrite=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User B',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        'Ids': set([1, 2, 3]),
 | 
				
			||||||
 | 
					        'PK': 7,
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key3'
 | 
				
			||||||
 | 
					    item = Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save() 
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan()
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(SentBy__eq='User B')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(Body__beginswith='http')
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(Ids__null=False)
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(Ids__null=True)
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(PK__between=[8, 9])
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    results = table.scan(PK__between=[5, 8])
 | 
				
			||||||
 | 
					    sum(1 for _ in results).should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_scan_with_undeclared_table():
 | 
				
			||||||
 | 
					    conn = boto.dynamodb2.layer1.DynamoDBConnection()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    conn.scan.when.called_with(
 | 
				
			||||||
 | 
					        table_name='undeclared-table',
 | 
				
			||||||
 | 
					        scan_filter={
 | 
				
			||||||
 | 
					            "SentBy": {
 | 
				
			||||||
 | 
					                "AttributeValueList": [{
 | 
				
			||||||
 | 
					                    "S": "User B"}
 | 
				
			||||||
 | 
					                ],
 | 
				
			||||||
 | 
					                "ComparisonOperator": "EQ"
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    ).should.throw(JSONResponseError)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_write_batch():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    with table.batch_write() as batch:
 | 
				
			||||||
 | 
					        batch.put_item(data={
 | 
				
			||||||
 | 
					            'forum_name': 'the-key',
 | 
				
			||||||
 | 
					            'subject': '123',
 | 
				
			||||||
 | 
					            'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					            'SentBy': 'User A',
 | 
				
			||||||
 | 
					            'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        })  
 | 
				
			||||||
 | 
					        batch.put_item(data={
 | 
				
			||||||
 | 
					            'forum_name': 'the-key2',
 | 
				
			||||||
 | 
					            'subject': '789',
 | 
				
			||||||
 | 
					            'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					            'SentBy': 'User B',
 | 
				
			||||||
 | 
					            'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        }) 
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.count().should.equal(2)
 | 
				
			||||||
 | 
					    with table.batch_write() as batch:
 | 
				
			||||||
 | 
					        batch.delete_item(
 | 
				
			||||||
 | 
					            forum_name='the-key',
 | 
				
			||||||
 | 
					            subject='789'
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    table.count().should.equal(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_batch_read():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key1'
 | 
				
			||||||
 | 
					    item = Item(table,item_data)     
 | 
				
			||||||
 | 
					    item.save()  
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item = Item(table,item_data)
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'the-key2'
 | 
				
			||||||
 | 
					    item.save(overwrite = True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    item_data = {
 | 
				
			||||||
 | 
					        'Body': 'http://url_to_lolcat.gif',
 | 
				
			||||||
 | 
					        'SentBy': 'User B',
 | 
				
			||||||
 | 
					        'ReceivedTime': '12/9/2011 11:36:03 PM',
 | 
				
			||||||
 | 
					        'Ids': set([1, 2, 3]),
 | 
				
			||||||
 | 
					        'PK': 7,
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    item = Item(table,item_data) 
 | 
				
			||||||
 | 
					    item_data['forum_name'] = 'another-key'
 | 
				
			||||||
 | 
					    item.save(overwrite = True)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    results = table.batch_get(keys=[
 | 
				
			||||||
 | 
					                {'forum_name': 'the-key1'},
 | 
				
			||||||
 | 
					                {'forum_name': 'another-key'}])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # Iterate through so that batch_item gets called
 | 
				
			||||||
 | 
					    count = len([x for x in results])
 | 
				
			||||||
 | 
					    count.should.equal(2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_key_fields():
 | 
				
			||||||
 | 
					    table = create_table()
 | 
				
			||||||
 | 
					    kf = table.get_key_fields()
 | 
				
			||||||
 | 
					    kf[0].should.equal('forum_name')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@mock_dynamodb2
 | 
				
			||||||
 | 
					def test_get_special_item():
 | 
				
			||||||
 | 
					    table = Table.create('messages', schema=[
 | 
				
			||||||
 | 
					        HashKey('date-joined')
 | 
				
			||||||
 | 
					    ], throughput={
 | 
				
			||||||
 | 
					        'read': 10,
 | 
				
			||||||
 | 
					        'write': 10,
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    data={
 | 
				
			||||||
 | 
					        'date-joined': 127549192,
 | 
				
			||||||
 | 
					        'SentBy': 'User A',
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    table.put_item(data = data)
 | 
				
			||||||
 | 
					    returned_item = table.get_item(**{'date-joined': 127549192})
 | 
				
			||||||
 | 
					    dict(returned_item).should.equal(data)
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
							
								
								
									
										18
									
								
								tests/test_dynamodb2/test_server.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								tests/test_dynamodb2/test_server.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,18 @@
 | 
				
			|||||||
 | 
					import sure  # noqa
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import moto.server as server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					'''
 | 
				
			||||||
 | 
					Test the different server responses
 | 
				
			||||||
 | 
					'''
 | 
				
			||||||
 | 
					server.configure_urls("dynamodb2")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def test_table_list():
 | 
				
			||||||
 | 
					    test_client = server.app.test_client()
 | 
				
			||||||
 | 
					    res = test_client.get('/')
 | 
				
			||||||
 | 
					    res.status_code.should.equal(404)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    headers = {'X-Amz-Target': 'TestTable.ListTables'}
 | 
				
			||||||
 | 
					    res = test_client.get('/', headers=headers)
 | 
				
			||||||
 | 
					    res.data.should.contain('TableNames')
 | 
				
			||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user