feat: Phase 4-1 테넌트 관리 백엔드 구현
- TenantService 생성 (CRUD, 통계, 복원/영구삭제) - API Controller 구현 (HTMX 요청 감지, HTML/JSON 이중 응답) - FormRequest 검증 (StoreTenantRequest, UpdateTenantRequest) - Tenant 모델 확장 (17개 필드, 관계 설정, accessor) - Department, Menu, Role 모델 복사 (admin → mng) - Web Controller 수정 (index/create/edit 화면) - MIGRATION_PLAN.md 작성 (HTMX + API 아키텍처)
This commit is contained in:
206
app/Http/Controllers/Api/Admin/TenantController.php
Normal file
206
app/Http/Controllers/Api/Admin/TenantController.php
Normal file
@@ -0,0 +1,206 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers\Api\Admin;
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Http\Requests\StoreTenantRequest;
|
||||
use App\Http\Requests\UpdateTenantRequest;
|
||||
use App\Services\TenantService;
|
||||
use Illuminate\Http\JsonResponse;
|
||||
use Illuminate\Http\Request;
|
||||
|
||||
class TenantController extends Controller
|
||||
{
|
||||
public function __construct(
|
||||
private readonly TenantService $tenantService
|
||||
) {}
|
||||
|
||||
/**
|
||||
* 테넌트 목록 조회
|
||||
*/
|
||||
public function index(Request $request): JsonResponse
|
||||
{
|
||||
$tenants = $this->tenantService->getTenants(
|
||||
$request->all(),
|
||||
$request->integer('per_page', 15)
|
||||
);
|
||||
|
||||
// HTMX 요청 시 HTML 반환
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'html' => view('tenants.partials.table', compact('tenants'))->render(),
|
||||
]);
|
||||
}
|
||||
|
||||
// 일반 요청 시 JSON 반환
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'data' => $tenants->items(),
|
||||
'meta' => [
|
||||
'current_page' => $tenants->currentPage(),
|
||||
'last_page' => $tenants->lastPage(),
|
||||
'per_page' => $tenants->perPage(),
|
||||
'total' => $tenants->total(),
|
||||
],
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 생성
|
||||
*/
|
||||
public function store(StoreTenantRequest $request): JsonResponse
|
||||
{
|
||||
$tenant = $this->tenantService->createTenant($request->validated());
|
||||
|
||||
// HTMX 요청 시 성공 메시지와 리다이렉트 헤더 반환
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 생성되었습니다.',
|
||||
'redirect' => route('tenants.index'),
|
||||
]);
|
||||
}
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 생성되었습니다.',
|
||||
'data' => $tenant,
|
||||
], 201);
|
||||
}
|
||||
|
||||
/**
|
||||
* 특정 테넌트 조회
|
||||
*/
|
||||
public function show(Request $request, int $id): JsonResponse
|
||||
{
|
||||
$tenant = $this->tenantService->getTenantById($id, true);
|
||||
|
||||
if (!$tenant) {
|
||||
return response()->json([
|
||||
'success' => false,
|
||||
'message' => '테넌트를 찾을 수 없습니다.',
|
||||
], 404);
|
||||
}
|
||||
|
||||
// HTMX 요청 시 HTML 반환
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'html' => view('tenants.partials.detail', compact('tenant'))->render(),
|
||||
]);
|
||||
}
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'data' => $tenant,
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 수정
|
||||
*/
|
||||
public function update(UpdateTenantRequest $request, int $id): JsonResponse
|
||||
{
|
||||
$this->tenantService->updateTenant($id, $request->validated());
|
||||
|
||||
// HTMX 요청 시 성공 메시지와 리다이렉트 헤더 반환
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 수정되었습니다.',
|
||||
'redirect' => route('tenants.index'),
|
||||
]);
|
||||
}
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 수정되었습니다.',
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 삭제 (Soft Delete)
|
||||
*/
|
||||
public function destroy(Request $request, int $id): JsonResponse
|
||||
{
|
||||
$this->tenantService->deleteTenant($id);
|
||||
|
||||
// HTMX 요청 시 테이블 행 제거 트리거
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 삭제되었습니다.',
|
||||
'action' => 'remove',
|
||||
]);
|
||||
}
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 삭제되었습니다.',
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 복원
|
||||
*/
|
||||
public function restore(Request $request, int $id): JsonResponse
|
||||
{
|
||||
$this->tenantService->restoreTenant($id);
|
||||
|
||||
// HTMX 요청 시 테이블 새로고침 트리거
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 복원되었습니다.',
|
||||
'action' => 'refresh',
|
||||
]);
|
||||
}
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 복원되었습니다.',
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 영구 삭제 (슈퍼관리자 전용)
|
||||
*/
|
||||
public function forceDestroy(Request $request, int $id): JsonResponse
|
||||
{
|
||||
// 슈퍼관리자 권한 체크
|
||||
if (!auth()->user()?->is_super_admin) {
|
||||
return response()->json([
|
||||
'success' => false,
|
||||
'message' => '권한이 없습니다.',
|
||||
], 403);
|
||||
}
|
||||
|
||||
$this->tenantService->forceDeleteTenant($id);
|
||||
|
||||
// HTMX 요청 시 테이블 행 제거 트리거
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 영구 삭제되었습니다.',
|
||||
'action' => 'remove',
|
||||
]);
|
||||
}
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => '테넌트가 영구 삭제되었습니다.',
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 통계 조회
|
||||
*/
|
||||
public function stats(Request $request): JsonResponse
|
||||
{
|
||||
$stats = $this->tenantService->getTenantStats();
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'data' => $stats,
|
||||
]);
|
||||
}
|
||||
}
|
||||
@@ -2,12 +2,48 @@
|
||||
|
||||
namespace App\Http\Controllers;
|
||||
|
||||
use App\Services\TenantService;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\View\View;
|
||||
|
||||
class TenantController extends Controller
|
||||
{
|
||||
public function __construct(
|
||||
private readonly TenantService $tenantService
|
||||
) {}
|
||||
|
||||
/**
|
||||
* 테넌트 전환
|
||||
* 테넌트 목록 (Blade 화면)
|
||||
*/
|
||||
public function index(Request $request): View
|
||||
{
|
||||
return view('tenants.index');
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 생성 화면
|
||||
*/
|
||||
public function create(): View
|
||||
{
|
||||
return view('tenants.create');
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 수정 화면
|
||||
*/
|
||||
public function edit(int $id): View
|
||||
{
|
||||
$tenant = $this->tenantService->getTenantById($id);
|
||||
|
||||
if (!$tenant) {
|
||||
abort(404, '테넌트를 찾을 수 없습니다.');
|
||||
}
|
||||
|
||||
return view('tenants.edit', compact('tenant'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 테넌트 전환 (기존 기능 유지)
|
||||
*/
|
||||
public function switch(Request $request)
|
||||
{
|
||||
|
||||
Reference in New Issue
Block a user