|
11 | 11 | use Felix_Arntz\AI_Services\Services\API\Enums\AI_Capability;
|
12 | 12 | use Felix_Arntz\AI_Services\Services\API\Helpers;
|
13 | 13 | use Felix_Arntz\AI_Services\Services\API\Types\Content;
|
| 14 | +use Felix_Arntz\AI_Services\Services\API\Types\Image_Generation_Config; |
| 15 | +use Felix_Arntz\AI_Services\Services\API\Types\Parts\File_Data_Part; |
| 16 | +use Felix_Arntz\AI_Services\Services\API\Types\Parts\Inline_Data_Part; |
14 | 17 | use Felix_Arntz\AI_Services\Services\API\Types\Parts\Text_Part;
|
15 | 18 | use Felix_Arntz\AI_Services\Services\API\Types\Text_Generation_Config;
|
16 | 19 | use Felix_Arntz\AI_Services\Services\API\Types\Tools;
|
17 | 20 | use Felix_Arntz\AI_Services\Services\Contracts\Generative_AI_Model;
|
18 | 21 | use Felix_Arntz\AI_Services\Services\Contracts\Generative_AI_Service;
|
| 22 | +use Felix_Arntz\AI_Services\Services\Contracts\With_Image_Generation; |
19 | 23 | use Felix_Arntz\AI_Services\Services\Contracts\With_Text_Generation;
|
20 | 24 | use Felix_Arntz\AI_Services\Services\Entities\Service_Entity;
|
21 | 25 | use Felix_Arntz\AI_Services\Services\Entities\Service_Entity_Query;
|
@@ -438,7 +442,7 @@ public function list_models( array $args, array $assoc_args ): void {
|
438 | 442 | public function generate_text( array $args, array $assoc_args ): void {
|
439 | 443 | list( $service_slug, $model_slug, $prompt ) = $this->parse_generate_positional_args( $args );
|
440 | 444 |
|
441 |
| - $model_params = $this->get_model_params( $model_slug, $assoc_args ); |
| 445 | + $model_params = $this->get_text_model_params( $model_slug, $assoc_args ); |
442 | 446 |
|
443 | 447 | $attachment_id = isset( $assoc_args['attachment-id'] ) ? (int) $assoc_args['attachment-id'] : 0;
|
444 | 448 |
|
@@ -471,6 +475,71 @@ public function generate_text( array $args, array $assoc_args ): void {
|
471 | 475 | }
|
472 | 476 | }
|
473 | 477 |
|
| 478 | + /** |
| 479 | + * Generates an image using a generative model from an available AI service. |
| 480 | + * |
| 481 | + * Only authorized users with sufficient permissions can use this command. |
| 482 | + * Provide the `--user` argument to specify the user. |
| 483 | + * |
| 484 | + * ## OPTIONS |
| 485 | + * |
| 486 | + * [<service>] |
| 487 | + * : The service to use. Can be omitted to use any available service. |
| 488 | + * |
| 489 | + * [<model>] |
| 490 | + * : The model to use from the service. Can be omitted to use any suitable model from the service. |
| 491 | + * |
| 492 | + * <prompt> |
| 493 | + * : The text prompt to generate an image. |
| 494 | + * |
| 495 | + * [--feature=<feature>] |
| 496 | + * : Required. Unique identifier of the feature that the model will be used for. |
| 497 | + * |
| 498 | + * [--system-instruction=<system-instruction>] |
| 499 | + * : System instruction for the model. |
| 500 | + * |
| 501 | + * [--<field>=<value>] |
| 502 | + * : Model generation config arguments. For example, `--size=2048x2048`. |
| 503 | + * |
| 504 | + * ## EXAMPLES |
| 505 | + * |
| 506 | + * wp ai-services generate-image google "Photorealistic image of a French bulldog wearing sunglasses in the forest." --feature=my-cli-test --user=admin > img_output.txt |
| 507 | + * wp ai-services generate-image openai dall-e-3 "Photorealistic image of a French bulldog wearing sunglasses in the forest." --feature=cli-example --user=admin > img_output.txt |
| 508 | + * wp ai-services generate-image openai "Photorealistic image of a French bulldog wearing sunglasses in the forest." --feature=cli-example --response-type=file_data --user=admin |
| 509 | + * |
| 510 | + * @subcommand generate-image |
| 511 | + * |
| 512 | + * @since n.e.x.t |
| 513 | + * |
| 514 | + * @param mixed[] $args List of the positional arguments. |
| 515 | + * @param array<string, mixed> $assoc_args Map of the associative arguments and their values. |
| 516 | + */ |
| 517 | + public function generate_image( array $args, array $assoc_args ): void { |
| 518 | + list( $service_slug, $model_slug, $prompt ) = $this->parse_generate_positional_args( $args ); |
| 519 | + |
| 520 | + $model_params = $this->get_image_model_params( $model_slug, $assoc_args ); |
| 521 | + |
| 522 | + if ( $service_slug ) { |
| 523 | + $service_args = $service_slug; |
| 524 | + } elseif ( isset( $model_params['capabilities'] ) ) { |
| 525 | + $service_args = array( 'capabilities' => $model_params['capabilities'] ); |
| 526 | + } else { |
| 527 | + $service_args = array(); |
| 528 | + } |
| 529 | + |
| 530 | + try { |
| 531 | + $service = $this->services_api->get_available_service( $service_args ); |
| 532 | + } catch ( InvalidArgumentException $e ) { |
| 533 | + WP_CLI::error( html_entity_decode( $e->getMessage() ) ); |
| 534 | + } |
| 535 | + |
| 536 | + $model = $this->get_model( $service, $model_params ); |
| 537 | + |
| 538 | + $content = Helpers::text_to_content( $prompt ); |
| 539 | + |
| 540 | + $this->generate_image_using_model( $model, $content ); |
| 541 | + } |
| 542 | + |
474 | 543 | /**
|
475 | 544 | * Parses the positional arguments for a generate command.
|
476 | 545 | *
|
@@ -595,15 +664,15 @@ private function get_model_sort_callback( string $orderby, string $order ): call
|
595 | 664 | }
|
596 | 665 |
|
597 | 666 | /**
|
598 |
| - * Gets the model parameters for the given model slug and associative arguments. |
| 667 | + * Gets the text generation model parameters for the given model slug and associative arguments. |
599 | 668 | *
|
600 | 669 | * @since n.e.x.t
|
601 | 670 | *
|
602 | 671 | * @param string|null $model_slug The model slug.
|
603 | 672 | * @param array<string, mixed> $assoc_args Map of the associative arguments and their values.
|
604 | 673 | * @return array<string, mixed> The model parameters, to retrieve a model.
|
605 | 674 | */
|
606 |
| - private function get_model_params( ?string $model_slug, array $assoc_args ): array { |
| 675 | + private function get_text_model_params( ?string $model_slug, array $assoc_args ): array { |
607 | 676 | // Assume any unknown arguments are generation configuration arguments.
|
608 | 677 | $generation_config_args = $assoc_args;
|
609 | 678 | $assoc_args = $this->parse_assoc_args(
|
@@ -644,6 +713,41 @@ private function get_model_params( ?string $model_slug, array $assoc_args ): arr
|
644 | 713 | );
|
645 | 714 | }
|
646 | 715 |
|
| 716 | + /** |
| 717 | + * Gets the image generation model parameters for the given model slug and associative arguments. |
| 718 | + * |
| 719 | + * @since n.e.x.t |
| 720 | + * |
| 721 | + * @param string|null $model_slug The model slug. |
| 722 | + * @param array<string, mixed> $assoc_args Map of the associative arguments and their values. |
| 723 | + * @return array<string, mixed> The model parameters, to retrieve a model. |
| 724 | + */ |
| 725 | + private function get_image_model_params( ?string $model_slug, array $assoc_args ): array { |
| 726 | + // Assume any unknown arguments are generation configuration arguments. |
| 727 | + $generation_config_args = $assoc_args; |
| 728 | + $assoc_args = $this->parse_assoc_args( |
| 729 | + $assoc_args, |
| 730 | + array( |
| 731 | + 'feature' => '', |
| 732 | + 'system-instruction' => '', |
| 733 | + ), |
| 734 | + $this->formatter_args |
| 735 | + ); |
| 736 | + $generation_config_args = $this->sanitize_generation_config_args( |
| 737 | + array_diff_key( $generation_config_args, $assoc_args ) |
| 738 | + ); |
| 739 | + |
| 740 | + $capabilities = array( AI_Capability::IMAGE_GENERATION ); |
| 741 | + |
| 742 | + return array( |
| 743 | + 'feature' => $assoc_args['feature'] ? $assoc_args['feature'] : null, |
| 744 | + 'model' => $model_slug, |
| 745 | + 'capabilities' => $capabilities, |
| 746 | + 'generationConfig' => Image_Generation_Config::from_array( $generation_config_args ), |
| 747 | + 'systemInstruction' => $assoc_args['system-instruction'] ? $assoc_args['system-instruction'] : null, |
| 748 | + ); |
| 749 | + } |
| 750 | + |
647 | 751 | /**
|
648 | 752 | * Sanitizes the generation configuration arguments.
|
649 | 753 | *
|
@@ -719,7 +823,7 @@ private function generate_text_using_model( Generative_AI_Model $model, Content
|
719 | 823 | } catch ( Generative_AI_Exception $e ) {
|
720 | 824 | WP_CLI::error(
|
721 | 825 | sprintf(
|
722 |
| - 'Generating content with model %1$s failed: %2$s', |
| 826 | + 'Generating text with model %1$s failed: %2$s', |
723 | 827 | $model->get_model_slug(),
|
724 | 828 | html_entity_decode( $e->getMessage() )
|
725 | 829 | )
|
@@ -770,7 +874,7 @@ private function stream_generate_text_using_model( Generative_AI_Model $model, C
|
770 | 874 | } catch ( Generative_AI_Exception $e ) {
|
771 | 875 | WP_CLI::error(
|
772 | 876 | sprintf(
|
773 |
| - 'Generating content with model %1$s failed: %2$s', |
| 877 | + 'Generating text with model %1$s failed: %2$s', |
774 | 878 | $model->get_model_slug(),
|
775 | 879 | html_entity_decode( $e->getMessage() )
|
776 | 880 | )
|
@@ -801,6 +905,55 @@ private function stream_generate_text_using_model( Generative_AI_Model $model, C
|
801 | 905 | }
|
802 | 906 | }
|
803 | 907 |
|
| 908 | + /** |
| 909 | + * Generates an image using the given generative model and prints it. |
| 910 | + * |
| 911 | + * @since n.e.x.t |
| 912 | + * |
| 913 | + * @param Generative_AI_Model $model The model to use. |
| 914 | + * @param Content $content Prompt for the content to generate. |
| 915 | + */ |
| 916 | + private function generate_image_using_model( Generative_AI_Model $model, Content $content ): void { |
| 917 | + if ( ! $model instanceof With_Image_Generation ) { |
| 918 | + WP_CLI::error( 'The model does not support image generation.' ); |
| 919 | + } |
| 920 | + |
| 921 | + try { |
| 922 | + $candidates = $model->generate_image( $content ); |
| 923 | + } catch ( Generative_AI_Exception $e ) { |
| 924 | + WP_CLI::error( |
| 925 | + sprintf( |
| 926 | + 'Generating image with model %1$s failed: %2$s', |
| 927 | + $model->get_model_slug(), |
| 928 | + html_entity_decode( $e->getMessage() ) |
| 929 | + ) |
| 930 | + ); |
| 931 | + } catch ( InvalidArgumentException $e ) { |
| 932 | + WP_CLI::error( |
| 933 | + sprintf( |
| 934 | + 'Invalid content provided to model %1$s: %2$s', |
| 935 | + $model->get_model_slug(), |
| 936 | + html_entity_decode( $e->getMessage() ) |
| 937 | + ) |
| 938 | + ); |
| 939 | + } |
| 940 | + |
| 941 | + $contents = Helpers::get_candidate_contents( $candidates ); |
| 942 | + foreach ( $contents as $content ) { |
| 943 | + // If the content is purely an image, print it directly, otherwise print the parts as structured JSON. |
| 944 | + $parts = $content->get_parts(); |
| 945 | + if ( count( $parts ) === 1 && $parts->get( 0 ) instanceof Inline_Data_Part ) { |
| 946 | + WP_CLI::print_value( $parts->get( 0 )->get_base64_data(), array( 'format' => 'table' ) ); |
| 947 | + continue; |
| 948 | + } |
| 949 | + if ( count( $parts ) === 1 && $parts->get( 0 ) instanceof File_Data_Part ) { |
| 950 | + WP_CLI::print_value( $parts->get( 0 )->get_file_uri(), array( 'format' => 'table' ) ); |
| 951 | + continue; |
| 952 | + } |
| 953 | + WP_CLI::print_value( $parts->to_array(), array( 'format' => 'json' ) ); |
| 954 | + } |
| 955 | + } |
| 956 | + |
804 | 957 | /**
|
805 | 958 | * Parses and validates the associative arguments.
|
806 | 959 | *
|
|
0 commit comments