De demo zoals die in het hoorcollege gegeven is.
Zorg voor de start van de demo er voor dat je database runt (mongod.exe). Start de applicatie middels npm start of nodemon.
We gebruiken in deze demo Express om een restful API te tonen. Je kan deze code bekijken, maar het is nog niet nodig om deze te begrijpen.
- Open /models/teacher.js
- Mongoose verwacht een schema. Deze kan je als volgt definiëren
var teacherSchema = new mongoose.Schema({});
mongoose.model('Teacher', teacherSchema);
- Het schema moet nu uitgebreid worden met properties. Vervang daarom het teacherschema met het volgende:
var teacherSchema = new mongoose.Schema({
_id: { type: String},
firstName: { type: String },
middleName: { type: String },
lastName: { type: String },
age: { type: Number },
isActive: { type: Boolean },
courses: [{ type: String }]
});
- Open Postman en post de volgende body naar http://localhost:3000/teachers Je ziet dat dit goed gaat, dit willen we niet omdat de teacher nu een in onze ogen invalide object is.
{
"_id": "test"
}
- We gaan het schema uitbreiden met inline validaties. Denk aan required, lowercase, min en max Vervang je schema door het volgende:
var teacherSchema = new mongoose.Schema({
// Schema including validation
_id: { type: String, required: true, lowercase: true },
firstName: { type: String, required: true },
middleName: { type: String },
lastName: { type: String, required: true },
age: {type: Number, min: 0, max: 100 },
isActive: { type: Boolean},
courses: [{ type: String, required: true }]
});
- Als we nu stap 4 opnieuw uitvoeren zul je zien dat je een foutmelding krijgt.
- We willen een pseudo-join uitvoeren. We willen de courses bij teachers kunnen ophalen. Geef dan bij je teacher aan dat hij verwijst naar courses.
courses: [{ type: String, required: true, ref: 'Course' /* Pseudo-joins */ }]
- Als je een nieuw GET-request doet dan zie je nog geen verschil. We moeten hem namelijk nog aangeven dat de course meegeladen moet worden. Open /routes/teachers.js en geef op regel 15 aan dat de courses gepopulated dienen te worden.
var result = Teacher.find(query)
.populate('courses');
- Nu zul je bij een GET-request wel de courses opgehaald zien worden.
- We kunnen nog extra validaties op paden uitvoeren. Dit doe je voornamelijk als je afhankelijke properties hebt. In het volgende voorbeeld mag je achternaam bijvoorbeeld niet hetzelfde als je voornaam zijn.
// Validation
teacherSchema.path('lastName').validate(function (val) {
return val && this.firstName != val;
}, 'Last name must differ from first name.');
- We kunnen berekende eigenschappen in onze objecten toevoegen. Dit zijn eigenschappen die niet in de database staan opgeslagen, maar wel teruggeven worden. Denk bijvoorbeeld aan een leeftijd op basis van een geboortedatum of een full name. Voeg de volgende code toe aan je teacher:
// Virtuals
teacherSchema.virtual('fullName').get(function(){
var fullName = this.firstName + ' ';
if(this.middleName && this.middleName.length){
fullName += this.middleName + ' ';
}
fullName += this.lastName;
return fullName;
});
- Als je nu een GET-request uitvoert zal je zien dat je deze property niet terugkrijgt. Voeg daarom aan je schema toe dat hij je virtuals wél in een JSON-object moet bijvoegen.
var teacherSchema = new mongoose.Schema({
// Hier staan al je eigenschappen.
},
// settings:
{
toObject: { virtuals: true },
toJSON: { virtuals: true }
});
- We kennen al de query builders van het Entity Framework en JQuery. Hier kan je methodes achter elkaar chainen om het op het eind pas uit te voeren. Mongoose heeft deze mogelijkheid ook. Denk bijvoorbeeld aan het volgende:
Teacher.find({ firstName: 'Martijn'})
.sort({ _id: 1 })
.limit(pageSize)
.skip(pageIndex * pageSize)
.exec(/* function met resultaat */);
- Deze kunnen we uitbreiden met custom query methodes. Bovenstaande code biedt bijvoorbeeld pagination aan. Hier kunnen we een eigen query method voor maken. Voeg de onderstaande code toe aan het TeacherSchema.
// Custom query function.
teacherSchema.query.byPage = function (pageSize, pageIndex) {
return this.find()
.limit(parseInt(pageSize))
.skip(pageIndex * pageSize);
};
Merk op dat we eerst een find() doen. Dit heeft geen effect op eerdere filters, dit is enkel nodig om een query-object te krijgen.
- Je kan nu in /routes/teachers.js je query uitbreiden. Dit hoeft niet eens na elkaar, je kan bijvoorbeeld eerst je parameters checken.
if (req.query.pageSize && req.query.pageIndex) {
result = result.byPage(req.query.pageSize, req.query.pageIndex);
}
Je kan je eigen middleware schrijven. Dit wordt uitgevoerd voor of na bepaalde acties. In ons voorbeeld is dit vóór het opslaan, maar je kan je misschien wel voorstellen dat nadat bijvoorbeeld User-object geassocieerd wordt met een nieuwe rol je de administrator een mailtje stuurt ter informatie.
- Voeg de volgende code toe aan je teacherschema:
// Middleware
teacherSchema.pre('save', function(next){
console.log('Teacher will be saved');
next();
console.log('Teacher is saved');
});
- Probeer nu een teacher op te slaan. Je zal zien dat je logs te zien zijn in je console.
Met MongooseJS kan je dus enorm veel logica van je applicatie in je models kwijt. Het idee is dus ook dat je in je routes weinig doet en dit voornamelijk op deze laag laat afhandelen.
MongooseJS is nog veel rijker dan bovengenoemde voorbeelden. Kijk voor de volledige documentatie daarom op http://mongoosejs.com/docs/guide.html