µWeb documentation - Model

The µWeb framework provides a model module with the intention of simplifying database access. The design goal is to provide a rich abstraction that

Making database interaction easier without restricting the abilities of the developer is our main goal. Some default mechanisms make assumptions on the way the database is organized, but these are well-documented, and it's entirely possible to change the behavior of these mechanisms.

Record

The basic idea of the Record class is that it is a container for your database records, with related records automatically loaded as needed, and custom methods that provide more info, child objects, etc. Outlined below are the default features available, with minimal configuration requirements.

Your first Record class

To create your own Record subclass, nothing is required beyond the class' name. The following example substitutes a complete working example:

1from uweb import model
2class Message(model.Record):
3  """Abstraction class for messages stored in the database."""

Loading fields from primary key

The Record class comes loaded with a way to load records from your database using the FromPrimary method. This is a classmethod available on the Record class and all your own subclasses, and when given a connection and primary key value, will load that record from the database. Provided you have a database that looks like this:

1-- TABLE `message`
2+----+--------+--------------------------------------------------+
3| ID | author | message                                          |
4+----+--------+--------------------------------------------------+
5|  1 | Elmer  | First message!                                   |
6|  2 | Bobby  | Robert'); DROP TABLE Students;--                 |
7|  3 | Elmer  | You didn't think it would be this easy, did you? |
8+----+--------+--------------------------------------------------+

You can load data from this table with the following code:

 1# The model:
 2from uweb import model
 3class Message(model.Record):
 4  """Abstraction class for messages stored in the database."""
 5
 6# Using this:
 7>>> message = Message.FromPrimary(db_conn, 1)
 8>>> print message
 9Message({'message': u'First message!', 'ID': 1L, 'author': u'Elmer'})

Changing the primary key field

By default, Record uses a primary key called 'ID'. You can change this to any value you like, and FromPrimary will automatically work based on that value, and all other methods and functionality of the class will also use this new definition (deleting, creating and auto-loading from related tables, which are all explained later).

To change the primary key field, create a class with a defined _PRIMARY_KEY class variable:

1from uweb import model
2class Country(model.Record):
3  """Abstraction class for a country table.
4
5  This class uses the ISO-3166-1 alpha2 country code as primary key.
6  """
7  _PRIMARY_KEY = 'alpha2'

Compound primary keys

The µWeb model also supports compound primary keys, with one limitation: AUTO_INCREMENT fields are not supported for creation of the Record, all values need to be provided for it.

Loading values from a compound primary keys works by passing a tuple instead of a single value:

 1# The model:
 2from uweb import model
 3class MonthReport(model.Record):
 4  """Abstraction class for the monthReport table.
 5
 6  This is keyed on a composite of both year and month, foregoing the need for a separate AUTO_INCREMENT field.
 7  """
 8  _PRIMARY_KEY = 'year', 'month'
 9
10# Using this:
11>>> report = MonthReport.FromPrimary(db_conn, (2012, 5))
12>>> print report
13Message({'report': 'Things went really well', 'month': 5, 'year': 2012})

Class and table relation

By default, the assumption is made that the table name is the same as the class name, with the first letter lowercase. The table related to the class Message would be message. To change this behavior, assign your own table name to the _TABLE class constant. This new table name will then be used in all built-in Record methods:

1from uweb import model
2class Message(model.Record):
3  """Abstraction class for messages stored in the database."""
4  _TABLE = 'MyMessage'

Alternatively, you can override the TableName class-method to alter the table-name transformation that is done.

Creating records

To create a record in the database, you can use the classmethod Create. This takes the connection and a dictionary of the keys and values that should be inserted into the database. Using the Message class we defined earlier, creating a new record is a relatively simple call:

1>>> message = Message.Create(db_conn, {'author': 'Bob', 'message': 'Another message'})
2>>> print message
3Message({'message': 'Another message', 'ID': 4L, 'author': 'Bob'})

N.B. Skipping fields that are optional in the database is allowed, but their default values assigned by the database will not be reflected in the object. That is, the record will not be reloaded after storing. The primary

Deleting records

Records can be deleted from the database either from a loaded object, or using the DeletePrimary classmethod. This latter removes the record from the database using the primary key to select it.

 1class Message(model.Record):
 2  """Abstraction class for messages records."""
 3
 4# Loading and deleting an active record.
 5>>> bad_record = Message.FromPrimary(db_connection, 3)
 6>>> bad_record.Delete()
 7
 8# Deleting a record based on its primary key.
 9>>> Message.DeletePrimary(db_connection, 2)

Listing all records

For situations where all records must be retrieved or processed, there is the List classmethod. This takes the connection as argument and iterates over all records in the database:

 1class Message(model.Record):
 2  """Abstraction class for messages records."""
 3
 4# List all messages:
 5>>> for message in Message.List(db_connection):
 6...   print message
 7...
 8Message({'message': u'First message!', 'ID': 1L, 'author': 1})
 9Message({'message': u"Robert'); DROP TABLE Students;--", 'ID': 2L, 'author': 2})
10Message({'message': u"You didn't think it would be this easy, did you?", 'ID': 3L, 'author': 1})

On-demand loading of referenced records.

In databases that are more complex than a single table (nearly ''all''), information is often normalized. That is, the author information in our previously demonstrated message table will be stored in a separate author table. The author field on message records will be a reference to a record in the author table.

Consider the following tables in your database:

 1-- TABLE `message`
 2+----+--------+--------------------------------------------------+
 3| ID | author | message                                          |
 4+----+--------+--------------------------------------------------+
 5|  1 |      1 | First message!                                   |
 6|  2 |      2 | Robert'); DROP TABLE Students;--                 |
 7|  3 |      1 | You didn't think it would be this easy, did you? |
 8+----+--------+--------------------------------------------------+
 9
10-- TABLE `author`
11+----+-------+--------------------+
12| ID | name  | emailAddress       |
13+----+-------+--------------------+
14|  1 | Elmer | elmer@underdark.nl |
15|  2 | Bobby | bobby@tables.com   |
16+----+-------+--------------------+

And the following class definitions in Python:

1from uweb import model
2class Author(model.Record):
3  """Abstraction class for author records."""
4
5class Message(model.Record):
6  """Abstraction class for messages records."""

This makes it possible to retrieve a message, and from that Message object, retrieve the author information. This is done when the information is requested, and not pre-loaded beforehand. This means that retrieving a thousand Message objects will not trigger an additional 1000 queries to retrieve the author information, if that information might not be used at all.

 1>>> message = Message.FromPrimary(db_conn, 1)
 2>>> message
 3Message({'message': u'First message!', 'ID': 1L, 'author': 1})
 4# This is the same message we saw before, without author information.
 5# However, retrieving the author field specifically, provides its record:
 6>>> message['author']
 7Author({'emailAddress': u'elmer@underdark.nl', 'ID': 1, 'name': u'Elmer'})
 8>>> message
 9Message({'message': u'First message!', 'ID': 1L,
10         'author': Author({'emailAddress': u'elmer@underdark.nl', 'ID': 1, 'name': u'Elmer'})})

This works on the assumption that any field name that is also the table name of another Record class, is a reference to that table. In the case of the example above: The message table contains a field author. There exists a Record subclass for that table (namely Author, table 'author'). The value of message['author'] (=1), is now used to load an Author record using the FromPrimary classmethod, with 1 as the primary key value.

  1. message['author'] uses the author field
  2. author table is represented by Author class
  3. message['author'] is replaced by Author.FromPrimary(db_connection, message['author']

Customize table-references

The auto-loading behavior can be modified using the _FOREIGN_RELATIONS class constant. This provides a mapping that specifies (and overrides) which Record classes should be used to resolve references from fields. The key for the mapping is a field name (string), and the corresponding value can be a class or None.

The following is an example case where the table names are plural, but the field names are singular:

 1from uweb import model
 2class Author(model.Record):
 3  """Abstraction class for author records."""
 4  _TABLE = 'authors'
 5
 6class Message(model.Record):
 7  """Abstraction class for messages records."""
 8  _TABLE = 'messages'
 9  _FOREIGN_RELATIONS = {'author': Author}

Loading child objects (1-to-n relations)

The model provides a generic method to retrieve child records (that is, 1 to n relations) of a record. The desired relations should have an associated Record class. The method to use is _Children, which is a private method of any Record class. As its argument, it needs the name of a child class. Returned is an iterator that yields instances of the given Record subclass.

Given its name and usage, the suggested usage of this is to wrap a more descriptive method around this:

 1from uweb import model
 2class Author(model.Record):
 3  """Abstraction class for author records."""
 4  def Messages(self):
 5    """Returns an iterator for all messages written by this author."""
 6    return self._Children(Message)
 7
 8class Message(model.Record):
 9  """Abstraction class for messages records."""
10
11# Caller code
12>>> elmer = Author.FromPrimary(db_connection, 1)
13>>> for message in elmer.Messages():
14...   print message
15Message({'message': u'First message!', 'ID': 1L,
16         'author': Author({'emailAddress': u'elmer@underdark.nl', 'ID': 1, 'name': u'Elmer'})})
17Message({'message': u"You didn't think it would be this easy, did you?", 'ID': 3L,
18         'author': Author({'emailAddress': u'elmer@underdark.nl', 'ID': 1, 'name': u'Elmer'})})
19# Reflowing to keep things legible

What you can see here is that all messages written by the given author are retrieved from the database, and presented. This is done with a single database query, where the child Record's table is searched for rows where the relation_field is equal to the parent Record's primary key value. This relation_field is an optional argument to the _Children method, and defaults to the class' table name.

N.B. print and the methods (iter)items, (iter)values all cause the object's foreign relations to be retrieved.

The same example, this time with pluralized table names:

 1class Author(model.Record):
 2  """Abstraction class for author records."""
 3  _TABLE = 'authors'
 4
 5  def Messages(self):
 6    """Returns an iterator for all messages written by this author."""
 7    return self._Children(Message, relation_field='author')
 8
 9class Message(model.Record):
10  """Abstraction class for messages records."""
11  _TABLE = 'messages'
12  _FOREIGN_RELATIONS = {'author': Author}

Updating a record

After loading a record, it can be altered, and saved. These changes (and optionally changes to nested records), will be committed to the database, and reflected in the current loaded record.

 1class Author(model.Record):
 2  """Abstraction class for author records."""
 3
 4class Message(model.Record):
 5  """Abstraction class for messages records."""
 6
 7>>> retort = Message.FromPrimary(db_connection, 3)
 8>>> retort['message'] = "Please go away Bobby."
 9>>> # Our changes are not yet reflected in the database:
10>>> print Message.FromPrimary(db_connection, 3)
11Message({'message': u"You didn't think it would be this easy, did you?", 'ID': 3L,
12         'author': Author({'emailAddress': u'elmer@underdark.nl', 'ID': 1, 'name': u'Elmer'})})
13>>> retort.Save()
14>>> # Now our changes are committed to the database:
15>>> print Message.FromPrimary(db_connection, 3)
16Message({'message': u'Please go away Bobby.', 'ID': 3L,
17         'author': Author({'emailAddress': u'elmer@underdark.nl', 'ID': 1, 'name': u'Elmer'})})

To save all changes in related fields, we can provide the named argument save_foreign and set it to True. This way we could alter both the author name and the message itself in one database transaction.

Comparisons

Equality

Records must pass the following criteria to be considered equal to one another.:
  1. Type: Two objects must be of the same type (class)
  2. Primary key: The primary key values must compare equal
  3. Foreign relations: Foreign relations must be the same. If these are not resolved in one object but are in the other, the primary key of the resolved object will be compared to the data of the other record.
  4. Data: All remaining data fields must be equal and symmetric (i.e. both objects describe the same fields)

Greater / smaller

Comparing two objects with one another to tell their relative order can only be done if they are of the same type. If they are, the comparison is done based on the primary key values of the records. In most cases this will result in an ordering similar to the database-insert order.

VersionedRecord

MongoRecord

blog comments powered by Disqus