A one day introductory workshop on Node!
By the end of this workshop you'll be able to answer the following questions:
- What is Node? Why do you need it?
- What is a server?
- What is npm? How do you use it?
- What are node modules?
- What is a package.json?
- What are 'module exports' and 'require'?
- What is a module?
We'll be doing the following things:
- Installing Node
- Creating a basic 'Hello world' Node server
- Learning how to use the 'fs' core module to read and write from the file system
- Setting up 'nodemon' for your development environment
- Creating a route for your server
In one sentence it's 'Javascript for the server'!!
Wikipedia says:
Node.js is a packaged compilation of Google’s V8 JavaScript engine, the libuv platform abstraction layer, and a core library, which is itself primarily written in JavaScript.
Node has an asynchronous, event-driven I/O model. Node is an interface to the V8 JavaScript runtime – the JavaScript interpreter that runs in the Chrome browser.
This is a pretty good intro video: What is Node.js Exactly? - a beginners introduction to Nodejs
Node v4.0.0 is now available and supports lots of ES6 features. (ES6 - aka ECMAScript 6 or ECMAScript 2015 - is a newer version of JavaScript with a bunch of new features. It's not 100% supported everywhere yet, but it will be eventually).
For a detailed description of node checkout @heron2014's ''what-is-node tutorial'.
If you want to learn ES6 Look up @benjaminlees's tutorial!
Servers are computer programs that receive requests from other programs, the clients and send back a response e.g share data, information or hardware and software resources.
In a typical web app a server could perform some of these functions:
- Handle manipulation of data in the database
- File manipulation
- Authentication
- Lots of secret logic
Client side code sends requests to a server which sends back data to the front end which can then be displayed.
Front end Javascript is executed in the site visitor's browser whereas server-side code runs on a site's web server.
Download Node from the NodeJS website - versions 4 and 6 are the long-term supported versions.
If Node is installed properly, you should be able to invoke the interactive node.js shell by typing node
into the command line. You can then type any Javascript code and it will be executed.
$ node
> console.log('Hello World');
Hello World
The shell is a great way to test simple one liners. In order to escape from the shell, simply press Ctrl + C.
Modules are just small programs you can integrate with the bigger program you are writing.
'Core' Node modules come with Node automatically. But there are thousands of open-source, 3rd-party Node modules that other clever people have written. You can download useful 3rd-party modules (also known as "packages") from the Node Package Manager.
The npm website says:
npm makes it easy for JavaScript developers to share and reuse code, and it makes it easy to update the code that you're sharing.
Node comes with npm installed. However, npm gets updated more frequently than Node does so make sure it's the latest version.
Check the version
$ npm --version
// the version should be 3.10.5
If it is not the latest version then type:
sudo npm install npm -g
By default, npm installs any dependency in the local mode - the node_modules directory in the folder where Node application is present. The -g
flag means npm will be installed globally so not just for a particular project.
Now you can use npm to create and install external modules for your project!
Type this command into your terminal in the folder you want to create the project.
npm init
This takes you through the process of creating a file called a package.json
which is a json object that contains information about your project and its dependencies. It looks a bit like this:
{
"name": "autocomplete-nikki",
"version": "1.0.2",
"description": "find words in English dictionary with autosuggestion!",
"main": "index.js",
"directories": {
"test": "test"
},
"scripts": {
"test": "node test/test.js"
},
"repository": {
"type": "git",
"url": "https://github.com/nikhilaravi/autocomplete.git"
},
"keywords": [
"autocomplete",
"english",
"dictionary",
"suggestions"
],
"author": "nikhilaravi",
"license": "ISC",
"bugs": {
"url": "https://github.com/nikhilaravi/autocomplete/issues"
},
"homepage": "https://github.com/nikhilaravi/autocomplete",
"engines": {
"node": ">= 0.10"
},
"dependencies": {
"express": "^4.14.0"
},
"devDependencies": {
"pre-commit": "^1.0.7"
}
}
The package.json is the file that makes it possible for others to install and run your project, once you've built it. When someone installs your project, npm will look through the package.json and install any modules that your project depends on, hence the name dependencies. devDependencies are modules you only need for development purposes (so testing frameworks, for example); for more in-depth explanation, read this Stack Overflow thread.
Key attributes in the package.json:
- name
- version
- description
- homepage
- author
- contributors
- scripts - these are commands that you define. You can run them from the command line using
npm run <NAMEOFSCRIPT>
. We'll use this to run tests and start our server - dependencies
- repository - repository type and url of the git repository
- main
- keywords
Remember:
- You can't add comments to a .json file!
- Watch out for trailing commas in your json object as this will cause errors when trying to
npm install
!
npm install <NAME OF MODULE> --save
This creates a folder called node_modules
within your project and installs the node module's files in it. The --save
flag saves the name of the module into the dependencies
in the package.json. You can leave out the --save
flag, but then you would have to remember to add the module to your package.json manually.
The version of the module is also shown using the 'Semver Rule'
Remember to add node_modules
to your .gitignore
file so you don't push all the node module files up to Github! The package.json is all someone would need to run your project. Once the package is in node_modules, you can use it in your code.
When you clone a Node.js project or pull down the latest version of your project from GitHub you should install any new node modules using the command.
npm install
or
npm i
This installs the modules listed in your package.json to the node_modules folder e.g.
var mandrill = require('mandrill'); //Mandrill is a module for setting up an email client
We've already mentioned two types of module:
- Core Node modules that come with Node.js
- 3rd-party modules that you can download and use into your code (sometimes called "packages")
There's a third type of module in Node - your own modules!
To keep your project organised, you might separate out some of your code into a different file. You can require
a file into another one, just as you would with a core Node module or a 3rd-party module. Hence, these are also called modules.
We will try this out now. Create a file called library.js
.
"use strict";
var books = {
"Emma": {
author: 'Jane Austen',
published: 'December 25, 1815'
},
"Harry Potter and the Goblet of Fire": {
author: 'JK Rowling',
published: 'July 8, 2000'
},
"Eloquent Javascript": {
author: 'Marijn Haverbeke',
published: '2011'
}
};
function getBookAuthor(name) {
return books[name].author
};
function getDatePublished(name) {
return books[name].published
};
To enable the functions to be used by other files, you need to export the functions.
Add this line to the end of library.js:
module.exports = {
getBookAuthor: getBookAuthor,
getDatePublished: getDatePublished
};
Other files can then import library.js
and use the methods returned.
In another file (e.g. librarian.js):
var Library = require('./library.js');
Library.getBookAuthor("Emma");
Node modules and JavaScript files do not need an extension (e.g. 'js') when being specified inside require()
. However it can be helpful to add '.js' to the end of your local JavaScript files so it's easier to differentiate between your own files and node modules.
E.g.
var http = require('http');
var myFile = require('myFile.js');
Inside the call to require
, the path to the file needs to be specified.
A required module prefixed with '/' is an absolute path to the file. For > example, require('/home/marco/foo.js') will load the file at /home/marco/foo.js.
A required module prefixed with './' is relative to the file calling > require(). That is, circle.js must be in the same directory as foo.js for > require('./circle') to find it.
Use the '../' prefix for a file in the directory above the current file.
Relative paths are explained in more detail in the Node Docs
Node.js has several modules compiled into the binary e.g. 'http', 'fs', 'querystring' . These are called 'core modules'.
Core modules are always preferentially loaded. For instance, require('http') will always return the built-in http module, even if there is a file by that name.
Create a file called server.js and add the following code:
var http = require('http');
// set the port for the server
var port = process.env.PORT || 8000;
This loads in the http
module which we will use to create a web server that processes requests using http.
When a request reaches the server, we need a way of responding to it. In comes the handler
function. This is just a function that takes in a request
and response
object and sends the response back to the client along with some information. We call it handler because it 'handles' requests and responses.
function handler(request, response) {
//display 'HELLO WORLD' when the user is on the home page
var url = request.url; //e.g. '/'
if (url.length === 1) {
response.writeHead(200, {"Content-Type": "text/html"});
response.end("HELLO WORLD!");
}
}
http.createServer(handler).listen(port);
console.log('node http server listening on http://localhost:' + port);
Inside the call to http.createServer()'
we pass in our handler function. The handler gets called every time someone connects to the app. Let's take a closer look at the two parameters the handler function takes:
-
request - this object contains the information about what the visitor asked for including name of the page that was requested, the settings, and any fields filled in on a form.
-
response - this is the object which contains the information that you send back to the user.
response.writeHead(200)
sends back a status code of 200 in the response header to the request to say that everything is okay. The response headers are used to describe the resource being fetched or the behavior of the server.
The status code is a 3-digit http status code, like 404. The second argument of the writeHead
function are the response headers. This function must be called before response.end()
.
Now start the server! In the command line type:
node server.js
In the browser navigate to http://localhost:8000
. You should see your 'Hello world' message!
We're going to create an index.html file and then serve it up when the user navigates to the home page. You can use the template below or create your own!
<!DOCTYPE html>
<html>
<body>
<h1>Hello</h1>
<img src="http://www.harmonycentral.com/forum/filedata/fetch?id=31139011&d=1398720429"/>
</body>
</html>
Back to our server file. Import the 'fs' core node module - this allows reading and writing to the file system.
Then read in the index.html file and save it as a variable. __dirname
is used to get the name of the directory that the currently file resides in.
Don't forget the '/' after __dirname
😉
var fs = require('fs');
fs.readFile(__dirname + '/index.html', function (error, index) {
// send the index back to the client!
});
Then send back the html file in the response, with response.write()
.
response.write(index);
response.end();
Psst! A shorthand version is to leave out response.write()
and pass the data directly into response.end()
. Change the call to response.end
to be:
response.end(index);
Restart the server and you should see the page from index.html!
For different requests you might want to carry out different functions or retrieve specific data. These can be specified through the URL of the request and you can create specific routes in your server to handle these requests.
Lets look at an example:
Lets say you have a button on your home page that when clicked sends an http request with a url of /cat
. It might look something like this:
var request = new XMLHttpRequest();
request.onreadystatechange = function() {
if(request.readyState === 4) {
displayImage(request.responseText);
};
};
request.open("GET", "/cat", true);
request.send();
Don't worry about the displayImage
function for now - let's pretend it takes html string with an image and adds it to the page.
On the server you would look at the url of the request and if it is /cat
you want to send back the cat image requested by the client:
function(request, response) {
var url = request.url
if (url.indexOf('/cat') > -1) {
// check if the url contains /cat and if so send back a link to a cat image e.g. from a database or an API
response.writeHead(200, {"Content-Type": "text/html"});
response.end('<img src="http://charts.stocktwits.com/production/original_24310845.jpg?1404265667"/>')
}
}
You can also have nested urls e.g. /cat/lion
. In the server you can split the request url to get this additional information:
var type = url.split('/')[2].toString();
When you add css or js files in your index.html they won't load the same as just a purely front end app. You need to create a route in your server to serve these files.
Create a main.css
file and add a link to this file to your index.html
page e.g.
<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" type="text/css" href="main.css">
</head>
<body>
<h1>Hello</h1>
<img src="http://www.harmonycentral.com/forum/filedata/fetch?id=31139011&d=1398720429"/>
</body>
</html>
Add this to your main.css
file so you can see that the changes have been applied in the browser:
h1 {
color: blue;
}
Add the following code to your server handler as a final else
branch.
fs.readFile(__dirname + url, function(error, file){
if (error){
console.log(error);
response.end();
} else {
var ext = url.split('.')[1];
response.writeHead(200, {'Content-Type' : 'text/' + ext});
response.end(file);
}
});
Okay let's break down this function:
- To load the main.css file the client sends a request to the server with a url of
main.css
. - The url is the filename, so the function can read the contents of the file from the filesystem.
- To know what type of file it is (e.g. css, js, html), we split the url on the
.
. So forurl = main.css
,url.split('.')
would result in an array with two elements:[main, css]
. We take the second element of the array and set this to be the content type in the response header. - Finally we send back the contents of the file in the response of the http request.
Instead of having to restart the server every time you change any of your code you can install a module that will monitor for any changes in your javascript files and automatically restart the server!
npm install -g nodemon
In the scripts part of your package.json add the following line:
"start": "nodemon server.js"
You can then start the server by typing npm start
in the command line.
If you make a post request and send some data with the request you need a way of reading the data on the server side. For this you need to listen for the 'data' event on the request.
var requestData = '';
request.on('data', function(chunk) {
requestData += chunk;
});
If you are using APIs, you don't want to push the API Keys up to Github. To keep the keys secret we want to save them as environment variables.
Follow @nelsonic's tutorial to learn how this works! https://github.com/dwyl/learn-environment-variables
npm install -g n
Use this to set which version of node you are running. In the future you may want to switch between versions of node or use io.js and you can easily do this by typing 'n' into the terminal and toggling the up and down arrows.