RethinkDB
Why another database?
A (not so) simple life of a web developer
WAMP / LAMP
(Windows|Linux) + Apache + MySQL + PHP
web server = Apache
database = MySQL
language = PHP
Database landscape
... and its evolution
So, why add to this madness?
RethinkDB
The open-source database for the realtime web
Few facts
- Written in C++
- First commit: October 3rd, 2009
- Current version: 2.2 (production-ready)
- > 11k stars on Github
Official drivers
- JavaScript
- Python
- Ruby
- Java (under development)
Document store
Stores plain JSONs
Proper NoSQL document
{
"first_name": "Paweł",
"last_name": "Świątkowski",
"born": 1988,
"company": "Codete",
"languages": ["Ruby", "JavaScript", "Erlang"],
"room": "Mad Max"
}
As opposed to...
{
"first_name": "Paweł",
"last_name": "Świątkowski",
"born": 1988,
"company_id": 12,
"language_ids": [1, 13, 55],
"room_id": 158
}
NoSQL databases are not relational databases equivalents!
Text
Text
(or are they?)
Features
A quick tour
ReQL
Expressive query language for difficult times
Principles
- Built with target language in mind
- Everything is chainable
- All queries execute on server
r.db('workers').table('companies').insert({
name: 'Codete', address: 'Olszańska 7', sector: 'IT'
}).run()
Query
Response
{
"deleted": 0,
"errors": 0,
"generated_keys": [
"77e35b16-f5bb-41f5-9344-0952199e5152"
],
"inserted": 1,
"replaced": 0,
"skipped": 0,
"unchanged": 0
}
r.db('workers').table('companies').filter({
sector: 'IT'
}).run()
Query
Response
{
"address": "Olszańska 7",
"id": "77e35b16-f5bb-41f5-9344-0952199e5152",
"name": "Codete",
"sector": "IT"
}
Other languages
Python
cursor = r.table("authors").filter(r.row["name"] == "William Adama").run()
for document in cursor:
print(document)
Ruby
cursor = r.table("authors").filter{|author| author["posts"].count > 2}.run
cursor.each{|document| p document}
Java
Cursor<Map<String, String>> oneEntryLambda =
r.db(dbName).table(tableName).filter(
table -> table.getField("field").eq("456")
).run(conn);
(very early version)
Clojure
(-> (r/db "test")
(r/table "authors")
(r/get-all ["crap"] {:index "genre"})
(r/filter (r/fn [row]
(r/eq "Stephenie Meyer" (r/get-field row "name"))))
(r/run conn))
PHP
$result = r\table("tablePhpTest")->map(function($x) {
return $x('someKey');
})->run($conn);
(apparently you can't write pretty code in PHP)
JOINS
Wait, joins in NoSQL?
r.db('workers').table('people').insert({
"first_name": "Paweł",
"last_name": "Świątkowski",
"born": 1988,
"company": "77e35b16-f5bb-41f5-9344-0952199e5152",
"languages": ["Ruby", "JavaScript", "Erlang"],
"room": "Mad Max"
})
Query
r.db('workers').table('people').eqJoin(
'company', r.db('workers').table('companies')).run()
Query
Response
{
"left": {
"born": 1988,
"company": "77e35b16-f5bb-41f5-9344-0952199e5152",
"first_name": "Paweł",
"id": "feee3558-a009-48fe-9380-217d2db562f4",
"languages": [
"Ruby",
"JavaScript",
"Erlang"
],
"last_name": "Świątkowski",
"room": "Mad Max"
},
"right": {
"address": "Olszańska 7",
"id": "77e35b16-f5bb-41f5-9344-0952199e5152",
"name": "Codete",
"sector": "IT"
}
}
r.db('workers').table('people').eqJoin(
'company', r.db('workers').table('companies')).zip().run()
Query
Response
{
"address": "Olszańska 7",
"born": 1988,
"company": "77e35b16-f5bb-41f5-9344-0952199e5152",
"first_name": "Paweł",
"id": "77e35b16-f5bb-41f5-9344-0952199e5152",
"languages": [
"Ruby",
"JavaScript",
"Erlang"
] ,
"last_name": "Świątkowski",
"name": "Codete",
"room": "Mad Max",
"sector": "IT"
}
What's more
Also outerJoin and innerJoin supported
Limitations
Very limited support for indices with join
(inherent to NoSQL databases)
Data types
not only JSON
All JSON data types, of course
Time
r.table('events').insert([
{id: 0, timestamp: new Date()},
{id: 1, timestamp: r.epochTime(1376436769.923)}
]).run()
r.table('events').between(r.epochTime(1376436769.913),
r.epochTime(1376436769.933), {index: 'timestamp'}
).run()
[{"id": 1, "timestamp": Date("2013-08-13T23:32:49.923Z")}]
Binary
You can upload files and store them in the database directly.
(... which is probably not the best idea, but you still can)
Geometry
r.table('geo').insert([
{
id: 1,
name: 'San Francisco',
location: r.point(-122.423246,37.779388)
},
{
id: 2,
name: 'San Diego',
location: r.point(-117.220406,32.719464)
}
])
r.table('geo').indexCreate('location', {geo: true})
r.table('geo').getNearest(
r.point(-122.422876,37.777128),
{maxDist: 100, unit: 'km', index: 'location'}
).run()
Query
Response
[{
"dist": 0.25295150950901096,
"doc": {
"id": 1,
"location": {
"$reql_type$": "GEOMETRY",
"coordinates": [
-122.423246,
37.779388
],
"type": "Point"
},
"name": "San Francisco"
}
}]
Geometry (more)
- line, polygon, circle
- distance
- intersection
- containment
- ...
Changefeeds
Real-time updates for documents
Subscribe to a feed with callback function
r.table('users').changes().run(conn, function(err, cursor) {
cursor.each(console.log);
})
{
old_val: null,
new_val: {id: 1, name: 'Slava', age: 31}
}
r.table('messages').filter(
r.row('room_id').eq('#general')
).changes().run(conn, callback)
How about filtering?
r.table('scores').changes().filter(
r.row('new_val')('score').gt(r.row('old_val')('score'))
)('new_val').run(conn, callback)
How about something crazy?
Administration
Powerful web administration console
Sharding and Replication
Automatic failover
(with enough shards and replicas)
Who's using it?
Platzi
Platzi is an education platform that streams live video alongside realtime discussion. RethinkDB powers live updates and instant messaging in their online courses. The Platzi team chose RethinkDB because its push model helped them enable realtime collaboration between students and educators.
ClientSuccess
ClientSuccess provides a cloud-based platform that helps SaaS providers manage their customer relationships. RethinkDB simplified the way that their application synchronizes data with Salesforce and other sources.
Jive
Jive Software is a provider of communication and collaboration solutions for business. Jive enables employees, partners and customers to work together.
Questions?
RethinkDB
By katafrakt
RethinkDB
- 1,567