11
11
use Felix_Arntz \AI_Services \Services \API \Enums \Content_Role ;
12
12
use Felix_Arntz \AI_Services \Services \API \Types \Candidates ;
13
13
use Felix_Arntz \AI_Services \Services \API \Types \Content ;
14
+ use Felix_Arntz \AI_Services \Services \API \Types \Parts ;
14
15
use Felix_Arntz \AI_Services \Services \API \Types \Parts \Text_Part ;
15
16
use Felix_Arntz \AI_Services \Services \Util \Formatter ;
16
17
use Generator ;
18
+ use WP_Post ;
17
19
18
20
/**
19
21
* Class providing static helper methods as part of the public API.
@@ -35,6 +37,56 @@ public static function text_to_content( string $text, string $role = Content_Rol
35
37
return Formatter::format_content ( $ text , $ role );
36
38
}
37
39
40
+ /**
41
+ * Converts a text string and attachment to a multimodal Content instance.
42
+ *
43
+ * The text will be included as a prompt as the first part of the content, and the attachment (e.g. an image or
44
+ * audio file) will be included as the second part.
45
+ *
46
+ * @since n.e.x.t
47
+ *
48
+ * @param string $text The text.
49
+ * @param int|WP_Post $attachment The attachment ID or object.
50
+ * @param string $role Optional. The role to use for the content. Default 'user'.
51
+ * @return Content The content instance.
52
+ */
53
+ public static function text_and_attachment_to_content ( string $ text , $ attachment , string $ role = Content_Role::USER ): Content {
54
+ if ( $ attachment instanceof WP_Post ) {
55
+ $ attachment_id = (int ) $ attachment ->ID ;
56
+ } else {
57
+ $ attachment_id = (int ) $ attachment ;
58
+ $ attachment = get_post ( $ attachment_id );
59
+ }
60
+
61
+ $ file = get_attached_file ( $ attachment_id );
62
+ $ large_size = image_get_intermediate_size ( $ attachment_id , 'large ' );
63
+ if ( $ large_size && isset ( $ large_size ['path ' ] ) ) {
64
+ // To get the absolute path to a sub-size file, we need to prepend the uploads dir.
65
+ if ( str_starts_with ( $ large_size ['path ' ], '/ ' ) ) {
66
+ $ file = $ large_size ['path ' ];
67
+ } else {
68
+ $ uploads = wp_get_upload_dir ();
69
+ if ( false === $ uploads ['error ' ] ) {
70
+ $ file = "{$ uploads ['basedir ' ]}/ {$ large_size ['path ' ]}" ;
71
+ }
72
+ }
73
+ }
74
+
75
+ $ mime_type = wp_check_filetype ( $ file );
76
+ if ( isset ( $ mime_type ['type ' ] ) ) {
77
+ $ mime_type = $ mime_type ['type ' ];
78
+ } else {
79
+ // Fallback that should never be needed.
80
+ $ mime_type = $ attachment ->post_mime_type ;
81
+ }
82
+
83
+ $ parts = new Parts ();
84
+ $ parts ->add_text_part ( $ text );
85
+ $ parts ->add_inline_data_part ( $ mime_type , self ::base64_encode_file ( $ file , $ mime_type ) );
86
+
87
+ return Formatter::format_content ( $ parts , $ role );
88
+ }
89
+
38
90
/**
39
91
* Converts a Content instance to a text string.
40
92
*
@@ -149,4 +201,29 @@ public static function get_candidate_contents( Candidates $candidates ): array {
149
201
public static function process_candidates_stream ( Generator $ generator ): Candidates_Stream_Processor { // phpcs:ignore Squiz.Commenting.FunctionComment.IncorrectTypeHint
150
202
return new Candidates_Stream_Processor ( $ generator );
151
203
}
204
+
205
+ /**
206
+ * Base64-encodes a file and returns its data URL.
207
+ *
208
+ * @since n.e.x.t
209
+ *
210
+ * @param string $file Absolute path to the file, or its URL.
211
+ * @param string $mime_type Optional. The MIME type of the file. If provided, the base64-encoded data URL will
212
+ * be prefixed with `data:{mime_type};base64,`. Default empty string.
213
+ * @return string The base64-encoded file data URL, or empty string on failure.
214
+ */
215
+ public static function base64_encode_file ( string $ file , string $ mime_type = '' ): string {
216
+ // phpcs:ignore WordPress.WP.AlternativeFunctions.file_get_contents_file_get_contents
217
+ $ binary_data = file_get_contents ( $ file );
218
+ if ( ! $ binary_data ) {
219
+ return '' ;
220
+ }
221
+
222
+ // phpcs:ignore WordPress.PHP.DiscouragedPHPFunctions.obfuscation_base64_encode
223
+ $ base64 = base64_encode ( $ binary_data );
224
+ if ( '' !== $ mime_type ) {
225
+ $ base64 = "data: $ mime_type;base64, $ base64 " ;
226
+ }
227
+ return $ base64 ;
228
+ }
152
229
}
0 commit comments