Skip to content

Latest commit

 

History

History
287 lines (238 loc) · 11.7 KB

README.md

File metadata and controls

287 lines (238 loc) · 11.7 KB

Koa Sample App (handlebars templating + RESTful API using MySQL, on Node.js)

This is the result of a self-learning exercise on how to put together a complete Node.js/io.js MySQL-driven Koa app.

When I started with Node.js (using Express), I found plenty of tutorials & examples on individual elements, but found it hard to stitch everything together; this was even more true with Koa. Being new to Node / Express / Koa, I found that understanding came very much by assembling all the different bits together.

While the Koa ‘hello world’ certainly doesn’t flatter to deceive, there’s obviously a long way to go after it. This does some of that. It is a template not a mini-tutorial. It puts a lot of the components of a complete system together: neither mini nor tutorial!

Having worked it all out, this is now largely a ‘note-to-self’ aide-memoire, but I hope it might help others in a similar situation. No one else will be doing all the same things, but seeing components operating together may be helpful.

It is of course simplistic, but unlike many tutorials, it assembles together many of the components of a complete system: in this case, basic interactive tools for viewing, adding, editing, and deleting (CRUD), with passport login/authentication, and a matching RESTful API to do the same (using basic access authentication). Many system may not require an API, but the API nicely doubles-up for ajax functions. Of course, real systems do much more, but generally build on these core functions.

The database includes a couple of tables with related data and referential integrity – one step beyond where most tutorials go. Hence code is included for handling basic validation and referential integrity errors returned from the database.

Otherwise I’ve stripped it down to essentials. There’s no pretty styling! There’s no great UI. There’s no client-side JavaScript effects, no ajax. Just the bare building-blocks.

There are also sample integration/acceptance tests using mocha / supertest / chai / cheerio.

I don’t have time to put together a full tutorial, but I’ve tried to make everything clear & well structured, and I’ve liberally commented the code.

Design choices

For all that it’s rather bleeding-edge, Koa makes development far simpler than classic callback-style Node.js with Express. Yield may be just a stepping-stone to ES7 await, but it works. JavaScript may be looked down on (misunderstood?) in some quarters, but I do find it vastly better to work with than PHP :)

With Node.js v12.0 (stable) out, and io.js as an alternative, I believe Koa can now be considered* for production use.

The app is built with a modular approach. There are three (composed) sub-apps: the bare bones of a public site, a web-based password-protected admin system using handlebars-templated html pages, and a REST API. Each of these is structured in a modular fashion; mostly each admin module has JavaScript handlers to handle GET and POST requests, and a set of handlebars templates; each API module has JavaScript handlers for GET, POST, PATCH, DELETE requests.

The highly-structured applications I work on require ACID SQL databases with referential integrity, so MongoDb was out for me. MySQL and PostgreSQL should be pretty similar, but PostgreSQL is not yet so well supported for Koa. (Actually, one of my first Koa applications is using MongoDB/monk; there are few changes – the models replicate monk functions, but add functionality to those functions which update the database).

For some people, a JavaScript framework will work better. If you’re happy to plan out your own preferred structure, designing your own patterns means one less component to learn / conform to.

There’s always things others would do differently. If you have better ways of doing things, it will either be down to my preferences, or my ignorance! If you notice anything I’ve got wrong or have real improvements, let me know.

Admin app

The admin app just does basic adding/editing/deleting. Any real app will do much more, but will generally include and build on these basics.

I find Handlebars offers a good minimal-logic templates (mustache is too limiting, but I like to work with HTML).

The main admin/app.js sets up the database connection, handlebars templating, passport authentication, 404/500 handling, etc.

The app is then divided into modules (members, teams in this case), each with a handler/controller and a set of templates. The handlers have one function for each method/route, and either render a view, redirect (e.g. after POST), or throw an error.

API

The API returns JSON or XML (or plain text) according to the Accepts request header.

The main api/app.js sets up the database connection, content negotiation, passport authentication, and 404/500 handling.

The api/members.js and api/teams.js then handle the API requests. I use PATCH in preference to PUT so that a subset of entity fields can be supplied (correctly, a PUT will set unsupplied fields to null); otherwise everything is very straightforward REST API, hopefully all following best practice.

Models

Models sit above the sub-apps, as they are used by both the admin app and the API.

I use light-weight models just to manage all operations which modify stored data in the database; individual handlers are responsible for obtaining data they require to render their templates (using SQL queries).

Dependencies

While very basic, this sample app incorporates together many of the components of a real application; as well as handlebars templates & MySQL, there’s static file serving, body-parser for post data, compression,passport logins with remember-me, logging, flash messages, etc, and mocha/chai/cheerio for testing (I’ve ignored i18n which would introduce considerable complexity).

Note that if you wish to set this up locally, you will need e.g. admin.localhost, api.localhost, and www.localhost set up in /etc/hosts. Currently, io.js is required as Node.js v12.0 doesn’t yet support template strings.

Demo

There is a running version of the app at koa-sample-app.movable-type.co.uk.

File structure

.
├── apps
│   ├── admin
│   │   ├── ajax
│   │   │   └── routes-ajax.js
│   │   ├── index
│   │   │   ├── templates
│   │   │   │   └── index.html
│   │   │   └── routes-index.js
│   │   ├── login
│   │   │   ├── templates
│   │   │   │   └── login.html
│   │   │   ├── handlers-login.js
│   │   │   └── routes-login.js
│   │   ├── logs
│   │   │   ├── templates
│   │   │   │   └── logs.html
│   │   │   └── routes-logs.js
│   │   ├── members
│   │   │   ├── templates
│   │   │   │   ├── members-add.html
│   │   │   │   ├── members-delete.html
│   │   │   │   ├── members-edit.html
│   │   │   │   ├── members-list.html
│   │   │   │   └── members-view.html
│   │   │   ├── handlers-members.js
│   │   │   └── routes-members.js
│   │   ├── teams
│   │   │   ├── templates
│   │   │   │   ├── teams-add.html
│   │   │   │   ├── teams-delete.html
│   │   │   │   ├── teams-edit.html
│   │   │   │   ├── teams-list.html
│   │   │   │   └── teams-view.html
│   │   │   ├── handlers-teams.js
│   │   │   └── routes-teams.js
│   │   ├── templates
│   │   │   ├── 400-bad-request.html
│   │   │   ├── 404-not-found.html
│   │   │   ├── 500-internal-server-error.html
│   │   │   ├── errpartial.html
│   │   │   └── navpartial.html
│   │   ├── app-admin.js
│   │   └── passport.js
│   ├── api
│   │   ├── app-api.js
│   │   ├── handlers-members.js
│   │   ├── handlers-teams.js
│   │   ├── routes-auth.js
│   │   ├── routes-members.js
│   │   ├── routes-root.js
│   │   ├── routes-teams.js
│   │   └── validate.js
│   └── www
│       ├── templates
│       │   ├── 404-not-found.html
│       │   ├── 500-internal-server-error.html
│       │   ├── contact.html
│       │   ├── index.html
│       │   └── navpartial.html
│       ├── app-www.js
│       ├── handlers-www.js
│       └── routes-www.js
├── config
│   ├── db-development.json
│   └── db-production.json
├── lib
│   └── lib.js
├── logs
├── models
│   ├── member.js
│   ├── modelerror.js
│   ├── team.js
│   ├── team-member.js
│   └── user.js
├── public
│   └── css
│       ├── app.css
│       ├── base.css
│       └── www.css
├── test
│   ├── admin.js
│   └── api.js
├─ app.js
├─ LICENSE
├─ package.json
└─ README.md

Database schema

-- Schema for ‘koa-sample-web-app-api-mysql’ app

create table Member (
  MemberId  integer unsigned not null auto_increment,
  Firstname text,
  Lastname  text,
  Email     text not null,
  primary key       (MemberId),
  unique  key Email (Email(24))
) engine=InnoDB charset=utf8 auto_increment=100001;

create table Team (
  TeamId integer unsigned not null auto_increment,
  Name   text not null,
  primary key (TeamId)
) engine=InnoDB charset=utf8 auto_increment=100001;

create table TeamMember (
  TeamMemberId integer unsigned not null auto_increment,
  MemberId     integer unsigned not null,
  TeamId       integer unsigned not null,
  JoinedOn     date not null,
  primary key            (TeamMemberId),
  key         MemberId   (MemberId),
  key         TeamId     (TeamId),
  unique key  TeamMember (MemberId,TeamId),
  constraint Fk_Team_TeamMember   foreign key (TeamId)   references Team   (TeamId),
  constraint Fk_Member_TeamMember foreign key (MemberId) references Member (MemberId)
) engine=InnoDB charset=utf8 auto_increment=100001;

create table User (
  UserId    integer unsigned not null auto_increment,
  Firstname text,
  Lastname  text,
  Email     text not null,
  Password  text,
  ApiToken  text,
  Role      text,
  primary key       (UserId),
  unique  key Email (Email(24))
) engine=InnoDB charset=utf8 auto_increment=100001;

Test data

-- Test data for ‘koa-sample-web-app-api-mysql’ app

INSERT INTO Member VALUES 
 (100001,'Juan Manuel','Fangio','[email protected]'),
 (100002,'Ayrton','Senna','[email protected]'),
 (100003,'Michael','Schumacher','[email protected]'),
 (100004,'Lewis','Hamilton','[email protected]');

INSERT INTO Team VALUES 
 (100001,'Ferrari'),
 (100002,'Mercedes'),
 (100003,'McLaren');
 
INSERT INTO TeamMember VALUES 
 (100001,100001,100001,'1956-01-22'),
 (100002,100001,100002,'1954-01-17'),
 (100003,100002,100003,'1988-04-03'),
 (100004,100003,100001,'1996-03-10'),
 (100005,100003,100002,'2010-03-14'),
 (100006,100004,100002,'2007-03-18'),
 (100007,100004,100003,'2013-03-17');
 
INSERT INTO User VALUES
  (100001,'Guest','User','[email protected]','$2a$12$G5op7sX70HUXfFbI8tPQuuhnWz4bwqbWQeIN9KFyklH5OhLgQbnU6',null,'guest'),
  (100002,'Admin','User','[email protected]','$2a$12$jEG0N4wNwuc20WQxN1VzduijVnlzLgBNn2N6Uq1pNjN45VhUyNf4W',null,'admin');

The full sample app is around 1,000 lines of JavaScript.