Skip to content

ga-wdi-lessons/mongoose-intro

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

56 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Intro to Mongoose

Learning Objectives

  • Differentiate between NoSQL and SQL databases
  • Explain what Mongoose is
  • Describe the role of Mongoose schema and models
  • Use Mongoose to perform CRUD functionality
  • List and describe common Mongoose queries
  • Persist data using Mongoose embedded documents

Opening Framing (10 minutes / 0:10)

In previous WDI units you've used ActiveRecord to interact with and perform CRUD actions on a SQL database through a Ruby back-end. Today, we'll be doing the equivalent with a tool called Mongoose on a NoSQL database using a Node back-end.

Before we dive into Mongoose, however, let's talk a bit about about last night's homework. You were tasked with looking through the Mongo lesson plan and familiarizing yourself with a NoSQL database. What are some of your takeaways?

What is a NoSQL database?

A NoSQL database is a non-relational database.

  • This means no explicit one-to-one, one-to-many and many-to-many relationships.
  • That being said, we can emulate these relationships in a NoSQL database.
How is a NoSQL database organized?

A NoSQL database can be organized into documents and collections.

  • Collections are the NoSQL equivalent of tables in a SQL database.
  • Documents are the NoSQL equivalent of a table row.

This is not the only way that a NoSQL database organizes data. Start here to learn more.

What is MongoDB?

A NoSQL database that stores information as JSON.

  • Technically, it's BJSON -- "binary JSON."
  • Think of this as taking the place of Postgres.

Why Use NoSQL/Mongo Over SQL?

It's flexible.

  • You don't need to follow a schema if you don't want to. This might be helpful with non-uniform data.
  • That being said, you can enforce consistency using schemas. In fact, we'll be doing that in today's class.

It's fast.

  • Data is "denormalized" in a NoSQL data, meaning that it's all in the same place.
  • For example, a post's comments will be nested directly within the post in the database.
  • This is unlike a relational database, in which we need to make queries to retrieve data connected through a relation.

Many web apps already implement object-oriented Javascript.

  • If we're using objects in both the back-end and front-end, that makes handling and sending data between the client and a database much easier.
  • No need for type conversion (e.g., making sure a Ruby hash is being served as JSON).

Example MongoDB Commands

Even though you won't be writing much Mongo in WDI, we will be using some MongoDB CLI commands to test what's in our database in this class. Examples include...

  • show dbs - Show a list of all databases
  • use database-name - Connect to a database
  • show collections - List the collections in a database
  • db.students.find() - List all students in a student collection

Mongoose (5 minutes / 0:15)

mongoose.js

"Let's face it, writing MongoDB validation, casting and business logic boilerplate is a drag. That's why we wrote Mongoose."

Mongoose is an ODM (Object Data Mapping), that allows us to encapsulate and model our data in our applications. It gives us access to additional helpers, functions and queries to simply and easily preform CRUD actions.

Mongoose will provide us with the similar functionality to interact with MongoDB and Express as Active Record did with PostgreSQL and Rails.

Active Record is an ORM (Object Relational Mapping). An ODM is the same thing without relations.

Here's an example of some Mongoose code pulled from their documentation.

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/test');

var Cat = mongoose.model('Cat', { name: String });

var kitty = new Cat({ name: 'Zilda' });
kitty.save(err => {
  if (err) // ...
  console.log('meow');
});

You Do: Initial Set Up for Reminders (5 minutes / 0:20)

During today's in-class exercises, you will be creating a two-model todo app using Mongoose and MongoDB.

It will have two models: Authors and Reminders.

  • Each Author has a name (string)
  • Each Reminder has a body (string)

Authors will have many Reminders, although we won't be implementing that using a SQL relationship.

Follow these steps...

  1. Clone this repo: $ git clone https://github.com/ga-wdi-exercises/reminders_mongo.git
  2. Make sure to checkout locally to the mongoose branch: $ git checkout mongoose
  3. Also make sure you are running $ mongod in a separate Terminal tab.

Starter Code

Solution Code

I Do: Mongoose and Connection Set Up (5 minutes / 0:25)

For today's in-class demonstrations, we will be creating an app that uses two models: Students and Projects. After each demo, you will apply the same functionality to your todo app.

This means you should not be following along when I am building the Student-Project app.

Let's begin by installing Mongoose...

$ npm install --save mongoose

In order to have access to mongoose in our application, we need to explicitly require mongoose and open a connection to the test database on locally...

// db/schema.js

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/students');

The name above students will be the name of the database stored in MongoDB

// db/schema.js
var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/students');

// Now that we're connected, let's save that connection to the database in a variable.
var db = mongoose.connection;

// Will log an error if db can't connect to MongoDB
db.on('error', err => {
  console.log(err);
});

// Will log "database has been connected" if it successfully connects.
db.once('open', () => {
  console.log("database has been connected!");
});

Now let's run our db/schema.js file...

$ node db/schema.js

You Do: Install Mongoose and Connection (5 min / 0:30)

Follow the instructions in the previous section to require mongoose in your Reminders app.

Solution

I Do: Mongoose Schema & Models (10 minutes / 0:40)

What Is a Mongoose Schema?

  • Schemas are used to define attributes and structure for our documents.
  • Each Schema maps to a MongoDB collection and defines the shape of the documents within that collection.

Here's an example of a Mongoose schema...

// db/schema.js

// First, we instantiate a namespace for our Schema constructor defined by mongoose.
var Schema = mongoose.Schema

var StudentSchema = new Schema({
  name: String,
  age: Number
});

var ProjectSchema = new Schema({
  title: String,
  unit: String
});

What Are Mongoose Models?

Mongoose Models will represent documents in our database.

  • Models are defined by passing a Schema instance to mongoose.model
// db/schema.js

var Schema = mongoose.Schema

var StudentSchema = new Schema({
  name: String,
  age: Number
});

var ProjectSchema = new Schema({
  title: String,
  unit: String
});

var Student = mongoose.model("Student", StudentSchema);
var Project = mongoose.model("Project", ProjectSchema)

.model() makes a copy of a schema.

  • The first argument is the singular name of the collection your model is for. Mongoose automatically looks for the plural version of your model name when creating a collection.
  • That means the Student model is for the students collection in the database.

Collections: Embedded Documents (10 minutes / 0:50)

Let's add another model to db/schema.js.

  • We will be adding a schema for Project since we want to create an application that tracks Students and Projects
  • Like a one-to-many relationship in a relational database, a Student will have many Projects.

With ActiveRecord, we defined a one-to-many relationship like so...

class Students < ActiveRecord::Base
  has_many :projects
end


class Projects < ActiveRecord::Base
  belongs_to :student
end

In Mongoose, we will do this using embedded documents.

I Do: Embedded Documents

Embedded Documents -- sometimes referred to as "sub-documents" -- are schemas of their own which are elements of a parent document's array

  • They contain all the same features as normal documents.
// db/schema.js

var Schema = mongoose.Schema

var ProjectSchema = new Schema({
  title: String,
  unit: String
})

var StudentSchema = new Schema({
  name: String,
  age: Number,
  projects: [ProjectSchema]
})

var Student = mongoose.model("Student", StudentSchema);
var Project = mongoose.model("Project", ProjectSchema);

The projects key of your StudentSchema documents will contain a special array that has specific methods to work with embedded documents.

The Project Schema must be defined prior to our main Student Schema.

Advantages

  • Easy to conceptualize and set up.
  • Can be accessed quickly.

Disadvantages

  • Don't scale well. Documents cannot exceed 16MB in size.

If you find that you are nesting documents within documents for 3+ levels, you should probably look into a relational database.

You Do: Set Up Schema and Models for Reminders (10 minutes / 1:00)

Use the previous section to step up your Reminder and Author schemas and models.

Place the following code at the bottom of your schema file. You'll know you set up your schema properly if, after running it, you see information about an Author printed to the console. Don't worry about what it means - we'll go over that soon in this lesson.

var john = new Author({name: "John"})
var newReminder = new Reminder({body: "Don't forget to set up your schema properly"})

john.reminders.push(newReminder)

john.save((err, author) => {
  if(err){
    console.error(err)
  } else {
    console.log(author)
    process.exit()
  }
})

Solution

Break (10 minutes / 1:10)

I Do: Create With Students and Projects (10 minutes / 1:20)

First let's create an instance of our Student model. Here's one way of doing it...

// db/schema.js

// First we create a new student. It's just like generating a new instance with a constructor function!
var anna = new Student({name: "Anna", age: 30});

// Then we save it to the database using .save
anna.save((err, student) => {
  if(err){
    console.log(err);
  }
  else{
    console.log(student);
  }
});

You can compare this to .new and .save in ActiveRecord.

We can also consolidate that into a single .create method, like so...

// db/schema.js

Student.create({ name: 'Anna', age: 30 }, (err, student) => {
  if (err){
    console.log(err);
  }
  else{
    console.log(student);
  }
});

Callback Functions

Oftentimes, when making a Mongoose query we will pass in a callback function. It will be passed two arguments: err and data.

  • err will contain an error message if something goes wrong with the Mongoose query
  • data contains the result of the Mongoose query
Why do you think callbacks might be necessary when using Mongoose?

Because these queries are asynchronous! We want to make sure the query has finished before we run any code that depends on the result.

Promises

If callbacks aren't your cup of tea, you can replace the callbacks we used above with promise methods.

var anna = new Student({name: "Anna", age: 30});

anna.save().then(student => {   // We don't pass in `err` as an argument here...
  console.log(student)
}).catch(err => {               // ...instead, we pass it in here
  console.log(err)              // If there's an error, this `.catch` method will be triggered
});

I Do: Add Embedded Documents

Next, let's create a Project...

// db/schema.js

var anna = new Student({name: "Anna", age: 30});
var project1 = new Project({title: "memory game", unit: "JS"});

// Now we add that project to a student's collection / array of projects.
anna.projects.push(project1)

// In order to save that project to the student, we need to call `.save` on the student -- not the project.
anna.save((err, student) => {
  if(err){
    console.log(err)
  } else {
    console.log(student + " was saved to our db!");
  }
});

I Do: Seed Data (10 minutes / 1:30)

Let's seed some data in our database. In order to do that, we need to first make sure we can connect schema.js to seeds.js. Let's add the following to db/schema.js...

// db/schema.js

// The rest of our schema code is up here...

// By adding `module.exports`, we can know reference these models in other files by requiring `schema.js`.
module.exports = {
  Student: Student,
  Project: Project
};

And add the following to db/seeds.js...

// db/seeds.js

var Schema = require("./schema.js");

var Student = Schema.Student
var Project = Schema.Project

Now let's call some methods in db/schema.js that will populate our database...

// db/seeds.js

var Schema = require("./schema.js");

var Student = Schema.Student
var Project = Schema.Project

// First we clear the database of existing students and projects.
Student.remove({}, err => {
  if(err){
    console.log(err)
  }
});

Project.remove({}, err => {
  if(err){
    console.log(err)
  }
});

// Now we generate instances of Student and Project.
var becky = new Student({name: "becky"})
var brandon = new Student({name: "brandon"})
var tom = new Student({name: "tom"})

var project1 = new Project({title: "Project 1", unit: "JS"})
var project2 = new Project({title: "Project 2", unit: "Rails"})
var project3 = new Project({title: "Project 3", unit: "Angular"})
var project4 = new Project({title: "Project 4", unit: "Express"})

var students = [becky, brandon, tom]
var projects = [project1, project2, project3, project4]

// Here we assign some projects to each student.
for(var i = 0; i < students.length; i++){
  students[i].projects.push(projects[i], projects[i+1])
  students[i].save((err, student) => {
    if (err){
      console.log(err)
    } else {
      console.log(student);
    }
  })
};

// ...or you could use forEach instead of a for loop...
students.forEach((student, i) => {
  student.projects.push(projects[i], projects[i+1])   // Assigning each student multiple projects
  student.save((err, student) => {
    if (err){
      console.log(err)
    } else {
      console.log(student);
    }
  })
})

Now, seed your database by running node db/seeds.js in your terminal. Use Ctrl + C to exit running Node.

Let's test if this all worked by opening Mongo in the Terminal...

$ mongo
$ show dbs
$ use students
$ show collections
$ db.students.find()

You Do: Add Seed Data and Create to Reminders (15 minutes / 1:45)

Now do the same thing with your Reminders app.

Solution

I Do: Mongoose Queries (10 minutes / 1:55)

Like Active Record, Mongoose provides us with a variety of helper methods that allow us to easily retrieve documents from our database.

Explore them using the Mongoose Queries Documentation.

// Finds all documents of a specified model type. We can pass in a key-value pair(s) to narrow down the search.
Model.find({}, callback)

// Finds a single model by its id.
Model.findById(someId, callback)

// Find a single model using a key-value pair(s).
Model.findOne({someKey: someValue}, callback)

// Removes documents that match a key-value pair(s).
Model.remove({someKey: someValue}, callback)

Let's use .find to implement index functionality. We'll do that in a controller file...

$ mkdir controllers
$ touch controllers/studentsController.js

We are adding a controllers directory and studentsController.js file to mimic how we might define a controller in an Express application. Like how our controllers helped us in Rails, we will be following similar REST conventions and using our controllers to listen for incoming requests and communication with our database.

// controllers/studentsController.js

var Schema = require("../db/schema.js");
var Student = Schema.Student;
var Project = Schema.Project;

var studentsController = {
  index(){
    Student.find({}, (err, students) => {
      console.log(students);
    });
  }
};

studentsController.index();

Run $ node controllers/studentsController.js in the terminal.

Now let's do show...

// controllers/studentsController.js

var studentsController = {
  index(){
    Student.find({}, (err, students) => {
      console.log(students);
    });
  },
  show(req){
    Student.findOne({name: req.name}, (err, student) => {
      console.log(student);
    });
  }
};

studentsController.show({name: "becky"});

req here stands for "request." While we don't have to name the argument that way, it is common practice. It represents information that is being sent to a server. In this case, it contains information about the data in question: a single student.

You Do: Index, Show, Update and Delete (15 minutes / 2:10)

Follow the above instructions to implement index and show for the Author model.

Then use Mongoose documentation to figure out how to update and delete authors. If the documentation proves difficult to navigate, don't be afraid to Google it! We'll go over how to update and delete after the exercise...

Index/Read Solution Update Solution Delete Solution

Bonus

If you finish early...

  • Create a controller method that returns the number of Authors or Reminders in the database
  • Create a controller method that clears all of an Author's Reminders (i.e., the author has no reminders)
  • Check out the Validations section at the end of this lesson

Break (5 minutes / 2:15)

I Do: Update & Delete (10 minutes / 2:25)

This is how to update...
// controllers/studentsController.js
var studentsController = {

  // This method takes two arguments: (1) the old instance and (2) what we want to update it with.
  update(req, update){
    Student.findOneAndUpdate({name: req.name}, {name: update.name}, {new: true}, (err, student) => {
      if(err) {
        console.log(err)
      }
      else {
        console.log(student);
      }
    });
  }
};

studentsController.update({name: "becky"}, {name: "Sarah"});

We are inserting {new: true} as an additional option. If we do not, we will get the old document as a return value -- not the updated one.

This is how to delete...
// controllers/studentsController.js

var studentsController = {
  destroy(req){
    Student.findOneAndRemove(req, (err, docs) => {
      if(err){
        console.log(err);
      }
      else{
        console.log(docs);
      }
    });
  }
};

studentsController.destroy({name: "brandon"});

Don't look at these while working on the previous exercise!


Bonus: Validations

Mongoose contains built-in validators and an option to create custom validators as well.

Validators are defined at the field level of a document and are executed when the document is being saved. If a validation error occurs, the save operation is aborted and the error is passed to the callback.

Built in Validators:

  • required and unique: used to validate the field existence in Mongoose, which is placed in your schema on the field you want to validate.

Example: Let's say you want to verify the existence of a username field and ensure it's unique before you save the user document.

var UserSchema = new Schema({
  ...
  username: {
    type: String,
    unique: true,
    required: true
  }
});

This will validate the existence and uniqueness of the username field when saving the document, thus preventing the saving of any document that doesn't contain that field.

  • match: type based validator for strings, placed in your fields for you schemas

Continuing off the above example, to validate your email field, you would need to change your UserSchema as follows:

var UserSchema = new Schema({
  username: {
    type: String,
    unique: true,
    required: true
  },
  email: {
    type: String,
    match: /.+\@.+\..+/
  }
});

The usage of a match validator here will make sure the email field value matches the given regex expression, thus preventing the saving of any document where the e-mail doesn't conform to a valid pattern!

  • enum: helps to define a set of strings that are only available for that field value.
var UserSchema = new Schema({
  username: {
    type: String,
    unique: true,
    required: true
  },
  email: {
    type: String,
    match: /.+\@.+\..+/
  },
  role: {
    type: String,
    enum: ['Admin', 'Owner', 'User']
  },

});

By Adding in enum, we are adding a validation to ensure only these three possible strings are saved in the document.

Custom Validations

We can also define our own validators by using the validate property.

This validate property value is typically an array consisting of a validation function.

For example, if we want to validate the length of your user's password. To do so, you would have to make these changes in your UserSchema:

var UserSchema = new Schema({
  ...
  password: {
    type: String,
    validate: [
      function(password) {
        return password.length >= 6;
      },
      'Password should be longer'
    ]
  },
});

This custom validator will make sure your user's password is at least six characters long or it will prevent it from saving the document

  • .pre validation: middleware that are functions which are passed control during execution of asynchronous functions.

By using .pre, these are executed before validations.

UserSchema.pre("save", function(next) {
    var self = this;

    UserModel.findOne({email : this.email}, 'email', function(err, results) {
        if(err) {
            next(err);
        } else if(results) {
            console.warn('results', results);
            self.invalidate("email", "email must be unique");
            next(new Error("email must be unique"));
        } else {
            next();
        }
    });
});

Bonus: Multiple Collections & References

Similar to how we use foreign keys to represent a one-to-many relationship in Postgres, we can add references to documents in other collections by storing an array of ObjectIds referencing document ids from another model.

// db/schema.js

var Schema = mongoose.Schema
var ObjectId = Schema.ObjectId

var ProjectSchema = new Schema({
  title: String,
  unit: String
});

var StudentSchema = new Schema({
  name: String,
  age: Number,
  projects: [ {type: Schema.ObjectId, ref: "Project"}]
});

var Student = mongoose.model("Student", StudentSchema);
var Project = mongoose.model("Project", ProjectSchema);

Since we are using an id to refer to other objects, we use the ObjectId type in the schema definition. The ref attribute must match the model used in the definition.

Advantages

  • Could offer greater flexibility with querying.
  • Might be a better decision for scaling.

Disadvantages

  • Requires more work. Need to find both documents that have the references (i.e., multiple queries).

Closing / Questions

  • How is Mongoose used to interact with MongoDB?
  • What are embedded documents in Mongoose?
  • Why do we create a Schema in Mongoose?
  • What do we need after Mongoose queries?
  • What are common built in validations for Mongoose? Why would we use them?

Homework

After this class you should be able to complete Emergency Compliment and Part I of YUM.

Additional Resources

About

[mongoose, mongo, nosql]

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published