Guia Simplificado: CRUD de Usuários com Spring Boot e H2 (Local)
Etapa 1: Criando o Projeto Vamos começar do zero usando o Spring Initializr, a ferramenta oficial para criar projetos Spring.
Configurações do Projeto:
Project: Maven
Language: Java
Spring Boot: 3.2.x ou superior
Project Metadata:
Group: br.com.cadastro
Artifact: usuarios
Package name: br.com.cadastro.usuarios
Packaging: Jar
Java: 21
Dependências Essenciais:
Spring Web: Para criar a aplicação web.
Spring Data JPA: Para salvar e buscar dados no banco.
Thymeleaf: Para construir nossas páginas HTML.
H2 Database: Nosso banco de dados em memória, que funciona localmente.
Spring Security: Para adicionar uma tela de login e proteger a aplicação.
Spring Boot DevTools: Para que o projeto reinicie automaticamente quando você alterar o código.
Clique em “GENERATE”, baixe o projeto, descompacte e abra na sua IDE.
Etapa 2: Estrutura e Modelo de Dados Uma boa organização ajuda muito. Dentro de src/main/java/br/com/cadastro/usuarios, vamos criar a seguinte estrutura de pacotes:
br.com.cadastro.usuarios ├── config/ # Configurações de segurança ├── controller/ # Controladores que lidam com as requisições web ├── entity/ # Nossas tabelas do banco de dados (Entidades) ├── repository/ # Para acesso aos dados └── service/ # Regras de negócio (neste caso, o serviço de autenticação)
2.1. Criando as Entidades (entity) Vamos definir as tabelas User (usuário) e Role (perfil, ex: ADMIN).
Arquivo: src/main/java/br/com/cadastro/usuarios/entity/Role.java
package br.com.cadastro.usuarios.entity;
import jakarta.persistence.*; import org.springframework.security.core.GrantedAuthority;
@Entity @Table(name = “roles”) public class Role implements GrantedAuthority { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id; private String name;
@Override
public String getAuthority() {
return name;
}
// Getters e Setters
public Long getId() { return id; }
public void setId(Long id) { this.id = id; }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
}
Arquivo: src/main/java/br/com/cadastro/usuarios/entity/User.java
package br.com.cadastro.usuarios.entity;
import jakarta.persistence.*; import org.springframework.security.core.GrantedAuthority; import org.springframework.security.core.userdetails.UserDetails;
import java.util.Collection; import java.util.Set;
@Entity @Table(name = “users”) public class User implements UserDetails { // UserDetails é do Spring Security @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id;
@Column(nullable = false, unique = true)
private String username;
@Column(nullable = false)
private String password;
@Column(nullable = false, unique = true)
private String email;
@ManyToMany(fetch = FetchType.EAGER)
@JoinTable(name = "user_roles",
joinColumns = @JoinColumn(name = "user_id"),
inverseJoinColumns = @JoinColumn(name = "role_id"))
private Set<Role> roles;
// Métodos exigidos pelo UserDetails
@Override public Collection<? extends GrantedAuthority> getAuthorities() { return this.roles; }
@Override public boolean isAccountNonExpired() { return true; }
@Override public boolean isAccountNonLocked() { return true; }
@Override public boolean isCredentialsNonExpired() { return true; }
@Override public boolean isEnabled() { return true; }
// Getters e Setters
public Long getId() { return id; }
public void setId(Long id) { this.id = id; }
public String getUsername() { return username; }
public void setUsername(String username) { this.username = username; }
public String getPassword() { return password; }
public void setPassword(String password) { this.password = password; }
public String getEmail() { return email; }
public void setEmail(String email) { this.email = email; }
public Set<Role> getRoles() { return roles; }
public void setRoles(Set<Role> roles) { this.roles = roles; }
}
2.2. Criando o Repositório (repository) Esta interface nos dará os métodos para salvar, buscar, deletar, etc., sem precisarmos escrever SQL.
Arquivo: src/main/java/br/com/cadastro/usuarios/repository/UserRepository.java
package br.com.cadastro.usuarios.repository;
import br.com.cadastro.usuarios.entity.User; import org.springframework.data.jpa.repository.JpaRepository; import java.util.Optional;
public interface UserRepository extends JpaRepository<User, Long> {
Optional
Etapa 3: Configurando o Banco de Dados H2 Simplificaremos o arquivo application.properties para usar apenas o H2.
Arquivo: src/main/resources/application.properties
URL de conexão para o banco H2 em um arquivo
spring.datasource.url=jdbcfile:./data/usuariosdb
Driver do H2
spring.datasource.driverClassName=org.h2.Driver
Usuário e senha do banco
spring.datasource.username=sa spring.datasource.password=password
Dialeto do Hibernate para H2
spring.jpa.database-platform=org.hibernate.dialect.H2Dialect
Habilita o console do H2 para visualizarmos o banco no navegador
spring.h2.console.enabled=true
Caminho para acessar o console (ex: http://localhost:8080/h2-console)
spring.h2.console.path=/h2-console
Gerencia a criação das tabelas automaticamente
spring.jpa.hibernate.ddl-auto=update
Etapa 4: Implementando a Segurança (Login Simples) Vamos configurar uma tela de login tradicional.
4.1. Configuração de Segurança Arquivo: src/main/java/br/com/cadastro/usuarios/config/SecurityConfig.java
package br.com.cadastro.usuarios.config;
import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.security.config.annotation.web.builders.HttpSecurity; import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity; import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.security.web.SecurityFilterChain;
@Configuration @EnableWebSecurity public class SecurityConfig {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception {
http
.authorizeHttpRequests(authorize -> authorize
// Permite acesso livre aos arquivos de CSS e JS
.requestMatchers("/css/**", "/js/**").permitAll()
// Qualquer outra requisição precisa de autenticação
.anyRequest().authenticated()
)
// Configura o formulário de login
.formLogin(form -> form
.loginPage("/login").permitAll() // Diz qual a URL da nossa página de login
.defaultSuccessUrl("/", true) // Para onde ir após o login com sucesso
)
// Configura o logout
.logout(logout -> logout.logoutSuccessUrl("/login"));
return http.build();
}
@Bean
public PasswordEncoder passwordEncoder() {
// Codificador de senhas. Essencial para a segurança.
return new BCryptPasswordEncoder();
}
}
4.2. Serviço de Autenticação Este serviço é responsável por buscar o usuário no banco de dados para o Spring Security.
Arquivo: src/main/java/br/com/cadastro/usuarios/service/AuthorizationService.java
package br.com.cadastro.usuarios.service;
import br.com.cadastro.usuarios.repository.UserRepository; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.security.core.userdetails.UserDetails; import org.springframework.security.core.userdetails.UserDetailsService; import org.springframework.security.core.userdetails.UsernameNotFoundException; import org.springframework.stereotype.Service;
@Service public class AuthorizationService implements UserDetailsService {
@Autowired
private UserRepository userRepository;
@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
return userRepository.findByUsername(username)
.orElseThrow(() -> new UsernameNotFoundException("Usuário não encontrado."));
}
}
Etapa 5: Criando os Controllers e as Páginas Agora vamos criar as telas e os controllers que as servem.
5.1. Controller para Login e Páginas Principais Arquivo: src/main/java/br/com/cadastro/usuarios/controller/UserController.java
package br.com.cadastro.usuarios.controller;
import br.com.cadastro.usuarios.entity.User; import br.com.cadastro.usuarios.repository.UserRepository; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestParam;
import java.util.Optional;
@Controller public class UserController {
@Autowired
private UserRepository userRepository;
@Autowired
private PasswordEncoder passwordEncoder;
@GetMapping("/")
public String index(Model model) {
model.addAttribute("users", userRepository.findAll());
model.addAttribute("userForm", new User()); // Objeto para o formulário
return "index";
}
@PostMapping("/save")
public String saveUser(User user) {
// Codifica a senha antes de salvar
user.setPassword(passwordEncoder.encode(user.getPassword()));
userRepository.save(user);
return "redirect:/"; // Redireciona para a página principal
}
@GetMapping("/edit/{id}")
public String editUser(@PathVariable("id") Long id, Model model) {
Optional<User> userOpt = userRepository.findById(id);
if (userOpt.isEmpty()) {
return "redirect:/"; // Se não achar, volta para o início
}
model.addAttribute("users", userRepository.findAll());
model.addAttribute("userForm", userOpt.get());
return "index"; // Reutiliza a mesma página, mas com o formulário preenchido
}
@GetMapping("/delete/{id}")
public String deleteUser(@PathVariable("id") Long id) {
userRepository.deleteById(id);
return "redirect:/";
}
@GetMapping("/login")
public String login() {
return "login";
}
}
5.2. Páginas HTML com Thymeleaf Crie a pasta templates dentro de src/main/resources.
Página de Login: src/main/resources/templates/login.html
Login
Página Principal: src/main/resources/templates/index.html
Novo Usuário
Editar Usuário
<!-- Tabela de Usuários -->
<div class="col-md-8">
<h4>Usuários Cadastrados</h4>
<table class="table table-hover">
<thead>
<tr>
<th>ID</th><th>Username</th><th>Email</th><th>Ações</th>
</tr>
</thead>
<tbody>
<tr th:each="user : ${users}">
<td th:text="${user.id}"></td>
<td th:text="${user.username}"></td>
<td th:text="${user.email}"></td>
<td>
<a th:href="@{/edit/{id}(id=${user.id})}" class="btn btn-sm btn-warning"><i class="bi bi-pencil-fill"></i></a>
<a th:href="@{/delete/{id}(id=${user.id})}" class="btn btn-sm btn-danger"><i class="bi bi-trash-fill"></i></a>
</td>
</tr>
</tbody>
</table>
</div>
</div>
</main>
Etapa 6: Criando um Usuário Administrador Inicial Para conseguirmos fazer login pela primeira vez, vamos criar um bean que insere um usuário “admin” quando a aplicação inicia.
Crie este arquivo: src/main/java/br/com/cadastro/usuarios/config/DataInitializer.java
package br.com.cadastro.usuarios.config;
import br.com.cadastro.usuarios.entity.User; import br.com.cadastro.usuarios.repository.UserRepository; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.CommandLineRunner; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.security.crypto.password.PasswordEncoder;
@Configuration public class DataInitializer {
@Autowired
private UserRepository userRepository;
@Autowired
private PasswordEncoder passwordEncoder;
@Bean
public CommandLineRunner initData() {
return args -> {
// Cria o usuário admin apenas se ele não existir
if (userRepository.findByUsername("admin").isEmpty()) {
User admin = new User();
admin.setUsername("admin");
admin.setPassword(passwordEncoder.encode("admin"));
admin.setEmail("admin@email.com");
// Aqui você poderia criar e associar roles (perfis)
userRepository.save(admin);
System.out.println(">>> Usuário 'admin' criado com senha 'admin'");
}
};
}
}
Etapa 7: Rodando o Projeto! Agora é a parte fácil!
Execute a Classe Principal: Encontre a classe UsuariosApplication.java e clique com o botão direito → “Run” (ou use o atalho da sua IDE).
Acesse a Aplicação: Abra seu navegador e vá para http://localhost:8080.
Você será redirecionado para a tela de login.
Use o usuário admin e senha admin para entrar.
Acesse o Console do Banco: Para ver as tabelas e dados que estão sendo criados, acesse http://localhost:8080/h2-console.
No campo JDBC URL, coloque jdbcfile:./data/usuariosdb.
Use o usuário sa e a senha password.
Clique em “Connect”.