Skip to content

17. [TD]: segurança do servidor web / jSON das eleições

Palavras-chave: arquitetura multicamadas, Spring, injeção de dependências, serviço web / jSON seguro, cliente / servidor

Aplicamos agora o que aprendemos no capítulo anterior ao TD das eleições. A arquitetura será a seguinte:

O processo decorrerá da seguinte forma:

  • programação do servidor;
  • programação do cliente sem a camada [ui], mas com um teste JUnit da camada [métier];
  • gravação do cliente com a camada [ui];

17.1. Support

 

Os projetos deste capítulo encontram-se na pasta [support / chap-17]. O script SQL serve para gerar a base de dados necessária para os testes.

17.2. A base de dados

A base de dados do servidor seguro deve agora conter as tabelas [USERS], [ROLES] e [USERS_ROLES]:

 

O script SQL para a criação da base de dados está disponível na documentação de suporte.

17.3. O servidor seguro

Para configurar o servidor seguro das eleições, basta copiar e colar o projeto de exemplo [intro-spring-security-server-01]:


Tarefa a realizar: renomeie os pacotes conforme indicado em [1].


Feito isto, alteramos o ficheiro [pom.xml] da seguinte forma:


<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>istia.st.elections</groupId>
    <artifactId>elections-security-webjson-metier-dao-spring-data</artifactId>
    <version>0.0.1-SNAPSHOT</version>

    <name>elections-security-webjson-metier-dao-spring-data</name>
    <description>elections spring security</description>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <java.version>1.8</java.version>
    </properties>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.2.7.RELEASE</version>
    </parent>

    <dependencies>
        <dependency>
            <groupId>istia.st.elections</groupId>
            <artifactId>elections-webjson-metier-dao-spring-data</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
        <!-- Spring Security -->
        ...
    </dependencies>
    <!-- plug-ins -->
    <build>
        ...
    </build>

</project>
  • nas linhas 23-27, substitua a dependência do projeto [intro-server-webjson-01] pela do projeto [elections-webjson-metier-dao-spring-data] abordado no parágrafo 12;
  • nas linhas 4 a 7, introduza as características do novo projeto;

Resta-nos agora alterar as classes de configuração do projeto:

[DaoConfig]


package elections.security.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;

@EnableJpaRepositories(basePackages = { "elections.security.repositories" })
@ComponentScan(basePackages = { "elections.security.dao" })
@Import({ elections.dao.config.DaoConfig.class })
public class DaoConfig {

    // constantes
    final static private String[] ENTITIES_PACKAGES = { "elections.dao.entities", "elections.security.entities" };


    @Bean
    public String[] packagesToScan() {
        return ENTITIES_PACKAGES;
    }

}

As alterações encontram-se nas seguintes linhas:

  • linhas 8, 9, 14: introduzir os nomes corretos dos pacotes;
  • linha 10: a classe importada é agora [elections.dao.config.DaoConfig] do projeto [elections-webjson-metier-dao-spring-data];

Nota: tal como foi referido, esta configuração só funciona se a classe [elections.dao.config.DaoConfig] tiver a anotação [@Configuration]. Verifique este ponto.

[SecurityConfig]


package elections.security.config;

...

@EnableWebSecurity
@ComponentScan(basePackages = { "elections.security.service" })
@Import({ WebConfig.class, DaoConfig.class })
public class SecurityConfig extends WebSecurityConfigurerAdapter {

    ...
}

As alterações encontram-se nas seguintes linhas:

  • linha 6: altere o nome do pacote;

É tudo. Faça um primeiro teste executando o teste JUnit []:

  

Execute a classe de arranque [Boot] e, em seguida, com a extensão Chrome [Advanced Rest Client], faça o seguinte teste:

Em [1], a solicitação. Em [3], a resposta. Em [2], o cabeçalho HTTP é o cabeçalho de autenticação do utilizador [admin, admin]: Authorization:Basic YWRtaW46YWRtaW4=

Veja agora as listas em competição:

17.4. O cliente do servidor seguro sem a camada [ui]

Faça um copiar/colar do projeto [elections-ui-metier-dao-webjson] para o projeto [elections-metier-dao-security-webjson].


Tarefa a realizar: no [1], renomeie os pacotes, se necessário, e elimine os da camada [ui] e das classes de arranque [boot].


Vamos agora proceder tal como foi feito no parágrafo 16.5.

17.4.1. Configuração do Maven

Apenas a parte relativa à identificação do projeto muda:


<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>istia.st.elections</groupId>
    <artifactId>elections-metier-dao-security-webjson</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <description>Client jUnit du serveur web / jSON</description>
    <name>elections-metier-dao-security-webjson</name>
...

17.4.2. Reestruturação da camada [métier]

  

A interface [IElectionsMetier] sofre as seguintes alterações:


package elections.security.client.metier;

import elections.security.client.entities.ListeElectorale;
import elections.security.client.entities.User;

public interface IElectionsMetier {

    // autenticação
    public void authenticate(User user);

    // obter as listas em competição
    public ListeElectorale[] getListesElectorales(User user);

    // o número de lugares a preencher
    public int getNbSiegesAPourvoir(User user);

    // o limiar eleitoral
    public double getSeuilElectoral(User user);

    // registo dos resultados
    public void recordResultats(User user, ListeElectorale[] listesElectorales);

    // cálculo dos lugares
    public ListeElectorale[] calculerSieges(User user, ListeElectorale[] listesElectorales);

}

Todos os métodos têm como primeiro parâmetro o utilizador que pretende utilizar os recursos do servidor seguro.

A implementação [ElectionsMetier] evolui da seguinte forma:


package elections.security.client.metier;

import java.io.IOException;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;

import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;

import elections.security.client.dao.IClientDao;
import elections.security.client.entities.ElectionsConfig;
import elections.security.client.entities.ElectionsException;
import elections.security.client.entities.ListeElectorale;
import elections.security.client.entities.User;

@Component
public class ElectionsMetier implements IElectionsMetier {

  @Autowired
  private IClientDao dao;
  @Autowired
  private ApplicationContext context;

  // configuração da eleição
  private ElectionsConfig electionsConfig;

  private ElectionsConfig getElectionsConfig(User user) {
    if(electionsConfig!=null){
      return electionsConfig;
    }
    // mapadores jSON
    ObjectMapper mapperResponse = context.getBean(ObjectMapper.class);
    try {
      // consulta
      Response<ElectionsConfig> response = mapperResponse.readValue(dao.getResponse(user, "/getElectionsConfig", null),
              new TypeReference<Response<ElectionsConfig>>() {
      });
      // erro?
      if (response.getStatus() != 0) {
        // é lançada 1 exceção
        throw new ElectionsException(response.getStatus(), response.getMessages());
      } else {
        electionsConfig = response.getBody();
        return electionsConfig;
      }
    } catch (ElectionsException e1) {
      throw e1;
    } catch (IOException | RuntimeException e2) {
      throw new ElectionsException(100, e2);
    }
  }

  @Override
  public ListeElectorale[] getListesElectorales(User user) {
   ...
  }

  @Override
  public int getNbSiegesAPourvoir(User user) {
    return getElectionsConfig(user).getNbSiegesAPourvoir();
  }

  @Override
  public double getSeuilElectoral(User user) {
    return getElectionsConfig(user).getSeuilElectoral();
  }

  @Override
  public void recordResultats(User user, ListeElectorale[] listesElectorales) {
    ...
  }

  @Override
  public ListeElectorale[] calculerSieges(User user, ListeElectorale[] listesElectorales) {
    ...
  }

  @Override
  public void authenticate(User user) {
    dao.getResponse(user, "/authenticate", null);
  }
}

Tarefa: complete o código acima.


17.4.3. Configuração do Spring

  

A classe [MetierConfig] é alterada da seguinte forma:


package elections.security.client.config;

...

@ComponentScan({ "elections.security.client.dao", "elections.security.client.metier" })
public class MetierConfig {

Na linha 5, os pacotes a explorar devem ser atualizados.

17.4.4. O teste JUnit da camada [métier]

  

A classe de teste [Test01] evolui da seguinte forma:


package elections.security.client.metier.junit;

import org.junit.Assert;
import org.junit.BeforeClass;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.SpringApplicationConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;

import elections.security.client.config.MetierConfig;
import elections.security.client.entities.ElectionsException;
import elections.security.client.entities.ListeElectorale;
import elections.security.client.entities.User;
import elections.security.client.metier.IElectionsMetier;

@SpringApplicationConfiguration(classes = MetierConfig.class)
@RunWith(SpringJUnit4ClassRunner.class)
public class Test01 {

    // camada [electionsMetier]
    @Autowired
    private IElectionsMetier electionsMetier;

    // mapeador jSON
    private final ObjectMapper mapper = new ObjectMapper();

    // utilizadores
    static private User admin;
    static private User user;
    static private User unknown;

    @BeforeClass
    public static void initTest() {
        admin = new User("admin", "admin");
        user = new User("user", "user");
        unknown = new User("x", "y");
    }

    @Test()
    public void checkUserUser() {
        ElectionsException se = null;
        try {
            electionsMetier.authenticate(user);
        } catch (ElectionsException e) {
            se = e;
        }
        Assert.assertNotNull(se);
        Assert.assertEquals("403 Forbidden", se.getErreurs().get(0));
    }

    @Test()
    public void checkUserUnknown() {
        ElectionsException se = null;
        try {
            electionsMetier.authenticate(unknown);
        } catch (ElectionsException e) {
            se = e;
        }
        Assert.assertNotNull(se);
        Assert.assertEquals("401 Unauthorized", se.getErreurs().get(0));
    }

    @Test()
    public void checkUserAdmin() {
        ElectionsException se = null;
        try {
            electionsMetier.authenticate(admin);
        } catch (ElectionsException e) {
            se = e;
        }
        Assert.assertNull(se);
    }
    
    /**
     * vérification 1 : méthode de calcul des sièges on fixe en dur les listes
     */
    @Test
    public void calculSieges1() {
        // cria-se a tabela das 7 listas de candidatos
        ListeElectorale[] listes = new ListeElectorale[7];
        listes[0] = new ListeElectorale("A", 32000, 0, false);
        listes[1] = new ListeElectorale("B", 25000, 0, false);
        listes[2] = new ListeElectorale("C", 16000, 0, false);
        listes[3] = new ListeElectorale("D", 12000, 0, false);
        listes[4] = new ListeElectorale("E", 8000, 0, false);
        listes[5] = new ListeElectorale("F", 4500, 0, false);
        listes[6] = new ListeElectorale("G", 2500, 0, false);
        // calculam-se os lugares de cada uma das listas
        listes = electionsMetier.calculerSieges(admin,listes);
        // verificam-se os resultados
        Assert.assertEquals(2, listes[0].getSieges());
        Assert.assertFalse(listes[0].isElimine());
        Assert.assertEquals(2, listes[1].getSieges());
        Assert.assertFalse(listes[1].isElimine());
        Assert.assertEquals(1, listes[2].getSieges());
        Assert.assertFalse(listes[2].isElimine());
        Assert.assertEquals(1, listes[3].getSieges());
        Assert.assertFalse(listes[3].isElimine());
        Assert.assertEquals(0, listes[4].getSieges());
        Assert.assertFalse(listes[4].isElimine());
        Assert.assertEquals(0, listes[5].getSieges());
        Assert.assertTrue(listes[5].isElimine());
        Assert.assertEquals(0, listes[6].getSieges());
        Assert.assertTrue(listes[6].isElimine());
    }

    /**
     * vérification 2 : méthode de calcul des sièges on demande les listes à la couche [metier] puis on fixe en dur les
     * voix
     */
    @Test
    public void calculSieges2() {
        // cria-se a tabela das 7 listas de candidatos
        ListeElectorale[] listes = electionsMetier.getListesElectorales(admin);
        // fixam-se os votos
        listes[0].setVoix(32000);
        listes[1].setVoix(25000);
        listes[2].setVoix(16000);
        listes[3].setVoix(12000);
        listes[4].setVoix(8000);
        listes[5].setVoix(4500);
        listes[6].setVoix(2500);
        // calculam-se os lugares obtidos por cada uma das listas
        listes = electionsMetier.calculerSieges(admin,listes);
        // verifica-se os resultados
        Assert.assertEquals(2, listes[0].getSieges());
        Assert.assertFalse(listes[0].isElimine());
        Assert.assertEquals(2, listes[1].getSieges());
        Assert.assertFalse(listes[1].isElimine());
        Assert.assertEquals(1, listes[2].getSieges());
        Assert.assertFalse(listes[2].isElimine());
        Assert.assertEquals(1, listes[3].getSieges());
        Assert.assertFalse(listes[3].isElimine());
        Assert.assertEquals(0, listes[4].getSieges());
        Assert.assertFalse(listes[4].isElimine());
        Assert.assertEquals(0, listes[5].getSieges());
        Assert.assertTrue(listes[5].isElimine());
        Assert.assertEquals(0, listes[6].getSieges());
        Assert.assertTrue(listes[6].isElimine());
    }

    /**
     * vérification 3 méthode de calcul des sièges on provoque une exception
     */
    @Test(expected = ElectionsException.class)
    public void calculSieges3() {
        // cria-se uma tabela com 24 listas de candidatos, cada uma com 1 voto
        ListeElectorale[] listes = new ListeElectorale[25];
        // as 25 listas terão o mesmo número de votos (4%)
        for (int i = 0; i < listes.length; i++) {
            listes[i] = new ListeElectorale("Liste" + (i + 1), 1, 0, false);
        }
        // cálculo dos assentos - normalmente deve ser ElectionsException
        // com um limiar eleitoral de 5%
        electionsMetier.calculerSieges(admin,listes);
    }

    /**
     * enregistrement des résultats de l'élection
     * 
     * @throws JsonProcessingException
     */
    @Test
    public void ecritureResultatsElections() throws JsonProcessingException {
        // cria-se a tabela das 7 listas de candidatos
        ListeElectorale[] listes = electionsMetier.getListesElectorales(admin);
        // definem-se os votos de forma fixa
        listes[0].setVoix(32000);
        listes[1].setVoix(25000);
        listes[2].setVoix(16000);
        listes[3].setVoix(12000);
        listes[4].setVoix(8000);
        listes[5].setVoix(4500);
        listes[6].setVoix(2500);
        // calculam-se os lugares obtidos por cada uma das listas
        listes = electionsMetier.calculerSieges(admin,listes);
        // exibem-se os resultados
        for (int i = 0; i < listes.length; i++) {
            System.out.println(mapper.writeValueAsString(listes[i]));
        }
        // os resultados são gravados na base de dados
        electionsMetier.recordResultats(admin,listes);
        // verifica-se os resultados
        listes = electionsMetier.getListesElectorales(admin);
        // são apresentados os resultados
        for (int i = 0; i < listes.length; i++) {
            System.out.println(mapper.writeValueAsString(listes[i]));
        }
        Assert.assertEquals(2, listes[0].getSieges());
        Assert.assertFalse(listes[0].isElimine());
        Assert.assertEquals(2, listes[1].getSieges());
        Assert.assertFalse(listes[1].isElimine());
        Assert.assertEquals(1, listes[2].getSieges());
        Assert.assertFalse(listes[2].isElimine());
        Assert.assertEquals(1, listes[3].getSieges());
        Assert.assertFalse(listes[3].isElimine());
        Assert.assertEquals(0, listes[4].getSieges());
        Assert.assertFalse(listes[4].isElimine());
        Assert.assertEquals(0, listes[5].getSieges());
        Assert.assertTrue(listes[5].isElimine());
        Assert.assertEquals(0, listes[6].getSieges());
        Assert.assertTrue(listes[6].isElimine());
    }
}

Tarefa a realizar: compreenda este teste e verifique se ele é bem-sucedido.


17.5. O cliente do servidor seguro com uma camada [console]

Com o NetBeans, abrimos os projetos Maven [elections-metier-dao-security-webjson] e [elections-ui-metier-dao-webjson] e, em seguida, criamos um novo projeto [elections-console-metier-dao-security-webjson] através de uma cópia/colagem do projeto [elections-ui-metier-dao-webjson]:

A maioria das classes do novo projeto [3] provém do projeto [elections-ui-metier-dao-webjson] e [2], que era o cliente do servidor web / jSON não seguro:

Siga estes passos para criar o novo projeto Maven:

  • faça uma cópia e cole o projeto [elections-ui-metier-dao-webjson] no projeto [elections-ui-metier-dao-security-webjson];
  • no novo projeto:
    • elimine os pacotes [elections.client.dao, elections.client.metier, elections.client.entities];
    • no pacote [elections.client.ui], mantenha apenas a classe console [ElectionsConsole] e a sua interface [IElectionsUI];
    • renomeie os pacotes conforme indicado para [3];
    • resolva os problemas de importação nas várias classes através de [clic droit / Fix Imports];

Nesta fase, o novo projeto apresenta numerosos erros.

17.5.1. Configuração do Maven

O ficheiro [pom.xml] do novo projeto é o seguinte:


<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>istia.st.elections</groupId>
    <artifactId>elections-console-metier-dao-security-webjson</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>elections-console-metier-dao-security-webjson</name>
    <description>couche console du client web / jSON</description>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <java.version>1.8</java.version>
    </properties>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.2.7.RELEASE</version>
    </parent>

    <dependencies>
        <dependency>
            <groupId>istia.st.elections</groupId>
            <artifactId>elections-metier-dao-security-webjson</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.18.1</version>
            </plugin>
        </plugins>
    </build>

</project>
  • linhas 4-8: a identidade Maven do novo projeto;
  • linhas 22-26: a dependência do projeto [elections-metier-dao-security-webjson] que acabámos de criar no parágrafo 17.4, e que fornece as camadas [DAO] e [métier];

17.5.2. Configuração do Spring

  

A classe [UiConfig] é a seguinte:


package elections.security.client.config;

import elections.security.client.entities.User;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;

@Import(MetierConfig.class)
@ComponentScan(basePackages = {"elections.security.client.console","elections.security.client.swing"})
public class UiConfig {

  // administrador
  private final User ADMIN = new User("admin", "admin");

  @Bean
  private User admin() {
    return ADMIN;
  }

}
  • linha 8: importa-se a classe [elections.security.client.config.MetierConfig] do projeto [elections-metier-dao-security-webjson] analisado no parágrafo 17.4;
  • linha 9: declaram-se os pacotes onde se encontram os beans Spring;
  • linhas 12-18: o bean [admin] corresponde ao utilizador [admin, admin], que será utilizado pela aplicação de consola. Recorde-se que este é o único utilizador autorizado a consultar o servidor web seguro /jSON;

17.5.3. A aplicação de arranque da consola

  

A classe [ElectionsConsole] evolui da seguinte forma:


package elections.security.client.console;

import java.util.Arrays;
import java.util.Comparator;
import java.util.Scanner;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import elections.security.client.entities.ElectionsException;
import elections.security.client.entities.ListeElectorale;
import elections.security.client.entities.User;
import elections.security.client.metier.IElectionsMetier;

@Component
public class ElectionsConsole implements IElectionsUI {

    @Autowired
    private IElectionsMetier electionsMetier;

  @Autowired
  private User admin;
  
    @Override
    public void run() {
        // as listas em competição
        ListeElectorale[] listes;
        // introdução de dados
        try (Scanner clavier = new Scanner(System.in)) {
            // solicitam-se as listas em competição à camada [metier]
            listes = electionsMetier.getListesElectorales(admin);
            // efetua-se a contagem dos votos
            System.out.println("Il y a " + listes.length
                    + " listes en compétition. Veuillez indiquer le nombre de voix de chacune d'elles :");
            for (int i = 0; i < listes.length; i++) {
                boolean saisieOK = false;
                while (!saisieOK) {
                    System.out.print("Nombre de voix de la liste [" + listes[i].getNom() + "] : ");
                    try {
                        listes[i].setVoix(Integer.parseInt(clavier.nextLine()));
                        saisieOK = true;
                    } catch (ElectionsException | NumberFormatException ex) {
                        System.out.println("Nombre de voix incorrect. Veuillez recommencer");
                    }
                }
            }
        }
        // efetua-se o cálculo dos lugares
        listes=electionsMetier.calculerSieges(admin,listes);
        // regista-se os resultados
        electionsMetier.recordResultats(admin,listes);
        // ordena-se as listas por ordem decrescente de votos
        Arrays.sort(listes, new CompareListesElectorales());
        // são apresentadas
        System.out.println("\nRésultats de l'élection\n");
        for (int i = 0; i < listes.length; i++) {
            System.out.println(listes[i]);
        }
    }

    // classe de comparação de listas eleitorais
    class CompareListesElectorales implements Comparator<ListeElectorale> {

        // comparação de duas listas de candidatos com base no número de votos
        @Override
        public int compare(ListeElectorale listeElectorale1, ListeElectorale listeElectorale2) {
            // comparam-se os votos destas duas listas
            int nbVoix1 = listeElectorale1.getVoix();
            int nbVoix2 = listeElectorale2.getVoix();
            if (nbVoix1 < nbVoix2) {
                return +1;
            } else {
                if (nbVoix1 > nbVoix2)
                    return -1;
                else
                    return 0;
            }
        }
    }

}

A classe ElectionsConsole sofre poucas alterações. Basta ter em conta que, agora, os métodos da camada [métier] exigem, como primeiro parâmetro, um utilizador (linhas 31, 49, 51). Este é fornecido pela injeção nas linhas 21-22. O utilizador injetado é aquele autorizado a consultar o servidor web / jSON.


Tarefa a realizar: teste a aplicação de consola (não se esqueça de iniciar previamente o servidor seguro).


17.6. O cliente do servidor seguro com uma camada [swing]

Criamos um novo projeto NetBeans [elections-swing-metier-dao-security-webjson] através de um copiar/colar do projeto [elections-ui-metier-dao-webjson]:

No novo projeto [2]:

  • elimine os pacotes [elections.client.dao, elections.client.metier, elections.client.entities];
  • no pacote [elections.client.ui], elimine as classes [ElectionsConsole, IElectionsUI];
  • renomeie os pacotes conforme indicado em [2];
  • no pacote [elections.security.client.swing], renomeie a classe [AbstractElectionsSwing], que implementa a interface gráfica, para [AbstractElectionsMainForm], e a classe [ElectionsSwing], que implementa os gestores de eventos dainterface gráfica para [ElectionsMainForm];
  • resolva os problemas de importação nas várias classes através de [clic droit / Fix Imports];

Nesta fase, existem muitos erros.

17.6.1. Configuração do Maven

O ficheiro [pom.xml] evolui da seguinte forma:


<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>istia.st.elections</groupId>
    <artifactId>elections-swing-metier-dao-security-webjson</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>elections-swing-metier-dao-security-webjson</name>
    <description>couche swing du client web / jSON</description>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <java.version>1.8</java.version>
    </properties>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.2.7.RELEASE</version>
    </parent>

    <dependencies>
        <dependency>
            <groupId>istia.st.elections</groupId>
            <artifactId>elections-console-metier-dao-security-webjson</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.18.1</version>
            </plugin>
        </plugins>
    </build>

</project>
  • linhas 4-8: a identidade Maven do novo projeto;
  • linhas 22-26: uma dependência do projeto console que acabámos de analisar no parágrafo 17.5;

17.6.2. Configuração do Spring

Este projeto utiliza a configuração Spring do projeto «console» que importa (ver parágrafo 17.5.2).

17.6.3. As vistas da aplicação Swing

 

Este projeto irá utilizar duas vistas:

  • a vista de início de sessão [1] é nova. Serve para autenticar o utilizador que pretende utilizar a aplicação. Utiliza as classes:
    • [AbstractElectionsConnectForm], que implementa a vista;
    • [ElectionsConnectForm], que gere os eventos da vista;
  • a vista [2] já é conhecida. É a que tem sido utilizada até agora. Utiliza as seguintes classes:
    • [AbstractElectionsMainForm], que implementa a vista;
    • [ElectionsMainForm], que gere os eventos da vista;

17.6.4. A sessão

  

Quando uma aplicação Swing tem várias vistas, é necessário encontrar uma forma de uma vista poder passar informação para outra vista. Utilizamos um conceito empregue no desenvolvimento web: a sessão, que permite que as vistas associadas a um utilizador específico partilhem informação. Este conceito será implementado aqui com a ajuda de um singleton Spring que será injetado nas duas classes que gerem os eventos das duas vistas:


package elections.security.client.swing;

import elections.security.client.entities.User;
import org.springframework.stereotype.Component;

@Component
public class UiSession {
  
    // o utilizador que está conectado
  private User user;
  
  // getters e setters
... 
}
  • linha 6: a classe [UiSession] é um componente Spring;
  • linha 10: serve apenas para uma coisa: memorizar o utilizador que inicia sessão na vista [1]. A vista [2], que precisa de saber quem é esse utilizador, irá buscá-lo aí;

17.6.5. A classe de arranque

  

A classe de inicialização da aplicação gráfica é a seguinte:


package elections.security.client.boot;

import elections.security.client.console.IElectionsUI;

public class BootElectionsSwing extends AbstractBootElections {
    public static void main(String[] arguments) {
        new BootElectionsSwing().run();
    }

    @Override
    protected IElectionsUI getUI() {
        return ctx.getBean("electionsConnectForm", IElectionsUI.class);
    }
}
  • linha 5: a classe [BootElectionsSwing] estende a classe [AbstractBootElections] definida no projeto de consola presente nas dependências do projeto;
  • linhas 10-13: a classe [BootElectionsSwing] irá apresentar a vista de início de sessão [ElectionsConnectForm];
  • linhas 6-8: é o método [run] desta classe que será executado;

17.6.6. A classe [ElectionsMainForm]

A classe [ElectionsMainForm] é aquela que anteriormente se chamava [ElectionsSwing]. Ela gere os eventos da vista [AbstractElectionsMainForm]. O seu código evolui da seguinte forma:


package elections.security.client.swing;

import elections.security.client.console.IElectionsUI;
import java.awt.Dimension;
import java.awt.Toolkit;
import java.util.ArrayList;
import java.util.List;

import javax.swing.DefaultListModel;
import javax.swing.JLabel;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import elections.security.client.entities.ElectionsException;
import elections.security.client.entities.ListeElectorale;
import elections.security.client.entities.User;
import elections.security.client.metier.IElectionsMetier;

@Component
public class ElectionsMainForm extends AbstractElectionsMainForm implements IElectionsUI {

  private static final long serialVersionUID = 1L;

  // referência na camada [métier]
  @Autowired
  private IElectionsMetier metier;

  // sessão UI
  @Autowired
  private UiSession uiSession;

  // utilizador conectado
  private User user;

  // modelos das listas JList
  private DefaultListModel<String> modèleNomsVoix = null;
  private DefaultListModel<String> modèleRésultats = null;

  // as listas em competição
  private ListeElectorale[] listes;

  // listas introduzidas pelo utilizador
  private final List<ListeElectorale> listesSaisies = new ArrayList<>();
  private ListeElectorale[] tListesSaisies;

  // inicializações
  @Override
  protected void init() {
    // geração de componentes pela classe pai
    super.init();
    // inicializações locais
    modèleNomsVoix = new DefaultListModel<>();
    jListNomsVoix.setModel(modèleNomsVoix);
    modèleRésultats = new DefaultListModel<>();
    jListResultats.setModel(modèleRésultats);
    String info;
    boolean erreur = false;
    try {
      // solicitam-se as listas à camada [métier]
      listes = metier.getListesElectorales(user);
      // associa-se os nomes das listas ao menu suspenso jComboBoxNomsListes
      for (int i = 0; i < listes.length; i++) {
        jComboBoxNomsListes.addItem(String.format("%s - %s", listes[i].getId(), listes[i].getNom()));
      }
      // bem como os parâmetros da eleição
      int nbSiegesAPourvoir = metier.getNbSiegesAPourvoir(user);
      double seuilElectoral = metier.getSeuilElectoral(user);
      // inicializam-se os rótulos associados a estas duas informações
      jLabelSAP.setText(jLabelSAP.getText() + nbSiegesAPourvoir);
      jLabelSE.setText(jLabelSE.getText() + seuilElectoral);
      // mensagem de sucesso
      info = "Source de données lue avec succès";
    } catch (ElectionsException ex1) {
      // regista-se o erro
      info = getInfoForException("Les erreurs suivantes se sont produites :", ex1);
      erreur = true;
    } catch (RuntimeException ex2) {
      // regista-se o erro
      info = getInfoForException("Les erreurs suivantes se sont produites :", ex2);
      erreur = true;
    }
    // erro?
    if (erreur) {
      // exibe a informação
      jTextPaneMessages.setText(info);
      jTextPaneMessages.setCaretPosition(0);
      return;
    } else {
      jTextPaneMessages.setText("");
    }
    // estado do formulário
    Utilitaires.setEnabled(new JLabel[]{jLabelAjouter, jLabelCalculer, jLabelEnregistrer, jLabelSupprimer}, false);
    Utilitaires.setEnabled(
            new JMenuItem[]{jMenuItemAjouter, jMenuItemCalculer, jMenuItemEnregistrer, jMenuItemSupprimer}, false);
    // centrar a janela
    Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
    Dimension frameSize = getSize();
    if (frameSize.height > screenSize.height) {
      frameSize.height = screenSize.height;
    }
    if (frameSize.width > screenSize.width) {
      frameSize.width = screenSize.width;
    }
    setLocation((screenSize.width - frameSize.width) / 2, (screenSize.height - frameSize.height) / 2);

  }
...

  @Override
  public void run() {
    // memorização do utilizador
    user = uiSession.getUser();
    // inicialização da janela
    init();
    setVisible(true);
  }
}
  • linhas 32-33: injeção da sessão da aplicação;
  • linhas 112-119: a vista será ativada, tal como anteriormente, através do seu método [run];
  • linha 115: recupera-se o utilizador da sessão. Este terá sido inserido na sessão pelo código da vista de início de sessão [ElectionsConnectForm]. O código é então alterado para que o primeiro parâmetro das chamadas à camada [métier] seja esse utilizador (linhas 63, 69-70, por exemplo);

17.6.7. A vista [AbstractElectionsConnectForm]

  

Crie com o NetBeans o seguinte formulário:

 

A classe não irá implementar por si própria o método que gere o clique na opção de menu [Connexion]. Irá recorrer a um método [doConnecter] declarado como abstrato e a classe da vista será, por sua vez, declarada como abstrata. Será eliminado o método estático [main], que foi gerado automaticamente.

17.6.8. Gestão dos eventos da vista de ligação

  

A classe [ElectionsConnectForm] implementa a gestão de eventos da vista de ligação da seguinte forma:


package elections.security.client.swing;

import elections.security.client.console.IElectionsUI;
import elections.security.client.entities.ElectionsException;
import elections.security.client.entities.User;
import elections.security.client.metier.IElectionsMetier;
import java.awt.Dimension;
import java.awt.Toolkit;
import javax.swing.SwingUtilities;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Component
public class ElectionsConnectForm extends AbstractElectionsConnectForm implements IElectionsUI {

    private static final long serialVersionUID = 1L;

    // referência na camada [métier]
  @Autowired
  private IElectionsMetier metier;

  // utilizador conectado
  private User user;

  // formulário principal
  @Autowired
  private ElectionsMainForm electionsMainForm;
  
  // sessão UI
  @Autowired
  private UiSession uiSession;
  
  @Override
  protected void doConnect() {
    String info = null;
    try {
      if (isPageValid()) {
        // autenticação do utilizador
        metier.authenticate(user);
        // o utilizador é guardado na sessão
        uiSession.setUser(user);
        // a página de início de sessão é ocultada
        setVisible(false);
        // a vista principal é apresentada
        electionsMainForm.run();
      }
    } catch (ElectionsException ex1) {
      // regista-se o erro
      info = getInfoForException("Les erreurs suivantes se sont produites :", ex1);
    } catch (RuntimeException ex2) {
      // regista-se o erro
      info = getInfoForException("Les erreurs suivantes se sont produites :", ex2);
    }
    // erro?
    if (info != null) {
      // a informação é apresentada
      jTextPaneErreurs.setText(info);
      jTextPaneErreurs.setCaretPosition(0);
    }
  }

  // inicializações
  @Override
  protected void init() {
    // geração dos componentes pela classe pai
    super.init();
    // inicializações locais
...
    // centrar a janela
...
  }

  @Override
  public void run() {
    // exibe a interface gráfica
    SwingUtilities.invokeLater(new Runnable() {
      public void run() {
        init();
        setVisible(true);
      }
    });
  }

  private boolean isPageValid() {
    ...
  }

  private String getInfoForException(String message, ElectionsException ex) {
...
  }

  private String getInfoForException(String message, RuntimeException ex) {
...
  }

}
  • linhas 73-82: o método [run], que será chamado pela classe de inicialização [BootElectionsSwing];
  • linhas 26-27: injeção de uma referência à vista n.º 2, que deverá ser apresentada se o utilizador for reconhecido;
  • linhas 30-31: inserção da sessão da aplicação;
  • linha 84: o método [isPageValid] faz duas coisas:
    • verifica se o identificador não está vazio (a palavra-passe pode estar);
    • instancia o campo «User» da linha 23 com os dados introduzidos;

Tarefa: complete o código da classe.



Tarefa: Teste a aplicação.