Skip to content
Alex Browne edited this page May 4, 2015 · 6 revisions

This wiki page describes how Zoom works under the hood. In particular, it describes what commands Zoom sends to Redis when you call Save, Find, etc. and how indexes and queries work.

Basic Operations

Save

When you call Save, a few things happen:

First, if the model does not have an id, Zoom randomly generates one and assigns it via the SetId method. The id is generated here and consists of the current unix time with millisecond precision (encoded in base-36) and a randomly generated string that is 16 characters long.

Second, the exported fields in the struct are converted to a form Redis can understand, namely they are all converted to a slice of bytes. Primitive fields (string, int, uint, float64, etc) are converted via the redis driver. Non-nil pointer fields are dereferenced and their underlying values are stored. Nil pointer fields are stored as "NULL". All other fields are converted to binary encoding via the gob package. The conversion occurs here. Once all the fields are converted, they are stored in Redis hash via the HMSET command. The key used for the model hash is exposed via the ModelKey method.

Third, the model id is added to a set of all models for a specific type via the SADD command. The key used for all the model ids is exposed via the AllIndexKey method. This set is used for the FindAll method as well as a starting point for unordered queries.

Fourth, any indexed fields identified by the struct tag zoom:"index" will be indexed and stored in a sorted set. The key used for the sorted set for each field index is exposed via the FieldIndexKey method. See below for more information about how indexes work.

Here is a more concrete example. Let's say you have a Person type:

type Person struct {
    Name string
    Age  int     `zoom:"index"`
    zoom.DefaultData
}

And you create and save a new person like so:

person := &Person{
    Name: "Bob",
    Age: 27,
}
if err := zoom.Save(person); err != nil {
    // handle err
}

When you call Save, a random id will be generated for the model. For the sake of simplicity let's say the id that was generated was "foo". Here are the Redis commands that will be run:

# Add the fields to the main model hash
HMSET Person:foo Name Bob Age 27
# Add the model id to the set of all person ids
SADD Person:all foo
# Index the Age field in a sorted set
ZADD Person:Age 27 foo

You can change the name that Zoom uses for a given field with the redis struct tag. So if we wanted the Age field to be stored as a lowercase field in Redis, we could use the following:

type Person struct {
    Name string
    Age  int     `redis:"age"`
    zoom.DefaultData
}

Using the struct tag redis:"-" has special meaning and tells Zoom that you do not want the field to be stored in Redis at all.

Find

You might be able to guess how find works based on the description of Save. Find simply gets all the fields from the hash in Redis and scans them into a model type. As a more concrete example, the following code:

p := &Person{}
if err := Persons.Find("foo", p); err != nil {
    // handle error
}

Will result in the following Redis command:

HMGET Person:foo Name Age

Then Zoom will simply convert the reply from Redis back into the types of the original fields and set the field values using reflection.

Delete

Again, the inner-workings of Delete might be obvious if you understand Save. The following happens when you delete a model:

  1. The main hash for the model is deleted with the DEL command.
  2. The model is removed from the set of all model ids for a specific type.
  3. The model is removed from any field indexes.

So if you run the following code:

if _, err := Persons.Delete("foo"); err != nil {
    // handle error
}

Here are the Redis commands that will be executed:

DEL Person:foo
SREM Person:all foo
ZREM Person:Age foo

Queries and Indexes

For these examples we're going to define a new model type:

type IndexedModel struct {
    Int    int    `zoom:"index"`
    Bool   bool   `zoom:"index"`
    String string `zoom:"index"`
}

Numeric Indexes

You already have seen a little bit about how numeric indexes work. Zoom stores numeric indexes as a sorted set, where each score corresponds to a field value, and each member is the id of some model. So if we saved a model like so:

model := &IndexedModel{
    Int: 5,
}
if err := IndexedModels.Save(model); err != nil {
   // handle error
}

The index would be saved with the following Redis command:

ZADD IndexedModel:Int 5 foo

Note that we are again pretending that the id assigned to the model was "foo" for simplicity. The same basic rule applies for other types of numeric fields, such as floats and uints.

Bool Indexes

Indexes on bool fields work very similar to numeric fields. There is just an extra conversion step where true is converted to a score of 1 and false is converted to a score of 0. So if we saved a model with the Bool field set like so:

model := &IndexedModel{
    Bool: true,
}
if err := IndexedModels.Save(model); err != nil {
   // handle error
}

The bool index would be saved with the following Redis command:

ZADD IndexedModel:Bool 1 foo

String Indexes

String indexes are a little different, because we cannot use strings as scores for sorted sets in Redis. So instead, Zoom relies on a workaround. String indexes are stored as sorted sets where all the scores are 0 and each member has the following format: value\0id, where \0 is the NULL character. So if we saved a model with the String field set like so:

model := &IndexedModel{
    String: "bar",
}
if err := IndexedModels.Save(model); err != nil {
   // handle error
}

The string index would be saved with the following Redis command:

ZADD IndexedModel:String 0 "bar\x00foo"
Clone this wiki locally