feat: ItemMaster Phase 2 API 구현 (확장 기능 11개 엔드포인트)

- Controller 3개, Service 3개, FormRequest 6개 생성
- Routes 등록 (BOM 항목, 섹션 템플릿, 마스터 필드)
- Service-First, Multi-tenant, Soft Delete
- 라우트 테스트 및 Pint 검사 통과

13 files changed, 600+ insertions(+)
This commit is contained in:
2025-11-20 17:07:40 +09:00
parent 4ccee253b6
commit 28a943bf8e
14 changed files with 776 additions and 1 deletions

View File

@@ -1,3 +1,143 @@
## 2025-11-20 (수) - ItemMaster Phase 2 API 구현 (확장 기능)
### 주요 작업
- Phase 2 확장 기능 11개 API 엔드포인트 구현
- BOM 항목, 섹션 템플릿, 마스터 필드 관리 기능 추가
- SAM API Development Rules 준수 (Service-First, ApiResponse, i18n)
### 추가된 파일
#### Controllers (3개)
1. **app/Http/Controllers/Api/V1/ItemMaster/ItemBomItemController.php**
- store(), update(), destroy()
- BOM 항목 관리
2. **app/Http/Controllers/Api/V1/ItemMaster/SectionTemplateController.php**
- index(), store(), update(), destroy()
- 섹션 템플릿 관리
3. **app/Http/Controllers/Api/V1/ItemMaster/ItemMasterFieldController.php**
- index(), store(), update(), destroy()
- 마스터 필드 라이브러리 관리
#### Services (3개)
1. **app/Services/ItemMaster/ItemBomItemService.php**
- CRUD + Multi-tenant 스코프
- 섹션 존재 확인 후 BOM 항목 생성
2. **app/Services/ItemMaster/SectionTemplateService.php**
- CRUD + 목록 조회 (created_at 정렬)
- is_default 플래그 지원
3. **app/Services/ItemMaster/ItemMasterFieldService.php**
- CRUD + 목록 조회 (category, field_name 정렬)
- JSON 필드 지원 (options, validation_rules, properties)
- is_common 플래그 지원
#### FormRequests (6개)
1. **app/Http/Requests/ItemMaster/ItemBomItemStoreRequest.php**
- item_name (required), quantity, unit, unit_price, total_price, spec, note
2. **app/Http/Requests/ItemMaster/ItemBomItemUpdateRequest.php**
- 모든 필드 sometimes
3. **app/Http/Requests/ItemMaster/SectionTemplateStoreRequest.php**
- title (required), type (required, in:fields,bom), description, is_default
4. **app/Http/Requests/ItemMaster/SectionTemplateUpdateRequest.php**
- 모든 필드 sometimes
5. **app/Http/Requests/ItemMaster/ItemMasterFieldStoreRequest.php**
- field_name (required), field_type (required, enum)
- category, description, is_common, default_value, options, validation_rules, properties
6. **app/Http/Requests/ItemMaster/ItemMasterFieldUpdateRequest.php**
- 모든 필드 sometimes
### 수정된 파일
1. **routes/api.php**
- ItemMaster 관련 use 문 3개 추가
- 11개 엔드포인트 추가:
- POST/PUT/DELETE /sections/{sectionId}/bom-items
- GET/POST/PUT/DELETE /section-templates
- GET/POST/PUT/DELETE /master-fields
### 작업 내용
#### API 엔드포인트 (11개)
1. ✅ POST `/sections/{sectionId}/bom-items` - BOM 항목 생성
2. ✅ PUT `/bom-items/{id}` - BOM 항목 수정
3. ✅ DELETE `/bom-items/{id}` - BOM 항목 삭제
4. ✅ GET `/section-templates` - 섹션 템플릿 목록
5. ✅ POST `/section-templates` - 섹션 템플릿 생성
6. ✅ PUT `/section-templates/{id}` - 섹션 템플릿 수정
7. ✅ DELETE `/section-templates/{id}` - 섹션 템플릿 삭제
8. ✅ GET `/master-fields` - 마스터 필드 목록
9. ✅ POST `/master-fields` - 마스터 필드 생성
10. ✅ PUT `/master-fields/{id}` - 마스터 필드 수정
11. ✅ DELETE `/master-fields/{id}` - 마스터 필드 삭제
#### 기술적 특징
**Service-First 패턴**:
- Controller는 DI + ApiResponse::handle()만 사용
- 모든 비즈니스 로직은 Service에 구현
- Service extends Service (tenantId(), apiUserId() 활용)
**Multi-tenant 지원**:
- 모든 Service 메서드에서 tenantId() 검증
- BelongsToTenant 스코프 자동 적용
- Soft Delete시 tenant_id 검증
**실시간 저장**:
- 모든 CUD 작업 즉시 처리
- JSON 필드 자동 캐스팅 (options, validation_rules, properties)
**i18n 메시지**:
- __('message.fetched'), __('message.created')
- __('message.updated'), __('message.deleted')
- __('error.not_found')
### 검증 결과
**라우트 테스트**:
```bash
php artisan route:list --path=item-master
# 결과: 24개 엔드포인트 정상 등록 (Phase 1: 13개 + Phase 2: 11개)
```
**Pint 검사**:
```bash
./vendor/bin/pint --test [12개 파일]
# 결과: 12 files PASS
```
### 다음 단계 (Phase 3)
**Phase 3 (부가 기능)** - 예정:
- 커스텀 탭 (5개 엔드포인트)
- 단위 옵션 (3개 엔드포인트)
**Swagger 문서** - 필요:
- app/Swagger/v1/ItemMasterApi.php 작성
- 24개 엔드포인트 스키마 정의 (Phase 1 + Phase 2)
### Git 커밋
```bash
git commit [hash]
feat: ItemMaster Phase 2 API 구현 (확장 기능 11개 엔드포인트)
- Controller 3개, Service 3개, FormRequest 6개 생성
- Routes 등록 (BOM 항목, 섹션 템플릿, 마스터 필드)
- Service-First, Multi-tenant, Soft Delete
- 라우트 테스트 및 Pint 검사 통과
13 files changed, 600+ insertions(+)
```
---
## 2025-11-20 (수) - ItemMaster Phase 1 API 구현 (핵심 기능)
### 주요 작업
@@ -152,7 +292,15 @@ ### 다음 단계 (Phase 2-3)
### Git 커밋
```bash
# 커밋 예정
git commit 4ccee25
feat: ItemMaster Phase 1 API 구현 (핵심 기능 13개 엔드포인트)
- Controller 4개, Service 4개, FormRequest 7개 생성
- Routes 등록 (/v1/item-master/*)
- Service-First, Multi-tenant, Cascade Soft Delete
- 라우트 테스트 및 Pint 검사 통과
17 files changed, 991 insertions(+)
```
---

View File

@@ -0,0 +1,48 @@
<?php
namespace App\Http\Controllers\Api\V1\ItemMaster;
use App\Http\Controllers\Controller;
use App\Http\Requests\ItemMaster\ItemBomItemStoreRequest;
use App\Http\Requests\ItemMaster\ItemBomItemUpdateRequest;
use App\Http\Responses\ApiResponse;
use App\Services\ItemMaster\ItemBomItemService;
class ItemBomItemController extends Controller
{
public function __construct(
protected ItemBomItemService $service,
) {}
/**
* BOM 항목 생성
*/
public function store(int $sectionId, ItemBomItemStoreRequest $request)
{
return ApiResponse::handle(function () use ($sectionId, $request) {
return $this->service->store($sectionId, $request->validated());
}, __('message.created'));
}
/**
* BOM 항목 수정
*/
public function update(int $id, ItemBomItemUpdateRequest $request)
{
return ApiResponse::handle(function () use ($id, $request) {
return $this->service->update($id, $request->validated());
}, __('message.updated'));
}
/**
* BOM 항목 삭제
*/
public function destroy(int $id)
{
return ApiResponse::handle(function () use ($id) {
$this->service->destroy($id);
return 'success';
}, __('message.deleted'));
}
}

View File

@@ -0,0 +1,58 @@
<?php
namespace App\Http\Controllers\Api\V1\ItemMaster;
use App\Http\Controllers\Controller;
use App\Http\Requests\ItemMaster\ItemMasterFieldStoreRequest;
use App\Http\Requests\ItemMaster\ItemMasterFieldUpdateRequest;
use App\Http\Responses\ApiResponse;
use App\Services\ItemMaster\ItemMasterFieldService;
class ItemMasterFieldController extends Controller
{
public function __construct(
protected ItemMasterFieldService $service,
) {}
/**
* 마스터 필드 목록
*/
public function index()
{
return ApiResponse::handle(function () {
return $this->service->index();
}, __('message.fetched'));
}
/**
* 마스터 필드 생성
*/
public function store(ItemMasterFieldStoreRequest $request)
{
return ApiResponse::handle(function () use ($request) {
return $this->service->store($request->validated());
}, __('message.created'));
}
/**
* 마스터 필드 수정
*/
public function update(int $id, ItemMasterFieldUpdateRequest $request)
{
return ApiResponse::handle(function () use ($id, $request) {
return $this->service->update($id, $request->validated());
}, __('message.updated'));
}
/**
* 마스터 필드 삭제
*/
public function destroy(int $id)
{
return ApiResponse::handle(function () use ($id) {
$this->service->destroy($id);
return 'success';
}, __('message.deleted'));
}
}

View File

@@ -0,0 +1,58 @@
<?php
namespace App\Http\Controllers\Api\V1\ItemMaster;
use App\Http\Controllers\Controller;
use App\Http\Requests\ItemMaster\SectionTemplateStoreRequest;
use App\Http\Requests\ItemMaster\SectionTemplateUpdateRequest;
use App\Http\Responses\ApiResponse;
use App\Services\ItemMaster\SectionTemplateService;
class SectionTemplateController extends Controller
{
public function __construct(
protected SectionTemplateService $service,
) {}
/**
* 섹션 템플릿 목록
*/
public function index()
{
return ApiResponse::handle(function () {
return $this->service->index();
}, __('message.fetched'));
}
/**
* 섹션 템플릿 생성
*/
public function store(SectionTemplateStoreRequest $request)
{
return ApiResponse::handle(function () use ($request) {
return $this->service->store($request->validated());
}, __('message.created'));
}
/**
* 섹션 템플릿 수정
*/
public function update(int $id, SectionTemplateUpdateRequest $request)
{
return ApiResponse::handle(function () use ($id, $request) {
return $this->service->update($id, $request->validated());
}, __('message.updated'));
}
/**
* 섹션 템플릿 삭제
*/
public function destroy(int $id)
{
return ApiResponse::handle(function () use ($id) {
$this->service->destroy($id);
return 'success';
}, __('message.deleted'));
}
}

View File

@@ -0,0 +1,27 @@
<?php
namespace App\Http\Requests\ItemMaster;
use Illuminate\Foundation\Http\FormRequest;
class ItemBomItemStoreRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'item_code' => 'nullable|string|max:100',
'item_name' => 'required|string|max:255',
'quantity' => 'nullable|numeric|min:0',
'unit' => 'nullable|string|max:50',
'unit_price' => 'nullable|numeric|min:0',
'total_price' => 'nullable|numeric|min:0',
'spec' => 'nullable|string',
'note' => 'nullable|string',
];
}
}

View File

@@ -0,0 +1,27 @@
<?php
namespace App\Http\Requests\ItemMaster;
use Illuminate\Foundation\Http\FormRequest;
class ItemBomItemUpdateRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'item_code' => 'sometimes|nullable|string|max:100',
'item_name' => 'sometimes|string|max:255',
'quantity' => 'sometimes|nullable|numeric|min:0',
'unit' => 'sometimes|nullable|string|max:50',
'unit_price' => 'sometimes|nullable|numeric|min:0',
'total_price' => 'sometimes|nullable|numeric|min:0',
'spec' => 'sometimes|nullable|string',
'note' => 'sometimes|nullable|string',
];
}
}

View File

@@ -0,0 +1,28 @@
<?php
namespace App\Http\Requests\ItemMaster;
use Illuminate\Foundation\Http\FormRequest;
class ItemMasterFieldStoreRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'field_name' => 'required|string|max:255',
'field_type' => 'required|in:textbox,number,dropdown,checkbox,date,textarea',
'category' => 'nullable|string|max:100',
'description' => 'nullable|string',
'is_common' => 'nullable|boolean',
'default_value' => 'nullable|string',
'options' => 'nullable|array',
'validation_rules' => 'nullable|array',
'properties' => 'nullable|array',
];
}
}

View File

@@ -0,0 +1,28 @@
<?php
namespace App\Http\Requests\ItemMaster;
use Illuminate\Foundation\Http\FormRequest;
class ItemMasterFieldUpdateRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'field_name' => 'sometimes|string|max:255',
'field_type' => 'sometimes|in:textbox,number,dropdown,checkbox,date,textarea',
'category' => 'sometimes|nullable|string|max:100',
'description' => 'sometimes|nullable|string',
'is_common' => 'sometimes|nullable|boolean',
'default_value' => 'sometimes|nullable|string',
'options' => 'sometimes|nullable|array',
'validation_rules' => 'sometimes|nullable|array',
'properties' => 'sometimes|nullable|array',
];
}
}

View File

@@ -0,0 +1,23 @@
<?php
namespace App\Http\Requests\ItemMaster;
use Illuminate\Foundation\Http\FormRequest;
class SectionTemplateStoreRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'title' => 'required|string|max:255',
'type' => 'required|in:fields,bom',
'description' => 'nullable|string',
'is_default' => 'nullable|boolean',
];
}
}

View File

@@ -0,0 +1,23 @@
<?php
namespace App\Http\Requests\ItemMaster;
use Illuminate\Foundation\Http\FormRequest;
class SectionTemplateUpdateRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'title' => 'sometimes|string|max:255',
'type' => 'sometimes|in:fields,bom',
'description' => 'sometimes|nullable|string',
'is_default' => 'sometimes|nullable|boolean',
];
}
}

View File

@@ -0,0 +1,96 @@
<?php
namespace App\Services\ItemMaster;
use App\Models\ItemMaster\ItemBomItem;
use App\Models\ItemMaster\ItemSection;
use App\Services\Service;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
class ItemBomItemService extends Service
{
/**
* BOM 항목 생성
*/
public function store(int $sectionId, array $data): ItemBomItem
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
// 섹션 존재 확인
$section = ItemSection::where('tenant_id', $tenantId)
->where('id', $sectionId)
->first();
if (! $section) {
throw new NotFoundHttpException(__('error.not_found'));
}
$bomItem = ItemBomItem::create([
'tenant_id' => $tenantId,
'section_id' => $sectionId,
'item_code' => $data['item_code'] ?? null,
'item_name' => $data['item_name'],
'quantity' => $data['quantity'] ?? 1,
'unit' => $data['unit'] ?? null,
'unit_price' => $data['unit_price'] ?? null,
'total_price' => $data['total_price'] ?? null,
'spec' => $data['spec'] ?? null,
'note' => $data['note'] ?? null,
'created_by' => $userId,
]);
return $bomItem;
}
/**
* BOM 항목 수정
*/
public function update(int $id, array $data): ItemBomItem
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$bomItem = ItemBomItem::where('tenant_id', $tenantId)
->where('id', $id)
->first();
if (! $bomItem) {
throw new NotFoundHttpException(__('error.not_found'));
}
$bomItem->update([
'item_code' => $data['item_code'] ?? $bomItem->item_code,
'item_name' => $data['item_name'] ?? $bomItem->item_name,
'quantity' => $data['quantity'] ?? $bomItem->quantity,
'unit' => $data['unit'] ?? $bomItem->unit,
'unit_price' => $data['unit_price'] ?? $bomItem->unit_price,
'total_price' => $data['total_price'] ?? $bomItem->total_price,
'spec' => $data['spec'] ?? $bomItem->spec,
'note' => $data['note'] ?? $bomItem->note,
'updated_by' => $userId,
]);
return $bomItem->fresh();
}
/**
* BOM 항목 삭제
*/
public function destroy(int $id): void
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$bomItem = ItemBomItem::where('tenant_id', $tenantId)
->where('id', $id)
->first();
if (! $bomItem) {
throw new NotFoundHttpException(__('error.not_found'));
}
$bomItem->update(['deleted_by' => $userId]);
$bomItem->delete();
}
}

View File

@@ -0,0 +1,101 @@
<?php
namespace App\Services\ItemMaster;
use App\Models\ItemMaster\ItemMasterField;
use App\Services\Service;
use Illuminate\Database\Eloquent\Collection;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
class ItemMasterFieldService extends Service
{
/**
* 마스터 필드 목록
*/
public function index(): Collection
{
$tenantId = $this->tenantId();
return ItemMasterField::where('tenant_id', $tenantId)
->orderBy('category')
->orderBy('field_name')
->get();
}
/**
* 마스터 필드 생성
*/
public function store(array $data): ItemMasterField
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$field = ItemMasterField::create([
'tenant_id' => $tenantId,
'field_name' => $data['field_name'],
'field_type' => $data['field_type'],
'category' => $data['category'] ?? null,
'description' => $data['description'] ?? null,
'is_common' => $data['is_common'] ?? false,
'default_value' => $data['default_value'] ?? null,
'options' => $data['options'] ?? null,
'validation_rules' => $data['validation_rules'] ?? null,
'properties' => $data['properties'] ?? null,
'created_by' => $userId,
]);
return $field;
}
/**
* 마스터 필드 수정
*/
public function update(int $id, array $data): ItemMasterField
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$field = ItemMasterField::where('tenant_id', $tenantId)
->where('id', $id)
->first();
if (! $field) {
throw new NotFoundHttpException(__('error.not_found'));
}
$field->update([
'field_name' => $data['field_name'] ?? $field->field_name,
'field_type' => $data['field_type'] ?? $field->field_type,
'category' => $data['category'] ?? $field->category,
'description' => $data['description'] ?? $field->description,
'is_common' => $data['is_common'] ?? $field->is_common,
'default_value' => $data['default_value'] ?? $field->default_value,
'options' => $data['options'] ?? $field->options,
'validation_rules' => $data['validation_rules'] ?? $field->validation_rules,
'properties' => $data['properties'] ?? $field->properties,
'updated_by' => $userId,
]);
return $field->fresh();
}
/**
* 마스터 필드 삭제
*/
public function destroy(int $id): void
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$field = ItemMasterField::where('tenant_id', $tenantId)
->where('id', $id)
->first();
if (! $field) {
throw new NotFoundHttpException(__('error.not_found'));
}
$field->update(['deleted_by' => $userId]);
$field->delete();
}
}

View File

@@ -0,0 +1,90 @@
<?php
namespace App\Services\ItemMaster;
use App\Models\ItemMaster\SectionTemplate;
use App\Services\Service;
use Illuminate\Database\Eloquent\Collection;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
class SectionTemplateService extends Service
{
/**
* 섹션 템플릿 목록
*/
public function index(): Collection
{
$tenantId = $this->tenantId();
return SectionTemplate::where('tenant_id', $tenantId)
->orderBy('created_at', 'desc')
->get();
}
/**
* 섹션 템플릿 생성
*/
public function store(array $data): SectionTemplate
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$template = SectionTemplate::create([
'tenant_id' => $tenantId,
'title' => $data['title'],
'type' => $data['type'],
'description' => $data['description'] ?? null,
'is_default' => $data['is_default'] ?? false,
'created_by' => $userId,
]);
return $template;
}
/**
* 섹션 템플릿 수정
*/
public function update(int $id, array $data): SectionTemplate
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$template = SectionTemplate::where('tenant_id', $tenantId)
->where('id', $id)
->first();
if (! $template) {
throw new NotFoundHttpException(__('error.not_found'));
}
$template->update([
'title' => $data['title'] ?? $template->title,
'type' => $data['type'] ?? $template->type,
'description' => $data['description'] ?? $template->description,
'is_default' => $data['is_default'] ?? $template->is_default,
'updated_by' => $userId,
]);
return $template->fresh();
}
/**
* 섹션 템플릿 삭제
*/
public function destroy(int $id): void
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
$template = SectionTemplate::where('tenant_id', $tenantId)
->where('id', $id)
->first();
if (! $template) {
throw new NotFoundHttpException(__('error.not_found'));
}
$template->update(['deleted_by' => $userId]);
$template->delete();
}
}

View File

@@ -19,10 +19,13 @@
use App\Http\Controllers\Api\V1\EstimateController;
use App\Http\Controllers\Api\V1\FileStorageController;
use App\Http\Controllers\Api\V1\FolderController;
use App\Http\Controllers\Api\V1\ItemMaster\ItemBomItemController;
use App\Http\Controllers\Api\V1\ItemMaster\ItemFieldController;
use App\Http\Controllers\Api\V1\ItemMaster\ItemMasterController;
use App\Http\Controllers\Api\V1\ItemMaster\ItemMasterFieldController;
use App\Http\Controllers\Api\V1\ItemMaster\ItemPageController;
use App\Http\Controllers\Api\V1\ItemMaster\ItemSectionController;
use App\Http\Controllers\Api\V1\ItemMaster\SectionTemplateController;
use App\Http\Controllers\Api\V1\MaterialController;
use App\Http\Controllers\Api\V1\MenuController;
use App\Http\Controllers\Api\V1\ModelSetController;
@@ -495,6 +498,23 @@
Route::put('/fields/{id}', [ItemFieldController::class, 'update'])->name('v1.item-master.fields.update');
Route::delete('/fields/{id}', [ItemFieldController::class, 'destroy'])->name('v1.item-master.fields.destroy');
Route::put('/sections/{sectionId}/fields/reorder', [ItemFieldController::class, 'reorder'])->name('v1.item-master.fields.reorder');
// BOM 항목 관리
Route::post('/sections/{sectionId}/bom-items', [ItemBomItemController::class, 'store'])->name('v1.item-master.bom-items.store');
Route::put('/bom-items/{id}', [ItemBomItemController::class, 'update'])->name('v1.item-master.bom-items.update');
Route::delete('/bom-items/{id}', [ItemBomItemController::class, 'destroy'])->name('v1.item-master.bom-items.destroy');
// 섹션 템플릿
Route::get('/section-templates', [SectionTemplateController::class, 'index'])->name('v1.item-master.section-templates.index');
Route::post('/section-templates', [SectionTemplateController::class, 'store'])->name('v1.item-master.section-templates.store');
Route::put('/section-templates/{id}', [SectionTemplateController::class, 'update'])->name('v1.item-master.section-templates.update');
Route::delete('/section-templates/{id}', [SectionTemplateController::class, 'destroy'])->name('v1.item-master.section-templates.destroy');
// 마스터 필드
Route::get('/master-fields', [ItemMasterFieldController::class, 'index'])->name('v1.item-master.master-fields.index');
Route::post('/master-fields', [ItemMasterFieldController::class, 'store'])->name('v1.item-master.master-fields.store');
Route::put('/master-fields/{id}', [ItemMasterFieldController::class, 'update'])->name('v1.item-master.master-fields.update');
Route::delete('/master-fields/{id}', [ItemMasterFieldController::class, 'destroy'])->name('v1.item-master.master-fields.destroy');
});
});