4. Execução da aplicação
Pretendemos agora operar a aplicação fora dos controladores IDE e STS (para o servidor) e do Webstorm (para o cliente).
4.1. Implantação do serviço web num servidor Tomcat
Vimos no parágrafo 2.11.9 como criar um arquivo WAR para o Tomcat. Repetimos aqui o procedimento. Em primeiro lugar, para preservar o que já existe, duplicamos o projeto Eclipse [rdvmedecins-webapi-v3] para [rdvmedecins-webapi-v4].
![]() |
O ficheiro [pom.xml] é alterado da seguinte forma:
<modelVersion>4.0.0</modelVersion>
<groupId>istia.st.spring4.mvc</groupId>
<artifactId>rdvmedecins-webapi-v4</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>war</packaging>
<name>rdvmedecins-webapi-v3</name>
<description>Gestion de RV Médecins</description>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.0.0.RELEASE</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-security</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-tomcat</artifactId>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>istia.st.spring4.rdvmedecins</groupId>
<artifactId>rdvmedecins-metier-dao-v2</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
</dependencies>
As alterações devem ser feitas em dois locais:
- linha 5: é necessário indicar que se vai gerar um arquivo WAR (Web ARchive);
- linhas 23-27: é necessário adicionar uma dependência do artefacto [spring-boot-starter-tomcat]. Este artefacto inclui todas as classes do Tomcat nas dependências do projeto;
- linha 26: este artefacto é o [provided], ou seja, os arquivos correspondentes não serão incluídos no WAR gerado. Com efeito, esses arquivos estarão disponíveis no servidor Tomcat onde a aplicação será executada;
Além disso, é necessário configurar a aplicação web. Na ausência do ficheiro [web.xml], isto é feito com uma classe que herda de [SpringBootServletInitializer]:
![]() |
A classe [ApplicationInitializer] é a seguinte:
package rdvmedecins.web.config;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.boot.context.web.SpringBootServletInitializer;
public class ApplicationInitializer extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
return application.sources(AppConfig.class);
}
}
- linha 6: a classe [ApplicationInitializer] estende a classe [SpringBootServletInitializer];
- linha 8: o método [configure] é redefinido (linha 7);
- linha 9: é fornecida a classe [AppConfig], que configura o projeto;
Feito isto, poderá ser necessário atualizar o projeto Maven (eu tive de o fazer): [clic droit sur projet / Maven / Update project] ou [Alt-F5].
Para executar o projeto, pode-se proceder da seguinte forma:
![]() |
- no [1], executa-se o projeto num dos servidores registados no IDE Eclipse;
- no [2], seleciona-se o [tc Server Developer], que está presente por predefinição. Trata-se de uma variante do Tomcat;
Obtém-se o seguinte resultado:
![]() |
Isto é normal. Recorde-se que o serviço web não tem URL nem [/] nos seus métodos. Quando se tenta o URL e o [/getAllMedecins], obtém-se a seguinte resposta:
![]() |
Isto é normal. O serviço web está protegido.
Agora, vamos iniciar o cliente [rdvmedecins-angular-v2] no WebStorm:
![]() |
No [1], inserimos o URL do novo serviço web [http://localhost:8080/rdvmedecins-webapi-v4]. Obtemos o seguinte resultado:

Para executar a aplicação fora do IDE STS, existem várias soluções. Aqui está uma delas.
Descarregue uma versão do Tomcat [http://tomcat.apache.org/download-80.cgi] (julho de 2014):
![]() |
Selecione em [1] uma versão compactada e descompacte-a em [2]. Volte a STS:
![]() |
- no separador [Servers], clica-se com o botão direito do rato na aplicação [rdvmedecins-webapi-v4] e seleciona-se a opção [Browse Deployment Location];
- em [4]: copie a pasta [rdvmedecins-webapi-v4];
![]() |
- no [5], cole a pasta [rdvmedecins-webapi-v4] na pasta [webapps] do Tomcat;
- em [6], executa-se o ficheiro de comando [startup.bat] (o servidor Tomcat integrado em STS deve estar parado). Abre-se uma janela DOS para apresentar os registos do Tomcat. Estes devem indicar que a aplicação [rdvmedecins-webapi-v4] foi iniciada.
Para verificar isso, execute novamente o cliente Angular [rdvmedecins-angular-v2] no Webstorm:
![]() |
No [1], insere-se o URL do novo serviço web [http://localhost:8080/rdvmedecins-webapi-v4]. Obtém-se o seguinte resultado:

4.2. Implantação do cliente Angular no servidor Tomcat
Agora que o serviço web foi implementado no Tomcat, vamos implementar o cliente Angular num servidor também. Pode muito bem ser o servidor que já aloja o serviço web. Vamos seguir por este caminho.
Em primeiro lugar, duplicamos o cliente [rdvmedecins-angular-v2] para [rdvmedecins-angular-v3] e efetuamos as seguintes alterações:
![]() |
- em [1], tudo foi movido para uma pasta [app] ;
- em [1], foi eliminada a pasta [bower-components], que continha as várias bibliotecas CSS e JS necessárias para o projeto. Todos estes elementos foram copiados para a pasta [lib] [2];
- no [1], o ficheiro [app.html] foi renomeado para [index.html];
O ficheiro [index.html] foi alterado para ter em conta as alterações nos caminhos dos recursos utilizados:
<!DOCTYPE html>
<html ng-app="rdvmedecins">
<head>
<title>RdvMedecins</title>
...
<!-- o CSS -->
...
<link href="lib/bootstrap-theme.min.css" rel="stylesheet"/>
<link href="lib/bootstrap-select.min.css" rel="stylesheet"/>
</head>
<!-- controlador [appCtrl], modelo [app] -->
<body ng-controller="appCtrl">
<div class="container">
...
</div>
<!-- Núcleo do Bootstrap JavaScript ================================================== -->
<script type="text/javascript" src="lib/jquery.min.js"></script>
<script type="text/javascript" src="lib/bootstrap.min.js"></script>
<script type="text/javascript" src="lib/bootstrap-select.min.js"></script>
<script type="text/javascript" src="lib/footable.js"></script>
<!-- AngularJS -->
<script type="text/javascript" src="lib/angular.min.js"></script>
<script type="text/javascript" src="lib/ui-bootstrap-tpls.min.js"></script>
<script type="text/javascript" src="lib/angular-route.min.js"></script>
<script type="text/javascript" src="lib/angular-translate.min.js"></script>
<script type="text/javascript" src="lib/angular-base64.min.js"></script>
<!-- módulos -->
...
<!-- serviços -->
...
<!-- diretivas -->
...
<!-- controladores -->
....
</body>
</html>
Além disso, o controlador [loginCtrl] foi alterado para apontar para o servidor correto, de modo a evitar que o utilizador tenha de introduzir o seu URL:
// credenciais
app.serverUrl = "http://localhost:8080/rdvmedecins-webapi-v4";
app.username = "admin";
app.password = "admin";
Feito isto, vamos executar o ficheiro [index.html]:
![]() | ![]() |
Em seguida, liguemo-nos ao serviço web. Deve funcionar. Depois de verificado isto, desliguemos o servidor Tomcat. Vamos reutilizar o servidor integrado do STS.
No STS, vamos copiar todo o conteúdo da pasta [rdvmedecins-angular-v3/app] para a pasta [webapp] do projeto [rdvmedecins-webapi-v4] (separador Navigator) [1]:
![]() |
Feito isto, inicie o [2], o servidor VMware do STS e, em seguida, solicite o URL [http://localhost:8080/rdvmedecins-webapi-v4/app/index.html]:
![]() |
Temos um problema de direitos no [3]. Não é de admirar, uma vez que protegemos o serviço web. Temos de declarar que o acesso ao ficheiro [/app/index.html] é livre. Voltemos ao Eclipse:
![]() |
Recordamos que os direitos de acesso foram definidos na classe [SecurityConfig]. Vamos alterá-la da seguinte forma:
@Override
protected void configure(HttpSecurity http) throws Exception {
// CSRF
http.csrf().disable();
// a palavra-passe é transmitida através do cabeçalho Authorization: Basic xxxx
http.httpBasic();
// o método HTTP OPTIONS deve ser autorizado para todos
http.authorizeRequests() //
.antMatchers(HttpMethod.OPTIONS, "/", "/**").permitAll();
// a pasta [app] está acessível a todos
http.authorizeRequests() //
.antMatchers(HttpMethod.GET, "/app", "/app/**").permitAll();
// apenas a função ADMIN pode utilizar a aplicação
http.authorizeRequests() //
.antMatchers("/", "/**") // todos os URL
.hasRole("ADMIN");
}
- linhas 11-12: autorizamos todos a ler a pasta [app] e o seu conteúdo. Para o fazer, inspiramo-nos nas linhas anteriores.
Agora, reiniciemos o servidor Tomcat de STS e, em seguida, solicitemos novamente o URL [http://localhost:8080/rdvmedecins-webapi-v4/app/index.html]:

Desta vez, está tudo bem.
4.3. Os cabeçalhos CORS
Talvez se lembrem que tivemos bastante dificuldade em lidar com os cabeçalhos CORS. No exemplo anterior:
- o serviço web está em URL [http://localhost:8080/rdvmedecins-webapi-v4];
- o cliente HTML está no URL [http://localhost:8080/rdvmedecins-webapi-v4/app];
O cliente HTML e o serviço web encontram-se, portanto, no mesmo servidor [http://localhost:8080]. Não há, portanto, conflitos CORS, uma vez que estes só ocorrem quando o cliente e o servidor não se encontram no mesmo domínio. Deveria ser possível verificar isso. Voltamos ao STS:
![]() |
A geração ou não dos cabeçalhos CORS é controlada por um valor booleano definido na classe [ApplicationModel]:
// dados de configuração
private boolean CORSneeded = true;
Definimos o valor booleano acima para «false», reiniciamos o serviço web e voltamos a solicitar o URL [http://localhost:8080/rdvmedecins-webapi-v4/app/index.html]. Verificamos que a aplicação funciona.
4.4. Implementação do cliente Angular num tablet Android
A ferramenta [Phonegap] [http://phonegap.com/] permite gerar um executável para dispositivos móveis (Android, IoS, Windows 8, ...) a partir de uma aplicação HTML / JS / CSS. Existem várias formas de atingir este objetivo. Utilizamos a mais simples: uma ferramenta disponível online no site da Phonegap [http://build.phonegap.com/apps].
![]() |
- antes de [1], poderá ser necessário criar uma conta;
- em [1], começamos;
- em [2], escolhe-se um plano gratuito que permite apenas uma aplicação Phonegap;
![]() |
- em [3], descarregue a aplicação compactada [4] (a pasta [app] criada no parágrafo 4.2 está compactada);
![]() |
- em [5], atribua um nome à aplicação;
- no [6], compila-se a aplicação. Esta operação pode demorar 1 minuto. Aguarde até que os ícones das diferentes plataformas móveis indiquem que a compilação está concluída;
![]() |
- apenas os binários para Android ([7]) e para Windows ([8]) foram gerados;
- clique em [7] para descarregar o ficheiro binário do Android;
![]() |
- em [9], o ficheiro binário [apk] descarregado;
Inicie um emulador [GenyMotion] para um tablet Android (ver parágrafo 6.4):
![]() |
Acima, iniciamos um emulador de tablet com o API 16 do Android. Assim que o emulador estiver a funcionar,
- desbloqueie-o arrastando o fecho (se houver) para o lado e, em seguida, soltando-o;
- com o rato, arraste o ficheiro [PGBuildApp-debug.apk] que descarregou e solte-o no emulador. Este será então instalado e executado;
![]() |
É necessário alterar o URL para [1]. Para tal, numa janela de comando, digite o comando [ipconfig] (linha 1 abaixo), que irá apresentar os diferentes endereços IP do seu computador:
C:\Users\Serge Tahé>ipconfig
Configuration IP de Windows
Carte réseau sans fil Connexion au réseau local* 15 :
Statut du média. . . . . . . . . . . . : Média déconnecté
Suffixe DNS propre à la connexion. . . :
Carte Ethernet Connexion au réseau local :
Suffixe DNS propre à la connexion. . . : ad.univ-angers.fr
Adresse IPv6 de liaison locale. . . . .: fe80::698b:455a:925:6b13%4
Adresse IPv4. . . . . . . . . . . . . .: 172.19.81.34
Masque de sous-réseau. . . . . . . . . : 255.255.0.0
Passerelle par défaut. . . . . . . . . : 172.19.0.254
Carte réseau sans fil Wi-Fi :
Statut du média. . . . . . . . . . . . : Média déconnecté
Suffixe DNS propre à la connexion. . . :
...
Anote quer o endereço Wi-Fi IP (linhas 6-9), quer o endereço na rede local IP (linhas 11-17). Em seguida, utilize este endereço IP no URL do servidor web:
![]() |
Feito isto, ligue-se ao serviço web:
![]() |
Teste a aplicação no emulador. Deve funcionar. Do lado do servidor, é possível autorizar ou não os cabeçalhos CORS na classe [ApplicationModel]:
// dados de configuração
private boolean CORSneeded = false;
Isto não tem importância para a aplicação Android. Esta não é executada num navegador. Ora, a exigência dos cabeçalhos CORS provém do navegador e não do servidor.
4.5. Implantação do cliente Angular no emulador de um smartphone Android
Repetimos a operação anterior com um emulador de smartphone. Queremos verificar como o nosso cliente se comporta em ecrãs pequenos:
![]() |
- em [1], iniciamos um emulador de smartphone;
- em [2] e [3], a barra de navegação foi recolhida num menu;
![]() |
- em [4], efetua-se o início de sessão;
- em [5], a lista e o calendário estão um abaixo do outro, em vez de estarem lado a lado;
![]() |
- em [6], solicita-se a agenda;
- em [7], como o ecrã é demasiado pequeno, parte dos horários fica oculta. Foi a biblioteca [footable] que realizou este trabalho;
![]() |
- em [8], a mesma vista que anteriormente, mas desta vez com um compromisso.
No final, a nossa aplicação adapta-se bastante bem ao smartphone. Certamente poderia ser melhor, mas continua a ser utilizável.





























