Last weekend I participated in a hack day with some colleagues and one of them decided that it would be a good idea to use MongoDB. Since I had never used it, I thought it would be fun to learn a little about it. Here I’m going to write about the things that I learned.
Installation
I chose to use Docker because it makes everything easier. The only thing to keep in mind when using Docker for a database is that you need to store the data files outside the container so they don’t disappear when the container is destroyed. This is enough to get the MongoDB running in a container and making sure the data is persisted in $(PWD)/data in the host:
1
docker run --name my-mongo -v $(PWD)/data:/data/db -d mongo:3.3
First steps
Once we have mongo running we need to open a shell in the running container so we can play with it:
1
docker exec -it my-mongo bash
Then you can get a mongo cli client by running:
1
mongo shell
The mongo shell works similarly to mysql-client. The first thing you probably want to do is check the databases that already exist:
1
2
3
> show dbs
admin 0.000GB
local 0.000GB
You can switch to a DB:
1
> use admin
And see the collections (similar to MySQL tables):
1
2
> show collections;
system.version
To create a database you have to move into the database and then create a collection in it:
1
2
> use temp;
> createCollection('users');
Now you will see your database in the list:
1
2
3
4
> show dbs;
admin 0.000GB
local 0.000GB
temp 0.000GB
And the collection:
1
2
> show collections;
users
Collections, are different to MySQL tables in that they don’t have schemas. You can save data without defining what kind of data you are saving. Lets add some users:
1
2
3
> db.users.insert({'name': 'Pancho Villa', 'email': '[email protected]'});
> db.users.insert({'name': 'Miguel Hidalgo', 'phone': '12345678'});
> db.users.insert({'name': 'Benito Juarez', 'email': '[email protected]', 'weight': '83kg'});
As you can see, I can insert any data I want. It doesn’t matter if the fields are the same. We can see all the documents in the collection by using find:
1
2
3
4
> db.users.find();
{ "_id" : ObjectId("57d3bd869c149b0f37405e87"), "name" : "Pancho Villa", "email" : "[email protected]" }
{ "_id" : ObjectId("57d3bfbc9c149b0f37405e88"), "name" : "Miguel Hidalgo", "phone" : "12345678" }
{ "_id" : ObjectId("57d3bfed9c149b0f37405e89"), "name" : "Benito Juarez", "email" : "[email protected]", "weight" : "83kg" }
An _id
field is added automatically to each document. This id is used when you want to make relationships between documents in other collections.
Now that we have documents in our collection we probably want to be able to find them. We already saw how to use the find command in its simplest form, but lets say we only want to get an element with an specific id. We can do it this way:
1
2
> db.users.find('57d3bfbc9c149b0f37405e89');
{ "_id" : ObjectId("57d3bfed9c149b0f37405e89"), "name" : "Benito Juarez", "email" : "[email protected]", "weight" : "83kg" }
It is important to mention that there is no index in this field, so a linear search is done. I’ll explain how to create indexes later in this post.
If you want to search by a different field, you can do:
1
2
> db.users.find({'name': 'Pancho Villa'});
{ "_id" : ObjectId("57d3bd869c149b0f37405e87"), "name" : "Pancho Villa", "email" : "[email protected]" }
The MongoDB documentation explains more advanced queries in detail. I prefer not to duplicate all that information.
Another common operation is updating a document. An update by default will only update one document and will also replace the whole document by default. For example, executing this command:
1
> db.users.update({'_id': ObjectId("57d3bd869c149b0f37405e87")}, {'weight': '79kg'});
Will probably not have the effect you expect:
1
2
> db.users.find('57d3bd869c149b0f37405e87');
{ "_id" : ObjectId("57d3bd869c149b0f37405e87"), 'weight': '79kg' }
This could be an unpleasant surprise, so be careful with your updates. To just update the fields without replacing the whole document we need to use update operators:
1
> db.users.update({'_id': ObjectId("57d3bd869c149b0f37405e87")}, {$set: {'weight': '70kg'}});
In the previous example, $set is an update operator and tells MongoDB how to make the update.
To complete the standard CRUD operation set, we need to be able to delete an element. For this we use the remove command:
1
> db.users.remove({'_id': ObjectId("57d3bfab9c149b0f37405e88")});
It works similar to a find and will delete all documents matching the query.
Document validation
Some times you want to enforce certain fields to contain data in certain formats. For this you can use document validation. You can for example update the users table to only accept numbers lower than 300 in the age field:
1
> db.runCommand({'collMod': 'users', validator: {'age': {$type: 'int', $lt: 300}}});
If you try to insert a document with an invalid age, you will get an error:
1
2
3
4
5
6
7
8
> db.users.insert({'name': 'Josefa Ortiz', 'age': 700});
WriteResult({
"nInserted" : 0,
"writeError" : {
"code" : 121,
"errmsg" : "Document failed validation"
}
})
One gotcha about this is that you now have to use NumberInt function when you insert a new document:
1
> db.users.insert({'name': 'Josefa Ortiz', 'age': NumberInt(70)});
Indexing
Without the ability to index fields, all our queries would be too slow as we add more data into our collections. Indexes are necessary to be able to perform searches efficiently.
To create a simple index on the age field of of users collection we can do:
1
> db.users.createIndex({age: 1});
The value 1 means that it will be an ascending index. If we specified -1, it would be a descending index. For a single column index, this doesn’t really make a difference, the index can be traversed equally in any direction.
There are other types of indexes in MongoDB. You can learn more about them in their documentation.
Conclusion
Although there are a lot more features in MongoDB, this should be enough to start using it in a project. As I learn more about it, I will write more articles with the parts that I find interesting.
databases
]