'required', 'service.docker_compose' => 'required', 'service.name' => ValidationPatterns::nameRules(), 'service.description' => ValidationPatterns::descriptionRules(), 'service.connect_to_docker_network' => 'nullable', ]; // Add dynamic field rules foreach ($this->fields ?? collect() as $key => $field) { $rules = data_get($field, 'rules', 'nullable'); $baseRules["fields.$key.value"] = $rules; } return $baseRules; } protected function messages(): array { return array_merge( ValidationPatterns::combinedMessages(), [ 'service.name.required' => 'The Name field is required.', 'service.name.regex' => 'The Name may only contain letters, numbers, spaces, dashes (-), underscores (_), dots (.), slashes (/), colons (:), and parentheses ().', 'service.description.regex' => 'The Description contains invalid characters. Only letters, numbers, spaces, and common punctuation (- _ . : / () \' " , ! ? @ # % & + = [] {} | ~ ` *) are allowed.', 'service.docker_compose_raw.required' => 'The Docker Compose Raw field is required.', 'service.docker_compose.required' => 'The Docker Compose field is required.', ] ); } public $validationAttributes = []; public function mount() { $this->fields = collect([]); $extraFields = $this->service->extraFields(); foreach ($extraFields as $serviceName => $fields) { foreach ($fields as $fieldKey => $field) { $key = data_get($field, 'key'); $value = data_get($field, 'value'); $rules = data_get($field, 'rules', 'nullable'); $isPassword = data_get($field, 'isPassword', false); $customHelper = data_get($field, 'customHelper', false); $this->fields->put($key, [ 'serviceName' => $serviceName, 'key' => $key, 'name' => $fieldKey, 'value' => $value, 'isPassword' => $isPassword, 'rules' => $rules, 'customHelper' => $customHelper, ]); $this->validationAttributes["fields.$key.value"] = $fieldKey; } } $this->fields = $this->fields->groupBy('serviceName')->map(function ($group) { return $group->sortBy(function ($field) { return data_get($field, 'isPassword') ? 1 : 0; })->mapWithKeys(function ($field) { return [$field['key'] => $field]; }); })->flatMap(function ($group) { return $group; }); } public function saveCompose($raw) { $this->service->docker_compose_raw = $raw; $this->submit(notify: true); } public function instantSave() { $this->service->save(); $this->dispatch('success', 'Service settings saved.'); } public function submit($notify = true) { try { $this->validate(); $this->service->save(); $this->service->saveExtraFields($this->fields); $this->service->parse(); $this->service->refresh(); $this->service->saveComposeConfigs(); $this->dispatch('refreshEnvs'); $this->dispatch('refreshServices'); $notify && $this->dispatch('success', 'Service saved.'); } catch (\Throwable $e) { return handleError($e, $this); } finally { if (is_null($this->service->config_hash)) { $this->service->isConfigurationChanged(true); } else { $this->dispatch('configurationChanged'); } } } public function render() { return view('livewire.project.service.stack-form'); } }