refactor(form-builder): consolidate subject-type allow-list into purpose registry

Q6 of ARCH-CONSOLIDATION-ADDENDUM-2026-04-24: the allowed
`form_submissions.subject_type` values are now derived from
`PurposeRegistry::allSubjectTypes()` instead of the parallel
`config/form_subjects.php` file.

- CreateFormSubmissionRequest validates `subject_type` against the
  registry via constructor-injected PurposeRegistry.
- FormSubmissionController and FormValueService resolve the subject
  FQCN through `Relation::getMorphedModel()` — the morph-map is the
  single source of truth for alias → model mapping.
- `config/form_subjects.php` is deleted. `MorphMapAlignmentTest` keeps
  the registry and morph-map aligned going forward.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-04-24 14:35:48 +02:00
parent b9343f6eec
commit ab67ed46ca
4 changed files with 13 additions and 61 deletions

View File

@@ -14,6 +14,7 @@ use App\Models\FormBuilder\FormSubmission;
use App\Models\Organisation;
use App\Services\FormBuilder\FormSubmissionService;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\Relation;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\Resources\Json\AnonymousResourceCollection;
use Illuminate\Support\Facades\Gate;
@@ -103,9 +104,8 @@ final class FormSubmissionController extends Controller
if ($type === '' || $id === '') {
return null;
}
$map = (array) config('form_subjects');
$class = $map[$type]['model'] ?? null;
if ($class === null) {
$class = Relation::getMorphedModel($type);
if ($class === null || ! class_exists($class)) {
return null;
}

View File

@@ -4,11 +4,18 @@ declare(strict_types=1);
namespace App\Http\Requests\Api\V1\FormBuilder;
use App\FormBuilder\Purposes\PurposeRegistry;
use Illuminate\Foundation\Http\FormRequest;
use Illuminate\Validation\Rule;
final class CreateFormSubmissionRequest extends FormRequest
{
public function __construct(
private readonly PurposeRegistry $purposeRegistry,
) {
parent::__construct();
}
public function authorize(): bool
{
return true;
@@ -19,10 +26,8 @@ final class CreateFormSubmissionRequest extends FormRequest
*/
public function rules(): array
{
$allowedSubjects = array_keys((array) config('form_subjects', []));
return [
'subject_type' => ['nullable', Rule::in($allowedSubjects)],
'subject_type' => ['nullable', Rule::in($this->purposeRegistry->allSubjectTypes())],
'subject_id' => ['nullable', 'string', 'max:30', 'required_with:subject_type'],
'idempotency_key' => ['nullable', 'string', 'max:30'],
'is_test' => ['boolean'],

View File

@@ -340,9 +340,8 @@ final class FormValueService
}
if ($subjectType === $entity) {
$map = config('form_subjects');
$model = $map[$entity]['model'] ?? null;
if ($model === null) {
$model = \Illuminate\Database\Eloquent\Relations\Relation::getMorphedModel($entity);
if ($model === null || ! class_exists($model)) {
return null;
}