백엔드/Java
메뉴 권한 관리 구조 쿼리 (MariaDB - WITH RECURSIVE)
david100gom
2024. 10. 26. 14:10
MyBatis와 MySQL을 사용하여 메뉴 기반 권한 관리 시스템의 구조를 설정하고 쿼리를 작성해 보겠습니다. 역할(Role)과 메뉴(Menu)를 연결하여 특정 역할에 따라 접근할 수 있는 메뉴를 관리하는 구조입니다.
1. 데이터베이스 테이블 구조 예시
다음은 메뉴(Menu), 역할(Role), 권한(Permission)을 관리하기 위한 기본 테이블 구조입니다.
1.1. 메뉴 테이블 (Menus)
CREATE TABLE Menus (
menu_id INT PRIMARY KEY AUTO_INCREMENT,
menu_name VARCHAR(100) NOT NULL,
parent_menu_id INT,
url VARCHAR(255),
order_index INT,
FOREIGN KEY (parent_menu_id) REFERENCES Menus(menu_id) ON DELETE SET NULL
);
1.2. 역할 테이블 (Roles)
CREATE TABLE Roles (
role_id INT PRIMARY KEY AUTO_INCREMENT,
role_name VARCHAR(50) NOT NULL UNIQUE
);
1.3. 메뉴 권한 테이블 (Role_Menus)
Role_Menus 테이블은 특정 역할이 어떤 메뉴에 접근할 수 있는지를 관리합니다.
CREATE TABLE Role_Menus (
role_menu_id INT PRIMARY KEY AUTO_INCREMENT,
role_id INT,
menu_id INT,
FOREIGN KEY (role_id) REFERENCES Roles(role_id) ON DELETE CASCADE,
FOREIGN KEY (menu_id) REFERENCES Menus(menu_id) ON DELETE CASCADE
);
2. 재귀 쿼리 작성 (MyBatis Mapper)
MyBatis에서 역할에 따른 메뉴 트리 구조를 조회하는 쿼리를 작성합니다. 예를 들어, 특정 역할 ID를 입력 받아 해당 역할의 메뉴 트리 구조를 반환하도록 하겠습니다.
MenuMapper.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.example.mapper.MenuMapper">
<!-- 특정 역할(role_id)에 따른 메뉴 트리 구조 쿼리 -->
<select id="getMenuTreeByRole" parameterType="int" resultType="com.example.model.Menu">
WITH RECURSIVE MenuTree AS (
-- 최상위 메뉴 선택
SELECT
m.menu_id,
m.menu_name,
m.parent_menu_id,
m.url,
m.order_index,
CAST(LPAD(m.order_index, 2, '0') as CHAR(256)) AS full_idx,
CAST(m.menu_name AS CHAR(1000)) AS path,
1 AS depth
FROM Menus m
JOIN Role_Menus rm ON m.menu_id = rm.menu_id
WHERE rm.role_id = #{roleId} AND m.parent_menu_id = '0'
UNION ALL
-- 하위 메뉴 재귀적으로 조회
SELECT
m.menu_id,
m.menu_name,
m.parent_menu_id,
m.url,
m.order_index,
CONCAT(mt.full_idx, '-', LPAD(m.order_index, 2, '0')) AS full_idx,
CONCAT(mt.path, ' > ', m.menu_name) AS path,
mt.depth + 1 AS depth
FROM Menus m
JOIN MenuTree mt ON m.parent_menu_id = mt.menu_id
JOIN Role_Menus rm ON m.menu_id = rm.menu_id
WHERE rm.role_id = #{roleId}
)
-- 결과 정렬
SELECT menu_id,
menu_name,
parent_menu_id,
url,
full_idx,
order_idx,
path,
depth
FROM MenuTree
ORDER BY full_idx;
</select>
</mapper>
3. 매핑할 자바 모델 클래스 (예: Menu.java)
이 쿼리 결과를 받을 Menu 클래스에는 menu_id, menu_name, parent_menu_id, url, order_index, path, depth 필드를 포함합니다.
package com.example.model;
public class Menu {
private int menuId;
private String menuName;
private Integer parentMenuId;
private String url;
private int orderIndex;
private String path;
private int depth;
// Getters and Setters
// ...
}
4. 매퍼 인터페이스 (예: MenuMapper.java)
MyBatis에서 쿼리를 호출하기 위한 매퍼 인터페이스를 정의합니다.
package com.example.mapper;
import com.example.model.Menu;
import java.util.List;
public interface MenuMapper {
List<Menu> getMenuTreeByRole(int roleId);
}
5. 서비스에서 역할 기반 메뉴 트리 호출하기
서비스 계층에서 MenuMapper의 getMenuTreeByRole() 메서드를 호출하여 특정 역할에 따른 메뉴 목록을 가져올 수 있습니다.
package com.example.service;
import com.example.mapper.MenuMapper;
import com.example.model.Menu;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;
@Service
public class MenuService {
@Autowired
private MenuMapper menuMapper;
public List<Menu> getMenuTreeByRole(int roleId) {
return menuMapper.getMenuTreeByRole(roleId);
}
}
6. 결과 예시
getMenuTreeByRole(roleId) 메서드를 호출하면 roleId에 따라 접근 가능한 메뉴 트리 구조를 반환합니다. 예를 들어, roleId = 1로 조회했을 때 다음과 같은 결과가 나올 수 있습니다.
menu_id | menu_name | parent_menu_id | full_idx | order_index | path | depth |
1 | Dashboard | NULL | 01 | 1 | Dashboard | 1 |
2 | Reports | NULL | 02 | 2 | Reports | 1 |
3 | Sales | 2 | 02_01 | 1 | Reports > Sales | 2 |
4 | Marketing | 2 | 02_02 | 2 | Reports > Marketing | 2 |
5 | SEO | 4 | 02_02_01 | 1 | Reports > Marketing > SEO | 3 |
이와 같이 특정 역할에 따라 접근할 수 있는 메뉴의 트리 구조를 가져올 수 있습니다.