feat(labor): 노임관리 API 구현

- Labor 모델 (BelongsToTenant, SoftDeletes)
- LaborController 7개 엔드포인트
- LaborService 비즈니스 로직
- FormRequest 4개 (Index/Store/Update/BulkDelete)
- 마이그레이션 및 라우트 등록

API: GET/POST /labor, GET/PUT/DELETE /labor/{id}, DELETE /labor/bulk, GET /labor/stats

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
2026-01-11 23:29:32 +09:00
parent ceb7798c28
commit f59dd1b9fb
11 changed files with 703 additions and 0 deletions

View File

@@ -0,0 +1,31 @@
<?php
namespace App\Http\Requests\Labor;
use Illuminate\Foundation\Http\FormRequest;
class LaborBulkDeleteRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'ids' => ['required', 'array', 'min:1'],
'ids.*' => ['required', 'integer', 'exists:labors,id'],
];
}
public function messages(): array
{
return [
'ids.required' => __('validation.required', ['attribute' => '삭제 대상']),
'ids.array' => __('validation.array', ['attribute' => '삭제 대상']),
'ids.min' => __('validation.min.array', ['attribute' => '삭제 대상', 'min' => 1]),
'ids.*.exists' => __('validation.exists', ['attribute' => '노임 ID']),
];
}
}

View File

@@ -0,0 +1,37 @@
<?php
namespace App\Http\Requests\Labor;
use Illuminate\Foundation\Http\FormRequest;
class LaborIndexRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'search' => ['nullable', 'string', 'max:100'],
'category' => ['nullable', 'string', 'in:가로,세로할증'],
'status' => ['nullable', 'string', 'in:사용,중지'],
'start_date' => ['nullable', 'date'],
'end_date' => ['nullable', 'date', 'after_or_equal:start_date'],
'page' => ['nullable', 'integer', 'min:1'],
'per_page' => ['nullable', 'integer', 'min:1', 'max:100'],
'sort_by' => ['nullable', 'string', 'in:created_at,labor_number,category,min_m,max_m,labor_price'],
'sort_dir' => ['nullable', 'string', 'in:asc,desc'],
];
}
public function messages(): array
{
return [
'category.in' => __('validation.in', ['attribute' => '구분']),
'status.in' => __('validation.in', ['attribute' => '상태']),
'end_date.after_or_equal' => __('validation.after_or_equal', ['attribute' => '종료일', 'date' => '시작일']),
];
}
}

View File

@@ -0,0 +1,39 @@
<?php
namespace App\Http\Requests\Labor;
use Illuminate\Foundation\Http\FormRequest;
class LaborStoreRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'labor_number' => ['required', 'string', 'max:50'],
'category' => ['required', 'string', 'in:가로,세로할증'],
'min_m' => ['required', 'numeric', 'min:0', 'max:9999.99'],
'max_m' => ['required', 'numeric', 'min:0', 'max:9999.99', 'gte:min_m'],
'labor_price' => ['nullable', 'integer', 'min:0'],
'status' => ['required', 'string', 'in:사용,중지'],
];
}
public function messages(): array
{
return [
'labor_number.required' => __('validation.required', ['attribute' => '노임번호']),
'category.required' => __('validation.required', ['attribute' => '구분']),
'category.in' => __('validation.in', ['attribute' => '구분']),
'min_m.required' => __('validation.required', ['attribute' => '최소(m)']),
'max_m.required' => __('validation.required', ['attribute' => '최대(m)']),
'max_m.gte' => __('validation.gte.numeric', ['attribute' => '최대(m)', 'value' => '최소(m)']),
'status.required' => __('validation.required', ['attribute' => '상태']),
'status.in' => __('validation.in', ['attribute' => '상태']),
];
}
}

View File

@@ -0,0 +1,39 @@
<?php
namespace App\Http\Requests\Labor;
use Illuminate\Foundation\Http\FormRequest;
class LaborUpdateRequest extends FormRequest
{
public function authorize(): bool
{
return true;
}
public function rules(): array
{
return [
'labor_number' => ['sometimes', 'required', 'string', 'max:50'],
'category' => ['sometimes', 'required', 'string', 'in:가로,세로할증'],
'min_m' => ['sometimes', 'required', 'numeric', 'min:0', 'max:9999.99'],
'max_m' => ['sometimes', 'required', 'numeric', 'min:0', 'max:9999.99', 'gte:min_m'],
'labor_price' => ['nullable', 'integer', 'min:0'],
'status' => ['sometimes', 'required', 'string', 'in:사용,중지'],
];
}
public function messages(): array
{
return [
'labor_number.required' => __('validation.required', ['attribute' => '노임번호']),
'category.required' => __('validation.required', ['attribute' => '구분']),
'category.in' => __('validation.in', ['attribute' => '구분']),
'min_m.required' => __('validation.required', ['attribute' => '최소(m)']),
'max_m.required' => __('validation.required', ['attribute' => '최대(m)']),
'max_m.gte' => __('validation.gte.numeric', ['attribute' => '최대(m)', 'value' => '최소(m)']),
'status.required' => __('validation.required', ['attribute' => '상태']),
'status.in' => __('validation.in', ['attribute' => '상태']),
];
}
}