Data Types

DynamoDB supports three different data types: STRING, NUMBER, and BINARY. It also supports sets of these types: STRING_SET, NUMBER_SET, BINARY_SET.

You can use these values directly for the model declarations, though they require an import:

from flywheel import Model, Field, STRING, NUMBER

class Tweet(Model):
    userid = Field(data_type=STRING, hash_key=True)
    id = Field(data_type=STRING, range_key=True)
    ts = Field(data_type=NUMBER, index='ts-index')
    text = Field(data_type=STRING)

There are other settings for data_type that are represented by python primitives. Some of them (like unicode) are functionally equivalent to the DynamoDB option (STRING). Others, like int, enforce an additional application-level constraint on the data. Each option works transparently, so a datetime field would be set with datetime objects and you could query against it using other datetime‘s.

Below is a table of python types, how they are stored in DynamoDB, and any special notes. For more information, the code for data types is located in types.

Type Dynamo Type Description
unicode STRING Basic STRING type. This is the default for fields
str BINARY  
int NUMBER Enforces integer constraint on data
float NUMBER  
set *_SET This will use the appropriate type of DynamoDB set
bool NUMBER  
datetime NUMBER datetimes MUST be provided in UTC
date NUMBER dates MUST be provided in UTC
Decimal NUMBER If you need decimal precision in your application
dict STRING Stored as json-encoded string
list STRING Stored as json-encoded string
S3Type STRING Stores the S3 key path as a string

If you attempt to set a field with a type that doesn’t match, it will raise a TypeError. If a field was created with coerce=True it will first attempt to convert the value to the correct type. This means you could set an int field with the value "123" and it would perform the conversion for you.

Note

Certain fields will auto-coerce specific data types. For example, a str field will auto-encode a unicode to utf-8 even if coerce=False. Similarly, a unicode field will auto-decode a str value to a unicode string.

Warning

If an int field is set to coerce values, it will still refuse to drop floating point data. This has the following effect:

>>> class Game(Model):
...    title = Field(hash_key=True)
...    points = Field(data_type=int, coerce=True)

>>> mygame = Game()
>>> mygame.points = 1.8
ValueError: Field 'points' refusing to convert 1.8 to int! Results in data loss!

Set types

If you define a set field with no additional parameters Field(data_type=set), flywheel will ensure that the field is a set, but will perform no type checking on the items within the set. This should work fine for basic uses when you are storing a number or string, but sets are able to contain any data type listed in the table above (and any custom type you declare). All you have to do is specify it in the data_type like so:

from flywheel import Model, Field, set_
from datetime import date

class Location(Model):
    name = Field(hash_key=True)
    events = Field(data_type=set_(date))

If you don’t want to import set_, you can use an equivalent expression with the python frozenset builtin:

events = Field(data_type=frozenset([date]))

Advanced Types

S3 Keys

You can use S3Type to quickly and easily reference S3 values from your model objects. This type will store the S3 key in Dynamo and put a Key object in your model.

from flywheel.fields.types import S3Type

class Image(Model):
    user = Field(hash_key=True)
    name = Field(range_key=True)
    taken = Field(data_type=datetime, index='taken-index')
    data = Composite('user', 'name', data_type=S3Type('my_image_bucket'),
                     merge=lambda *a: '/'.join(a))

    def __init__(self, user, name, taken):
        self.user = user
        self.name = name
        self.taken = taken

You can use this class like so:

>>> img = Image('Rob', 'Big Sur.jpg', datetime.utcnow())
>>> img.data.set_contents_from_filename(img.name)
>>> engine.save(img)

It will store the image data in the S3 bucket named my_image_bucket and use the path Rob/Big Sur.jpg. See Composite Fields for more about how the key path is generated.

Custom Types

You can define your own custom data types and make them available across all of your models. All you need to do is create a subclass of TypeDefinition. Let’s make a type that will store any python object in pickled format.

from flywheel.fields.types import TypeDefinition, BINARY, Binary
import cPickle as pickle

class PickleType(TypeDefinition):
    data_type = pickle #  name you use to reference this type
    aliases = ['pickle'] # alternate names that reference this type
    ddb_data_type = BINARY # data type of the field in dynamo

    def coerce(self, value, force):
        # Perform no type checking because we can pickle ANYTHING
        return value

    def ddb_dump(self, value):
        # Pickle and convert to a Binary object for boto
        return Binary(pickle.dumps(value))

    def ddb_load(self, value):
        # Convert from a Binary object and unpickle
        return pickle.loads(value.value)

Now that you have your type definition, you can either use it directly in your code:

class MyModel(Model):
    myobj = Field(data_type=PickleType())

Or you can register it globally and reference it by its data_type or any aliases that were defined.

from flywheel.fields.types import register_type

register_type(PickleType)

class MyModel(Model):
    myobj = Field(data_type='pickle')
Read the Docs v: 0.1.3
Versions
latest
0.1.3
0.1.2
0.1.1
0.1.0
Downloads
On Read the Docs
Project Home
Builds

Free document hosting provided by Read the Docs.