ํ์ ๊ฐ์
Code
Preview
1.
๋ฉ์ธ ํ๋ฉด
2.
ํ์ ๊ฐ์
์์ ํ๋ก์ธ์ค
1.
ํ๋ก์ ํธ ์์ฑ
2.
ํ๋ก์ ํธ ์ค์
3.
ERD
4.
ํ
์ด๋ธ ์ ์
5.
๋๋ฉ์ธ
6.
๋ฐ์ดํฐ
7.
์๋น์ค
8.
์คํ๋ง ์ํ๋ฆฌํฐ ์ค์
9.
์์ฒญ ๊ฒฝ๋ก ๋งคํ
Preview
โข
๋ฉ์ธ ํ๋ฉด
โข
ํ์ ๊ฐ์
๋ฉ์ธ ํ๋ฉด
ํ์ ๊ฐ์
์์ ํ๋ก์ธ์ค
1.
ํ๋ก์ ํธ ์์ฑ
โข
โข
spring boot version : 3.x.x
โข
spring security version : 6.x.x
โข
์์กด์ฑ ์ค์
โฆ
Spring Web
โฆ
Spring Boot DevTools
โฆ
Spring Security
โฆ
Lombok
โฆ
Thymeleaf
โฆ
MySQL Driver
โฆ
Mybatis Framework
2.
ํ๋ก์ ํธ ์ค์
โข
โฆ
๋ฐ์ดํฐ ์์ค ์ค์
โฆ
MyBatis ์ค์
4.
ํ
์ด๋ธ ์ ์
โข
USER
โข
USER_AUTH
5.
๋๋ฉ์ธ
โข
Users.java
โข
UserAuth.java
6.
๋ฐ์ดํฐ
โข
UserMapper.xml
โข
UserMapper.java
7.
์๋น์ค
โข
UserService.java
โข
UserServiceImpl.java
8.
์คํ๋ง ์ํ๋ฆฌํฐ ์ค์
โข
โข
9.
์์ฒญ ๊ฒฝ๋ก ๋งคํ
โข
โฆ
๋ฉ์ธ ํ๋ฉด
โช
GET
โช
โช
โฆ
ํ์ ๊ฐ์
ํ๋ฉด
โช
GET
โช
โช
โฆ
ํ์ ๊ฐ์
์ฒ๋ฆฌ
โช
POST
โช
โช
โช
โฆ
์์ด๋ ์ค๋ณต ํ์ธ
โช
POST
โช
โช
โช
ํ๋ก์ ํธ ์์ฑ
build.gradle
plugins {
id 'java'
id 'war'
id 'org.springframework.boot' version '3.5.9'
id 'io.spring.dependency-management' version '1.1.7'
}
group = 'com.aloha'
version = '0.0.1-SNAPSHOT'
description = 'Demo project for Spring Boot'
java {
toolchain {
languageVersion = JavaLanguageVersion.of(23)
}
}
configurations {
compileOnly {
extendsFrom annotationProcessor
}
}
repositories {
mavenCentral()
}
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-security'
implementation 'org.springframework.boot:spring-boot-starter-thymeleaf'
implementation 'org.springframework.boot:spring-boot-starter-web'
implementation 'org.mybatis.spring.boot:mybatis-spring-boot-starter:3.0.5'
implementation 'org.thymeleaf.extras:thymeleaf-extras-springsecurity6'
compileOnly 'org.projectlombok:lombok'
developmentOnly 'org.springframework.boot:spring-boot-devtools'
runtimeOnly 'com.mysql:mysql-connector-j'
annotationProcessor 'org.projectlombok:lombok'
providedRuntime 'org.springframework.boot:spring-boot-starter-tomcat'
testImplementation 'org.springframework.boot:spring-boot-starter-test'
testImplementation 'org.mybatis.spring.boot:mybatis-spring-boot-starter-test:3.0.5'
testImplementation 'org.springframework.security:spring-security-test'
testRuntimeOnly 'org.junit.platform:junit-platform-launcher'
}
tasks.named('test') {
useJUnitPlatform()
}
Java
๋ณต์ฌ
Spring Boot Version
Language
Group Id
Artifact Id
packaging type
Java version
์์กด์ฑ ์ค์
1.
Spring Web
2.
Spring boot devtools
3.
Lombok
4.
Thymeleaf
5.
Spring Security
6.
MySQL Driver
7.
Mybatis Framework
Spring Web
Spring boot devtools
Lombok
Thymeleaf
Spring Security
MySQL Driver
Mybatis Framework
ํ๋ก์ ํธ ์ค์
application.properties
โข
๋ฐ์ดํฐ ์์ค ์ค์
โข
MyBatis ์ค์
spring.application.name=security6
# ๋ฐ์ดํฐ ์์ค - MySQL
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.url=jdbc:mysql://127.0.0.1:3306/aloha?serverTimezone=Asia/Seoul&allowPublicKeyRetrieval=true&useSSL=false&autoReconnection=true&autoReconnection=true
spring.datasource.username=aloha
spring.datasource.password=123456
# Mybatis ์ค์
mybatis.configuration.map-underscore-to-camel-case=true
mybatis.type-aliases-package=com.aloha.security6.domain
mybatis.mapper-locations=classpath:mybatis/mapper/**/**.xml
Markdown
๋ณต์ฌ
ERD
ํ ์ด๋ธ ์ ์
โข
USER
โข
USER_AUTH
USER
-- user : ํ์
DROP TABLE IF EXISTS `user`;
CREATE TABLE `user` (
`no` BIGINT NOT NULL AUTO_INCREMENT PRIMARY KEY COMMENT 'ํ์๋ฒํธ',
`id` VARCHAR(64) NOT NULL UNIQUE COMMENT '์์ด๋',
`username` VARCHAR(100) NOT NULL COMMENT '์ฌ์ฉ์๋ช
',
`password` VARCHAR(200) NOT NULL COMMENT '๋น๋ฐ๋ฒํธ',
`name` VARCHAR(100) NOT NULL COMMENT '์ด๋ฆ',
`email` VARCHAR(200) DEFAULT NULL COMMENT '์ด๋ฉ์ผ',
`created_at` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '์์ฑ์ผ์',
`updated_at` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '์์ ์ผ์',
`enabled` INT DEFAULT 1 COMMENT 'ํ์ฑํ์ฌ๋ถ'
) COMMENT='ํ์';
SQL
๋ณต์ฌ
USER_AUTH
-- user_auth : ํ์๊ถํ
DROP TABLE IF EXISTS `user_auth`;
CREATE TABLE `user_auth` (
`no` BIGINT NOT NULL AUTO_INCREMENT PRIMARY KEY COMMENT '๊ถํ๋ฒํธ',
`id` VARCHAR(64) NOT NULL COMMENT '์ฌ์ฉ์ID (UK)',
`username` VARCHAR(100) NOT NULL COMMENT '์ฌ์ฉ์๋ช
',
`auth` VARCHAR(100) NOT NULL COMMENT '๊ถํ (ROLE_USER, ROLE_ADMIN, ...)',
`created_at` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '์์ฑ์ผ์',
`updated_at` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '์์ ์ผ์'
) COMMENT='ํ์๊ถํ';
SQL
๋ณต์ฌ
๋๋ฉ์ธ
โข
Users.java
โข
UserAuth.java
Users.java
@Data
@Builder
@AllArgsConstructor
public class Users {
private Long no;
@Builder.Default
private String id = UUID.randomUUID().toString();
private String username;
private String password;
private String name;
private String email;
private Date createdAt;
private Date updatedAt;
private int enabled;
private List<UserAuth> authList;
public Users() {
this.id = UUID.randomUUID().toString();
}
}
Java
๋ณต์ฌ
UserAuth.java
@Data
@Builder
@AllArgsConstructor
public class UserAuth {
private Long no;
@Builder.Default
private String id = UUID.randomUUID().toString();
private String username;
private String auth;
private Date createdAt;
private Date updatedAt;
public UserAuth() {
this.id = UUID.randomUUID().toString();
}
}
Java
๋ณต์ฌ
๋ฐ์ดํฐ
โข
UserMapper.xml
โข
UserMapper.java
UserMapper.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.aloha.security.mapper.UserMapper">
<resultMap id="UserMap" type="Users">
<id property="no" column="no"/> <!-- PK -->
<result property="id" column="id"/>
<result property="username" column="username"/>
<result property="password" column="password"/>
<result property="name" column="name"/>
<result property="email" column="email"/>
<result property="enabled" column="enabled"/>
<result property="createdAt" column="created_at"/>
<result property="updatedAt" column="updated_at"/>
<!-- JOIN ์ฟผ๋ฆฌ๋ฅผ ์ง๊ณ AuthMap ์ฐ๊ฒฐํ๋ ๋ฐฉ์ -->
<!-- <collection property="authList" resultMap="AuthMap"></collection> -->
<!-- ๋ณ๋ SELECT ๋ก ์ฐ๊ฒฐ -->
<collection property="authList" ofType="UserAuth"
select="selectAuth"
column="username"></collection>
</resultMap>
<resultMap id="AuthMap" type="UserAuth">
<result property="no" column="no" />
<result property="username" column="username" />
<result property="auth" column="auth" />
</resultMap>
<!-- ํ์ ๊ฐ์
-->
<insert id="join">
INSERT INTO user ( id, username, password, name, email )
VALUES ( #{id}, #{username}, #{password}, #{name}, #{email} )
</insert>
<!-- ํ์ ๊ถํ ๋ฑ๋ก -->
<insert id="insertAuth">
INSERT INTO user_auth ( id, username, auth )
VALUES ( #{id}, #{username}, #{auth} )
</insert>
<!-- ํ์ ์กฐํ -->
<!-- <select id="select" resultMap="UserMap">
SELECT u.*
,auth
FROM user u
LEFT JOIN user_auth auth ON u.username = auth.username
WHERE u.username = #{username}
</select> -->
<select id="select" resultMap="UserMap">
SELECT *
FROM user
WHERE username = #{username}
</select>
<select id="selectAuth" resultType="UserAuth">
SELECT *
FROM user_auth
WHERE username = #{username}
</select>
</mapper>
XML
๋ณต์ฌ
UserMapper.java
@Mapper
public interface UserMapper {
// ํ์ ์กฐํ
public Users select(String id) throws Exception;
// ํ์ ๊ฐ์
public int join(Users user) throws Exception;
// ํ์ ์์
public int update(Users user) throws Exception;
// ํ์ ๊ถํ ๋ฑ๋ก
public int insertAuth(UserAuth userAuth) throws Exception;
}
Java
๋ณต์ฌ
์๋น์ค
โข
UserService.java
โข
UserServiceImpl.java
UserService.java
public interface UserService {
// ์กฐํ
public Users select(String username) throws Exception;
// ํ์ ๊ฐ์
public int join(Users user) throws Exception;
// ํ์ ์์
public int update(Users user) throws Exception;
// ํ์ ๊ถํ ๋ฑ๋ก
public int insertAuth(UserAuth userAuth) throws Exception;
}
Java
๋ณต์ฌ
UserServiceImpl.java
@Service
@RequiredArgsConstructor
public class UserServiceImpl implements UserService {
// @Autowired private UserMapper userMapper;
// @Autowired private PasswordEncoder passwordEncoder;
private final UserMapper userMapper;
private final PasswordEncoder passwordEncoder;
@Override
public Users select(String username) throws Exception {
Users user = userMapper.select(username);
return user;
}
@Override
@Transactional
public int join(Users user) throws Exception {
String username = user.getUsername();
String password = user.getPassword();
// 123456 -> ๐ F123N905123890N3138932N4 (์ํธํ)
String encodedPassword = passwordEncoder.encode(password); // ๐ ๋น๋ฐ๋ฒํธ ์ํธํ
user.setPassword(encodedPassword);
// ํ์ ๋ฑ๋ก
int result = userMapper.join(user);
if( result > 0 ) {
// ํ์ ๊ธฐ๋ณธ ๊ถํ ๋ฑ๋ก
UserAuth userAuth = new UserAuth();
userAuth.setUsername(username);
userAuth.setAuth("ROLE_USER");
result = userMapper.insertAuth(userAuth);
}
return result;
}
@Override
public int update(Users user) throws Exception {
// ๋น๋ฐ๋ฒํธ ๋ณ๊ฒฝํ๋ ๊ฒฝ์ฐ ์ํธํ ์ฒ๋ฆฌ
String password = user.getPassword();
if( password != null && !password.isEmpty() ) {
String encodedPassword = passwordEncoder.encode(password); // ๐ ๋น๋ฐ๋ฒํธ ์ํธํ
user.setPassword(encodedPassword);
}
int result = userMapper.update(user);
return result;
}
@Override
public int insertAuth(UserAuth userAuth) throws Exception {
int result = userMapper.insertAuth(userAuth);
return result;
}
}
Java
๋ณต์ฌ
์คํ๋ง ์ํ๋ฆฌํฐ ์ค์
โข
โข
~/config/SecurityConfig.java
@Configuration
@EnableWebSecurity
public class SecurityConfig {
// ์คํ๋ง ์ํ๋ฆฌํฐ ์ค์ ๋ฉ์๋
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception {
/// โ
์ธ๊ฐ ์ค์
http.authorizeHttpRequests(auth -> auth
.requestMatchers("/**").permitAll());
return http.build();
}
/**
* ๐ ์ํธํ ๋ฐฉ์ ๋น ๋ฑ๋ก
* @return
*/
@Bean
public PasswordEncoder passwordEncoder() {
return new BCryptPasswordEncoder();
}
}
Java
๋ณต์ฌ
์์ฒญ ๊ฒฝ๋ก ๋งคํ
โข
โฆ
๋ฉ์ธ ํ๋ฉด
โช
GET
โช
โช
โฆ
ํ์ ๊ฐ์
ํ๋ฉด
โช
GET
โช
โช
โฆ
ํ์ ๊ฐ์
์ฒ๋ฆฌ
โช
POST
โช
โช
โช
โฆ
์์ด๋ ์ค๋ณต ํ์ธ
โช
POST
โช
โช
โช
~/controller/HomeController.java
@Slf4j
@Controller
public class HomeController {
@Autowired
private UserService userService;
/**
* ๋ฉ์ธ ํ๋ฉด
* ๐ [GET] - /
* ๐ index.html
* @return
*/
@GetMapping("")
public String home() {
log.info(":::::::::: ๋ฉ์ธ ํ๋ฉด ::::::::::");
return "index";
}
/**
* ํ์ ๊ฐ์
ํ๋ฉด
* ๐ [GET] - /join
* ๐ join.html
* @return
*/
@GetMapping("/join")
public String join() {
log.info(":::::::::: ํ์ ๊ฐ์
ํ๋ฉด ::::::::::");
return "join";
}
/**
* ํ์ ๊ฐ์
์ฒ๋ฆฌ
* ๐ [POST] - /join
* โก โญ /login
* โ /join?error
* @param user
* @return
* @throws Exception
*/
@PostMapping("/join")
public String joinPro(Users user) throws Exception {
log.info(":::::::::: ํ์ ๊ฐ์
์ฒ๋ฆฌ ::::::::::");
log.info("user : " + user);
int result = userService.join(user);
if( result > 0 ) {
return "redirect:/login";
}
return "redirect:/join?error";
}
/**
* ์์ด๋ ์ค๋ณต ๊ฒ์ฌ
* @param username
* @return
* @throws Exception
*/
@ResponseBody
@GetMapping("/check/{username}")
public ResponseEntity<Boolean> userCheck(@PathVariable("username") String username) throws Exception {
log.info("์์ด๋ ์ค๋ณต ํ์ธ : " + username);
Users user = userService.select(username);
// ์์ด๋ ์ค๋ณต
if( user != null ) {
log.info("์ค๋ณต๋ ์์ด๋ ์
๋๋ค - " + username);
return new ResponseEntity<>(false, HttpStatus.OK);
}
// ์ฌ์ฉ ๊ฐ๋ฅํ ์์ด๋์
๋๋ค.
log.info("์ฌ์ฉ ๊ฐ๋ฅํ ์์ด๋ ์
๋๋ค." + username);
return new ResponseEntity<>(true, HttpStatus.OK);
}
}
Java
๋ณต์ฌ
index.html
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org"
xmlns:sec="http://www.thymeleaf.org/thymeleaf-extras-springsecurity6">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>OAuth</title>
<!-- bootstrap css -->
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.1/dist/css/bootstrap.min.css" rel="stylesheet">
</head>
<body>
<div class="container col-12 col-lg-4">
<div class="px-4 py-5 mt-5 text-center">
<h1 class="display-5 fw-bold text-body-emphasis">๋ฉ์ธ ํ๋ฉด</h1>
</div>
<!-- ๋น ๋ก๊ทธ์ธ ์ -->
<th:block sec:authorize="isAnonymous()">
<div class="d-grid gap-2">
<a href="/login" class="btn btn-lg btn-primary">๋ก๊ทธ์ธ</a>
<a href="/join" class="btn btn-lg btn-success">ํ์๊ฐ์
</a>
</div>
</th:block>
<!-- ๋ก๊ทธ์ธ ์ -->
<th:block sec:authorize="isAuthenticated()">
<div class="card">
<div class="inner p-4">
<div class="d-flex flex-column align-items-center">
<div class="item my-2">
<img th:src="${user.profile}" alt="ํ๋กํ" class="rounded-circle">
</div>
<div class="item my-2">
<h3 th:text="${user.name}"></h3>
</div>
<div class="item my-2">
<h5 th:text="${user.email}"></h5>
</div>
<div class="item my-2 w-100">
<span sec:authentication="principal">์ธ์ฆ๋ ์ฌ์ฉ์</span>
</div>
</div>
</div>
</div>
<form action="/logout" method="post">
<!-- CSRF TOKEN -->
<input type="hidden" th:name="${_csrf.parameterName}" th:value="${_csrf.token}">
<div class="d-grid gap-2">
<button type="submit" class="btn btn-lg btn-primary">๋ก๊ทธ์์</button>
</div>
</form>
</th:block>
</div>
<!-- bootstrap js -->
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.1/dist/js/bootstrap.bundle.min.js"></script>
</body>
</html>
HTML
๋ณต์ฌ
join.html
๋น๋๊ธฐ ์์ฒญ์ JavaScript ๋ก XMLHttpRequest, fetch, axios ๋ฑ์ ์ฌ์ฉํ ์ ์์ต๋๋ค.
๋น๋๊ธฐ ๊ด๋ จ ๋ด์ฉ์ ์๋ ํ์ด์ง๋ฅผ ์ฐธ์กฐํด๋ณด์ธ์~!
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org"
xmlns:sec="http://www.thymeleaf.org/thymeleaf-extras-springsecurity6">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>ํ์๊ฐ์
</title>
<!-- bootstrap css -->
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.1/dist/css/bootstrap.min.css" rel="stylesheet">
</head>
<body>
<div class="container col-12 col-lg-4">
<div class="px-4 py-5 mt-5 text-center">
<h1 class="display-5 fw-bold text-body-emphasis">ํ์ ๊ฐ์
</h1>
</div>
<!-- ํ์๊ฐ์
์์ญ -->
<main class="form-signin login-box w-100 m-auto">
<form id="form" action="/join" method="post" class="needs-validation" onsubmit="return checkSubmit(event)">
<!-- CSRF TOKEN -->
<input type="hidden" th:name="${_csrf.parameterName}" th:value="${_csrf.token}">
<div class="input-group my-2">
<div class="form-floating" id="box-id">
<input type="text" class="form-control" id="username" name="username" value="" placeholder="์์ด๋" autofocus>
<label for="username">์์ด๋</label>
</div>
<div class="input-group-append">
<button type="button" class="btn btn-lg btn-outline-secondary h-100" onclick="checkId()" >์ค๋ณตํ์ธ</button>
</div>
</div>
<div class="form-floating my-2">
<input type="password" class="form-control" id="password" name="password" placeholder="๋น๋ฐ๋ฒํธ">
<label for="password">๋น๋ฐ๋ฒํธ</label>
</div>
<div class="form-floating my-2">
<input type="password" class="form-control" id="passwordCheck" name="passwordCheck" placeholder="๋น๋ฐ๋ฒํธ ํ์ธ">
<label for="passwordCheck">๋น๋ฐ๋ฒํธ ํ์ธ</label>
</div>
<div class="form-floating my-2">
<input type="text" class="form-control" id="name" name="name" value="" placeholder="์ด๋ฆ"
autofocus="">
<label for="name">์ด๋ฆ</label>
</div>
<div class="form-floating my-2">
<input type="text" class="form-control" id="email" name="email" value="" placeholder="์ด๋ฉ์ผ"
autofocus="">
<label for="email">์ด๋ฉ์ผ</label>
</div>
<div class="d-grid gap-2">
<button class="btn btn-lg btn-primary w-100 py-2" type="submit">ํ์๊ฐ์
</button>
<a href="/" class="btn btn-lg btn-success w-100 py-2">๋ฉ์ธ</a>
</div>
</form>
</main>
</div>
<!-- bootstrap js -->
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.1/dist/js/bootstrap.bundle.min.js"></script>
<script>
// ๐ CRSF TOKEN
const csrfToken = "[[${_csrf.token}]]"
/*
์์ด๋ ์ค๋ณต ํ์ธ
*/
async function checkId() {
const username = document.getElementById("username").value;
// null ๋๋ undefined
if (!username) {
alert("์์ด๋๋ฅผ ์
๋ ฅํด์ฃผ์ธ์");
return;
}
try {
// ์์ด๋ ์ค๋ณต ํ์ธ
const response = await fetch(`/check/${username}`, {
method: 'GET',
headers: {
'X-CSRF-TOKEN': csrfToken
}
});
if (response.ok) {
const result = await response.text();
let boxId = document.getElementById('box-id');
if (result === 'true') {
alert('์ฌ์ฉ ๊ฐ๋ฅํ ์์ด๋์
๋๋ค.');
boxId.classList.remove('needs-validation');
boxId.classList.add('was-validated');
return true;
} else {
alert('์ค๋ณต๋ ์์ด๋์
๋๋ค.');
boxId.classList.remove('was-validated');
boxId.classList.add('needs-validation');
}
return false;
} else {
alert('์์ด๋ ์ค๋ณต ํ์ธ ์ค ์ค๋ฅ๊ฐ ๋ฐ์ํ์ต๋๋ค.');
return false;
}
} catch (error) {
console.error('Error:', error);
alert('์์ด๋ ์ค๋ณต ํ์ธ ์ค ์ค๋ฅ๊ฐ ๋ฐ์ํ์ต๋๋ค.');
return false;
}
}
/*
์ ์ถ ํ์ธ
- ์์ด๋ ์ค๋ณต ์ฒดํฌ
*/
async function checkSubmit(event) {
event.preventDefault(); // ํผ ์ ์ถ ๋ฐฉ์ง
// ์์ด๋ ์ค๋ณต ์ฒดํฌ
const isIdAvailable = await checkId();
if (!isIdAvailable) {
return;
}
document.getElementById("form").submit();
}
</script>
</body>
</html>
HTML
๋ณต์ฌ




























