Skip to content

Latest commit

 

History

History
363 lines (290 loc) · 9.82 KB

README.md

File metadata and controls

363 lines (290 loc) · 9.82 KB

Cocktail

Cute abstract database for Erlang inspired by KVS.

Installing

In rebar.config:

{cocktail, ".*", {git, "git://github.com/tapsters/cocktail", {tag, "master"}}}

Starting up

In order to start working with Cocktail you should call ctail:init/0, but some backends require to make additional function calls.

For example builtin backend for Mnesia requires to call ctail_mnesia:join/0 or ctail_mnesia:join/1.

After performing all required calls you will have ready to use schema in your database (database, tables and indexes).

Example:

1> ctail:init().
ok
2> ctail_mnesia:join().
ok

Choosing backend

You can specify backend you want use in sys.config:

[
  {cocktail, [
    {backend, ctail_mnesia},
  ]}
]

Coctail uses Mnesia by default.

List of available backends:

If you want create your own backend, you should implement ctail_backend behaviour.

Working with schema

Get tables list:

1> ctail:dir().
["feed","user"]

Destroy all tables:

1> ctail:destroy().
ok

Basic type for Cocktail is record. You should describe table for all your records. To do this you should implement ctail_schema behaviour:

-module(user).
-behaviour(ctail_schema).

-include_lib("cocktail/include/ctail.hrl").

-export([meta/0]).

meta() ->
  #schema{name=myschema, tables=[
    %% #table.name should be the same as record name
    #table{name=user, fields=record_info(fields, user), keys = [email]}
  ]}.

And then add your module to sys.config:

[
  {cocktail, [
    {schema, [user]},
  ]}
]

Backends implement two schema related functions: create_table/1 and add_table_index/2.

Raw operations

Cocktail provides you with following functions to work with raw data:

  • next_id/2
  • put/1
  • delete/2
  • get/2
  • index/3
  • all/1
  • count/1

When you persist new item to database you should specify id. Backends implement next_id/2 to handle it. Example:

ctail:put(#user{id=ctail:next_id(user,1), firstName="Bob"}). 

Also you can specify increment but it's ignored by some backends:

ctail:put(#user{id=ctail:next_id(user,1), firstName="Bob"}). 

Raw operations usage example:

1> ctail:put(#user{id=ctail:next_id(user,1), firstName="Bob", status=0}).
ok
2> ctail:put(#user{id=ctail:next_id(user,1), firstName="Fred", status=0}).
ok
3> ctail:put(#user{id=ctail:next_id(user,1), firstName="Pet", status=1}).
ok

4> JohnId = ctail:next_id(user, 1).
4
5> ctail:put(#user{id=JohnId, firstName="John"}).
ok
6> ctail:get(user, JohnId).
{ok,#user{id = 4,firstName = "John",lastName = undefined,
                 username = undefined,status = undefined}}
7> ctail:delete(user, JohnId).
ok

8> ctail:index(user, status, 0).
[#user{id = 1,firstName = "Bob",lastName = undefined,
              username = undefined,status = 0}
 #user{id = 2,firstName = "Fred",lastName = undefined,
              username = undefined,status = 0}]

9> ctail:all(user).
[#user{id = 1,firstName = "Bob",lastName = undefined,
              username = undefined,status = 0}
 #user{id = 2,firstName = "Fred",lastName = undefined,
              username = undefined,status = 0}
 #user{id = 3,firstName = "Pet",lastName = undefined,
              username = undefined,status = 1}}]

10> ctail:count(user).
3

Chain operations

There are two domains provided by Cocktail to work with chains (linked lists): container and iterator.

Container holds list's head and count (we call it 'tail'):

-define(CONTAINER, 
  id, 
  top, 
  count=0
).

-record(container, {?CONTAINER}).
-record(feed,      {?CONTAINER}).

You can store containers in you own table[s], but Coctail provides you with builtin feed table.

Iterator is like superclass for your records. It defines base fields and adds some fields needed by chain operations:

-define(ITERATOR(ContainerName), 
  id, 
  version,
  container=ContainerName, 
  feed_id, 
  prev, 
  next
).

-record(iterator,  {?ITERATOR(undefined)}).

In the above example ContainerName is name of the table where you want to store containers for this record. If #iterator.container equals to undefined, your record's 'tail' will be stored in builtin container (feed).

If you want to your record be ready to use in chain operations, you should include ITERATOR macro as first item in your record definition:

-record(message, {
  ?ITERATOR(feed),
  origin, 
  payload, 
  createdAt
}).

Cocktail provides you with following functions to work with chains:

  • create/2
  • add/1
  • link/1
  • feed/3
  • entries/4
  • remove/2

Chain operations usage example:

1> ctail:get(feed, {chat,1}).
{error,not_found}

2> ctail_feed:add(#message{id=ctail:next_id(message,1), feed_id={chat,1}, origin=1, 
                      payload="Hello, Mike"}).
{ok,#message{id = 1,version = undefined,container = feed,
             feed_id = {chat,1},
             prev = undefined,next = undefined,origin = 1,
             payload = "Hello, Mike",createdAt = undefined}}

3> ctail:get(feed, {chat,1}).
{ok,#feed{id = {chat,1},top = 1,count = 1}

4> ctail_feed:add(#message{id=ctail:next_id(message,1), feed_id={chat,1}, origin=1, 
                      payload="How are you?"}).
{ok,#message{id = 2,version = undefined,container = feed,
             feed_id = {chat,1},
             prev = 1,next = undefined,origin = 1,
             payload = "How are you?",createdAt = undefined}}
             
5> ctail_feed:add(#message{id=ctail:next_id(message,1), feed_id={chat,1}, origin=1, 
                      payload="O rly?"}).
{ok,#message{id = 3,version = undefined,container = feed,
             feed_id = {chat,1},
             prev = 2,next = undefined,origin = 1,
             payload = "O rly?",createdAt = undefined}}

6> ctail:get(feed, {chat,1}).
{ok,#feed{id = {chat,1},top = 3,count = 3}

7> ctail_feed:get(message, {chat,1}, -1).
[#message{id = 1,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = undefined,next = 2,origin = 1,
          payload = "Hello, Mike",createdAt = undefined},
 #message{id = 2,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 1,next = 3,origin = 1,
          payload = "How are you?",createdAt = undefined},
 #message{id = 3,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 2,next = undefined,origin = 1,
          payload = "O rly?",createdAt = undefined}]

8> ctail_feed:get(message, {chat,1}, 2).
[#message{id = 2,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 1,next = 3,origin = 1,
          payload = "How are you?",createdAt = undefined},
 #message{id = 3,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 2,next = undefined,origin = 1,
          payload = "O rly?",createdAt = undefined}]

9> ctail:entries(message, 3, 2, #iterator.prev).
[#message{id = 2,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 1,next = 3,origin = 1,
          payload = "How are you?",createdAt = undefined},
 #message{id = 3,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 2,next = undefined,origin = 1,
          payload = "O rly?",createdAt = undefined}]

10> ctail:entries(message, 1, 2, #iterator.next).
[#message{id = 1,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = undefined,next = 2,origin = 1,
          payload = "Hello, Mike",createdAt = undefined},
 #message{id = 2,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 1,next = 3,origin = 1,
          payload = "How are you?",createdAt = undefined}]

11> ctail_feed:remove(message, 2).
ok

12> ctail:get(feed, {chat,1}).
{ok,#feed{id = {chat,1},top = 3,count = 2}

13>ctail_feed:get(message, {chat,1}, -1).
[#message{id = 1,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = undefined,next = 3,origin = 1,
          payload = "Hello, Mike",createdAt = undefined},
 #message{id = 3,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 1,next = undefined,origin = 1,
          payload = "O rly?",createdAt = undefined}]

14> Message = #message{id=ctail:next_id(message,1), feed_id={chat,1}, 
                       origin=1, payload="New message"}.
#message{id = 4,version = undefined,container = feed,
         feed_id = {chat,1},
         prev = undefined,next = undefined,origin = 1,
         payload = "New message",createdAt = undefined}

15> ctail:put(Message).
ok

16> ctail_feed:get(message, {chat,1}, -1).
[#message{id = 1,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = undefined,next = 3,origin = 1,
          payload = "Hello, Mike",createdAt = undefined},
 #message{id = 3,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 1,next = undefined,origin = 1,
          payload = "O rly?",createdAt = undefined}]

17> ctail:link(Message).
{ok,#message{id = 4,version = undefined,container = feed,
             feed_id = {chat,1},
             prev = 3,next = undefined,origin = 1,
             payload = "New message",createdAt = undefined}}

18> ctail_feed:get(message, {chat,1}, -1).
[#message{id = 1,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = undefined,next = 3,origin = 1,
          payload = "Hello, Mike",createdAt = undefined},
 #message{id = 3,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 1,next = 4,origin = 1,
          payload = "O rly?",createdAt = undefined},
 #message{id = 4,version = undefined,container = feed,
          feed_id = {chat,1},
          prev = 3,next = undefined,origin = 1,
          payload = "New message",createdAt = undefined}]