Criando aplicações web com Node.js Charles Viegas [email protected] @charlesviegas Referências • nodejs.org • http://expressjs.com/ • https://github.com/expressjs/body-parser • https://github.com/expressjs/session • Livro: Node.js: aplicações web real-time com Node.js – Caio Ribeiro – Casa do código Código fonte da solução https://github.com/charlesviegas/minicurso-nodejs.git INTRODUÇÃO AO NODE.JS O que é o Node.js? • Javascript no lado do servidor • Construído sobre o engine V8 do Chrome • Executa por linha de comando • Projetado para alta concorrência • Single thread • Não bloqueante • Utiliza o framework CommonsJS Não bloqueia IO • Esperar pedidos de I/O degrada a performance. • O Node usa evento em JS para anexar retornos a chamadas de IO. Exemplo BLOQUEANTE NÃO BLOQUEANTE ... String sql = "SELECT c FROM Contato"; … var callback = function(err, rows) { if(err) throw err; console.log(rows); }); Query q= em.createQuery(sql); List result = query.getResultList(); ... con.query("SELECT * FROM Contato", callback); … Event Loop • Ao invés de threads, o Node usa um loop de eventos, aliviando o overhead da troca de contexto. Instalação e Configuração • Acesse o site http://nodejs.org • Faça download do executável e instale de forma padrão • Verifique se a instalação funcionou corretamente executando os comandos: – node –version – npm --version Prática 1 - Olá Javaneiros • Abra o terminal e navegue para a pasta: – curso\pratica1 • Execute o comando: node olajavaneiros.js – Observe a saída no próprio terminal • Execute o comando: node olajavaneiroshttp.js – Abra seu navegador e acesse a url impressa no terminal NPM • Node Package Manager – Gerencia os módulos do node, baixando da web os módulos e suas dependências • Baseado em linha de comando – npm install nome_do_modulo --save – npm install – g nome_do_modulo – npm remove nome_do_modulo – npm update nome_do_modulo – npm list package.json • O Node utiliza o arquivo package.json para descrever os metadados do projeto. – Nome, versão, dependências • Deve estar presente na pasta raiz de cada projeto. package.json { "name": “nome_do_projeto", "description": "Meu primeiro app em Node.js", "author": “Charles Viegas <[email protected]>", "version": "1.0.0", "private": true, "dependencies": { "modulo-1": "1.0.0", "modulo-2": "~1.0.0", "modulo-3": ">=1.0.0" }, "devDependencies": { "modulo-4": "*" } } Prática 2 - Criando o package.json • Durante o curso iremos criar uma pequena aplicação para controlar uma lista de presentes para casamento. • Crie a pasta raiz: – Nome: casalista • Crie dentro da pasta o arquivo: – Nome: package.json – Conteúdo: name, description, author, version e private • Execute dentro da pasta raiz o seguinte comando: – npm install Prática 2 - Criando o package.json { "name": “casalista", "description": “App para lista de casamento em Node.js", "author": “Seu Nome <[email protected]>", "version": "1.0.0", "private": true } Objetos globais • global: assim como nos browsers o Node também possui um contexto de alto nível. • process: contém os dados do processo de execução. • console: usado para imprimir mensagens nas saídas de fluxo padrões. • Buffer: usado para manipular dados binários. • require: usado para importação de módulos Objetos globais • __filename: o nome do arquivo que está sendo executado • __dirname: o nome do diretório que reside o arquivo que está sendo executado. • module: uma referência para o módulo corrente. CommonJS • O Node utiliza nativamente o padrão CommonJS para organização e carregamento dos módulos. modulo1.js modulo2.js var m2 = require(“./modulo2.js”); m2(“Ola Javaneiros”); module.exports = function(msg){ console.log(msg); } DOMINANDO EXPRESS O que é o Express? • Escrever código usando a API HTTP do Node é muito trabalhoso. • O Express é um módulo para desenvolvimento de aplicações web de grande escala. Características do Express • MVR (Model-View-Routes) • MVC (Model-View-Controller) • Roteamento de urls via callbacks • Middleware • Interface RESTFul • Suporte a File Uploads • Integração com Template Engines Módulos do Express • A partir da versão 4, o Express passou a ser distribuído em vários módulos, sendo eles: – express: Módulo principal do express. – express-load: Carregamento dinâmico de arquivos JS. – express-session: Controle de sessões do usuário. – method-override: Traduz método POST para PUT e DELETE. – body-parser: Parser de campos do html para objeto JSON. – cookie-parser: Parser de cookie para objeto JSON. – ejs: Engine de template. Prática 3 - Instalando o express • Altere o seu package.json incluindo como dependência as últimas versões dos módulos: – express, express-load, express-session, method-override, bodyparser, cookie-parser e ejs • Dentro da pasta raiz execute o comando: – npm install • O npm vai criar a pasta node_modules e dentro delas todos os seus módulos Prática 4 – Criando as pastas do projeto • Copie para a pasta raiz do seu projeto a estrutura de pastas e arquivos definida em curso\pratica4 Prática 4 – Criando as pastas do projeto • controllers: vai conter os nossos controladores (MVC). • routes: vai conter os nossos roteadores. • public: vai conter os css e imagens. • views: vai conter as nossas views (MVC). • app.js: arquivo JS principal do nosso projeto. Atenção: por questões de simplificação não teremos o model (MVC) neste curso. Iniciando um servidor Express // Importa e instancia um servidor express var express = require("express"), app = express(); // Configurações futuras // Sobe o servidor HTTP na posta 3000 app.listen(3000, function () { console.log(“Servidor está no ar."); }); Prática 5 – Iniciando um servidor express • Conforme exemplo anterior altere o arquivo app.js • Execute o comando: – node app.js • Abra o navegador no endereço: http://localhost:3000 • Você deve receber a mensagem: Cannot GET / Definindo views ... // Configurações futuras // Define a pasta onde irão ficar as views app.set("views", __dirname + "/views"); // Define o template engine usado nas views app.set("view engine", "ejs"); ... Definindo conteúdo estático // Configurações futuras ... // Define a pasta public para conteúdo estático app.use(express.static(__dirname + "/public")); ... Definindo as rota inicial // Configurações futuras ... // Responde ao acessar o localhost:3000/ app.use("/", function(req, res){ res.send("Ola Javaneiros"); }); ... Prática 6 – Definindo views, conteúdo estático e rota inicial • Altere o arquivo app.js com os passos feitos nos últimos três slides. • Reinicie o node visualizar as alterações. • Para ver os resultados, abra seu navegador no endereço http://localhost:3000/ • Para testar o conteúdo estático abra também o endereço http://localhost:3000/app/css/style.css Express-load • Utilizaremos o express-load que carrega modulos dentro da instância do express simplementes informando a pasta onde estas scripts estão. • Portanto, ao invés de usar o require(“nome_do_modulo”), nós utilizaremos o express-load para esta função. Express-load // Importa o módulo express-load var express = require("express"), load = require("express-load"), app = express(); ... // Carrega todas as scripts da pasta controller e routes load("controllers") .then("routes") .into(app); Prática 7 – Express-load • Adicione o express-load conforme slides anteriores. • Reinicie o node visualizar as alterações. Criando Roteadores • É comum que os roteamento no express fiquem separados em módulos externos para melhorar a legibilidade do código. • Portanto iremos remover o código abaixo da app.js: app.use("/", function(req, res){ res.send("Ola Javaneiros"); }); Criando Roteadores • E criaremos um módulo a parte só para roteamento. module.exports = function (app) { var home = app.controllers.home; app.get("/", home.index); app.post("/entrar", home.entrar); }; Criando Controladores • Os controladores controlam a sequência de execução e a navegação entre as páginas. • Também é recomendável a separação da sua lógica num módulo separado. module.exports = function (app) { var HomeController = { index: function (req, res) { res.render("home"); }, entrar: function (req, res) { } }; }; return HomeController; Prática 8 – Roteadores e Controladores • Crie os roteadores e controladores conforme slides anteriores. Criando views com EJS • Para criação das Views utilizaremos o EJS. • O EJS permite imprimir o conteúdo de objetos JSON em partes do html • Também permite a inclusão de loops, estruturas condições e includes Criando views com EJS • Organizaremos nosso código da seguinte forma: – header.ejs – footer.ejs – home.ejs header.ejs <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content="width=device-width, initial-scale=1"> <title>Javaneiros - Lista de presentes de casamento</title> <link rel="stylesheet" href="/bootstrap/css/bootstrap.css"/> <link rel="stylesheet" href="/bootstrap/css/bootstrap-theme.css"/> <link rel="stylesheet" href="/app/css/style.css" /> </head> <body> footer.ejs <footer class="navbar navbar-default navbar-fixed-bottom"> <div class="container"> Javaneiros 2014 </div> </footer> <script src="/bootstrap/js/jquery.js"></script> <script src="/bootstrap/js/bootstrap.js"></script> </body> </html> home.ejs <% include ./header %> <div class="container"> <form action="/entrar" method="post" role="form" class="form-signin"> <h2>Lista de presentes</h2> <input type="text" name="noivo" placeholder="Noivo" class="form-control" required="required" autofocus="autofocus"/> <input type="text" name="noiva" placeholder="Noiva" class="form-control" required="required"/> <button type="submit" class="btn btn-lg btn-primary btn-block">Ver lista</button> </form> </div> <% include ./footer %> Prática 9 – Views • Crie as views header, footer e home conforme slides anteriores. Body-parser • O express utiliza o modulo body-parser para realizar a conversão dos dados submetidos por um formulário html em objeto JSON. Body-parser var express = require("express"), bodyParser = require("body-parser"), load = require("express-load"), app = express(); … // Inclui o parser para formato json app.use(bodyParser.json()); // Inclui parser para urlencoded app.use(bodyParser.urlencoded({extended: true})); ... Express-session • Recurso que permite manter dados do usuário no servidor. • Para acessar o objeto a ser mantido na sessão basta acessar req.session Express-session var express = require("express"), bodyParser = require("body-parser"), load = require("express-load"), expressSession = require("express-session"), app = express(); … app.use(expressSession({ secret: "keyboard cat", resave: false, saveUninitialized: true })); ... Prática 10 – Body-parser e Session • Adicione o body-parser e o express-session conforme slides anteriores. Manipulando session • Agora iremos gravar na sessão do usuário os dados submetidos pela tela de home. • Para isso, iremos alterar o controlador do home. Manipulando Session ... var HomeController = { ... entrar: function (req, res) { var lista = { noivo : req.body.noivo, noiva : req.body.noiva, presentes : [] }; req.session.lista = lista; res.redirect("/presentes"); } }; ... Prática 11 – Manipulando session • Complete o código do HomeController conforme slide anterior. • Após submeter o formulário você verá a seguinte mensagem: Cannot GET /presentes routers\presente.js module.exports = function (app) { var presentes = app.controllers.presentes; app.get("/presentes", presentes.index); app.get("/comprar/:id", presentes.comprar); app.post("/presentes", presentes.incluir); }; controllers\presente.js module.exports = function (app) { var PresentesController = { index: function (req, res) { res.render("presentes", {presentes: req.session.lista.presentes}); }, incluir: function (req, res) { var lista = req.session.lista; var presente = { nome: req.body.nome, valor: req.body.valor }; lista.presentes.push(presente); res.redirect("/presentes"); }, comprar: function (req, res) { var id = req.params.id; req.session.lista.presentes[id].comprado = true; res.redirect("/presentes"); } }; return PresentesController; }; view\presentes.ejs <% include ./header %> <div class="container"> <div class="page-header"> <h1>Lista de presentes</h1> </div> <div class="row"> <!– ****************** Formulário --> </div> <div class="row"> <!– ****************** Tabela --> </div> <div class="row"> <a href="/">Sair</a> </div> </div> <% include ./footer %> view\presentes.ejs - formulário <div class="col-xs-12"> <form action="/presentes" method="post" role="form"> <div class="row"> <div class="col-xs-4"> <input type="text" name=“nome" placeholder="Presente" class="form-control"/> </div> <div class="col-xs-4"> <input type="text" name=“valor" placeholder="Valor" class="form-control"/> </div> <div class="col-xs-4"> <button type="submit" class="btn btn-default">Incluir</button> </div> </div> </form> </div> view\presentes.ejs - tabela <div class="col-xs-12"> <table class="table table-bordered table-striped"> <thead><tr> <th>Presente</th> <th>Valor</th><th>Ação</th> </tr></thead> <tbody> <% presentes.forEach(function(presente, index) { %> <tr> <td><%- presente.nome %></td> <td><%- presente.valor %></td> <td> <% if (!presente.comprado) {%> <a href="/comprar/<%- index %>">Comprar</a> <%}%> </td> </tr> <% }) %> </tbody> </table> </div> Prática 12 – Final • Crie o roteador, controlador e a view para a tela de presentes conforme slides anteriores.