From 5604d1d076128fb7f0dc1dbc409954cea5e944cd Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 14 Aug 2025 10:44:11 -0600 Subject: [PATCH 01/47] feat: first draft of PromptBuilder --- src/Builders/PromptBuilder.php | 613 +++++++++++++++++++++++++++++++++ 1 file changed, 613 insertions(+) create mode 100644 src/Builders/PromptBuilder.php diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php new file mode 100644 index 00000000..17a27930 --- /dev/null +++ b/src/Builders/PromptBuilder.php @@ -0,0 +1,613 @@ + The inferred required capabilities. + */ + protected array $inferredCapabilities = []; + + /** + * @var array The inferred required options. + */ + protected array $inferredOptions = []; + + /** + * @var Message|null The system instruction message. + */ + protected ?Message $systemInstruction = null; + + /** + * Constructor. + * + * @since n.e.x.t + * + * @param string|Message|null $text Optional initial text or message. + */ + public function __construct($text = null) + { + $this->modelConfig = new ModelConfig(); + + if ($text !== null) { + if ($text instanceof Message) { + $this->messages[] = $text; + // Infer chat history capability if multiple messages + if (count($this->messages) > 0) { + $this->addInferredCapability(CapabilityEnum::chatHistory()); + } + } else { + $this->withText($text); + } + } + } + + /** + * Adds text to the current message. + * + * @since n.e.x.t + * + * @param string $text The text to add. + * @return self + */ + public function withText(string $text): self + { + $this->currentParts[] = new MessagePart($text); + $this->addInferredCapability(CapabilityEnum::textGeneration()); + return $this; + } + + /** + * Adds an inline image to the current message. + * + * @since n.e.x.t + * + * @param string $base64Blob The base64-encoded image data. + * @param string $mimeType The MIME type of the image. + * @return self + */ + public function withInlineImage(string $base64Blob, string $mimeType): self + { + // Create data URI format for inline image + $dataUri = 'data:' . $mimeType . ';base64,' . $base64Blob; + $file = new File($dataUri, $mimeType); + $this->currentParts[] = new MessagePart($file); + $this->addMultimodalInputCapability(); + return $this; + } + + /** + * Adds a remote image to the current message. + * + * @since n.e.x.t + * + * @param string $uri The URI of the remote image. + * @param string $mimeType The MIME type of the image. + * @return self + */ + public function withRemoteImage(string $uri, string $mimeType): self + { + $file = new File($uri, $mimeType); + $this->currentParts[] = new MessagePart($file); + $this->addMultimodalInputCapability(); + return $this; + } + + /** + * Adds an image file to the current message. + * + * @since n.e.x.t + * + * @param File $file The image file. + * @return self + */ + public function withImageFile(File $file): self + { + $this->currentParts[] = new MessagePart($file); + $this->addMultimodalInputCapability(); + return $this; + } + + /** + * Adds an audio file to the current message. + * + * @since n.e.x.t + * + * @param File $file The audio file. + * @return self + */ + public function withAudioFile(File $file): self + { + $this->currentParts[] = new MessagePart($file); + $this->addMultimodalInputCapability('audio'); + return $this; + } + + /** + * Adds a video file to the current message. + * + * @since n.e.x.t + * + * @param File $file The video file. + * @return self + */ + public function withVideoFile(File $file): self + { + $this->currentParts[] = new MessagePart($file); + $this->addMultimodalInputCapability('video'); + return $this; + } + + /** + * Adds a function response to the current message. + * + * @since n.e.x.t + * + * @param FunctionResponse $functionResponse The function response. + * @return self + */ + public function withFunctionResponse(FunctionResponse $functionResponse): self + { + $this->currentParts[] = new MessagePart($functionResponse); + return $this; + } + + /** + * Adds message parts to the current message. + * + * @since n.e.x.t + * + * @param MessagePart ...$parts The message parts to add. + * @return self + */ + public function withMessageParts(MessagePart ...$parts): self + { + foreach ($parts as $part) { + $this->currentParts[] = $part; + // Infer capabilities based on part type + if ($part->getFile() !== null) { + $this->addMultimodalInputCapability(); + } + } + return $this; + } + + /** + * Adds conversation history messages. + * + * @since n.e.x.t + * + * @param Message ...$messages The messages to add to history. + * @return self + */ + public function withHistory(Message ...$messages): self + { + // Save current parts as a user message if any exist + if (!empty($this->currentParts)) { + $this->messages[] = new UserMessage($this->currentParts); + $this->currentParts = []; + } + + foreach ($messages as $message) { + $this->messages[] = $message; + } + + $this->addInferredCapability(CapabilityEnum::chatHistory()); + return $this; + } + + /** + * Sets the model to use for generation. + * + * @since n.e.x.t + * + * @param ModelInterface $model The model to use. + * @return self + */ + public function usingModel(ModelInterface $model): self + { + $this->model = $model; + return $this; + } + + /** + * Sets the system instruction. + * + * @since n.e.x.t + * + * @param string|MessagePart[]|Message $systemInstruction The system instruction. + * @return self + */ + public function usingSystemInstruction($systemInstruction): self + { + $systemInstructionText = ''; + + if ($systemInstruction instanceof Message) { + $this->systemInstruction = $systemInstruction; + // Extract text from message parts for ModelConfig + foreach ($systemInstruction->getParts() as $part) { + if ($part->getText() !== null) { + $systemInstructionText .= $part->getText() . ' '; + } + } + } elseif (is_string($systemInstruction)) { + $this->systemInstruction = new Message( + MessageRoleEnum::system(), + [new MessagePart($systemInstruction)] + ); + $systemInstructionText = $systemInstruction; + } elseif (is_array($systemInstruction)) { + $this->systemInstruction = new Message( + MessageRoleEnum::system(), + $systemInstruction + ); + // Extract text from message parts + foreach ($systemInstruction as $part) { + if ($part instanceof MessagePart && $part->getText() !== null) { + $systemInstructionText .= $part->getText() . ' '; + } + } + } + + if (!empty($systemInstructionText)) { + $this->modelConfig->setSystemInstruction(trim($systemInstructionText)); + } + return $this; + } + + /** + * Sets the maximum number of tokens to generate. + * + * @since n.e.x.t + * + * @param int $maxTokens The maximum number of tokens. + * @return self + */ + public function usingMaxTokens(int $maxTokens): self + { + $this->modelConfig->setMaxTokens($maxTokens); + return $this; + } + + /** + * Sets the temperature for generation. + * + * @since n.e.x.t + * + * @param float $temperature The temperature value. + * @return self + */ + public function usingTemperature(float $temperature): self + { + $this->modelConfig->setTemperature($temperature); + return $this; + } + + /** + * Sets the top-p value for generation. + * + * @since n.e.x.t + * + * @param float $topP The top-p value. + * @return self + */ + public function usingTopP(float $topP): self + { + $this->modelConfig->setTopP($topP); + return $this; + } + + /** + * Sets the top-k value for generation. + * + * @since n.e.x.t + * + * @param int $topK The top-k value. + * @return self + */ + public function usingTopK(int $topK): self + { + $this->modelConfig->setTopK($topK); + return $this; + } + + /** + * Sets stop sequences for generation. + * + * @since n.e.x.t + * + * @param string ...$stopSequences The stop sequences. + * @return self + */ + public function usingStopSequences(string ...$stopSequences): self + { + $this->modelConfig->setCustomOption('stopSequences', $stopSequences); + return $this; + } + + /** + * Sets the number of candidates to generate. + * + * @since n.e.x.t + * + * @param int $candidateCount The number of candidates. + * @return self + */ + public function usingCandidateCount(int $candidateCount): self + { + $this->modelConfig->setCandidateCount($candidateCount); + return $this; + } + + /** + * Sets the output MIME type. + * + * @since n.e.x.t + * + * @param string $mimeType The MIME type. + * @return self + */ + public function usingOutputMime(string $mimeType): self + { + $this->modelConfig->setOutputMimeType($mimeType); + $this->inferredOptions[OptionEnum::outputMimeType()->value] = $mimeType; + return $this; + } + + /** + * Sets the output schema. + * + * @since n.e.x.t + * + * @param array $schema The output schema. + * @return self + */ + public function usingOutputSchema(array $schema): self + { + $this->modelConfig->setOutputSchema($schema); + $this->inferredOptions[OptionEnum::outputSchema()->value] = true; + return $this; + } + + /** + * Sets the output modalities. + * + * @since n.e.x.t + * + * @param ModalityEnum ...$modalities The output modalities. + * @return self + */ + public function usingOutputModalities(ModalityEnum ...$modalities): self + { + $this->modelConfig->setOutputModalities($modalities); + return $this; + } + + /** + * Configures the prompt for JSON response output. + * + * @since n.e.x.t + * + * @param array|null $schema Optional JSON schema. + * @return self + */ + public function asJsonResponse(?array $schema = null): self + { + $this->usingOutputMime('application/json'); + if ($schema !== null) { + $this->usingOutputSchema($schema); + } + return $this; + } + + /** + * Gets the inferred model requirements based on prompt features. + * + * @since n.e.x.t + * + * @return ModelRequirements The inferred requirements. + */ + public function getModelRequirements(): ModelRequirements + { + $requiredOptions = []; + foreach ($this->inferredOptions as $name => $value) { + $requiredOptions[] = new RequiredOption($name, $value); + } + + return new ModelRequirements( + $this->inferredCapabilities, + $requiredOptions + ); + } + + /** + * Checks if the current prompt is supported by the selected model. + * + * @since n.e.x.t + * + * @return bool True if supported, false otherwise. + */ + public function isSupported(): bool + { + if ($this->model === null) { + // Without a model selected, we can't determine support + return true; + } + + $requirements = $this->getModelRequirements(); + return $this->model->metadata()->meetsRequirements($requirements); + } + + /** + * Generates text from the prompt. + * + * @since n.e.x.t + * + * @return string The generated text. + * @throws InvalidArgumentException If the selected model doesn't meet requirements. + */ + public function generateText(): string + { + $this->validateModel(); + $this->prepareMessages(); + + // This is a placeholder - actual implementation would call the model + throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); + } + + /** + * Generates multiple text candidates from the prompt. + * + * @since n.e.x.t + * + * @param int|null $candidateCount The number of candidates to generate. + * @return list The generated texts. + * @throws InvalidArgumentException If the selected model doesn't meet requirements. + */ + public function generateTexts(?int $candidateCount = null): array + { + if ($candidateCount !== null) { + $this->usingCandidateCount($candidateCount); + } + + $this->validateModel(); + $this->prepareMessages(); + + // This is a placeholder - actual implementation would call the model + throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); + } + + /** + * Adds an inferred capability. + * + * @since n.e.x.t + * + * @param CapabilityEnum $capability The capability to add. + * @return void + */ + protected function addInferredCapability(CapabilityEnum $capability): void + { + if (!in_array($capability, $this->inferredCapabilities, true)) { + $this->inferredCapabilities[] = $capability; + } + } + + /** + * Adds multimodal input capability based on content type. + * + * @since n.e.x.t + * + * @param string $type The content type (image, audio, video). + * @return void + */ + protected function addMultimodalInputCapability(string $type = 'image'): void + { + $this->addInferredCapability(CapabilityEnum::textGeneration()); + + // Add input modalities requirement + $modalityKey = OptionEnum::inputModalities()->value; + $currentModalities = isset($this->inferredOptions[$modalityKey]) + && is_array($this->inferredOptions[$modalityKey]) + ? $this->inferredOptions[$modalityKey] + : ['text']; + + if (!in_array($type, $currentModalities, true)) { + $currentModalities[] = $type; + $this->inferredOptions[$modalityKey] = $currentModalities; + } + } + + /** + * Validates that the selected model meets requirements. + * + * @since n.e.x.t + * + * @return void + * @throws InvalidArgumentException If model doesn't meet requirements. + */ + protected function validateModel(): void + { + if ($this->model === null) { + // Model selection would happen here via registry + return; + } + + $requirements = $this->getModelRequirements(); + if (!$this->model->metadata()->meetsRequirements($requirements)) { + throw new InvalidArgumentException( + sprintf( + 'The selected model "%s" does not meet the required capabilities and options for this prompt.', + $this->model->metadata()->getId() + ) + ); + } + } + + /** + * Prepares the final messages array for sending to the model. + * + * @since n.e.x.t + * + * @return void + */ + protected function prepareMessages(): void + { + // Add current parts as a user message if any exist + if (!empty($this->currentParts)) { + $this->messages[] = new UserMessage($this->currentParts); + $this->currentParts = []; + } + + // Add system instruction if present + if ($this->systemInstruction !== null) { + array_unshift($this->messages, $this->systemInstruction); + } + } +} From 3e436ee55aebd0009ebc9b451ec8a9982823221f Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 09:37:39 -0600 Subject: [PATCH 02/47] feat: adds Prompts utility class --- src/Common/Utilities/Prompts.php | 174 ++++++++++++++++ tests/unit/Common/Utilities/PromptsTest.php | 214 ++++++++++++++++++++ 2 files changed, 388 insertions(+) create mode 100644 src/Common/Utilities/Prompts.php create mode 100644 tests/unit/Common/Utilities/PromptsTest.php diff --git a/src/Common/Utilities/Prompts.php b/src/Common/Utilities/Prompts.php new file mode 100644 index 00000000..f834d2ac --- /dev/null +++ b/src/Common/Utilities/Prompts.php @@ -0,0 +1,174 @@ +|list $prompt The prompt to normalize. + * @return list The normalized list of messages. + * @throws InvalidArgumentException If the prompt format is invalid. + */ + // phpcs:enable Generic.Files.LineLength.TooLong + public static function normalizeToMessages($prompt): array + { + // 1. Check if it's already a list of Messages + if (self::isMessagesList($prompt)) { + return $prompt; + } + + // 2. Check if it's a single Message + if ($prompt instanceof Message) { + return [$prompt]; + } + + // 3. Check if it's a MessageArrayShape (single message as array) + if (self::isMessageArrayShape($prompt)) { + return [Message::fromArray($prompt)]; + } + + // 4. If it's not an array, wrap it in an array + if (!is_array($prompt)) { + $prompt = [$prompt]; + } + + // 5. Loop through the array and handle conversions - all become parts of a single UserMessage + $parts = []; + + foreach ($prompt as $item) { + if (is_string($item)) { + $parts[] = new MessagePart($item); + } elseif ($item instanceof MessagePart) { + $parts[] = $item; + } elseif (self::isMessagePartArrayShape($item)) { + $parts[] = MessagePart::fromArray($item); + } else { + $type = is_object($item) ? get_class($item) : gettype($item); + throw new InvalidArgumentException( + sprintf('Invalid item type %s in prompt.', $type) + ); + } + } + + return [new UserMessage($parts)]; + } + + /** + * Checks if the value is a list of Message objects. + * + * @since n.e.x.t + * + * @param mixed $value The value to check. + * @return bool True if the value is a list of Message objects. + * + * @phpstan-assert-if-true list $value + */ + private static function isMessagesList($value): bool + { + if (!is_array($value) || empty($value) || !array_is_list($value)) { + return false; + } + + // Check if all items are Messages + foreach ($value as $item) { + if (!($item instanceof Message)) { + return false; + } + } + + return true; + } + + /** + * Checks if the value is a MessageArrayShape. + * + * @since n.e.x.t + * + * @param mixed $value The value to check. + * @return bool True if the value is a MessageArrayShape. + * + * @phpstan-assert-if-true MessageArrayShape $value + */ + private static function isMessageArrayShape($value): bool + { + if (!is_array($value)) { + return false; + } + + // Must have required keys + if (!isset($value['role']) || !isset($value['parts'])) { + return false; + } + + // Role must be a string + if (!is_string($value['role'])) { + return false; + } + + // Parts must be an array + if (!is_array($value['parts'])) { + return false; + } + + return true; + } + + /** + * Checks if the value is a MessagePartArrayShape. + * + * @since n.e.x.t + * + * @param mixed $value The value to check. + * @return bool True if the value is a MessagePartArrayShape. + * + * @phpstan-assert-if-true MessagePartArrayShape $value + */ + private static function isMessagePartArrayShape($value): bool + { + if (!is_array($value)) { + return false; + } + + // Channel is optional but if present must be a string + if (isset($value['channel']) && !is_string($value['channel'])) { + return false; + } + + // Must have exactly one of the content fields: text, file, functionCall, or functionResponse + // This matches the logic in MessagePart::fromArray() + $contentFields = [ + isset($value['text']), + isset($value['file']), + isset($value['functionCall']), + isset($value['functionResponse']) + ]; + + // Count how many are true - must be exactly 1 + return count(array_filter($contentFields)) === 1; + } +} diff --git a/tests/unit/Common/Utilities/PromptsTest.php b/tests/unit/Common/Utilities/PromptsTest.php new file mode 100644 index 00000000..4356b477 --- /dev/null +++ b/tests/unit/Common/Utilities/PromptsTest.php @@ -0,0 +1,214 @@ +assertCount(1, $result); + $this->assertInstanceOf(UserMessage::class, $result[0]); + + $parts = $result[0]->getParts(); + $this->assertCount(1, $parts); + $this->assertEquals('Hello, world!', $parts[0]->getText()); + } + + /** + * Tests normalizing a single MessagePart to messages. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithMessagePart(): void + { + $part = new MessagePart('Test content'); + $result = Prompts::normalizeToMessages($part); + + $this->assertCount(1, $result); + $this->assertInstanceOf(UserMessage::class, $result[0]); + + $parts = $result[0]->getParts(); + $this->assertCount(1, $parts); + $this->assertEquals('Test content', $parts[0]->getText()); + } + + /** + * Tests normalizing a single Message. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithMessage(): void + { + $message = new UserMessage([new MessagePart('Test message')]); + $result = Prompts::normalizeToMessages($message); + + $this->assertCount(1, $result); + $this->assertSame($message, $result[0]); + } + + /** + * Tests normalizing a MessageArrayShape. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithMessageArrayShape(): void + { + $arrayShape = [ + 'role' => 'user', + 'parts' => [ + [ + 'channel' => 'content', + 'type' => 'text', + 'text' => 'Hello from array' + ] + ] + ]; + + $result = Prompts::normalizeToMessages($arrayShape); + + $this->assertCount(1, $result); + $this->assertInstanceOf(UserMessage::class, $result[0]); + + $parts = $result[0]->getParts(); + $this->assertCount(1, $parts); + $this->assertEquals('Hello from array', $parts[0]->getText()); + } + + /** + * Tests normalizing a list of strings to messages. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithListOfStrings(): void + { + $result = Prompts::normalizeToMessages(['First part', 'Second part', 'Third part']); + + $this->assertCount(1, $result); + $this->assertInstanceOf(UserMessage::class, $result[0]); + + $parts = $result[0]->getParts(); + $this->assertCount(3, $parts); + $this->assertEquals('First part', $parts[0]->getText()); + $this->assertEquals('Second part', $parts[1]->getText()); + $this->assertEquals('Third part', $parts[2]->getText()); + } + + /** + * Tests normalizing a list of MessageParts to messages. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithListOfMessageParts(): void + { + $parts = [ + new MessagePart('Part 1'), + new MessagePart('Part 2') + ]; + + $result = Prompts::normalizeToMessages($parts); + + $this->assertCount(1, $result); + $this->assertInstanceOf(UserMessage::class, $result[0]); + + $messageParts = $result[0]->getParts(); + $this->assertCount(2, $messageParts); + $this->assertEquals('Part 1', $messageParts[0]->getText()); + $this->assertEquals('Part 2', $messageParts[1]->getText()); + } + + /** + * Tests normalizing a mixed list of strings and MessageParts. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithMixedList(): void + { + $items = [ + 'String part', + new MessagePart('MessagePart object'), + [ + 'channel' => 'content', + 'type' => 'text', + 'text' => 'Array shape part' + ] + ]; + + $result = Prompts::normalizeToMessages($items); + + $this->assertCount(1, $result); + $this->assertInstanceOf(UserMessage::class, $result[0]); + + $parts = $result[0]->getParts(); + $this->assertCount(3, $parts); + $this->assertEquals('String part', $parts[0]->getText()); + $this->assertEquals('MessagePart object', $parts[1]->getText()); + $this->assertEquals('Array shape part', $parts[2]->getText()); + } + + /** + * Tests normalizing a list of Messages. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithListOfMessages(): void + { + $messages = [ + new UserMessage([new MessagePart('First message')]), + new UserMessage([new MessagePart('Second message')]) + ]; + + $result = Prompts::normalizeToMessages($messages); + + $this->assertCount(2, $result); + $this->assertSame($messages[0], $result[0]); + $this->assertSame($messages[1], $result[1]); + } + + /** + * Tests that invalid input throws an exception. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithInvalidInput(): void + { + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Invalid item type integer in prompt.'); + + Prompts::normalizeToMessages(123); + } + + /** + * Tests that invalid item in list throws an exception. + * + * @since n.e.x.t + */ + public function testNormalizeToMessagesWithInvalidItemInList(): void + { + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Invalid item type integer in prompt.'); + + Prompts::normalizeToMessages(['Valid string', 123]); + } +} From 73c2a096183664c8a46fcab839d56210f82daa51 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 12:22:13 -0600 Subject: [PATCH 03/47] feat: expands builder message handling --- src/Builders/PromptBuilder.php | 207 ++++++++++---------- src/Common/Utilities/Prompts.php | 6 +- src/Messages/DTO/Message.php | 16 ++ tests/unit/Messages/DTO/MessageTest.php | 28 +++ tests/unit/Messages/DTO/UserMessageTest.php | 20 ++ 5 files changed, 175 insertions(+), 102 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 17a27930..129e16f9 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -5,6 +5,7 @@ namespace WordPress\AiClient\Builders; use InvalidArgumentException; +use WordPress\AiClient\Common\Utilities\Prompts; use WordPress\AiClient\Files\DTO\File; use WordPress\AiClient\Messages\DTO\Message; use WordPress\AiClient\Messages\DTO\MessagePart; @@ -15,8 +16,8 @@ use WordPress\AiClient\Providers\Models\DTO\ModelConfig; use WordPress\AiClient\Providers\Models\DTO\ModelRequirements; use WordPress\AiClient\Providers\Models\DTO\RequiredOption; -use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; use WordPress\AiClient\Providers\Models\Enums\OptionEnum; +use WordPress\AiClient\Providers\ProviderRegistry; use WordPress\AiClient\Tools\DTO\FunctionResponse; /** @@ -27,18 +28,22 @@ * requirements based on the features used in the prompt. * * @since n.e.x.t + * + * @phpstan-import-type MessageArrayShape from Message + * @phpstan-import-type MessagePartArrayShape from MessagePart */ class PromptBuilder { /** - * @var Message[] The messages in the conversation history. + * @var ProviderRegistry The provider registry for finding suitable models. + * @phpstan-ignore-next-line */ - protected array $messages = []; + private ProviderRegistry $registry; /** - * @var MessagePart[] The parts of the current message being built. + * @var list The messages in the conversation. */ - protected array $currentParts = []; + protected array $messages = []; /** * @var ModelInterface|null The model to use for generation. @@ -50,11 +55,6 @@ class PromptBuilder */ protected ModelConfig $modelConfig; - /** - * @var list The inferred required capabilities. - */ - protected array $inferredCapabilities = []; - /** * @var array The inferred required options. */ @@ -65,28 +65,67 @@ class PromptBuilder */ protected ?Message $systemInstruction = null; + // phpcs:disable Generic.Files.LineLength.TooLong /** * Constructor. * * @since n.e.x.t * - * @param string|Message|null $text Optional initial text or message. + * @param ProviderRegistry $registry The provider registry for finding suitable models. + * @param string|MessagePart|Message|MessageArrayShape|list|list|null $prompt + * Optional initial prompt content. */ - public function __construct($text = null) + // phpcs:enable Generic.Files.LineLength.TooLong + public function __construct(ProviderRegistry $registry, $prompt = null) { + $this->registry = $registry; $this->modelConfig = new ModelConfig(); - if ($text !== null) { - if ($text instanceof Message) { - $this->messages[] = $text; - // Infer chat history capability if multiple messages - if (count($this->messages) > 0) { - $this->addInferredCapability(CapabilityEnum::chatHistory()); + if ($prompt === null) { + return; + } + + // Check if it's a single Message - add to messages + if ($prompt instanceof Message) { + $this->messages[] = $prompt; + return; + } + + // Check if it's a list of Messages - set as messages + if (Prompts::isMessagesList($prompt)) { + $this->messages = $prompt; + return; + } + + // Check if it's a MessageArrayShape - add to messages + if (Prompts::isMessageArrayShape($prompt)) { + $this->messages[] = Message::fromArray($prompt); + return; + } + + // Everything else becomes a UserMessage with parts + $parts = []; + + if (is_string($prompt)) { + $parts[] = new MessagePart($prompt); + } elseif ($prompt instanceof MessagePart) { + $parts[] = $prompt; + } elseif (is_array($prompt)) { + // It's a list of strings/MessageParts/MessagePartArrayShapes + foreach ($prompt as $item) { + if (is_string($item)) { + $parts[] = new MessagePart($item); + } elseif ($item instanceof MessagePart) { + $parts[] = $item; + } elseif (Prompts::isMessagePartArrayShape($item)) { + $parts[] = MessagePart::fromArray($item); } - } else { - $this->withText($text); } } + + if (!empty($parts)) { + $this->messages[] = new UserMessage($parts); + } } /** @@ -99,8 +138,8 @@ public function __construct($text = null) */ public function withText(string $text): self { - $this->currentParts[] = new MessagePart($text); - $this->addInferredCapability(CapabilityEnum::textGeneration()); + $part = new MessagePart($text); + $this->appendPartToMessages($part); return $this; } @@ -118,8 +157,8 @@ public function withInlineImage(string $base64Blob, string $mimeType): self // Create data URI format for inline image $dataUri = 'data:' . $mimeType . ';base64,' . $base64Blob; $file = new File($dataUri, $mimeType); - $this->currentParts[] = new MessagePart($file); - $this->addMultimodalInputCapability(); + $part = new MessagePart($file); + $this->appendPartToMessages($part); return $this; } @@ -135,8 +174,8 @@ public function withInlineImage(string $base64Blob, string $mimeType): self public function withRemoteImage(string $uri, string $mimeType): self { $file = new File($uri, $mimeType); - $this->currentParts[] = new MessagePart($file); - $this->addMultimodalInputCapability(); + $part = new MessagePart($file); + $this->appendPartToMessages($part); return $this; } @@ -150,8 +189,8 @@ public function withRemoteImage(string $uri, string $mimeType): self */ public function withImageFile(File $file): self { - $this->currentParts[] = new MessagePart($file); - $this->addMultimodalInputCapability(); + $part = new MessagePart($file); + $this->appendPartToMessages($part); return $this; } @@ -165,8 +204,8 @@ public function withImageFile(File $file): self */ public function withAudioFile(File $file): self { - $this->currentParts[] = new MessagePart($file); - $this->addMultimodalInputCapability('audio'); + $part = new MessagePart($file); + $this->appendPartToMessages($part); return $this; } @@ -180,8 +219,8 @@ public function withAudioFile(File $file): self */ public function withVideoFile(File $file): self { - $this->currentParts[] = new MessagePart($file); - $this->addMultimodalInputCapability('video'); + $part = new MessagePart($file); + $this->appendPartToMessages($part); return $this; } @@ -195,7 +234,8 @@ public function withVideoFile(File $file): self */ public function withFunctionResponse(FunctionResponse $functionResponse): self { - $this->currentParts[] = new MessagePart($functionResponse); + $part = new MessagePart($functionResponse); + $this->appendPartToMessages($part); return $this; } @@ -210,11 +250,7 @@ public function withFunctionResponse(FunctionResponse $functionResponse): self public function withMessageParts(MessagePart ...$parts): self { foreach ($parts as $part) { - $this->currentParts[] = $part; - // Infer capabilities based on part type - if ($part->getFile() !== null) { - $this->addMultimodalInputCapability(); - } + $this->appendPartToMessages($part); } return $this; } @@ -229,17 +265,10 @@ public function withMessageParts(MessagePart ...$parts): self */ public function withHistory(Message ...$messages): self { - // Save current parts as a user message if any exist - if (!empty($this->currentParts)) { - $this->messages[] = new UserMessage($this->currentParts); - $this->currentParts = []; - } - foreach ($messages as $message) { $this->messages[] = $message; } - $this->addInferredCapability(CapabilityEnum::chatHistory()); return $this; } @@ -257,6 +286,20 @@ public function usingModel(ModelInterface $model): self return $this; } + /** + * Sets a different provider registry. + * + * @since n.e.x.t + * + * @param ProviderRegistry $registry The provider registry to use. + * @return self + */ + public function usingRegistry(ProviderRegistry $registry): self + { + $this->registry = $registry; + return $this; + } + /** * Sets the system instruction. * @@ -461,8 +504,9 @@ public function getModelRequirements(): ModelRequirements $requiredOptions[] = new RequiredOption($name, $value); } + // TODO: Derive capabilities from messages and parts return new ModelRequirements( - $this->inferredCapabilities, + [], $requiredOptions ); } @@ -496,7 +540,6 @@ public function isSupported(): bool public function generateText(): string { $this->validateModel(); - $this->prepareMessages(); // This is a placeholder - actual implementation would call the model throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); @@ -518,50 +561,35 @@ public function generateTexts(?int $candidateCount = null): array } $this->validateModel(); - $this->prepareMessages(); // This is a placeholder - actual implementation would call the model throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); } /** - * Adds an inferred capability. + * Appends a MessagePart to the messages array. * - * @since n.e.x.t - * - * @param CapabilityEnum $capability The capability to add. - * @return void - */ - protected function addInferredCapability(CapabilityEnum $capability): void - { - if (!in_array($capability, $this->inferredCapabilities, true)) { - $this->inferredCapabilities[] = $capability; - } - } - - /** - * Adds multimodal input capability based on content type. + * If the last message has a user role, the part is added to it. + * Otherwise, a new UserMessage is created with the part. * * @since n.e.x.t * - * @param string $type The content type (image, audio, video). + * @param MessagePart $part The part to append. * @return void */ - protected function addMultimodalInputCapability(string $type = 'image'): void + protected function appendPartToMessages(MessagePart $part): void { - $this->addInferredCapability(CapabilityEnum::textGeneration()); - - // Add input modalities requirement - $modalityKey = OptionEnum::inputModalities()->value; - $currentModalities = isset($this->inferredOptions[$modalityKey]) - && is_array($this->inferredOptions[$modalityKey]) - ? $this->inferredOptions[$modalityKey] - : ['text']; - - if (!in_array($type, $currentModalities, true)) { - $currentModalities[] = $type; - $this->inferredOptions[$modalityKey] = $currentModalities; + $lastMessage = end($this->messages); + + if ($lastMessage instanceof Message && $lastMessage->getRole()->isUser()) { + // Replace the last message with a new one containing the appended part + array_pop($this->messages); + $this->messages[] = $lastMessage->withPart($part); + return; } + + // Create new UserMessage with the part + $this->messages[] = new UserMessage([$part]); } /** @@ -575,7 +603,9 @@ protected function addMultimodalInputCapability(string $type = 'image'): void protected function validateModel(): void { if ($this->model === null) { - // Model selection would happen here via registry + // TODO: Use $this->registry to find a suitable model based on requirements + // $requirements = $this->getModelRequirements(); + // $this->model = $this->registry->findModel($requirements); return; } @@ -589,25 +619,4 @@ protected function validateModel(): void ); } } - - /** - * Prepares the final messages array for sending to the model. - * - * @since n.e.x.t - * - * @return void - */ - protected function prepareMessages(): void - { - // Add current parts as a user message if any exist - if (!empty($this->currentParts)) { - $this->messages[] = new UserMessage($this->currentParts); - $this->currentParts = []; - } - - // Add system instruction if present - if ($this->systemInstruction !== null) { - array_unshift($this->messages, $this->systemInstruction); - } - } } diff --git a/src/Common/Utilities/Prompts.php b/src/Common/Utilities/Prompts.php index f834d2ac..18807b34 100644 --- a/src/Common/Utilities/Prompts.php +++ b/src/Common/Utilities/Prompts.php @@ -88,7 +88,7 @@ public static function normalizeToMessages($prompt): array * * @phpstan-assert-if-true list $value */ - private static function isMessagesList($value): bool + public static function isMessagesList($value): bool { if (!is_array($value) || empty($value) || !array_is_list($value)) { return false; @@ -114,7 +114,7 @@ private static function isMessagesList($value): bool * * @phpstan-assert-if-true MessageArrayShape $value */ - private static function isMessageArrayShape($value): bool + public static function isMessageArrayShape($value): bool { if (!is_array($value)) { return false; @@ -148,7 +148,7 @@ private static function isMessageArrayShape($value): bool * * @phpstan-assert-if-true MessagePartArrayShape $value */ - private static function isMessagePartArrayShape($value): bool + public static function isMessagePartArrayShape($value): bool { if (!is_array($value)) { return false; diff --git a/src/Messages/DTO/Message.php b/src/Messages/DTO/Message.php index d68b0b94..240608cf 100644 --- a/src/Messages/DTO/Message.php +++ b/src/Messages/DTO/Message.php @@ -76,6 +76,22 @@ public function getParts(): array return $this->parts; } + /** + * Returns a new instance with the given part appended. + * + * @since n.e.x.t + * + * @param MessagePart $part The part to append. + * @return Message A new instance with the part appended. + */ + public function withPart(MessagePart $part): Message + { + $newParts = $this->parts; + $newParts[] = $part; + + return new Message($this->role, $newParts); + } + /** * {@inheritDoc} * diff --git a/tests/unit/Messages/DTO/MessageTest.php b/tests/unit/Messages/DTO/MessageTest.php index c5a26105..b4d1116a 100644 --- a/tests/unit/Messages/DTO/MessageTest.php +++ b/tests/unit/Messages/DTO/MessageTest.php @@ -319,4 +319,32 @@ public function testImplementsWithArrayTransformationInterface(): void $message ); } + + /** + * Tests that withPart creates a new instance with the part appended. + * + * @since n.e.x.t + */ + public function testWithPartCreatesNewInstance(): void + { + $original = new Message( + MessageRoleEnum::user(), + [new MessagePart('Original text')] + ); + + $newPart = new MessagePart('Additional text'); + $updated = $original->withPart($newPart); + + // Assert that a new instance was created + $this->assertNotSame($original, $updated); + + // Assert original is unchanged + $this->assertCount(1, $original->getParts()); + $this->assertEquals('Original text', $original->getParts()[0]->getText()); + + // Assert updated has both parts + $this->assertCount(2, $updated->getParts()); + $this->assertEquals('Original text', $updated->getParts()[0]->getText()); + $this->assertEquals('Additional text', $updated->getParts()[1]->getText()); + } } diff --git a/tests/unit/Messages/DTO/UserMessageTest.php b/tests/unit/Messages/DTO/UserMessageTest.php index cc811243..16428e9d 100644 --- a/tests/unit/Messages/DTO/UserMessageTest.php +++ b/tests/unit/Messages/DTO/UserMessageTest.php @@ -6,6 +6,7 @@ use PHPUnit\Framework\TestCase; use WordPress\AiClient\Files\DTO\File; +use WordPress\AiClient\Messages\DTO\Message; use WordPress\AiClient\Messages\DTO\MessagePart; use WordPress\AiClient\Messages\DTO\UserMessage; use WordPress\AiClient\Messages\Enums\MessagePartTypeEnum; @@ -310,4 +311,23 @@ public function testImplementsWithArrayTransformationInterface(): void $message = new UserMessage([new MessagePart('test')]); $this->assertImplementsArrayTransformation($message); } + + /** + * Tests that withPart returns a base Message with user role. + * + * @since n.e.x.t + */ + public function testWithPartReturnsBaseMessage(): void + { + $original = new UserMessage([new MessagePart('User text')]); + $updated = $original->withPart(new MessagePart('More text')); + + $this->assertInstanceOf(Message::class, $updated); + $this->assertNotInstanceOf(UserMessage::class, $updated); + $this->assertNotSame($original, $updated); + $this->assertCount(2, $updated->getParts()); + $this->assertEquals(MessageRoleEnum::user(), $updated->getRole()); + $this->assertEquals('User text', $updated->getParts()[0]->getText()); + $this->assertEquals('More text', $updated->getParts()[1]->getText()); + } } From c91e520b1791f2539d205d125d79bb230a506ddd Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 13:57:52 -0600 Subject: [PATCH 04/47] feat: improves model requirement system --- src/Builders/PromptBuilder.php | 147 ++++++++++++++++++++++++++++++--- 1 file changed, 137 insertions(+), 10 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 129e16f9..ce4d2829 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -16,6 +16,7 @@ use WordPress\AiClient\Providers\Models\DTO\ModelConfig; use WordPress\AiClient\Providers\Models\DTO\ModelRequirements; use WordPress\AiClient\Providers\Models\DTO\RequiredOption; +use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; use WordPress\AiClient\Providers\Models\Enums\OptionEnum; use WordPress\AiClient\Providers\ProviderRegistry; use WordPress\AiClient\Tools\DTO\FunctionResponse; @@ -36,7 +37,6 @@ class PromptBuilder { /** * @var ProviderRegistry The provider registry for finding suitable models. - * @phpstan-ignore-next-line */ private ProviderRegistry $registry; @@ -499,14 +499,72 @@ public function asJsonResponse(?array $schema = null): self */ public function getModelRequirements(): ModelRequirements { + $capabilities = []; + $inputModalities = []; + + // Always need text generation capability + $capabilities[] = CapabilityEnum::textGeneration(); + + // Check if we have chat history (multiple messages) + if (count($this->messages) > 1) { + $capabilities[] = CapabilityEnum::chatHistory(); + } + + // Analyze all messages to determine required input modalities + foreach ($this->messages as $message) { + foreach ($message->getParts() as $part) { + // Check for text input + if ($part->getText() !== null) { + $inputModalities[ModalityEnum::text()->value] = ModalityEnum::text(); + } + + // Check for file inputs + if ($part->getFile() !== null) { + $mimeType = $part->getFile()->getMimeType(); + + // Determine modality based on MIME type + if (strpos($mimeType, 'image/') === 0) { + $inputModalities[ModalityEnum::image()->value] = ModalityEnum::image(); + } elseif (strpos($mimeType, 'audio/') === 0) { + $inputModalities[ModalityEnum::audio()->value] = ModalityEnum::audio(); + } elseif (strpos($mimeType, 'video/') === 0) { + $inputModalities[ModalityEnum::video()->value] = ModalityEnum::video(); + } elseif ( + strpos($mimeType, 'application/pdf') === 0 || + strpos($mimeType, 'application/msword') === 0 || + strpos($mimeType, 'application/vnd.openxmlformats-officedocument') === 0 || + strpos($mimeType, 'text/plain') === 0 + ) { + $inputModalities[ModalityEnum::document()->value] = ModalityEnum::document(); + } + } + + // Check for function calls/responses (these might require special capabilities) + if ($part->getFunctionCall() !== null || $part->getFunctionResponse() !== null) { + // Function calling capability would go here if we had it in CapabilityEnum + // For now, we'll just note this requires text generation + } + } + } + + // Build required options $requiredOptions = []; + + // Add input modalities if we have non-text inputs + if (count($inputModalities) > 0) { + $requiredOptions[] = new RequiredOption( + OptionEnum::inputModalities()->value, + array_values($inputModalities) + ); + } + + // Add other inferred options foreach ($this->inferredOptions as $name => $value) { $requiredOptions[] = new RequiredOption($name, $value); } - // TODO: Derive capabilities from messages and parts return new ModelRequirements( - [], + $capabilities, $requiredOptions ); } @@ -535,10 +593,11 @@ public function isSupported(): bool * @since n.e.x.t * * @return string The generated text. - * @throws InvalidArgumentException If the selected model doesn't meet requirements. + * @throws InvalidArgumentException If the prompt or model validation fails. */ public function generateText(): string { + $this->validateMessages(); $this->validateModel(); // This is a placeholder - actual implementation would call the model @@ -552,7 +611,7 @@ public function generateText(): string * * @param int|null $candidateCount The number of candidates to generate. * @return list The generated texts. - * @throws InvalidArgumentException If the selected model doesn't meet requirements. + * @throws InvalidArgumentException If the prompt or model validation fails. */ public function generateTexts(?int $candidateCount = null): array { @@ -560,6 +619,7 @@ public function generateTexts(?int $candidateCount = null): array $this->usingCandidateCount($candidateCount); } + $this->validateMessages(); $this->validateModel(); // This is a placeholder - actual implementation would call the model @@ -598,18 +658,43 @@ protected function appendPartToMessages(MessagePart $part): void * @since n.e.x.t * * @return void - * @throws InvalidArgumentException If model doesn't meet requirements. + * @throws InvalidArgumentException If model doesn't meet requirements or no suitable model found. */ protected function validateModel(): void { + $requirements = $this->getModelRequirements(); + + // If no model is specified, find one that meets requirements if ($this->model === null) { - // TODO: Use $this->registry to find a suitable model based on requirements - // $requirements = $this->getModelRequirements(); - // $this->model = $this->registry->findModel($requirements); + $modelsMetadata = $this->registry->findModelsMetadataForSupport($requirements); + + if (empty($modelsMetadata)) { + throw new InvalidArgumentException( + 'No models found that support the required capabilities and options for this prompt. ' . + 'Required capabilities: ' . implode(', ', array_map(function ($cap) { + return $cap->value; + }, $requirements->getRequiredCapabilities())) . + '. Required options: ' . implode(', ', array_map(function ($opt) { + return $opt->getName() . '=' . json_encode($opt->getValue()); + }, $requirements->getRequiredOptions())) + ); + } + + // Get the first available model from the first provider + $firstProviderModels = $modelsMetadata[0]; + $firstModelMetadata = $firstProviderModels->getModels()[0]; + + // Get the model instance from the provider + $this->model = $this->registry->getProviderModel( + $firstProviderModels->getProvider()->getId(), + $firstModelMetadata->getId(), + $this->modelConfig + ); + return; } - $requirements = $this->getModelRequirements(); + // Validate existing model meets requirements if (!$this->model->metadata()->meetsRequirements($requirements)) { throw new InvalidArgumentException( sprintf( @@ -619,4 +704,46 @@ protected function validateModel(): void ); } } + + /** + * Validates the messages array for prompt generation. + * + * Ensures that: + * - The first message is a user or system message + * - The last message is a user message + * - The last message has parts + * + * @since n.e.x.t + * + * @return void + * @throws InvalidArgumentException If validation fails. + */ + private function validateMessages(): void + { + if (empty($this->messages)) { + throw new InvalidArgumentException( + 'Cannot generate from an empty prompt. Add content using withText() or similar methods.' + ); + } + + $firstMessage = reset($this->messages); + if (!$firstMessage->getRole()->isUser() && !$firstMessage->getRole()->isSystem()) { + throw new InvalidArgumentException( + 'The first message must be from a user or system role, not from ' . $firstMessage->getRole()->value + ); + } + + $lastMessage = end($this->messages); + if (!$lastMessage->getRole()->isUser()) { + throw new InvalidArgumentException( + 'The last message must be from a user role, not from ' . $lastMessage->getRole()->value + ); + } + + if (empty($lastMessage->getParts())) { + throw new InvalidArgumentException( + 'The last message must have content parts. Add content using withText() or similar methods.' + ); + } + } } From 765460829ca1be7934ae1e5c1f878a087f916639 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 15:45:16 -0600 Subject: [PATCH 05/47] feat: adds DTO::isArrayShape --- src/Builders/PromptBuilder.php | 4 +- src/Common/AbstractDataTransferObject.php | 18 ++++- .../WithArrayTransformationInterface.php | 11 +++ src/Common/Utilities/Prompts.php | 72 +------------------ 4 files changed, 32 insertions(+), 73 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index ce4d2829..bf1c875d 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -98,7 +98,7 @@ public function __construct(ProviderRegistry $registry, $prompt = null) } // Check if it's a MessageArrayShape - add to messages - if (Prompts::isMessageArrayShape($prompt)) { + if (is_array($prompt) && Message::isArrayShape($prompt)) { $this->messages[] = Message::fromArray($prompt); return; } @@ -117,7 +117,7 @@ public function __construct(ProviderRegistry $registry, $prompt = null) $parts[] = new MessagePart($item); } elseif ($item instanceof MessagePart) { $parts[] = $item; - } elseif (Prompts::isMessagePartArrayShape($item)) { + } elseif (is_array($item) && MessagePart::isArrayShape($item)) { $parts[] = MessagePart::fromArray($item); } } diff --git a/src/Common/AbstractDataTransferObject.php b/src/Common/AbstractDataTransferObject.php index 302506dc..11baddd4 100644 --- a/src/Common/AbstractDataTransferObject.php +++ b/src/Common/AbstractDataTransferObject.php @@ -37,7 +37,7 @@ abstract class AbstractDataTransferObject implements * * @since n.e.x.t * - * @param TArrayShape $data The array data to validate. + * @param array $data The array data to validate. * @param string[] $requiredKeys The keys that must be present. * @throws InvalidArgumentException If any required key is missing. */ @@ -62,6 +62,22 @@ protected static function validateFromArrayData(array $data, array $requiredKeys } } + /** + * {@inheritDoc} + * + * @since n.e.x.t + */ + public static function isArrayShape(array $array): bool + { + try { + /** @var TArrayShape $array */ + static::fromArray($array); + return true; + } catch (InvalidArgumentException $e) { + return false; + } + } + /** * Converts the object to a JSON-serializable format. * diff --git a/src/Common/Contracts/WithArrayTransformationInterface.php b/src/Common/Contracts/WithArrayTransformationInterface.php index 257647f0..5d158e1b 100644 --- a/src/Common/Contracts/WithArrayTransformationInterface.php +++ b/src/Common/Contracts/WithArrayTransformationInterface.php @@ -31,4 +31,15 @@ public function toArray(): array; * @return self The created instance. */ public static function fromArray(array $array): self; + + /** + * Checks if the array is a valid shape for this object. + * + * @since n.e.x.t + * + * @param array $array The array to check. + * @return bool True if the array is a valid shape. + * @phpstan-assert-if-true TArrayShape $array + */ + public static function isArrayShape(array $array): bool; } diff --git a/src/Common/Utilities/Prompts.php b/src/Common/Utilities/Prompts.php index 18807b34..ae46eb35 100644 --- a/src/Common/Utilities/Prompts.php +++ b/src/Common/Utilities/Prompts.php @@ -48,7 +48,7 @@ public static function normalizeToMessages($prompt): array } // 3. Check if it's a MessageArrayShape (single message as array) - if (self::isMessageArrayShape($prompt)) { + if (is_array($prompt) && Message::isArrayShape($prompt)) { return [Message::fromArray($prompt)]; } @@ -65,7 +65,7 @@ public static function normalizeToMessages($prompt): array $parts[] = new MessagePart($item); } elseif ($item instanceof MessagePart) { $parts[] = $item; - } elseif (self::isMessagePartArrayShape($item)) { + } elseif (is_array($item) && MessagePart::isArrayShape($item)) { $parts[] = MessagePart::fromArray($item); } else { $type = is_object($item) ? get_class($item) : gettype($item); @@ -103,72 +103,4 @@ public static function isMessagesList($value): bool return true; } - - /** - * Checks if the value is a MessageArrayShape. - * - * @since n.e.x.t - * - * @param mixed $value The value to check. - * @return bool True if the value is a MessageArrayShape. - * - * @phpstan-assert-if-true MessageArrayShape $value - */ - public static function isMessageArrayShape($value): bool - { - if (!is_array($value)) { - return false; - } - - // Must have required keys - if (!isset($value['role']) || !isset($value['parts'])) { - return false; - } - - // Role must be a string - if (!is_string($value['role'])) { - return false; - } - - // Parts must be an array - if (!is_array($value['parts'])) { - return false; - } - - return true; - } - - /** - * Checks if the value is a MessagePartArrayShape. - * - * @since n.e.x.t - * - * @param mixed $value The value to check. - * @return bool True if the value is a MessagePartArrayShape. - * - * @phpstan-assert-if-true MessagePartArrayShape $value - */ - public static function isMessagePartArrayShape($value): bool - { - if (!is_array($value)) { - return false; - } - - // Channel is optional but if present must be a string - if (isset($value['channel']) && !is_string($value['channel'])) { - return false; - } - - // Must have exactly one of the content fields: text, file, functionCall, or functionResponse - // This matches the logic in MessagePart::fromArray() - $contentFields = [ - isset($value['text']), - isset($value['file']), - isset($value['functionCall']), - isset($value['functionResponse']) - ]; - - // Count how many are true - must be exactly 1 - return count(array_filter($contentFields)) === 1; - } } From c996c451dde7aea51b538ce9954c1433f16b1d1a Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 16:04:22 -0600 Subject: [PATCH 06/47] test: adds isArrayShape tests --- tests/traits/ArrayTransformationTestTrait.php | 30 ++++++++++++ tests/unit/Messages/DTO/MessageTest.php | 47 +++++++++++++++++++ 2 files changed, 77 insertions(+) diff --git a/tests/traits/ArrayTransformationTestTrait.php b/tests/traits/ArrayTransformationTestTrait.php index 09f175ba..8165232f 100644 --- a/tests/traits/ArrayTransformationTestTrait.php +++ b/tests/traits/ArrayTransformationTestTrait.php @@ -83,4 +83,34 @@ protected function assertArrayNotHasKeys(array $array, array $unexpectedKeys): v $this->assertArrayNotHasKey($key, $array, "Array should not contain key: {$key}"); } } + + /** + * Tests isArrayShape with valid and invalid arrays. + * + * @param string $className The class name to test. + * @param array $validArray A valid array that should pass isArrayShape. + * @param array[] $invalidArrays Arrays that should fail isArrayShape. + * @return void + */ + protected function assertIsArrayShapeValidation(string $className, array $validArray, array $invalidArrays): void + { + // Test valid array + $this->assertTrue( + $className::isArrayShape($validArray), + 'isArrayShape() should return true for valid array structure' + ); + + // Test that fromArray works with the valid array (ensures consistency) + $instance = $className::fromArray($validArray); + $this->assertInstanceOf($className, $instance); + + // Test invalid arrays + foreach ($invalidArrays as $description => $invalidArray) { + $this->assertFalse( + $className::isArrayShape($invalidArray), + "isArrayShape() should return false for: {$description}" + ); + } + } + } diff --git a/tests/unit/Messages/DTO/MessageTest.php b/tests/unit/Messages/DTO/MessageTest.php index b4d1116a..32da9277 100644 --- a/tests/unit/Messages/DTO/MessageTest.php +++ b/tests/unit/Messages/DTO/MessageTest.php @@ -10,6 +10,7 @@ use WordPress\AiClient\Messages\DTO\MessagePart; use WordPress\AiClient\Messages\Enums\MessagePartTypeEnum; use WordPress\AiClient\Messages\Enums\MessageRoleEnum; +use WordPress\AiClient\Tests\traits\ArrayTransformationTestTrait; use WordPress\AiClient\Tools\DTO\FunctionCall; use WordPress\AiClient\Tools\DTO\FunctionResponse; @@ -18,6 +19,7 @@ */ class MessageTest extends TestCase { + use ArrayTransformationTestTrait; /** * Tests creating Message with single text part. * @@ -186,6 +188,51 @@ public function testMessageWithManyParts(): void $this->assertEquals('Part number 99', $message->getParts()[99]->getText()); } + /** + * Tests isArrayShape validation. + * + * @return void + */ + public function testIsArrayShapeValidation(): void + { + $validArray = [ + Message::KEY_ROLE => MessageRoleEnum::user()->value, + Message::KEY_PARTS => [ + [ + MessagePart::KEY_TYPE => MessagePartTypeEnum::text()->value, + MessagePart::KEY_TEXT => 'Test message' + ] + ] + ]; + + $invalidArrays = [ + 'missing role' => [ + Message::KEY_PARTS => [ + [ + MessagePart::KEY_TYPE => MessagePartTypeEnum::text()->value, + MessagePart::KEY_TEXT => 'Test message' + ] + ] + ], + 'missing parts' => [ + Message::KEY_ROLE => MessageRoleEnum::user()->value + ], + 'invalid role value' => [ + Message::KEY_ROLE => 'invalid_role', + Message::KEY_PARTS => [ + [ + MessagePart::KEY_TYPE => MessagePartTypeEnum::text()->value, + MessagePart::KEY_TEXT => 'Test message' + ] + ] + ], + 'empty array' => [], + 'non-associative array' => ['user', 'parts'] + ]; + + $this->assertIsArrayShapeValidation(Message::class, $validArray, $invalidArrays); + } + /** * Tests preserving part order. * From 8f390d6af4b16d4874817874faf958a8806d2695 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 16:26:06 -0600 Subject: [PATCH 07/47] test: adds some isArrayShape tests --- tests/traits/ArrayTransformationTestTrait.php | 1 - tests/unit/Messages/DTO/MessageTest.php | 1 + 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/traits/ArrayTransformationTestTrait.php b/tests/traits/ArrayTransformationTestTrait.php index 8165232f..d670c0f2 100644 --- a/tests/traits/ArrayTransformationTestTrait.php +++ b/tests/traits/ArrayTransformationTestTrait.php @@ -112,5 +112,4 @@ protected function assertIsArrayShapeValidation(string $className, array $validA ); } } - } diff --git a/tests/unit/Messages/DTO/MessageTest.php b/tests/unit/Messages/DTO/MessageTest.php index 32da9277..0edc5f85 100644 --- a/tests/unit/Messages/DTO/MessageTest.php +++ b/tests/unit/Messages/DTO/MessageTest.php @@ -20,6 +20,7 @@ class MessageTest extends TestCase { use ArrayTransformationTestTrait; + /** * Tests creating Message with single text part. * From e8ac1c286ef008742b59380306b1d58cda13ff4e Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 16:26:21 -0600 Subject: [PATCH 08/47] refactor: cleans up file type checking --- src/Builders/PromptBuilder.php | 15 +++++---------- 1 file changed, 5 insertions(+), 10 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index bf1c875d..5d51a90d 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -520,21 +520,16 @@ public function getModelRequirements(): ModelRequirements // Check for file inputs if ($part->getFile() !== null) { - $mimeType = $part->getFile()->getMimeType(); + $mimeType = $part->getFile()->getMimeTypeObject(); // Determine modality based on MIME type - if (strpos($mimeType, 'image/') === 0) { + if ($mimeType->isImage()) { $inputModalities[ModalityEnum::image()->value] = ModalityEnum::image(); - } elseif (strpos($mimeType, 'audio/') === 0) { + } elseif ($mimeType->isAudio()) { $inputModalities[ModalityEnum::audio()->value] = ModalityEnum::audio(); - } elseif (strpos($mimeType, 'video/') === 0) { + } elseif ($mimeType->isVideo()) { $inputModalities[ModalityEnum::video()->value] = ModalityEnum::video(); - } elseif ( - strpos($mimeType, 'application/pdf') === 0 || - strpos($mimeType, 'application/msword') === 0 || - strpos($mimeType, 'application/vnd.openxmlformats-officedocument') === 0 || - strpos($mimeType, 'text/plain') === 0 - ) { + } elseif ($mimeType->isDocument() || $mimeType->isText()) { $inputModalities[ModalityEnum::document()->value] = ModalityEnum::document(); } } From 9638c412dbe91bb2fdf7653f073d1a7e64a9adc2 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 16:31:47 -0600 Subject: [PATCH 09/47] feat: adds type checking methods to File DTO --- src/Files/DTO/File.php | 12 ++++ tests/unit/Files/DTO/FileTest.php | 115 ++++++++++++++++++++++++++++++ 2 files changed, 127 insertions(+) diff --git a/src/Files/DTO/File.php b/src/Files/DTO/File.php index 4a9ad809..e98cf935 100644 --- a/src/Files/DTO/File.php +++ b/src/Files/DTO/File.php @@ -286,6 +286,18 @@ public function isText(): bool return $this->mimeType->isText(); } + /** + * Checks if the file is a document. + * + * @since n.e.x.t + * + * @return bool True if the file is a document. + */ + public function isDocument(): bool + { + return $this->mimeType->isDocument(); + } + /** * Checks if the file is a specific MIME type. * diff --git a/tests/unit/Files/DTO/FileTest.php b/tests/unit/Files/DTO/FileTest.php index 9ed4e6d7..52a9730f 100644 --- a/tests/unit/Files/DTO/FileTest.php +++ b/tests/unit/Files/DTO/FileTest.php @@ -405,4 +405,119 @@ public function testImplementsWithArrayTransformationInterface(): void $file ); } + + /** + * Tests isImage passthrough method. + * + * @return void + */ + public function testIsImage(): void + { + $imageFile = new File('https://example.com/test.jpg'); + $this->assertTrue($imageFile->isImage()); + + $pngBase64 = 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNkYPhfDwAChwGA60e6kgAAAABJRU5ErkJggg=='; + $pngFile = new File('data:image/png;base64,' . $pngBase64); + $this->assertTrue($pngFile->isImage()); + + $textFile = new File('https://example.com/test.txt', 'text/plain'); + $this->assertFalse($textFile->isImage()); + } + + /** + * Tests isVideo passthrough method. + * + * @return void + */ + public function testIsVideo(): void + { + $videoFile = new File('https://example.com/test.mp4'); + $this->assertTrue($videoFile->isVideo()); + + $aviFile = new File('https://example.com/test.avi'); + $this->assertTrue($aviFile->isVideo()); + + $imageFile = new File('https://example.com/test.jpg'); + $this->assertFalse($imageFile->isVideo()); + } + + /** + * Tests isAudio passthrough method. + * + * @return void + */ + public function testIsAudio(): void + { + $audioFile = new File('https://example.com/test.mp3'); + $this->assertTrue($audioFile->isAudio()); + + $wavFile = new File('https://example.com/test.wav'); + $this->assertTrue($wavFile->isAudio()); + + $imageFile = new File('https://example.com/test.jpg'); + $this->assertFalse($imageFile->isAudio()); + } + + /** + * Tests isText passthrough method. + * + * @return void + */ + public function testIsText(): void + { + $textFile = new File('https://example.com/test.txt'); + $this->assertTrue($textFile->isText()); + + $csvFile = new File('https://example.com/test.csv'); + $this->assertTrue($csvFile->isText()); + + $htmlFile = new File('https://example.com/test.html'); + $this->assertTrue($htmlFile->isText()); + + $imageFile = new File('https://example.com/test.jpg'); + $this->assertFalse($imageFile->isText()); + } + + /** + * Tests isDocument passthrough method. + * + * @return void + */ + public function testIsDocument(): void + { + $pdfFile = new File('https://example.com/test.pdf'); + $this->assertTrue($pdfFile->isDocument()); + + $docFile = new File('https://example.com/test.doc'); + $this->assertTrue($docFile->isDocument()); + + $docxFile = new File('https://example.com/test.docx'); + $this->assertTrue($docxFile->isDocument()); + + $imageFile = new File('https://example.com/test.jpg'); + $this->assertFalse($imageFile->isDocument()); + + $audioFile = new File('https://example.com/test.mp3'); + $this->assertFalse($audioFile->isDocument()); + } + + /** + * Tests isMimeType passthrough method. + * + * @return void + */ + public function testIsMimeType(): void + { + $imageFile = new File('https://example.com/test.jpg'); + $this->assertTrue($imageFile->isMimeType('image')); + $this->assertFalse($imageFile->isMimeType('video')); + + $videoFile = new File('https://example.com/test.mp4'); + $this->assertTrue($videoFile->isMimeType('video')); + $this->assertFalse($videoFile->isMimeType('audio')); + + $textFile = new File('https://example.com/test.txt'); + $this->assertTrue($textFile->isMimeType('text')); + $this->assertFalse($textFile->isMimeType('image')); + } } From 0da377edb084d54207ac982637f1f6a07384b135 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 16:31:57 -0600 Subject: [PATCH 10/47] refactor: uses new file type checking --- src/Builders/PromptBuilder.php | 14 ++++++-------- 1 file changed, 6 insertions(+), 8 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 5d51a90d..cf71c183 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -519,17 +519,15 @@ public function getModelRequirements(): ModelRequirements } // Check for file inputs - if ($part->getFile() !== null) { - $mimeType = $part->getFile()->getMimeTypeObject(); - - // Determine modality based on MIME type - if ($mimeType->isImage()) { + $file = $part->getFile(); + if ($file !== null) { + if ($file->isImage()) { $inputModalities[ModalityEnum::image()->value] = ModalityEnum::image(); - } elseif ($mimeType->isAudio()) { + } elseif ($file->isAudio()) { $inputModalities[ModalityEnum::audio()->value] = ModalityEnum::audio(); - } elseif ($mimeType->isVideo()) { + } elseif ($file->isVideo()) { $inputModalities[ModalityEnum::video()->value] = ModalityEnum::video(); - } elseif ($mimeType->isDocument() || $mimeType->isText()) { + } elseif ($file->isDocument() || $file->isText()) { $inputModalities[ModalityEnum::document()->value] = ModalityEnum::document(); } } From eb9f197b0a75aa96f350e80ff57148c4e431e8f7 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 17:04:03 -0600 Subject: [PATCH 11/47] refactor: removes inferred property in favor of late checking --- src/Builders/PromptBuilder.php | 23 +++++++++++++---------- 1 file changed, 13 insertions(+), 10 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index cf71c183..67629b33 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -55,11 +55,6 @@ class PromptBuilder */ protected ModelConfig $modelConfig; - /** - * @var array The inferred required options. - */ - protected array $inferredOptions = []; - /** * @var Message|null The system instruction message. */ @@ -440,7 +435,6 @@ public function usingCandidateCount(int $candidateCount): self public function usingOutputMime(string $mimeType): self { $this->modelConfig->setOutputMimeType($mimeType); - $this->inferredOptions[OptionEnum::outputMimeType()->value] = $mimeType; return $this; } @@ -455,7 +449,6 @@ public function usingOutputMime(string $mimeType): self public function usingOutputSchema(array $schema): self { $this->modelConfig->setOutputSchema($schema); - $this->inferredOptions[OptionEnum::outputSchema()->value] = true; return $this; } @@ -551,9 +544,19 @@ public function getModelRequirements(): ModelRequirements ); } - // Add other inferred options - foreach ($this->inferredOptions as $name => $value) { - $requiredOptions[] = new RequiredOption($name, $value); + // Check ModelConfig for output requirements + if ($this->modelConfig->getOutputMimeType() !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::outputMimeType()->value, + $this->modelConfig->getOutputMimeType() + ); + } + + if ($this->modelConfig->getOutputSchema() !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::outputSchema()->value, + true + ); } return new ModelRequirements( From 88242f6b340c1f2ff77c8016c7a0c4f0177e8f7d Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 17:13:07 -0600 Subject: [PATCH 12/47] feat: adds means of converting ModelConfig to requirements --- src/Builders/PromptBuilder.php | 19 +--- src/Providers/Models/DTO/ModelConfig.php | 126 +++++++++++++++++++++++ 2 files changed, 128 insertions(+), 17 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 67629b33..dea46008 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -533,8 +533,8 @@ public function getModelRequirements(): ModelRequirements } } - // Build required options - $requiredOptions = []; + // Build required options from ModelConfig + $requiredOptions = $this->modelConfig->toRequiredOptions(); // Add input modalities if we have non-text inputs if (count($inputModalities) > 0) { @@ -544,21 +544,6 @@ public function getModelRequirements(): ModelRequirements ); } - // Check ModelConfig for output requirements - if ($this->modelConfig->getOutputMimeType() !== null) { - $requiredOptions[] = new RequiredOption( - OptionEnum::outputMimeType()->value, - $this->modelConfig->getOutputMimeType() - ); - } - - if ($this->modelConfig->getOutputSchema() !== null) { - $requiredOptions[] = new RequiredOption( - OptionEnum::outputSchema()->value, - true - ); - } - return new ModelRequirements( $capabilities, $requiredOptions diff --git a/src/Providers/Models/DTO/ModelConfig.php b/src/Providers/Models/DTO/ModelConfig.php index e3504232..52a23192 100644 --- a/src/Providers/Models/DTO/ModelConfig.php +++ b/src/Providers/Models/DTO/ModelConfig.php @@ -9,6 +9,7 @@ use WordPress\AiClient\Files\Enums\FileTypeEnum; use WordPress\AiClient\Files\Enums\MediaOrientationEnum; use WordPress\AiClient\Messages\Enums\ModalityEnum; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; use WordPress\AiClient\Tools\DTO\FunctionDeclaration; use WordPress\AiClient\Tools\DTO\WebSearch; @@ -913,6 +914,131 @@ static function (FunctionDeclaration $function_declaration): array { return $data; } + /** + * Converts the model configuration to required options. + * + * @since n.e.x.t + * + * @return list The required options. + */ + public function toRequiredOptions(): array + { + $requiredOptions = []; + + // Map properties that have corresponding OptionEnum values + if ($this->outputModalities !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::outputModalities()->value, + $this->outputModalities + ); + } + + if ($this->systemInstruction !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::systemInstruction()->value, + $this->systemInstruction + ); + } + + if ($this->candidateCount !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::candidateCount()->value, + $this->candidateCount + ); + } + + if ($this->maxTokens !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::maxTokens()->value, + $this->maxTokens + ); + } + + if ($this->temperature !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::temperature()->value, + $this->temperature + ); + } + + if ($this->topP !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::topP()->value, + $this->topP + ); + } + + if ($this->topK !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::topK()->value, + $this->topK + ); + } + + if ($this->outputMimeType !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::outputMimeType()->value, + $this->outputMimeType + ); + } + + if ($this->outputSchema !== null) { + $requiredOptions[] = new RequiredOption( + OptionEnum::outputSchema()->value, + true // Just indicate that schema is required + ); + } + + // Handle properties without OptionEnum values as custom options + // These would need to be handled specially by providers + if ($this->stopSequences !== null) { + $requiredOptions[] = new RequiredOption('stop_sequences', $this->stopSequences); + } + + if ($this->presencePenalty !== null) { + $requiredOptions[] = new RequiredOption('presence_penalty', $this->presencePenalty); + } + + if ($this->frequencyPenalty !== null) { + $requiredOptions[] = new RequiredOption('frequency_penalty', $this->frequencyPenalty); + } + + if ($this->logprobs !== null) { + $requiredOptions[] = new RequiredOption('logprobs', $this->logprobs); + } + + if ($this->topLogprobs !== null) { + $requiredOptions[] = new RequiredOption('top_logprobs', $this->topLogprobs); + } + + if ($this->functionDeclarations !== null) { + $requiredOptions[] = new RequiredOption('function_declarations', true); + } + + if ($this->webSearch !== null) { + $requiredOptions[] = new RequiredOption('web_search', true); + } + + if ($this->outputFileType !== null) { + $requiredOptions[] = new RequiredOption('output_file_type', $this->outputFileType->value); + } + + if ($this->outputMediaOrientation !== null) { + $requiredOptions[] = new RequiredOption('output_media_orientation', $this->outputMediaOrientation->value); + } + + if ($this->outputMediaAspectRatio !== null) { + $requiredOptions[] = new RequiredOption('output_media_aspect_ratio', $this->outputMediaAspectRatio); + } + + // Add custom options as individual RequiredOptions + foreach ($this->customOptions as $key => $value) { + $requiredOptions[] = new RequiredOption($key, $value); + } + + return $requiredOptions; + } + /** * {@inheritDoc} * From 2a65902b1454cf850ec17cd34a9c35c70791f112 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 17:37:10 -0600 Subject: [PATCH 13/47] refactor: cleans up how system instructions work --- src/Builders/PromptBuilder.php | 161 +++++++++++++++++++++------------ 1 file changed, 103 insertions(+), 58 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index dea46008..1ff95b48 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -55,11 +55,6 @@ class PromptBuilder */ protected ModelConfig $modelConfig; - /** - * @var Message|null The system instruction message. - */ - protected ?Message $systemInstruction = null; - // phpcs:disable Generic.Files.LineLength.TooLong /** * Constructor. @@ -80,12 +75,6 @@ public function __construct(ProviderRegistry $registry, $prompt = null) return; } - // Check if it's a single Message - add to messages - if ($prompt instanceof Message) { - $this->messages[] = $prompt; - return; - } - // Check if it's a list of Messages - set as messages if (Prompts::isMessagesList($prompt)) { $this->messages = $prompt; @@ -98,29 +87,9 @@ public function __construct(ProviderRegistry $registry, $prompt = null) return; } - // Everything else becomes a UserMessage with parts - $parts = []; - - if (is_string($prompt)) { - $parts[] = new MessagePart($prompt); - } elseif ($prompt instanceof MessagePart) { - $parts[] = $prompt; - } elseif (is_array($prompt)) { - // It's a list of strings/MessageParts/MessagePartArrayShapes - foreach ($prompt as $item) { - if (is_string($item)) { - $parts[] = new MessagePart($item); - } elseif ($item instanceof MessagePart) { - $parts[] = $item; - } elseif (is_array($item) && MessagePart::isArrayShape($item)) { - $parts[] = MessagePart::fromArray($item); - } - } - } - - if (!empty($parts)) { - $this->messages[] = new UserMessage($parts); - } + // Parse it as a user message + $userMessage = $this->parseMessage($prompt, MessageRoleEnum::user()); + $this->messages[] = $userMessage; } /** @@ -295,48 +264,53 @@ public function usingRegistry(ProviderRegistry $registry): self return $this; } + // phpcs:disable Generic.Files.LineLength.TooLong /** * Sets the system instruction. * + * If a system message already exists at the beginning of the messages array, + * it will be updated. Otherwise, a new system message will be prepended. + * * @since n.e.x.t * - * @param string|MessagePart[]|Message $systemInstruction The system instruction. + * @param string|MessagePart|MessagePartArrayShape|list|Message $systemInstruction + * The system instruction. * @return self + * @throws InvalidArgumentException If the input type is not supported. */ + // phpcs:enable Generic.Files.LineLength.TooLong public function usingSystemInstruction($systemInstruction): self { - $systemInstructionText = ''; + // Parse the input into a system message + $systemMessage = $this->parseMessage($systemInstruction, MessageRoleEnum::system()); - if ($systemInstruction instanceof Message) { - $this->systemInstruction = $systemInstruction; - // Extract text from message parts for ModelConfig - foreach ($systemInstruction->getParts() as $part) { - if ($part->getText() !== null) { - $systemInstructionText .= $part->getText() . ' '; - } + // Extract text from message parts for ModelConfig + $systemInstructionText = ''; + foreach ($systemMessage->getParts() as $part) { + if ($part->getText() !== null) { + $systemInstructionText .= $part->getText() . ' '; } - } elseif (is_string($systemInstruction)) { - $this->systemInstruction = new Message( - MessageRoleEnum::system(), - [new MessagePart($systemInstruction)] - ); - $systemInstructionText = $systemInstruction; - } elseif (is_array($systemInstruction)) { - $this->systemInstruction = new Message( + } + + // Check if the first message is a system message + if (!empty($this->messages) && $this->messages[0]->getRole()->isSystem()) { + // Update the existing system message by appending new parts + $existingParts = $this->messages[0]->getParts(); + $newParts = $systemMessage->getParts(); + $this->messages[0] = new Message( MessageRoleEnum::system(), - $systemInstruction + array_merge($existingParts, $newParts) ); - // Extract text from message parts - foreach ($systemInstruction as $part) { - if ($part instanceof MessagePart && $part->getText() !== null) { - $systemInstructionText .= $part->getText() . ' '; - } - } + } else { + // Prepend the new system message + array_unshift($this->messages, $systemMessage); } + // Update ModelConfig with system instruction text if (!empty($systemInstructionText)) { $this->modelConfig->setSystemInstruction(trim($systemInstructionText)); } + return $this; } @@ -686,6 +660,77 @@ protected function validateModel(): void } } + /** + * Parses various input types into a Message with the given role. + * + * @since n.e.x.t + * + * @param mixed $input The input to parse. + * @param MessageRoleEnum $role The role for the message. + * @return Message The parsed message. + * @throws InvalidArgumentException If the input type is not supported or results in empty message. + */ + private function parseMessage($input, MessageRoleEnum $role): Message + { + // Handle Message input directly + if ($input instanceof Message) { + return $input; + } + + // Handle single MessagePart + if ($input instanceof MessagePart) { + return new Message($role, [$input]); + } + + // Handle string input + if (is_string($input)) { + if (trim($input) === '') { + throw new InvalidArgumentException('Cannot create a message from an empty string.'); + } + return new Message($role, [new MessagePart($input)]); + } + + // Handle array input + if (!is_array($input)) { + throw new InvalidArgumentException( + 'Input must be a string, MessagePart, MessagePartArrayShape, ' . + 'a list of string|MessagePart|MessagePartArrayShape, or a Message instance.' + ); + } + + // Check if it's a MessagePartArrayShape + if (MessagePart::isArrayShape($input)) { + return new Message($role, [MessagePart::fromArray($input)]); + } + + // It should be a list of string|MessagePart|MessagePartArrayShape + if (!array_is_list($input)) { + throw new InvalidArgumentException('Array input must be a list array.'); + } + + // Empty array check + if (empty($input)) { + throw new InvalidArgumentException('Cannot create a message from an empty array.'); + } + + $parts = []; + foreach ($input as $item) { + if (is_string($item)) { + $parts[] = new MessagePart($item); + } elseif ($item instanceof MessagePart) { + $parts[] = $item; + } elseif (is_array($item) && MessagePart::isArrayShape($item)) { + $parts[] = MessagePart::fromArray($item); + } else { + throw new InvalidArgumentException( + 'Array items must be strings, MessagePart instances, or MessagePartArrayShape.' + ); + } + } + + return new Message($role, $parts); + } + /** * Validates the messages array for prompt generation. * From f480ce10a04484bfddd1c21acdbdaab709e71904 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 17:50:34 -0600 Subject: [PATCH 14/47] refactor: switches validating to getting a model --- src/Builders/PromptBuilder.php | 81 ++++++++++++++++++---------------- 1 file changed, 43 insertions(+), 38 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 1ff95b48..be781467 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -553,7 +553,7 @@ public function isSupported(): bool public function generateText(): string { $this->validateMessages(); - $this->validateModel(); + $model = $this->getModel(); // This is a placeholder - actual implementation would call the model throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); @@ -575,7 +575,7 @@ public function generateTexts(?int $candidateCount = null): array } $this->validateMessages(); - $this->validateModel(); + $model = $this->getModel(); // This is a placeholder - actual implementation would call the model throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); @@ -608,56 +608,61 @@ protected function appendPartToMessages(MessagePart $part): void } /** - * Validates that the selected model meets requirements. + * Gets the model to use for generation. + * + * If a model has been explicitly set, validates it meets requirements and returns it. + * Otherwise, finds a suitable model based on the prompt requirements. * * @since n.e.x.t * - * @return void - * @throws InvalidArgumentException If model doesn't meet requirements or no suitable model found. + * @param ModelRequirements|null $requirements Optional requirements to use. If not provided, will be inferred. + * @return ModelInterface The model to use. + * @throws InvalidArgumentException If no suitable model is found or set model doesn't meet requirements. */ - protected function validateModel(): void + public function getModel(?ModelRequirements $requirements = null): ModelInterface { - $requirements = $this->getModelRequirements(); - - // If no model is specified, find one that meets requirements - if ($this->model === null) { - $modelsMetadata = $this->registry->findModelsMetadataForSupport($requirements); + if ($requirements === null) { + $requirements = $this->getModelRequirements(); + } - if (empty($modelsMetadata)) { + // If a model has been explicitly set, validate and return it + if ($this->model !== null) { + if (!$this->model->metadata()->meetsRequirements($requirements)) { throw new InvalidArgumentException( - 'No models found that support the required capabilities and options for this prompt. ' . - 'Required capabilities: ' . implode(', ', array_map(function ($cap) { - return $cap->value; - }, $requirements->getRequiredCapabilities())) . - '. Required options: ' . implode(', ', array_map(function ($opt) { - return $opt->getName() . '=' . json_encode($opt->getValue()); - }, $requirements->getRequiredOptions())) + sprintf( + 'The selected model "%s" does not meet the required capabilities and options for this prompt.', + $this->model->metadata()->getId() + ) ); } - - // Get the first available model from the first provider - $firstProviderModels = $modelsMetadata[0]; - $firstModelMetadata = $firstProviderModels->getModels()[0]; - - // Get the model instance from the provider - $this->model = $this->registry->getProviderModel( - $firstProviderModels->getProvider()->getId(), - $firstModelMetadata->getId(), - $this->modelConfig - ); - - return; + return $this->model; } - // Validate existing model meets requirements - if (!$this->model->metadata()->meetsRequirements($requirements)) { + // Find a suitable model based on requirements + $modelsMetadata = $this->registry->findModelsMetadataForSupport($requirements); + + if (empty($modelsMetadata)) { throw new InvalidArgumentException( - sprintf( - 'The selected model "%s" does not meet the required capabilities and options for this prompt.', - $this->model->metadata()->getId() - ) + 'No models found that support the required capabilities and options for this prompt. ' . + 'Required capabilities: ' . implode(', ', array_map(function ($cap) { + return $cap->value; + }, $requirements->getRequiredCapabilities())) . + '. Required options: ' . implode(', ', array_map(function ($opt) { + return $opt->getName() . '=' . json_encode($opt->getValue()); + }, $requirements->getRequiredOptions())) ); } + + // Get the first available model from the first provider + $firstProviderModels = $modelsMetadata[0]; + $firstModelMetadata = $firstProviderModels->getModels()[0]; + + // Get the model instance from the provider + return $this->registry->getProviderModel( + $firstProviderModels->getProvider()->getId(), + $firstModelMetadata->getId(), + $this->modelConfig + ); } /** From 59cc9dfce29bc5f964008b9e55d7cf5eace239b7 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Thu, 21 Aug 2025 18:00:52 -0600 Subject: [PATCH 15/47] fix: always sets the model config --- src/Builders/PromptBuilder.php | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index be781467..b7315d0f 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -553,7 +553,7 @@ public function isSupported(): bool public function generateText(): string { $this->validateMessages(); - $model = $this->getModel(); + $model = $this->getConfiguredModel(); // This is a placeholder - actual implementation would call the model throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); @@ -575,7 +575,7 @@ public function generateTexts(?int $candidateCount = null): array } $this->validateMessages(); - $model = $this->getModel(); + $model = $this->getConfiguredModel(); // This is a placeholder - actual implementation would call the model throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); @@ -619,7 +619,7 @@ protected function appendPartToMessages(MessagePart $part): void * @return ModelInterface The model to use. * @throws InvalidArgumentException If no suitable model is found or set model doesn't meet requirements. */ - public function getModel(?ModelRequirements $requirements = null): ModelInterface + public function getConfiguredModel(?ModelRequirements $requirements = null): ModelInterface { if ($requirements === null) { $requirements = $this->getModelRequirements(); @@ -635,6 +635,7 @@ public function getModel(?ModelRequirements $requirements = null): ModelInterfac ) ); } + $this->model->setConfig($this->modelConfig); return $this->model; } From 01f294b19055fefa0d9a04cc42cac8cf67dd000e Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 12:36:01 -0600 Subject: [PATCH 16/47] feat: adds generate methods --- src/Builders/PromptBuilder.php | 463 ++++++++++++++++++++++- src/Providers/Models/DTO/ModelConfig.php | 30 ++ 2 files changed, 476 insertions(+), 17 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index b7315d0f..6480682b 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -5,6 +5,7 @@ namespace WordPress\AiClient\Builders; use InvalidArgumentException; +use RuntimeException; use WordPress\AiClient\Common\Utilities\Prompts; use WordPress\AiClient\Files\DTO\File; use WordPress\AiClient\Messages\DTO\Message; @@ -18,7 +19,12 @@ use WordPress\AiClient\Providers\Models\DTO\RequiredOption; use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; use WordPress\AiClient\Providers\Models\Enums\OptionEnum; +use WordPress\AiClient\Providers\Models\ImageGeneration\Contracts\ImageGenerationModelInterface; +use WordPress\AiClient\Providers\Models\SpeechGeneration\Contracts\SpeechGenerationModelInterface; +use WordPress\AiClient\Providers\Models\TextGeneration\Contracts\TextGenerationModelInterface; +use WordPress\AiClient\Providers\Models\TextToSpeechConversion\Contracts\TextToSpeechConversionModelInterface; use WordPress\AiClient\Providers\ProviderRegistry; +use WordPress\AiClient\Results\DTO\GenerativeAiResult; use WordPress\AiClient\Tools\DTO\FunctionResponse; /** @@ -542,6 +548,176 @@ public function isSupported(): bool return $this->model->metadata()->meetsRequirements($requirements); } + /** + * Generates a result from the prompt. + * + * This is the primary execution method that generates a result (containing + * potentially multiple candidates) based on the configured output modality. + * + * @since n.e.x.t + * + * @return GenerativeAiResult The generated result containing candidates. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If the model doesn't support the configured output modality. + */ + public function generateResult(): GenerativeAiResult + { + $this->validateMessages(); + $model = $this->getConfiguredModel(); + + // Get the configured output modalities + $outputModalities = $this->modelConfig->getOutputModalities(); + + // Default to text if no output modality is specified + if ($outputModalities === null || empty($outputModalities)) { + $outputModalities = [ModalityEnum::text()]; + } + + // Multi-modal output (multiple modalities) uses TextGenerationModelInterface + if (count($outputModalities) > 1) { + if (!$model instanceof TextGenerationModelInterface) { + throw new RuntimeException( + sprintf( + 'Model "%s" does not support multi-modal generation.', + $model->metadata()->getId() + ) + ); + } + return $model->generateTextResult($this->messages); + } + + // Single modality routing + $outputModality = $outputModalities[0]; + + // Route to the appropriate generation method based on output modality + if ($outputModality->isText()) { + if (!$model instanceof TextGenerationModelInterface) { + throw new RuntimeException( + sprintf( + 'Model "%s" does not support text generation.', + $model->metadata()->getId() + ) + ); + } + return $model->generateTextResult($this->messages); + } + + if ($outputModality->isImage()) { + if (!$model instanceof ImageGenerationModelInterface) { + throw new RuntimeException( + sprintf( + 'Model "%s" does not support image generation.', + $model->metadata()->getId() + ) + ); + } + return $model->generateImageResult($this->messages); + } + + if ($outputModality->isAudio()) { + if (!$model instanceof SpeechGenerationModelInterface) { + throw new RuntimeException( + sprintf( + 'Model "%s" does not support speech/audio generation.', + $model->metadata()->getId() + ) + ); + } + return $model->generateSpeechResult($this->messages); + } + + // TODO: Add support for video output modality when interface is available + throw new RuntimeException( + sprintf('Output modality "%s" is not yet supported.', $outputModality->value) + ); + } + + /** + * Generates a text result from the prompt. + * + * @since n.e.x.t + * + * @return GenerativeAiResult The generated result containing text candidates. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If the model doesn't support text generation. + */ + public function generateTextResult(): GenerativeAiResult + { + // Include text in output modalities + $this->modelConfig->includeOutputModality(ModalityEnum::text()); + + // Generate and return the result + return $this->generateResult(); + } + + /** + * Generates an image result from the prompt. + * + * @since n.e.x.t + * + * @return GenerativeAiResult The generated result containing image candidates. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If the model doesn't support image generation. + */ + public function generateImageResult(): GenerativeAiResult + { + // Include image in output modalities + $this->modelConfig->includeOutputModality(ModalityEnum::image()); + + // Generate and return the result + return $this->generateResult(); + } + + /** + * Generates a speech result from the prompt. + * + * @since n.e.x.t + * + * @return GenerativeAiResult The generated result containing speech audio candidates. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If the model doesn't support speech generation. + */ + public function generateSpeechResult(): GenerativeAiResult + { + // Include audio in output modalities + $this->modelConfig->includeOutputModality(ModalityEnum::audio()); + + // Generate and return the result + return $this->generateResult(); + } + + /** + * Converts text to speech and returns the result. + * + * @since n.e.x.t + * + * @return GenerativeAiResult The generated result containing speech audio candidates. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If the model doesn't support text-to-speech conversion. + */ + public function convertTextToSpeechResult(): GenerativeAiResult + { + // Include audio in output modalities + $this->modelConfig->includeOutputModality(ModalityEnum::audio()); + + // Get the configured model + $model = $this->getConfiguredModel(); + + // Ensure the model supports text-to-speech conversion + if (!$model instanceof TextToSpeechConversionModelInterface) { + throw new RuntimeException( + sprintf( + 'Model "%s" does not support text-to-speech conversion.', + $model->metadata()->getId() + ) + ); + } + + // Validate messages and convert + $this->validateMessages(); + return $model->convertTextToSpeechResult($this->messages); + } + /** * Generates text from the prompt. * @@ -552,11 +728,27 @@ public function isSupported(): bool */ public function generateText(): string { - $this->validateMessages(); - $model = $this->getConfiguredModel(); + // Generate text result and extract text from first candidate + $result = $this->generateTextResult(); + $candidates = $result->getCandidates(); + + if (empty($candidates)) { + throw new RuntimeException('No candidates were generated.'); + } + + // Get the text from the first message part + $message = $candidates[0]->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + throw new RuntimeException('Generated message contains no parts.'); + } + + $text = $parts[0]->getText(); + if ($text === null) { + throw new RuntimeException('Generated message part contains no text.'); + } - // This is a placeholder - actual implementation would call the model - throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); + return $text; } /** @@ -574,11 +766,256 @@ public function generateTexts(?int $candidateCount = null): array $this->usingCandidateCount($candidateCount); } - $this->validateMessages(); - $model = $this->getConfiguredModel(); + // Generate text result + $results = $this->generateTextResult(); + $candidates = $results->getCandidates(); + + // Extract text from each candidate + $texts = []; + foreach ($candidates as $candidate) { + $message = $candidate->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + continue; + } + + $text = $parts[0]->getText(); + if ($text !== null) { + $texts[] = $text; + } + } + + if (empty($texts)) { + throw new RuntimeException('No text was generated from any candidates.'); + } + + return $texts; + } + + /** + * Generates an image from the prompt. + * + * @since n.e.x.t + * + * @return File The generated image file. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If no image is generated. + */ + public function generateImage(): File + { + // Generate image result and extract image from first candidate + $result = $this->generateImageResult(); + $candidates = $result->getCandidates(); + + if (empty($candidates)) { + throw new RuntimeException('No candidates were generated.'); + } - // This is a placeholder - actual implementation would call the model - throw new \RuntimeException('Not implemented yet - requires AiClient integration.'); + // Get the image file from the first message part + $message = $candidates[0]->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + throw new RuntimeException('Generated message contains no parts.'); + } + + $file = $parts[0]->getFile(); + if ($file === null) { + throw new RuntimeException('Generated message part contains no image file.'); + } + + return $file; + } + + /** + * Generates multiple images from the prompt. + * + * @since n.e.x.t + * + * @param int|null $candidateCount The number of images to generate. + * @return list The generated image files. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If no images are generated. + */ + public function generateImages(?int $candidateCount = null): array + { + if ($candidateCount !== null) { + $this->usingCandidateCount($candidateCount); + } + + // Generate image result + $results = $this->generateImageResult(); + $candidates = $results->getCandidates(); + + // Extract image files from each candidate + $images = []; + foreach ($candidates as $candidate) { + $message = $candidate->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + continue; + } + + $file = $parts[0]->getFile(); + if ($file !== null) { + $images[] = $file; + } + } + + if (empty($images)) { + throw new RuntimeException('No images were generated from any candidates.'); + } + + return $images; + } + + /** + * Converts text to speech. + * + * @since n.e.x.t + * + * @return File The generated speech audio file. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If no audio is generated. + */ + public function convertTextToSpeech(): File + { + // Convert text to speech and extract audio from first candidate + $result = $this->convertTextToSpeechResult(); + $candidates = $result->getCandidates(); + + if (empty($candidates)) { + throw new RuntimeException('No candidates were generated.'); + } + + $message = $candidates[0]->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + throw new RuntimeException('Generated message contains no parts.'); + } + + $file = $parts[0]->getFile(); + if ($file === null) { + throw new RuntimeException('Generated message part contains no audio file.'); + } + + return $file; + } + + /** + * Converts text to multiple speech outputs. + * + * @since n.e.x.t + * + * @param int|null $candidateCount The number of speech outputs to generate. + * @return list The generated speech audio files. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If no audio is generated. + */ + public function convertTextToSpeeches(?int $candidateCount = null): array + { + if ($candidateCount !== null) { + $this->usingCandidateCount($candidateCount); + } + + // Convert text to speech + $result = $this->convertTextToSpeechResult(); + + // Extract audio files from each candidate + $audioFiles = []; + foreach ($result->getCandidates() as $candidate) { + $message = $candidate->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + continue; + } + + $file = $parts[0]->getFile(); + if ($file !== null) { + $audioFiles[] = $file; + } + } + + if (empty($audioFiles)) { + throw new RuntimeException('No audio files were generated from any candidates.'); + } + + return $audioFiles; + } + + /** + * Generates speech from the prompt. + * + * @since n.e.x.t + * + * @return File The generated speech audio file. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If no audio is generated. + */ + public function generateSpeech(): File + { + // Generate speech result and extract audio from first candidate + $result = $this->generateSpeechResult(); + $candidates = $result->getCandidates(); + + if (empty($candidates)) { + throw new RuntimeException('No candidates were generated.'); + } + + // Get the audio file from the first message part + $message = $candidates[0]->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + throw new RuntimeException('Generated message contains no parts.'); + } + + $file = $parts[0]->getFile(); + if ($file === null) { + throw new RuntimeException('Generated message part contains no audio file.'); + } + + return $file; + } + + /** + * Generates multiple speech outputs from the prompt. + * + * @since n.e.x.t + * + * @param int|null $candidateCount The number of speech outputs to generate. + * @return list The generated speech audio files. + * @throws InvalidArgumentException If the prompt or model validation fails. + * @throws RuntimeException If no audio is generated. + */ + public function generateSpeeches(?int $candidateCount = null): array + { + if ($candidateCount !== null) { + $this->usingCandidateCount($candidateCount); + } + + // Generate speech result + $result = $this->generateSpeechResult(); + $candidates = $result->getCandidates(); + + // Extract audio files from each candidate + $audioFiles = []; + foreach ($candidates as $candidate) { + $message = $candidate->getMessage(); + $parts = $message->getParts(); + if (empty($parts)) { + continue; + } + + $file = $parts[0]->getFile(); + if ($file !== null) { + $audioFiles[] = $file; + } + } + + if (empty($audioFiles)) { + throw new RuntimeException('No audio files were generated from any candidates.'); + } + + return $audioFiles; } /** @@ -625,16 +1062,8 @@ public function getConfiguredModel(?ModelRequirements $requirements = null): Mod $requirements = $this->getModelRequirements(); } - // If a model has been explicitly set, validate and return it + // If a model has been explicitly set, return it if ($this->model !== null) { - if (!$this->model->metadata()->meetsRequirements($requirements)) { - throw new InvalidArgumentException( - sprintf( - 'The selected model "%s" does not meet the required capabilities and options for this prompt.', - $this->model->metadata()->getId() - ) - ); - } $this->model->setConfig($this->modelConfig); return $this->model; } diff --git a/src/Providers/Models/DTO/ModelConfig.php b/src/Providers/Models/DTO/ModelConfig.php index 52a23192..c7f9afcf 100644 --- a/src/Providers/Models/DTO/ModelConfig.php +++ b/src/Providers/Models/DTO/ModelConfig.php @@ -203,6 +203,36 @@ public function getOutputModalities(): ?array return $this->outputModalities; } + /** + * Includes an output modality if not already present. + * + * Adds the given modality to the output modalities list if it's not + * already included. If output modalities is null, initializes it with + * the given modality. + * + * @since n.e.x.t + * + * @param ModalityEnum $modality The modality to include. + */ + public function includeOutputModality(ModalityEnum $modality): void + { + // Initialize if null + if ($this->outputModalities === null) { + $this->outputModalities = [$modality]; + return; + } + + // Check if modality already exists + foreach ($this->outputModalities as $existingModality) { + if ($existingModality->value === $modality->value) { + return; // Already included + } + } + + // Add the modality + $this->outputModalities[] = $modality; + } + /** * Sets the system instruction. * From 3fd1fc2fb17b9de99881ac6e1dbc2504b39a2d1d Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 12:37:31 -0600 Subject: [PATCH 17/47] refactor: tweaks enum comparison --- src/Providers/Models/DTO/ModelConfig.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Providers/Models/DTO/ModelConfig.php b/src/Providers/Models/DTO/ModelConfig.php index c7f9afcf..585dbe2e 100644 --- a/src/Providers/Models/DTO/ModelConfig.php +++ b/src/Providers/Models/DTO/ModelConfig.php @@ -224,7 +224,7 @@ public function includeOutputModality(ModalityEnum $modality): void // Check if modality already exists foreach ($this->outputModalities as $existingModality) { - if ($existingModality->value === $modality->value) { + if ($existingModality === $modality) { return; // Already included } } From c56bf85a9676e471ec83fe0e230a2dc65439eeeb Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 12:54:47 -0600 Subject: [PATCH 18/47] test: adds includeOutputModality tests --- .../Providers/Models/DTO/ModelConfigTest.php | 139 ++++++++++++++++++ 1 file changed, 139 insertions(+) diff --git a/tests/unit/Providers/Models/DTO/ModelConfigTest.php b/tests/unit/Providers/Models/DTO/ModelConfigTest.php index 7f5b353b..7866928a 100644 --- a/tests/unit/Providers/Models/DTO/ModelConfigTest.php +++ b/tests/unit/Providers/Models/DTO/ModelConfigTest.php @@ -654,4 +654,143 @@ public function testSetCustomOption(): void $restored = ModelConfig::fromArray($array); $this->assertEquals($customOptions, $restored->getCustomOptions()); } + + /** + * Tests includeOutputModality method. + * + * @return void + */ + public function testIncludeOutputModality(): void + { + $config = new ModelConfig(); + + // Test adding modality to null output modalities + $this->assertNull($config->getOutputModalities()); + $config->includeOutputModality(ModalityEnum::text()); + $modalities = $config->getOutputModalities(); + $this->assertCount(1, $modalities); + $this->assertTrue($modalities[0]->isText()); + + // Test adding a different modality + $config->includeOutputModality(ModalityEnum::image()); + $modalities = $config->getOutputModalities(); + $this->assertCount(2, $modalities); + $this->assertTrue($modalities[0]->isText()); + $this->assertTrue($modalities[1]->isImage()); + + // Test adding a duplicate modality (should not add) + $config->includeOutputModality(ModalityEnum::text()); + $modalities = $config->getOutputModalities(); + $this->assertCount(2, $modalities); + $this->assertTrue($modalities[0]->isText()); + $this->assertTrue($modalities[1]->isImage()); + + // Test adding another unique modality + $config->includeOutputModality(ModalityEnum::audio()); + $modalities = $config->getOutputModalities(); + $this->assertCount(3, $modalities); + $this->assertTrue($modalities[0]->isText()); + $this->assertTrue($modalities[1]->isImage()); + $this->assertTrue($modalities[2]->isAudio()); + + // Test that duplicate modalities are not added (different instance, same value) + $config->includeOutputModality(ModalityEnum::image()); + $modalities = $config->getOutputModalities(); + $this->assertCount(3, $modalities); + } + + /** + * Tests includeOutputModality with existing modalities set via setOutputModalities. + * + * @return void + */ + public function testIncludeOutputModalityWithExistingModalitiesSet(): void + { + $config = new ModelConfig(); + + // Set initial modalities + $config->setOutputModalities([ModalityEnum::text(), ModalityEnum::video()]); + $modalities = $config->getOutputModalities(); + $this->assertCount(2, $modalities); + + // Include a new modality + $config->includeOutputModality(ModalityEnum::image()); + $modalities = $config->getOutputModalities(); + $this->assertCount(3, $modalities); + $this->assertTrue($modalities[0]->isText()); + $this->assertTrue($modalities[1]->isVideo()); + $this->assertTrue($modalities[2]->isImage()); + + // Include an existing modality (should not add) + $config->includeOutputModality(ModalityEnum::video()); + $modalities = $config->getOutputModalities(); + $this->assertCount(3, $modalities); + } + + /** + * Tests includeOutputModality preserves modality order. + * + * @return void + */ + public function testIncludeOutputModalityPreservesOrder(): void + { + $config = new ModelConfig(); + + // Add modalities in specific order + $config->includeOutputModality(ModalityEnum::audio()); + $config->includeOutputModality(ModalityEnum::document()); + $config->includeOutputModality(ModalityEnum::text()); + $config->includeOutputModality(ModalityEnum::image()); + + $modalities = $config->getOutputModalities(); + $this->assertCount(4, $modalities); + $this->assertTrue($modalities[0]->isAudio()); + $this->assertTrue($modalities[1]->isDocument()); + $this->assertTrue($modalities[2]->isText()); + $this->assertTrue($modalities[3]->isImage()); + + // Try to add existing modalities in different order (should not change) + $config->includeOutputModality(ModalityEnum::text()); + $config->includeOutputModality(ModalityEnum::audio()); + + $modalities = $config->getOutputModalities(); + $this->assertCount(4, $modalities); + $this->assertTrue($modalities[0]->isAudio()); + $this->assertTrue($modalities[1]->isDocument()); + $this->assertTrue($modalities[2]->isText()); + $this->assertTrue($modalities[3]->isImage()); + } + + /** + * Tests includeOutputModality handles all modality types. + * + * @return void + */ + public function testIncludeOutputModalityHandlesAllModalityTypes(): void + { + $config = new ModelConfig(); + + // Test all available modality types + $allModalities = [ + ModalityEnum::text(), + ModalityEnum::image(), + ModalityEnum::audio(), + ModalityEnum::video(), + ModalityEnum::document() + ]; + + foreach ($allModalities as $modality) { + $config->includeOutputModality($modality); + } + + $modalities = $config->getOutputModalities(); + $this->assertCount(5, $modalities); + + // Verify all modalities are present + $this->assertTrue($modalities[0]->isText()); + $this->assertTrue($modalities[1]->isImage()); + $this->assertTrue($modalities[2]->isAudio()); + $this->assertTrue($modalities[3]->isVideo()); + $this->assertTrue($modalities[4]->isDocument()); + } } From 539a278ae8f89c779a998785bc6fdf452a2c472a Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 13:19:52 -0600 Subject: [PATCH 19/47] refactor: uses system instructions instead of message --- src/Builders/PromptBuilder.php | 49 ++++++---------------------------- 1 file changed, 8 insertions(+), 41 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 6480682b..59b029cc 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -270,53 +270,20 @@ public function usingRegistry(ProviderRegistry $registry): self return $this; } - // phpcs:disable Generic.Files.LineLength.TooLong /** * Sets the system instruction. * - * If a system message already exists at the beginning of the messages array, - * it will be updated. Otherwise, a new system message will be prepended. + * System instructions are stored in the model configuration and guide + * the AI model's behavior throughout the conversation. * * @since n.e.x.t * - * @param string|MessagePart|MessagePartArrayShape|list|Message $systemInstruction - * The system instruction. + * @param string $systemInstruction The system instruction text. * @return self - * @throws InvalidArgumentException If the input type is not supported. */ - // phpcs:enable Generic.Files.LineLength.TooLong - public function usingSystemInstruction($systemInstruction): self + public function usingSystemInstruction(string $systemInstruction): self { - // Parse the input into a system message - $systemMessage = $this->parseMessage($systemInstruction, MessageRoleEnum::system()); - - // Extract text from message parts for ModelConfig - $systemInstructionText = ''; - foreach ($systemMessage->getParts() as $part) { - if ($part->getText() !== null) { - $systemInstructionText .= $part->getText() . ' '; - } - } - - // Check if the first message is a system message - if (!empty($this->messages) && $this->messages[0]->getRole()->isSystem()) { - // Update the existing system message by appending new parts - $existingParts = $this->messages[0]->getParts(); - $newParts = $systemMessage->getParts(); - $this->messages[0] = new Message( - MessageRoleEnum::system(), - array_merge($existingParts, $newParts) - ); - } else { - // Prepend the new system message - array_unshift($this->messages, $systemMessage); - } - - // Update ModelConfig with system instruction text - if (!empty($systemInstructionText)) { - $this->modelConfig->setSystemInstruction(trim($systemInstructionText)); - } - + $this->modelConfig->setSystemInstruction($systemInstruction); return $this; } @@ -1170,7 +1137,7 @@ private function parseMessage($input, MessageRoleEnum $role): Message * Validates the messages array for prompt generation. * * Ensures that: - * - The first message is a user or system message + * - The first message is a user message * - The last message is a user message * - The last message has parts * @@ -1188,9 +1155,9 @@ private function validateMessages(): void } $firstMessage = reset($this->messages); - if (!$firstMessage->getRole()->isUser() && !$firstMessage->getRole()->isSystem()) { + if (!$firstMessage->getRole()->isUser()) { throw new InvalidArgumentException( - 'The first message must be from a user or system role, not from ' . $firstMessage->getRole()->value + 'The first message must be from a user role, not from ' . $firstMessage->getRole()->value ); } From e86190e11e066b50589c63cb7e3c27f79300f3b5 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 13:40:59 -0600 Subject: [PATCH 20/47] test: adds a lot of Prompt Builder tests --- tests/unit/Builders/PromptBuilderTest.php | 2523 +++++++++++++++++++++ 1 file changed, 2523 insertions(+) create mode 100644 tests/unit/Builders/PromptBuilderTest.php diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php new file mode 100644 index 00000000..d2bb0c20 --- /dev/null +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -0,0 +1,2523 @@ +registry = $this->createMock(ProviderRegistry::class); + } + + /** + * Tests constructor with no prompt. + * + * @return void + */ + public function testConstructorWithNoPrompt(): void + { + $builder = new PromptBuilder($this->registry); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + + $this->assertEmpty($messagesProperty->getValue($builder)); + } + + /** + * Tests constructor with string prompt. + * + * @return void + */ + public function testConstructorWithStringPrompt(): void + { + $builder = new PromptBuilder($this->registry, 'Hello, world!'); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertEquals('Hello, world!', $messages[0]->getParts()[0]->getText()); + } + + /** + * Tests constructor with MessagePart prompt. + * + * @return void + */ + public function testConstructorWithMessagePartPrompt(): void + { + $part = new MessagePart('Test message'); + $builder = new PromptBuilder($this->registry, $part); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertEquals('Test message', $messages[0]->getParts()[0]->getText()); + } + + /** + * Tests constructor with Message prompt. + * + * @return void + */ + public function testConstructorWithMessagePrompt(): void + { + $message = new UserMessage([new MessagePart('User message')]); + $builder = new PromptBuilder($this->registry, $message); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertSame($message, $messages[0]); + } + + /** + * Tests constructor with list of Messages. + * + * @return void + */ + public function testConstructorWithMessagesList(): void + { + $messages = [ + new UserMessage([new MessagePart('First')]), + new ModelMessage([new MessagePart('Second')]), + new UserMessage([new MessagePart('Third')]) + ]; + $builder = new PromptBuilder($this->registry, $messages); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $actualMessages = $messagesProperty->getValue($builder); + + $this->assertCount(3, $actualMessages); + $this->assertSame($messages, $actualMessages); + } + + /** + * Tests constructor with MessageArrayShape. + * + * @return void + */ + public function testConstructorWithMessageArrayShape(): void + { + $messageArray = [ + 'role' => 'user', + 'parts' => [ + ['type' => 'text', 'text' => 'Hello from array'] + ] + ]; + $builder = new PromptBuilder($this->registry, $messageArray); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertEquals('Hello from array', $messages[0]->getParts()[0]->getText()); + } + + /** + * Tests withText method. + * + * @return void + */ + public function testWithText(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->withText('Some text'); + + $this->assertSame($builder, $result); // Test fluent interface + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertEquals('Some text', $messages[0]->getParts()[0]->getText()); + } + + /** + * Tests withText appends to existing user message. + * + * @return void + */ + public function testWithTextAppendsToExistingUserMessage(): void + { + $builder = new PromptBuilder($this->registry, 'Initial text'); + $builder->withText(' Additional text'); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $parts = $messages[0]->getParts(); + $this->assertCount(2, $parts); + $this->assertEquals('Initial text', $parts[0]->getText()); + $this->assertEquals(' Additional text', $parts[1]->getText()); + } + + /** + * Tests withInlineImage method. + * + * @return void + */ + public function testWithInlineImage(): void + { + $builder = new PromptBuilder($this->registry); + $base64 = 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNkYPhfDwAChwGA60e6kgAAAABJRU5ErkJggg=='; + $result = $builder->withInlineImage($base64, 'image/png'); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $file = $messages[0]->getParts()[0]->getFile(); + $this->assertInstanceOf(File::class, $file); + $this->assertEquals('data:image/png;base64,' . $base64, $file->getUri()); + $this->assertEquals('image/png', $file->getMimeType()); + } + + /** + * Tests withRemoteImage method. + * + * @return void + */ + public function testWithRemoteImage(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->withRemoteImage('https://example.com/image.jpg', 'image/jpeg'); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $file = $messages[0]->getParts()[0]->getFile(); + $this->assertInstanceOf(File::class, $file); + $this->assertEquals('https://example.com/image.jpg', $file->getUri()); + $this->assertEquals('image/jpeg', $file->getMimeType()); + } + + /** + * Tests withImageFile method. + * + * @return void + */ + public function testWithImageFile(): void + { + $file = new File('https://example.com/test.png', 'image/png'); + $builder = new PromptBuilder($this->registry); + $result = $builder->withImageFile($file); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); + } + + /** + * Tests withAudioFile method. + * + * @return void + */ + public function testWithAudioFile(): void + { + $file = new File('https://example.com/audio.mp3', 'audio/mp3'); + $builder = new PromptBuilder($this->registry); + $result = $builder->withAudioFile($file); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); + } + + /** + * Tests withVideoFile method. + * + * @return void + */ + public function testWithVideoFile(): void + { + $file = new File('https://example.com/video.mp4', 'video/mp4'); + $builder = new PromptBuilder($this->registry); + $result = $builder->withVideoFile($file); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); + } + + /** + * Tests withFunctionResponse method. + * + * @return void + */ + public function testWithFunctionResponse(): void + { + $functionResponse = new FunctionResponse('func_id', 'func_name', ['result' => 'data']); + $builder = new PromptBuilder($this->registry); + $result = $builder->withFunctionResponse($functionResponse); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertSame($functionResponse, $messages[0]->getParts()[0]->getFunctionResponse()); + } + + /** + * Tests withMessageParts method. + * + * @return void + */ + public function testWithMessageParts(): void + { + $part1 = new MessagePart('Part 1'); + $part2 = new MessagePart('Part 2'); + $part3 = new MessagePart('Part 3'); + + $builder = new PromptBuilder($this->registry); + $result = $builder->withMessageParts($part1, $part2, $part3); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $parts = $messages[0]->getParts(); + $this->assertCount(3, $parts); + $this->assertEquals('Part 1', $parts[0]->getText()); + $this->assertEquals('Part 2', $parts[1]->getText()); + $this->assertEquals('Part 3', $parts[2]->getText()); + } + + /** + * Tests withHistory method. + * + * @return void + */ + public function testWithHistory(): void + { + $history = [ + new UserMessage([new MessagePart('User 1')]), + new ModelMessage([new MessagePart('Model 1')]), + new UserMessage([new MessagePart('User 2')]) + ]; + + $builder = new PromptBuilder($this->registry); + $result = $builder->withHistory(...$history); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(3, $messages); + $this->assertEquals('User 1', $messages[0]->getParts()[0]->getText()); + $this->assertEquals('Model 1', $messages[1]->getParts()[0]->getText()); + $this->assertEquals('User 2', $messages[2]->getParts()[0]->getText()); + } + + /** + * Tests usingModel method. + * + * @return void + */ + public function testUsingModel(): void + { + $model = $this->createMock(ModelInterface::class); + $builder = new PromptBuilder($this->registry); + $result = $builder->usingModel($model); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $modelProperty = $reflection->getProperty('model'); + $modelProperty->setAccessible(true); + + $this->assertSame($model, $modelProperty->getValue($builder)); + } + + /** + * Tests usingRegistry method. + * + * @return void + */ + public function testUsingRegistry(): void + { + $newRegistry = $this->createMock(ProviderRegistry::class); + $builder = new PromptBuilder($this->registry); + $result = $builder->usingRegistry($newRegistry); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $registryProperty = $reflection->getProperty('registry'); + $registryProperty->setAccessible(true); + + $this->assertSame($newRegistry, $registryProperty->getValue($builder)); + } + + /** + * Tests usingSystemInstruction method. + * + * @return void + */ + public function testUsingSystemInstruction(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingSystemInstruction('You are a helpful assistant.'); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals('You are a helpful assistant.', $config->getSystemInstruction()); + } + + /** + * Tests usingMaxTokens method. + * + * @return void + */ + public function testUsingMaxTokens(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingMaxTokens(1000); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals(1000, $config->getMaxTokens()); + } + + /** + * Tests usingTemperature method. + * + * @return void + */ + public function testUsingTemperature(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingTemperature(0.7); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals(0.7, $config->getTemperature()); + } + + /** + * Tests usingTopP method. + * + * @return void + */ + public function testUsingTopP(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingTopP(0.9); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals(0.9, $config->getTopP()); + } + + /** + * Tests usingTopK method. + * + * @return void + */ + public function testUsingTopK(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingTopK(40); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals(40, $config->getTopK()); + } + + /** + * Tests usingStopSequences method. + * + * @return void + */ + public function testUsingStopSequences(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingStopSequences('STOP', 'END', '###'); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $customOptions = $config->getCustomOptions(); + $this->assertArrayHasKey('stopSequences', $customOptions); + $this->assertEquals(['STOP', 'END', '###'], $customOptions['stopSequences']); + } + + /** + * Tests usingCandidateCount method. + * + * @return void + */ + public function testUsingCandidateCount(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingCandidateCount(3); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals(3, $config->getCandidateCount()); + } + + /** + * Tests usingOutputMime method. + * + * @return void + */ + public function testUsingOutputMime(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingOutputMime('application/json'); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals('application/json', $config->getOutputMimeType()); + } + + /** + * Tests usingOutputSchema method. + * + * @return void + */ + public function testUsingOutputSchema(): void + { + $schema = [ + 'type' => 'object', + 'properties' => [ + 'name' => ['type' => 'string'] + ] + ]; + + $builder = new PromptBuilder($this->registry); + $result = $builder->usingOutputSchema($schema); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals($schema, $config->getOutputSchema()); + } + + /** + * Tests usingOutputModalities method. + * + * @return void + */ + public function testUsingOutputModalities(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->usingOutputModalities( + ModalityEnum::text(), + ModalityEnum::image() + ); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $modalities = $config->getOutputModalities(); + $this->assertCount(2, $modalities); + $this->assertTrue($modalities[0]->isText()); + $this->assertTrue($modalities[1]->isImage()); + } + + /** + * Tests asJsonResponse method. + * + * @return void + */ + public function testAsJsonResponse(): void + { + $builder = new PromptBuilder($this->registry); + $result = $builder->asJsonResponse(); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals('application/json', $config->getOutputMimeType()); + } + + /** + * Tests asJsonResponse with schema. + * + * @return void + */ + public function testAsJsonResponseWithSchema(): void + { + $schema = ['type' => 'array']; + $builder = new PromptBuilder($this->registry); + $result = $builder->asJsonResponse($schema); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals('application/json', $config->getOutputMimeType()); + $this->assertEquals($schema, $config->getOutputSchema()); + } + + /** + * Tests getModelRequirements with basic text prompt. + * + * @return void + */ + public function testGetModelRequirementsBasicText(): void + { + $builder = new PromptBuilder($this->registry, 'Simple text'); + $requirements = $builder->getModelRequirements(); + + $capabilities = $requirements->getRequiredCapabilities(); + $this->assertCount(1, $capabilities); + $this->assertTrue($capabilities[0]->isTextGeneration()); + + $options = $requirements->getRequiredOptions(); + // Should have input modalities with text + $inputModalitiesFound = false; + foreach ($options as $option) { + if ($option->getName() === OptionEnum::inputModalities()->value) { + $inputModalitiesFound = true; + $modalities = $option->getValue(); + $this->assertCount(1, $modalities); + $this->assertTrue($modalities[0]->isText()); + } + } + $this->assertTrue($inputModalitiesFound); + } + + /** + * Tests getModelRequirements with chat history. + * + * @return void + */ + public function testGetModelRequirementsWithChatHistory(): void + { + $builder = new PromptBuilder($this->registry); + $builder->withHistory( + new UserMessage([new MessagePart('Hello')]), + new ModelMessage([new MessagePart('Hi there')]), + new UserMessage([new MessagePart('How are you?')]) + ); + + $requirements = $builder->getModelRequirements(); + $capabilities = $requirements->getRequiredCapabilities(); + + // Should have text generation and chat history capabilities + $this->assertCount(2, $capabilities); + $hasTextGeneration = false; + $hasChatHistory = false; + foreach ($capabilities as $capability) { + if ($capability->isTextGeneration()) { + $hasTextGeneration = true; + } + if ($capability->isChatHistory()) { + $hasChatHistory = true; + } + } + $this->assertTrue($hasTextGeneration); + $this->assertTrue($hasChatHistory); + } + + /** + * Tests getModelRequirements with multimodal input. + * + * @return void + */ + public function testGetModelRequirementsWithMultimodalInput(): void + { + $builder = new PromptBuilder($this->registry); + $builder->withText('Describe this image') + ->withRemoteImage('https://example.com/image.jpg', 'image/jpeg'); + + $requirements = $builder->getModelRequirements(); + $options = $requirements->getRequiredOptions(); + + // Find input modalities option + $inputModalities = null; + foreach ($options as $option) { + if ($option->getName() === OptionEnum::inputModalities()->value) { + $inputModalities = $option->getValue(); + break; + } + } + + $this->assertNotNull($inputModalities); + $this->assertCount(2, $inputModalities); + + $hasText = false; + $hasImage = false; + foreach ($inputModalities as $modality) { + if ($modality->isText()) { + $hasText = true; + } + if ($modality->isImage()) { + $hasImage = true; + } + } + $this->assertTrue($hasText); + $this->assertTrue($hasImage); + } + + /** + * Tests isSupported without model. + * + * @return void + */ + public function testIsSupportedWithoutModel(): void + { + $builder = new PromptBuilder($this->registry, 'Test'); + + // Without a model, should return true (can't determine support) + $this->assertTrue($builder->isSupported()); + } + + /** + * Tests isSupported with compatible model. + * + * @return void + */ + public function testIsSupportedWithCompatibleModel(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ModelInterface::class); + $model->method('metadata')->willReturn($metadata); + + $builder = new PromptBuilder($this->registry, 'Test'); + $builder->usingModel($model); + + $this->assertTrue($builder->isSupported()); + } + + /** + * Tests isSupported with incompatible model. + * + * @return void + */ + public function testIsSupportedWithIncompatibleModel(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('meetsRequirements')->willReturn(false); + + $model = $this->createMock(ModelInterface::class); + $model->method('metadata')->willReturn($metadata); + + $builder = new PromptBuilder($this->registry, 'Test'); + $builder->usingModel($model); + + $this->assertFalse($builder->isSupported()); + } + + /** + * Tests validateMessages with empty messages throws exception. + * + * @return void + */ + public function testValidateMessagesEmptyThrowsException(): void + { + $builder = new PromptBuilder($this->registry); + + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Cannot generate from an empty prompt'); + + $builder->generateResult(); + } + + /** + * Tests validateMessages with non-user first message throws exception. + * + * @return void + */ + public function testValidateMessagesNonUserFirstThrowsException(): void + { + $builder = new PromptBuilder($this->registry, [ + new ModelMessage([new MessagePart('Model says hi')]), + new UserMessage([new MessagePart('User response')]) + ]); + + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('The first message must be from a user role'); + + $builder->generateResult(); + } + + /** + * Tests validateMessages with non-user last message throws exception. + * + * @return void + */ + public function testValidateMessagesNonUserLastThrowsException(): void + { + $builder = new PromptBuilder($this->registry, [ + new UserMessage([new MessagePart('User says hi')]), + new ModelMessage([new MessagePart('Model response')]) + ]); + + // Add a user message to make it valid, then add model message + $builder->withHistory(new ModelMessage([new MessagePart('Another model message')])); + + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('The last message must be from a user role'); + + $builder->generateResult(); + } + + /** + * Tests parseMessage with empty string throws exception. + * + * @return void + */ + public function testParseMessageEmptyStringThrowsException(): void + { + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Cannot create a message from an empty string'); + + new PromptBuilder($this->registry, ' '); + } + + /** + * Tests parseMessage with empty array throws exception. + * + * @return void + */ + public function testParseMessageEmptyArrayThrowsException(): void + { + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Cannot create a message from an empty array'); + + new PromptBuilder($this->registry, []); + } + + /** + * Tests parseMessage with invalid type throws exception. + * + * @return void + */ + public function testParseMessageInvalidTypeThrowsException(): void + { + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Input must be a string, MessagePart, MessagePartArrayShape'); + + new PromptBuilder($this->registry, 123); + } + + /** + * Tests chaining multiple operations. + * + * @return void + */ + public function testMethodChaining(): void + { + $model = $this->createMock(ModelInterface::class); + + $builder = new PromptBuilder($this->registry); + $result = $builder + ->withText('Start of prompt') + ->withRemoteImage('https://example.com/img.jpg', 'image/jpeg') + ->usingModel($model) + ->usingSystemInstruction('Be helpful') + ->usingMaxTokens(500) + ->usingTemperature(0.8) + ->usingTopP(0.95) + ->usingTopK(50) + ->usingCandidateCount(2) + ->asJsonResponse(); + + $this->assertSame($builder, $result); + + $reflection = new \ReflectionClass($builder); + + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + $this->assertCount(1, $messages); + $this->assertCount(2, $messages[0]->getParts()); // Text and image + + $modelProperty = $reflection->getProperty('model'); + $modelProperty->setAccessible(true); + $this->assertSame($model, $modelProperty->getValue($builder)); + + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals('Be helpful', $config->getSystemInstruction()); + $this->assertEquals(500, $config->getMaxTokens()); + $this->assertEquals(0.8, $config->getTemperature()); + $this->assertEquals(0.95, $config->getTopP()); + $this->assertEquals(50, $config->getTopK()); + $this->assertEquals(2, $config->getCandidateCount()); + $this->assertEquals('application/json', $config->getOutputMimeType()); + } + + /** + * Tests generateResult with text output modality. + * + * @return void + */ + public function testGenerateResultWithTextModality(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Test prompt'); + $builder->usingModel($model); + + $actualResult = $builder->generateResult(); + $this->assertSame($result, $actualResult); + } + + /** + * Tests generateResult with image output modality. + * + * @return void + */ + public function testGenerateResultWithImageModality(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ImageGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateImageResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate an image'); + $builder->usingModel($model); + $builder->usingOutputModalities(ModalityEnum::image()); + + $actualResult = $builder->generateResult(); + $this->assertSame($result, $actualResult); + } + + /** + * Tests generateResult with audio output modality. + * + * @return void + */ + public function testGenerateResultWithAudioModality(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(SpeechGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateSpeechResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate speech'); + $builder->usingModel($model); + $builder->usingOutputModalities(ModalityEnum::audio()); + + $actualResult = $builder->generateResult(); + $this->assertSame($result, $actualResult); + } + + /** + * Tests generateResult with multimodal output. + * + * @return void + */ + public function testGenerateResultWithMultimodalOutput(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate multimodal'); + $builder->usingModel($model); + $builder->usingOutputModalities(ModalityEnum::text(), ModalityEnum::image()); + + $actualResult = $builder->generateResult(); + $this->assertSame($result, $actualResult); + } + + /** + * Tests generateResult throws exception when model doesn't support modality. + * + * @return void + */ + public function testGenerateResultThrowsExceptionForUnsupportedModality(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + // Model that only implements ModelInterface, not TextGenerationModelInterface + $model = $this->createMock(ModelInterface::class); + $model->method('metadata')->willReturn($metadata); + + $builder = new PromptBuilder($this->registry, 'Test prompt'); + $builder->usingModel($model); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('Model "test-model" does not support text generation'); + + $builder->generateResult(); + } + + /** + * Tests generateResult throws exception for unsupported output modality. + * + * @return void + */ + public function testGenerateResultThrowsExceptionForUnsupportedOutputModality(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ModelInterface::class); + $model->method('metadata')->willReturn($metadata); + + $builder = new PromptBuilder($this->registry, 'Test prompt'); + $builder->usingModel($model); + $builder->usingOutputModalities(ModalityEnum::video()); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('Output modality "video" is not yet supported'); + + $builder->generateResult(); + } + + /** + * Tests generateTextResult method. + * + * @return void + */ + public function testGenerateTextResult(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Test prompt'); + $builder->usingModel($model); + + $actualResult = $builder->generateTextResult(); + $this->assertSame($result, $actualResult); + + // Verify text modality was included + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $modalities = $config->getOutputModalities(); + $this->assertNotNull($modalities); + $this->assertTrue($modalities[0]->isText()); + } + + /** + * Tests generateImageResult method. + * + * @return void + */ + public function testGenerateImageResult(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ImageGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateImageResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate image'); + $builder->usingModel($model); + + $actualResult = $builder->generateImageResult(); + $this->assertSame($result, $actualResult); + + // Verify image modality was included + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $modalities = $config->getOutputModalities(); + $this->assertNotNull($modalities); + $this->assertTrue($modalities[0]->isImage()); + } + + /** + * Tests generateSpeechResult method. + * + * @return void + */ + public function testGenerateSpeechResult(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(SpeechGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateSpeechResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate speech'); + $builder->usingModel($model); + + $actualResult = $builder->generateSpeechResult(); + $this->assertSame($result, $actualResult); + + // Verify audio modality was included + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $modalities = $config->getOutputModalities(); + $this->assertNotNull($modalities); + $this->assertTrue($modalities[0]->isAudio()); + } + + /** + * Tests convertTextToSpeechResult method. + * + * @return void + */ + public function testConvertTextToSpeechResult(): void + { + $result = $this->createMock(GenerativeAiResult::class); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextToSpeechConversionModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('convertTextToSpeechResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Convert to speech'); + $builder->usingModel($model); + + $actualResult = $builder->convertTextToSpeechResult(); + $this->assertSame($result, $actualResult); + + // Verify audio modality was included + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $modalities = $config->getOutputModalities(); + $this->assertNotNull($modalities); + $this->assertTrue($modalities[0]->isAudio()); + } + + /** + * Tests convertTextToSpeechResult throws exception for unsupported model. + * + * @return void + */ + public function testConvertTextToSpeechResultThrowsExceptionForUnsupportedModel(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + // Model that doesn't implement TextToSpeechConversionModelInterface + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + + $builder = new PromptBuilder($this->registry, 'Convert to speech'); + $builder->usingModel($model); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('Model "test-model" does not support text-to-speech conversion'); + + $builder->convertTextToSpeechResult(); + } + + /** + * Tests generateText method. + * + * @return void + */ + public function testGenerateText(): void + { + $messagePart = new MessagePart('Generated text content'); + $message = new Message(MessageRoleEnum::model(), [$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate text'); + $builder->usingModel($model); + + $text = $builder->generateText(); + $this->assertEquals('Generated text content', $text); + } + + /** + * Tests generateText throws exception when no candidates. + * + * @return void + */ + public function testGenerateTextThrowsExceptionWhenNoCandidates(): void + { + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate text'); + $builder->usingModel($model); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('No candidates were generated'); + + $builder->generateText(); + } + + /** + * Tests generateText throws exception when message has no parts. + * + * @return void + */ + public function testGenerateTextThrowsExceptionWhenNoParts(): void + { + $message = new Message(MessageRoleEnum::model(), []); + $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate text'); + $builder->usingModel($model); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('Generated message contains no parts'); + + $builder->generateText(); + } + + /** + * Tests generateText throws exception when part has no text. + * + * @return void + */ + public function testGenerateTextThrowsExceptionWhenPartHasNoText(): void + { + $file = new File('https://example.com/image.jpg', 'image/jpeg'); + $messagePart = new MessagePart($file); + $message = new Message(MessageRoleEnum::model(), [$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate text'); + $builder->usingModel($model); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('Generated message part contains no text'); + + $builder->generateText(); + } + + /** + * Tests generateTexts method. + * + * @return void + */ + public function testGenerateTexts(): void + { + $candidates = [ + new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart('Text 1')]), + FinishReasonEnum::stop(), + 10 + ), + new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart('Text 2')]), + FinishReasonEnum::stop(), + 10 + ), + new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart('Text 3')]), + FinishReasonEnum::stop(), + 10 + ) + ]; + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn($candidates); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate texts'); + $builder->usingModel($model); + + $texts = $builder->generateTexts(3); + + $this->assertCount(3, $texts); + $this->assertEquals('Text 1', $texts[0]); + $this->assertEquals('Text 2', $texts[1]); + $this->assertEquals('Text 3', $texts[2]); + + // Verify candidate count was set + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $this->assertEquals(3, $config->getCandidateCount()); + } + + /** + * Tests generateTexts throws exception when no text generated. + * + * @return void + */ + public function testGenerateTextsThrowsExceptionWhenNoTextGenerated(): void + { + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateTextResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate texts'); + $builder->usingModel($model); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('No text was generated from any candidates'); + + $builder->generateTexts(); + } + + /** + * Tests generateImage method. + * + * @return void + */ + public function testGenerateImage(): void + { + $file = new File('https://example.com/generated.jpg', 'image/jpeg'); + $messagePart = new MessagePart($file); + $message = new Message(MessageRoleEnum::model(), [$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ImageGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateImageResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate image'); + $builder->usingModel($model); + + $generatedFile = $builder->generateImage(); + $this->assertSame($file, $generatedFile); + } + + /** + * Tests generateImage throws exception when no image file. + * + * @return void + */ + public function testGenerateImageThrowsExceptionWhenNoFile(): void + { + $messagePart = new MessagePart('Text instead of image'); + $message = new Message(MessageRoleEnum::model(), [$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ImageGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateImageResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate image'); + $builder->usingModel($model); + + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('Generated message part contains no image file'); + + $builder->generateImage(); + } + + /** + * Tests generateImages method. + * + * @return void + */ + public function testGenerateImages(): void + { + $files = [ + new File('https://example.com/img1.jpg', 'image/jpeg'), + new File('https://example.com/img2.jpg', 'image/jpeg'), + ]; + + $candidates = []; + foreach ($files as $file) { + $candidates[] = new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart($file)]), + FinishReasonEnum::stop(), + 10 + ); + } + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn($candidates); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ImageGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateImageResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate images'); + $builder->usingModel($model); + + $generatedFiles = $builder->generateImages(2); + + $this->assertCount(2, $generatedFiles); + $this->assertSame($files[0], $generatedFiles[0]); + $this->assertSame($files[1], $generatedFiles[1]); + } + + /** + * Tests convertTextToSpeech method. + * + * @return void + */ + public function testConvertTextToSpeech(): void + { + $file = new File('https://example.com/audio.mp3', 'audio/mp3'); + $messagePart = new MessagePart($file); + $message = new Message(MessageRoleEnum::model(), [$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextToSpeechConversionModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('convertTextToSpeechResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Convert this text'); + $builder->usingModel($model); + + $audioFile = $builder->convertTextToSpeech(); + $this->assertSame($file, $audioFile); + } + + /** + * Tests convertTextToSpeeches method. + * + * @return void + */ + public function testConvertTextToSpeeches(): void + { + $files = [ + new File('https://example.com/audio1.mp3', 'audio/mp3'), + new File('https://example.com/audio2.mp3', 'audio/mp3'), + ]; + + $candidates = []; + foreach ($files as $file) { + $candidates[] = new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart($file)]), + FinishReasonEnum::stop(), + 10 + ); + } + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn($candidates); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(TextToSpeechConversionModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('convertTextToSpeechResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Convert this text'); + $builder->usingModel($model); + + $audioFiles = $builder->convertTextToSpeeches(2); + + $this->assertCount(2, $audioFiles); + $this->assertSame($files[0], $audioFiles[0]); + $this->assertSame($files[1], $audioFiles[1]); + } + + /** + * Tests generateSpeech method. + * + * @return void + */ + public function testGenerateSpeech(): void + { + $file = new File('https://example.com/speech.mp3', 'audio/mp3'); + $messagePart = new MessagePart($file); + $message = new Message(MessageRoleEnum::model(), [$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(SpeechGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateSpeechResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate speech'); + $builder->usingModel($model); + + $speechFile = $builder->generateSpeech(); + $this->assertSame($file, $speechFile); + } + + /** + * Tests generateSpeeches method. + * + * @return void + */ + public function testGenerateSpeeches(): void + { + $files = [ + new File('https://example.com/speech1.mp3', 'audio/mp3'), + new File('https://example.com/speech2.mp3', 'audio/mp3'), + new File('https://example.com/speech3.mp3', 'audio/mp3'), + ]; + + $candidates = []; + foreach ($files as $file) { + $candidates[] = new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart($file)]), + FinishReasonEnum::stop(), + 10 + ); + } + + $result = $this->createMock(GenerativeAiResult::class); + $result->method('getCandidates')->willReturn($candidates); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(SpeechGenerationModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->method('generateSpeechResult')->willReturn($result); + + $builder = new PromptBuilder($this->registry, 'Generate speech'); + $builder->usingModel($model); + + $speechFiles = $builder->generateSpeeches(3); + + $this->assertCount(3, $speechFiles); + $this->assertSame($files[0], $speechFiles[0]); + $this->assertSame($files[1], $speechFiles[1]); + $this->assertSame($files[2], $speechFiles[2]); + } + + /** + * Tests getConfiguredModel with explicitly set model. + * + * @return void + */ + public function testGetConfiguredModelWithExplicitModel(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createMock(ModelInterface::class); + $model->method('metadata')->willReturn($metadata); + $model->expects($this->once())->method('setConfig')->with($this->isInstanceOf(ModelConfig::class)); + + $builder = new PromptBuilder($this->registry, 'Test'); + $builder->usingModel($model); + + $reflection = new \ReflectionClass($builder); + $method = $reflection->getMethod('getConfiguredModel'); + $method->setAccessible(true); + + $configuredModel = $method->invoke($builder); + $this->assertSame($model, $configuredModel); + } + + /** + * Tests getConfiguredModel throws exception when model doesn't meet requirements. + * + * @return void + */ + public function testGetConfiguredModelThrowsExceptionWhenModelDoesntMeetRequirements(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('meetsRequirements')->willReturn(false); + $metadata->method('getId')->willReturn('incompatible-model'); + + $model = $this->createMock(ModelInterface::class); + $model->method('metadata')->willReturn($metadata); + + $builder = new PromptBuilder($this->registry, 'Test'); + $builder->usingModel($model); + + $reflection = new \ReflectionClass($builder); + $method = $reflection->getMethod('getConfiguredModel'); + $method->setAccessible(true); + + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('The selected model "incompatible-model" does not meet the required capabilities'); + + $method->invoke($builder); + } + + /** + * Tests getConfiguredModel finds model from registry. + * + * @return void + */ + public function testGetConfiguredModelFindsModelFromRegistry(): void + { + $modelMetadata = $this->createMock(ModelMetadata::class); + $modelMetadata->method('getId')->willReturn('found-model'); + + $providerMetadata = $this->createMock(ProviderMetadata::class); + $providerMetadata->method('getId')->willReturn('test-provider'); + + $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); + $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); + $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); + + $model = $this->createMock(ModelInterface::class); + + $this->registry->method('findModelsMetadataForSupport') + ->willReturn([$providerModelsMetadata]); + + $this->registry->method('getProviderModel') + ->with('test-provider', 'found-model', $this->isInstanceOf(ModelConfig::class)) + ->willReturn($model); + + $builder = new PromptBuilder($this->registry, 'Test'); + + $reflection = new \ReflectionClass($builder); + $method = $reflection->getMethod('getConfiguredModel'); + $method->setAccessible(true); + + $configuredModel = $method->invoke($builder); + $this->assertSame($model, $configuredModel); + } + + /** + * Tests getConfiguredModel throws exception when no models found. + * + * @return void + */ + public function testGetConfiguredModelThrowsExceptionWhenNoModelsFound(): void + { + $this->registry->method('findModelsMetadataForSupport')->willReturn([]); + + $builder = new PromptBuilder($this->registry, 'Test'); + + $reflection = new \ReflectionClass($builder); + $method = $reflection->getMethod('getConfiguredModel'); + $method->setAccessible(true); + + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('No models found that support the required capabilities'); + + $method->invoke($builder); + } + + /** + * Tests appendPartToMessages creates new user message when empty. + * + * @return void + */ + public function testAppendPartToMessagesCreatesNewUserMessage(): void + { + $builder = new PromptBuilder($this->registry); + + $reflection = new \ReflectionClass($builder); + $method = $reflection->getMethod('appendPartToMessages'); + $method->setAccessible(true); + + $part = new MessagePart('Test part'); + $method->invoke($builder, $part); + + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertEquals('Test part', $messages[0]->getParts()[0]->getText()); + } + + /** + * Tests appendPartToMessages appends to existing user message. + * + * @return void + */ + public function testAppendPartToMessagesAppendsToExistingUserMessage(): void + { + $builder = new PromptBuilder($this->registry, 'Initial'); + + $reflection = new \ReflectionClass($builder); + $method = $reflection->getMethod('appendPartToMessages'); + $method->setAccessible(true); + + $part = new MessagePart('Additional'); + $method->invoke($builder, $part); + + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $parts = $messages[0]->getParts(); + $this->assertCount(2, $parts); + $this->assertEquals('Initial', $parts[0]->getText()); + $this->assertEquals('Additional', $parts[1]->getText()); + } + + /** + * Tests appendPartToMessages creates new message when last is model message. + * + * @return void + */ + public function testAppendPartToMessagesCreatesNewMessageWhenLastIsModel(): void + { + $builder = new PromptBuilder($this->registry, [ + new UserMessage([new MessagePart('User')]), + new ModelMessage([new MessagePart('Model')]) + ]); + + $reflection = new \ReflectionClass($builder); + $method = $reflection->getMethod('appendPartToMessages'); + $method->setAccessible(true); + + $part = new MessagePart('New user message'); + $method->invoke($builder, $part); + + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(3, $messages); + $this->assertInstanceOf(UserMessage::class, $messages[2]); + $this->assertEquals('New user message', $messages[2]->getParts()[0]->getText()); + } + + /** + * Tests complex multimodal prompt building. + * + * @return void + */ + public function testComplexMultimodalPromptBuilding(): void + { + $file1 = new File('https://example.com/img1.jpg', 'image/jpeg'); + $file2 = new File('https://example.com/audio.mp3', 'audio/mp3'); + $functionResponse = new FunctionResponse('func1', 'getData', ['result' => 'data']); + + $builder = new PromptBuilder($this->registry); + $builder->withText('Analyze this data:') + ->withImageFile($file1) + ->withText(' and this audio:') + ->withAudioFile($file2) + ->withFunctionResponse($functionResponse) + ->withHistory( + new UserMessage([new MessagePart('Previous question')]), + new ModelMessage([new MessagePart('Previous answer')]) + ) + ->withText(' Final instruction'); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + // Should have 3 messages: 2 from history + 1 current + $this->assertCount(3, $messages); + + // Check history messages + $this->assertEquals('Previous question', $messages[0]->getParts()[0]->getText()); + $this->assertEquals('Previous answer', $messages[1]->getParts()[0]->getText()); + + // Check current message has all parts + $currentParts = $messages[2]->getParts(); + $this->assertCount(6, $currentParts); + $this->assertEquals('Analyze this data:', $currentParts[0]->getText()); + $this->assertSame($file1, $currentParts[1]->getFile()); + $this->assertEquals(' and this audio:', $currentParts[2]->getText()); + $this->assertSame($file2, $currentParts[3]->getFile()); + $this->assertSame($functionResponse, $currentParts[4]->getFunctionResponse()); + $this->assertEquals(' Final instruction', $currentParts[5]->getText()); + } + + /** + * Tests includeOutputModality preserves existing modalities. + * + * @return void + */ + public function testIncludeOutputModalityPreservesExisting(): void + { + $builder = new PromptBuilder($this->registry, 'Test'); + + // Set initial modality + $builder->usingOutputModalities(ModalityEnum::audio()); + + // Generate text should add text modality, not replace audio + $builder->generateTextResult(); + + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $modalities = $config->getOutputModalities(); + $this->assertCount(2, $modalities); + $this->assertTrue($modalities[0]->isAudio()); + $this->assertTrue($modalities[1]->isText()); + } + + /** + * Tests constructor with list of string parts. + * + * @return void + */ + public function testConstructorWithStringPartsList(): void + { + $builder = new PromptBuilder($this->registry, ['Part 1', 'Part 2', 'Part 3']); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $this->assertInstanceOf(UserMessage::class, $messages[0]); + $parts = $messages[0]->getParts(); + $this->assertCount(3, $parts); + $this->assertEquals('Part 1', $parts[0]->getText()); + $this->assertEquals('Part 2', $parts[1]->getText()); + $this->assertEquals('Part 3', $parts[2]->getText()); + } + + /** + * Tests constructor with mixed parts list. + * + * @return void + */ + public function testConstructorWithMixedPartsList(): void + { + $part1 = new MessagePart('Part 1'); + $part2Array = ['type' => 'text', 'text' => 'Part 2']; + + $builder = new PromptBuilder($this->registry, ['String part', $part1, $part2Array]); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + + $this->assertCount(1, $messages); + $parts = $messages[0]->getParts(); + $this->assertCount(3, $parts); + $this->assertEquals('String part', $parts[0]->getText()); + $this->assertEquals('Part 1', $parts[1]->getText()); + $this->assertEquals('Part 2', $parts[2]->getText()); + } + + /** + * Tests parseMessage with non-list array throws exception. + * + * @return void + */ + public function testParseMessageWithNonListArrayThrowsException(): void + { + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Array input must be a list array'); + + new PromptBuilder($this->registry, ['key' => 'value']); + } + + /** + * Tests parseMessage with invalid array item throws exception. + * + * @return void + */ + public function testParseMessageWithInvalidArrayItemThrowsException(): void + { + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('Array items must be strings, MessagePart instances, or MessagePartArrayShape'); + + new PromptBuilder($this->registry, ['valid string', 123, 'another string']); + } + + /** + * Tests getModelRequirements with all file types. + * + * @return void + */ + public function testGetModelRequirementsWithAllFileTypes(): void + { + $builder = new PromptBuilder($this->registry); + $builder->withText('Analyze:') + ->withRemoteImage('https://example.com/img.jpg', 'image/jpeg') + ->withAudioFile(new File('https://example.com/audio.mp3', 'audio/mp3')) + ->withVideoFile(new File('https://example.com/video.mp4', 'video/mp4')) + ->withImageFile(new File('https://example.com/doc.pdf', 'application/pdf')); + + $requirements = $builder->getModelRequirements(); + $options = $requirements->getRequiredOptions(); + + // Find input modalities + $inputModalities = null; + foreach ($options as $option) { + if ($option->getName() === OptionEnum::inputModalities()->value) { + $inputModalities = $option->getValue(); + break; + } + } + + $this->assertNotNull($inputModalities); + + // Check all modality types are present + $modalityTypes = []; + foreach ($inputModalities as $modality) { + $modalityTypes[] = $modality->value; + } + + $this->assertContains('text', $modalityTypes); + $this->assertContains('image', $modalityTypes); + $this->assertContains('audio', $modalityTypes); + $this->assertContains('video', $modalityTypes); + $this->assertContains('document', $modalityTypes); + } + + /** + * Tests getModelRequirements includes config options. + * + * @return void + */ + public function testGetModelRequirementsIncludesConfigOptions(): void + { + $builder = new PromptBuilder($this->registry, 'Test'); + $builder->usingMaxTokens(1000) + ->usingTemperature(0.7) + ->usingOutputModalities(ModalityEnum::text(), ModalityEnum::image()) + ->asJsonResponse(['type' => 'object']); + + $requirements = $builder->getModelRequirements(); + $options = $requirements->getRequiredOptions(); + + // Check that config options are included + $optionNames = array_map(function ($option) { + return $option->getName(); + }, $options); + + $this->assertContains(OptionEnum::maxTokens()->value, $optionNames); + $this->assertContains(OptionEnum::temperature()->value, $optionNames); + $this->assertContains(OptionEnum::outputModalities()->value, $optionNames); + $this->assertContains(OptionEnum::outputMimeType()->value, $optionNames); + $this->assertContains(OptionEnum::outputSchema()->value, $optionNames); + } + + /** + * Tests last message must have parts validation. + * + * @return void + */ + public function testValidateMessagesLastMessageMustHaveParts(): void + { + // Create a message with empty parts + $emptyMessage = new UserMessage([]); + + $builder = new PromptBuilder($this->registry, [ + new UserMessage([new MessagePart('First')]), + new ModelMessage([new MessagePart('Response')]), + $emptyMessage + ]); + + $this->expectException(InvalidArgumentException::class); + $this->expectExceptionMessage('The last message must have content parts'); + + $builder->generateResult(); + } + + /** + * Tests generateImageResult method creates proper operation. + * + * @return void + */ + public function testGenerateImageResultCreatesProperOperation(): void + { + $operation = $this->createMock(OperationInterface::class); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->expects($this->once()) + ->method('generate') + ->with( + $this->isInstanceOf(Prompt::class), + $this->callback(function ($config) { + $modalities = $config->getOutputModalities(); + return count($modalities) === 1 && $modalities[0]->isImage(); + }) + ) + ->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate an image'); + $builder->usingModel($model); + + $result = $builder->generateImageResult(); + $this->assertSame($operation, $result); + } + + /** + * Tests generateAudioResult method creates proper operation. + * + * @return void + */ + public function testGenerateAudioResultCreatesProperOperation(): void + { + $operation = $this->createMock(OperationInterface::class); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->expects($this->once()) + ->method('generate') + ->with( + $this->isInstanceOf(Prompt::class), + $this->callback(function ($config) { + $modalities = $config->getOutputModalities(); + return count($modalities) === 1 && $modalities[0]->isAudio(); + }) + ) + ->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate audio'); + $builder->usingModel($model); + + $result = $builder->generateAudioResult(); + $this->assertSame($operation, $result); + } + + /** + * Tests generateVideoResult method creates proper operation. + * + * @return void + */ + public function testGenerateVideoResultCreatesProperOperation(): void + { + $operation = $this->createMock(OperationInterface::class); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->expects($this->once()) + ->method('generate') + ->with( + $this->isInstanceOf(Prompt::class), + $this->callback(function ($config) { + $modalities = $config->getOutputModalities(); + return count($modalities) === 1 && $modalities[0]->isVideo(); + }) + ) + ->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate video'); + $builder->usingModel($model); + + $result = $builder->generateVideoResult(); + $this->assertSame($operation, $result); + } + + /** + * Tests generateImage shorthand method returns file directly. + * + * @return void + */ + public function testGenerateImageReturnsFileDirectly(): void + { + $file = new File('https://example.com/generated.jpg', 'image/jpeg'); + + $candidate = $this->createMock(CandidateInterface::class); + $candidate->method('getPart')->willReturn($file); + + $result = $this->createMock(GenerativeAiResultInterface::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $operation = $this->createMock(OperationInterface::class); + $operation->method('getResult')->willReturn($result); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->method('generate')->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate an image'); + $builder->usingModel($model); + + $generatedFile = $builder->generateImage(); + $this->assertSame($file, $generatedFile); + } + + /** + * Tests generateAudio shorthand method returns file directly. + * + * @return void + */ + public function testGenerateAudioReturnsFileDirectly(): void + { + $file = new File('https://example.com/generated.mp3', 'audio/mp3'); + + $candidate = $this->createMock(CandidateInterface::class); + $candidate->method('getPart')->willReturn($file); + + $result = $this->createMock(GenerativeAiResultInterface::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $operation = $this->createMock(OperationInterface::class); + $operation->method('getResult')->willReturn($result); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->method('generate')->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate audio'); + $builder->usingModel($model); + + $generatedFile = $builder->generateAudio(); + $this->assertSame($file, $generatedFile); + } + + /** + * Tests generateVideo shorthand method returns file directly. + * + * @return void + */ + public function testGenerateVideoReturnsFileDirectly(): void + { + $file = new File('https://example.com/generated.mp4', 'video/mp4'); + + $candidate = $this->createMock(CandidateInterface::class); + $candidate->method('getPart')->willReturn($file); + + $result = $this->createMock(GenerativeAiResultInterface::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $operation = $this->createMock(OperationInterface::class); + $operation->method('getResult')->willReturn($result); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->method('generate')->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate video'); + $builder->usingModel($model); + + $generatedFile = $builder->generateVideo(); + $this->assertSame($file, $generatedFile); + } + + /** + * Tests generation method with multiple output modalities. + * + * @return void + */ + public function testGenerationWithMultipleOutputModalities(): void + { + $operation = $this->createMock(OperationInterface::class); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->expects($this->once()) + ->method('generate') + ->with( + $this->isInstanceOf(Prompt::class), + $this->callback(function ($config) { + $modalities = $config->getOutputModalities(); + return count($modalities) === 3 && + $modalities[0]->isText() && + $modalities[1]->isImage() && + $modalities[2]->isAudio(); + }) + ) + ->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate multimodal'); + $builder->usingModel($model) + ->usingOutputModalities( + ModalityEnum::text(), + ModalityEnum::image(), + ModalityEnum::audio() + ); + + $result = $builder->generateResult(); + $this->assertSame($operation, $result); + } + + /** + * Tests streaming generation methods. + * + * @return void + */ + public function testStreamingGenerationMethods(): void + { + $streamingOperation = $this->createMock(OperationInterface::class); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->expects($this->exactly(4)) + ->method('generateStream') + ->with( + $this->isInstanceOf(Prompt::class), + $this->isInstanceOf(ModelConfig::class) + ) + ->willReturn($streamingOperation); + + $builder = new PromptBuilder($this->registry, 'Stream content'); + $builder->usingModel($model); + + // Test all streaming methods + $this->assertSame($streamingOperation, $builder->streamResult()); + $this->assertSame($streamingOperation, $builder->streamTextResult()); + $this->assertSame($streamingOperation, $builder->streamImageResult()); + $this->assertSame($streamingOperation, $builder->streamAudioResult()); + } + + /** + * Tests generateText with no candidates throws exception. + * + * @return void + */ + public function testGenerateTextWithNoCandidatesThrowsException(): void + { + $result = $this->createMock(GenerativeAiResultInterface::class); + $result->method('getCandidates')->willReturn([]); + + $operation = $this->createMock(OperationInterface::class); + $operation->method('getResult')->willReturn($result); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->method('generate')->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate text'); + $builder->usingModel($model); + + $this->expectException(\RuntimeException::class); + $this->expectExceptionMessage('No candidates returned from generation'); + + $builder->generateText(); + } + + /** + * Tests generateText with non-string part throws exception. + * + * @return void + */ + public function testGenerateTextWithNonStringPartThrowsException(): void + { + $file = new File('https://example.com/file.jpg', 'image/jpeg'); + + $candidate = $this->createMock(CandidateInterface::class); + $candidate->method('getPart')->willReturn($file); + + $result = $this->createMock(GenerativeAiResultInterface::class); + $result->method('getCandidates')->willReturn([$candidate]); + + $operation = $this->createMock(OperationInterface::class); + $operation->method('getResult')->willReturn($result); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->method('generate')->willReturn($operation); + + $builder = new PromptBuilder($this->registry, 'Generate text'); + $builder->usingModel($model); + + $this->expectException(\RuntimeException::class); + $this->expectExceptionMessage('Expected string part but got different type'); + + $builder->generateText(); + } + + /** + * Tests chain generation with multiple prompts. + * + * @return void + */ + public function testChainGenerationWithMultiplePrompts(): void + { + // First generation + $firstCandidate = $this->createMock(CandidateInterface::class); + $firstCandidate->method('getPart')->willReturn('First response'); + + $firstResult = $this->createMock(GenerativeAiResultInterface::class); + $firstResult->method('getCandidates')->willReturn([$firstCandidate]); + + $firstOperation = $this->createMock(OperationInterface::class); + $firstOperation->method('getResult')->willReturn($firstResult); + + // Second generation + $secondCandidate = $this->createMock(CandidateInterface::class); + $secondCandidate->method('getPart')->willReturn('Second response'); + + $secondResult = $this->createMock(GenerativeAiResultInterface::class); + $secondResult->method('getCandidates')->willReturn([$secondCandidate]); + + $secondOperation = $this->createMock(OperationInterface::class); + $secondOperation->method('getResult')->willReturn($secondResult); + + $model = $this->createMock(GenerativeModelInterface::class); + $model->expects($this->exactly(2)) + ->method('generate') + ->willReturnOnConsecutiveCalls($firstOperation, $secondOperation); + + $builder = new PromptBuilder($this->registry, 'First prompt'); + $builder->usingModel($model); + + $firstText = $builder->generateText(); + $this->assertEquals('First response', $firstText); + + // Continue with second prompt + $builder->withModelResponse($firstText) + ->withText('Second prompt'); + + $secondText = $builder->generateText(); + $this->assertEquals('Second response', $secondText); + } +} \ No newline at end of file From 29b59fed26b5901d2c500c1fbdc235b4373545d1 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 15:18:43 -0600 Subject: [PATCH 21/47] test: fixes failing builder tests --- tests/unit/Builders/PromptBuilderTest.php | 862 +++++++++++++--------- 1 file changed, 505 insertions(+), 357 deletions(-) diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index d2bb0c20..10ee15af 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -4,6 +4,7 @@ namespace WordPress\AiClient\Tests\unit\Builders; +use Generator; use InvalidArgumentException; use PHPUnit\Framework\TestCase; use RuntimeException; @@ -19,7 +20,7 @@ use WordPress\AiClient\Providers\Models\DTO\ModelConfig; use WordPress\AiClient\Providers\Models\DTO\ModelMetadata; use WordPress\AiClient\Providers\Models\DTO\ModelRequirements; -use WordPress\AiClient\Providers\Models\DTO\ProviderModelsMetadata; +use WordPress\AiClient\Providers\DTO\ProviderModelsMetadata; use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; use WordPress\AiClient\Providers\Models\Enums\OptionEnum; use WordPress\AiClient\Providers\Models\ImageGeneration\Contracts\ImageGenerationModelInterface; @@ -33,6 +34,8 @@ use WordPress\AiClient\Results\DTO\TokenUsage; use WordPress\AiClient\Results\Enums\FinishReasonEnum; use WordPress\AiClient\Tools\DTO\FunctionResponse; +use WordPress\AiClient\Operations\Contracts\OperationInterface; +use WordPress\AiClient\Results\Contracts\ResultInterface; /** * @covers \WordPress\AiClient\Builders\PromptBuilder @@ -43,6 +46,183 @@ class PromptBuilderTest extends TestCase * @var ProviderRegistry */ private ProviderRegistry $registry; + + /** + * Creates a mock model that implements both ModelInterface and TextGenerationModelInterface. + * + * @param ModelMetadata $metadata The metadata for the model. + * @param GenerativeAiResult $result The result to return from generation. + * @return ModelInterface&TextGenerationModelInterface The mock model. + */ + private function createTextGenerationModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface + { + return new class($metadata, $result) implements ModelInterface, TextGenerationModelInterface { + private ModelMetadata $metadata; + private GenerativeAiResult $result; + private ModelConfig $config; + + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) + { + $this->metadata = $metadata; + $this->result = $result; + $this->config = new ModelConfig(); + } + + public function metadata(): ModelMetadata + { + return $this->metadata; + } + + public function setConfig(ModelConfig $config): void + { + $this->config = $config; + } + + public function getConfig(): ModelConfig + { + return $this->config; + } + + public function generateTextResult(array $prompt): GenerativeAiResult + { + return $this->result; + } + + public function streamGenerateTextResult(array $prompt): Generator + { + yield $this->result; + } + }; + } + + /** + * Creates a mock model that implements both ModelInterface and ImageGenerationModelInterface. + * + * @param ModelMetadata $metadata The metadata for the model. + * @param GenerativeAiResult $result The result to return from generation. + * @return ModelInterface&ImageGenerationModelInterface The mock model. + */ + private function createImageGenerationModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface + { + return new class($metadata, $result) implements ModelInterface, ImageGenerationModelInterface { + private ModelMetadata $metadata; + private GenerativeAiResult $result; + private ModelConfig $config; + + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) + { + $this->metadata = $metadata; + $this->result = $result; + $this->config = new ModelConfig(); + } + + public function metadata(): ModelMetadata + { + return $this->metadata; + } + + public function setConfig(ModelConfig $config): void + { + $this->config = $config; + } + + public function getConfig(): ModelConfig + { + return $this->config; + } + + public function generateImageResult(array $prompt): GenerativeAiResult + { + return $this->result; + } + }; + } + + /** + * Creates a mock model that implements both ModelInterface and SpeechGenerationModelInterface. + * + * @param ModelMetadata $metadata The metadata for the model. + * @param GenerativeAiResult $result The result to return from generation. + * @return ModelInterface&SpeechGenerationModelInterface The mock model. + */ + private function createSpeechGenerationModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface + { + return new class($metadata, $result) implements ModelInterface, SpeechGenerationModelInterface { + private ModelMetadata $metadata; + private GenerativeAiResult $result; + private ModelConfig $config; + + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) + { + $this->metadata = $metadata; + $this->result = $result; + $this->config = new ModelConfig(); + } + + public function metadata(): ModelMetadata + { + return $this->metadata; + } + + public function setConfig(ModelConfig $config): void + { + $this->config = $config; + } + + public function getConfig(): ModelConfig + { + return $this->config; + } + + public function generateSpeechResult(array $prompt): GenerativeAiResult + { + return $this->result; + } + }; + } + + /** + * Creates a mock model that implements both ModelInterface and TextToSpeechConversionModelInterface. + * + * @param ModelMetadata $metadata The metadata for the model. + * @param GenerativeAiResult $result The result to return from generation. + * @return ModelInterface&TextToSpeechConversionModelInterface The mock model. + */ + private function createTextToSpeechModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface + { + return new class($metadata, $result) implements ModelInterface, TextToSpeechConversionModelInterface { + private ModelMetadata $metadata; + private GenerativeAiResult $result; + private ModelConfig $config; + + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) + { + $this->metadata = $metadata; + $this->result = $result; + $this->config = new ModelConfig(); + } + + public function metadata(): ModelMetadata + { + return $this->metadata; + } + + public function setConfig(ModelConfig $config): void + { + $this->config = $config; + } + + public function getConfig(): ModelConfig + { + return $this->config; + } + + public function convertTextToSpeechResult(array $prompt): GenerativeAiResult + { + return $this->result; + } + }; + } /** * Sets up test fixtures. @@ -83,10 +263,13 @@ public function testConstructorWithStringPrompt(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ + /** @var list $messages */ + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); - $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Hello, world!', $messages[0]->getParts()[0]->getText()); } @@ -103,10 +286,13 @@ public function testConstructorWithMessagePartPrompt(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ + /** @var list $messages */ + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); - $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Test message', $messages[0]->getParts()[0]->getText()); } @@ -123,6 +309,8 @@ public function testConstructorWithMessagePrompt(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -146,6 +334,7 @@ public function testConstructorWithMessagesList(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $actualMessages */ $actualMessages = $messagesProperty->getValue($builder); $this->assertCount(3, $actualMessages); @@ -170,10 +359,12 @@ public function testConstructorWithMessageArrayShape(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); - $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Hello from array', $messages[0]->getParts()[0]->getText()); } @@ -192,6 +383,7 @@ public function testWithText(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -211,6 +403,7 @@ public function testWithTextAppendsToExistingUserMessage(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -236,12 +429,13 @@ public function testWithInlineImage(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); $file = $messages[0]->getParts()[0]->getFile(); $this->assertInstanceOf(File::class, $file); - $this->assertEquals('data:image/png;base64,' . $base64, $file->getUri()); + $this->assertEquals('data:image/png;base64,' . $base64, $file->getDataUri()); $this->assertEquals('image/png', $file->getMimeType()); } @@ -260,12 +454,13 @@ public function testWithRemoteImage(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); $file = $messages[0]->getParts()[0]->getFile(); $this->assertInstanceOf(File::class, $file); - $this->assertEquals('https://example.com/image.jpg', $file->getUri()); + $this->assertEquals('https://example.com/image.jpg', $file->getUrl()); $this->assertEquals('image/jpeg', $file->getMimeType()); } @@ -285,6 +480,7 @@ public function testWithImageFile(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -307,6 +503,7 @@ public function testWithAudioFile(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -329,6 +526,7 @@ public function testWithVideoFile(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -351,6 +549,7 @@ public function testWithFunctionResponse(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -376,6 +575,7 @@ public function testWithMessageParts(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -407,6 +607,7 @@ public function testWithHistory(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(3, $messages); @@ -432,7 +633,9 @@ public function testUsingModel(): void $modelProperty = $reflection->getProperty('model'); $modelProperty->setAccessible(true); - $this->assertSame($model, $modelProperty->getValue($builder)); + /** @var ModelInterface $actualModel */ + $actualModel = $modelProperty->getValue($builder); + $this->assertSame($model, $actualModel); } /** @@ -452,7 +655,9 @@ public function testUsingRegistry(): void $registryProperty = $reflection->getProperty('registry'); $registryProperty->setAccessible(true); - $this->assertSame($newRegistry, $registryProperty->getValue($builder)); + /** @var ProviderRegistry $actualRegistry */ + $actualRegistry = $registryProperty->getValue($builder); + $this->assertSame($newRegistry, $actualRegistry); } /** @@ -470,6 +675,7 @@ public function testUsingSystemInstruction(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals('You are a helpful assistant.', $config->getSystemInstruction()); @@ -490,6 +696,7 @@ public function testUsingMaxTokens(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals(1000, $config->getMaxTokens()); @@ -510,6 +717,7 @@ public function testUsingTemperature(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals(0.7, $config->getTemperature()); @@ -530,6 +738,7 @@ public function testUsingTopP(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals(0.9, $config->getTopP()); @@ -550,6 +759,7 @@ public function testUsingTopK(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals(40, $config->getTopK()); @@ -570,6 +780,7 @@ public function testUsingStopSequences(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $customOptions = $config->getCustomOptions(); @@ -592,6 +803,7 @@ public function testUsingCandidateCount(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals(3, $config->getCandidateCount()); @@ -612,6 +824,7 @@ public function testUsingOutputMime(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals('application/json', $config->getOutputMimeType()); @@ -639,6 +852,7 @@ public function testUsingOutputSchema(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals($schema, $config->getOutputSchema()); @@ -662,6 +876,7 @@ public function testUsingOutputModalities(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $modalities = $config->getOutputModalities(); @@ -685,6 +900,7 @@ public function testAsJsonResponse(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals('application/json', $config->getOutputMimeType()); @@ -706,6 +922,7 @@ public function testAsJsonResponseWithSchema(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals('application/json', $config->getOutputMimeType()); @@ -985,16 +1202,20 @@ public function testMethodChaining(): void $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); $this->assertCount(2, $messages[0]->getParts()); // Text and image $modelProperty = $reflection->getProperty('model'); $modelProperty->setAccessible(true); - $this->assertSame($model, $modelProperty->getValue($builder)); + /** @var ModelInterface $actualModel */ + $actualModel = $modelProperty->getValue($builder); + $this->assertSame($model, $actualModel); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals('Be helpful', $config->getSystemInstruction()); @@ -1019,9 +1240,7 @@ public function testGenerateResultWithTextModality(): void $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); @@ -1037,15 +1256,17 @@ public function testGenerateResultWithTextModality(): void */ public function testGenerateResultWithImageModality(): void { - $result = $this->createMock(GenerativeAiResult::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart(new File('', 'image/png'))]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(ImageGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateImageResult')->willReturn($result); + $model = $this->createImageGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate an image'); $builder->usingModel($model); @@ -1062,15 +1283,17 @@ public function testGenerateResultWithImageModality(): void */ public function testGenerateResultWithAudioModality(): void { - $result = $this->createMock(GenerativeAiResult::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(SpeechGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateSpeechResult')->willReturn($result); + $model = $this->createSpeechGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); @@ -1087,15 +1310,17 @@ public function testGenerateResultWithAudioModality(): void */ public function testGenerateResultWithMultimodalOutput(): void { - $result = $this->createMock(GenerativeAiResult::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart('Generated text')]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate multimodal'); $builder->usingModel($model); @@ -1160,15 +1385,17 @@ public function testGenerateResultThrowsExceptionForUnsupportedOutputModality(): */ public function testGenerateTextResult(): void { - $result = $this->createMock(GenerativeAiResult::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart('Generated text')]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); @@ -1180,6 +1407,7 @@ public function testGenerateTextResult(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $modalities = $config->getOutputModalities(); @@ -1194,15 +1422,17 @@ public function testGenerateTextResult(): void */ public function testGenerateImageResult(): void { - $result = $this->createMock(GenerativeAiResult::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart(new File('', 'image/png'))]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(ImageGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateImageResult')->willReturn($result); + $model = $this->createImageGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate image'); $builder->usingModel($model); @@ -1214,6 +1444,7 @@ public function testGenerateImageResult(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $modalities = $config->getOutputModalities(); @@ -1228,15 +1459,17 @@ public function testGenerateImageResult(): void */ public function testGenerateSpeechResult(): void { - $result = $this->createMock(GenerativeAiResult::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(SpeechGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateSpeechResult')->willReturn($result); + $model = $this->createSpeechGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); @@ -1248,6 +1481,7 @@ public function testGenerateSpeechResult(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $modalities = $config->getOutputModalities(); @@ -1262,15 +1496,17 @@ public function testGenerateSpeechResult(): void */ public function testConvertTextToSpeechResult(): void { - $result = $this->createMock(GenerativeAiResult::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextToSpeechConversionModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('convertTextToSpeechResult')->willReturn($result); + $model = $this->createTextToSpeechModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Convert to speech'); $builder->usingModel($model); @@ -1282,6 +1518,7 @@ public function testConvertTextToSpeechResult(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $modalities = $config->getOutputModalities(); @@ -1301,7 +1538,7 @@ public function testConvertTextToSpeechResultThrowsExceptionForUnsupportedModel( $metadata->method('meetsRequirements')->willReturn(true); // Model that doesn't implement TextToSpeechConversionModelInterface - $model = $this->createMock(TextGenerationModelInterface::class); + $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); $builder = new PromptBuilder($this->registry, 'Convert to speech'); @@ -1321,19 +1558,16 @@ public function testConvertTextToSpeechResultThrowsExceptionForUnsupportedModel( public function testGenerateText(): void { $messagePart = new MessagePart('Generated text content'); - $message = new Message(MessageRoleEnum::model(), [$messagePart]); - $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + $message = new ModelMessage([$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop()); - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); @@ -1349,16 +1583,47 @@ public function testGenerateText(): void */ public function testGenerateTextThrowsExceptionWhenNoCandidates(): void { - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([]); - + // Since GenerativeAiResult constructor requires at least one candidate, + // we need to create a mock that throws an exception or test a different scenario $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = new class($metadata) implements ModelInterface, TextGenerationModelInterface { + private ModelMetadata $metadata; + private ModelConfig $config; + + public function __construct(ModelMetadata $metadata) + { + $this->metadata = $metadata; + $this->config = new ModelConfig(); + } + + public function metadata(): ModelMetadata + { + return $this->metadata; + } + + public function setConfig(ModelConfig $config): void + { + $this->config = $config; + } + + public function getConfig(): ModelConfig + { + return $this->config; + } + + public function generateTextResult(array $prompt): GenerativeAiResult + { + throw new RuntimeException('No candidates were generated'); + } + + public function streamGenerateTextResult(array $prompt): Generator + { + yield from []; + } + }; $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); @@ -1376,19 +1641,16 @@ public function testGenerateTextThrowsExceptionWhenNoCandidates(): void */ public function testGenerateTextThrowsExceptionWhenNoParts(): void { - $message = new Message(MessageRoleEnum::model(), []); - $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + $message = new ModelMessage([]); + $candidate = new Candidate($message, FinishReasonEnum::stop()); - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); @@ -1408,19 +1670,16 @@ public function testGenerateTextThrowsExceptionWhenPartHasNoText(): void { $file = new File('https://example.com/image.jpg', 'image/jpeg'); $messagePart = new MessagePart($file); - $message = new Message(MessageRoleEnum::model(), [$messagePart]); - $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + $message = new ModelMessage([$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop()); - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); @@ -1440,32 +1699,26 @@ public function testGenerateTexts(): void { $candidates = [ new Candidate( - new Message(MessageRoleEnum::model(), [new MessagePart('Text 1')]), - FinishReasonEnum::stop(), - 10 + new ModelMessage([new MessagePart('Text 1')]), + FinishReasonEnum::stop() ), new Candidate( - new Message(MessageRoleEnum::model(), [new MessagePart('Text 2')]), - FinishReasonEnum::stop(), - 10 + new ModelMessage([new MessagePart('Text 2')]), + FinishReasonEnum::stop() ), new Candidate( - new Message(MessageRoleEnum::model(), [new MessagePart('Text 3')]), - FinishReasonEnum::stop(), - 10 + new ModelMessage([new MessagePart('Text 3')]), + FinishReasonEnum::stop() ) ]; - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn($candidates); + $result = new GenerativeAiResult('test-result-id', $candidates, new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate texts'); $builder->usingModel($model); @@ -1481,6 +1734,7 @@ public function testGenerateTexts(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $this->assertEquals(3, $config->getCandidateCount()); @@ -1493,16 +1747,45 @@ public function testGenerateTexts(): void */ public function testGenerateTextsThrowsExceptionWhenNoTextGenerated(): void { - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([]); - $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateTextResult')->willReturn($result); + $model = new class($metadata) implements ModelInterface, TextGenerationModelInterface { + private ModelMetadata $metadata; + private ModelConfig $config; + + public function __construct(ModelMetadata $metadata) + { + $this->metadata = $metadata; + $this->config = new ModelConfig(); + } + + public function metadata(): ModelMetadata + { + return $this->metadata; + } + + public function setConfig(ModelConfig $config): void + { + $this->config = $config; + } + + public function getConfig(): ModelConfig + { + return $this->config; + } + + public function generateTextResult(array $prompt): GenerativeAiResult + { + throw new RuntimeException('No text was generated from any candidates'); + } + + public function streamGenerateTextResult(array $prompt): Generator + { + yield from []; + } + }; $builder = new PromptBuilder($this->registry, 'Generate texts'); $builder->usingModel($model); @@ -1522,19 +1805,16 @@ public function testGenerateImage(): void { $file = new File('https://example.com/generated.jpg', 'image/jpeg'); $messagePart = new MessagePart($file); - $message = new Message(MessageRoleEnum::model(), [$messagePart]); - $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + $message = new ModelMessage([$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop()); - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(ImageGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateImageResult')->willReturn($result); + $model = $this->createImageGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate image'); $builder->usingModel($model); @@ -1551,19 +1831,16 @@ public function testGenerateImage(): void public function testGenerateImageThrowsExceptionWhenNoFile(): void { $messagePart = new MessagePart('Text instead of image'); - $message = new Message(MessageRoleEnum::model(), [$messagePart]); - $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + $message = new ModelMessage([$messagePart]); + $candidate = new Candidate($message, FinishReasonEnum::stop()); - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(ImageGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateImageResult')->willReturn($result); + $model = $this->createImageGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate image'); $builder->usingModel($model); @@ -1590,21 +1867,17 @@ public function testGenerateImages(): void foreach ($files as $file) { $candidates[] = new Candidate( new Message(MessageRoleEnum::model(), [new MessagePart($file)]), - FinishReasonEnum::stop(), - 10 + FinishReasonEnum::stop() ); } - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn($candidates); + $result = new GenerativeAiResult('test-result-id', $candidates, new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(ImageGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateImageResult')->willReturn($result); + $model = $this->createImageGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate images'); $builder->usingModel($model); @@ -1626,18 +1899,15 @@ public function testConvertTextToSpeech(): void $file = new File('https://example.com/audio.mp3', 'audio/mp3'); $messagePart = new MessagePart($file); $message = new Message(MessageRoleEnum::model(), [$messagePart]); - $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + $candidate = new Candidate($message, FinishReasonEnum::stop()); - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextToSpeechConversionModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('convertTextToSpeechResult')->willReturn($result); + $model = $this->createTextToSpeechModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Convert this text'); $builder->usingModel($model); @@ -1662,21 +1932,17 @@ public function testConvertTextToSpeeches(): void foreach ($files as $file) { $candidates[] = new Candidate( new Message(MessageRoleEnum::model(), [new MessagePart($file)]), - FinishReasonEnum::stop(), - 10 + FinishReasonEnum::stop() ); } - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn($candidates); + $result = new GenerativeAiResult('test-result-id', $candidates, new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(TextToSpeechConversionModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('convertTextToSpeechResult')->willReturn($result); + $model = $this->createTextToSpeechModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Convert this text'); $builder->usingModel($model); @@ -1698,18 +1964,15 @@ public function testGenerateSpeech(): void $file = new File('https://example.com/speech.mp3', 'audio/mp3'); $messagePart = new MessagePart($file); $message = new Message(MessageRoleEnum::model(), [$messagePart]); - $candidate = new Candidate($message, FinishReasonEnum::stop(), 10); + $candidate = new Candidate($message, FinishReasonEnum::stop()); - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(SpeechGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateSpeechResult')->willReturn($result); + $model = $this->createSpeechGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); @@ -1740,16 +2003,17 @@ public function testGenerateSpeeches(): void ); } - $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn($candidates); + $result = new GenerativeAiResult( + 'test-result-id', + $candidates, + new TokenUsage(100, 50, 150) + ); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(SpeechGenerationModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->method('generateSpeechResult')->willReturn($result); + $model = $this->createSpeechGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); @@ -1788,18 +2052,20 @@ public function testGetConfiguredModelWithExplicitModel(): void } /** - * Tests getConfiguredModel throws exception when model doesn't meet requirements. + * Tests getConfiguredModel returns explicitly set model. * * @return void */ - public function testGetConfiguredModelThrowsExceptionWhenModelDoesntMeetRequirements(): void + public function testGetConfiguredModelReturnsExplicitlySetModel(): void { $metadata = $this->createMock(ModelMetadata::class); - $metadata->method('meetsRequirements')->willReturn(false); - $metadata->method('getId')->willReturn('incompatible-model'); + $metadata->method('getId')->willReturn('explicit-model'); $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); + $model->expects($this->once()) + ->method('setConfig') + ->with($this->isInstanceOf(ModelConfig::class)); $builder = new PromptBuilder($this->registry, 'Test'); $builder->usingModel($model); @@ -1808,10 +2074,8 @@ public function testGetConfiguredModelThrowsExceptionWhenModelDoesntMeetRequirem $method = $reflection->getMethod('getConfiguredModel'); $method->setAccessible(true); - $this->expectException(InvalidArgumentException::class); - $this->expectExceptionMessage('The selected model "incompatible-model" does not meet the required capabilities'); - - $method->invoke($builder); + $result = $method->invoke($builder); + $this->assertSame($model, $result); } /** @@ -1889,10 +2153,12 @@ public function testAppendPartToMessagesCreatesNewUserMessage(): void $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); - $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Test part', $messages[0]->getParts()[0]->getText()); } @@ -1914,6 +2180,7 @@ public function testAppendPartToMessagesAppendsToExistingUserMessage(): void $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -1944,10 +2211,11 @@ public function testAppendPartToMessagesCreatesNewMessageWhenLastIsModel(): void $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(3, $messages); - $this->assertInstanceOf(UserMessage::class, $messages[2]); + $this->assertInstanceOf(Message::class, $messages[2]); $this->assertEquals('New user message', $messages[2]->getParts()[0]->getText()); } @@ -1977,24 +2245,29 @@ public function testComplexMultimodalPromptBuilding(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - // Should have 3 messages: 2 from history + 1 current - $this->assertCount(3, $messages); + // Should have 4 messages: 1 initial + 2 from history + 1 final + $this->assertCount(4, $messages); + + // Check first message with initial content + $firstParts = $messages[0]->getParts(); + $this->assertCount(5, $firstParts); // text, image, text, audio, function response + $this->assertEquals('Analyze this data:', $firstParts[0]->getText()); + $this->assertSame($file1, $firstParts[1]->getFile()); + $this->assertEquals(' and this audio:', $firstParts[2]->getText()); + $this->assertSame($file2, $firstParts[3]->getFile()); + $this->assertSame($functionResponse, $firstParts[4]->getFunctionResponse()); // Check history messages - $this->assertEquals('Previous question', $messages[0]->getParts()[0]->getText()); - $this->assertEquals('Previous answer', $messages[1]->getParts()[0]->getText()); + $this->assertEquals('Previous question', $messages[1]->getParts()[0]->getText()); + $this->assertEquals('Previous answer', $messages[2]->getParts()[0]->getText()); - // Check current message has all parts - $currentParts = $messages[2]->getParts(); - $this->assertCount(6, $currentParts); - $this->assertEquals('Analyze this data:', $currentParts[0]->getText()); - $this->assertSame($file1, $currentParts[1]->getFile()); - $this->assertEquals(' and this audio:', $currentParts[2]->getText()); - $this->assertSame($file2, $currentParts[3]->getFile()); - $this->assertSame($functionResponse, $currentParts[4]->getFunctionResponse()); - $this->assertEquals(' Final instruction', $currentParts[5]->getText()); + // Check final message + $finalParts = $messages[3]->getParts(); + $this->assertCount(1, $finalParts); + $this->assertEquals(' Final instruction', $finalParts[0]->getText()); } /** @@ -2004,7 +2277,20 @@ public function testComplexMultimodalPromptBuilding(): void */ public function testIncludeOutputModalityPreservesExisting(): void { + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart('Generated text')]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); + + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createTextGenerationModel($metadata, $result); + $builder = new PromptBuilder($this->registry, 'Test'); + $builder->usingModel($model); // Set initial modality $builder->usingOutputModalities(ModalityEnum::audio()); @@ -2015,6 +2301,7 @@ public function testIncludeOutputModalityPreservesExisting(): void $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); + /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); $modalities = $config->getOutputModalities(); @@ -2035,10 +2322,12 @@ public function testConstructorWithStringPartsList(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); - $this->assertInstanceOf(UserMessage::class, $messages[0]); + $this->assertInstanceOf(Message::class, $messages[0]); $parts = $messages[0]->getParts(); $this->assertCount(3, $parts); $this->assertEquals('Part 1', $parts[0]->getText()); @@ -2061,6 +2350,7 @@ public function testConstructorWithMixedPartsList(): void $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); + /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); @@ -2195,25 +2485,34 @@ public function testValidateMessagesLastMessageMustHaveParts(): void */ public function testGenerateImageResultCreatesProperOperation(): void { - $operation = $this->createMock(OperationInterface::class); + $result = new GenerativeAiResult( + 'test-result', + [new Candidate(new ModelMessage([new MessagePart(new File('', 'image/png'))]), FinishReasonEnum::stop())], + new TokenUsage(100, 50, 150) + ); - $model = $this->createMock(GenerativeModelInterface::class); - $model->expects($this->once()) - ->method('generate') - ->with( - $this->isInstanceOf(Prompt::class), - $this->callback(function ($config) { - $modalities = $config->getOutputModalities(); - return count($modalities) === 1 && $modalities[0]->isImage(); - }) - ) - ->willReturn($operation); + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); + + $model = $this->createImageGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate an image'); $builder->usingModel($model); - $result = $builder->generateImageResult(); - $this->assertSame($operation, $result); + $actualResult = $builder->generateImageResult(); + $this->assertSame($result, $actualResult); + + // Verify that image modality was included in the model config + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + /** @var ModelConfig $config */ + $config = $configProperty->getValue($builder); + + $outputModalities = $config->getOutputModalities(); + $this->assertCount(1, $outputModalities); + $this->assertTrue($outputModalities[0]->isImage()); } /** @@ -2223,25 +2522,7 @@ public function testGenerateImageResultCreatesProperOperation(): void */ public function testGenerateAudioResultCreatesProperOperation(): void { - $operation = $this->createMock(OperationInterface::class); - - $model = $this->createMock(GenerativeModelInterface::class); - $model->expects($this->once()) - ->method('generate') - ->with( - $this->isInstanceOf(Prompt::class), - $this->callback(function ($config) { - $modalities = $config->getOutputModalities(); - return count($modalities) === 1 && $modalities[0]->isAudio(); - }) - ) - ->willReturn($operation); - - $builder = new PromptBuilder($this->registry, 'Generate audio'); - $builder->usingModel($model); - - $result = $builder->generateAudioResult(); - $this->assertSame($operation, $result); + $this->markTestSkipped('generateAudioResult method does not exist yet'); } /** @@ -2251,25 +2532,7 @@ public function testGenerateAudioResultCreatesProperOperation(): void */ public function testGenerateVideoResultCreatesProperOperation(): void { - $operation = $this->createMock(OperationInterface::class); - - $model = $this->createMock(GenerativeModelInterface::class); - $model->expects($this->once()) - ->method('generate') - ->with( - $this->isInstanceOf(Prompt::class), - $this->callback(function ($config) { - $modalities = $config->getOutputModalities(); - return count($modalities) === 1 && $modalities[0]->isVideo(); - }) - ) - ->willReturn($operation); - - $builder = new PromptBuilder($this->registry, 'Generate video'); - $builder->usingModel($model); - - $result = $builder->generateVideoResult(); - $this->assertSame($operation, $result); + $this->markTestSkipped('generateVideoResult method does not exist yet'); } /** @@ -2280,18 +2543,18 @@ public function testGenerateVideoResultCreatesProperOperation(): void public function testGenerateImageReturnsFileDirectly(): void { $file = new File('https://example.com/generated.jpg', 'image/jpeg'); + $candidate = new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart($file)]), + FinishReasonEnum::stop() + ); - $candidate = $this->createMock(CandidateInterface::class); - $candidate->method('getPart')->willReturn($file); - - $result = $this->createMock(GenerativeAiResultInterface::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - $operation = $this->createMock(OperationInterface::class); - $operation->method('getResult')->willReturn($result); + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(GenerativeModelInterface::class); - $model->method('generate')->willReturn($operation); + $model = $this->createImageGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate an image'); $builder->usingModel($model); @@ -2307,25 +2570,7 @@ public function testGenerateImageReturnsFileDirectly(): void */ public function testGenerateAudioReturnsFileDirectly(): void { - $file = new File('https://example.com/generated.mp3', 'audio/mp3'); - - $candidate = $this->createMock(CandidateInterface::class); - $candidate->method('getPart')->willReturn($file); - - $result = $this->createMock(GenerativeAiResultInterface::class); - $result->method('getCandidates')->willReturn([$candidate]); - - $operation = $this->createMock(OperationInterface::class); - $operation->method('getResult')->willReturn($result); - - $model = $this->createMock(GenerativeModelInterface::class); - $model->method('generate')->willReturn($operation); - - $builder = new PromptBuilder($this->registry, 'Generate audio'); - $builder->usingModel($model); - - $generatedFile = $builder->generateAudio(); - $this->assertSame($file, $generatedFile); + $this->markTestSkipped('generateAudio method does not exist yet'); } /** @@ -2335,25 +2580,7 @@ public function testGenerateAudioReturnsFileDirectly(): void */ public function testGenerateVideoReturnsFileDirectly(): void { - $file = new File('https://example.com/generated.mp4', 'video/mp4'); - - $candidate = $this->createMock(CandidateInterface::class); - $candidate->method('getPart')->willReturn($file); - - $result = $this->createMock(GenerativeAiResultInterface::class); - $result->method('getCandidates')->willReturn([$candidate]); - - $operation = $this->createMock(OperationInterface::class); - $operation->method('getResult')->willReturn($result); - - $model = $this->createMock(GenerativeModelInterface::class); - $model->method('generate')->willReturn($operation); - - $builder = new PromptBuilder($this->registry, 'Generate video'); - $builder->usingModel($model); - - $generatedFile = $builder->generateVideo(); - $this->assertSame($file, $generatedFile); + $this->markTestSkipped('generateVideo method does not exist yet'); } /** @@ -2363,33 +2590,7 @@ public function testGenerateVideoReturnsFileDirectly(): void */ public function testGenerationWithMultipleOutputModalities(): void { - $operation = $this->createMock(OperationInterface::class); - - $model = $this->createMock(GenerativeModelInterface::class); - $model->expects($this->once()) - ->method('generate') - ->with( - $this->isInstanceOf(Prompt::class), - $this->callback(function ($config) { - $modalities = $config->getOutputModalities(); - return count($modalities) === 3 && - $modalities[0]->isText() && - $modalities[1]->isImage() && - $modalities[2]->isAudio(); - }) - ) - ->willReturn($operation); - - $builder = new PromptBuilder($this->registry, 'Generate multimodal'); - $builder->usingModel($model) - ->usingOutputModalities( - ModalityEnum::text(), - ModalityEnum::image(), - ModalityEnum::audio() - ); - - $result = $builder->generateResult(); - $this->assertSame($operation, $result); + $this->markTestSkipped('Operations-based generation not implemented yet'); } /** @@ -2399,25 +2600,7 @@ public function testGenerationWithMultipleOutputModalities(): void */ public function testStreamingGenerationMethods(): void { - $streamingOperation = $this->createMock(OperationInterface::class); - - $model = $this->createMock(GenerativeModelInterface::class); - $model->expects($this->exactly(4)) - ->method('generateStream') - ->with( - $this->isInstanceOf(Prompt::class), - $this->isInstanceOf(ModelConfig::class) - ) - ->willReturn($streamingOperation); - - $builder = new PromptBuilder($this->registry, 'Stream content'); - $builder->usingModel($model); - - // Test all streaming methods - $this->assertSame($streamingOperation, $builder->streamResult()); - $this->assertSame($streamingOperation, $builder->streamTextResult()); - $this->assertSame($streamingOperation, $builder->streamImageResult()); - $this->assertSame($streamingOperation, $builder->streamAudioResult()); + $this->markTestSkipped('Streaming methods do not exist yet'); } /** @@ -2427,20 +2610,21 @@ public function testStreamingGenerationMethods(): void */ public function testGenerateTextWithNoCandidatesThrowsException(): void { - $result = $this->createMock(GenerativeAiResultInterface::class); + // Create a mock result that returns empty candidates + $result = $this->createMock(GenerativeAiResult::class); $result->method('getCandidates')->willReturn([]); - $operation = $this->createMock(OperationInterface::class); - $operation->method('getResult')->willReturn($result); + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(GenerativeModelInterface::class); - $model->method('generate')->willReturn($operation); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - $this->expectException(\RuntimeException::class); - $this->expectExceptionMessage('No candidates returned from generation'); + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('No candidates were generated'); $builder->generateText(); } @@ -2453,24 +2637,24 @@ public function testGenerateTextWithNoCandidatesThrowsException(): void public function testGenerateTextWithNonStringPartThrowsException(): void { $file = new File('https://example.com/file.jpg', 'image/jpeg'); + $candidate = new Candidate( + new Message(MessageRoleEnum::model(), [new MessagePart($file)]), + FinishReasonEnum::stop() + ); - $candidate = $this->createMock(CandidateInterface::class); - $candidate->method('getPart')->willReturn($file); - - $result = $this->createMock(GenerativeAiResultInterface::class); - $result->method('getCandidates')->willReturn([$candidate]); + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - $operation = $this->createMock(OperationInterface::class); - $operation->method('getResult')->willReturn($result); + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('test-model'); + $metadata->method('meetsRequirements')->willReturn(true); - $model = $this->createMock(GenerativeModelInterface::class); - $model->method('generate')->willReturn($operation); + $model = $this->createTextGenerationModel($metadata, $result); $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - $this->expectException(\RuntimeException::class); - $this->expectExceptionMessage('Expected string part but got different type'); + $this->expectException(RuntimeException::class); + $this->expectExceptionMessage('Generated message part contains no text'); $builder->generateText(); } @@ -2482,42 +2666,6 @@ public function testGenerateTextWithNonStringPartThrowsException(): void */ public function testChainGenerationWithMultiplePrompts(): void { - // First generation - $firstCandidate = $this->createMock(CandidateInterface::class); - $firstCandidate->method('getPart')->willReturn('First response'); - - $firstResult = $this->createMock(GenerativeAiResultInterface::class); - $firstResult->method('getCandidates')->willReturn([$firstCandidate]); - - $firstOperation = $this->createMock(OperationInterface::class); - $firstOperation->method('getResult')->willReturn($firstResult); - - // Second generation - $secondCandidate = $this->createMock(CandidateInterface::class); - $secondCandidate->method('getPart')->willReturn('Second response'); - - $secondResult = $this->createMock(GenerativeAiResultInterface::class); - $secondResult->method('getCandidates')->willReturn([$secondCandidate]); - - $secondOperation = $this->createMock(OperationInterface::class); - $secondOperation->method('getResult')->willReturn($secondResult); - - $model = $this->createMock(GenerativeModelInterface::class); - $model->expects($this->exactly(2)) - ->method('generate') - ->willReturnOnConsecutiveCalls($firstOperation, $secondOperation); - - $builder = new PromptBuilder($this->registry, 'First prompt'); - $builder->usingModel($model); - - $firstText = $builder->generateText(); - $this->assertEquals('First response', $firstText); - - // Continue with second prompt - $builder->withModelResponse($firstText) - ->withText('Second prompt'); - - $secondText = $builder->generateText(); - $this->assertEquals('Second response', $secondText); + $this->markTestSkipped('Complex chaining with model response methods not fully implemented yet'); } } \ No newline at end of file From 0fd756e9620d19022891fb522bce3d6aaaaf27b7 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 15:18:56 -0600 Subject: [PATCH 22/47] refactor: removes the need for the Prompts utility class --- src/Builders/PromptBuilder.php | 29 ++- src/Common/Utilities/Prompts.php | 106 ---------- tests/unit/Common/Utilities/PromptsTest.php | 214 -------------------- 3 files changed, 27 insertions(+), 322 deletions(-) delete mode 100644 src/Common/Utilities/Prompts.php delete mode 100644 tests/unit/Common/Utilities/PromptsTest.php diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 59b029cc..5fb8aef8 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -6,7 +6,6 @@ use InvalidArgumentException; use RuntimeException; -use WordPress\AiClient\Common\Utilities\Prompts; use WordPress\AiClient\Files\DTO\File; use WordPress\AiClient\Messages\DTO\Message; use WordPress\AiClient\Messages\DTO\MessagePart; @@ -82,7 +81,7 @@ public function __construct(ProviderRegistry $registry, $prompt = null) } // Check if it's a list of Messages - set as messages - if (Prompts::isMessagesList($prompt)) { + if ($this->isMessagesList($prompt)) { $this->messages = $prompt; return; } @@ -1174,4 +1173,30 @@ private function validateMessages(): void ); } } + + /** + * Checks if the value is a list of Message objects. + * + * @since n.e.x.t + * + * @param mixed $value The value to check. + * @return bool True if the value is a list of Message objects. + * + * @phpstan-assert-if-true list $value + */ + private function isMessagesList($value): bool + { + if (!is_array($value) || empty($value) || !array_is_list($value)) { + return false; + } + + // Check if all items are Messages + foreach ($value as $item) { + if (!($item instanceof Message)) { + return false; + } + } + + return true; + } } diff --git a/src/Common/Utilities/Prompts.php b/src/Common/Utilities/Prompts.php deleted file mode 100644 index ae46eb35..00000000 --- a/src/Common/Utilities/Prompts.php +++ /dev/null @@ -1,106 +0,0 @@ -|list $prompt The prompt to normalize. - * @return list The normalized list of messages. - * @throws InvalidArgumentException If the prompt format is invalid. - */ - // phpcs:enable Generic.Files.LineLength.TooLong - public static function normalizeToMessages($prompt): array - { - // 1. Check if it's already a list of Messages - if (self::isMessagesList($prompt)) { - return $prompt; - } - - // 2. Check if it's a single Message - if ($prompt instanceof Message) { - return [$prompt]; - } - - // 3. Check if it's a MessageArrayShape (single message as array) - if (is_array($prompt) && Message::isArrayShape($prompt)) { - return [Message::fromArray($prompt)]; - } - - // 4. If it's not an array, wrap it in an array - if (!is_array($prompt)) { - $prompt = [$prompt]; - } - - // 5. Loop through the array and handle conversions - all become parts of a single UserMessage - $parts = []; - - foreach ($prompt as $item) { - if (is_string($item)) { - $parts[] = new MessagePart($item); - } elseif ($item instanceof MessagePart) { - $parts[] = $item; - } elseif (is_array($item) && MessagePart::isArrayShape($item)) { - $parts[] = MessagePart::fromArray($item); - } else { - $type = is_object($item) ? get_class($item) : gettype($item); - throw new InvalidArgumentException( - sprintf('Invalid item type %s in prompt.', $type) - ); - } - } - - return [new UserMessage($parts)]; - } - - /** - * Checks if the value is a list of Message objects. - * - * @since n.e.x.t - * - * @param mixed $value The value to check. - * @return bool True if the value is a list of Message objects. - * - * @phpstan-assert-if-true list $value - */ - public static function isMessagesList($value): bool - { - if (!is_array($value) || empty($value) || !array_is_list($value)) { - return false; - } - - // Check if all items are Messages - foreach ($value as $item) { - if (!($item instanceof Message)) { - return false; - } - } - - return true; - } -} diff --git a/tests/unit/Common/Utilities/PromptsTest.php b/tests/unit/Common/Utilities/PromptsTest.php deleted file mode 100644 index 4356b477..00000000 --- a/tests/unit/Common/Utilities/PromptsTest.php +++ /dev/null @@ -1,214 +0,0 @@ -assertCount(1, $result); - $this->assertInstanceOf(UserMessage::class, $result[0]); - - $parts = $result[0]->getParts(); - $this->assertCount(1, $parts); - $this->assertEquals('Hello, world!', $parts[0]->getText()); - } - - /** - * Tests normalizing a single MessagePart to messages. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithMessagePart(): void - { - $part = new MessagePart('Test content'); - $result = Prompts::normalizeToMessages($part); - - $this->assertCount(1, $result); - $this->assertInstanceOf(UserMessage::class, $result[0]); - - $parts = $result[0]->getParts(); - $this->assertCount(1, $parts); - $this->assertEquals('Test content', $parts[0]->getText()); - } - - /** - * Tests normalizing a single Message. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithMessage(): void - { - $message = new UserMessage([new MessagePart('Test message')]); - $result = Prompts::normalizeToMessages($message); - - $this->assertCount(1, $result); - $this->assertSame($message, $result[0]); - } - - /** - * Tests normalizing a MessageArrayShape. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithMessageArrayShape(): void - { - $arrayShape = [ - 'role' => 'user', - 'parts' => [ - [ - 'channel' => 'content', - 'type' => 'text', - 'text' => 'Hello from array' - ] - ] - ]; - - $result = Prompts::normalizeToMessages($arrayShape); - - $this->assertCount(1, $result); - $this->assertInstanceOf(UserMessage::class, $result[0]); - - $parts = $result[0]->getParts(); - $this->assertCount(1, $parts); - $this->assertEquals('Hello from array', $parts[0]->getText()); - } - - /** - * Tests normalizing a list of strings to messages. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithListOfStrings(): void - { - $result = Prompts::normalizeToMessages(['First part', 'Second part', 'Third part']); - - $this->assertCount(1, $result); - $this->assertInstanceOf(UserMessage::class, $result[0]); - - $parts = $result[0]->getParts(); - $this->assertCount(3, $parts); - $this->assertEquals('First part', $parts[0]->getText()); - $this->assertEquals('Second part', $parts[1]->getText()); - $this->assertEquals('Third part', $parts[2]->getText()); - } - - /** - * Tests normalizing a list of MessageParts to messages. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithListOfMessageParts(): void - { - $parts = [ - new MessagePart('Part 1'), - new MessagePart('Part 2') - ]; - - $result = Prompts::normalizeToMessages($parts); - - $this->assertCount(1, $result); - $this->assertInstanceOf(UserMessage::class, $result[0]); - - $messageParts = $result[0]->getParts(); - $this->assertCount(2, $messageParts); - $this->assertEquals('Part 1', $messageParts[0]->getText()); - $this->assertEquals('Part 2', $messageParts[1]->getText()); - } - - /** - * Tests normalizing a mixed list of strings and MessageParts. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithMixedList(): void - { - $items = [ - 'String part', - new MessagePart('MessagePart object'), - [ - 'channel' => 'content', - 'type' => 'text', - 'text' => 'Array shape part' - ] - ]; - - $result = Prompts::normalizeToMessages($items); - - $this->assertCount(1, $result); - $this->assertInstanceOf(UserMessage::class, $result[0]); - - $parts = $result[0]->getParts(); - $this->assertCount(3, $parts); - $this->assertEquals('String part', $parts[0]->getText()); - $this->assertEquals('MessagePart object', $parts[1]->getText()); - $this->assertEquals('Array shape part', $parts[2]->getText()); - } - - /** - * Tests normalizing a list of Messages. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithListOfMessages(): void - { - $messages = [ - new UserMessage([new MessagePart('First message')]), - new UserMessage([new MessagePart('Second message')]) - ]; - - $result = Prompts::normalizeToMessages($messages); - - $this->assertCount(2, $result); - $this->assertSame($messages[0], $result[0]); - $this->assertSame($messages[1], $result[1]); - } - - /** - * Tests that invalid input throws an exception. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithInvalidInput(): void - { - $this->expectException(InvalidArgumentException::class); - $this->expectExceptionMessage('Invalid item type integer in prompt.'); - - Prompts::normalizeToMessages(123); - } - - /** - * Tests that invalid item in list throws an exception. - * - * @since n.e.x.t - */ - public function testNormalizeToMessagesWithInvalidItemInList(): void - { - $this->expectException(InvalidArgumentException::class); - $this->expectExceptionMessage('Invalid item type integer in prompt.'); - - Prompts::normalizeToMessages(['Valid string', 123]); - } -} From 78fa2aadf8a63541933c86fed7c35dd3d9137bb6 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 15:38:05 -0600 Subject: [PATCH 23/47] feat: improves methods for checking if prompt is supported --- src/Builders/PromptBuilder.php | 83 ++++++++- tests/unit/Builders/PromptBuilderTest.php | 198 +++++++++++++++++++++- 2 files changed, 268 insertions(+), 13 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 5fb8aef8..6a0b995d 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -501,17 +501,90 @@ public function getModelRequirements(): ModelRequirements * * @since n.e.x.t * + * @param ModalityEnum|null $intendedOutput Optional output modality to check support for. * @return bool True if supported, false otherwise. */ - public function isSupported(): bool + public function isSupported(?ModalityEnum $intendedOutput = null): bool { - if ($this->model === null) { - // Without a model selected, we can't determine support + // If an intended output modality is specified, temporarily include it + $originalModalities = null; + if ($intendedOutput !== null) { + $originalModalities = $this->modelConfig->getOutputModalities(); + $this->modelConfig->includeOutputModality($intendedOutput); + } + + try { + // Try to get a configured model - this will throw if no suitable model exists + $this->getConfiguredModel(); return true; + } catch (InvalidArgumentException $e) { + return false; + } finally { + // Restore original modalities if we modified them + if ($originalModalities !== null) { + $this->modelConfig->setOutputModalities($originalModalities); + } } + } + + /** + * Checks if the prompt is supported for text generation. + * + * @since n.e.x.t + * + * @return bool True if text generation is supported. + */ + public function isSupportedForText(): bool + { + return $this->isSupported(ModalityEnum::text()); + } - $requirements = $this->getModelRequirements(); - return $this->model->metadata()->meetsRequirements($requirements); + /** + * Checks if the prompt is supported for image generation. + * + * @since n.e.x.t + * + * @return bool True if image generation is supported. + */ + public function isSupportedForImage(): bool + { + return $this->isSupported(ModalityEnum::image()); + } + + /** + * Checks if the prompt is supported for audio generation. + * + * @since n.e.x.t + * + * @return bool True if audio generation is supported. + */ + public function isSupportedForAudio(): bool + { + return $this->isSupported(ModalityEnum::audio()); + } + + /** + * Checks if the prompt is supported for video generation. + * + * @since n.e.x.t + * + * @return bool True if video generation is supported. + */ + public function isSupportedForVideo(): bool + { + return $this->isSupported(ModalityEnum::video()); + } + + /** + * Checks if the prompt is supported for speech generation. + * + * @since n.e.x.t + * + * @return bool True if speech generation is supported. + */ + public function isSupportedForSpeech(): bool + { + return $this->isSupported(ModalityEnum::audio()); } /** diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index 10ee15af..da4924dc 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -1037,9 +1037,18 @@ public function testGetModelRequirementsWithMultimodalInput(): void */ public function testIsSupportedWithoutModel(): void { + // Mock registry to return models + $providerMetadata = $this->createMock(ProviderMetadata::class); + $modelMetadata = $this->createMock(ModelMetadata::class); + $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); + $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); + $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); + $this->registry->method('findModelsMetadataForSupport')->willReturn([$providerModelsMetadata]); + $this->registry->method('getProviderModel')->willReturn($this->createMock(ModelInterface::class)); + $builder = new PromptBuilder($this->registry, 'Test'); - // Without a model, should return true (can't determine support) + // Without a model explicitly set, it should try to find one from registry $this->assertTrue($builder->isSupported()); } @@ -1051,14 +1060,18 @@ public function testIsSupportedWithoutModel(): void public function testIsSupportedWithCompatibleModel(): void { $metadata = $this->createMock(ModelMetadata::class); - $metadata->method('meetsRequirements')->willReturn(true); + $metadata->method('getId')->willReturn('test-model'); $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); + $model->expects($this->once()) + ->method('setConfig') + ->with($this->isInstanceOf(ModelConfig::class)); $builder = new PromptBuilder($this->registry, 'Test'); $builder->usingModel($model); + // With an explicitly set model, it should always return true $this->assertTrue($builder->isSupported()); } @@ -1069,15 +1082,12 @@ public function testIsSupportedWithCompatibleModel(): void */ public function testIsSupportedWithIncompatibleModel(): void { - $metadata = $this->createMock(ModelMetadata::class); - $metadata->method('meetsRequirements')->willReturn(false); - - $model = $this->createMock(ModelInterface::class); - $model->method('metadata')->willReturn($metadata); + // When no models are found in registry, it should return false + $this->registry->method('findModelsMetadataForSupport')->willReturn([]); $builder = new PromptBuilder($this->registry, 'Test'); - $builder->usingModel($model); + // Without any available models, it should return false $this->assertFalse($builder->isSupported()); } @@ -2668,4 +2678,176 @@ public function testChainGenerationWithMultiplePrompts(): void { $this->markTestSkipped('Complex chaining with model response methods not fully implemented yet'); } + + /** + * Tests isSupported with intended output modality. + * + * @return void + */ + public function testIsSupportedWithIntendedOutput(): void + { + $builder = new PromptBuilder($this->registry, 'Test prompt'); + + // Mock registry to return no models for image generation + $this->registry->method('findModelsMetadataForSupport') + ->willReturnCallback(function ($requirements) { + $options = $requirements->getRequiredOptions(); + foreach ($options as $option) { + if ($option->getName() === OptionEnum::outputModalities()->value) { + $modalities = $option->getValue(); + foreach ($modalities as $modality) { + if ($modality->isImage()) { + return []; // No models support image generation + } + } + } + } + // Return a mock model for text generation + $providerMetadata = $this->createMock(ProviderMetadata::class); + $modelMetadata = $this->createMock(ModelMetadata::class); + $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); + $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); + $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); + return [$providerModelsMetadata]; + }); + + // Text should be supported + $this->assertTrue($builder->isSupported(ModalityEnum::text())); + + // Image should not be supported + $this->assertFalse($builder->isSupported(ModalityEnum::image())); + } + + /** + * Tests isSupportedForText convenience method. + * + * @return void + */ + public function testIsSupportedForText(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('text-model'); + + $result = new GenerativeAiResult('test-id', [ + new Candidate( + new ModelMessage([new MessagePart('Test')]), + FinishReasonEnum::stop() + ) + ], new TokenUsage(10, 5, 15)); + + $model = $this->createTextGenerationModel($metadata, $result); + + $builder = new PromptBuilder($this->registry, 'Test prompt'); + $builder->usingModel($model); + + $this->assertTrue($builder->isSupportedForText()); + } + + /** + * Tests isSupportedForImage convenience method. + * + * @return void + */ + public function testIsSupportedForImage(): void + { + $builder = new PromptBuilder($this->registry, 'Generate an image'); + + // Mock registry to return no models for image generation + $this->registry->method('findModelsMetadataForSupport') + ->willReturn([]); + + $this->assertFalse($builder->isSupportedForImage()); + } + + /** + * Tests isSupportedForAudio convenience method. + * + * @return void + */ + public function testIsSupportedForAudio(): void + { + $builder = new PromptBuilder($this->registry, 'Generate audio'); + + // Mock registry to return no models for audio generation + $this->registry->method('findModelsMetadataForSupport') + ->willReturn([]); + + $this->assertFalse($builder->isSupportedForAudio()); + } + + /** + * Tests isSupportedForVideo convenience method. + * + * @return void + */ + public function testIsSupportedForVideo(): void + { + $builder = new PromptBuilder($this->registry, 'Generate video'); + + // Mock registry to return no models for video generation + $this->registry->method('findModelsMetadataForSupport') + ->willReturn([]); + + $this->assertFalse($builder->isSupportedForVideo()); + } + + /** + * Tests isSupportedForSpeech convenience method. + * + * @return void + */ + public function testIsSupportedForSpeech(): void + { + $metadata = $this->createMock(ModelMetadata::class); + $metadata->method('getId')->willReturn('speech-model'); + + $result = new GenerativeAiResult('test-id', [ + new Candidate( + new ModelMessage([new MessagePart(new File('https://example.com/speech.mp3', 'audio/mp3'))]), + FinishReasonEnum::stop() + ) + ], new TokenUsage(10, 5, 15)); + + $model = $this->createSpeechGenerationModel($metadata, $result); + + $builder = new PromptBuilder($this->registry, 'Generate speech'); + $builder->usingModel($model); + + $this->assertTrue($builder->isSupportedForSpeech()); + } + + /** + * Tests isSupported restores original modalities after check. + * + * @return void + */ + public function testIsSupportedRestoresOriginalModalities(): void + { + $builder = new PromptBuilder($this->registry, 'Test prompt'); + + // Set initial modality + $builder->usingOutputModalities(ModalityEnum::text()); + + // Mock registry to return models + $providerMetadata = $this->createMock(ProviderMetadata::class); + $modelMetadata = $this->createMock(ModelMetadata::class); + $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); + $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); + $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); + $this->registry->method('findModelsMetadataForSupport')->willReturn([$providerModelsMetadata]); + $this->registry->method('getProviderModel')->willReturn($this->createMock(ModelInterface::class)); + + // Check with image modality + $builder->isSupported(ModalityEnum::image()); + + // Verify original modality is restored + $reflection = new \ReflectionClass($builder); + $configProperty = $reflection->getProperty('modelConfig'); + $configProperty->setAccessible(true); + $config = $configProperty->getValue($builder); + + $modalities = $config->getOutputModalities(); + $this->assertCount(1, $modalities); + $this->assertTrue($modalities[0]->isText()); + } } \ No newline at end of file From 4d91dc439b55db7a47ed3f57f9eb99e0b8e569b2 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Fri, 22 Aug 2025 15:43:39 -0600 Subject: [PATCH 24/47] test: fixes a bajillion linting errors --- tests/unit/Builders/PromptBuilderTest.php | 824 +++++++++++----------- 1 file changed, 419 insertions(+), 405 deletions(-) diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index da4924dc..557c6c48 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -16,26 +16,22 @@ use WordPress\AiClient\Messages\DTO\UserMessage; use WordPress\AiClient\Messages\Enums\MessageRoleEnum; use WordPress\AiClient\Messages\Enums\ModalityEnum; +use WordPress\AiClient\Providers\DTO\ProviderMetadata; +use WordPress\AiClient\Providers\DTO\ProviderModelsMetadata; use WordPress\AiClient\Providers\Models\Contracts\ModelInterface; use WordPress\AiClient\Providers\Models\DTO\ModelConfig; use WordPress\AiClient\Providers\Models\DTO\ModelMetadata; -use WordPress\AiClient\Providers\Models\DTO\ModelRequirements; -use WordPress\AiClient\Providers\DTO\ProviderModelsMetadata; -use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; use WordPress\AiClient\Providers\Models\Enums\OptionEnum; use WordPress\AiClient\Providers\Models\ImageGeneration\Contracts\ImageGenerationModelInterface; use WordPress\AiClient\Providers\Models\SpeechGeneration\Contracts\SpeechGenerationModelInterface; use WordPress\AiClient\Providers\Models\TextGeneration\Contracts\TextGenerationModelInterface; use WordPress\AiClient\Providers\Models\TextToSpeechConversion\Contracts\TextToSpeechConversionModelInterface; use WordPress\AiClient\Providers\ProviderRegistry; -use WordPress\AiClient\Providers\DTO\ProviderMetadata; use WordPress\AiClient\Results\DTO\Candidate; use WordPress\AiClient\Results\DTO\GenerativeAiResult; use WordPress\AiClient\Results\DTO\TokenUsage; use WordPress\AiClient\Results\Enums\FinishReasonEnum; use WordPress\AiClient\Tools\DTO\FunctionResponse; -use WordPress\AiClient\Operations\Contracts\OperationInterface; -use WordPress\AiClient\Results\Contracts\ResultInterface; /** * @covers \WordPress\AiClient\Builders\PromptBuilder @@ -46,7 +42,7 @@ class PromptBuilderTest extends TestCase * @var ProviderRegistry */ private ProviderRegistry $registry; - + /** * Creates a mock model that implements both ModelInterface and TextGenerationModelInterface. * @@ -56,38 +52,38 @@ class PromptBuilderTest extends TestCase */ private function createTextGenerationModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface { - return new class($metadata, $result) implements ModelInterface, TextGenerationModelInterface { + return new class ($metadata, $result) implements ModelInterface, TextGenerationModelInterface { private ModelMetadata $metadata; private GenerativeAiResult $result; private ModelConfig $config; - + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) { $this->metadata = $metadata; $this->result = $result; $this->config = new ModelConfig(); } - + public function metadata(): ModelMetadata { return $this->metadata; } - + public function setConfig(ModelConfig $config): void { $this->config = $config; } - + public function getConfig(): ModelConfig { return $this->config; } - + public function generateTextResult(array $prompt): GenerativeAiResult { return $this->result; } - + public function streamGenerateTextResult(array $prompt): Generator { yield $this->result; @@ -104,33 +100,33 @@ public function streamGenerateTextResult(array $prompt): Generator */ private function createImageGenerationModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface { - return new class($metadata, $result) implements ModelInterface, ImageGenerationModelInterface { + return new class ($metadata, $result) implements ModelInterface, ImageGenerationModelInterface { private ModelMetadata $metadata; private GenerativeAiResult $result; private ModelConfig $config; - + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) { $this->metadata = $metadata; $this->result = $result; $this->config = new ModelConfig(); } - + public function metadata(): ModelMetadata { return $this->metadata; } - + public function setConfig(ModelConfig $config): void { $this->config = $config; } - + public function getConfig(): ModelConfig { return $this->config; } - + public function generateImageResult(array $prompt): GenerativeAiResult { return $this->result; @@ -147,33 +143,33 @@ public function generateImageResult(array $prompt): GenerativeAiResult */ private function createSpeechGenerationModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface { - return new class($metadata, $result) implements ModelInterface, SpeechGenerationModelInterface { + return new class ($metadata, $result) implements ModelInterface, SpeechGenerationModelInterface { private ModelMetadata $metadata; private GenerativeAiResult $result; private ModelConfig $config; - + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) { $this->metadata = $metadata; $this->result = $result; $this->config = new ModelConfig(); } - + public function metadata(): ModelMetadata { return $this->metadata; } - + public function setConfig(ModelConfig $config): void { $this->config = $config; } - + public function getConfig(): ModelConfig { return $this->config; } - + public function generateSpeechResult(array $prompt): GenerativeAiResult { return $this->result; @@ -190,33 +186,33 @@ public function generateSpeechResult(array $prompt): GenerativeAiResult */ private function createTextToSpeechModel(ModelMetadata $metadata, GenerativeAiResult $result): ModelInterface { - return new class($metadata, $result) implements ModelInterface, TextToSpeechConversionModelInterface { + return new class ($metadata, $result) implements ModelInterface, TextToSpeechConversionModelInterface { private ModelMetadata $metadata; private GenerativeAiResult $result; private ModelConfig $config; - + public function __construct(ModelMetadata $metadata, GenerativeAiResult $result) { $this->metadata = $metadata; $this->result = $result; $this->config = new ModelConfig(); } - + public function metadata(): ModelMetadata { return $this->metadata; } - + public function setConfig(ModelConfig $config): void { $this->config = $config; } - + public function getConfig(): ModelConfig { return $this->config; } - + public function convertTextToSpeechResult(array $prompt): GenerativeAiResult { return $this->result; @@ -243,11 +239,11 @@ protected function setUp(): void public function testConstructorWithNoPrompt(): void { $builder = new PromptBuilder($this->registry); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); - + $this->assertEmpty($messagesProperty->getValue($builder)); } @@ -259,7 +255,7 @@ public function testConstructorWithNoPrompt(): void public function testConstructorWithStringPrompt(): void { $builder = new PromptBuilder($this->registry, 'Hello, world!'); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); @@ -267,7 +263,7 @@ public function testConstructorWithStringPrompt(): void /** @var list $messages */ /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Hello, world!', $messages[0]->getParts()[0]->getText()); @@ -282,7 +278,7 @@ public function testConstructorWithMessagePartPrompt(): void { $part = new MessagePart('Test message'); $builder = new PromptBuilder($this->registry, $part); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); @@ -290,7 +286,7 @@ public function testConstructorWithMessagePartPrompt(): void /** @var list $messages */ /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Test message', $messages[0]->getParts()[0]->getText()); @@ -305,14 +301,14 @@ public function testConstructorWithMessagePrompt(): void { $message = new UserMessage([new MessagePart('User message')]); $builder = new PromptBuilder($this->registry, $message); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertSame($message, $messages[0]); } @@ -330,13 +326,13 @@ public function testConstructorWithMessagesList(): void new UserMessage([new MessagePart('Third')]) ]; $builder = new PromptBuilder($this->registry, $messages); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $actualMessages */ $actualMessages = $messagesProperty->getValue($builder); - + $this->assertCount(3, $actualMessages); $this->assertSame($messages, $actualMessages); } @@ -355,14 +351,14 @@ public function testConstructorWithMessageArrayShape(): void ] ]; $builder = new PromptBuilder($this->registry, $messageArray); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Hello from array', $messages[0]->getParts()[0]->getText()); @@ -377,15 +373,15 @@ public function testWithText(): void { $builder = new PromptBuilder($this->registry); $result = $builder->withText('Some text'); - + $this->assertSame($builder, $result); // Test fluent interface - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertEquals('Some text', $messages[0]->getParts()[0]->getText()); } @@ -399,13 +395,13 @@ public function testWithTextAppendsToExistingUserMessage(): void { $builder = new PromptBuilder($this->registry, 'Initial text'); $builder->withText(' Additional text'); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $parts = $messages[0]->getParts(); $this->assertCount(2, $parts); @@ -423,15 +419,15 @@ public function testWithInlineImage(): void $builder = new PromptBuilder($this->registry); $base64 = 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNkYPhfDwAChwGA60e6kgAAAABJRU5ErkJggg=='; $result = $builder->withInlineImage($base64, 'image/png'); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $file = $messages[0]->getParts()[0]->getFile(); $this->assertInstanceOf(File::class, $file); @@ -448,15 +444,15 @@ public function testWithRemoteImage(): void { $builder = new PromptBuilder($this->registry); $result = $builder->withRemoteImage('https://example.com/image.jpg', 'image/jpeg'); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $file = $messages[0]->getParts()[0]->getFile(); $this->assertInstanceOf(File::class, $file); @@ -474,15 +470,15 @@ public function testWithImageFile(): void $file = new File('https://example.com/test.png', 'image/png'); $builder = new PromptBuilder($this->registry); $result = $builder->withImageFile($file); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); } @@ -497,15 +493,15 @@ public function testWithAudioFile(): void $file = new File('https://example.com/audio.mp3', 'audio/mp3'); $builder = new PromptBuilder($this->registry); $result = $builder->withAudioFile($file); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); } @@ -520,15 +516,15 @@ public function testWithVideoFile(): void $file = new File('https://example.com/video.mp4', 'video/mp4'); $builder = new PromptBuilder($this->registry); $result = $builder->withVideoFile($file); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); } @@ -543,15 +539,15 @@ public function testWithFunctionResponse(): void $functionResponse = new FunctionResponse('func_id', 'func_name', ['result' => 'data']); $builder = new PromptBuilder($this->registry); $result = $builder->withFunctionResponse($functionResponse); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertSame($functionResponse, $messages[0]->getParts()[0]->getFunctionResponse()); } @@ -566,18 +562,18 @@ public function testWithMessageParts(): void $part1 = new MessagePart('Part 1'); $part2 = new MessagePart('Part 2'); $part3 = new MessagePart('Part 3'); - + $builder = new PromptBuilder($this->registry); $result = $builder->withMessageParts($part1, $part2, $part3); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $parts = $messages[0]->getParts(); $this->assertCount(3, $parts); @@ -598,18 +594,18 @@ public function testWithHistory(): void new ModelMessage([new MessagePart('Model 1')]), new UserMessage([new MessagePart('User 2')]) ]; - + $builder = new PromptBuilder($this->registry); $result = $builder->withHistory(...$history); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(3, $messages); $this->assertEquals('User 1', $messages[0]->getParts()[0]->getText()); $this->assertEquals('Model 1', $messages[1]->getParts()[0]->getText()); @@ -626,13 +622,13 @@ public function testUsingModel(): void $model = $this->createMock(ModelInterface::class); $builder = new PromptBuilder($this->registry); $result = $builder->usingModel($model); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $modelProperty = $reflection->getProperty('model'); $modelProperty->setAccessible(true); - + /** @var ModelInterface $actualModel */ $actualModel = $modelProperty->getValue($builder); $this->assertSame($model, $actualModel); @@ -648,13 +644,13 @@ public function testUsingRegistry(): void $newRegistry = $this->createMock(ProviderRegistry::class); $builder = new PromptBuilder($this->registry); $result = $builder->usingRegistry($newRegistry); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $registryProperty = $reflection->getProperty('registry'); $registryProperty->setAccessible(true); - + /** @var ProviderRegistry $actualRegistry */ $actualRegistry = $registryProperty->getValue($builder); $this->assertSame($newRegistry, $actualRegistry); @@ -669,15 +665,15 @@ public function testUsingSystemInstruction(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingSystemInstruction('You are a helpful assistant.'); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals('You are a helpful assistant.', $config->getSystemInstruction()); } @@ -690,15 +686,15 @@ public function testUsingMaxTokens(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingMaxTokens(1000); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals(1000, $config->getMaxTokens()); } @@ -711,15 +707,15 @@ public function testUsingTemperature(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingTemperature(0.7); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals(0.7, $config->getTemperature()); } @@ -732,15 +728,15 @@ public function testUsingTopP(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingTopP(0.9); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals(0.9, $config->getTopP()); } @@ -753,15 +749,15 @@ public function testUsingTopK(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingTopK(40); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals(40, $config->getTopK()); } @@ -774,15 +770,15 @@ public function testUsingStopSequences(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingStopSequences('STOP', 'END', '###'); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $customOptions = $config->getCustomOptions(); $this->assertArrayHasKey('stopSequences', $customOptions); $this->assertEquals(['STOP', 'END', '###'], $customOptions['stopSequences']); @@ -797,15 +793,15 @@ public function testUsingCandidateCount(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingCandidateCount(3); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals(3, $config->getCandidateCount()); } @@ -818,15 +814,15 @@ public function testUsingOutputMime(): void { $builder = new PromptBuilder($this->registry); $result = $builder->usingOutputMime('application/json'); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals('application/json', $config->getOutputMimeType()); } @@ -843,18 +839,18 @@ public function testUsingOutputSchema(): void 'name' => ['type' => 'string'] ] ]; - + $builder = new PromptBuilder($this->registry); $result = $builder->usingOutputSchema($schema); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals($schema, $config->getOutputSchema()); } @@ -870,15 +866,15 @@ public function testUsingOutputModalities(): void ModalityEnum::text(), ModalityEnum::image() ); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $modalities = $config->getOutputModalities(); $this->assertCount(2, $modalities); $this->assertTrue($modalities[0]->isText()); @@ -894,15 +890,15 @@ public function testAsJsonResponse(): void { $builder = new PromptBuilder($this->registry); $result = $builder->asJsonResponse(); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals('application/json', $config->getOutputMimeType()); } @@ -916,15 +912,15 @@ public function testAsJsonResponseWithSchema(): void $schema = ['type' => 'array']; $builder = new PromptBuilder($this->registry); $result = $builder->asJsonResponse($schema); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals('application/json', $config->getOutputMimeType()); $this->assertEquals($schema, $config->getOutputSchema()); } @@ -938,11 +934,11 @@ public function testGetModelRequirementsBasicText(): void { $builder = new PromptBuilder($this->registry, 'Simple text'); $requirements = $builder->getModelRequirements(); - + $capabilities = $requirements->getRequiredCapabilities(); $this->assertCount(1, $capabilities); $this->assertTrue($capabilities[0]->isTextGeneration()); - + $options = $requirements->getRequiredOptions(); // Should have input modalities with text $inputModalitiesFound = false; @@ -970,10 +966,10 @@ public function testGetModelRequirementsWithChatHistory(): void new ModelMessage([new MessagePart('Hi there')]), new UserMessage([new MessagePart('How are you?')]) ); - + $requirements = $builder->getModelRequirements(); $capabilities = $requirements->getRequiredCapabilities(); - + // Should have text generation and chat history capabilities $this->assertCount(2, $capabilities); $hasTextGeneration = false; @@ -1000,10 +996,10 @@ public function testGetModelRequirementsWithMultimodalInput(): void $builder = new PromptBuilder($this->registry); $builder->withText('Describe this image') ->withRemoteImage('https://example.com/image.jpg', 'image/jpeg'); - + $requirements = $builder->getModelRequirements(); $options = $requirements->getRequiredOptions(); - + // Find input modalities option $inputModalities = null; foreach ($options as $option) { @@ -1012,10 +1008,10 @@ public function testGetModelRequirementsWithMultimodalInput(): void break; } } - + $this->assertNotNull($inputModalities); $this->assertCount(2, $inputModalities); - + $hasText = false; $hasImage = false; foreach ($inputModalities as $modality) { @@ -1045,9 +1041,9 @@ public function testIsSupportedWithoutModel(): void $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); $this->registry->method('findModelsMetadataForSupport')->willReturn([$providerModelsMetadata]); $this->registry->method('getProviderModel')->willReturn($this->createMock(ModelInterface::class)); - + $builder = new PromptBuilder($this->registry, 'Test'); - + // Without a model explicitly set, it should try to find one from registry $this->assertTrue($builder->isSupported()); } @@ -1061,16 +1057,16 @@ public function testIsSupportedWithCompatibleModel(): void { $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); - + $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); $model->expects($this->once()) ->method('setConfig') ->with($this->isInstanceOf(ModelConfig::class)); - + $builder = new PromptBuilder($this->registry, 'Test'); $builder->usingModel($model); - + // With an explicitly set model, it should always return true $this->assertTrue($builder->isSupported()); } @@ -1084,9 +1080,9 @@ public function testIsSupportedWithIncompatibleModel(): void { // When no models are found in registry, it should return false $this->registry->method('findModelsMetadataForSupport')->willReturn([]); - + $builder = new PromptBuilder($this->registry, 'Test'); - + // Without any available models, it should return false $this->assertFalse($builder->isSupported()); } @@ -1099,10 +1095,10 @@ public function testIsSupportedWithIncompatibleModel(): void public function testValidateMessagesEmptyThrowsException(): void { $builder = new PromptBuilder($this->registry); - + $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('Cannot generate from an empty prompt'); - + $builder->generateResult(); } @@ -1117,10 +1113,10 @@ public function testValidateMessagesNonUserFirstThrowsException(): void new ModelMessage([new MessagePart('Model says hi')]), new UserMessage([new MessagePart('User response')]) ]); - + $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('The first message must be from a user role'); - + $builder->generateResult(); } @@ -1135,13 +1131,13 @@ public function testValidateMessagesNonUserLastThrowsException(): void new UserMessage([new MessagePart('User says hi')]), new ModelMessage([new MessagePart('Model response')]) ]); - + // Add a user message to make it valid, then add model message $builder->withHistory(new ModelMessage([new MessagePart('Another model message')])); - + $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('The last message must be from a user role'); - + $builder->generateResult(); } @@ -1154,7 +1150,7 @@ public function testParseMessageEmptyStringThrowsException(): void { $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('Cannot create a message from an empty string'); - + new PromptBuilder($this->registry, ' '); } @@ -1167,7 +1163,7 @@ public function testParseMessageEmptyArrayThrowsException(): void { $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('Cannot create a message from an empty array'); - + new PromptBuilder($this->registry, []); } @@ -1180,7 +1176,7 @@ public function testParseMessageInvalidTypeThrowsException(): void { $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('Input must be a string, MessagePart, MessagePartArrayShape'); - + new PromptBuilder($this->registry, 123); } @@ -1192,7 +1188,7 @@ public function testParseMessageInvalidTypeThrowsException(): void public function testMethodChaining(): void { $model = $this->createMock(ModelInterface::class); - + $builder = new PromptBuilder($this->registry); $result = $builder ->withText('Start of prompt') @@ -1205,29 +1201,29 @@ public function testMethodChaining(): void ->usingTopK(50) ->usingCandidateCount(2) ->asJsonResponse(); - + $this->assertSame($builder, $result); - + $reflection = new \ReflectionClass($builder); - + $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); $this->assertCount(2, $messages[0]->getParts()); // Text and image - + $modelProperty = $reflection->getProperty('model'); $modelProperty->setAccessible(true); /** @var ModelInterface $actualModel */ $actualModel = $modelProperty->getValue($builder); $this->assertSame($model, $actualModel); - + $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals('Be helpful', $config->getSystemInstruction()); $this->assertEquals(500, $config->getMaxTokens()); $this->assertEquals(0.8, $config->getTemperature()); @@ -1245,16 +1241,16 @@ public function testMethodChaining(): void public function testGenerateResultWithTextModality(): void { $result = $this->createMock(GenerativeAiResult::class); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); - + $actualResult = $builder->generateResult(); $this->assertSame($result, $actualResult); } @@ -1268,20 +1264,23 @@ public function testGenerateResultWithImageModality(): void { $result = new GenerativeAiResult( 'test-result', - [new Candidate(new ModelMessage([new MessagePart(new File('', 'image/png'))]), FinishReasonEnum::stop())], + [new Candidate( + new ModelMessage([new MessagePart(new File('', 'image/png'))]), + FinishReasonEnum::stop() + )], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createImageGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate an image'); $builder->usingModel($model); $builder->usingOutputModalities(ModalityEnum::image()); - + $actualResult = $builder->generateResult(); $this->assertSame($result, $actualResult); } @@ -1295,20 +1294,23 @@ public function testGenerateResultWithAudioModality(): void { $result = new GenerativeAiResult( 'test-result', - [new Candidate(new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), FinishReasonEnum::stop())], + [new Candidate( + new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), + FinishReasonEnum::stop() + )], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createSpeechGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); $builder->usingOutputModalities(ModalityEnum::audio()); - + $actualResult = $builder->generateResult(); $this->assertSame($result, $actualResult); } @@ -1325,17 +1327,17 @@ public function testGenerateResultWithMultimodalOutput(): void [new Candidate(new ModelMessage([new MessagePart('Generated text')]), FinishReasonEnum::stop())], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate multimodal'); $builder->usingModel($model); $builder->usingOutputModalities(ModalityEnum::text(), ModalityEnum::image()); - + $actualResult = $builder->generateResult(); $this->assertSame($result, $actualResult); } @@ -1350,17 +1352,17 @@ public function testGenerateResultThrowsExceptionForUnsupportedModality(): void $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + // Model that only implements ModelInterface, not TextGenerationModelInterface $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); - + $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Model "test-model" does not support text generation'); - + $builder->generateResult(); } @@ -1374,17 +1376,17 @@ public function testGenerateResultThrowsExceptionForUnsupportedOutputModality(): $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); - + $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); $builder->usingOutputModalities(ModalityEnum::video()); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Output modality "video" is not yet supported'); - + $builder->generateResult(); } @@ -1400,26 +1402,26 @@ public function testGenerateTextResult(): void [new Candidate(new ModelMessage([new MessagePart('Generated text')]), FinishReasonEnum::stop())], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); - + $actualResult = $builder->generateTextResult(); $this->assertSame($result, $actualResult); - + // Verify text modality was included $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $modalities = $config->getOutputModalities(); $this->assertNotNull($modalities); $this->assertTrue($modalities[0]->isText()); @@ -1434,29 +1436,32 @@ public function testGenerateImageResult(): void { $result = new GenerativeAiResult( 'test-result', - [new Candidate(new ModelMessage([new MessagePart(new File('', 'image/png'))]), FinishReasonEnum::stop())], + [new Candidate( + new ModelMessage([new MessagePart(new File('', 'image/png'))]), + FinishReasonEnum::stop() + )], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createImageGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate image'); $builder->usingModel($model); - + $actualResult = $builder->generateImageResult(); $this->assertSame($result, $actualResult); - + // Verify image modality was included $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $modalities = $config->getOutputModalities(); $this->assertNotNull($modalities); $this->assertTrue($modalities[0]->isImage()); @@ -1471,29 +1476,32 @@ public function testGenerateSpeechResult(): void { $result = new GenerativeAiResult( 'test-result', - [new Candidate(new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), FinishReasonEnum::stop())], + [new Candidate( + new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), + FinishReasonEnum::stop() + )], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createSpeechGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); - + $actualResult = $builder->generateSpeechResult(); $this->assertSame($result, $actualResult); - + // Verify audio modality was included $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $modalities = $config->getOutputModalities(); $this->assertNotNull($modalities); $this->assertTrue($modalities[0]->isAudio()); @@ -1508,29 +1516,32 @@ public function testConvertTextToSpeechResult(): void { $result = new GenerativeAiResult( 'test-result', - [new Candidate(new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), FinishReasonEnum::stop())], + [new Candidate( + new ModelMessage([new MessagePart(new File('data:audio/wav;base64,UklGRigE=', 'audio/wav'))]), + FinishReasonEnum::stop() + )], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextToSpeechModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Convert to speech'); $builder->usingModel($model); - + $actualResult = $builder->convertTextToSpeechResult(); $this->assertSame($result, $actualResult); - + // Verify audio modality was included $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $modalities = $config->getOutputModalities(); $this->assertNotNull($modalities); $this->assertTrue($modalities[0]->isAudio()); @@ -1546,17 +1557,17 @@ public function testConvertTextToSpeechResultThrowsExceptionForUnsupportedModel( $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + // Model that doesn't implement TextToSpeechConversionModelInterface $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); - + $builder = new PromptBuilder($this->registry, 'Convert to speech'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Model "test-model" does not support text-to-speech conversion'); - + $builder->convertTextToSpeechResult(); } @@ -1570,18 +1581,18 @@ public function testGenerateText(): void $messagePart = new MessagePart('Generated text content'); $message = new ModelMessage([$messagePart]); $candidate = new Candidate($message, FinishReasonEnum::stop()); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - + $text = $builder->generateText(); $this->assertEquals('Generated text content', $text); } @@ -1598,49 +1609,49 @@ public function testGenerateTextThrowsExceptionWhenNoCandidates(): void $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - - $model = new class($metadata) implements ModelInterface, TextGenerationModelInterface { + + $model = new class ($metadata) implements ModelInterface, TextGenerationModelInterface { private ModelMetadata $metadata; private ModelConfig $config; - + public function __construct(ModelMetadata $metadata) { $this->metadata = $metadata; $this->config = new ModelConfig(); } - + public function metadata(): ModelMetadata { return $this->metadata; } - + public function setConfig(ModelConfig $config): void { $this->config = $config; } - + public function getConfig(): ModelConfig { return $this->config; } - + public function generateTextResult(array $prompt): GenerativeAiResult { throw new RuntimeException('No candidates were generated'); } - + public function streamGenerateTextResult(array $prompt): Generator { yield from []; } }; - + $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('No candidates were generated'); - + $builder->generateText(); } @@ -1653,21 +1664,21 @@ public function testGenerateTextThrowsExceptionWhenNoParts(): void { $message = new ModelMessage([]); $candidate = new Candidate($message, FinishReasonEnum::stop()); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Generated message contains no parts'); - + $builder->generateText(); } @@ -1682,21 +1693,21 @@ public function testGenerateTextThrowsExceptionWhenPartHasNoText(): void $messagePart = new MessagePart($file); $message = new ModelMessage([$messagePart]); $candidate = new Candidate($message, FinishReasonEnum::stop()); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Generated message part contains no text'); - + $builder->generateText(); } @@ -1721,32 +1732,32 @@ public function testGenerateTexts(): void FinishReasonEnum::stop() ) ]; - + $result = new GenerativeAiResult('test-result-id', $candidates, new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate texts'); $builder->usingModel($model); - + $texts = $builder->generateTexts(3); - + $this->assertCount(3, $texts); $this->assertEquals('Text 1', $texts[0]); $this->assertEquals('Text 2', $texts[1]); $this->assertEquals('Text 3', $texts[2]); - + // Verify candidate count was set $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $this->assertEquals(3, $config->getCandidateCount()); } @@ -1760,49 +1771,49 @@ public function testGenerateTextsThrowsExceptionWhenNoTextGenerated(): void $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - - $model = new class($metadata) implements ModelInterface, TextGenerationModelInterface { + + $model = new class ($metadata) implements ModelInterface, TextGenerationModelInterface { private ModelMetadata $metadata; private ModelConfig $config; - + public function __construct(ModelMetadata $metadata) { $this->metadata = $metadata; $this->config = new ModelConfig(); } - + public function metadata(): ModelMetadata { return $this->metadata; } - + public function setConfig(ModelConfig $config): void { $this->config = $config; } - + public function getConfig(): ModelConfig { return $this->config; } - + public function generateTextResult(array $prompt): GenerativeAiResult { throw new RuntimeException('No text was generated from any candidates'); } - + public function streamGenerateTextResult(array $prompt): Generator { yield from []; } }; - + $builder = new PromptBuilder($this->registry, 'Generate texts'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('No text was generated from any candidates'); - + $builder->generateTexts(); } @@ -1817,18 +1828,18 @@ public function testGenerateImage(): void $messagePart = new MessagePart($file); $message = new ModelMessage([$messagePart]); $candidate = new Candidate($message, FinishReasonEnum::stop()); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createImageGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate image'); $builder->usingModel($model); - + $generatedFile = $builder->generateImage(); $this->assertSame($file, $generatedFile); } @@ -1843,21 +1854,21 @@ public function testGenerateImageThrowsExceptionWhenNoFile(): void $messagePart = new MessagePart('Text instead of image'); $message = new ModelMessage([$messagePart]); $candidate = new Candidate($message, FinishReasonEnum::stop()); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createImageGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate image'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Generated message part contains no image file'); - + $builder->generateImage(); } @@ -1872,7 +1883,7 @@ public function testGenerateImages(): void new File('https://example.com/img1.jpg', 'image/jpeg'), new File('https://example.com/img2.jpg', 'image/jpeg'), ]; - + $candidates = []; foreach ($files as $file) { $candidates[] = new Candidate( @@ -1880,20 +1891,20 @@ public function testGenerateImages(): void FinishReasonEnum::stop() ); } - + $result = new GenerativeAiResult('test-result-id', $candidates, new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createImageGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate images'); $builder->usingModel($model); - + $generatedFiles = $builder->generateImages(2); - + $this->assertCount(2, $generatedFiles); $this->assertSame($files[0], $generatedFiles[0]); $this->assertSame($files[1], $generatedFiles[1]); @@ -1910,18 +1921,18 @@ public function testConvertTextToSpeech(): void $messagePart = new MessagePart($file); $message = new Message(MessageRoleEnum::model(), [$messagePart]); $candidate = new Candidate($message, FinishReasonEnum::stop()); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextToSpeechModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Convert this text'); $builder->usingModel($model); - + $audioFile = $builder->convertTextToSpeech(); $this->assertSame($file, $audioFile); } @@ -1937,7 +1948,7 @@ public function testConvertTextToSpeeches(): void new File('https://example.com/audio1.mp3', 'audio/mp3'), new File('https://example.com/audio2.mp3', 'audio/mp3'), ]; - + $candidates = []; foreach ($files as $file) { $candidates[] = new Candidate( @@ -1945,20 +1956,20 @@ public function testConvertTextToSpeeches(): void FinishReasonEnum::stop() ); } - + $result = new GenerativeAiResult('test-result-id', $candidates, new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextToSpeechModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Convert this text'); $builder->usingModel($model); - + $audioFiles = $builder->convertTextToSpeeches(2); - + $this->assertCount(2, $audioFiles); $this->assertSame($files[0], $audioFiles[0]); $this->assertSame($files[1], $audioFiles[1]); @@ -1975,18 +1986,18 @@ public function testGenerateSpeech(): void $messagePart = new MessagePart($file); $message = new Message(MessageRoleEnum::model(), [$messagePart]); $candidate = new Candidate($message, FinishReasonEnum::stop()); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createSpeechGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); - + $speechFile = $builder->generateSpeech(); $this->assertSame($file, $speechFile); } @@ -2003,7 +2014,7 @@ public function testGenerateSpeeches(): void new File('https://example.com/speech2.mp3', 'audio/mp3'), new File('https://example.com/speech3.mp3', 'audio/mp3'), ]; - + $candidates = []; foreach ($files as $file) { $candidates[] = new Candidate( @@ -2012,24 +2023,24 @@ public function testGenerateSpeeches(): void 10 ); } - + $result = new GenerativeAiResult( 'test-result-id', $candidates, new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createSpeechGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); - + $speechFiles = $builder->generateSpeeches(3); - + $this->assertCount(3, $speechFiles); $this->assertSame($files[0], $speechFiles[0]); $this->assertSame($files[1], $speechFiles[1]); @@ -2045,18 +2056,18 @@ public function testGetConfiguredModelWithExplicitModel(): void { $metadata = $this->createMock(ModelMetadata::class); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); $model->expects($this->once())->method('setConfig')->with($this->isInstanceOf(ModelConfig::class)); - + $builder = new PromptBuilder($this->registry, 'Test'); $builder->usingModel($model); - + $reflection = new \ReflectionClass($builder); $method = $reflection->getMethod('getConfiguredModel'); $method->setAccessible(true); - + $configuredModel = $method->invoke($builder); $this->assertSame($model, $configuredModel); } @@ -2070,20 +2081,20 @@ public function testGetConfiguredModelReturnsExplicitlySetModel(): void { $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('explicit-model'); - + $model = $this->createMock(ModelInterface::class); $model->method('metadata')->willReturn($metadata); $model->expects($this->once()) ->method('setConfig') ->with($this->isInstanceOf(ModelConfig::class)); - + $builder = new PromptBuilder($this->registry, 'Test'); $builder->usingModel($model); - + $reflection = new \ReflectionClass($builder); $method = $reflection->getMethod('getConfiguredModel'); $method->setAccessible(true); - + $result = $method->invoke($builder); $this->assertSame($model, $result); } @@ -2097,29 +2108,29 @@ public function testGetConfiguredModelFindsModelFromRegistry(): void { $modelMetadata = $this->createMock(ModelMetadata::class); $modelMetadata->method('getId')->willReturn('found-model'); - + $providerMetadata = $this->createMock(ProviderMetadata::class); $providerMetadata->method('getId')->willReturn('test-provider'); - + $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); - + $model = $this->createMock(ModelInterface::class); - + $this->registry->method('findModelsMetadataForSupport') ->willReturn([$providerModelsMetadata]); - + $this->registry->method('getProviderModel') ->with('test-provider', 'found-model', $this->isInstanceOf(ModelConfig::class)) ->willReturn($model); - + $builder = new PromptBuilder($this->registry, 'Test'); - + $reflection = new \ReflectionClass($builder); $method = $reflection->getMethod('getConfiguredModel'); $method->setAccessible(true); - + $configuredModel = $method->invoke($builder); $this->assertSame($model, $configuredModel); } @@ -2132,16 +2143,16 @@ public function testGetConfiguredModelFindsModelFromRegistry(): void public function testGetConfiguredModelThrowsExceptionWhenNoModelsFound(): void { $this->registry->method('findModelsMetadataForSupport')->willReturn([]); - + $builder = new PromptBuilder($this->registry, 'Test'); - + $reflection = new \ReflectionClass($builder); $method = $reflection->getMethod('getConfiguredModel'); $method->setAccessible(true); - + $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('No models found that support the required capabilities'); - + $method->invoke($builder); } @@ -2153,20 +2164,20 @@ public function testGetConfiguredModelThrowsExceptionWhenNoModelsFound(): void public function testAppendPartToMessagesCreatesNewUserMessage(): void { $builder = new PromptBuilder($this->registry); - + $reflection = new \ReflectionClass($builder); $method = $reflection->getMethod('appendPartToMessages'); $method->setAccessible(true); - + $part = new MessagePart('Test part'); $method->invoke($builder, $part); - + $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertInstanceOf(Message::class, $messages[0]); $this->assertEquals('Test part', $messages[0]->getParts()[0]->getText()); @@ -2180,19 +2191,19 @@ public function testAppendPartToMessagesCreatesNewUserMessage(): void public function testAppendPartToMessagesAppendsToExistingUserMessage(): void { $builder = new PromptBuilder($this->registry, 'Initial'); - + $reflection = new \ReflectionClass($builder); $method = $reflection->getMethod('appendPartToMessages'); $method->setAccessible(true); - + $part = new MessagePart('Additional'); $method->invoke($builder, $part); - + $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $parts = $messages[0]->getParts(); $this->assertCount(2, $parts); @@ -2211,19 +2222,19 @@ public function testAppendPartToMessagesCreatesNewMessageWhenLastIsModel(): void new UserMessage([new MessagePart('User')]), new ModelMessage([new MessagePart('Model')]) ]); - + $reflection = new \ReflectionClass($builder); $method = $reflection->getMethod('appendPartToMessages'); $method->setAccessible(true); - + $part = new MessagePart('New user message'); $method->invoke($builder, $part); - + $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(3, $messages); $this->assertInstanceOf(Message::class, $messages[2]); $this->assertEquals('New user message', $messages[2]->getParts()[0]->getText()); @@ -2239,7 +2250,7 @@ public function testComplexMultimodalPromptBuilding(): void $file1 = new File('https://example.com/img1.jpg', 'image/jpeg'); $file2 = new File('https://example.com/audio.mp3', 'audio/mp3'); $functionResponse = new FunctionResponse('func1', 'getData', ['result' => 'data']); - + $builder = new PromptBuilder($this->registry); $builder->withText('Analyze this data:') ->withImageFile($file1) @@ -2251,16 +2262,16 @@ public function testComplexMultimodalPromptBuilding(): void new ModelMessage([new MessagePart('Previous answer')]) ) ->withText(' Final instruction'); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + // Should have 4 messages: 1 initial + 2 from history + 1 final $this->assertCount(4, $messages); - + // Check first message with initial content $firstParts = $messages[0]->getParts(); $this->assertCount(5, $firstParts); // text, image, text, audio, function response @@ -2269,11 +2280,11 @@ public function testComplexMultimodalPromptBuilding(): void $this->assertEquals(' and this audio:', $firstParts[2]->getText()); $this->assertSame($file2, $firstParts[3]->getFile()); $this->assertSame($functionResponse, $firstParts[4]->getFunctionResponse()); - + // Check history messages $this->assertEquals('Previous question', $messages[1]->getParts()[0]->getText()); $this->assertEquals('Previous answer', $messages[2]->getParts()[0]->getText()); - + // Check final message $finalParts = $messages[3]->getParts(); $this->assertCount(1, $finalParts); @@ -2292,28 +2303,28 @@ public function testIncludeOutputModalityPreservesExisting(): void [new Candidate(new ModelMessage([new MessagePart('Generated text')]), FinishReasonEnum::stop())], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Test'); $builder->usingModel($model); - + // Set initial modality $builder->usingOutputModalities(ModalityEnum::audio()); - + // Generate text should add text modality, not replace audio $builder->generateTextResult(); - + $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $modalities = $config->getOutputModalities(); $this->assertCount(2, $modalities); $this->assertTrue($modalities[0]->isAudio()); @@ -2328,14 +2339,14 @@ public function testIncludeOutputModalityPreservesExisting(): void public function testConstructorWithStringPartsList(): void { $builder = new PromptBuilder($this->registry, ['Part 1', 'Part 2', 'Part 3']); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $this->assertInstanceOf(Message::class, $messages[0]); $parts = $messages[0]->getParts(); @@ -2354,15 +2365,15 @@ public function testConstructorWithMixedPartsList(): void { $part1 = new MessagePart('Part 1'); $part2Array = ['type' => 'text', 'text' => 'Part 2']; - + $builder = new PromptBuilder($this->registry, ['String part', $part1, $part2Array]); - + $reflection = new \ReflectionClass($builder); $messagesProperty = $reflection->getProperty('messages'); $messagesProperty->setAccessible(true); /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - + $this->assertCount(1, $messages); $parts = $messages[0]->getParts(); $this->assertCount(3, $parts); @@ -2380,7 +2391,7 @@ public function testParseMessageWithNonListArrayThrowsException(): void { $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('Array input must be a list array'); - + new PromptBuilder($this->registry, ['key' => 'value']); } @@ -2393,7 +2404,7 @@ public function testParseMessageWithInvalidArrayItemThrowsException(): void { $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('Array items must be strings, MessagePart instances, or MessagePartArrayShape'); - + new PromptBuilder($this->registry, ['valid string', 123, 'another string']); } @@ -2410,10 +2421,10 @@ public function testGetModelRequirementsWithAllFileTypes(): void ->withAudioFile(new File('https://example.com/audio.mp3', 'audio/mp3')) ->withVideoFile(new File('https://example.com/video.mp4', 'video/mp4')) ->withImageFile(new File('https://example.com/doc.pdf', 'application/pdf')); - + $requirements = $builder->getModelRequirements(); $options = $requirements->getRequiredOptions(); - + // Find input modalities $inputModalities = null; foreach ($options as $option) { @@ -2422,15 +2433,15 @@ public function testGetModelRequirementsWithAllFileTypes(): void break; } } - + $this->assertNotNull($inputModalities); - + // Check all modality types are present $modalityTypes = []; foreach ($inputModalities as $modality) { $modalityTypes[] = $modality->value; } - + $this->assertContains('text', $modalityTypes); $this->assertContains('image', $modalityTypes); $this->assertContains('audio', $modalityTypes); @@ -2450,15 +2461,15 @@ public function testGetModelRequirementsIncludesConfigOptions(): void ->usingTemperature(0.7) ->usingOutputModalities(ModalityEnum::text(), ModalityEnum::image()) ->asJsonResponse(['type' => 'object']); - + $requirements = $builder->getModelRequirements(); $options = $requirements->getRequiredOptions(); - + // Check that config options are included $optionNames = array_map(function ($option) { return $option->getName(); }, $options); - + $this->assertContains(OptionEnum::maxTokens()->value, $optionNames); $this->assertContains(OptionEnum::temperature()->value, $optionNames); $this->assertContains(OptionEnum::outputModalities()->value, $optionNames); @@ -2475,16 +2486,16 @@ public function testValidateMessagesLastMessageMustHaveParts(): void { // Create a message with empty parts $emptyMessage = new UserMessage([]); - + $builder = new PromptBuilder($this->registry, [ new UserMessage([new MessagePart('First')]), new ModelMessage([new MessagePart('Response')]), $emptyMessage ]); - + $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('The last message must have content parts'); - + $builder->generateResult(); } @@ -2497,29 +2508,32 @@ public function testGenerateImageResultCreatesProperOperation(): void { $result = new GenerativeAiResult( 'test-result', - [new Candidate(new ModelMessage([new MessagePart(new File('', 'image/png'))]), FinishReasonEnum::stop())], + [new Candidate( + new ModelMessage([new MessagePart(new File('', 'image/png'))]), + FinishReasonEnum::stop() + )], new TokenUsage(100, 50, 150) ); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createImageGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate an image'); $builder->usingModel($model); - + $actualResult = $builder->generateImageResult(); $this->assertSame($result, $actualResult); - + // Verify that image modality was included in the model config $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); /** @var ModelConfig $config */ $config = $configProperty->getValue($builder); - + $outputModalities = $config->getOutputModalities(); $this->assertCount(1, $outputModalities); $this->assertTrue($outputModalities[0]->isImage()); @@ -2557,18 +2571,18 @@ public function testGenerateImageReturnsFileDirectly(): void new Message(MessageRoleEnum::model(), [new MessagePart($file)]), FinishReasonEnum::stop() ); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createImageGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate an image'); $builder->usingModel($model); - + $generatedFile = $builder->generateImage(); $this->assertSame($file, $generatedFile); } @@ -2623,19 +2637,19 @@ public function testGenerateTextWithNoCandidatesThrowsException(): void // Create a mock result that returns empty candidates $result = $this->createMock(GenerativeAiResult::class); $result->method('getCandidates')->willReturn([]); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('No candidates were generated'); - + $builder->generateText(); } @@ -2651,21 +2665,21 @@ public function testGenerateTextWithNonStringPartThrowsException(): void new Message(MessageRoleEnum::model(), [new MessagePart($file)]), FinishReasonEnum::stop() ); - + $result = new GenerativeAiResult('test-result', [$candidate], new TokenUsage(100, 50, 150)); - + $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); $metadata->method('meetsRequirements')->willReturn(true); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate text'); $builder->usingModel($model); - + $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Generated message part contains no text'); - + $builder->generateText(); } @@ -2687,7 +2701,7 @@ public function testChainGenerationWithMultiplePrompts(): void public function testIsSupportedWithIntendedOutput(): void { $builder = new PromptBuilder($this->registry, 'Test prompt'); - + // Mock registry to return no models for image generation $this->registry->method('findModelsMetadataForSupport') ->willReturnCallback(function ($requirements) { @@ -2710,10 +2724,10 @@ public function testIsSupportedWithIntendedOutput(): void $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); return [$providerModelsMetadata]; }); - + // Text should be supported $this->assertTrue($builder->isSupported(ModalityEnum::text())); - + // Image should not be supported $this->assertFalse($builder->isSupported(ModalityEnum::image())); } @@ -2727,19 +2741,19 @@ public function testIsSupportedForText(): void { $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('text-model'); - + $result = new GenerativeAiResult('test-id', [ new Candidate( new ModelMessage([new MessagePart('Test')]), FinishReasonEnum::stop() ) ], new TokenUsage(10, 5, 15)); - + $model = $this->createTextGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); - + $this->assertTrue($builder->isSupportedForText()); } @@ -2751,11 +2765,11 @@ public function testIsSupportedForText(): void public function testIsSupportedForImage(): void { $builder = new PromptBuilder($this->registry, 'Generate an image'); - + // Mock registry to return no models for image generation $this->registry->method('findModelsMetadataForSupport') ->willReturn([]); - + $this->assertFalse($builder->isSupportedForImage()); } @@ -2767,11 +2781,11 @@ public function testIsSupportedForImage(): void public function testIsSupportedForAudio(): void { $builder = new PromptBuilder($this->registry, 'Generate audio'); - + // Mock registry to return no models for audio generation $this->registry->method('findModelsMetadataForSupport') ->willReturn([]); - + $this->assertFalse($builder->isSupportedForAudio()); } @@ -2783,11 +2797,11 @@ public function testIsSupportedForAudio(): void public function testIsSupportedForVideo(): void { $builder = new PromptBuilder($this->registry, 'Generate video'); - + // Mock registry to return no models for video generation $this->registry->method('findModelsMetadataForSupport') ->willReturn([]); - + $this->assertFalse($builder->isSupportedForVideo()); } @@ -2800,19 +2814,19 @@ public function testIsSupportedForSpeech(): void { $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('speech-model'); - + $result = new GenerativeAiResult('test-id', [ new Candidate( new ModelMessage([new MessagePart(new File('https://example.com/speech.mp3', 'audio/mp3'))]), FinishReasonEnum::stop() ) ], new TokenUsage(10, 5, 15)); - + $model = $this->createSpeechGenerationModel($metadata, $result); - + $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); - + $this->assertTrue($builder->isSupportedForSpeech()); } @@ -2824,10 +2838,10 @@ public function testIsSupportedForSpeech(): void public function testIsSupportedRestoresOriginalModalities(): void { $builder = new PromptBuilder($this->registry, 'Test prompt'); - + // Set initial modality $builder->usingOutputModalities(ModalityEnum::text()); - + // Mock registry to return models $providerMetadata = $this->createMock(ProviderMetadata::class); $modelMetadata = $this->createMock(ModelMetadata::class); @@ -2836,18 +2850,18 @@ public function testIsSupportedRestoresOriginalModalities(): void $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); $this->registry->method('findModelsMetadataForSupport')->willReturn([$providerModelsMetadata]); $this->registry->method('getProviderModel')->willReturn($this->createMock(ModelInterface::class)); - + // Check with image modality $builder->isSupported(ModalityEnum::image()); - + // Verify original modality is restored $reflection = new \ReflectionClass($builder); $configProperty = $reflection->getProperty('modelConfig'); $configProperty->setAccessible(true); $config = $configProperty->getValue($builder); - + $modalities = $config->getOutputModalities(); $this->assertCount(1, $modalities); $this->assertTrue($modalities[0]->isText()); } -} \ No newline at end of file +} From d976ccfe53bc1cb8cebac98c091d7c1689d1068e Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Sat, 23 Aug 2025 10:41:39 -0600 Subject: [PATCH 25/47] fix: uses output schema as require option value --- src/Providers/Models/DTO/ModelConfig.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Providers/Models/DTO/ModelConfig.php b/src/Providers/Models/DTO/ModelConfig.php index 585dbe2e..69e29a2e 100644 --- a/src/Providers/Models/DTO/ModelConfig.php +++ b/src/Providers/Models/DTO/ModelConfig.php @@ -1015,7 +1015,7 @@ public function toRequiredOptions(): array if ($this->outputSchema !== null) { $requiredOptions[] = new RequiredOption( OptionEnum::outputSchema()->value, - true // Just indicate that schema is required + $this->outputSchema ); } From bba08ddcb9213f23f4954320f6cf7e8181fb63a8 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Sat, 23 Aug 2025 10:56:07 -0600 Subject: [PATCH 26/47] refactor: moves includeOutputModalities to PromptBuilder --- src/Builders/PromptBuilder.php | 52 ++++++- src/Providers/Models/DTO/ModelConfig.php | 30 ---- .../Providers/Models/DTO/ModelConfigTest.php | 139 ------------------ 3 files changed, 47 insertions(+), 174 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 6a0b995d..7fa7634b 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -510,7 +510,7 @@ public function isSupported(?ModalityEnum $intendedOutput = null): bool $originalModalities = null; if ($intendedOutput !== null) { $originalModalities = $this->modelConfig->getOutputModalities(); - $this->modelConfig->includeOutputModality($intendedOutput); + $this->includeOutputModalities($intendedOutput); } try { @@ -683,7 +683,7 @@ public function generateResult(): GenerativeAiResult public function generateTextResult(): GenerativeAiResult { // Include text in output modalities - $this->modelConfig->includeOutputModality(ModalityEnum::text()); + $this->includeOutputModalities(ModalityEnum::text()); // Generate and return the result return $this->generateResult(); @@ -701,7 +701,7 @@ public function generateTextResult(): GenerativeAiResult public function generateImageResult(): GenerativeAiResult { // Include image in output modalities - $this->modelConfig->includeOutputModality(ModalityEnum::image()); + $this->includeOutputModalities(ModalityEnum::image()); // Generate and return the result return $this->generateResult(); @@ -719,7 +719,7 @@ public function generateImageResult(): GenerativeAiResult public function generateSpeechResult(): GenerativeAiResult { // Include audio in output modalities - $this->modelConfig->includeOutputModality(ModalityEnum::audio()); + $this->includeOutputModalities(ModalityEnum::audio()); // Generate and return the result return $this->generateResult(); @@ -737,7 +737,7 @@ public function generateSpeechResult(): GenerativeAiResult public function convertTextToSpeechResult(): GenerativeAiResult { // Include audio in output modalities - $this->modelConfig->includeOutputModality(ModalityEnum::audio()); + $this->includeOutputModalities(ModalityEnum::audio()); // Get the configured model $model = $this->getConfiguredModel(); @@ -1272,4 +1272,46 @@ private function isMessagesList($value): bool return true; } + + /** + * Includes output modalities if not already present. + * + * Adds the given modalities to the output modalities list if they're not + * already included. If output modalities is null, initializes it with + * the given modalities. + * + * @since n.e.x.t + * + * @param ModalityEnum ...$modalities The modalities to include. + * @return void + */ + private function includeOutputModalities(ModalityEnum ...$modalities): void + { + $existing = $this->modelConfig->getOutputModalities(); + + // Initialize if null + if ($existing === null) { + $this->modelConfig->setOutputModalities($modalities); + return; + } + + // Build a set of existing modality values for O(1) lookup + $existingValues = []; + foreach ($existing as $existingModality) { + $existingValues[$existingModality->value] = true; + } + + // Add new modalities that don't exist + $toAdd = []; + foreach ($modalities as $modality) { + if (!isset($existingValues[$modality->value])) { + $toAdd[] = $modality; + } + } + + // Update if we have new modalities to add + if (!empty($toAdd)) { + $this->modelConfig->setOutputModalities(array_merge($existing, $toAdd)); + } + } } diff --git a/src/Providers/Models/DTO/ModelConfig.php b/src/Providers/Models/DTO/ModelConfig.php index 69e29a2e..59194504 100644 --- a/src/Providers/Models/DTO/ModelConfig.php +++ b/src/Providers/Models/DTO/ModelConfig.php @@ -203,36 +203,6 @@ public function getOutputModalities(): ?array return $this->outputModalities; } - /** - * Includes an output modality if not already present. - * - * Adds the given modality to the output modalities list if it's not - * already included. If output modalities is null, initializes it with - * the given modality. - * - * @since n.e.x.t - * - * @param ModalityEnum $modality The modality to include. - */ - public function includeOutputModality(ModalityEnum $modality): void - { - // Initialize if null - if ($this->outputModalities === null) { - $this->outputModalities = [$modality]; - return; - } - - // Check if modality already exists - foreach ($this->outputModalities as $existingModality) { - if ($existingModality === $modality) { - return; // Already included - } - } - - // Add the modality - $this->outputModalities[] = $modality; - } - /** * Sets the system instruction. * diff --git a/tests/unit/Providers/Models/DTO/ModelConfigTest.php b/tests/unit/Providers/Models/DTO/ModelConfigTest.php index 7866928a..7f5b353b 100644 --- a/tests/unit/Providers/Models/DTO/ModelConfigTest.php +++ b/tests/unit/Providers/Models/DTO/ModelConfigTest.php @@ -654,143 +654,4 @@ public function testSetCustomOption(): void $restored = ModelConfig::fromArray($array); $this->assertEquals($customOptions, $restored->getCustomOptions()); } - - /** - * Tests includeOutputModality method. - * - * @return void - */ - public function testIncludeOutputModality(): void - { - $config = new ModelConfig(); - - // Test adding modality to null output modalities - $this->assertNull($config->getOutputModalities()); - $config->includeOutputModality(ModalityEnum::text()); - $modalities = $config->getOutputModalities(); - $this->assertCount(1, $modalities); - $this->assertTrue($modalities[0]->isText()); - - // Test adding a different modality - $config->includeOutputModality(ModalityEnum::image()); - $modalities = $config->getOutputModalities(); - $this->assertCount(2, $modalities); - $this->assertTrue($modalities[0]->isText()); - $this->assertTrue($modalities[1]->isImage()); - - // Test adding a duplicate modality (should not add) - $config->includeOutputModality(ModalityEnum::text()); - $modalities = $config->getOutputModalities(); - $this->assertCount(2, $modalities); - $this->assertTrue($modalities[0]->isText()); - $this->assertTrue($modalities[1]->isImage()); - - // Test adding another unique modality - $config->includeOutputModality(ModalityEnum::audio()); - $modalities = $config->getOutputModalities(); - $this->assertCount(3, $modalities); - $this->assertTrue($modalities[0]->isText()); - $this->assertTrue($modalities[1]->isImage()); - $this->assertTrue($modalities[2]->isAudio()); - - // Test that duplicate modalities are not added (different instance, same value) - $config->includeOutputModality(ModalityEnum::image()); - $modalities = $config->getOutputModalities(); - $this->assertCount(3, $modalities); - } - - /** - * Tests includeOutputModality with existing modalities set via setOutputModalities. - * - * @return void - */ - public function testIncludeOutputModalityWithExistingModalitiesSet(): void - { - $config = new ModelConfig(); - - // Set initial modalities - $config->setOutputModalities([ModalityEnum::text(), ModalityEnum::video()]); - $modalities = $config->getOutputModalities(); - $this->assertCount(2, $modalities); - - // Include a new modality - $config->includeOutputModality(ModalityEnum::image()); - $modalities = $config->getOutputModalities(); - $this->assertCount(3, $modalities); - $this->assertTrue($modalities[0]->isText()); - $this->assertTrue($modalities[1]->isVideo()); - $this->assertTrue($modalities[2]->isImage()); - - // Include an existing modality (should not add) - $config->includeOutputModality(ModalityEnum::video()); - $modalities = $config->getOutputModalities(); - $this->assertCount(3, $modalities); - } - - /** - * Tests includeOutputModality preserves modality order. - * - * @return void - */ - public function testIncludeOutputModalityPreservesOrder(): void - { - $config = new ModelConfig(); - - // Add modalities in specific order - $config->includeOutputModality(ModalityEnum::audio()); - $config->includeOutputModality(ModalityEnum::document()); - $config->includeOutputModality(ModalityEnum::text()); - $config->includeOutputModality(ModalityEnum::image()); - - $modalities = $config->getOutputModalities(); - $this->assertCount(4, $modalities); - $this->assertTrue($modalities[0]->isAudio()); - $this->assertTrue($modalities[1]->isDocument()); - $this->assertTrue($modalities[2]->isText()); - $this->assertTrue($modalities[3]->isImage()); - - // Try to add existing modalities in different order (should not change) - $config->includeOutputModality(ModalityEnum::text()); - $config->includeOutputModality(ModalityEnum::audio()); - - $modalities = $config->getOutputModalities(); - $this->assertCount(4, $modalities); - $this->assertTrue($modalities[0]->isAudio()); - $this->assertTrue($modalities[1]->isDocument()); - $this->assertTrue($modalities[2]->isText()); - $this->assertTrue($modalities[3]->isImage()); - } - - /** - * Tests includeOutputModality handles all modality types. - * - * @return void - */ - public function testIncludeOutputModalityHandlesAllModalityTypes(): void - { - $config = new ModelConfig(); - - // Test all available modality types - $allModalities = [ - ModalityEnum::text(), - ModalityEnum::image(), - ModalityEnum::audio(), - ModalityEnum::video(), - ModalityEnum::document() - ]; - - foreach ($allModalities as $modality) { - $config->includeOutputModality($modality); - } - - $modalities = $config->getOutputModalities(); - $this->assertCount(5, $modalities); - - // Verify all modalities are present - $this->assertTrue($modalities[0]->isText()); - $this->assertTrue($modalities[1]->isImage()); - $this->assertTrue($modalities[2]->isAudio()); - $this->assertTrue($modalities[3]->isVideo()); - $this->assertTrue($modalities[4]->isDocument()); - } } From 9281c54b7ce3b3e533d4e6f409814cd72bea6fbe Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Sat, 23 Aug 2025 11:02:22 -0600 Subject: [PATCH 27/47] refactor: improves handling of message arrays --- src/Builders/PromptBuilder.php | 23 +++++++++++------------ 1 file changed, 11 insertions(+), 12 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 7fa7634b..e623f7e9 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -86,12 +86,6 @@ public function __construct(ProviderRegistry $registry, $prompt = null) return; } - // Check if it's a MessageArrayShape - add to messages - if (is_array($prompt) && Message::isArrayShape($prompt)) { - $this->messages[] = Message::fromArray($prompt); - return; - } - // Parse it as a user message $userMessage = $this->parseMessage($prompt, MessageRoleEnum::user()); $this->messages[] = $userMessage; @@ -1140,11 +1134,11 @@ public function getConfiguredModel(?ModelRequirements $requirements = null): Mod * @since n.e.x.t * * @param mixed $input The input to parse. - * @param MessageRoleEnum $role The role for the message. + * @param MessageRoleEnum $defaultRole The role for the message if not specified by input. * @return Message The parsed message. * @throws InvalidArgumentException If the input type is not supported or results in empty message. */ - private function parseMessage($input, MessageRoleEnum $role): Message + private function parseMessage($input, MessageRoleEnum $defaultRole): Message { // Handle Message input directly if ($input instanceof Message) { @@ -1153,7 +1147,7 @@ private function parseMessage($input, MessageRoleEnum $role): Message // Handle single MessagePart if ($input instanceof MessagePart) { - return new Message($role, [$input]); + return new Message($defaultRole, [$input]); } // Handle string input @@ -1161,7 +1155,7 @@ private function parseMessage($input, MessageRoleEnum $role): Message if (trim($input) === '') { throw new InvalidArgumentException('Cannot create a message from an empty string.'); } - return new Message($role, [new MessagePart($input)]); + return new Message($defaultRole, [new MessagePart($input)]); } // Handle array input @@ -1172,9 +1166,14 @@ private function parseMessage($input, MessageRoleEnum $role): Message ); } + // Handle MessageArrayShape input + if (Message::isArrayShape($input)) { + return Message::fromArray($input); + } + // Check if it's a MessagePartArrayShape if (MessagePart::isArrayShape($input)) { - return new Message($role, [MessagePart::fromArray($input)]); + return new Message($defaultRole, [MessagePart::fromArray($input)]); } // It should be a list of string|MessagePart|MessagePartArrayShape @@ -1202,7 +1201,7 @@ private function parseMessage($input, MessageRoleEnum $role): Message } } - return new Message($role, $parts); + return new Message($defaultRole, $parts); } /** From d7320ce93c4b4ece0409acfc44cf340d11e64197 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Sat, 23 Aug 2025 11:26:09 -0600 Subject: [PATCH 28/47] refactor: consolidates builder file methods --- src/Builders/PromptBuilder.php | 79 ++++------------------- tests/unit/Builders/PromptBuilderTest.php | 76 +++++++++------------- 2 files changed, 41 insertions(+), 114 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index e623f7e9..7b8ff215 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -107,85 +107,30 @@ public function withText(string $text): self } /** - * Adds an inline image to the current message. + * Adds a file to the current message. * - * @since n.e.x.t - * - * @param string $base64Blob The base64-encoded image data. - * @param string $mimeType The MIME type of the image. - * @return self - */ - public function withInlineImage(string $base64Blob, string $mimeType): self - { - // Create data URI format for inline image - $dataUri = 'data:' . $mimeType . ';base64,' . $base64Blob; - $file = new File($dataUri, $mimeType); - $part = new MessagePart($file); - $this->appendPartToMessages($part); - return $this; - } - - /** - * Adds a remote image to the current message. - * - * @since n.e.x.t - * - * @param string $uri The URI of the remote image. - * @param string $mimeType The MIME type of the image. - * @return self - */ - public function withRemoteImage(string $uri, string $mimeType): self - { - $file = new File($uri, $mimeType); - $part = new MessagePart($file); - $this->appendPartToMessages($part); - return $this; - } - - /** - * Adds an image file to the current message. + * Accepts: + * - File object + * - URL string (remote file) + * - Base64-encoded data string + * - Data URI string (data:mime/type;base64,data) + * - Local file path string * * @since n.e.x.t * - * @param File $file The image file. + * @param string|File $file The file (File object or string representation). + * @param string|null $mimeType The MIME type (optional, ignored if File object provided). * @return self + * @throws InvalidArgumentException If the file is invalid or MIME type cannot be determined. */ - public function withImageFile(File $file): self + public function withFile($file, ?string $mimeType = null): self { + $file = $file instanceof File ? $file : new File($file, $mimeType); $part = new MessagePart($file); $this->appendPartToMessages($part); return $this; } - /** - * Adds an audio file to the current message. - * - * @since n.e.x.t - * - * @param File $file The audio file. - * @return self - */ - public function withAudioFile(File $file): self - { - $part = new MessagePart($file); - $this->appendPartToMessages($part); - return $this; - } - - /** - * Adds a video file to the current message. - * - * @since n.e.x.t - * - * @param File $file The video file. - * @return self - */ - public function withVideoFile(File $file): self - { - $part = new MessagePart($file); - $this->appendPartToMessages($part); - return $this; - } /** * Adds a function response to the current message. diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index 557c6c48..b9db5aca 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -410,15 +410,15 @@ public function testWithTextAppendsToExistingUserMessage(): void } /** - * Tests withInlineImage method. + * Tests withFile method with base64 data. * * @return void */ - public function testWithInlineImage(): void + public function testWithInlineFile(): void { $builder = new PromptBuilder($this->registry); $base64 = 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNkYPhfDwAChwGA60e6kgAAAABJRU5ErkJggg=='; - $result = $builder->withInlineImage($base64, 'image/png'); + $result = $builder->withFile($base64, 'image/png'); $this->assertSame($builder, $result); @@ -436,14 +436,14 @@ public function testWithInlineImage(): void } /** - * Tests withRemoteImage method. + * Tests withFile method with remote URL. * * @return void */ - public function testWithRemoteImage(): void + public function testWithRemoteFile(): void { $builder = new PromptBuilder($this->registry); - $result = $builder->withRemoteImage('https://example.com/image.jpg', 'image/jpeg'); + $result = $builder->withFile('https://example.com/image.jpg', 'image/jpeg'); $this->assertSame($builder, $result); @@ -461,15 +461,15 @@ public function testWithRemoteImage(): void } /** - * Tests withImageFile method. + * Tests withFile with data URI. * * @return void */ - public function testWithImageFile(): void + public function testWithInlineFileDataUri(): void { - $file = new File('https://example.com/test.png', 'image/png'); $builder = new PromptBuilder($this->registry); - $result = $builder->withImageFile($file); + $dataUri = ''; + $result = $builder->withFile($dataUri); $this->assertSame($builder, $result); @@ -480,42 +480,21 @@ public function testWithImageFile(): void $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); - $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); - } - - /** - * Tests withAudioFile method. - * - * @return void - */ - public function testWithAudioFile(): void - { - $file = new File('https://example.com/audio.mp3', 'audio/mp3'); - $builder = new PromptBuilder($this->registry); - $result = $builder->withAudioFile($file); - - $this->assertSame($builder, $result); - - $reflection = new \ReflectionClass($builder); - $messagesProperty = $reflection->getProperty('messages'); - $messagesProperty->setAccessible(true); - /** @var list $messages */ - $messages = $messagesProperty->getValue($builder); - - $this->assertCount(1, $messages); - $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); + $file = $messages[0]->getParts()[0]->getFile(); + $this->assertInstanceOf(File::class, $file); + $this->assertEquals('image/jpeg', $file->getMimeType()); } /** - * Tests withVideoFile method. + * Tests withFile with URL without explicit MIME type. * * @return void */ - public function testWithVideoFile(): void + public function testWithRemoteFileWithoutMimeType(): void { - $file = new File('https://example.com/video.mp4', 'video/mp4'); $builder = new PromptBuilder($this->registry); - $result = $builder->withVideoFile($file); + // File extension should be used to determine MIME type + $result = $builder->withFile('https://example.com/audio.mp3'); $this->assertSame($builder, $result); @@ -526,7 +505,10 @@ public function testWithVideoFile(): void $messages = $messagesProperty->getValue($builder); $this->assertCount(1, $messages); - $this->assertSame($file, $messages[0]->getParts()[0]->getFile()); + $file = $messages[0]->getParts()[0]->getFile(); + $this->assertInstanceOf(File::class, $file); + $this->assertEquals('https://example.com/audio.mp3', $file->getUrl()); + $this->assertEquals('audio/mpeg', $file->getMimeType()); } /** @@ -995,7 +977,7 @@ public function testGetModelRequirementsWithMultimodalInput(): void { $builder = new PromptBuilder($this->registry); $builder->withText('Describe this image') - ->withRemoteImage('https://example.com/image.jpg', 'image/jpeg'); + ->withFile('https://example.com/image.jpg', 'image/jpeg'); $requirements = $builder->getModelRequirements(); $options = $requirements->getRequiredOptions(); @@ -1192,7 +1174,7 @@ public function testMethodChaining(): void $builder = new PromptBuilder($this->registry); $result = $builder ->withText('Start of prompt') - ->withRemoteImage('https://example.com/img.jpg', 'image/jpeg') + ->withFile('https://example.com/img.jpg', 'image/jpeg') ->usingModel($model) ->usingSystemInstruction('Be helpful') ->usingMaxTokens(500) @@ -2253,9 +2235,9 @@ public function testComplexMultimodalPromptBuilding(): void $builder = new PromptBuilder($this->registry); $builder->withText('Analyze this data:') - ->withImageFile($file1) + ->withFile($file1) ->withText(' and this audio:') - ->withAudioFile($file2) + ->withFile($file2) ->withFunctionResponse($functionResponse) ->withHistory( new UserMessage([new MessagePart('Previous question')]), @@ -2417,10 +2399,10 @@ public function testGetModelRequirementsWithAllFileTypes(): void { $builder = new PromptBuilder($this->registry); $builder->withText('Analyze:') - ->withRemoteImage('https://example.com/img.jpg', 'image/jpeg') - ->withAudioFile(new File('https://example.com/audio.mp3', 'audio/mp3')) - ->withVideoFile(new File('https://example.com/video.mp4', 'video/mp4')) - ->withImageFile(new File('https://example.com/doc.pdf', 'application/pdf')); + ->withFile('https://example.com/img.jpg', 'image/jpeg') + ->withFile('https://example.com/audio.mp3', 'audio/mp3') + ->withFile('https://example.com/video.mp4', 'video/mp4') + ->withFile('https://example.com/doc.pdf', 'application/pdf'); $requirements = $builder->getModelRequirements(); $options = $requirements->getRequiredOptions(); From cb54e0b3395b85dbe97153dcb11fe5ad7f7d80ce Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Sat, 23 Aug 2025 11:26:26 -0600 Subject: [PATCH 29/47] feat: adds file isInline and isRemote methods --- src/Files/DTO/File.php | 24 +++++++++++++++ tests/unit/Files/DTO/FileTest.php | 49 +++++++++++++++++++++++++++++++ 2 files changed, 73 insertions(+) diff --git a/src/Files/DTO/File.php b/src/Files/DTO/File.php index e98cf935..ee731ff9 100644 --- a/src/Files/DTO/File.php +++ b/src/Files/DTO/File.php @@ -174,6 +174,30 @@ public function getFileType(): FileTypeEnum return $this->fileType; } + /** + * Checks if the file is an inline file. + * + * @since n.e.x.t + * + * @return bool True if the file is inline (base64/data URI). + */ + public function isInline(): bool + { + return $this->fileType->isInline(); + } + + /** + * Checks if the file is a remote file. + * + * @since n.e.x.t + * + * @return bool True if the file is remote (URL). + */ + public function isRemote(): bool + { + return $this->fileType->isRemote(); + } + /** * Gets the URL for remote files. * diff --git a/tests/unit/Files/DTO/FileTest.php b/tests/unit/Files/DTO/FileTest.php index 52a9730f..84d6adcf 100644 --- a/tests/unit/Files/DTO/FileTest.php +++ b/tests/unit/Files/DTO/FileTest.php @@ -520,4 +520,53 @@ public function testIsMimeType(): void $this->assertTrue($textFile->isMimeType('text')); $this->assertFalse($textFile->isMimeType('image')); } + + /** + * Tests isInline method for inline files. + * + * @return void + */ + public function testIsInlineForInlineFiles(): void + { + // Test with base64 data + $base64File = new File('SGVsbG8gV29ybGQ=', 'text/plain'); + $this->assertTrue($base64File->isInline()); + $this->assertFalse($base64File->isRemote()); + + // Test with data URI + $dataUri = '' + . 'AAADUJEQVR42mNkYPhfDwAChwGA60e6kgAAAABJRU5ErkJggg=='; + $dataUriFile = new File($dataUri); + $this->assertTrue($dataUriFile->isInline()); + $this->assertFalse($dataUriFile->isRemote()); + + // Test with local file (becomes inline) + $tempFile = tempnam(sys_get_temp_dir(), 'test'); + file_put_contents($tempFile, 'test content'); + try { + $localFile = new File($tempFile, 'text/plain'); + $this->assertTrue($localFile->isInline()); + $this->assertFalse($localFile->isRemote()); + } finally { + unlink($tempFile); + } + } + + /** + * Tests isRemote method for remote files. + * + * @return void + */ + public function testIsRemoteForRemoteFiles(): void + { + // Test with URL + $urlFile = new File('https://example.com/image.jpg', 'image/jpeg'); + $this->assertTrue($urlFile->isRemote()); + $this->assertFalse($urlFile->isInline()); + + // Test with URL without explicit MIME type + $urlFileNoMime = new File('https://example.com/document.pdf'); + $this->assertTrue($urlFileNoMime->isRemote()); + $this->assertFalse($urlFileNoMime->isInline()); + } } From f324b961a0956aa7bc645d8d4908d0ee2e820d44 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Mon, 25 Aug 2025 10:21:33 -0600 Subject: [PATCH 30/47] refactor: removes usingRegistry method for now --- src/Builders/PromptBuilder.php | 14 -------------- tests/unit/Builders/PromptBuilderTest.php | 22 ---------------------- 2 files changed, 36 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 7b8ff215..65b0fe32 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -194,20 +194,6 @@ public function usingModel(ModelInterface $model): self return $this; } - /** - * Sets a different provider registry. - * - * @since n.e.x.t - * - * @param ProviderRegistry $registry The provider registry to use. - * @return self - */ - public function usingRegistry(ProviderRegistry $registry): self - { - $this->registry = $registry; - return $this; - } - /** * Sets the system instruction. * diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index b9db5aca..51429508 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -616,28 +616,6 @@ public function testUsingModel(): void $this->assertSame($model, $actualModel); } - /** - * Tests usingRegistry method. - * - * @return void - */ - public function testUsingRegistry(): void - { - $newRegistry = $this->createMock(ProviderRegistry::class); - $builder = new PromptBuilder($this->registry); - $result = $builder->usingRegistry($newRegistry); - - $this->assertSame($builder, $result); - - $reflection = new \ReflectionClass($builder); - $registryProperty = $reflection->getProperty('registry'); - $registryProperty->setAccessible(true); - - /** @var ProviderRegistry $actualRegistry */ - $actualRegistry = $registryProperty->getValue($builder); - $this->assertSame($newRegistry, $actualRegistry); - } - /** * Tests usingSystemInstruction method. * From a07c2428a8d7010b09d8a38e748d50f549afa1d5 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Mon, 25 Aug 2025 10:33:40 -0600 Subject: [PATCH 31/47] feat: validates message parts --- src/Messages/DTO/Message.php | 29 ++++ tests/unit/Messages/DTO/MessageTest.php | 141 +++++++++++++++++-- tests/unit/Messages/DTO/ModelMessageTest.php | 6 +- 3 files changed, 158 insertions(+), 18 deletions(-) diff --git a/src/Messages/DTO/Message.php b/src/Messages/DTO/Message.php index 240608cf..c07ca9e5 100644 --- a/src/Messages/DTO/Message.php +++ b/src/Messages/DTO/Message.php @@ -4,6 +4,7 @@ namespace WordPress\AiClient\Messages\DTO; +use InvalidArgumentException; use WordPress\AiClient\Common\AbstractDataTransferObject; use WordPress\AiClient\Messages\Enums\MessageRoleEnum; @@ -45,11 +46,13 @@ class Message extends AbstractDataTransferObject * * @param MessageRoleEnum $role The role of the message sender. * @param MessagePart[] $parts The parts that make up this message. + * @throws InvalidArgumentException If parts contain invalid content for the role. */ public function __construct(MessageRoleEnum $role, array $parts) { $this->role = $role; $this->parts = $parts; + $this->validateParts(); } /** @@ -83,6 +86,7 @@ public function getParts(): array * * @param MessagePart $part The part to append. * @return Message A new instance with the part appended. + * @throws InvalidArgumentException If the part is invalid for the role. */ public function withPart(MessagePart $part): Message { @@ -92,6 +96,31 @@ public function withPart(MessagePart $part): Message return new Message($this->role, $newParts); } + /** + * Validates that the message parts are appropriate for the message role. + * + * @since n.e.x.t + * + * @return void + * @throws InvalidArgumentException If validation fails. + */ + private function validateParts(): void + { + foreach ($this->parts as $part) { + if ($this->role->isUser() && $part->getType()->isFunctionCall()) { + throw new InvalidArgumentException( + 'User messages cannot contain function calls.' + ); + } + + if ($this->role->isModel() && $part->getType()->isFunctionResponse()) { + throw new InvalidArgumentException( + 'Model messages cannot contain function responses.' + ); + } + } + } + /** * {@inheritDoc} * diff --git a/tests/unit/Messages/DTO/MessageTest.php b/tests/unit/Messages/DTO/MessageTest.php index 0edc5f85..db77cad8 100644 --- a/tests/unit/Messages/DTO/MessageTest.php +++ b/tests/unit/Messages/DTO/MessageTest.php @@ -108,28 +108,40 @@ public function roleProvider(): array */ public function testComplexMessageWithAllPartTypes(): void { - $role = MessageRoleEnum::model(); + // Test with user role since it can have function responses but not function calls + $role = MessageRoleEnum::user(); $parts = [ - new MessagePart('I\'ll help you with that. Let me search for the information.'), - new MessagePart(new FunctionCall('search_123', 'webSearch', ['query' => 'latest PHP news'])), + new MessagePart('I need help with searching.'), new MessagePart(new FunctionResponse('search_123', 'webSearch', ['results' => ['item1', 'item2']])), - new MessagePart('Based on my search, here are the latest PHP news:'), + new MessagePart('Here is additional information:'), new MessagePart(new File('data:text/plain;base64,SGVsbG8=', 'text/plain')), ]; $message = new Message($role, $parts); - $this->assertCount(5, $message->getParts()); + $this->assertCount(4, $message->getParts()); // Verify each part type $this->assertEquals( - 'I\'ll help you with that. Let me search for the information.', + 'I need help with searching.', $message->getParts()[0]->getText() ); - $this->assertInstanceOf(FunctionCall::class, $message->getParts()[1]->getFunctionCall()); - $this->assertInstanceOf(FunctionResponse::class, $message->getParts()[2]->getFunctionResponse()); - $this->assertEquals('Based on my search, here are the latest PHP news:', $message->getParts()[3]->getText()); - $this->assertInstanceOf(File::class, $message->getParts()[4]->getFile()); + $this->assertInstanceOf(FunctionResponse::class, $message->getParts()[1]->getFunctionResponse()); + $this->assertEquals('Here is additional information:', $message->getParts()[2]->getText()); + $this->assertInstanceOf(File::class, $message->getParts()[3]->getFile()); + + // Also test model role with function calls + $modelRole = MessageRoleEnum::model(); + $modelParts = [ + new MessagePart('I\'ll help you with that. Let me search for the information.'), + new MessagePart(new FunctionCall('search_123', 'webSearch', ['query' => 'latest PHP news'])), + new MessagePart('Based on my search, here are the latest PHP news:'), + ]; + + $modelMessage = new Message($modelRole, $modelParts); + + $this->assertCount(3, $modelMessage->getParts()); + $this->assertInstanceOf(FunctionCall::class, $modelMessage->getParts()[1]->getFunctionCall()); } /** @@ -258,13 +270,13 @@ public function testPreservesPartOrder(): void } /** - * Tests model message with function response. + * Tests that user message can have function response. * * @return void */ - public function testModelMessageWithFunctionResponse(): void + public function testUserMessageWithFunctionResponse(): void { - $role = MessageRoleEnum::model(); + $role = MessageRoleEnum::user(); $functionResponse = new FunctionResponse( 'calc_123', 'calculate', @@ -274,7 +286,7 @@ public function testModelMessageWithFunctionResponse(): void $message = new Message($role, [$part]); - $this->assertTrue($message->getRole()->isModel()); + $this->assertTrue($message->getRole()->isUser()); $this->assertNotNull($message->getParts()[0]->getFunctionResponse()); } @@ -395,4 +407,105 @@ public function testWithPartCreatesNewInstance(): void $this->assertEquals('Original text', $updated->getParts()[0]->getText()); $this->assertEquals('Additional text', $updated->getParts()[1]->getText()); } + + /** + * Tests that user messages cannot contain function call parts. + * + * @return void + */ + public function testUserMessageCannotContainFunctionCall(): void + { + $this->expectException(\InvalidArgumentException::class); + $this->expectExceptionMessage('User messages cannot contain function calls.'); + + $functionCall = new FunctionCall('testFunc', 'test', ['param' => 'value']); + $part = new MessagePart($functionCall); + + new Message(MessageRoleEnum::user(), [$part]); + } + + /** + * Tests that model messages cannot contain function response parts. + * + * @return void + */ + public function testModelMessageCannotContainFunctionResponse(): void + { + $this->expectException(\InvalidArgumentException::class); + $this->expectExceptionMessage('Model messages cannot contain function responses.'); + + $functionResponse = new FunctionResponse('resp1', 'test', ['result' => 'value']); + $part = new MessagePart($functionResponse); + + new Message(MessageRoleEnum::model(), [$part]); + } + + /** + * Tests that user messages can contain function response parts. + * + * @return void + */ + public function testUserMessageCanContainFunctionResponse(): void + { + $functionResponse = new FunctionResponse('resp1', 'test', ['result' => 'value']); + $part = new MessagePart($functionResponse); + + $message = new Message(MessageRoleEnum::user(), [$part]); + + $this->assertCount(1, $message->getParts()); + $this->assertSame($functionResponse, $message->getParts()[0]->getFunctionResponse()); + } + + /** + * Tests that model messages can contain function call parts. + * + * @return void + */ + public function testModelMessageCanContainFunctionCall(): void + { + $functionCall = new FunctionCall('call1', 'test', ['param' => 'value']); + $part = new MessagePart($functionCall); + + $message = new Message(MessageRoleEnum::model(), [$part]); + + $this->assertCount(1, $message->getParts()); + $this->assertSame($functionCall, $message->getParts()[0]->getFunctionCall()); + } + + /** + * Tests that withPart validates the new part against the role. + * + * @return void + */ + public function testWithPartValidatesAgainstRole(): void + { + $message = new Message(MessageRoleEnum::user(), [new MessagePart('Initial text')]); + + $this->expectException(\InvalidArgumentException::class); + $this->expectExceptionMessage('User messages cannot contain function calls.'); + + $functionCall = new FunctionCall('call1', 'test', ['param' => 'value']); + $invalidPart = new MessagePart($functionCall); + + $message->withPart($invalidPart); + } + + /** + * Tests validation with multiple parts including invalid ones. + * + * @return void + */ + public function testValidationWithMixedParts(): void + { + $this->expectException(\InvalidArgumentException::class); + $this->expectExceptionMessage('User messages cannot contain function calls.'); + + $parts = [ + new MessagePart('Text part'), + new MessagePart(new File('https://example.com/image.jpg', 'image/jpeg')), + new MessagePart(new FunctionCall('call1', 'test', [])), // Invalid for user role + ]; + + new Message(MessageRoleEnum::user(), $parts); + } } diff --git a/tests/unit/Messages/DTO/ModelMessageTest.php b/tests/unit/Messages/DTO/ModelMessageTest.php index f315defe..c7c25e25 100644 --- a/tests/unit/Messages/DTO/ModelMessageTest.php +++ b/tests/unit/Messages/DTO/ModelMessageTest.php @@ -12,7 +12,6 @@ use WordPress\AiClient\Messages\Enums\MessageRoleEnum; use WordPress\AiClient\Tests\traits\ArrayTransformationTestTrait; use WordPress\AiClient\Tools\DTO\FunctionCall; -use WordPress\AiClient\Tools\DTO\FunctionResponse; /** * @covers \WordPress\AiClient\Messages\DTO\ModelMessage @@ -92,13 +91,12 @@ public function testWithVariousContentTypes(): void { $file = new File('https://example.com/image.jpg', 'image/jpeg'); $functionCall = new FunctionCall('func_123', 'search', ['q' => 'test']); - $functionResponse = new FunctionResponse('func_123', 'search', ['results' => []]); $parts = [ new MessagePart('I found the following:'), new MessagePart($file), new MessagePart($functionCall), - new MessagePart($functionResponse), + new MessagePart('Here are the results based on my search.'), ]; $message = new ModelMessage($parts); @@ -106,7 +104,7 @@ public function testWithVariousContentTypes(): void $this->assertEquals('I found the following:', $message->getParts()[0]->getText()); $this->assertSame($file, $message->getParts()[1]->getFile()); $this->assertSame($functionCall, $message->getParts()[2]->getFunctionCall()); - $this->assertSame($functionResponse, $message->getParts()[3]->getFunctionResponse()); + $this->assertEquals('Here are the results based on my search.', $message->getParts()[3]->getText()); } /** From 0d39e740901a4028ac72de19dfcc3b6a418e401e Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Mon, 25 Aug 2025 10:47:25 -0600 Subject: [PATCH 32/47] fix: prepends message in withHistory --- src/Builders/PromptBuilder.php | 8 +- src/Messages/DTO/ModelMessage.php | 4 + src/Messages/DTO/UserMessage.php | 4 + tests/unit/Builders/PromptBuilderTest.php | 103 +++++++++++++++----- tests/unit/Messages/DTO/UserMessageTest.php | 6 +- 5 files changed, 93 insertions(+), 32 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 65b0fe32..66632178 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -166,6 +166,9 @@ public function withMessageParts(MessagePart ...$parts): self /** * Adds conversation history messages. * + * Historical messages are prepended to the beginning of the message list, + * before the current message being built. + * * @since n.e.x.t * * @param Message ...$messages The messages to add to history. @@ -173,9 +176,8 @@ public function withMessageParts(MessagePart ...$parts): self */ public function withHistory(Message ...$messages): self { - foreach ($messages as $message) { - $this->messages[] = $message; - } + // Prepend the history messages to the beginning of the messages array + $this->messages = array_merge($messages, $this->messages); return $this; } diff --git a/src/Messages/DTO/ModelMessage.php b/src/Messages/DTO/ModelMessage.php index cf67b79c..2b160cc3 100644 --- a/src/Messages/DTO/ModelMessage.php +++ b/src/Messages/DTO/ModelMessage.php @@ -12,6 +12,10 @@ * This is a convenience class that automatically sets the role to MODEL. * Model messages contain the AI's responses. * + * Important: Do not rely on `instanceof ModelMessage` to determine the message role. + * This is merely a helper class for construction. Always use `$message->getRole()` + * to check the role of a message. + * * @since n.e.x.t */ class ModelMessage extends Message diff --git a/src/Messages/DTO/UserMessage.php b/src/Messages/DTO/UserMessage.php index c0cb931f..84aded05 100644 --- a/src/Messages/DTO/UserMessage.php +++ b/src/Messages/DTO/UserMessage.php @@ -11,6 +11,10 @@ * * This is a convenience class that automatically sets the role to USER. * + * Important: Do not rely on `instanceof UserMessage` to determine the message role. + * This is merely a helper class for construction. Always use `$message->getRole()` + * to check the role of a message. + * * @since n.e.x.t */ class UserMessage extends Message diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index 51429508..744ff8d7 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -1087,13 +1087,23 @@ public function testValidateMessagesNonUserFirstThrowsException(): void */ public function testValidateMessagesNonUserLastThrowsException(): void { - $builder = new PromptBuilder($this->registry, [ - new UserMessage([new MessagePart('User says hi')]), - new ModelMessage([new MessagePart('Model response')]) - ]); + // Start with a user message + $builder = new PromptBuilder($this->registry); + $builder->withText('Initial user message'); + + // Add history that will make the last message a model message + $builder->withHistory( + new UserMessage([new MessagePart('Historical user message')]), + new ModelMessage([new MessagePart('Historical model response')]) + ); - // Add a user message to make it valid, then add model message - $builder->withHistory(new ModelMessage([new MessagePart('Another model message')])); + // Now add a model message manually to be the last message + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + $messages = $messagesProperty->getValue($builder); + $messages[] = new ModelMessage([new MessagePart('Final model message')]); + $messagesProperty->setValue($builder, $messages); $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('The last message must be from a user role'); @@ -2229,26 +2239,67 @@ public function testComplexMultimodalPromptBuilding(): void /** @var list $messages */ $messages = $messagesProperty->getValue($builder); - // Should have 4 messages: 1 initial + 2 from history + 1 final - $this->assertCount(4, $messages); - - // Check first message with initial content - $firstParts = $messages[0]->getParts(); - $this->assertCount(5, $firstParts); // text, image, text, audio, function response - $this->assertEquals('Analyze this data:', $firstParts[0]->getText()); - $this->assertSame($file1, $firstParts[1]->getFile()); - $this->assertEquals(' and this audio:', $firstParts[2]->getText()); - $this->assertSame($file2, $firstParts[3]->getFile()); - $this->assertSame($functionResponse, $firstParts[4]->getFunctionResponse()); - - // Check history messages - $this->assertEquals('Previous question', $messages[1]->getParts()[0]->getText()); - $this->assertEquals('Previous answer', $messages[2]->getParts()[0]->getText()); - - // Check final message - $finalParts = $messages[3]->getParts(); - $this->assertCount(1, $finalParts); - $this->assertEquals(' Final instruction', $finalParts[0]->getText()); + // Should have 3 messages: 2 from history + 1 current being built + $this->assertCount(3, $messages); + + // Check history messages (now at the beginning) + $this->assertEquals('Previous question', $messages[0]->getParts()[0]->getText()); + $this->assertEquals('Previous answer', $messages[1]->getParts()[0]->getText()); + + // Check current message being built (now at the end) + $currentParts = $messages[2]->getParts(); + $this->assertCount(6, $currentParts); // text, image, text, audio, function response, final text + $this->assertEquals('Analyze this data:', $currentParts[0]->getText()); + $this->assertSame($file1, $currentParts[1]->getFile()); + $this->assertEquals(' and this audio:', $currentParts[2]->getText()); + $this->assertSame($file2, $currentParts[3]->getFile()); + $this->assertSame($functionResponse, $currentParts[4]->getFunctionResponse()); + $this->assertEquals(' Final instruction', $currentParts[5]->getText()); + } + + /** + * Tests that withHistory prepends messages to the beginning. + * + * @return void + */ + public function testWithHistoryPrependsMessages(): void + { + $builder = new PromptBuilder($this->registry); + + // Start building current message + $builder->withText('Current message content'); + + // Add history + $builder->withHistory( + new UserMessage([new MessagePart('First history message')]), + new ModelMessage([new MessagePart('Second history message')]) + ); + + // Add more to current message + $builder->withText(' with additional content'); + + $reflection = new \ReflectionClass($builder); + $messagesProperty = $reflection->getProperty('messages'); + $messagesProperty->setAccessible(true); + /** @var list $messages */ + $messages = $messagesProperty->getValue($builder); + + // Should have 3 messages: 2 history + 1 current + $this->assertCount(3, $messages); + + // History should be at the beginning + $this->assertTrue($messages[0]->getRole()->isUser()); + $this->assertEquals('First history message', $messages[0]->getParts()[0]->getText()); + + $this->assertTrue($messages[1]->getRole()->isModel()); + $this->assertEquals('Second history message', $messages[1]->getParts()[0]->getText()); + + // Current message should be at the end + $this->assertTrue($messages[2]->getRole()->isUser()); + $currentParts = $messages[2]->getParts(); + $this->assertCount(2, $currentParts); + $this->assertEquals('Current message content', $currentParts[0]->getText()); + $this->assertEquals(' with additional content', $currentParts[1]->getText()); } /** diff --git a/tests/unit/Messages/DTO/UserMessageTest.php b/tests/unit/Messages/DTO/UserMessageTest.php index 16428e9d..5b5df92f 100644 --- a/tests/unit/Messages/DTO/UserMessageTest.php +++ b/tests/unit/Messages/DTO/UserMessageTest.php @@ -313,20 +313,20 @@ public function testImplementsWithArrayTransformationInterface(): void } /** - * Tests that withPart returns a base Message with user role. + * Tests that withPart returns a new Message with user role. * * @since n.e.x.t */ - public function testWithPartReturnsBaseMessage(): void + public function testWithPartReturnsNewMessage(): void { $original = new UserMessage([new MessagePart('User text')]); $updated = $original->withPart(new MessagePart('More text')); $this->assertInstanceOf(Message::class, $updated); - $this->assertNotInstanceOf(UserMessage::class, $updated); $this->assertNotSame($original, $updated); $this->assertCount(2, $updated->getParts()); $this->assertEquals(MessageRoleEnum::user(), $updated->getRole()); + $this->assertTrue($updated->getRole()->isUser()); $this->assertEquals('User text', $updated->getParts()[0]->getText()); $this->assertEquals('More text', $updated->getParts()[1]->getText()); } From 7d25a0389c9634aab1826da23d72f3c25a638ad5 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Mon, 25 Aug 2025 14:03:20 -0600 Subject: [PATCH 33/47] feat: turns OptionEnum into a superset of ModelConfig --- src/Common/AbstractEnum.php | 14 +- src/Providers/Models/Enums/OptionEnum.php | 126 +++++++++++------- .../Providers/Models/Enums/OptionEnumTest.php | 86 ++++++++++-- 3 files changed, 165 insertions(+), 61 deletions(-) diff --git a/src/Common/AbstractEnum.php b/src/Common/AbstractEnum.php index c8d816c0..a339c74e 100644 --- a/src/Common/AbstractEnum.php +++ b/src/Common/AbstractEnum.php @@ -51,7 +51,7 @@ abstract class AbstractEnum /** * @var array> Cache for reflection data. */ - private static array $cache = []; + protected static array $cache = []; /** * @var array> Cache for enum instances. @@ -138,7 +138,7 @@ final public static function from(string $value): self */ final public static function tryFrom(string $value): ?self { - $constants = self::getConstants(); + $constants = static::getConstants(); foreach ($constants as $name => $constantValue) { if ($constantValue === $value) { return self::getInstance($constantValue, $name); @@ -157,7 +157,7 @@ final public static function tryFrom(string $value): ?self final public static function cases(): array { $cases = []; - $constants = self::getConstants(); + $constants = static::getConstants(); foreach ($constants as $name => $value) { $cases[] = self::getInstance($value, $name); } @@ -203,7 +203,7 @@ final public function is(self $other): bool */ final public static function getValues(): array { - return array_values(self::getConstants()); + return array_values(static::getConstants()); } /** @@ -253,7 +253,7 @@ private static function getInstance(string $value, string $name): self * @return array Map of constant names to values. * @throws RuntimeException If invalid constant found. */ - final protected static function getConstants(): array + protected static function getConstants(): array { $className = static::class; @@ -312,7 +312,7 @@ final public function __call(string $name, array $arguments): bool // Handle is* methods if (strpos($name, 'is') === 0) { $constantName = self::camelCaseToConstant(substr($name, 2)); - $constants = self::getConstants(); + $constants = static::getConstants(); if (isset($constants[$constantName])) { return $this->value === $constants[$constantName]; @@ -337,7 +337,7 @@ final public function __call(string $name, array $arguments): bool final public static function __callStatic(string $name, array $arguments): self { $constantName = self::camelCaseToConstant($name); - $constants = self::getConstants(); + $constants = static::getConstants(); if (isset($constants[$constantName])) { return self::getInstance($constants[$constantName], $constantName); diff --git a/src/Providers/Models/Enums/OptionEnum.php b/src/Providers/Models/Enums/OptionEnum.php index 96fd37a8..dc7ba698 100644 --- a/src/Providers/Models/Enums/OptionEnum.php +++ b/src/Providers/Models/Enums/OptionEnum.php @@ -4,83 +4,117 @@ namespace WordPress\AiClient\Providers\Models\Enums; +use ReflectionClass; use WordPress\AiClient\Common\AbstractEnum; +use WordPress\AiClient\Providers\Models\DTO\ModelConfig; /** * Enum for model options. * - * @since n.e.x.t + * This enum dynamically includes all options from ModelConfig KEY_* constants + * in addition to the explicitly defined constants below. * + * Explicitly defined option (not in ModelConfig): * @method static self inputModalities() Creates an instance for INPUT_MODALITIES option. - * @method static self outputModalities() Creates an instance for OUTPUT_MODALITIES option. - * @method static self systemInstruction() Creates an instance for SYSTEM_INSTRUCTION option. + * @method bool isInputModalities() Checks if the option is INPUT_MODALITIES. + * + * Dynamically loaded from ModelConfig KEY_* constants: * @method static self candidateCount() Creates an instance for CANDIDATE_COUNT option. + * @method static self customOptions() Creates an instance for CUSTOM_OPTIONS option. + * @method static self frequencyPenalty() Creates an instance for FREQUENCY_PENALTY option. + * @method static self functionDeclarations() Creates an instance for FUNCTION_DECLARATIONS option. + * @method static self logprobs() Creates an instance for LOGPROBS option. * @method static self maxTokens() Creates an instance for MAX_TOKENS option. + * @method static self outputFileType() Creates an instance for OUTPUT_FILE_TYPE option. + * @method static self outputMediaAspectRatio() Creates an instance for OUTPUT_MEDIA_ASPECT_RATIO option. + * @method static self outputMediaOrientation() Creates an instance for OUTPUT_MEDIA_ORIENTATION option. + * @method static self outputMimeType() Creates an instance for OUTPUT_MIME_TYPE option. + * @method static self outputModalities() Creates an instance for OUTPUT_MODALITIES option. + * @method static self outputSchema() Creates an instance for OUTPUT_SCHEMA option. + * @method static self presencePenalty() Creates an instance for PRESENCE_PENALTY option. + * @method static self stopSequences() Creates an instance for STOP_SEQUENCES option. + * @method static self systemInstruction() Creates an instance for SYSTEM_INSTRUCTION option. * @method static self temperature() Creates an instance for TEMPERATURE option. * @method static self topK() Creates an instance for TOP_K option. + * @method static self topLogprobs() Creates an instance for TOP_LOGPROBS option. * @method static self topP() Creates an instance for TOP_P option. - * @method static self outputMimeType() Creates an instance for OUTPUT_MIME_TYPE option. - * @method static self outputSchema() Creates an instance for OUTPUT_SCHEMA option. - * @method bool isInputModalities() Checks if the option is INPUT_MODALITIES. - * @method bool isOutputModalities() Checks if the option is OUTPUT_MODALITIES. - * @method bool isSystemInstruction() Checks if the option is SYSTEM_INSTRUCTION. + * @method static self webSearch() Creates an instance for WEB_SEARCH option. * @method bool isCandidateCount() Checks if the option is CANDIDATE_COUNT. + * @method bool isCustomOptions() Checks if the option is CUSTOM_OPTIONS. + * @method bool isFrequencyPenalty() Checks if the option is FREQUENCY_PENALTY. + * @method bool isFunctionDeclarations() Checks if the option is FUNCTION_DECLARATIONS. + * @method bool isLogprobs() Checks if the option is LOGPROBS. * @method bool isMaxTokens() Checks if the option is MAX_TOKENS. + * @method bool isOutputFileType() Checks if the option is OUTPUT_FILE_TYPE. + * @method bool isOutputMediaAspectRatio() Checks if the option is OUTPUT_MEDIA_ASPECT_RATIO. + * @method bool isOutputMediaOrientation() Checks if the option is OUTPUT_MEDIA_ORIENTATION. + * @method bool isOutputMimeType() Checks if the option is OUTPUT_MIME_TYPE. + * @method bool isOutputModalities() Checks if the option is OUTPUT_MODALITIES. + * @method bool isOutputSchema() Checks if the option is OUTPUT_SCHEMA. + * @method bool isPresencePenalty() Checks if the option is PRESENCE_PENALTY. + * @method bool isStopSequences() Checks if the option is STOP_SEQUENCES. + * @method bool isSystemInstruction() Checks if the option is SYSTEM_INSTRUCTION. * @method bool isTemperature() Checks if the option is TEMPERATURE. * @method bool isTopK() Checks if the option is TOP_K. + * @method bool isTopLogprobs() Checks if the option is TOP_LOGPROBS. * @method bool isTopP() Checks if the option is TOP_P. - * @method bool isOutputMimeType() Checks if the option is OUTPUT_MIME_TYPE. - * @method bool isOutputSchema() Checks if the option is OUTPUT_SCHEMA. + * @method bool isWebSearch() Checks if the option is WEB_SEARCH. + * + * @since n.e.x.t */ class OptionEnum extends AbstractEnum { /** * Input modalities option. + * + * This constant is not in ModelConfig as it's derived from message content, + * not configured directly. */ public const INPUT_MODALITIES = 'input_modalities'; /** - * Output modalities option. - */ - public const OUTPUT_MODALITIES = 'output_modalities'; - - /** - * System instruction option. + * Gets the constants for this enum. + * + * Overrides the parent method to dynamically add constants from ModelConfig + * that are prefixed with KEY_. These are transformed to remove the KEY_ prefix + * and converted to snake_case values. + * + * @since n.e.x.t + * + * @return array The enum constants. */ - public const SYSTEM_INSTRUCTION = 'system_instruction'; + protected static function getConstants(): array + { + // Check if we already have cached constants for this class + $className = static::class; + if (isset(self::$cache[$className])) { + return self::$cache[$className]; + } - /** - * Candidate count option. - */ - public const CANDIDATE_COUNT = 'candidate_count'; - - /** - * Maximum tokens option. - */ - public const MAX_TOKENS = 'max_tokens'; + // Start with the constants defined in this class + $constants = parent::getConstants(); - /** - * Temperature option. - */ - public const TEMPERATURE = 'temperature'; + // Use reflection to get all constants from ModelConfig + $modelConfigReflection = new ReflectionClass(ModelConfig::class); + $modelConfigConstants = $modelConfigReflection->getConstants(); - /** - * Top K option. - */ - public const TOP_K = 'top_k'; + // Add ModelConfig constants that start with KEY_ + foreach ($modelConfigConstants as $constantName => $constantValue) { + if (strpos($constantName, 'KEY_') === 0) { + // Remove KEY_ prefix to get the enum constant name + $enumConstantName = substr($constantName, 4); - /** - * Top P option. - */ - public const TOP_P = 'top_p'; + // The value is the snake_case version stored in ModelConfig + // ModelConfig already stores these as snake_case strings + if (is_string($constantValue)) { + $constants[$enumConstantName] = $constantValue; + } + } + } - /** - * Output MIME type option. - */ - public const OUTPUT_MIME_TYPE = 'output_mime_type'; + // Cache the combined constants + self::$cache[$className] = $constants; - /** - * Output schema option. - */ - public const OUTPUT_SCHEMA = 'output_schema'; + return $constants; + } } diff --git a/tests/unit/Providers/Models/Enums/OptionEnumTest.php b/tests/unit/Providers/Models/Enums/OptionEnumTest.php index 06d40d73..dd82b746 100644 --- a/tests/unit/Providers/Models/Enums/OptionEnumTest.php +++ b/tests/unit/Providers/Models/Enums/OptionEnumTest.php @@ -33,16 +33,30 @@ protected function getEnumClass(): string protected function getExpectedValues(): array { return [ + // Explicitly defined constant (not in ModelConfig) 'INPUT_MODALITIES' => 'input_modalities', - 'OUTPUT_MODALITIES' => 'output_modalities', - 'SYSTEM_INSTRUCTION' => 'system_instruction', - 'CANDIDATE_COUNT' => 'candidate_count', - 'MAX_TOKENS' => 'max_tokens', + + // Dynamically added from ModelConfig KEY_* constants + 'OUTPUT_MODALITIES' => 'outputModalities', + 'SYSTEM_INSTRUCTION' => 'systemInstruction', + 'CANDIDATE_COUNT' => 'candidateCount', + 'MAX_TOKENS' => 'maxTokens', 'TEMPERATURE' => 'temperature', - 'TOP_K' => 'top_k', - 'TOP_P' => 'top_p', - 'OUTPUT_MIME_TYPE' => 'output_mime_type', - 'OUTPUT_SCHEMA' => 'output_schema', + 'TOP_P' => 'topP', + 'TOP_K' => 'topK', + 'STOP_SEQUENCES' => 'stopSequences', + 'PRESENCE_PENALTY' => 'presencePenalty', + 'FREQUENCY_PENALTY' => 'frequencyPenalty', + 'LOGPROBS' => 'logprobs', + 'TOP_LOGPROBS' => 'topLogprobs', + 'FUNCTION_DECLARATIONS' => 'functionDeclarations', + 'WEB_SEARCH' => 'webSearch', + 'OUTPUT_FILE_TYPE' => 'outputFileType', + 'OUTPUT_MIME_TYPE' => 'outputMimeType', + 'OUTPUT_SCHEMA' => 'outputSchema', + 'OUTPUT_MEDIA_ORIENTATION' => 'outputMediaOrientation', + 'OUTPUT_MEDIA_ASPECT_RATIO' => 'outputMediaAspectRatio', + 'CUSTOM_OPTIONS' => 'customOptions', ]; } @@ -65,4 +79,60 @@ public function testSpecificEnumMethods(): void $this->assertTrue($outputSchema->isOutputSchema()); $this->assertFalse($outputSchema->isOutputMimeType()); } + + /** + * Tests that dynamically loaded constants from ModelConfig work. + * + * @return void + */ + public function testDynamicallyLoadedConstants(): void + { + // Test a dynamically loaded constant + $stopSequences = OptionEnum::stopSequences(); + $this->assertInstanceOf(OptionEnum::class, $stopSequences); + $this->assertEquals('stopSequences', $stopSequences->value); + $this->assertTrue($stopSequences->isStopSequences()); + $this->assertFalse($stopSequences->isTemperature()); + + // Test another dynamically loaded constant + $presencePenalty = OptionEnum::presencePenalty(); + $this->assertInstanceOf(OptionEnum::class, $presencePenalty); + $this->assertEquals('presencePenalty', $presencePenalty->value); + $this->assertTrue($presencePenalty->isPresencePenalty()); + $this->assertFalse($presencePenalty->isFrequencyPenalty()); + + // Test that all expected dynamic constants are available + $this->assertInstanceOf(OptionEnum::class, OptionEnum::frequencyPenalty()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::logprobs()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::topLogprobs()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::functionDeclarations()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::webSearch()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::outputFileType()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::outputMediaOrientation()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::outputMediaAspectRatio()); + $this->assertInstanceOf(OptionEnum::class, OptionEnum::customOptions()); + } + + /** + * Tests that getValues includes all dynamically loaded constants. + * + * @return void + */ + public function testGetValuesIncludesDynamicConstants(): void + { + $values = OptionEnum::getValues(); + + // Check that dynamic values are included + $this->assertContains('stopSequences', $values); + $this->assertContains('presencePenalty', $values); + $this->assertContains('frequencyPenalty', $values); + $this->assertContains('logprobs', $values); + $this->assertContains('topLogprobs', $values); + $this->assertContains('functionDeclarations', $values); + $this->assertContains('webSearch', $values); + $this->assertContains('outputFileType', $values); + $this->assertContains('outputMediaOrientation', $values); + $this->assertContains('outputMediaAspectRatio', $values); + $this->assertContains('customOptions', $values); + } } From 3c6d64168ac3e43c11442dba044a923e10709f1a Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 08:32:57 -0700 Subject: [PATCH 34/47] refactor: switches required and supports options to use enum as name --- src/Builders/PromptBuilder.php | 4 +- src/Providers/Models/DTO/ModelConfig.php | 42 +++--- src/Providers/Models/DTO/ModelMetadata.php | 6 +- src/Providers/Models/DTO/RequiredOption.php | 18 +-- src/Providers/Models/DTO/SupportedOption.php | 18 +-- tests/unit/Builders/PromptBuilderTest.php | 20 +-- .../DTO/ProviderModelsMetadataTest.php | 11 +- .../Models/DTO/ModelMetadataTest.php | 51 +++---- .../Models/DTO/ModelRequirementsTest.php | 55 +++---- .../Models/DTO/RequiredOptionTest.php | 135 +++++++++--------- .../Models/DTO/SupportedOptionTest.php | 81 +++++------ 11 files changed, 227 insertions(+), 214 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 66632178..0ea73136 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -412,7 +412,7 @@ public function getModelRequirements(): ModelRequirements // Add input modalities if we have non-text inputs if (count($inputModalities) > 0) { $requiredOptions[] = new RequiredOption( - OptionEnum::inputModalities()->value, + OptionEnum::inputModalities(), array_values($inputModalities) ); } @@ -1044,7 +1044,7 @@ public function getConfiguredModel(?ModelRequirements $requirements = null): Mod return $cap->value; }, $requirements->getRequiredCapabilities())) . '. Required options: ' . implode(', ', array_map(function ($opt) { - return $opt->getName() . '=' . json_encode($opt->getValue()); + return $opt->getName()->value . '=' . json_encode($opt->getValue()); }, $requirements->getRequiredOptions())) ); } diff --git a/src/Providers/Models/DTO/ModelConfig.php b/src/Providers/Models/DTO/ModelConfig.php index 59194504..9e2fd5c6 100644 --- a/src/Providers/Models/DTO/ModelConfig.php +++ b/src/Providers/Models/DTO/ModelConfig.php @@ -928,63 +928,63 @@ public function toRequiredOptions(): array // Map properties that have corresponding OptionEnum values if ($this->outputModalities !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::outputModalities()->value, + OptionEnum::outputModalities(), $this->outputModalities ); } if ($this->systemInstruction !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::systemInstruction()->value, + OptionEnum::systemInstruction(), $this->systemInstruction ); } if ($this->candidateCount !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::candidateCount()->value, + OptionEnum::candidateCount(), $this->candidateCount ); } if ($this->maxTokens !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::maxTokens()->value, + OptionEnum::maxTokens(), $this->maxTokens ); } if ($this->temperature !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::temperature()->value, + OptionEnum::temperature(), $this->temperature ); } if ($this->topP !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::topP()->value, + OptionEnum::topP(), $this->topP ); } if ($this->topK !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::topK()->value, + OptionEnum::topK(), $this->topK ); } if ($this->outputMimeType !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::outputMimeType()->value, + OptionEnum::outputMimeType(), $this->outputMimeType ); } if ($this->outputSchema !== null) { $requiredOptions[] = new RequiredOption( - OptionEnum::outputSchema()->value, + OptionEnum::outputSchema(), $this->outputSchema ); } @@ -992,48 +992,50 @@ public function toRequiredOptions(): array // Handle properties without OptionEnum values as custom options // These would need to be handled specially by providers if ($this->stopSequences !== null) { - $requiredOptions[] = new RequiredOption('stop_sequences', $this->stopSequences); + $requiredOptions[] = new RequiredOption(OptionEnum::stopSequences(), $this->stopSequences); } if ($this->presencePenalty !== null) { - $requiredOptions[] = new RequiredOption('presence_penalty', $this->presencePenalty); + $requiredOptions[] = new RequiredOption(OptionEnum::presencePenalty(), $this->presencePenalty); } if ($this->frequencyPenalty !== null) { - $requiredOptions[] = new RequiredOption('frequency_penalty', $this->frequencyPenalty); + $requiredOptions[] = new RequiredOption(OptionEnum::frequencyPenalty(), $this->frequencyPenalty); } if ($this->logprobs !== null) { - $requiredOptions[] = new RequiredOption('logprobs', $this->logprobs); + $requiredOptions[] = new RequiredOption(OptionEnum::logprobs(), $this->logprobs); } if ($this->topLogprobs !== null) { - $requiredOptions[] = new RequiredOption('top_logprobs', $this->topLogprobs); + $requiredOptions[] = new RequiredOption(OptionEnum::topLogprobs(), $this->topLogprobs); } if ($this->functionDeclarations !== null) { - $requiredOptions[] = new RequiredOption('function_declarations', true); + $requiredOptions[] = new RequiredOption(OptionEnum::functionDeclarations(), true); } if ($this->webSearch !== null) { - $requiredOptions[] = new RequiredOption('web_search', true); + $requiredOptions[] = new RequiredOption(OptionEnum::webSearch(), true); } if ($this->outputFileType !== null) { - $requiredOptions[] = new RequiredOption('output_file_type', $this->outputFileType->value); + $requiredOptions[] = new RequiredOption(OptionEnum::outputFileType(), $this->outputFileType->value); } if ($this->outputMediaOrientation !== null) { - $requiredOptions[] = new RequiredOption('output_media_orientation', $this->outputMediaOrientation->value); + $requiredOptions[] = new RequiredOption(OptionEnum::outputMediaOrientation(), $this->outputMediaOrientation->value); } if ($this->outputMediaAspectRatio !== null) { - $requiredOptions[] = new RequiredOption('output_media_aspect_ratio', $this->outputMediaAspectRatio); + $requiredOptions[] = new RequiredOption(OptionEnum::outputMediaAspectRatio(), $this->outputMediaAspectRatio); } // Add custom options as individual RequiredOptions + // Custom options don't have predefined OptionEnum values, so we use the customOptions enum + // with the actual key-value pair as the value foreach ($this->customOptions as $key => $value) { - $requiredOptions[] = new RequiredOption($key, $value); + $requiredOptions[] = new RequiredOption(OptionEnum::customOptions(), [$key => $value]); } return $requiredOptions; diff --git a/src/Providers/Models/DTO/ModelMetadata.php b/src/Providers/Models/DTO/ModelMetadata.php index a5429103..23bb335e 100644 --- a/src/Providers/Models/DTO/ModelMetadata.php +++ b/src/Providers/Models/DTO/ModelMetadata.php @@ -98,7 +98,7 @@ public function __construct(string $id, string $name, array $supportedCapabiliti // Build options map for efficient lookups foreach ($supportedOptions as $option) { - $this->optionsMap[$option->getName()] = $option; + $this->optionsMap[$option->getName()->value] = $option; } } @@ -229,11 +229,11 @@ public function meetsRequirements(ModelRequirements $requirements): bool // Check if all required options are supported with the specified values foreach ($requirements->getRequiredOptions() as $requiredOption) { // Use map lookup instead of linear search - if (!isset($this->optionsMap[$requiredOption->getName()])) { + if (!isset($this->optionsMap[$requiredOption->getName()->value])) { return false; } - $supportedOption = $this->optionsMap[$requiredOption->getName()]; + $supportedOption = $this->optionsMap[$requiredOption->getName()->value]; // Check if the required value is supported by this option if (!$supportedOption->isSupportedValue($requiredOption->getValue())) { diff --git a/src/Providers/Models/DTO/RequiredOption.php b/src/Providers/Models/DTO/RequiredOption.php index ee79c7f5..d7d26291 100644 --- a/src/Providers/Models/DTO/RequiredOption.php +++ b/src/Providers/Models/DTO/RequiredOption.php @@ -5,6 +5,7 @@ namespace WordPress\AiClient\Providers\Models\DTO; use WordPress\AiClient\Common\AbstractDataTransferObject; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; /** * Represents an option that the implementing code requires the model to support. @@ -27,9 +28,9 @@ class RequiredOption extends AbstractDataTransferObject public const KEY_VALUE = 'value'; /** - * @var string The option name. + * @var OptionEnum The option name. */ - protected string $name; + protected OptionEnum $name; /** * @var mixed The value that the model must support for this option. @@ -41,10 +42,10 @@ class RequiredOption extends AbstractDataTransferObject * * @since n.e.x.t * - * @param string $name The option name. + * @param OptionEnum $name The option name. * @param mixed $value The value that the model must support for this option. */ - public function __construct(string $name, $value) + public function __construct(OptionEnum $name, $value) { $this->name = $name; $this->value = $value; @@ -55,9 +56,9 @@ public function __construct(string $name, $value) * * @since n.e.x.t * - * @return string The option name. + * @return OptionEnum The option name. */ - public function getName(): string + public function getName(): OptionEnum { return $this->name; } @@ -86,6 +87,7 @@ public static function getJsonSchema(): array 'properties' => [ self::KEY_NAME => [ 'type' => 'string', + 'enum' => OptionEnum::getValues(), 'description' => 'The option name.', ], self::KEY_VALUE => [ @@ -114,7 +116,7 @@ public static function getJsonSchema(): array public function toArray(): array { return [ - self::KEY_NAME => $this->name, + self::KEY_NAME => $this->name->value, self::KEY_VALUE => $this->value, ]; } @@ -129,7 +131,7 @@ public static function fromArray(array $array): self static::validateFromArrayData($array, [self::KEY_NAME, self::KEY_VALUE]); return new self( - $array[self::KEY_NAME], + OptionEnum::from($array[self::KEY_NAME]), $array[self::KEY_VALUE] ); } diff --git a/src/Providers/Models/DTO/SupportedOption.php b/src/Providers/Models/DTO/SupportedOption.php index ddf76b62..1a4d121f 100644 --- a/src/Providers/Models/DTO/SupportedOption.php +++ b/src/Providers/Models/DTO/SupportedOption.php @@ -6,6 +6,7 @@ use InvalidArgumentException; use WordPress\AiClient\Common\AbstractDataTransferObject; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; /** * Represents a supported configuration option for an AI model. @@ -28,9 +29,9 @@ class SupportedOption extends AbstractDataTransferObject public const KEY_SUPPORTED_VALUES = 'supportedValues'; /** - * @var string The option name. + * @var OptionEnum The option name. */ - protected string $name; + protected OptionEnum $name; /** * @var list|null The supported values for this option. @@ -42,12 +43,12 @@ class SupportedOption extends AbstractDataTransferObject * * @since n.e.x.t * - * @param string $name The option name. + * @param OptionEnum $name The option name. * @param list|null $supportedValues The supported values for this option, or null if any value is supported. * * @throws InvalidArgumentException If supportedValues is not null and not a list. */ - public function __construct(string $name, ?array $supportedValues = null) + public function __construct(OptionEnum $name, ?array $supportedValues = null) { if ($supportedValues !== null && !array_is_list($supportedValues)) { throw new InvalidArgumentException('Supported values must be a list array.'); @@ -62,9 +63,9 @@ public function __construct(string $name, ?array $supportedValues = null) * * @since n.e.x.t * - * @return string The option name. + * @return OptionEnum The option name. */ - public function getName(): string + public function getName(): OptionEnum { return $this->name; } @@ -111,6 +112,7 @@ public static function getJsonSchema(): array 'properties' => [ self::KEY_NAME => [ 'type' => 'string', + 'enum' => OptionEnum::getValues(), 'description' => 'The option name.', ], self::KEY_SUPPORTED_VALUES => [ @@ -142,7 +144,7 @@ public static function getJsonSchema(): array public function toArray(): array { $data = [ - self::KEY_NAME => $this->name, + self::KEY_NAME => $this->name->value, ]; if ($this->supportedValues !== null) { @@ -164,7 +166,7 @@ public static function fromArray(array $array): self static::validateFromArrayData($array, [self::KEY_NAME]); return new self( - $array[self::KEY_NAME], + OptionEnum::from($array[self::KEY_NAME]), $array[self::KEY_SUPPORTED_VALUES] ?? null ); } diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index 744ff8d7..744fd231 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -903,7 +903,7 @@ public function testGetModelRequirementsBasicText(): void // Should have input modalities with text $inputModalitiesFound = false; foreach ($options as $option) { - if ($option->getName() === OptionEnum::inputModalities()->value) { + if ($option->getName()->equals(OptionEnum::inputModalities())) { $inputModalitiesFound = true; $modalities = $option->getValue(); $this->assertCount(1, $modalities); @@ -963,7 +963,7 @@ public function testGetModelRequirementsWithMultimodalInput(): void // Find input modalities option $inputModalities = null; foreach ($options as $option) { - if ($option->getName() === OptionEnum::inputModalities()->value) { + if ($option->getName()->equals(OptionEnum::inputModalities())) { $inputModalities = $option->getValue(); break; } @@ -2439,7 +2439,7 @@ public function testGetModelRequirementsWithAllFileTypes(): void // Find input modalities $inputModalities = null; foreach ($options as $option) { - if ($option->getName() === OptionEnum::inputModalities()->value) { + if ($option->getName()->equals(OptionEnum::inputModalities())) { $inputModalities = $option->getValue(); break; } @@ -2477,15 +2477,15 @@ public function testGetModelRequirementsIncludesConfigOptions(): void $options = $requirements->getRequiredOptions(); // Check that config options are included - $optionNames = array_map(function ($option) { + $optionEnums = array_map(function ($option) { return $option->getName(); }, $options); - $this->assertContains(OptionEnum::maxTokens()->value, $optionNames); - $this->assertContains(OptionEnum::temperature()->value, $optionNames); - $this->assertContains(OptionEnum::outputModalities()->value, $optionNames); - $this->assertContains(OptionEnum::outputMimeType()->value, $optionNames); - $this->assertContains(OptionEnum::outputSchema()->value, $optionNames); + $this->assertContains(OptionEnum::maxTokens(), $optionEnums); + $this->assertContains(OptionEnum::temperature(), $optionEnums); + $this->assertContains(OptionEnum::outputModalities(), $optionEnums); + $this->assertContains(OptionEnum::outputMimeType(), $optionEnums); + $this->assertContains(OptionEnum::outputSchema(), $optionEnums); } /** @@ -2718,7 +2718,7 @@ public function testIsSupportedWithIntendedOutput(): void ->willReturnCallback(function ($requirements) { $options = $requirements->getRequiredOptions(); foreach ($options as $option) { - if ($option->getName() === OptionEnum::outputModalities()->value) { + if ($option->getName()->equals(OptionEnum::outputModalities())) { $modalities = $option->getValue(); foreach ($modalities as $modality) { if ($modality->isImage()) { diff --git a/tests/unit/Providers/DTO/ProviderModelsMetadataTest.php b/tests/unit/Providers/DTO/ProviderModelsMetadataTest.php index 0141cc60..cd46472e 100644 --- a/tests/unit/Providers/DTO/ProviderModelsMetadataTest.php +++ b/tests/unit/Providers/DTO/ProviderModelsMetadataTest.php @@ -12,6 +12,7 @@ use WordPress\AiClient\Providers\Models\DTO\ModelMetadata; use WordPress\AiClient\Providers\Models\DTO\SupportedOption; use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; /** * @covers \WordPress\AiClient\Providers\DTO\ProviderModelsMetadata @@ -41,7 +42,7 @@ private function createModelMetadata(string $id, string $name): ModelMetadata $id, $name, [CapabilityEnum::textGeneration(), CapabilityEnum::chatHistory()], - [new SupportedOption('temperature', [0.0, 0.5, 1.0, 1.5, 2.0])] + [new SupportedOption(OptionEnum::temperature(), [0.0, 0.5, 1.0, 1.5, 2.0])] ); } @@ -178,7 +179,7 @@ public function testFromArray(): void ModelMetadata::KEY_SUPPORTED_CAPABILITIES => ['text_generation', 'chat_history'], ModelMetadata::KEY_SUPPORTED_OPTIONS => [ [ - SupportedOption::KEY_NAME => 'max_tokens', + SupportedOption::KEY_NAME => OptionEnum::maxTokens()->value, SupportedOption::KEY_SUPPORTED_VALUES => [100, 1000, 10000] ] ] @@ -295,15 +296,15 @@ public function testWithMultipleModelsAndCapabilities(): void CapabilityEnum::chatHistory() ], [ - new SupportedOption('resolution', ['256x256', '512x512', '1024x1024']), - new SupportedOption('style', ['realistic', 'artistic', 'cartoon']) + new SupportedOption(OptionEnum::outputSchema(), ['256x256', '512x512', '1024x1024']), + new SupportedOption(OptionEnum::outputSchema(), ['realistic', 'artistic', 'cartoon']) ] ), new ModelMetadata( 'embedding', 'Embedding Model', [CapabilityEnum::embeddingGeneration()], - [new SupportedOption('dimensions', [256, 512, 1024])] + [new SupportedOption(OptionEnum::outputSchema(), [256, 512, 1024])] ) ]; diff --git a/tests/unit/Providers/Models/DTO/ModelMetadataTest.php b/tests/unit/Providers/Models/DTO/ModelMetadataTest.php index 7954d6f6..52d7daf5 100644 --- a/tests/unit/Providers/Models/DTO/ModelMetadataTest.php +++ b/tests/unit/Providers/Models/DTO/ModelMetadataTest.php @@ -9,6 +9,7 @@ use WordPress\AiClient\Providers\Models\DTO\ModelMetadata; use WordPress\AiClient\Providers\Models\DTO\SupportedOption; use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; /** * @covers \WordPress\AiClient\Providers\Models\DTO\ModelMetadata @@ -22,7 +23,7 @@ class ModelMetadataTest extends TestCase */ private function createSampleOption(): SupportedOption { - return new SupportedOption('temperature', [0.0, 0.5, 1.0, 1.5, 2.0]); + return new SupportedOption(OptionEnum::temperature(), [0.0, 0.5, 1.0, 1.5, 2.0]); } /** @@ -40,8 +41,8 @@ public function testConstructorAndGetters(): void CapabilityEnum::textGeneration() ]; $options = [ - new SupportedOption('temperature', [0.0, 0.7, 1.0, 2.0]), - new SupportedOption('max_tokens', [100, 1000, 4000]) + new SupportedOption(OptionEnum::temperature(), [0.0, 0.7, 1.0, 2.0]), + new SupportedOption(OptionEnum::maxTokens(), [100, 1000, 4000]) ]; $metadata = new ModelMetadata($id, $name, $capabilities, $options); @@ -137,8 +138,8 @@ public function testToArray(): void 'Claude 2', [CapabilityEnum::textGeneration(), CapabilityEnum::chatHistory()], [ - new SupportedOption('max_tokens', [100, 1000, 10000]), - new SupportedOption('temperature', [0.0, 1.0]) + new SupportedOption(OptionEnum::maxTokens(), [100, 1000, 10000]), + new SupportedOption(OptionEnum::temperature(), [0.0, 1.0]) ] ); @@ -149,12 +150,12 @@ public function testToArray(): void $this->assertEquals('Claude 2', $array[ModelMetadata::KEY_NAME]); $this->assertEquals(['text_generation', 'chat_history'], $array[ModelMetadata::KEY_SUPPORTED_CAPABILITIES]); $this->assertCount(2, $array[ModelMetadata::KEY_SUPPORTED_OPTIONS]); - $this->assertEquals('max_tokens', $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::maxTokens()->value, $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME]); $this->assertEquals( [100, 1000, 10000], $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_SUPPORTED_VALUES] ); - $this->assertEquals('temperature', $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][1][SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::temperature()->value, $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][1][SupportedOption::KEY_NAME]); $this->assertEquals( [0.0, 1.0], $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][1][SupportedOption::KEY_SUPPORTED_VALUES] @@ -174,11 +175,11 @@ public function testFromArray(): void ModelMetadata::KEY_SUPPORTED_CAPABILITIES => ['text_generation', 'chat_history', 'embedding_generation'], ModelMetadata::KEY_SUPPORTED_OPTIONS => [ [ - SupportedOption::KEY_NAME => 'temperature', + SupportedOption::KEY_NAME => OptionEnum::temperature()->value, SupportedOption::KEY_SUPPORTED_VALUES => [0.1, 0.5, 0.9] ], [ - SupportedOption::KEY_NAME => 'top_p', + SupportedOption::KEY_NAME => OptionEnum::topP()->value, SupportedOption::KEY_SUPPORTED_VALUES => [0.5, 0.9, 0.95] ] ] @@ -198,9 +199,9 @@ public function testFromArray(): void $options = $metadata->getSupportedOptions(); $this->assertCount(2, $options); - $this->assertEquals('temperature', $options[0]->getName()); + $this->assertEquals(OptionEnum::temperature()->value, $options[0]->getName()); $this->assertEquals([0.1, 0.5, 0.9], $options[0]->getSupportedValues()); - $this->assertEquals('top_p', $options[1]->getName()); + $this->assertEquals(OptionEnum::topP()->value, $options[1]->getName()); $this->assertEquals([0.5, 0.9, 0.95], $options[1]->getSupportedValues()); } @@ -220,8 +221,8 @@ public function testArrayRoundTrip(): void CapabilityEnum::textToSpeechConversion() ], [ - new SupportedOption('resolution', ['256x256', '512x512', '1024x1024']), - new SupportedOption('voice', ['alloy', 'echo', 'fable', 'onyx', 'nova', 'shimmer']) + new SupportedOption(OptionEnum::outputSchema(), ['256x256', '512x512', '1024x1024']), + new SupportedOption(OptionEnum::outputSchema(), ['alloy', 'echo', 'fable', 'onyx', 'nova', 'shimmer']) ] ); @@ -258,7 +259,7 @@ public function testJsonSerialize(): void 'json-model', 'JSON Test Model', [CapabilityEnum::embeddingGeneration()], - [new SupportedOption('dimensions', [256, 512, 1024])] + [new SupportedOption(OptionEnum::outputSchema(), [256, 512, 1024])] ); $json = json_encode($metadata); @@ -270,7 +271,7 @@ public function testJsonSerialize(): void $this->assertEquals('JSON Test Model', $decoded[ModelMetadata::KEY_NAME]); $this->assertEquals(['embedding_generation'], $decoded[ModelMetadata::KEY_SUPPORTED_CAPABILITIES]); $this->assertCount(1, $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS]); - $this->assertEquals('dimensions', $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::outputSchema()->value, $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME]); $this->assertEquals( [256, 512, 1024], $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_SUPPORTED_VALUES] @@ -322,13 +323,13 @@ public function testWithAllCapabilities(): void public function testWithComplexSupportedOptions(): void { $options = [ - new SupportedOption('string_values', ['option1', 'option2', 'option3']), - new SupportedOption('numeric_values', [1, 2, 3, 4, 5]), - new SupportedOption('float_values', [0.1, 0.5, 0.9]), - new SupportedOption('boolean_values', [true, false]), - new SupportedOption('mixed_values', ['text', 123, true, null]), - new SupportedOption('nested_arrays', [['a', 'b'], ['c', 'd']]), - new SupportedOption('objects', [['key' => 'value'], ['another' => 'object']]) + new SupportedOption(OptionEnum::outputSchema(), ['option1', 'option2', 'option3']), + new SupportedOption(OptionEnum::outputSchema(), [1, 2, 3, 4, 5]), + new SupportedOption(OptionEnum::temperature(), [0.1, 0.5, 0.9]), + new SupportedOption(OptionEnum::outputSchema(), [true, false]), + new SupportedOption(OptionEnum::outputSchema(), ['text', 123, true, null]), + new SupportedOption(OptionEnum::outputSchema(), [['a', 'b'], ['c', 'd']]), + new SupportedOption(OptionEnum::customOptions(), [['key' => 'value'], ['another' => 'object']]) ]; $metadata = new ModelMetadata('complex-model', 'Complex Model', [], $options); @@ -378,7 +379,7 @@ public function testSpecialCharactersInNames(): void 'special-model-123', 'Model with "quotes" & special ', [CapabilityEnum::textGeneration()], - [new SupportedOption('option_with_underscore', ['value'])] + [new SupportedOption(OptionEnum::outputSchema(), ['value'])] ); $array = $metadata->toArray(); @@ -404,8 +405,8 @@ public function testArrayValuesProperlyIndexed(): void CapabilityEnum::embeddingGeneration() ], [ - new SupportedOption('opt1', [1, 2, 3]), - new SupportedOption('opt2', ['a', 'b', 'c']) + new SupportedOption(OptionEnum::maxTokens(), [1, 2, 3]), + new SupportedOption(OptionEnum::outputSchema(), ['a', 'b', 'c']) ] ); diff --git a/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php b/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php index 706859a5..ca773138 100644 --- a/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php +++ b/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php @@ -9,6 +9,7 @@ use WordPress\AiClient\Providers\Models\DTO\ModelRequirements; use WordPress\AiClient\Providers\Models\DTO\RequiredOption; use WordPress\AiClient\Providers\Models\Enums\CapabilityEnum; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; /** * @covers \WordPress\AiClient\Providers\Models\DTO\ModelRequirements @@ -27,8 +28,8 @@ public function testConstructorAndGetters(): void CapabilityEnum::chatHistory() ]; $options = [ - new RequiredOption('temperature', 0.7), - new RequiredOption('max_tokens', 1000) + new RequiredOption(OptionEnum::temperature(), 0.7), + new RequiredOption(OptionEnum::maxTokens(), 1000) ]; $requirements = new ModelRequirements($capabilities, $options); @@ -109,8 +110,8 @@ public function testToArray(): void $requirements = new ModelRequirements( [CapabilityEnum::imageGeneration(), CapabilityEnum::textGeneration()], [ - new RequiredOption('resolution', '1024x1024'), - new RequiredOption('style', 'realistic') + new RequiredOption(OptionEnum::outputSchema(), '1024x1024'), + new RequiredOption(OptionEnum::outputSchema(), 'realistic') ] ); @@ -122,9 +123,9 @@ public function testToArray(): void $array[ModelRequirements::KEY_REQUIRED_CAPABILITIES] ); $this->assertCount(2, $array[ModelRequirements::KEY_REQUIRED_OPTIONS]); - $this->assertEquals('resolution', $array[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_NAME]); + $this->assertEquals(OptionEnum::outputSchema()->value, $array[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_NAME]); $this->assertEquals('1024x1024', $array[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_VALUE]); - $this->assertEquals('style', $array[ModelRequirements::KEY_REQUIRED_OPTIONS][1][RequiredOption::KEY_NAME]); + $this->assertEquals(OptionEnum::outputSchema()->value, $array[ModelRequirements::KEY_REQUIRED_OPTIONS][1][RequiredOption::KEY_NAME]); $this->assertEquals('realistic', $array[ModelRequirements::KEY_REQUIRED_OPTIONS][1][RequiredOption::KEY_VALUE]); } @@ -139,11 +140,11 @@ public function testFromArray(): void ModelRequirements::KEY_REQUIRED_CAPABILITIES => ['text_generation', 'chat_history', 'embedding_generation'], ModelRequirements::KEY_REQUIRED_OPTIONS => [ [ - RequiredOption::KEY_NAME => 'response_format', + RequiredOption::KEY_NAME => OptionEnum::outputSchema()->value, RequiredOption::KEY_VALUE => ['type' => 'json_object'] ], [ - RequiredOption::KEY_NAME => 'temperature', + RequiredOption::KEY_NAME => OptionEnum::temperature()->value, RequiredOption::KEY_VALUE => 0.5 ] ] @@ -161,9 +162,9 @@ public function testFromArray(): void $options = $requirements->getRequiredOptions(); $this->assertCount(2, $options); - $this->assertEquals('response_format', $options[0]->getName()); + $this->assertEquals(OptionEnum::outputSchema()->value, $options[0]->getName()); $this->assertEquals(['type' => 'json_object'], $options[0]->getValue()); - $this->assertEquals('temperature', $options[1]->getName()); + $this->assertEquals(OptionEnum::temperature()->value, $options[1]->getName()); $this->assertEquals(0.5, $options[1]->getValue()); } @@ -181,9 +182,9 @@ public function testArrayRoundTrip(): void CapabilityEnum::musicGeneration() ], [ - new RequiredOption('voice', 'alloy'), - new RequiredOption('language', 'en-US'), - new RequiredOption('sample_rate', 44100) + new RequiredOption(OptionEnum::outputSchema(), 'alloy'), + new RequiredOption(OptionEnum::outputSchema(), 'en-US'), + new RequiredOption(OptionEnum::outputSchema(), 44100) ] ); @@ -215,7 +216,7 @@ public function testJsonSerialize(): void { $requirements = new ModelRequirements( [CapabilityEnum::embeddingGeneration()], - [new RequiredOption('dimensions', 1536)] + [new RequiredOption(OptionEnum::outputSchema(), 1536)] ); $json = json_encode($requirements); @@ -226,7 +227,7 @@ public function testJsonSerialize(): void $this->assertEquals(['embedding_generation'], $decoded[ModelRequirements::KEY_REQUIRED_CAPABILITIES]); $this->assertCount(1, $decoded[ModelRequirements::KEY_REQUIRED_OPTIONS]); $this->assertEquals( - 'dimensions', + OptionEnum::outputSchema()->value, $decoded[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_NAME] ); $this->assertEquals(1536, $decoded[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_VALUE]); @@ -271,13 +272,13 @@ public function testWithAllCapabilityTypes(): void public function testWithVariousOptionValueTypes(): void { $options = [ - new RequiredOption('string_option', 'text value'), - new RequiredOption('int_option', 42), - new RequiredOption('float_option', 3.14), - new RequiredOption('bool_option', true), - new RequiredOption('null_option', null), - new RequiredOption('array_option', ['a', 'b', 'c']), - new RequiredOption('object_option', ['key' => 'value', 'nested' => ['inner' => true]]) + new RequiredOption(OptionEnum::outputSchema(), 'text value'), + new RequiredOption(OptionEnum::outputSchema(), 42), + new RequiredOption(OptionEnum::temperature(), 3.14), + new RequiredOption(OptionEnum::outputSchema(), true), + new RequiredOption(OptionEnum::outputSchema(), null), + new RequiredOption(OptionEnum::outputSchema(), ['a', 'b', 'c']), + new RequiredOption(OptionEnum::customOptions(), ['key' => 'value', 'nested' => ['inner' => true]]) ]; $requirements = new ModelRequirements([], $options); @@ -323,15 +324,15 @@ public function testOnlyOptionsNoCapabilities(): void $requirements = new ModelRequirements( [], [ - new RequiredOption('api_key', 'secret-key'), - new RequiredOption('base_url', 'https://api.example.com') + new RequiredOption(OptionEnum::outputSchema(), 'secret-key'), + new RequiredOption(OptionEnum::outputSchema(), 'https://api.example.com') ] ); $array = $requirements->toArray(); $this->assertEquals([], $array['requiredCapabilities']); $this->assertCount(2, $array['requiredOptions']); - $this->assertEquals('api_key', $array['requiredOptions'][0]['name']); + $this->assertEquals(OptionEnum::outputSchema()->value, $array['requiredOptions'][0]['name']); $this->assertEquals('secret-key', $array['requiredOptions'][0]['value']); } @@ -349,8 +350,8 @@ public function testArrayValuesProperlyIndexed(): void CapabilityEnum::embeddingGeneration() ], [ - new RequiredOption('opt1', 'val1'), - new RequiredOption('opt2', 'val2') + new RequiredOption(OptionEnum::outputSchema(), 'val1'), + new RequiredOption(OptionEnum::outputSchema(), 'val2') ] ); diff --git a/tests/unit/Providers/Models/DTO/RequiredOptionTest.php b/tests/unit/Providers/Models/DTO/RequiredOptionTest.php index b636fd7e..6324bf3c 100644 --- a/tests/unit/Providers/Models/DTO/RequiredOptionTest.php +++ b/tests/unit/Providers/Models/DTO/RequiredOptionTest.php @@ -7,6 +7,7 @@ use JsonSerializable; use PHPUnit\Framework\TestCase; use WordPress\AiClient\Providers\Models\DTO\RequiredOption; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; /** * @covers \WordPress\AiClient\Providers\Models\DTO\RequiredOption @@ -20,12 +21,12 @@ class RequiredOptionTest extends TestCase */ public function testConstructorAndGettersWithStringValue(): void { - $name = 'api_key'; + $name = OptionEnum::maxTokens(); $value = 'secret-key-123'; $option = new RequiredOption($name, $value); - $this->assertEquals($name, $option->getName()); + $this->assertSame($name, $option->getName()); $this->assertEquals($value, $option->getValue()); } @@ -36,9 +37,9 @@ public function testConstructorAndGettersWithStringValue(): void */ public function testWithIntegerValue(): void { - $option = new RequiredOption('max_tokens', 1000); + $option = new RequiredOption(OptionEnum::maxTokens(), 1000); - $this->assertEquals('max_tokens', $option->getName()); + $this->assertSame(OptionEnum::maxTokens(), $option->getName()); $this->assertEquals(1000, $option->getValue()); $this->assertIsInt($option->getValue()); } @@ -50,9 +51,9 @@ public function testWithIntegerValue(): void */ public function testWithFloatValue(): void { - $option = new RequiredOption('temperature', 0.7); + $option = new RequiredOption(OptionEnum::temperature(), 0.7); - $this->assertEquals('temperature', $option->getName()); + $this->assertSame(OptionEnum::temperature(), $option->getName()); $this->assertEquals(0.7, $option->getValue()); $this->assertIsFloat($option->getValue()); } @@ -64,14 +65,14 @@ public function testWithFloatValue(): void */ public function testWithBooleanValue(): void { - $optionTrue = new RequiredOption('stream', true); - $optionFalse = new RequiredOption('logprobs', false); + $optionTrue = new RequiredOption(OptionEnum::webSearch(), true); + $optionFalse = new RequiredOption(OptionEnum::logprobs(), false); - $this->assertEquals('stream', $optionTrue->getName()); + $this->assertSame(OptionEnum::webSearch(), $optionTrue->getName()); $this->assertTrue($optionTrue->getValue()); $this->assertIsBool($optionTrue->getValue()); - $this->assertEquals('logprobs', $optionFalse->getName()); + $this->assertSame(OptionEnum::logprobs(), $optionFalse->getName()); $this->assertFalse($optionFalse->getValue()); $this->assertIsBool($optionFalse->getValue()); } @@ -83,9 +84,9 @@ public function testWithBooleanValue(): void */ public function testWithNullValue(): void { - $option = new RequiredOption('optional_field', null); + $option = new RequiredOption(OptionEnum::outputSchema(), null); - $this->assertEquals('optional_field', $option->getName()); + $this->assertSame(OptionEnum::outputSchema(), $option->getName()); $this->assertNull($option->getValue()); } @@ -97,9 +98,9 @@ public function testWithNullValue(): void public function testWithArrayValue(): void { $arrayValue = ['option1', 'option2', 'option3']; - $option = new RequiredOption('allowed_values', $arrayValue); + $option = new RequiredOption(OptionEnum::stopSequences(), $arrayValue); - $this->assertEquals('allowed_values', $option->getName()); + $this->assertSame(OptionEnum::stopSequences(), $option->getName()); $this->assertEquals($arrayValue, $option->getValue()); $this->assertIsArray($option->getValue()); } @@ -121,9 +122,9 @@ public function testWithObjectValue(): void ] ] ]; - $option = new RequiredOption('response_format', $objectValue); + $option = new RequiredOption(OptionEnum::outputSchema(), $objectValue); - $this->assertEquals('response_format', $option->getName()); + $this->assertSame(OptionEnum::outputSchema(), $option->getName()); $this->assertEquals($objectValue, $option->getValue()); $this->assertIsArray($option->getValue()); } @@ -147,6 +148,8 @@ public function testGetJsonSchema(): void // Check name property $this->assertEquals('string', $schema['properties'][RequiredOption::KEY_NAME]['type']); + $this->assertArrayHasKey('enum', $schema['properties'][RequiredOption::KEY_NAME]); + $this->assertIsArray($schema['properties'][RequiredOption::KEY_NAME]['enum']); $this->assertEquals('The option name.', $schema['properties'][RequiredOption::KEY_NAME]['description']); // Check value property with oneOf @@ -178,50 +181,50 @@ public function testGetJsonSchema(): void public function testToArrayWithDifferentValueTypes(): void { // String value - $stringOption = new RequiredOption('string_opt', 'value'); + $stringOption = new RequiredOption(OptionEnum::maxTokens(), 'value'); $this->assertEquals( - [RequiredOption::KEY_NAME => 'string_opt', RequiredOption::KEY_VALUE => 'value'], + [RequiredOption::KEY_NAME => 'maxTokens', RequiredOption::KEY_VALUE => 'value'], $stringOption->toArray() ); // Number values - $intOption = new RequiredOption('int_opt', 42); + $intOption = new RequiredOption(OptionEnum::candidateCount(), 42); $this->assertEquals( - [RequiredOption::KEY_NAME => 'int_opt', RequiredOption::KEY_VALUE => 42], + [RequiredOption::KEY_NAME => 'candidateCount', RequiredOption::KEY_VALUE => 42], $intOption->toArray() ); - $floatOption = new RequiredOption('float_opt', 3.14); + $floatOption = new RequiredOption(OptionEnum::temperature(), 3.14); $this->assertEquals( - [RequiredOption::KEY_NAME => 'float_opt', RequiredOption::KEY_VALUE => 3.14], + [RequiredOption::KEY_NAME => 'temperature', RequiredOption::KEY_VALUE => 3.14], $floatOption->toArray() ); // Boolean value - $boolOption = new RequiredOption('bool_opt', true); + $boolOption = new RequiredOption(OptionEnum::webSearch(), true); $this->assertEquals( - [RequiredOption::KEY_NAME => 'bool_opt', RequiredOption::KEY_VALUE => true], + [RequiredOption::KEY_NAME => 'webSearch', RequiredOption::KEY_VALUE => true], $boolOption->toArray() ); // Null value - $nullOption = new RequiredOption('null_opt', null); + $nullOption = new RequiredOption(OptionEnum::outputSchema(), null); $this->assertEquals( - [RequiredOption::KEY_NAME => 'null_opt', RequiredOption::KEY_VALUE => null], + [RequiredOption::KEY_NAME => 'outputSchema', RequiredOption::KEY_VALUE => null], $nullOption->toArray() ); // Array value - $arrayOption = new RequiredOption('array_opt', [1, 2, 3]); + $arrayOption = new RequiredOption(OptionEnum::stopSequences(), [1, 2, 3]); $this->assertEquals( - [RequiredOption::KEY_NAME => 'array_opt', RequiredOption::KEY_VALUE => [1, 2, 3]], + [RequiredOption::KEY_NAME => 'stopSequences', RequiredOption::KEY_VALUE => [1, 2, 3]], $arrayOption->toArray() ); // Object value - $objectOption = new RequiredOption('object_opt', ['key' => 'value']); + $objectOption = new RequiredOption(OptionEnum::outputSchema(), ['key' => 'value']); $this->assertEquals( - [RequiredOption::KEY_NAME => 'object_opt', RequiredOption::KEY_VALUE => ['key' => 'value']], + [RequiredOption::KEY_NAME => 'outputSchema', RequiredOption::KEY_VALUE => ['key' => 'value']], $objectOption->toArray() ); } @@ -235,51 +238,51 @@ public function testFromArrayWithDifferentValueTypes(): void { // String value $stringOption = RequiredOption::fromArray( - [RequiredOption::KEY_NAME => 'str', RequiredOption::KEY_VALUE => 'test'] + [RequiredOption::KEY_NAME => 'maxTokens', RequiredOption::KEY_VALUE => 'test'] ); - $this->assertEquals('str', $stringOption->getName()); + $this->assertEquals(OptionEnum::maxTokens(), $stringOption->getName()); $this->assertEquals('test', $stringOption->getValue()); // Integer value $intOption = RequiredOption::fromArray( - [RequiredOption::KEY_NAME => 'num', RequiredOption::KEY_VALUE => 100] + [RequiredOption::KEY_NAME => 'candidateCount', RequiredOption::KEY_VALUE => 100] ); - $this->assertEquals('num', $intOption->getName()); + $this->assertEquals(OptionEnum::candidateCount(), $intOption->getName()); $this->assertEquals(100, $intOption->getValue()); // Float value $floatOption = RequiredOption::fromArray( - [RequiredOption::KEY_NAME => 'float', RequiredOption::KEY_VALUE => 1.5] + [RequiredOption::KEY_NAME => 'temperature', RequiredOption::KEY_VALUE => 1.5] ); - $this->assertEquals('float', $floatOption->getName()); + $this->assertEquals(OptionEnum::temperature(), $floatOption->getName()); $this->assertEquals(1.5, $floatOption->getValue()); // Boolean value $boolOption = RequiredOption::fromArray( - [RequiredOption::KEY_NAME => 'bool', RequiredOption::KEY_VALUE => false] + [RequiredOption::KEY_NAME => 'logprobs', RequiredOption::KEY_VALUE => false] ); - $this->assertEquals('bool', $boolOption->getName()); + $this->assertEquals(OptionEnum::logprobs(), $boolOption->getName()); $this->assertFalse($boolOption->getValue()); // Null value $nullOption = RequiredOption::fromArray( - [RequiredOption::KEY_NAME => 'nullable', RequiredOption::KEY_VALUE => null] + [RequiredOption::KEY_NAME => 'outputSchema', RequiredOption::KEY_VALUE => null] ); - $this->assertEquals('nullable', $nullOption->getName()); + $this->assertEquals(OptionEnum::outputSchema(), $nullOption->getName()); $this->assertNull($nullOption->getValue()); // Array value $arrayOption = RequiredOption::fromArray( - [RequiredOption::KEY_NAME => 'arr', RequiredOption::KEY_VALUE => ['a', 'b', 'c']] + [RequiredOption::KEY_NAME => 'stopSequences', RequiredOption::KEY_VALUE => ['a', 'b', 'c']] ); - $this->assertEquals('arr', $arrayOption->getName()); + $this->assertEquals(OptionEnum::stopSequences(), $arrayOption->getName()); $this->assertEquals(['a', 'b', 'c'], $arrayOption->getValue()); // Object value $objectOption = RequiredOption::fromArray( - [RequiredOption::KEY_NAME => 'obj', RequiredOption::KEY_VALUE => ['nested' => ['deep' => true]]] + [RequiredOption::KEY_NAME => 'outputSchema', RequiredOption::KEY_VALUE => ['nested' => ['deep' => true]]] ); - $this->assertEquals('obj', $objectOption->getName()); + $this->assertEquals(OptionEnum::outputSchema(), $objectOption->getName()); $this->assertEquals(['nested' => ['deep' => true]], $objectOption->getValue()); } @@ -291,13 +294,13 @@ public function testFromArrayWithDifferentValueTypes(): void public function testArrayRoundTrip(): void { $testCases = [ - new RequiredOption('string', 'hello world'), - new RequiredOption('integer', 42), - new RequiredOption('float', 99.99), - new RequiredOption('boolean', true), - new RequiredOption('null', null), - new RequiredOption('array', ['one', 'two', 'three']), - new RequiredOption('object', ['type' => 'config', 'enabled' => true, 'settings' => ['a' => 1, 'b' => 2]]) + new RequiredOption(OptionEnum::maxTokens(), 'hello world'), + new RequiredOption(OptionEnum::candidateCount(), 42), + new RequiredOption(OptionEnum::temperature(), 99.99), + new RequiredOption(OptionEnum::webSearch(), true), + new RequiredOption(OptionEnum::outputSchema(), null), + new RequiredOption(OptionEnum::stopSequences(), ['one', 'two', 'three']), + new RequiredOption(OptionEnum::customOptions(), ['type' => 'config', 'enabled' => true, 'settings' => ['a' => 1, 'b' => 2]]) ]; foreach ($testCases as $original) { @@ -316,41 +319,41 @@ public function testArrayRoundTrip(): void */ public function testJsonSerialize(): void { - $option = new RequiredOption('json_test', ['enabled' => true, 'count' => 5]); + $option = new RequiredOption(OptionEnum::outputSchema(), ['enabled' => true, 'count' => 5]); $json = json_encode($option); $decoded = json_decode($json, true); $this->assertIsString($json); $this->assertIsArray($decoded); - $this->assertEquals('json_test', $decoded[RequiredOption::KEY_NAME]); + $this->assertEquals('outputSchema', $decoded[RequiredOption::KEY_NAME]); $this->assertEquals(['enabled' => true, 'count' => 5], $decoded[RequiredOption::KEY_VALUE]); } /** - * Tests with empty string name. + * Tests with custom options enum. * * @return void */ - public function testWithEmptyStringName(): void + public function testWithCustomOptions(): void { - $option = new RequiredOption('', 'value'); + $option = new RequiredOption(OptionEnum::customOptions(), ['key' => 'value']); - $this->assertEquals('', $option->getName()); - $this->assertEquals('value', $option->getValue()); + $this->assertEquals(OptionEnum::customOptions(), $option->getName()); + $this->assertEquals(['key' => 'value'], $option->getValue()); } /** - * Tests with special characters in name. + * Tests with input modalities enum. * * @return void */ - public function testWithSpecialCharactersInName(): void + public function testWithInputModalitiesEnum(): void { - $option = new RequiredOption('option-with_special.chars', 'value'); + $option = new RequiredOption(OptionEnum::inputModalities(), ['text', 'image']); - $this->assertEquals('option-with_special.chars', $option->getName()); - $this->assertEquals('value', $option->getValue()); + $this->assertEquals(OptionEnum::inputModalities(), $option->getName()); + $this->assertEquals(['text', 'image'], $option->getValue()); } /** @@ -373,7 +376,7 @@ public function testWithDeeplyNestedArrayValue(): void ] ]; - $option = new RequiredOption('nested_config', $deeplyNested); + $option = new RequiredOption(OptionEnum::outputSchema(), $deeplyNested); $array = $option->toArray(); $this->assertEquals($deeplyNested, $array['value']); @@ -401,7 +404,7 @@ public function testWithMixedArrayValue(): void ['another', 'array'] ]; - $option = new RequiredOption('mixed_types', $mixedArray); + $option = new RequiredOption(OptionEnum::customOptions(), $mixedArray); $this->assertEquals($mixedArray, $option->getValue()); // Verify exact types are preserved @@ -423,7 +426,7 @@ public function testWithMixedArrayValue(): void */ public function testImplementsCorrectInterfaces(): void { - $option = new RequiredOption('test', 'value'); + $option = new RequiredOption(OptionEnum::maxTokens(), 'value'); $this->assertInstanceOf( \WordPress\AiClient\Common\Contracts\WithArrayTransformationInterface::class, diff --git a/tests/unit/Providers/Models/DTO/SupportedOptionTest.php b/tests/unit/Providers/Models/DTO/SupportedOptionTest.php index 49e313a2..79c44ca1 100644 --- a/tests/unit/Providers/Models/DTO/SupportedOptionTest.php +++ b/tests/unit/Providers/Models/DTO/SupportedOptionTest.php @@ -7,6 +7,7 @@ use JsonSerializable; use PHPUnit\Framework\TestCase; use WordPress\AiClient\Providers\Models\DTO\SupportedOption; +use WordPress\AiClient\Providers\Models\Enums\OptionEnum; /** * @covers \WordPress\AiClient\Providers\Models\DTO\SupportedOption @@ -20,12 +21,12 @@ class SupportedOptionTest extends TestCase */ public function testConstructorAndGettersWithStringValues(): void { - $name = 'model'; - $values = ['gpt-3.5-turbo', 'gpt-4', 'gpt-4-turbo']; + $name = OptionEnum::outputModalities(); + $values = ['text', 'image', 'audio']; $option = new SupportedOption($name, $values); - $this->assertEquals($name, $option->getName()); + $this->assertSame($name, $option->getName()); $this->assertEquals($values, $option->getSupportedValues()); } @@ -36,7 +37,7 @@ public function testConstructorAndGettersWithStringValues(): void */ public function testIsSupportedValue(): void { - $option = new SupportedOption('temperature', [0.0, 0.5, 1.0, 1.5, 2.0]); + $option = new SupportedOption(OptionEnum::temperature(), [0.0, 0.5, 1.0, 1.5, 2.0]); $this->assertTrue($option->isSupportedValue(0.0)); $this->assertTrue($option->isSupportedValue(0.5)); @@ -57,9 +58,9 @@ public function testIsSupportedValue(): void */ public function testWithIntegerValues(): void { - $option = new SupportedOption('max_tokens', [100, 500, 1000, 2000, 4000]); + $option = new SupportedOption(OptionEnum::maxTokens(), [100, 500, 1000, 2000, 4000]); - $this->assertEquals('max_tokens', $option->getName()); + $this->assertSame(OptionEnum::maxTokens(), $option->getName()); $this->assertEquals([100, 500, 1000, 2000, 4000], $option->getSupportedValues()); $this->assertTrue($option->isSupportedValue(100)); @@ -75,9 +76,9 @@ public function testWithIntegerValues(): void */ public function testWithBooleanValues(): void { - $option = new SupportedOption('stream', [true, false]); + $option = new SupportedOption(OptionEnum::webSearch(), [true, false]); - $this->assertEquals('stream', $option->getName()); + $this->assertSame(OptionEnum::webSearch(), $option->getName()); $this->assertEquals([true, false], $option->getSupportedValues()); $this->assertTrue($option->isSupportedValue(true)); @@ -93,7 +94,7 @@ public function testWithBooleanValues(): void */ public function testWithNullValue(): void { - $option = new SupportedOption('optional_param', ['value1', 'value2', null]); + $option = new SupportedOption(OptionEnum::outputSchema(), ['value1', 'value2', null]); $this->assertTrue($option->isSupportedValue('value1')); $this->assertTrue($option->isSupportedValue('value2')); @@ -108,9 +109,9 @@ public function testWithNullValue(): void */ public function testWithArrayValues(): void { - $option = new SupportedOption('dimensions', [[256, 256], [512, 512], [1024, 1024]]); + $option = new SupportedOption(OptionEnum::outputMediaAspectRatio(), [[256, 256], [512, 512], [1024, 1024]]); - $this->assertEquals('dimensions', $option->getName()); + $this->assertSame(OptionEnum::outputMediaAspectRatio(), $option->getName()); $supportedValues = $option->getSupportedValues(); $this->assertCount(3, $supportedValues); @@ -129,7 +130,7 @@ public function testWithObjectValues(): void { $format1 = ['type' => 'json_object']; $format2 = ['type' => 'text']; - $option = new SupportedOption('response_format', [$format1, $format2]); + $option = new SupportedOption(OptionEnum::outputSchema(), [$format1, $format2]); $this->assertTrue($option->isSupportedValue(['type' => 'json_object'])); $this->assertTrue($option->isSupportedValue(['type' => 'text'])); @@ -186,11 +187,11 @@ public function testGetJsonSchema(): void */ public function testToArray(): void { - $option = new SupportedOption('style', ['realistic', 'artistic', 'cartoon', 'abstract']); + $option = new SupportedOption(OptionEnum::outputFileType(), ['realistic', 'artistic', 'cartoon', 'abstract']); $array = $option->toArray(); $this->assertIsArray($array); - $this->assertEquals('style', $array[SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::outputFileType()->value, $array[SupportedOption::KEY_NAME]); $this->assertEquals( ['realistic', 'artistic', 'cartoon', 'abstract'], $array[SupportedOption::KEY_SUPPORTED_VALUES] @@ -206,14 +207,14 @@ public function testToArray(): void public function testFromArray(): void { $data = [ - SupportedOption::KEY_NAME => 'voice', + SupportedOption::KEY_NAME => OptionEnum::outputFileType()->value, SupportedOption::KEY_SUPPORTED_VALUES => ['alloy', 'echo', 'fable', 'onyx', 'nova', 'shimmer'] ]; $option = SupportedOption::fromArray($data); $this->assertInstanceOf(SupportedOption::class, $option); - $this->assertEquals('voice', $option->getName()); + $this->assertEquals(OptionEnum::outputFileType()->value, $option->getName()->value); $this->assertEquals(['alloy', 'echo', 'fable', 'onyx', 'nova', 'shimmer'], $option->getSupportedValues()); } @@ -225,7 +226,7 @@ public function testFromArray(): void public function testArrayRoundTrip(): void { $original = new SupportedOption( - 'complex_option', + OptionEnum::customOptions(), [ 'string', 123, @@ -241,7 +242,7 @@ public function testArrayRoundTrip(): void $array = $original->toArray(); $restored = SupportedOption::fromArray($array); - $this->assertEquals($original->getName(), $restored->getName()); + $this->assertSame($original->getName(), $restored->getName()); $this->assertEquals($original->getSupportedValues(), $restored->getSupportedValues()); // Verify each value type is preserved @@ -263,14 +264,14 @@ public function testArrayRoundTrip(): void */ public function testJsonSerialize(): void { - $option = new SupportedOption('quality', ['low', 'medium', 'high', 'ultra']); + $option = new SupportedOption(OptionEnum::candidateCount(), ['low', 'medium', 'high', 'ultra']); $json = json_encode($option); $decoded = json_decode($json, true); $this->assertIsString($json); $this->assertIsArray($decoded); - $this->assertEquals('quality', $decoded[SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::candidateCount()->value, $decoded[SupportedOption::KEY_NAME]); $this->assertEquals(['low', 'medium', 'high', 'ultra'], $decoded[SupportedOption::KEY_SUPPORTED_VALUES]); } @@ -281,9 +282,9 @@ public function testJsonSerialize(): void */ public function testWithEmptySupportedValues(): void { - $option = new SupportedOption('empty_option', []); + $option = new SupportedOption(OptionEnum::stopSequences(), []); - $this->assertEquals('empty_option', $option->getName()); + $this->assertSame(OptionEnum::stopSequences(), $option->getName()); $this->assertEquals([], $option->getSupportedValues()); $this->assertFalse($option->isSupportedValue('anything')); $this->assertFalse($option->isSupportedValue(null)); @@ -296,7 +297,7 @@ public function testWithEmptySupportedValues(): void */ public function testWithDuplicateValues(): void { - $option = new SupportedOption('duplicates', ['a', 'b', 'a', 'c', 'b']); + $option = new SupportedOption(OptionEnum::stopSequences(), ['a', 'b', 'a', 'c', 'b']); $this->assertEquals(['a', 'b', 'a', 'c', 'b'], $option->getSupportedValues()); $this->assertTrue($option->isSupportedValue('a')); @@ -311,11 +312,11 @@ public function testWithDuplicateValues(): void */ public function testWithSpecialCharactersInName(): void { - $option = new SupportedOption('option-with_special.chars:test', ['value1', 'value2']); + $option = new SupportedOption(OptionEnum::customOptions(), ['value1', 'value2']); - $this->assertEquals('option-with_special.chars:test', $option->getName()); + $this->assertSame(OptionEnum::customOptions(), $option->getName()); $array = $option->toArray(); - $this->assertEquals('option-with_special.chars:test', $array[SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::customOptions()->value, $array[SupportedOption::KEY_NAME]); } /** @@ -325,7 +326,7 @@ public function testWithSpecialCharactersInName(): void */ public function testStrictTypeCheckingInIsSupportedValue(): void { - $option = new SupportedOption('mixed', [0, '0', false, '', null]); + $option = new SupportedOption(OptionEnum::customOptions(), [0, '0', false, '', null]); // Each value should only match itself exactly $this->assertTrue($option->isSupportedValue(0)); @@ -347,7 +348,7 @@ public function testStrictTypeCheckingInIsSupportedValue(): void */ public function testArrayValuesProperlyIndexed(): void { - $option = new SupportedOption('indexed', ['first', 'second', 'third']); + $option = new SupportedOption(OptionEnum::stopSequences(), ['first', 'second', 'third']); $array = $option->toArray(); // Ensure supportedValues array has numeric keys starting from 0 @@ -378,7 +379,7 @@ public function testWithDeeplyNestedStructures(): void ] ]; - $option = new SupportedOption('nested_configs', $deeplyNested); + $option = new SupportedOption(OptionEnum::outputSchema(), $deeplyNested); $this->assertTrue($option->isSupportedValue($deeplyNested[0])); $this->assertTrue($option->isSupportedValue($deeplyNested[1])); @@ -396,7 +397,7 @@ public function testWithDeeplyNestedStructures(): void */ public function testImplementsCorrectInterfaces(): void { - $option = new SupportedOption('test', ['value']); + $option = new SupportedOption(OptionEnum::maxTokens(), ['value']); $this->assertInstanceOf( \WordPress\AiClient\Common\Contracts\WithArrayTransformationInterface::class, @@ -419,9 +420,9 @@ public function testImplementsCorrectInterfaces(): void */ public function testWithNullSupportedValues(): void { - $option = new SupportedOption('any_value_option'); + $option = new SupportedOption(OptionEnum::temperature()); - $this->assertEquals('any_value_option', $option->getName()); + $this->assertSame(OptionEnum::temperature(), $option->getName()); $this->assertNull($option->getSupportedValues()); // Any value should be supported when supportedValues is null @@ -443,11 +444,11 @@ public function testWithNullSupportedValues(): void */ public function testToArrayWithNullSupportedValues(): void { - $option = new SupportedOption('flexible_option'); + $option = new SupportedOption(OptionEnum::topP()); $array = $option->toArray(); $this->assertIsArray($array); - $this->assertEquals('flexible_option', $array[SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::topP()->value, $array[SupportedOption::KEY_NAME]); $this->assertArrayNotHasKey(SupportedOption::KEY_SUPPORTED_VALUES, $array); $this->assertCount(1, $array); } @@ -460,13 +461,13 @@ public function testToArrayWithNullSupportedValues(): void public function testFromArrayWithMissingSupportedValues(): void { $data = [ - SupportedOption::KEY_NAME => 'open_option' + SupportedOption::KEY_NAME => OptionEnum::temperature()->value ]; $option = SupportedOption::fromArray($data); $this->assertInstanceOf(SupportedOption::class, $option); - $this->assertEquals('open_option', $option->getName()); + $this->assertEquals(OptionEnum::temperature()->value, $option->getName()->value); $this->assertNull($option->getSupportedValues()); $this->assertTrue($option->isSupportedValue('anything')); } @@ -478,12 +479,12 @@ public function testFromArrayWithMissingSupportedValues(): void */ public function testRoundTripWithNullSupportedValues(): void { - $original = new SupportedOption('unrestricted'); + $original = new SupportedOption(OptionEnum::topK()); $array = $original->toArray(); $restored = SupportedOption::fromArray($array); - $this->assertEquals($original->getName(), $restored->getName()); + $this->assertSame($original->getName(), $restored->getName()); $this->assertEquals($original->getSupportedValues(), $restored->getSupportedValues()); $this->assertNull($restored->getSupportedValues()); } @@ -495,14 +496,14 @@ public function testRoundTripWithNullSupportedValues(): void */ public function testJsonSerializationWithNullSupportedValues(): void { - $option = new SupportedOption('json_option'); + $option = new SupportedOption(OptionEnum::customOptions()); $json = json_encode($option); $decoded = json_decode($json, true); $this->assertIsString($json); $this->assertIsArray($decoded); - $this->assertEquals('json_option', $decoded[SupportedOption::KEY_NAME]); + $this->assertEquals(OptionEnum::customOptions()->value, $decoded[SupportedOption::KEY_NAME]); $this->assertArrayNotHasKey(SupportedOption::KEY_SUPPORTED_VALUES, $decoded); } From 3396da69c0143c59eff3a73304137734224fa7a1 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 09:01:30 -0700 Subject: [PATCH 35/47] refactor: tightens AbstractEnum cache scope --- src/Common/AbstractEnum.php | 87 +++++++++++-------- src/Providers/Models/DTO/ModelConfig.php | 10 ++- src/Providers/Models/Enums/OptionEnum.php | 18 ++-- .../Models/DTO/ModelMetadataTest.php | 15 +++- .../Models/DTO/ModelRequirementsTest.php | 10 ++- .../Models/DTO/RequiredOptionTest.php | 5 +- 6 files changed, 89 insertions(+), 56 deletions(-) diff --git a/src/Common/AbstractEnum.php b/src/Common/AbstractEnum.php index a339c74e..6b4a2cf4 100644 --- a/src/Common/AbstractEnum.php +++ b/src/Common/AbstractEnum.php @@ -51,7 +51,7 @@ abstract class AbstractEnum /** * @var array> Cache for reflection data. */ - protected static array $cache = []; + private static array $cache = []; /** * @var array> Cache for enum instances. @@ -253,48 +253,65 @@ private static function getInstance(string $value, string $name): self * @return array Map of constant names to values. * @throws RuntimeException If invalid constant found. */ - protected static function getConstants(): array + final protected static function getConstants(): array { $className = static::class; if (!isset(self::$cache[$className])) { - $reflection = new ReflectionClass($className); - $constants = $reflection->getConstants(); - - // Validate all constants - $enumConstants = []; - foreach ($constants as $name => $value) { - // Check if constant name follows uppercase snake_case pattern - if (!preg_match('/^[A-Z][A-Z0-9_]*$/', $name)) { - throw new RuntimeException( - sprintf( - 'Invalid enum constant name "%s" in %s. Constants must be UPPER_SNAKE_CASE.', - $name, - $className - ) - ); - } - - // Check if value is valid type - if (!is_string($value)) { - throw new RuntimeException( - sprintf( - 'Invalid enum value type for constant %s::%s. ' . - 'Only string values are allowed, %s given.', - $className, - $name, - gettype($value) - ) - ); - } - - $enumConstants[$name] = $value; + self::$cache[$className] = static::determineClassEnumerations($className); + } + + return self::$cache[$className]; + } + + /** + * Determines the class enumerations by reflecting on class constants. + * + * This method can be overridden by subclasses to customize how + * enumerations are determined (e.g., to add dynamic constants). + * + * @since n.e.x.t + * + * @param class-string $className The fully qualified class name. + * @return array Map of constant names to values. + * @throws RuntimeException If invalid constant found. + */ + protected static function determineClassEnumerations(string $className): array + { + $reflection = new ReflectionClass($className); + $constants = $reflection->getConstants(); + + // Validate all constants + $enumConstants = []; + foreach ($constants as $name => $value) { + // Check if constant name follows uppercase snake_case pattern + if (!preg_match('/^[A-Z][A-Z0-9_]*$/', $name)) { + throw new RuntimeException( + sprintf( + 'Invalid enum constant name "%s" in %s. Constants must be UPPER_SNAKE_CASE.', + $name, + $className + ) + ); + } + + // Check if value is valid type + if (!is_string($value)) { + throw new RuntimeException( + sprintf( + 'Invalid enum value type for constant %s::%s. ' . + 'Only string values are allowed, %s given.', + $className, + $name, + gettype($value) + ) + ); } - self::$cache[$className] = $enumConstants; + $enumConstants[$name] = $value; } - return self::$cache[$className]; + return $enumConstants; } /** diff --git a/src/Providers/Models/DTO/ModelConfig.php b/src/Providers/Models/DTO/ModelConfig.php index 9e2fd5c6..9555be05 100644 --- a/src/Providers/Models/DTO/ModelConfig.php +++ b/src/Providers/Models/DTO/ModelConfig.php @@ -1024,11 +1024,17 @@ public function toRequiredOptions(): array } if ($this->outputMediaOrientation !== null) { - $requiredOptions[] = new RequiredOption(OptionEnum::outputMediaOrientation(), $this->outputMediaOrientation->value); + $requiredOptions[] = new RequiredOption( + OptionEnum::outputMediaOrientation(), + $this->outputMediaOrientation->value + ); } if ($this->outputMediaAspectRatio !== null) { - $requiredOptions[] = new RequiredOption(OptionEnum::outputMediaAspectRatio(), $this->outputMediaAspectRatio); + $requiredOptions[] = new RequiredOption( + OptionEnum::outputMediaAspectRatio(), + $this->outputMediaAspectRatio + ); } // Add custom options as individual RequiredOptions diff --git a/src/Providers/Models/Enums/OptionEnum.php b/src/Providers/Models/Enums/OptionEnum.php index dc7ba698..508cc67d 100644 --- a/src/Providers/Models/Enums/OptionEnum.php +++ b/src/Providers/Models/Enums/OptionEnum.php @@ -73,7 +73,7 @@ class OptionEnum extends AbstractEnum public const INPUT_MODALITIES = 'input_modalities'; /** - * Gets the constants for this enum. + * Determines the class enumerations by reflecting on class constants. * * Overrides the parent method to dynamically add constants from ModelConfig * that are prefixed with KEY_. These are transformed to remove the KEY_ prefix @@ -81,18 +81,13 @@ class OptionEnum extends AbstractEnum * * @since n.e.x.t * + * @param class-string $className The fully qualified class name. * @return array The enum constants. */ - protected static function getConstants(): array + protected static function determineClassEnumerations(string $className): array { - // Check if we already have cached constants for this class - $className = static::class; - if (isset(self::$cache[$className])) { - return self::$cache[$className]; - } - - // Start with the constants defined in this class - $constants = parent::getConstants(); + // Start with the constants defined in this class using parent method + $constants = parent::determineClassEnumerations($className); // Use reflection to get all constants from ModelConfig $modelConfigReflection = new ReflectionClass(ModelConfig::class); @@ -112,9 +107,6 @@ protected static function getConstants(): array } } - // Cache the combined constants - self::$cache[$className] = $constants; - return $constants; } } diff --git a/tests/unit/Providers/Models/DTO/ModelMetadataTest.php b/tests/unit/Providers/Models/DTO/ModelMetadataTest.php index 52d7daf5..766e2dd1 100644 --- a/tests/unit/Providers/Models/DTO/ModelMetadataTest.php +++ b/tests/unit/Providers/Models/DTO/ModelMetadataTest.php @@ -150,12 +150,18 @@ public function testToArray(): void $this->assertEquals('Claude 2', $array[ModelMetadata::KEY_NAME]); $this->assertEquals(['text_generation', 'chat_history'], $array[ModelMetadata::KEY_SUPPORTED_CAPABILITIES]); $this->assertCount(2, $array[ModelMetadata::KEY_SUPPORTED_OPTIONS]); - $this->assertEquals(OptionEnum::maxTokens()->value, $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME]); + $this->assertEquals( + OptionEnum::maxTokens()->value, + $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME] + ); $this->assertEquals( [100, 1000, 10000], $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_SUPPORTED_VALUES] ); - $this->assertEquals(OptionEnum::temperature()->value, $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][1][SupportedOption::KEY_NAME]); + $this->assertEquals( + OptionEnum::temperature()->value, + $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][1][SupportedOption::KEY_NAME] + ); $this->assertEquals( [0.0, 1.0], $array[ModelMetadata::KEY_SUPPORTED_OPTIONS][1][SupportedOption::KEY_SUPPORTED_VALUES] @@ -271,7 +277,10 @@ public function testJsonSerialize(): void $this->assertEquals('JSON Test Model', $decoded[ModelMetadata::KEY_NAME]); $this->assertEquals(['embedding_generation'], $decoded[ModelMetadata::KEY_SUPPORTED_CAPABILITIES]); $this->assertCount(1, $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS]); - $this->assertEquals(OptionEnum::outputSchema()->value, $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME]); + $this->assertEquals( + OptionEnum::outputSchema()->value, + $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_NAME] + ); $this->assertEquals( [256, 512, 1024], $decoded[ModelMetadata::KEY_SUPPORTED_OPTIONS][0][SupportedOption::KEY_SUPPORTED_VALUES] diff --git a/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php b/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php index ca773138..21f71b2b 100644 --- a/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php +++ b/tests/unit/Providers/Models/DTO/ModelRequirementsTest.php @@ -123,9 +123,15 @@ public function testToArray(): void $array[ModelRequirements::KEY_REQUIRED_CAPABILITIES] ); $this->assertCount(2, $array[ModelRequirements::KEY_REQUIRED_OPTIONS]); - $this->assertEquals(OptionEnum::outputSchema()->value, $array[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_NAME]); + $this->assertEquals( + OptionEnum::outputSchema()->value, + $array[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_NAME] + ); $this->assertEquals('1024x1024', $array[ModelRequirements::KEY_REQUIRED_OPTIONS][0][RequiredOption::KEY_VALUE]); - $this->assertEquals(OptionEnum::outputSchema()->value, $array[ModelRequirements::KEY_REQUIRED_OPTIONS][1][RequiredOption::KEY_NAME]); + $this->assertEquals( + OptionEnum::outputSchema()->value, + $array[ModelRequirements::KEY_REQUIRED_OPTIONS][1][RequiredOption::KEY_NAME] + ); $this->assertEquals('realistic', $array[ModelRequirements::KEY_REQUIRED_OPTIONS][1][RequiredOption::KEY_VALUE]); } diff --git a/tests/unit/Providers/Models/DTO/RequiredOptionTest.php b/tests/unit/Providers/Models/DTO/RequiredOptionTest.php index 6324bf3c..3b20551b 100644 --- a/tests/unit/Providers/Models/DTO/RequiredOptionTest.php +++ b/tests/unit/Providers/Models/DTO/RequiredOptionTest.php @@ -300,7 +300,10 @@ public function testArrayRoundTrip(): void new RequiredOption(OptionEnum::webSearch(), true), new RequiredOption(OptionEnum::outputSchema(), null), new RequiredOption(OptionEnum::stopSequences(), ['one', 'two', 'three']), - new RequiredOption(OptionEnum::customOptions(), ['type' => 'config', 'enabled' => true, 'settings' => ['a' => 1, 'b' => 2]]) + new RequiredOption( + OptionEnum::customOptions(), + ['type' => 'config', 'enabled' => true, 'settings' => ['a' => 1, 'b' => 2]] + ) ]; foreach ($testCases as $original) { From 5d081b43fbc95a71561ab541addaa90ffa53fbd1 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 09:22:34 -0700 Subject: [PATCH 36/47] fix: corrects isSupported to use capabilities --- src/Builders/PromptBuilder.php | 96 +++++++++---- tests/unit/Builders/PromptBuilderTest.php | 163 +++------------------- 2 files changed, 85 insertions(+), 174 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 0ea73136..ea358a52 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -428,29 +428,47 @@ public function getModelRequirements(): ModelRequirements * * @since n.e.x.t * - * @param ModalityEnum|null $intendedOutput Optional output modality to check support for. + * @param CapabilityEnum|null $intendedCapability Optional capability to check support for. * @return bool True if supported, false otherwise. */ - public function isSupported(?ModalityEnum $intendedOutput = null): bool + private function isSupported(?CapabilityEnum $intendedCapability = null): bool { - // If an intended output modality is specified, temporarily include it - $originalModalities = null; - if ($intendedOutput !== null) { - $originalModalities = $this->modelConfig->getOutputModalities(); - $this->includeOutputModalities($intendedOutput); + // If a model has been explicitly set, we assume it meets the requirements + if ($this->model !== null) { + return true; + } + + // Build requirements with the intended capability if specified + $requirements = $this->getModelRequirements(); + + if ($intendedCapability !== null) { + $capabilities = $requirements->getRequiredCapabilities(); + + // Check if capability is already present + $hasCapability = false; + foreach ($capabilities as $capability) { + if ($capability->equals($intendedCapability)) { + $hasCapability = true; + break; + } + } + + // Add the capability if not already present + if (!$hasCapability) { + $capabilities[] = $intendedCapability; + $requirements = new ModelRequirements( + $capabilities, + $requirements->getRequiredOptions() + ); + } } try { - // Try to get a configured model - this will throw if no suitable model exists - $this->getConfiguredModel(); - return true; + // Check if any models support these requirements + $models = $this->registry->findModelsMetadataForSupport($requirements); + return !empty($models); } catch (InvalidArgumentException $e) { return false; - } finally { - // Restore original modalities if we modified them - if ($originalModalities !== null) { - $this->modelConfig->setOutputModalities($originalModalities); - } } } @@ -461,9 +479,9 @@ public function isSupported(?ModalityEnum $intendedOutput = null): bool * * @return bool True if text generation is supported. */ - public function isSupportedForText(): bool + public function isSupportedForTextGeneration(): bool { - return $this->isSupported(ModalityEnum::text()); + return $this->isSupported(CapabilityEnum::textGeneration()); } /** @@ -473,21 +491,21 @@ public function isSupportedForText(): bool * * @return bool True if image generation is supported. */ - public function isSupportedForImage(): bool + public function isSupportedForImageGeneration(): bool { - return $this->isSupported(ModalityEnum::image()); + return $this->isSupported(CapabilityEnum::imageGeneration()); } /** - * Checks if the prompt is supported for audio generation. + * Checks if the prompt is supported for text to speech conversion. * * @since n.e.x.t * - * @return bool True if audio generation is supported. + * @return bool True if text to speech conversion is supported. */ - public function isSupportedForAudio(): bool + public function isSupportedForTextToSpeechConversion(): bool { - return $this->isSupported(ModalityEnum::audio()); + return $this->isSupported(CapabilityEnum::textToSpeechConversion()); } /** @@ -497,9 +515,9 @@ public function isSupportedForAudio(): bool * * @return bool True if video generation is supported. */ - public function isSupportedForVideo(): bool + public function isSupportedForVideoGeneration(): bool { - return $this->isSupported(ModalityEnum::video()); + return $this->isSupported(CapabilityEnum::videoGeneration()); } /** @@ -509,9 +527,33 @@ public function isSupportedForVideo(): bool * * @return bool True if speech generation is supported. */ - public function isSupportedForSpeech(): bool + public function isSupportedForSpeechGeneration(): bool + { + return $this->isSupported(CapabilityEnum::speechGeneration()); + } + + /** + * Checks if the prompt is supported for music generation. + * + * @since n.e.x.t + * + * @return bool True if music generation is supported. + */ + public function isSupportedForMusicGeneration(): bool + { + return $this->isSupported(CapabilityEnum::musicGeneration()); + } + + /** + * Checks if the prompt is supported for embedding generation. + * + * @since n.e.x.t + * + * @return bool True if embedding generation is supported. + */ + public function isSupportedForEmbeddingGeneration(): bool { - return $this->isSupported(ModalityEnum::audio()); + return $this->isSupported(CapabilityEnum::embeddingGeneration()); } /** diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index 744fd231..ef260799 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -986,66 +986,6 @@ public function testGetModelRequirementsWithMultimodalInput(): void $this->assertTrue($hasImage); } - /** - * Tests isSupported without model. - * - * @return void - */ - public function testIsSupportedWithoutModel(): void - { - // Mock registry to return models - $providerMetadata = $this->createMock(ProviderMetadata::class); - $modelMetadata = $this->createMock(ModelMetadata::class); - $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); - $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); - $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); - $this->registry->method('findModelsMetadataForSupport')->willReturn([$providerModelsMetadata]); - $this->registry->method('getProviderModel')->willReturn($this->createMock(ModelInterface::class)); - - $builder = new PromptBuilder($this->registry, 'Test'); - - // Without a model explicitly set, it should try to find one from registry - $this->assertTrue($builder->isSupported()); - } - - /** - * Tests isSupported with compatible model. - * - * @return void - */ - public function testIsSupportedWithCompatibleModel(): void - { - $metadata = $this->createMock(ModelMetadata::class); - $metadata->method('getId')->willReturn('test-model'); - - $model = $this->createMock(ModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->expects($this->once()) - ->method('setConfig') - ->with($this->isInstanceOf(ModelConfig::class)); - - $builder = new PromptBuilder($this->registry, 'Test'); - $builder->usingModel($model); - - // With an explicitly set model, it should always return true - $this->assertTrue($builder->isSupported()); - } - - /** - * Tests isSupported with incompatible model. - * - * @return void - */ - public function testIsSupportedWithIncompatibleModel(): void - { - // When no models are found in registry, it should return false - $this->registry->method('findModelsMetadataForSupport')->willReturn([]); - - $builder = new PromptBuilder($this->registry, 'Test'); - - // Without any available models, it should return false - $this->assertFalse($builder->isSupported()); - } /** * Tests validateMessages with empty messages throws exception. @@ -2704,44 +2644,6 @@ public function testChainGenerationWithMultiplePrompts(): void $this->markTestSkipped('Complex chaining with model response methods not fully implemented yet'); } - /** - * Tests isSupported with intended output modality. - * - * @return void - */ - public function testIsSupportedWithIntendedOutput(): void - { - $builder = new PromptBuilder($this->registry, 'Test prompt'); - - // Mock registry to return no models for image generation - $this->registry->method('findModelsMetadataForSupport') - ->willReturnCallback(function ($requirements) { - $options = $requirements->getRequiredOptions(); - foreach ($options as $option) { - if ($option->getName()->equals(OptionEnum::outputModalities())) { - $modalities = $option->getValue(); - foreach ($modalities as $modality) { - if ($modality->isImage()) { - return []; // No models support image generation - } - } - } - } - // Return a mock model for text generation - $providerMetadata = $this->createMock(ProviderMetadata::class); - $modelMetadata = $this->createMock(ModelMetadata::class); - $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); - $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); - $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); - return [$providerModelsMetadata]; - }); - - // Text should be supported - $this->assertTrue($builder->isSupported(ModalityEnum::text())); - - // Image should not be supported - $this->assertFalse($builder->isSupported(ModalityEnum::image())); - } /** * Tests isSupportedForText convenience method. @@ -2752,6 +2654,7 @@ public function testIsSupportedForText(): void { $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('text-model'); + $metadata->method('meetsRequirements')->willReturn(true); $result = new GenerativeAiResult('test-id', [ new Candidate( @@ -2765,15 +2668,15 @@ public function testIsSupportedForText(): void $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); - $this->assertTrue($builder->isSupportedForText()); + $this->assertTrue($builder->isSupportedForTextGeneration()); } /** - * Tests isSupportedForImage convenience method. + * Tests isSupportedForImageGeneration convenience method. * * @return void */ - public function testIsSupportedForImage(): void + public function testIsSupportedForImageGeneration(): void { $builder = new PromptBuilder($this->registry, 'Generate an image'); @@ -2781,31 +2684,31 @@ public function testIsSupportedForImage(): void $this->registry->method('findModelsMetadataForSupport') ->willReturn([]); - $this->assertFalse($builder->isSupportedForImage()); + $this->assertFalse($builder->isSupportedForImageGeneration()); } /** - * Tests isSupportedForAudio convenience method. + * Tests isSupportedForTextToSpeechConversion convenience method. * * @return void */ - public function testIsSupportedForAudio(): void + public function testIsSupportedForTextToSpeechConversion(): void { $builder = new PromptBuilder($this->registry, 'Generate audio'); - // Mock registry to return no models for audio generation + // Mock registry to return no models for text to speech conversion $this->registry->method('findModelsMetadataForSupport') ->willReturn([]); - $this->assertFalse($builder->isSupportedForAudio()); + $this->assertFalse($builder->isSupportedForTextToSpeechConversion()); } /** - * Tests isSupportedForVideo convenience method. + * Tests isSupportedForVideoGeneration convenience method. * * @return void */ - public function testIsSupportedForVideo(): void + public function testIsSupportedForVideoGeneration(): void { $builder = new PromptBuilder($this->registry, 'Generate video'); @@ -2813,18 +2716,19 @@ public function testIsSupportedForVideo(): void $this->registry->method('findModelsMetadataForSupport') ->willReturn([]); - $this->assertFalse($builder->isSupportedForVideo()); + $this->assertFalse($builder->isSupportedForVideoGeneration()); } /** - * Tests isSupportedForSpeech convenience method. + * Tests isSupportedForSpeechGeneration convenience method. * * @return void */ - public function testIsSupportedForSpeech(): void + public function testIsSupportedForSpeechGeneration(): void { $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('speech-model'); + $metadata->method('meetsRequirements')->willReturn(true); $result = new GenerativeAiResult('test-id', [ new Candidate( @@ -2838,41 +2742,6 @@ public function testIsSupportedForSpeech(): void $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); - $this->assertTrue($builder->isSupportedForSpeech()); - } - - /** - * Tests isSupported restores original modalities after check. - * - * @return void - */ - public function testIsSupportedRestoresOriginalModalities(): void - { - $builder = new PromptBuilder($this->registry, 'Test prompt'); - - // Set initial modality - $builder->usingOutputModalities(ModalityEnum::text()); - - // Mock registry to return models - $providerMetadata = $this->createMock(ProviderMetadata::class); - $modelMetadata = $this->createMock(ModelMetadata::class); - $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); - $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); - $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); - $this->registry->method('findModelsMetadataForSupport')->willReturn([$providerModelsMetadata]); - $this->registry->method('getProviderModel')->willReturn($this->createMock(ModelInterface::class)); - - // Check with image modality - $builder->isSupported(ModalityEnum::image()); - - // Verify original modality is restored - $reflection = new \ReflectionClass($builder); - $configProperty = $reflection->getProperty('modelConfig'); - $configProperty->setAccessible(true); - $config = $configProperty->getValue($builder); - - $modalities = $config->getOutputModalities(); - $this->assertCount(1, $modalities); - $this->assertTrue($modalities[0]->isText()); + $this->assertTrue($builder->isSupportedForSpeechGeneration()); } } From cb7fb19c6e562b78dedd2fa622e193fbadfa1b6c Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 09:37:56 -0700 Subject: [PATCH 37/47] refactor: corrects input modality collection --- src/Builders/PromptBuilder.php | 34 ++++++++++++++++------------------ 1 file changed, 16 insertions(+), 18 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index ea358a52..77ed8c38 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -380,21 +380,24 @@ public function getModelRequirements(): ModelRequirements foreach ($this->messages as $message) { foreach ($message->getParts() as $part) { // Check for text input - if ($part->getText() !== null) { - $inputModalities[ModalityEnum::text()->value] = ModalityEnum::text(); + if ($part->getType()->isText()) { + $inputModalities[] = ModalityEnum::text(); } // Check for file inputs - $file = $part->getFile(); - if ($file !== null) { - if ($file->isImage()) { - $inputModalities[ModalityEnum::image()->value] = ModalityEnum::image(); - } elseif ($file->isAudio()) { - $inputModalities[ModalityEnum::audio()->value] = ModalityEnum::audio(); - } elseif ($file->isVideo()) { - $inputModalities[ModalityEnum::video()->value] = ModalityEnum::video(); - } elseif ($file->isDocument() || $file->isText()) { - $inputModalities[ModalityEnum::document()->value] = ModalityEnum::document(); + if ($part->getType()->isFile()) { + $file = $part->getFile(); + + if ($file !== null) { + if ($file->isImage()) { + $inputModalities[] = ModalityEnum::image(); + } elseif ($file->isAudio()) { + $inputModalities[] = ModalityEnum::audio(); + } elseif ($file->isVideo()) { + $inputModalities[] = ModalityEnum::video(); + } elseif ($file->isDocument() || $file->isText()) { + $inputModalities[] = ModalityEnum::document(); + } } } @@ -410,12 +413,7 @@ public function getModelRequirements(): ModelRequirements $requiredOptions = $this->modelConfig->toRequiredOptions(); // Add input modalities if we have non-text inputs - if (count($inputModalities) > 0) { - $requiredOptions[] = new RequiredOption( - OptionEnum::inputModalities(), - array_values($inputModalities) - ); - } + $requiredOptions[] = new RequiredOption(OptionEnum::inputModalities(), $inputModalities); return new ModelRequirements( $capabilities, From 2c7b4436a1b00417034ebeb7e71546acdc9678ac Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 10:12:59 -0700 Subject: [PATCH 38/47] feat: adds function call checking to requirements --- src/Builders/PromptBuilder.php | 48 ++++++++++++++++++++++++++++++---- 1 file changed, 43 insertions(+), 5 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 77ed8c38..852828db 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -377,6 +377,7 @@ public function getModelRequirements(): ModelRequirements } // Analyze all messages to determine required input modalities + $hasFunctionCallingCapability = false; foreach ($this->messages as $message) { foreach ($message->getParts() as $part) { // Check for text input @@ -402,9 +403,8 @@ public function getModelRequirements(): ModelRequirements } // Check for function calls/responses (these might require special capabilities) - if ($part->getFunctionCall() !== null || $part->getFunctionResponse() !== null) { - // Function calling capability would go here if we had it in CapabilityEnum - // For now, we'll just note this requires text generation + if ($part->getType()->isFunctionCall() || $part->getType()->isFunctionResponse()) { + $hasFunctionCallingCapability = true; } } } @@ -412,8 +412,19 @@ public function getModelRequirements(): ModelRequirements // Build required options from ModelConfig $requiredOptions = $this->modelConfig->toRequiredOptions(); - // Add input modalities if we have non-text inputs - $requiredOptions[] = new RequiredOption(OptionEnum::inputModalities(), $inputModalities); + if ($hasFunctionCallingCapability) { + // Add function declarations option if we have function calls/responses + $requiredOptions = $this->includeInRequiredOptions( + $requiredOptions, + new RequiredOption(OptionEnum::functionDeclarations(), true) + ); + } + + // Add input modalities if we have any inputs + $requiredOptions = $this->includeInRequiredOptions( + $requiredOptions, + new RequiredOption(OptionEnum::inputModalities(), $inputModalities) + ); return new ModelRequirements( $capabilities, @@ -1245,6 +1256,33 @@ private function isMessagesList($value): bool return true; } + /** + * Includes a required option in the list if not already present. + * + * Checks if a RequiredOption with the same name already exists in the list. + * If not, adds the new option. Returns the updated list. + * + * @since n.e.x.t + * + * @param list $options The existing list of required options. + * @param RequiredOption $option The option to potentially add. + * @return list The updated list of required options. + */ + private function includeInRequiredOptions(array $options, RequiredOption $option): array + { + // Check if an option with the same name already exists + foreach ($options as $existingOption) { + if ($existingOption->getName()->equals($option->getName())) { + // Option already exists, return unchanged list + return $options; + } + } + + // Add the new option + $options[] = $option; + return $options; + } + /** * Includes output modalities if not already present. * From b1a3f4477019002cd31dae60e47e605382f3ef2e Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 10:44:06 -0700 Subject: [PATCH 39/47] refactor: goes back to checking if model meets requirements in support check --- src/Builders/PromptBuilder.php | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 852828db..576c6d2f 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -442,11 +442,6 @@ public function getModelRequirements(): ModelRequirements */ private function isSupported(?CapabilityEnum $intendedCapability = null): bool { - // If a model has been explicitly set, we assume it meets the requirements - if ($this->model !== null) { - return true; - } - // Build requirements with the intended capability if specified $requirements = $this->getModelRequirements(); @@ -472,6 +467,11 @@ private function isSupported(?CapabilityEnum $intendedCapability = null): bool } } + // If the model has been set, check if it meets the requirements + if ($this->model !== null) { + return $this->model->metadata()->meetsRequirements($requirements); + } + try { // Check if any models support these requirements $models = $this->registry->findModelsMetadataForSupport($requirements); From f366cdd91f470f6928351cff4ef472c7efd8fe61 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 11:11:52 -0700 Subject: [PATCH 40/47] refactor: adjusts generateResult to use capability --- src/Builders/PromptBuilder.php | 115 +++++++++++++++++++-------------- 1 file changed, 66 insertions(+), 49 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 576c6d2f..4646b604 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -569,33 +569,63 @@ public function isSupportedForEmbeddingGeneration(): bool * Generates a result from the prompt. * * This is the primary execution method that generates a result (containing - * potentially multiple candidates) based on the configured output modality. + * potentially multiple candidates) based on the specified capability or + * the configured output modality. * * @since n.e.x.t * + * @param CapabilityEnum|null $capability Optional capability to use for generation. + * If null, capability is inferred from output modality. * @return GenerativeAiResult The generated result containing candidates. * @throws InvalidArgumentException If the prompt or model validation fails. - * @throws RuntimeException If the model doesn't support the configured output modality. + * @throws RuntimeException If the model doesn't support the required capability. */ - public function generateResult(): GenerativeAiResult + public function generateResult(?CapabilityEnum $capability = null): GenerativeAiResult { $this->validateMessages(); $model = $this->getConfiguredModel(); - // Get the configured output modalities - $outputModalities = $this->modelConfig->getOutputModalities(); + // If capability is not provided, infer it from output modalities + if ($capability === null) { + // Get the configured output modalities + $outputModalities = $this->modelConfig->getOutputModalities(); - // Default to text if no output modality is specified - if ($outputModalities === null || empty($outputModalities)) { - $outputModalities = [ModalityEnum::text()]; + // Default to text if no output modality is specified + if ($outputModalities === null || empty($outputModalities)) { + $outputModalities = [ModalityEnum::text()]; + } + + // Multi-modal output (multiple modalities) defaults to text generation. This is temporary + // as a multi-modal interface will be implemented in the future. + if (count($outputModalities) > 1) { + $capability = CapabilityEnum::textGeneration(); + } else { + // Infer capability from single output modality + $outputModality = $outputModalities[0]; + + if ($outputModality->isText()) { + $capability = CapabilityEnum::textGeneration(); + } elseif ($outputModality->isImage()) { + $capability = CapabilityEnum::imageGeneration(); + } elseif ($outputModality->isAudio()) { + $capability = CapabilityEnum::speechGeneration(); + } elseif ($outputModality->isVideo()) { + $capability = CapabilityEnum::videoGeneration(); + } else { + // For unsupported modalities, provide a clear error message + throw new RuntimeException( + sprintf('Output modality "%s" is not yet supported.', $outputModality->value) + ); + } + } } - // Multi-modal output (multiple modalities) uses TextGenerationModelInterface - if (count($outputModalities) > 1) { + // Route to the appropriate generation method based on capability + if ($capability->isTextGeneration()) { if (!$model instanceof TextGenerationModelInterface) { throw new RuntimeException( sprintf( - 'Model "%s" does not support multi-modal generation.', + 'Model "%s" does not support text generation.', $model->metadata()->getId() ) ); @@ -603,39 +633,35 @@ public function generateResult(): GenerativeAiResult return $model->generateTextResult($this->messages); } - // Single modality routing - $outputModality = $outputModalities[0]; - - // Route to the appropriate generation method based on output modality - if ($outputModality->isText()) { - if (!$model instanceof TextGenerationModelInterface) { + if ($capability->isImageGeneration()) { + if (!$model instanceof ImageGenerationModelInterface) { throw new RuntimeException( sprintf( - 'Model "%s" does not support text generation.', + 'Model "%s" does not support image generation.', $model->metadata()->getId() ) ); } - return $model->generateTextResult($this->messages); + return $model->generateImageResult($this->messages); } - if ($outputModality->isImage()) { - if (!$model instanceof ImageGenerationModelInterface) { + if ($capability->isTextToSpeechConversion()) { + if (!$model instanceof TextToSpeechConversionModelInterface) { throw new RuntimeException( sprintf( - 'Model "%s" does not support image generation.', + 'Model "%s" does not support text-to-speech conversion.', $model->metadata()->getId() ) ); } - return $model->generateImageResult($this->messages); + return $model->convertTextToSpeechResult($this->messages); } - if ($outputModality->isAudio()) { + if ($capability->isSpeechGeneration()) { if (!$model instanceof SpeechGenerationModelInterface) { throw new RuntimeException( sprintf( - 'Model "%s" does not support speech/audio generation.', + 'Model "%s" does not support speech generation.', $model->metadata()->getId() ) ); @@ -643,9 +669,14 @@ public function generateResult(): GenerativeAiResult return $model->generateSpeechResult($this->messages); } - // TODO: Add support for video output modality when interface is available + if ($capability->isVideoGeneration()) { + // Video generation is not yet implemented + throw new RuntimeException('Output modality "video" is not yet supported.'); + } + + // TODO: Add support for other capabilities when interfaces are available throw new RuntimeException( - sprintf('Output modality "%s" is not yet supported.', $outputModality->value) + sprintf('Capability "%s" is not yet supported for generation.', $capability->value) ); } @@ -663,8 +694,8 @@ public function generateTextResult(): GenerativeAiResult // Include text in output modalities $this->includeOutputModalities(ModalityEnum::text()); - // Generate and return the result - return $this->generateResult(); + // Generate and return the result with text generation capability + return $this->generateResult(CapabilityEnum::textGeneration()); } /** @@ -681,8 +712,8 @@ public function generateImageResult(): GenerativeAiResult // Include image in output modalities $this->includeOutputModalities(ModalityEnum::image()); - // Generate and return the result - return $this->generateResult(); + // Generate and return the result with image generation capability + return $this->generateResult(CapabilityEnum::imageGeneration()); } /** @@ -699,8 +730,8 @@ public function generateSpeechResult(): GenerativeAiResult // Include audio in output modalities $this->includeOutputModalities(ModalityEnum::audio()); - // Generate and return the result - return $this->generateResult(); + // Generate and return the result with speech generation capability + return $this->generateResult(CapabilityEnum::speechGeneration()); } /** @@ -717,22 +748,8 @@ public function convertTextToSpeechResult(): GenerativeAiResult // Include audio in output modalities $this->includeOutputModalities(ModalityEnum::audio()); - // Get the configured model - $model = $this->getConfiguredModel(); - - // Ensure the model supports text-to-speech conversion - if (!$model instanceof TextToSpeechConversionModelInterface) { - throw new RuntimeException( - sprintf( - 'Model "%s" does not support text-to-speech conversion.', - $model->metadata()->getId() - ) - ); - } - - // Validate messages and convert - $this->validateMessages(); - return $model->convertTextToSpeechResult($this->messages); + // Generate and return the result with text-to-speech conversion capability + return $this->generateResult(CapabilityEnum::textToSpeechConversion()); } /** From 4f4779ca54d1982c4b4b80f76348ee524e9cda58 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 11:13:59 -0700 Subject: [PATCH 41/47] refactor: improves variable naming --- src/Builders/PromptBuilder.php | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 4646b604..85a2103b 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -377,7 +377,7 @@ public function getModelRequirements(): ModelRequirements } // Analyze all messages to determine required input modalities - $hasFunctionCallingCapability = false; + $hasFunctionMessageParts = false; foreach ($this->messages as $message) { foreach ($message->getParts() as $part) { // Check for text input @@ -404,7 +404,7 @@ public function getModelRequirements(): ModelRequirements // Check for function calls/responses (these might require special capabilities) if ($part->getType()->isFunctionCall() || $part->getType()->isFunctionResponse()) { - $hasFunctionCallingCapability = true; + $hasFunctionMessageParts = true; } } } @@ -412,7 +412,7 @@ public function getModelRequirements(): ModelRequirements // Build required options from ModelConfig $requiredOptions = $this->modelConfig->toRequiredOptions(); - if ($hasFunctionCallingCapability) { + if ($hasFunctionMessageParts) { // Add function declarations option if we have function calls/responses $requiredOptions = $this->includeInRequiredOptions( $requiredOptions, From 5d9a2a989642ea6739567ffa4fc48f8928978cb8 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 11:24:48 -0700 Subject: [PATCH 42/47] refactor: cleans up generate methods big time --- src/Builders/PromptBuilder.php | 185 ++------------------------------- 1 file changed, 8 insertions(+), 177 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 85a2103b..2464aca0 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -762,27 +762,7 @@ public function convertTextToSpeechResult(): GenerativeAiResult */ public function generateText(): string { - // Generate text result and extract text from first candidate - $result = $this->generateTextResult(); - $candidates = $result->getCandidates(); - - if (empty($candidates)) { - throw new RuntimeException('No candidates were generated.'); - } - - // Get the text from the first message part - $message = $candidates[0]->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - throw new RuntimeException('Generated message contains no parts.'); - } - - $text = $parts[0]->getText(); - if ($text === null) { - throw new RuntimeException('Generated message part contains no text.'); - } - - return $text; + return $this->generateTextResult()->toText(); } /** @@ -801,29 +781,7 @@ public function generateTexts(?int $candidateCount = null): array } // Generate text result - $results = $this->generateTextResult(); - $candidates = $results->getCandidates(); - - // Extract text from each candidate - $texts = []; - foreach ($candidates as $candidate) { - $message = $candidate->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - continue; - } - - $text = $parts[0]->getText(); - if ($text !== null) { - $texts[] = $text; - } - } - - if (empty($texts)) { - throw new RuntimeException('No text was generated from any candidates.'); - } - - return $texts; + return $this->generateTextResult()->toTexts(); } /** @@ -837,27 +795,7 @@ public function generateTexts(?int $candidateCount = null): array */ public function generateImage(): File { - // Generate image result and extract image from first candidate - $result = $this->generateImageResult(); - $candidates = $result->getCandidates(); - - if (empty($candidates)) { - throw new RuntimeException('No candidates were generated.'); - } - - // Get the image file from the first message part - $message = $candidates[0]->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - throw new RuntimeException('Generated message contains no parts.'); - } - - $file = $parts[0]->getFile(); - if ($file === null) { - throw new RuntimeException('Generated message part contains no image file.'); - } - - return $file; + return $this->generateImageResult()->toFile(); } /** @@ -876,30 +814,7 @@ public function generateImages(?int $candidateCount = null): array $this->usingCandidateCount($candidateCount); } - // Generate image result - $results = $this->generateImageResult(); - $candidates = $results->getCandidates(); - - // Extract image files from each candidate - $images = []; - foreach ($candidates as $candidate) { - $message = $candidate->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - continue; - } - - $file = $parts[0]->getFile(); - if ($file !== null) { - $images[] = $file; - } - } - - if (empty($images)) { - throw new RuntimeException('No images were generated from any candidates.'); - } - - return $images; + return $this->generateImageResult()->toFiles(); } /** @@ -913,26 +828,7 @@ public function generateImages(?int $candidateCount = null): array */ public function convertTextToSpeech(): File { - // Convert text to speech and extract audio from first candidate - $result = $this->convertTextToSpeechResult(); - $candidates = $result->getCandidates(); - - if (empty($candidates)) { - throw new RuntimeException('No candidates were generated.'); - } - - $message = $candidates[0]->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - throw new RuntimeException('Generated message contains no parts.'); - } - - $file = $parts[0]->getFile(); - if ($file === null) { - throw new RuntimeException('Generated message part contains no audio file.'); - } - - return $file; + return $this->convertTextToSpeechResult()->toFile(); } /** @@ -951,29 +847,7 @@ public function convertTextToSpeeches(?int $candidateCount = null): array $this->usingCandidateCount($candidateCount); } - // Convert text to speech - $result = $this->convertTextToSpeechResult(); - - // Extract audio files from each candidate - $audioFiles = []; - foreach ($result->getCandidates() as $candidate) { - $message = $candidate->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - continue; - } - - $file = $parts[0]->getFile(); - if ($file !== null) { - $audioFiles[] = $file; - } - } - - if (empty($audioFiles)) { - throw new RuntimeException('No audio files were generated from any candidates.'); - } - - return $audioFiles; + return $this->convertTextToSpeechResult()->toFiles(); } /** @@ -987,27 +861,7 @@ public function convertTextToSpeeches(?int $candidateCount = null): array */ public function generateSpeech(): File { - // Generate speech result and extract audio from first candidate - $result = $this->generateSpeechResult(); - $candidates = $result->getCandidates(); - - if (empty($candidates)) { - throw new RuntimeException('No candidates were generated.'); - } - - // Get the audio file from the first message part - $message = $candidates[0]->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - throw new RuntimeException('Generated message contains no parts.'); - } - - $file = $parts[0]->getFile(); - if ($file === null) { - throw new RuntimeException('Generated message part contains no audio file.'); - } - - return $file; + return $this->generateSpeechResult()->toFile(); } /** @@ -1026,30 +880,7 @@ public function generateSpeeches(?int $candidateCount = null): array $this->usingCandidateCount($candidateCount); } - // Generate speech result - $result = $this->generateSpeechResult(); - $candidates = $result->getCandidates(); - - // Extract audio files from each candidate - $audioFiles = []; - foreach ($candidates as $candidate) { - $message = $candidate->getMessage(); - $parts = $message->getParts(); - if (empty($parts)) { - continue; - } - - $file = $parts[0]->getFile(); - if ($file !== null) { - $audioFiles[] = $file; - } - } - - if (empty($audioFiles)) { - throw new RuntimeException('No audio files were generated from any candidates.'); - } - - return $audioFiles; + return $this->generateSpeechResult()->toFiles(); } /** From e7ffc3ce0ac9d5c51620901be6b789e2d06868a0 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 11:37:30 -0700 Subject: [PATCH 43/47] refactor: uses consistent to- naming convention --- src/Builders/PromptBuilder.php | 8 ++++---- tests/unit/Builders/PromptBuilderTest.php | 4 ++-- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 2464aca0..3ec1e0b1 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -305,7 +305,7 @@ public function usingCandidateCount(int $candidateCount): self * @param string $mimeType The MIME type. * @return self */ - public function usingOutputMime(string $mimeType): self + public function asOutputMimeType(string $mimeType): self { $this->modelConfig->setOutputMimeType($mimeType); return $this; @@ -319,7 +319,7 @@ public function usingOutputMime(string $mimeType): self * @param array $schema The output schema. * @return self */ - public function usingOutputSchema(array $schema): self + public function asOutputSchema(array $schema): self { $this->modelConfig->setOutputSchema($schema); return $this; @@ -349,9 +349,9 @@ public function usingOutputModalities(ModalityEnum ...$modalities): self */ public function asJsonResponse(?array $schema = null): self { - $this->usingOutputMime('application/json'); + $this->asOutputMimeType('application/json'); if ($schema !== null) { - $this->usingOutputSchema($schema); + $this->asOutputSchema($schema); } return $this; } diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index ef260799..475be23d 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -773,7 +773,7 @@ public function testUsingCandidateCount(): void public function testUsingOutputMime(): void { $builder = new PromptBuilder($this->registry); - $result = $builder->usingOutputMime('application/json'); + $result = $builder->asOutputMimeType('application/json'); $this->assertSame($builder, $result); @@ -801,7 +801,7 @@ public function testUsingOutputSchema(): void ]; $builder = new PromptBuilder($this->registry); - $result = $builder->usingOutputSchema($schema); + $result = $builder->asOutputSchema($schema); $this->assertSame($builder, $result); From 98dd0a2820b2add1b02a7e4a02cd14da0350106f Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 13:42:55 -0700 Subject: [PATCH 44/47] chore: corrects array types as lists --- src/Results/DTO/GenerativeAiResult.php | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/src/Results/DTO/GenerativeAiResult.php b/src/Results/DTO/GenerativeAiResult.php index ba7882cc..229099a2 100644 --- a/src/Results/DTO/GenerativeAiResult.php +++ b/src/Results/DTO/GenerativeAiResult.php @@ -268,7 +268,7 @@ public function toMessage(): Message * * @since n.e.x.t * - * @return string[] Array of text content. + * @return list Array of text content. */ public function toTexts(): array { @@ -291,7 +291,7 @@ public function toTexts(): array * * @since n.e.x.t * - * @return File[] Array of files. + * @return list Array of files. */ public function toFiles(): array { @@ -314,7 +314,7 @@ public function toFiles(): array * * @since n.e.x.t * - * @return File[] Array of image files. + * @return list Array of image files. */ public function toImageFiles(): array { @@ -329,7 +329,7 @@ public function toImageFiles(): array * * @since n.e.x.t * - * @return File[] Array of audio files. + * @return list Array of audio files. */ public function toAudioFiles(): array { @@ -344,7 +344,7 @@ public function toAudioFiles(): array * * @since n.e.x.t * - * @return File[] Array of video files. + * @return list Array of video files. */ public function toVideoFiles(): array { @@ -359,11 +359,11 @@ public function toVideoFiles(): array * * @since n.e.x.t * - * @return Message[] Array of messages. + * @return list Array of messages. */ public function toMessages(): array { - return array_map(fn(Candidate $candidate) => $candidate->getMessage(), $this->candidates); + return array_values(array_map(fn(Candidate $candidate) => $candidate->getMessage(), $this->candidates)); } /** From d5d782ea1d70ddaca270d4442fdf732940bf48c8 Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 13:57:09 -0700 Subject: [PATCH 45/47] refactor: adjusts isSupported to work more like generateResults --- src/Builders/PromptBuilder.php | 124 +++++---- tests/unit/Builders/PromptBuilderTest.php | 296 +--------------------- 2 files changed, 66 insertions(+), 354 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 3ec1e0b1..6b14d81a 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -361,16 +361,14 @@ public function asJsonResponse(?array $schema = null): self * * @since n.e.x.t * + * @param CapabilityEnum $capability The capability the model must support. * @return ModelRequirements The inferred requirements. */ - public function getModelRequirements(): ModelRequirements + private function getModelRequirements(CapabilityEnum $capability): ModelRequirements { - $capabilities = []; + $capabilities = [$capability]; $inputModalities = []; - // Always need text generation capability - $capabilities[] = CapabilityEnum::textGeneration(); - // Check if we have chat history (multiple messages) if (count($this->messages) > 1) { $capabilities[] = CapabilityEnum::chatHistory(); @@ -432,6 +430,49 @@ public function getModelRequirements(): ModelRequirements ); } + /** + * Infers the capability from configured output modalities. + * + * @since n.e.x.t + * + * @return CapabilityEnum The inferred capability. + * @throws RuntimeException If the output modality is not supported. + */ + private function inferCapabilityFromOutputModalities(): CapabilityEnum + { + // Get the configured output modalities + $outputModalities = $this->modelConfig->getOutputModalities(); + + // Default to text if no output modality is specified + if ($outputModalities === null || empty($outputModalities)) { + return CapabilityEnum::textGeneration(); + } + + // Multi-modal output (multiple modalities) defaults to text generation. This is temporary + // as a multi-modal interface will be implemented in the future. + if (count($outputModalities) > 1) { + return CapabilityEnum::textGeneration(); + } + + // Infer capability from single output modality + $outputModality = $outputModalities[0]; + + if ($outputModality->isText()) { + return CapabilityEnum::textGeneration(); + } elseif ($outputModality->isImage()) { + return CapabilityEnum::imageGeneration(); + } elseif ($outputModality->isAudio()) { + return CapabilityEnum::speechGeneration(); + } elseif ($outputModality->isVideo()) { + return CapabilityEnum::videoGeneration(); + } else { + // For unsupported modalities, provide a clear error message + throw new RuntimeException( + sprintf('Output modality "%s" is not yet supported.', $outputModality->value) + ); + } + } + /** * Checks if the current prompt is supported by the selected model. * @@ -442,31 +483,14 @@ public function getModelRequirements(): ModelRequirements */ private function isSupported(?CapabilityEnum $intendedCapability = null): bool { - // Build requirements with the intended capability if specified - $requirements = $this->getModelRequirements(); - - if ($intendedCapability !== null) { - $capabilities = $requirements->getRequiredCapabilities(); - - // Check if capability is already present - $hasCapability = false; - foreach ($capabilities as $capability) { - if ($capability->equals($intendedCapability)) { - $hasCapability = true; - break; - } - } - - // Add the capability if not already present - if (!$hasCapability) { - $capabilities[] = $intendedCapability; - $requirements = new ModelRequirements( - $capabilities, - $requirements->getRequiredOptions() - ); - } + // If no intended capability provided, infer from output modalities + if ($intendedCapability === null) { + $intendedCapability = $this->inferCapabilityFromOutputModalities(); } + // Build requirements with the specified capability + $requirements = $this->getModelRequirements($intendedCapability); + // If the model has been set, check if it meets the requirements if ($this->model !== null) { return $this->model->metadata()->meetsRequirements($requirements); @@ -477,6 +501,7 @@ private function isSupported(?CapabilityEnum $intendedCapability = null): bool $models = $this->registry->findModelsMetadataForSupport($requirements); return !empty($models); } catch (InvalidArgumentException $e) { + // No models support the requirements return false; } } @@ -583,43 +608,14 @@ public function isSupportedForEmbeddingGeneration(): bool public function generateResult(?CapabilityEnum $capability = null): GenerativeAiResult { $this->validateMessages(); - $model = $this->getConfiguredModel(); // If capability is not provided, infer it from output modalities if ($capability === null) { - // Get the configured output modalities - $outputModalities = $this->modelConfig->getOutputModalities(); - - // Default to text if no output modality is specified - if ($outputModalities === null || empty($outputModalities)) { - $outputModalities = [ModalityEnum::text()]; - } - - // Multi-modal output (multiple modalities) defaults to text generation. This is temporary - // as a multi-modal interface will be implemented in the future. - if (count($outputModalities) > 1) { - $capability = CapabilityEnum::textGeneration(); - } else { - // Infer capability from single output modality - $outputModality = $outputModalities[0]; - - if ($outputModality->isText()) { - $capability = CapabilityEnum::textGeneration(); - } elseif ($outputModality->isImage()) { - $capability = CapabilityEnum::imageGeneration(); - } elseif ($outputModality->isAudio()) { - $capability = CapabilityEnum::speechGeneration(); - } elseif ($outputModality->isVideo()) { - $capability = CapabilityEnum::videoGeneration(); - } else { - // For unsupported modalities, provide a clear error message - throw new RuntimeException( - sprintf('Output modality "%s" is not yet supported.', $outputModality->value) - ); - } - } + $capability = $this->inferCapabilityFromOutputModalities(); } + $model = $this->getConfiguredModel($capability); + // Route to the appropriate generation method based on capability if ($capability->isTextGeneration()) { if (!$model instanceof TextGenerationModelInterface) { @@ -917,15 +913,13 @@ protected function appendPartToMessages(MessagePart $part): void * * @since n.e.x.t * - * @param ModelRequirements|null $requirements Optional requirements to use. If not provided, will be inferred. + * @param CapabilityEnum $capability The capability the model will be using. * @return ModelInterface The model to use. * @throws InvalidArgumentException If no suitable model is found or set model doesn't meet requirements. */ - public function getConfiguredModel(?ModelRequirements $requirements = null): ModelInterface + private function getConfiguredModel(CapabilityEnum $capability): ModelInterface { - if ($requirements === null) { - $requirements = $this->getModelRequirements(); - } + $requirements = $this->getModelRequirements($capability); // If a model has been explicitly set, return it if ($this->model !== null) { diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index 475be23d..89b6f29e 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -16,12 +16,9 @@ use WordPress\AiClient\Messages\DTO\UserMessage; use WordPress\AiClient\Messages\Enums\MessageRoleEnum; use WordPress\AiClient\Messages\Enums\ModalityEnum; -use WordPress\AiClient\Providers\DTO\ProviderMetadata; -use WordPress\AiClient\Providers\DTO\ProviderModelsMetadata; use WordPress\AiClient\Providers\Models\Contracts\ModelInterface; use WordPress\AiClient\Providers\Models\DTO\ModelConfig; use WordPress\AiClient\Providers\Models\DTO\ModelMetadata; -use WordPress\AiClient\Providers\Models\Enums\OptionEnum; use WordPress\AiClient\Providers\Models\ImageGeneration\Contracts\ImageGenerationModelInterface; use WordPress\AiClient\Providers\Models\SpeechGeneration\Contracts\SpeechGenerationModelInterface; use WordPress\AiClient\Providers\Models\TextGeneration\Contracts\TextGenerationModelInterface; @@ -885,107 +882,6 @@ public function testAsJsonResponseWithSchema(): void $this->assertEquals($schema, $config->getOutputSchema()); } - /** - * Tests getModelRequirements with basic text prompt. - * - * @return void - */ - public function testGetModelRequirementsBasicText(): void - { - $builder = new PromptBuilder($this->registry, 'Simple text'); - $requirements = $builder->getModelRequirements(); - - $capabilities = $requirements->getRequiredCapabilities(); - $this->assertCount(1, $capabilities); - $this->assertTrue($capabilities[0]->isTextGeneration()); - - $options = $requirements->getRequiredOptions(); - // Should have input modalities with text - $inputModalitiesFound = false; - foreach ($options as $option) { - if ($option->getName()->equals(OptionEnum::inputModalities())) { - $inputModalitiesFound = true; - $modalities = $option->getValue(); - $this->assertCount(1, $modalities); - $this->assertTrue($modalities[0]->isText()); - } - } - $this->assertTrue($inputModalitiesFound); - } - - /** - * Tests getModelRequirements with chat history. - * - * @return void - */ - public function testGetModelRequirementsWithChatHistory(): void - { - $builder = new PromptBuilder($this->registry); - $builder->withHistory( - new UserMessage([new MessagePart('Hello')]), - new ModelMessage([new MessagePart('Hi there')]), - new UserMessage([new MessagePart('How are you?')]) - ); - - $requirements = $builder->getModelRequirements(); - $capabilities = $requirements->getRequiredCapabilities(); - - // Should have text generation and chat history capabilities - $this->assertCount(2, $capabilities); - $hasTextGeneration = false; - $hasChatHistory = false; - foreach ($capabilities as $capability) { - if ($capability->isTextGeneration()) { - $hasTextGeneration = true; - } - if ($capability->isChatHistory()) { - $hasChatHistory = true; - } - } - $this->assertTrue($hasTextGeneration); - $this->assertTrue($hasChatHistory); - } - - /** - * Tests getModelRequirements with multimodal input. - * - * @return void - */ - public function testGetModelRequirementsWithMultimodalInput(): void - { - $builder = new PromptBuilder($this->registry); - $builder->withText('Describe this image') - ->withFile('https://example.com/image.jpg', 'image/jpeg'); - - $requirements = $builder->getModelRequirements(); - $options = $requirements->getRequiredOptions(); - - // Find input modalities option - $inputModalities = null; - foreach ($options as $option) { - if ($option->getName()->equals(OptionEnum::inputModalities())) { - $inputModalities = $option->getValue(); - break; - } - } - - $this->assertNotNull($inputModalities); - $this->assertCount(2, $inputModalities); - - $hasText = false; - $hasImage = false; - foreach ($inputModalities as $modality) { - if ($modality->isText()) { - $hasText = true; - } - if ($modality->isImage()) { - $hasImage = true; - } - } - $this->assertTrue($hasText); - $this->assertTrue($hasImage); - } - /** * Tests validateMessages with empty messages throws exception. @@ -1587,7 +1483,7 @@ public function testGenerateTextThrowsExceptionWhenNoParts(): void $builder->usingModel($model); $this->expectException(RuntimeException::class); - $this->expectExceptionMessage('Generated message contains no parts'); + $this->expectExceptionMessage('No text content found in first candidate'); $builder->generateText(); } @@ -1616,7 +1512,7 @@ public function testGenerateTextThrowsExceptionWhenPartHasNoText(): void $builder->usingModel($model); $this->expectException(RuntimeException::class); - $this->expectExceptionMessage('Generated message part contains no text'); + $this->expectExceptionMessage('No text content found in first candidate'); $builder->generateText(); } @@ -1777,7 +1673,7 @@ public function testGenerateImageThrowsExceptionWhenNoFile(): void $builder->usingModel($model); $this->expectException(RuntimeException::class); - $this->expectExceptionMessage('Generated message part contains no image file'); + $this->expectExceptionMessage('No file content found in first candidate'); $builder->generateImage(); } @@ -1957,115 +1853,6 @@ public function testGenerateSpeeches(): void $this->assertSame($files[2], $speechFiles[2]); } - /** - * Tests getConfiguredModel with explicitly set model. - * - * @return void - */ - public function testGetConfiguredModelWithExplicitModel(): void - { - $metadata = $this->createMock(ModelMetadata::class); - $metadata->method('meetsRequirements')->willReturn(true); - - $model = $this->createMock(ModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->expects($this->once())->method('setConfig')->with($this->isInstanceOf(ModelConfig::class)); - - $builder = new PromptBuilder($this->registry, 'Test'); - $builder->usingModel($model); - - $reflection = new \ReflectionClass($builder); - $method = $reflection->getMethod('getConfiguredModel'); - $method->setAccessible(true); - - $configuredModel = $method->invoke($builder); - $this->assertSame($model, $configuredModel); - } - - /** - * Tests getConfiguredModel returns explicitly set model. - * - * @return void - */ - public function testGetConfiguredModelReturnsExplicitlySetModel(): void - { - $metadata = $this->createMock(ModelMetadata::class); - $metadata->method('getId')->willReturn('explicit-model'); - - $model = $this->createMock(ModelInterface::class); - $model->method('metadata')->willReturn($metadata); - $model->expects($this->once()) - ->method('setConfig') - ->with($this->isInstanceOf(ModelConfig::class)); - - $builder = new PromptBuilder($this->registry, 'Test'); - $builder->usingModel($model); - - $reflection = new \ReflectionClass($builder); - $method = $reflection->getMethod('getConfiguredModel'); - $method->setAccessible(true); - - $result = $method->invoke($builder); - $this->assertSame($model, $result); - } - - /** - * Tests getConfiguredModel finds model from registry. - * - * @return void - */ - public function testGetConfiguredModelFindsModelFromRegistry(): void - { - $modelMetadata = $this->createMock(ModelMetadata::class); - $modelMetadata->method('getId')->willReturn('found-model'); - - $providerMetadata = $this->createMock(ProviderMetadata::class); - $providerMetadata->method('getId')->willReturn('test-provider'); - - $providerModelsMetadata = $this->createMock(ProviderModelsMetadata::class); - $providerModelsMetadata->method('getProvider')->willReturn($providerMetadata); - $providerModelsMetadata->method('getModels')->willReturn([$modelMetadata]); - - $model = $this->createMock(ModelInterface::class); - - $this->registry->method('findModelsMetadataForSupport') - ->willReturn([$providerModelsMetadata]); - - $this->registry->method('getProviderModel') - ->with('test-provider', 'found-model', $this->isInstanceOf(ModelConfig::class)) - ->willReturn($model); - - $builder = new PromptBuilder($this->registry, 'Test'); - - $reflection = new \ReflectionClass($builder); - $method = $reflection->getMethod('getConfiguredModel'); - $method->setAccessible(true); - - $configuredModel = $method->invoke($builder); - $this->assertSame($model, $configuredModel); - } - - /** - * Tests getConfiguredModel throws exception when no models found. - * - * @return void - */ - public function testGetConfiguredModelThrowsExceptionWhenNoModelsFound(): void - { - $this->registry->method('findModelsMetadataForSupport')->willReturn([]); - - $builder = new PromptBuilder($this->registry, 'Test'); - - $reflection = new \ReflectionClass($builder); - $method = $reflection->getMethod('getConfiguredModel'); - $method->setAccessible(true); - - $this->expectException(InvalidArgumentException::class); - $this->expectExceptionMessage('No models found that support the required capabilities'); - - $method->invoke($builder); - } - /** * Tests appendPartToMessages creates new user message when empty. * @@ -2359,75 +2146,6 @@ public function testParseMessageWithInvalidArrayItemThrowsException(): void new PromptBuilder($this->registry, ['valid string', 123, 'another string']); } - /** - * Tests getModelRequirements with all file types. - * - * @return void - */ - public function testGetModelRequirementsWithAllFileTypes(): void - { - $builder = new PromptBuilder($this->registry); - $builder->withText('Analyze:') - ->withFile('https://example.com/img.jpg', 'image/jpeg') - ->withFile('https://example.com/audio.mp3', 'audio/mp3') - ->withFile('https://example.com/video.mp4', 'video/mp4') - ->withFile('https://example.com/doc.pdf', 'application/pdf'); - - $requirements = $builder->getModelRequirements(); - $options = $requirements->getRequiredOptions(); - - // Find input modalities - $inputModalities = null; - foreach ($options as $option) { - if ($option->getName()->equals(OptionEnum::inputModalities())) { - $inputModalities = $option->getValue(); - break; - } - } - - $this->assertNotNull($inputModalities); - - // Check all modality types are present - $modalityTypes = []; - foreach ($inputModalities as $modality) { - $modalityTypes[] = $modality->value; - } - - $this->assertContains('text', $modalityTypes); - $this->assertContains('image', $modalityTypes); - $this->assertContains('audio', $modalityTypes); - $this->assertContains('video', $modalityTypes); - $this->assertContains('document', $modalityTypes); - } - - /** - * Tests getModelRequirements includes config options. - * - * @return void - */ - public function testGetModelRequirementsIncludesConfigOptions(): void - { - $builder = new PromptBuilder($this->registry, 'Test'); - $builder->usingMaxTokens(1000) - ->usingTemperature(0.7) - ->usingOutputModalities(ModalityEnum::text(), ModalityEnum::image()) - ->asJsonResponse(['type' => 'object']); - - $requirements = $builder->getModelRequirements(); - $options = $requirements->getRequiredOptions(); - - // Check that config options are included - $optionEnums = array_map(function ($option) { - return $option->getName(); - }, $options); - - $this->assertContains(OptionEnum::maxTokens(), $optionEnums); - $this->assertContains(OptionEnum::temperature(), $optionEnums); - $this->assertContains(OptionEnum::outputModalities(), $optionEnums); - $this->assertContains(OptionEnum::outputMimeType(), $optionEnums); - $this->assertContains(OptionEnum::outputSchema(), $optionEnums); - } - /** * Tests last message must have parts validation. * @@ -2585,9 +2303,9 @@ public function testStreamingGenerationMethods(): void */ public function testGenerateTextWithNoCandidatesThrowsException(): void { - // Create a mock result that returns empty candidates + // Create a mock result that throws when toText is called $result = $this->createMock(GenerativeAiResult::class); - $result->method('getCandidates')->willReturn([]); + $result->method('toText')->willThrowException(new RuntimeException('No text content found in first candidate')); $metadata = $this->createMock(ModelMetadata::class); $metadata->method('getId')->willReturn('test-model'); @@ -2599,7 +2317,7 @@ public function testGenerateTextWithNoCandidatesThrowsException(): void $builder->usingModel($model); $this->expectException(RuntimeException::class); - $this->expectExceptionMessage('No candidates were generated'); + $this->expectExceptionMessage('No text content found in first candidate'); $builder->generateText(); } @@ -2629,7 +2347,7 @@ public function testGenerateTextWithNonStringPartThrowsException(): void $builder->usingModel($model); $this->expectException(RuntimeException::class); - $this->expectExceptionMessage('Generated message part contains no text'); + $this->expectExceptionMessage('No text content found in first candidate'); $builder->generateText(); } From b8dcc954f4ee8fed36710a4af2c80f6c30b0422d Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 14:05:09 -0700 Subject: [PATCH 46/47] chore: provides Prompt type --- src/Builders/PromptBuilder.php | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 6b14d81a..1891453f 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -37,6 +37,8 @@ * * @phpstan-import-type MessageArrayShape from Message * @phpstan-import-type MessagePartArrayShape from MessagePart + * + * @phpstan-type Prompt string|MessagePart|Message|MessageArrayShape|list|list|null */ class PromptBuilder { @@ -67,7 +69,7 @@ class PromptBuilder * @since n.e.x.t * * @param ProviderRegistry $registry The provider registry for finding suitable models. - * @param string|MessagePart|Message|MessageArrayShape|list|list|null $prompt + * @param Prompt $prompt * Optional initial prompt content. */ // phpcs:enable Generic.Files.LineLength.TooLong From 32e1fb89de98e9b3b812f92ca5f098baa146c19d Mon Sep 17 00:00:00 2001 From: Jason Adams Date: Tue, 26 Aug 2025 14:05:53 -0700 Subject: [PATCH 47/47] refactor: renames output modalities method to use as- prefix --- src/Builders/PromptBuilder.php | 2 +- tests/unit/Builders/PromptBuilderTest.php | 12 ++++++------ 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/src/Builders/PromptBuilder.php b/src/Builders/PromptBuilder.php index 1891453f..83721788 100644 --- a/src/Builders/PromptBuilder.php +++ b/src/Builders/PromptBuilder.php @@ -335,7 +335,7 @@ public function asOutputSchema(array $schema): self * @param ModalityEnum ...$modalities The output modalities. * @return self */ - public function usingOutputModalities(ModalityEnum ...$modalities): self + public function asOutputModalities(ModalityEnum ...$modalities): self { $this->modelConfig->setOutputModalities($modalities); return $this; diff --git a/tests/unit/Builders/PromptBuilderTest.php b/tests/unit/Builders/PromptBuilderTest.php index 89b6f29e..0740c2ac 100644 --- a/tests/unit/Builders/PromptBuilderTest.php +++ b/tests/unit/Builders/PromptBuilderTest.php @@ -819,7 +819,7 @@ public function testUsingOutputSchema(): void public function testUsingOutputModalities(): void { $builder = new PromptBuilder($this->registry); - $result = $builder->usingOutputModalities( + $result = $builder->asOutputModalities( ModalityEnum::text(), ModalityEnum::image() ); @@ -1085,7 +1085,7 @@ public function testGenerateResultWithImageModality(): void $builder = new PromptBuilder($this->registry, 'Generate an image'); $builder->usingModel($model); - $builder->usingOutputModalities(ModalityEnum::image()); + $builder->asOutputModalities(ModalityEnum::image()); $actualResult = $builder->generateResult(); $this->assertSame($result, $actualResult); @@ -1115,7 +1115,7 @@ public function testGenerateResultWithAudioModality(): void $builder = new PromptBuilder($this->registry, 'Generate speech'); $builder->usingModel($model); - $builder->usingOutputModalities(ModalityEnum::audio()); + $builder->asOutputModalities(ModalityEnum::audio()); $actualResult = $builder->generateResult(); $this->assertSame($result, $actualResult); @@ -1142,7 +1142,7 @@ public function testGenerateResultWithMultimodalOutput(): void $builder = new PromptBuilder($this->registry, 'Generate multimodal'); $builder->usingModel($model); - $builder->usingOutputModalities(ModalityEnum::text(), ModalityEnum::image()); + $builder->asOutputModalities(ModalityEnum::text(), ModalityEnum::image()); $actualResult = $builder->generateResult(); $this->assertSame($result, $actualResult); @@ -1188,7 +1188,7 @@ public function testGenerateResultThrowsExceptionForUnsupportedOutputModality(): $builder = new PromptBuilder($this->registry, 'Test prompt'); $builder->usingModel($model); - $builder->usingOutputModalities(ModalityEnum::video()); + $builder->asOutputModalities(ModalityEnum::video()); $this->expectException(RuntimeException::class); $this->expectExceptionMessage('Output modality "video" is not yet supported'); @@ -2052,7 +2052,7 @@ public function testIncludeOutputModalityPreservesExisting(): void $builder->usingModel($model); // Set initial modality - $builder->usingOutputModalities(ModalityEnum::audio()); + $builder->asOutputModalities(ModalityEnum::audio()); // Generate text should add text modality, not replace audio $builder->generateTextResult();