Files
sam-react-prod/src/components/settings/PermissionManagement/actions.ts
유병철 cbb38d48b9 refactor(WEB): 전체 actions.ts에 공통 API 유틸 적용
- buildApiUrl / executePaginatedAction 패턴으로 전환 (40+ actions 파일)
- 직접 URLSearchParams 조립 → buildApiUrl 유틸 사용
- 수동 페이지네이션 메타 변환 → executePaginatedAction 자동 처리
- HandoverReportDocumentModal, OrderDocumentModal 개선
- 급여관리 SalaryManagement 코드 개선
- CLAUDE.md Server Action 공통 유틸 규칙 정리

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-12 20:59:59 +09:00

137 lines
4.8 KiB
TypeScript

'use server';
import { executeServerAction, type ActionResult } from '@/lib/api/execute-server-action';
import { buildApiUrl } from '@/lib/api/query-params';
import { revalidatePath } from 'next/cache';
import type { Role, RoleStats, PermissionMatrix, MenuTreeItem, PaginatedResponse } from './types';
// ========== Role CRUD ==========
export async function fetchRoles(params?: {
page?: number; size?: number; q?: string; is_hidden?: boolean;
}): Promise<ActionResult<PaginatedResponse<Role>>> {
return executeServerAction<PaginatedResponse<Role>>({
url: buildApiUrl('/api/v1/roles', {
page: params?.page,
per_page: params?.size,
q: params?.q,
is_hidden: params?.is_hidden,
}),
errorMessage: '역할 목록 조회에 실패했습니다.',
});
}
export async function fetchRole(id: number): Promise<ActionResult<Role>> {
return executeServerAction<Role>({
url: buildApiUrl(`/api/v1/roles/${id}`),
errorMessage: '역할 조회에 실패했습니다.',
});
}
export async function createRole(data: {
name: string; description?: string; is_hidden?: boolean;
}): Promise<ActionResult<Role>> {
const result = await executeServerAction<Role>({
url: buildApiUrl('/api/v1/roles'),
method: 'POST',
body: data,
errorMessage: '역할 생성에 실패했습니다.',
});
if (result.success) revalidatePath('/settings/permissions');
return result;
}
export async function updateRole(id: number, data: {
name?: string; description?: string; is_hidden?: boolean;
}): Promise<ActionResult<Role>> {
const result = await executeServerAction<Role>({
url: buildApiUrl(`/api/v1/roles/${id}`),
method: 'PATCH',
body: data,
errorMessage: '역할 수정에 실패했습니다.',
});
if (result.success) {
revalidatePath('/settings/permissions');
revalidatePath(`/settings/permissions/${id}`);
}
return result;
}
export async function deleteRole(id: number): Promise<ActionResult> {
const result = await executeServerAction({
url: buildApiUrl(`/api/v1/roles/${id}`),
method: 'DELETE',
errorMessage: '역할 삭제에 실패했습니다.',
});
if (result.success) revalidatePath('/settings/permissions');
return result;
}
export async function fetchRoleStats(): Promise<ActionResult<RoleStats>> {
return executeServerAction<RoleStats>({
url: buildApiUrl('/api/v1/roles/stats'),
errorMessage: '역할 통계 조회에 실패했습니다.',
});
}
export async function fetchActiveRoles(): Promise<ActionResult<Role[]>> {
return executeServerAction<Role[]>({
url: buildApiUrl('/api/v1/roles/active'),
errorMessage: '활성 역할 목록 조회에 실패했습니다.',
});
}
// ========== Permission Matrix ==========
export async function fetchPermissionMenus(): Promise<ActionResult<{
menus: MenuTreeItem[]; permission_types: string[];
}>> {
return executeServerAction({
url: buildApiUrl('/api/v1/role-permissions/menus'),
errorMessage: '메뉴 트리 조회에 실패했습니다.',
});
}
export async function fetchPermissionMatrix(roleId: number): Promise<ActionResult<PermissionMatrix>> {
return executeServerAction<PermissionMatrix>({
url: buildApiUrl(`/api/v1/roles/${roleId}/permissions/matrix`),
errorMessage: '권한 매트릭스 조회에 실패했습니다.',
});
}
export async function togglePermission(roleId: number, menuId: number, permissionType: string): Promise<ActionResult<{
granted: boolean; propagated_to: number[];
}>> {
const result = await executeServerAction<{ granted: boolean; propagated_to: number[] }>({
url: buildApiUrl(`/api/v1/roles/${roleId}/permissions/toggle`),
method: 'POST',
body: { menu_id: menuId, permission_type: permissionType },
errorMessage: '권한 토글에 실패했습니다.',
});
if (result.success) revalidatePath(`/settings/permissions/${roleId}`);
return result;
}
async function rolePermissionAction(roleId: number, action: string, errorMessage: string): Promise<ActionResult<{ count: number }>> {
const result = await executeServerAction<{ count: number }>({
url: buildApiUrl(`/api/v1/roles/${roleId}/permissions/${action}`),
method: 'POST',
errorMessage,
});
if (result.success) revalidatePath(`/settings/permissions/${roleId}`);
return result;
}
export async function allowAllPermissions(roleId: number): Promise<ActionResult<{ count: number }>> {
return rolePermissionAction(roleId, 'allow-all', '전체 허용에 실패했습니다.');
}
export async function denyAllPermissions(roleId: number): Promise<ActionResult<{ count: number }>> {
return rolePermissionAction(roleId, 'deny-all', '전체 거부에 실패했습니다.');
}
export async function resetPermissions(roleId: number): Promise<ActionResult<{ count: number }>> {
return rolePermissionAction(roleId, 'reset', '권한 초기화에 실패했습니다.');
}