feat: 단가 관리 API 구현 및 Flow Tester 호환성 개선
- Price, PriceRevision 모델 추가 (PriceHistory 대체) - PricingService: CRUD, 원가 조회, 확정 기능 - PricingController: statusCode 파라미터로 201 반환 지원 - NotFoundHttpException(404) 적용 (존재하지 않는 리소스) - FormRequest 분리 (Store, Update, Index, Cost, ByItems) - Swagger 문서 업데이트 - ApiResponse::handle()에 statusCode 옵션 추가 - prices/price_revisions 마이그레이션 및 데이터 이관
This commit is contained in:
@@ -4,93 +4,124 @@
|
||||
|
||||
use App\Helpers\ApiResponse;
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Services\Pricing\PricingService;
|
||||
use Illuminate\Http\Request;
|
||||
use App\Http\Requests\Pricing\PriceByItemsRequest;
|
||||
use App\Http\Requests\Pricing\PriceCostRequest;
|
||||
use App\Http\Requests\Pricing\PriceIndexRequest;
|
||||
use App\Http\Requests\Pricing\PriceStoreRequest;
|
||||
use App\Http\Requests\Pricing\PriceUpdateRequest;
|
||||
use App\Services\PricingService;
|
||||
|
||||
class PricingController extends Controller
|
||||
{
|
||||
protected PricingService $service;
|
||||
|
||||
public function __construct(PricingService $service)
|
||||
{
|
||||
$this->service = $service;
|
||||
}
|
||||
public function __construct(
|
||||
protected PricingService $service
|
||||
) {}
|
||||
|
||||
/**
|
||||
* 가격 이력 목록 조회
|
||||
* 단가 목록 조회
|
||||
*/
|
||||
public function index(Request $request)
|
||||
public function index(PriceIndexRequest $request)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($request) {
|
||||
$filters = $request->only([
|
||||
'item_type_code',
|
||||
'item_id',
|
||||
'price_type_code',
|
||||
'client_group_id',
|
||||
'date',
|
||||
]);
|
||||
$perPage = (int) ($request->input('size') ?? 15);
|
||||
|
||||
$data = $this->service->listPrices($filters, $perPage);
|
||||
$data = $this->service->index($request->validated());
|
||||
|
||||
return ['data' => $data, 'message' => __('message.fetched')];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 단일 항목 가격 조회
|
||||
* 단가 상세 조회
|
||||
*/
|
||||
public function show(Request $request)
|
||||
public function show(int $id)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($request) {
|
||||
$itemType = $request->input('item_type'); // PRODUCT | MATERIAL
|
||||
$itemId = (int) $request->input('item_id');
|
||||
$clientId = $request->input('client_id') ? (int) $request->input('client_id') : null;
|
||||
$date = $request->input('date') ?? null;
|
||||
return ApiResponse::handle(function () use ($id) {
|
||||
$data = $this->service->show($id);
|
||||
|
||||
$result = $this->service->getItemPrice($itemType, $itemId, $clientId, $date);
|
||||
|
||||
return ['data' => $result, 'message' => __('message.fetched')];
|
||||
return ['data' => $data, 'message' => __('message.fetched')];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 여러 항목 일괄 가격 조회
|
||||
* 단가 등록
|
||||
*/
|
||||
public function bulk(Request $request)
|
||||
public function store(PriceStoreRequest $request)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($request) {
|
||||
$items = $request->input('items'); // [['item_type' => 'PRODUCT', 'item_id' => 1], ...]
|
||||
$clientId = $request->input('client_id') ? (int) $request->input('client_id') : null;
|
||||
$date = $request->input('date') ?? null;
|
||||
$data = $this->service->store($request->validated());
|
||||
|
||||
$result = $this->service->getBulkItemPrices($items, $clientId, $date);
|
||||
|
||||
return ['data' => $result, 'message' => __('message.fetched')];
|
||||
return ['data' => $data, 'message' => __('message.created'), 'statusCode' => 201];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 가격 등록/수정
|
||||
* 단가 수정
|
||||
*/
|
||||
public function upsert(Request $request)
|
||||
public function update(PriceUpdateRequest $request, int $id)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($request) {
|
||||
$data = $this->service->upsertPrice($request->all());
|
||||
return ApiResponse::handle(function () use ($request, $id) {
|
||||
$data = $this->service->update($id, $request->validated());
|
||||
|
||||
return ['data' => $data, 'message' => __('message.created')];
|
||||
return ['data' => $data, 'message' => __('message.updated')];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 가격 삭제
|
||||
* 단가 삭제
|
||||
*/
|
||||
public function destroy(int $id)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($id) {
|
||||
$this->service->deletePrice($id);
|
||||
$this->service->destroy($id);
|
||||
|
||||
return ['data' => null, 'message' => __('message.deleted')];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 단가 확정
|
||||
*/
|
||||
public function finalize(int $id)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($id) {
|
||||
$data = $this->service->finalize($id);
|
||||
|
||||
return ['data' => $data, 'message' => __('message.pricing.finalized')];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 품목별 단가 현황 조회
|
||||
*/
|
||||
public function byItems(PriceByItemsRequest $request)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($request) {
|
||||
$data = $this->service->byItems($request->validated());
|
||||
|
||||
return ['data' => $data, 'message' => __('message.fetched')];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 변경 이력 조회
|
||||
*/
|
||||
public function revisions(int $id)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($id) {
|
||||
$data = $this->service->revisions($id);
|
||||
|
||||
return ['data' => $data, 'message' => __('message.fetched')];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 원가 조회 (수입검사 > 표준원가 fallback)
|
||||
*/
|
||||
public function cost(PriceCostRequest $request)
|
||||
{
|
||||
return ApiResponse::handle(function () use ($request) {
|
||||
$data = $this->service->getCost($request->validated());
|
||||
|
||||
return ['data' => $data, 'message' => __('message.fetched')];
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
24
app/Http/Requests/Pricing/PriceByItemsRequest.php
Normal file
24
app/Http/Requests/Pricing/PriceByItemsRequest.php
Normal file
@@ -0,0 +1,24 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests\Pricing;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
|
||||
class PriceByItemsRequest extends FormRequest
|
||||
{
|
||||
public function authorize(): bool
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public function rules(): array
|
||||
{
|
||||
return [
|
||||
'items' => 'required|array|min:1|max:100',
|
||||
'items.*.item_type_code' => 'required|string|in:PRODUCT,MATERIAL',
|
||||
'items.*.item_id' => 'required|integer',
|
||||
'client_group_id' => 'nullable|integer',
|
||||
'date' => 'nullable|date',
|
||||
];
|
||||
}
|
||||
}
|
||||
22
app/Http/Requests/Pricing/PriceCostRequest.php
Normal file
22
app/Http/Requests/Pricing/PriceCostRequest.php
Normal file
@@ -0,0 +1,22 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests\Pricing;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
|
||||
class PriceCostRequest extends FormRequest
|
||||
{
|
||||
public function authorize(): bool
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public function rules(): array
|
||||
{
|
||||
return [
|
||||
'item_type_code' => 'required|string|in:PRODUCT,MATERIAL',
|
||||
'item_id' => 'required|integer',
|
||||
'date' => 'nullable|date',
|
||||
];
|
||||
}
|
||||
}
|
||||
27
app/Http/Requests/Pricing/PriceIndexRequest.php
Normal file
27
app/Http/Requests/Pricing/PriceIndexRequest.php
Normal file
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests\Pricing;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
|
||||
class PriceIndexRequest extends FormRequest
|
||||
{
|
||||
public function authorize(): bool
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public function rules(): array
|
||||
{
|
||||
return [
|
||||
'size' => 'nullable|integer|min:1|max:100',
|
||||
'page' => 'nullable|integer|min:1',
|
||||
'q' => 'nullable|string|max:100',
|
||||
'item_type_code' => 'nullable|string|in:PRODUCT,MATERIAL',
|
||||
'item_id' => 'nullable|integer',
|
||||
'client_group_id' => 'nullable',
|
||||
'status' => 'nullable|string|in:draft,active,inactive,finalized',
|
||||
'valid_at' => 'nullable|date',
|
||||
];
|
||||
}
|
||||
}
|
||||
51
app/Http/Requests/Pricing/PriceStoreRequest.php
Normal file
51
app/Http/Requests/Pricing/PriceStoreRequest.php
Normal file
@@ -0,0 +1,51 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests\Pricing;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
use Illuminate\Validation\Rule;
|
||||
|
||||
class PriceStoreRequest extends FormRequest
|
||||
{
|
||||
public function authorize(): bool
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public function rules(): array
|
||||
{
|
||||
return [
|
||||
// 품목 연결 (필수)
|
||||
'item_type_code' => 'required|string|in:PRODUCT,MATERIAL',
|
||||
'item_id' => 'required|integer',
|
||||
'client_group_id' => 'nullable|integer',
|
||||
|
||||
// 원가 정보
|
||||
'purchase_price' => 'nullable|numeric|min:0',
|
||||
'processing_cost' => 'nullable|numeric|min:0',
|
||||
'loss_rate' => 'nullable|numeric|min:0|max:100',
|
||||
|
||||
// 판매가 정보
|
||||
'margin_rate' => 'nullable|numeric|min:0|max:100',
|
||||
'sales_price' => 'nullable|numeric|min:0',
|
||||
'rounding_rule' => ['nullable', Rule::in(['round', 'ceil', 'floor'])],
|
||||
'rounding_unit' => ['nullable', Rule::in([1, 10, 100, 1000])],
|
||||
|
||||
// 메타 정보
|
||||
'supplier' => 'nullable|string|max:255',
|
||||
'effective_from' => 'required|date',
|
||||
'effective_to' => 'nullable|date|after_or_equal:effective_from',
|
||||
'note' => 'nullable|string|max:1000',
|
||||
|
||||
// 상태
|
||||
'status' => ['nullable', Rule::in(['draft', 'active', 'inactive'])],
|
||||
];
|
||||
}
|
||||
|
||||
public function messages(): array
|
||||
{
|
||||
return [
|
||||
'effective_to.after_or_equal' => __('error.pricing.effective_to_must_be_after_from'),
|
||||
];
|
||||
}
|
||||
}
|
||||
54
app/Http/Requests/Pricing/PriceUpdateRequest.php
Normal file
54
app/Http/Requests/Pricing/PriceUpdateRequest.php
Normal file
@@ -0,0 +1,54 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests\Pricing;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
use Illuminate\Validation\Rule;
|
||||
|
||||
class PriceUpdateRequest extends FormRequest
|
||||
{
|
||||
public function authorize(): bool
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public function rules(): array
|
||||
{
|
||||
return [
|
||||
// 품목 연결 (선택)
|
||||
'item_type_code' => 'sometimes|string|in:PRODUCT,MATERIAL',
|
||||
'item_id' => 'sometimes|integer',
|
||||
'client_group_id' => 'nullable|integer',
|
||||
|
||||
// 원가 정보
|
||||
'purchase_price' => 'nullable|numeric|min:0',
|
||||
'processing_cost' => 'nullable|numeric|min:0',
|
||||
'loss_rate' => 'nullable|numeric|min:0|max:100',
|
||||
|
||||
// 판매가 정보
|
||||
'margin_rate' => 'nullable|numeric|min:0|max:100',
|
||||
'sales_price' => 'nullable|numeric|min:0',
|
||||
'rounding_rule' => ['nullable', Rule::in(['round', 'ceil', 'floor'])],
|
||||
'rounding_unit' => ['nullable', Rule::in([1, 10, 100, 1000])],
|
||||
|
||||
// 메타 정보
|
||||
'supplier' => 'nullable|string|max:255',
|
||||
'effective_from' => 'sometimes|date',
|
||||
'effective_to' => 'nullable|date|after_or_equal:effective_from',
|
||||
'note' => 'nullable|string|max:1000',
|
||||
|
||||
// 상태
|
||||
'status' => ['nullable', Rule::in(['draft', 'active', 'inactive'])],
|
||||
|
||||
// 변경 사유 (리비전 기록용)
|
||||
'change_reason' => 'nullable|string|max:500',
|
||||
];
|
||||
}
|
||||
|
||||
public function messages(): array
|
||||
{
|
||||
return [
|
||||
'effective_to.after_or_equal' => __('error.pricing.effective_to_must_be_after_from'),
|
||||
];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user