Skip to content

CSCI-E32/Week-13

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

24 Commits
 
 
 
 
 
 
 
 

Repository files navigation

Week 13: Node and Express and Angular's $resource

because, just because

A Simple Server

the most basic express application

  1. Install express on the server's package.json
  • npm install express
  1. Require express
  • var express = require('express');
  1. Create an express "app"
  • var app = express();
  1. Use the app to create a route to the root ('/')
app.get('/', function(req, res) {
  res.send('okay, yeah');
});

Note: res.send(something) is just an express wrapper for res.end(something), so it's doing a write(something) followed by an end(). It also does some encoding and status code checks.

Send as json

together

  1. Take the array from the ListController in the angular app
  • var people = [ ... ];
  1. Send the json
  • app.send(people);
  • `app.send(JSON.stringify(people));``
  1. Use app.json() to "send" the array
  • app.json(people);
  • send() is actually smart enough to know when you're sending json, but it's nice to be explicit
  1. Check it in a browser

Logging Middleware

together

  1. Use an anonymous function before any routes
  • app.use(function(req, res, next){ ... })
  • this is a pipeline for the request and response streams
  • the function will require a next(); function to be called if you want anything to happen after it
  • if you place it after any routes, it will never run
  1. Console log the date, the method and the url
  • console.log((new Date()).toString() + " " + req.method + " " + req.url)

Note: This is not a great logging solution, it's not handling errors or status codes. For that you'll want a real express middleware logging solution. It's just logging is such a nice and easy thing to show you some easy middleware.

Routing

together

  1. Create a route for a listing
  • app.get('/people', function(req, res){ ... });
  • the root route is fine for us right now, but it's not RESTful
  • RESTful does have a few rules, like having a single "resource" for making queries against

An Angular app for this!

together

  1. Look at the angular app
  2. Run the angular app
  • npm install
  • gulp

$resource service

together

  1. Add ng-resource to the project
  • bower install angular-resource
  1. Include it as a dependency on the app (in app.js)
  • angular.module('resthitter', ['ui.bootstrap', 'xeditable', 'ngRoute', 'ngResource']).config({ ... });
  1. Create a People service
  • create a new file under the services directory, People.js
  • these are normally created as factories
  • angular.module('resthitter').factory('People', ['$resource', function($resource){ ... }]);
  1. Add the $resource dependency
  • $resource is a wrapper for $http
    • it is meant to allow you to easily set up hitting restful APIs
    • it has 4 main functions: query(), get(), save(), and delete() (and remove())
    • these map to the primary RESTful methods you'll be using (GET all, GET a single, POST a new entry or change, and DELETE a single
    • $resource docs
  1. Return a $resource object
  • return $resource('http://localhost:7000/people/:id');
  • Note: the full url that can be parsed internally by $resource to give it the right urls for each RESTful action
  • Also note: $resource is as simple as it can be out of the box, but you can extend it to whatever you need, (for example, if you need it to make a PUT request against an API)

Use that service in the ListController

together

  1. Inject People into ListController
  2. Use People.query();
  • var people = People.query();
  1. Check your network activity in the browser and you will see it made a request
  2. Make the edits to have that data display in the template's ng-repeat

Add CORS to your API with middleware

together

  1. Create a new middleware option
  • app.use(function(req, res, next){ ... });
  1. Add the following two headers:
  • res.header("Access-Control-Allow-Origin", "*");
  • res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
  1. Call next(); to pass it through to the next middleware
  • next();
app.use(function(req, res, next) {
  res.header("Access-Control-Allow-Origin", "*");
  res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
  next();
});

Note: Cross Origin Resource Sharing is the most accepted way to subvert the Single Origin Policy, which is a security policy to prevent malicious actions. It's up to the server, and it's generally accepted that you're not going to be allowing * unless you're a truly public API -- usually a read-only API will do that. Usually people will at least limit the sharing to *.domain.com.

Fill out the Detail page

together

  1. Get the value of the current page's id from the $routeParams
  • inject $routeParams
  • current id is $routeParams.id
    • it's getting this from the $routeProvider config in app.js
  1. Set the value of the person variable to the $resource.get() for the current id
  • var person = People.get({id: $routeParams.id});
  • that object parameter is the parameter set in the $resource definition in the service
  • so it's equivalent to sending a GET request to: http://localhost:7000/people/1
  1. Make sure the appropriate value gets displayed on the template

Create the single route

together

  1. Create a new route, similar to the format of our angular $routeParams and $resource
  • app.get('/people/:id', function(req, res){ ... });
  • the id is now available on req.params.id, courtesy of express
  1. If we find that req.params.id in the people array, return the corresponding person object
  • forEach or some other loopish structure
    • people.forEach(function(person){ ... });
    • for(var index = 0; i < people.length; i++){ ... }
  • an if checking the id of each array element
    • if(person.id == req.params.id){ ... }
  • return the matching element
    • return person;
  1. If we don't find it, send a 404
  • res.status(404).end();
app.get('/people/:id', function(req, res){
  people.forEach(function(person){
    if(person.id == req.params.id){
      res.json(person);
    }
  });
  res.status(404).end();
});

Note: forEach, (as well as any vanilla loop) is going to be a synchronous process. This means it will be blocking. This is partly why people store data in noSQL solutions when working with Node, so lookups can be offloaded to Mongo or Redis or whatever with an async call. But we can also "simulate asynchronisity" by simply wrapping the interior with a setTimeout()... But then the 404 will get called before it has a chance to check them, so you'll have to handle that somehow. You could use a lib for handling async functions or even a promise pattern -- or just use a counter. But these are the problems running Node has.

var counter = 0;
people.forEach(function(person){
  setTimeout(function(){
    if(person.id == req.params.id){
      res.json(person);
    }
    counter++;
    if(counter == people.length){
      // then we've gone through them all and it wasn't found
      res.status(404).end();
    }
  }, 10);
});

Save an edit

together

  1. Since we have included xeditable, we have some directives available to us
  • add the attribute editable-text to the name and email texts
  • use them the same way you would use ng-model
  • <h1 editable-text="dc.person.name">{{ dc.person.name }}</h1>
  • Hint: wrap the editable field in a <span> to avoid the line across the page
  1. Create a save button
  • use ng-click to have that button call a save function in your controller
  1. You can call a save directly from the existing $resource you got from the .get() in that controller
  • dc.person.$save();
  • Note: this returns a promise, so we can chain it with a .then(success, failure);
  • use this success function to console log something to let us know if we're succeeding!
dc.person.$save().then(function(){
 console.log("success!");
});

Note: this can also be done with a call to the $resource factory

  • Name.save({}, dc.person, function(){ console.log("success!"); });
  • where the first param is $resource params
  • the second param is the POST's body
    • in this case, we're posting the json for the person
  • the third param is a success callback

Set up the post on the server

together

  1. Set up an app.post for the /people route
  • app.post('/people', function(req, res){ ... });
  1. Install body-parser
  • npm install body-parser
  • this is express middleware
  • we need this to get the POST data
  1. Require the body-parser
  • var bodyParser = require('body-parser');
  1. Use the body-parser to parse for json
  • app.use(bodyParser.json());
  • now in our routes, req.body has been populated with parsed json, assuming that's what was posted
  1. Get the person object from req.body
  • var person = req.body;

Find and update the person in the people array

on your own

  1. Loop through the array in some way (for() or .forEach() or whatever)
  2. Find the match on id
  3. Set that index to the post data we just got
  4. Don't forget to end the response!
  • res.end();

Redirect to the List

together

  1. Inject the $location service to DetailController
  • .controller('DetailController', ['People', '$location', function(People, $location){ ... }]);
  • $location is an angular wrapper for window.location
  1. Use the .then() promise chained success function to change the location
  • $location.path('/');

Add a new item

together

  1. Add a button to the list view
  • actually, a link will be more to the point, but we can make it look like a button
  • <a class="btn btn-info" href="/#/new">new</a>
  1. Create the new route -- just have it go to the detail template
  • .when('/new', { ... })
  • just populate the route the same as the edit route
  1. In the Controller, check for an undefined $routeParam.id
  • if($routeParam.id === undefined){ ... }
  1. If it's not defined, create a new People object and populate it with some default values
  • var dc.person = new Person();
  • dc.person.name = "placeholder";
  • dc.person.twitter = "@placeholder";
  • if it is defined, just do what we were doing before
  1. Try it

Fix the POST server route

together

  1. If it gets through the forEach, instead of throwing a 404, check if the id is undefined
  • if(postedPerson.id === undefined){ ... }
  1. Give the posted person an appropriate id
  • postedPerson.id = people.length + 1;
  1. Add the postedPerson to the people array
  • people.push(postedPerson);
  1. What are the obvious flaws with this?
  • hint: if we added a delete...

Delete an entry

together

  1. Add a delete button to the list template with an ng-click
  • <button class="btn" ng-click="lc.delete(person.id)">delete</button>
  1. Create the delete(id) function
  • lc.delete = function(id){ ... };
  1. Send a DELETE request to the server
  • People.delete({id: id});
  • remember, this is like sending a DELETE request to http://localhost:7000/people/:id
  1. Try it
  2. Alter CORS to allow DELETE
  • res.header('Access-Control-Allow-Methods', 'POST, GET, PUT, DELETE, OPTIONS');
  1. Create a delete route on the server
  • app.delete('/people/:id', function(req, res){ ... })
  1. Find and delete the entry
  • people.splice(index);
  • don't forget to res.end()
  1. Return a 404 if you didn't find it
  • res.status(404).end();
  1. In the controller, after a delete, refresh the data
  • People.delete({id: id}, function(){ ... });
  • lc.people = People.query();
  • Note: in a real scenario, you probably wouldn't want to be hammering the server -- if your DELETE came back with a success, you should remove it from the existing lc.people array
    • thankfully, this isn't real. Go home.

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published