/learn-node-from-scratch

My notes on Wes Bos' "Learn Node" online course (Node.js, Express, MongoDB).

Primary LanguageCSSMIT LicenseMIT

Learn Node

These are my notes for Wes Bos' Learn Node online class. Its goal is to teach you to build a restaurant reviewing web app (front- and back-end) using NodeJS, Express and MongoDB.

He is teaching the class from the perspective of a front-end developer wanting to become "full stack". As such, he provides a huge number of "starter files" including custom javascript helper functions, tons of dependencies and CSS "magic".

As my background is in back-end development and I prefer to learn from first principles, I reduced the codebase to the minimum needed for the concepts/examples taught in the given lesson. (You can check out the respective state of the codebase / course by looking at the git tags, e.g. video-10 will contain all the code you'll need to run the examples up to and including the 10th video of the course.)

Table of Contents

Table of contents generated with markdown-toc

01 - Getting Setup

Install NodeJS

  • install NodeJS 7.6 or above with npm 4 or above
# check node version
$ node -v
v10.15.3

$ npm -v
6.10.3

Install packages in the starter files directory

$ sudo npm install -g npm
$ npm install
...
added 1179 packages from 817 contributors and audited 10758 packages in 166.636s
found 27 vulnerabilities (10 low, 10 moderate, 7 high)
  run `npm audit fix` to fix them, or `npm audit` for details

$ npm audit fix
+ express-session@1.17.0
+ pug@2.0.4
+ axios@0.19.0
+ body-parser@1.19.0
+ express@4.17.1
+ moment@2.24.0
+ mongoose@5.7.4
added 13 packages from 14 contributors, removed 17 packages and updated 38 packages in 44.992s
fixed 22 of 27 vulnerabilities in 10758 scanned packages
  2 vulnerabilities required manual review and could not be updated
  2 package updates for 3 vulnerabilities involved breaking changes
  (use `npm audit fix --force` to install breaking changes; or refer to `npm audit` for steps to fix these manually)

$ npm audit fix --force
+ css-loader@3.2.0
+ juice@5.2.0
added 83 packages from 69 contributors, removed 75 packages and updated 16 packages in 39.571s
fixed 3 of 5 vulnerabilities in 10800 scanned packages
  2 vulnerabilities required manual review and could not be updated
  2 package updates for 3 vulnerabilities involved breaking changes
  (installed due to `--force` option)

# With NodeJS, you're always doomed!
$ npm audit
┌───────────────┬──────────────────────────────────────────────────────────────┐
│ Low           │ Regular Expression Denial of Service                         │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Package       │ timespan                                                     │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Patched in    │ No patch available                                           │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Dependency of │ forever                                                      │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Path          │ forever > timespan                                           │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ More info     │ https://npmjs.com/advisories/533                             │
└───────────────┴──────────────────────────────────────────────────────────────┘
┌───────────────┬──────────────────────────────────────────────────────────────┐
│ Low           │ Regular Expression Denial of Service                         │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Package       │ braces                                                       │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Patched in    │ >=2.3.1                                                      │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Dependency of │ forever                                                      │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ Path          │ forever > forever-monitor > chokidar > anymatch > micromatch │
│               │ > braces                                                     │
├───────────────┼──────────────────────────────────────────────────────────────┤
│ More info     │ https://npmjs.com/advisories/786                             │
└───────────────┴──────────────────────────────────────────────────────────────┘
found 2 low severity vulnerabilities in 10315 scanned packages
  2 vulnerabilities require manual review. See the full report for details.

Configure IDE

  • we need support for .pug (HTML template) files
    • vscodium (free/libre version of Micro$oft VSCode) provides this

02 - Setting up Mongo DB

  • we'll use MongoDB 3.2+ as our document store / database
  • set up MongoDB locally

Install on Ubuntu / Linux Mint

Install on arch linux

https://wiki.archlinux.org/index.php/MongoDB

sudo mkdir -p /data/db
sudo mongod --fork --logpath /tmp/mongo.log

Configure MongoDB

  • default local URL: mongodb://127.0.0.1:27017

  • copy variables.env.samples to variables.env

  • in variables.env, replace DATABASE value with mongodb://$USERNAME:$PASSWORD@127.0.0.1:27017/$DATABASENAME

    • add a user/password using the guide above
    • add a database using either MongoDB Compass or robo3t (and add the user to it)
  • add variables.env to your .gitignore file

  • test, if mongodb is working

$ mongo -u $USERNAME -p --authenticationDatabase admin
> exit
bye
  • installl MongoDB Compass GUI and connect to the database
    • this is an Electron app
    • try robo3t instead (open-source, writting in C++/Qt)

03 - Starter Files and Environmental Variables

app.js

  • imports all the JS/node libraries that we'll use
  • loads environment variables from variables.env

start.js

  • entrypoint for the app
  • connects to our MongoDB database, writes errors to console
  • listens for GET requests on the configured port (default: 7777)

Start our example app

  • npm start
    • looks up scripts section in ``package.json
    • calls node ./start.js and lots of other stuff (a file-change watcher, webpack etc)

04 - Core Concept - Routing

How are URLs routed in ExpressJS?

routes/index.js

  • contains a route for /
const express = require('express');
const router = express.Router();

router.get('/', (req, res) => {
  res.send('Hey! It works!');
});

module.exports = router;
  • the routes are then imported into app.js and used there
const routes = require('./routes/index');

const app = express();

# all routes starting with '/' will be handled by these routes
app.use('/', routes);

# we could have multiple route handlers:
# app.use('/admin', adminRoutes);
  • URL query params are available in req.query

05 - Core Concept - Templating

  • res.render renders templates in a response

  • we'll use the pug templating language (original name: jade)

    • templates are stored in ./views folder
  • pug uses significant whitespace for nesting tags

  • comments must be on their own line

hello.pug

h1 Hello, world!

// comment shown in HTML
//- comment not shown in HTML

//- <div class="wrapper">
div.wrapper 
  p This is a simple paragraph.

  //- <p id="second"></p>
  p#second Yet another paragraph.
  
  //- <img class="animal" src="dog.jpg" alt="Dog"/>
  img.animal(src="dog.jpg" alt="Dog")

Put variables / request params into template

// take a param from the render function and put it into a string
h2 #{name} is #{age} years old.

// when putting params into tags, we need to use ES6 string literals.
img.animal(src="dog.jpg" alt=`This is ${name}'s dog.`)

Use base templates

If you have a layout.pug template with a section block content, you can fill/override it in another template like this:

extends layout

block content
  h1 This is the headline!

06 - Core Concept - Template Helpers

helpers.js contains functions and constants that can be used as variables in pug templates.

  • is imported in app.js and stored in the variable h
  • helpers are made available in templates by putting them into a middleware
const helpers = require('./helpers');

app.use((req, res, next) => {
  res.locals.h = helpers;
  next();
});

For example, siteName from helpers.js

exports.siteName = `Now That's Delicious!`;

is now available in templates by using:

html
  head
    title= `${h.siteName}`

moment.js

  • relative date/timestamp library, can be used in templates like this:
p The sale ends in #{h.moment().endOf('day').fromNow()}

07 - Core Concept - Controllers and the MVC Pattern

To follow the MVC pattern, we will outsource the functions that are called in the routes into ./controllers, i.e. we turn this:

router.get('/', (req, res) => {
  res.render('hello', {
    name: req.query.name,
    age: req.query.age
  });
});

into this:

const storeController = require('../controllers/storeController')
router.get('/', storeController.homePage);

and putting the function into ./controllers/storeController.js:

exports.homePage = (req, res) => {
  res.render('hello', {
    name: req.query.name,
    age: req.query.age,
  });
}

08 - Core Concept - Middleware and Error Handling

  • middleware: function that get called for each incoming request before the response is sent back
    • the output of one middleware is handed over to the input of the next middleware by calling next()

Route-specific middleware

  • is only called for specific routes
// example middleware that adds a name to each incoming request
exports.myMiddleware = (req, res, next) => {
  req.name = 'John';
  next();
};

This can now be called in a route before the actual response:

router.get('/', storeController.myMiddleware, storeController.homePage);

Since myMiddleware hands over its output using next() and is called before homePage, the latter function can access the req.name value:

exports.homePage = (req, res) => {
  console.log(req.name);
  res.render('index');
};

Global middleware

  • is called for every request that comes into the application before it get to the router

app.js

  • contains all global middleware for this app, i.e. all calls to app.use(...) are invoking some kind of middleware for each incoming request

cookie-parser

We can set a cookie in any middleware like this:

exports.myMiddleware = (req, res, next) => {
  req.name = 'John';
  res.cookie('name', 'evil-cookie', {maxAge: 1000000});
  next();
};

It will be added to res.cookies (which can be viewed in Chrome DevTools

Application >> Storage >> Cookies) by cookie-parser, which is added as a global middleware in app.js:

const cookieParser = require('cookie-parser');

app.use(cookieParser());

09 - Creating our Store Model

  • unlike Elasticsearch, MongoDB uses schema by default

models/Store.js

const storeSchema = new mongoose.Schema({
  name: {
    type: String,
    trim: true,
    required: 'Please enter a store name!' // true would work, but emits an unhelpful error msg
  },
  slug: String,
  description: {
    type: String,
    trim: true
  },
  tags: [String]
});

We add middleware to create slugs for userfriendly URLs automatically. Our model(s) are then imported once in start.js as a singleton.

require('./models/Store');

10 - Saving Stores and using Mixins

  • goal: create the /add page with a form for adding stores/restaurants to our database
  • we'll not yet store it in the db, but merely return the form data back as JSON after the form is POSTed

Mixins

Mixins are templates that can be imported into other templates and used like functions with parameters, e.g. the mixin views/mixins/_storeForms.pug:

mixin storeForm(store = {})
    p #{store.name}

has a store parameter. It can be used in views/editStore.pug like this:

extends layout
include mixins/_storeForm

block content
    div
        +storeForm({name: 'Joe's Pizza'})

Sending and parsing HTML forms

We can create a form in pug that POSTs data to the URL /add:

mixin storeForm(store = {})
    form(action="/add" method="POST")

        label(for="name") Store Name
        // 'name' field value has to correspond to the schema we want to store
        // this in later on
        input(type="text" name="name")
        
        input(type="submit" value="Save")

To make this work, we need to add a post handler to the /add route:

router.get('/add', storeController.addStore);
router.post('/add', storeController.createStore);

For now, the createStore controller will just respond with the form fields/values:

exports.createStore = (req, res) => {
    // will not work without the 'body-parser' middleware in app.js
    res.json(req.body);
};

11 - Using Async Await

We need to look at async and await, before we can work with the mongogoose MongoDB client.

Since we're using ES Promises with mongoose:

// start.js
mongoose.Promise = global.Promise;

we can use async/await when saving data to MongoDB. (We don't necessarily want to wait for it to finish, and we also want to avoid "callback hell".)

Here's our controller. It reads the form data from the request body. Since we use a strict schema, we don't have to validate the data (at least for now):

// Import our database schema from models/Store.js.
const Store = mongoose.model('Store');

exports.createStore = async (req, res) => {
   const store = new Store(req.body);
   await store.save();
    res.redirect('/');
};

The save() operation might fail, but Wes doesn't want to wrap it in a try/catch block, so he uses some middleware error catching magic to avoid it. It is defined in handlers/errorHandlers.js:

rts.catchErrors = (fn) => {
  return function(req, res, next) {
    return fn(req, res, next).catch(next);
  };
};

and is imported and wrapped around the createStore controller in routes/index.js:

// { foo } object destructuring: allows us to just import 'catchErrors'
// instead of importing 'errorHandlers' and then referencing 'errorHandlers.catchErrors'
const { catchErrors } = require('../handlers/errorHandlers')

router.post('/add', catchErrors(storeController.createStore));

12 - Flash Messages

  • goal: add color-highlighted messages to an existing page (e.g. to signal that an order has been submitted successfully) -- in contrast to creating/redirecting the user to a new page

We need to add the connect-flash middleware to app.js (it depends on session support, which depends on a session store):

const session = require('express-session');
const mongoose = require('mongoose');
const MongoStore = require('connect-mongo')(session); // MongoDB session store
const flash = require('connect-flash');

// Sessions allow us to store data on visitors from request to request
// This keeps users logged in and allows us to send flash messages
app.use(session({
  secret: process.env.SECRET,
  key: process.env.KEY,
  resave: false,
  saveUninitialized: false,
  store: new MongoStore({ mongooseConnection: mongoose.connection })
}));

app.use(flash());

To make the flash messages trigger our CSS, we need to store them where they can be checked by our templates (i.e. in our pass-variables-to-templates middleware):

app.use((req, res, next) => {
  [...]
  res.locals.flashes = req.flash();
  next();
});

Wes built his own flash message handling in layout.pug, which looks like this:

    block messages
      if locals.flashes
        .inner
          .flash-messages
            - const categories = Object.keys(locals.flashes)
            each category in categories
              each message in flashes[category]
                .flash(class=`flash--${category}`)

                  // "!= message" means HTML-parse "message" instead of showing it verbatim
                  p.flash__text!= message

                  // remove the message on button click
                  button.flash__remove(onClick="this.parentElement.remove()") &times;

After all this setup, we can use finally flashes, but they will only show up in the response to the next request (or when we redirect the current request)!

exports.createStore = async (req, res) => {
    const store = new Store(req.body);
    await store.save();
    
    req.flash('success', `Successfully Created ${store.name}. Care to leave a review?`);
    // redirect to the page of the store we just created
    res.redirect(`/store/${store.slug}`);
};

13 - Querying our Database for Stores

  • goal: list all stores when hitting the homepage or /stores

First, we create a reusable "store overview" as a mixin (views/mixins/_storeCard.pug):

mixin storeCard(store = {})
    .store
        //- String interpolation with fallback: ${store.photo || 'store.png'} .
        //- If the store has no photo, show store.png instead.
        img(src=`/uploads/${store.photo || 'store.png'}`)
        
        h2.title
            a(href=`/store/${store.slug}`)

        .store__details
            p= store.description

We can now use it in the stores.pug view:

extends layout

include mixins/_storeCard

block content
    .inner
    h2= title

    .stores
        each store in stores
            +storeCard(store)

The getStores controller retrieves all stores from the db and renders them in the stores template:

exports.getStores = async (req, res) => {
    const stores = await Store.find();
    res.render('stores', {title: 'Stores', stores: stores});
};

Finally, we make the stores view available under the / and /stores routes. Since our db query is async, we need to wrap the controllers in our catchErrors() middleware:

router.get('/', catchErrors(storeController.getStores));
router.get('/stores', catchErrors(storeController.getStores));

14 - Creating an Editing Flow for Stores

  • goal: edit existing stores and update them in the database

This lesson introduces wildcard routes, e.g. /stores/:id/edit which use URLs like /stores/123/edit. The ID 123 will be stored in req.params.id.

We'll add two such routes, one for GETting the editable page of a store and another for POSTing the edited store (thereby updating the store's information in the database):

router.get('/stores/:id/edit', catchErrors(storeController.editStore));
router.post('/add/:id', catchErrors(storeController.updateStore));

Now, we need to add the controllers that these routes call. editStore fetches the store with the given ID and renders it in the editStore view:

exports.editStore = async (req, res) => {
   // .findOne() returns (a Promise of ) the store with the given ID from MongoDB
    const store = await Store.findOne({ _id: req.params.id });

    res.render('editStore', {
        title: `Edit ${store.name}`,
        store: store,
    });
};

We can use the existing editStore view without change for both adding new and editing existing stores:

extends layout

include mixins/_storeForm

block content
    .inner
        h2= title
        +storeForm(store)

but we need to adapt the storeForm mixin slightly to make it work for both cases.

The action /add/${store._id || ''} means that the form is POSTed to /add/ if the given store has no _id field (i.e. if it is a new store) and to /add/${store._id} otherwise. In the case of an existing store, we prefill the inputandtextarea` fields and set the checkboxes for all tags that were previously checked:

mixin storeForm(store = {})
    form(action=`/add/${store._id || ''}` method="POST" class="card")

        label(for="name") Store Name
        input(type="text" name="name" value=store.name)

        label(for="description") Description
        //- unlike <input>, <textarea> has no "value" field
        textarea(name="description" value=store.description)= store.description

        //- define choices of tags to add to the restaurant (list of checkboxes)
        - const choices = ['Wifi', 'Open Late', '24/7', 'Vegan', 'Cats']

        //- store all existing tags in an array. create a new array if
        //- this is a new store.
        - const tags = store.tags || []

        ul.tags
            each choice in choices
                .tag.tag__choice

                    //- checked=(tags.includes(choice)) means we're calling JS
                    //- to see if the tags array includes the given choice.
                    //- If so, the box will be checked.
                    input(type="checkbox" id=choice value=choice name="tags" checked=(tags.includes(choice)))
                    label(for=choice) #{choice}
        input(type="submit" value="Save" class="button")

The updateStore controller updates the given store (req.params.id) in MongoDB and redirects to that store's edit page afterwards:

exports.updateStore = async (req, res) => {
    // MongoDB: find and update the store in one go
    const query = { _id: req.params.id };
    const update = req.body;
    const options = {
        new: true, // return updated store instead of the old one
        runValidators: true // always run the model validators
                            // (by default, they're only run during model creation!)
    }

    // findOneAndUpdate() does only run, if we call exec() on it
    const store = await Store.findOneAndUpdate(query, update, options).exec();

    res.redirect(`/stores/${store._id}/edit`);
};

Finally, we will add an "edit" button to each store card, which links to that store's edit page:

mixin storeCard(store = {})
    .store
        .store__hero
            .store__actions
                a(href=`/stores/${store._id}/edit`)
                    button edit

15 - Saving Lat and Lng for each store

  • custom data types in MongoDB

In the storeForm mixin, we use location[address] for the name field of the <input> element.

        label(for="address")
        input(type="text" id="address" name="location[address]")

We can only do this because we use the body-parser middleware in app.js with the extended: tree setting, which allows us to use inputs with nested data in them (location[address] refers to location.address in our Store MongoDB schema).

app.use(bodyParser.urlencoded({ extended: true }));

What if we try to access store.location.address but store.location doesn't exist?

input(type="text" id="address" name="location[address]" value=store.location.address)

16 - Geocoding Data with Google Maps

17 - Quick Data Visualization Tip

18 - Uploading and Resizing Images with Middleware

19 - Routing and Templating Single Stores

20 - Using Pre-Save hooks to make Unique Slugs

21 - Custom MongoDB Aggregations

22 - Multiple Query Promises with Async Await

Hints for creating the whole thing from scratch

  • run npm init to create a package.json
  • create a route and a minimal start.js

Get rid of MongoDB warnings

DeprecationWarning: current URL string parser is deprecated, and will be removed in a future version. To use the new parser, pass option { useNewUrlParser: true } to MongoClient.connect.

DeprecationWarning: current Server Discovery and Monitoring engine is deprecated, and will be removed in a future version. To use the new Server Discover and Monitoring engine, pass option { useUnifiedTopology: true } to the
 MongoClient constructor.
  • can be avoided by setting
mongoose.connect(process.env.DATABASE, {
  useNewUrlParser: true,
  useUnifiedTopology: true
});

Hot reloading

  • uneccessary bloat: we need nodemon to make this work, but neither concurrently (adds 59 packages) or webpack (adds 178 packages) are needed

Slug creation

In Video 12, Wes claims that we need to combine store creation and saving into const store = await (new Store(req.body)).save() in order to make store.slug available. This is not true. It will just work the way it was:

exports.createStore = async (req, res) => {
    const store = new Store(req.body);
    await store.save();

    req.flash('success', `Successfully Created ${store.name}. Care to leave a review?`);
    // redirect to the page of the store we just created
    res.redirect(`/store/${store.slug}`);
};