PluginProbe ʕ •ᴥ•ʔ
AI Engine – The Chatbot, AI Framework & MCP for WordPress / 3.3.2
AI Engine – The Chatbot, AI Framework & MCP for WordPress v3.3.2
3.5.8 3.5.7 3.5.6 3.5.5 3.5.4 3.5.3 3.5.2 3.5.1 3.5.0 3.4.9 3.4.8 3.4.7 0.2.1 1.6.91 0.2.2 1.6.92 0.2.3 1.6.93 0.2.4 1.6.94 0.2.5 1.6.95 0.2.6 1.6.96 0.2.7 1.6.97 0.2.8 1.6.98 0.2.9 1.6.99 0.3.0 1.7.0 0.3.1 1.7.1 0.3.2 1.7.2 0.3.3 1.7.3 0.3.4 1.7.4 0.3.5 1.7.5 0.3.6 1.7.6 0.4.0 1.7.7 0.4.1 1.7.8 0.4.2 1.7.9 0.4.3 1.8.0 0.4.4 1.8.1 0.4.5 1.8.2 0.4.6 1.8.3 0.4.7 1.8.4 0.4.8 1.8.5 0.4.9 1.8.6 0.5.0 1.8.7 0.5.1 1.8.8 0.5.2 1.8.9 0.5.3 1.9.0 0.5.4 1.9.1 0.5.5 1.9.2 0.5.6 1.9.3 0.5.7 1.9.4 0.5.8 1.9.5 0.5.9 1.9.6 0.6.0 1.9.7 0.6.1 1.9.8 0.6.2 1.9.81 0.6.3 1.9.82 0.6.4 1.9.83 0.6.5 1.9.84 0.6.6 1.9.85 0.6.7 1.9.86 0.6.8 1.9.87 0.6.9 1.9.88 0.7.0 1.9.89 0.7.1 1.9.90 0.7.2 1.9.91 0.7.3 1.9.92 0.7.4 1.9.93 0.7.5 1.9.94 0.7.6 1.9.95 0.7.7 1.9.96 0.7.8 1.9.97 0.7.9 1.9.98 0.8.0 1.9.99 0.8.1 2.0.0 0.8.2 2.0.1 0.8.3 2.0.2 0.8.4 2.0.3 0.8.5 2.0.4 0.8.6 2.0.5 0.8.7 2.0.6 0.8.8 2.0.7 0.8.9 2.0.8 0.9.0 2.0.9 0.9.2 2.1.0 0.9.3 2.1.1 0.9.4 2.1.2 0.9.5 2.1.3 0.9.6 2.1.4 0.9.7 2.1.5 0.9.8 2.1.6 0.9.81 2.1.7 0.9.82 2.1.8 0.9.83 2.1.9 0.9.84 2.2.0 0.9.85 2.2.1 0.9.86 2.2.2 0.9.87 2.2.3 0.9.88 2.2.4 0.9.89 2.2.5 0.9.9 2.2.51 0.9.91 2.2.52 0.9.92 2.2.53 0.9.93 2.2.54 0.9.94 2.2.56 0.9.95 2.2.57 0.9.96 2.2.6 0.9.97 2.2.60 0.9.98 2.2.61 0.9.99 2.2.62 1.0.0 2.2.63 1.0.01 2.2.70 1.0.1 2.2.80 1.0.2 2.2.81 1.0.3 2.2.90 1.0.4 2.2.91 1.0.5 2.2.92 1.0.6 2.2.93 1.0.7 2.2.94 1.0.8 2.2.95 1.0.9 2.3.0 1.1.0 2.3.1 1.1.1 2.3.2 1.1.2 2.3.3 1.1.3 2.3.4 1.1.4 2.3.5 1.1.5 2.3.6 1.1.6 2.3.7 1.1.7 2.3.8 1.1.8 2.3.9 1.1.9 2.4.0 1.2.0 2.4.1 1.2.1 2.4.2 1.2.2 2.4.3 1.2.21 2.4.4 1.2.3 2.4.5 1.2.30 2.4.6 1.3.0 2.4.7 1.3.1 2.4.8 1.3.2 2.4.9 1.3.3 2.5.0 1.3.31 2.5.1 1.3.32 2.5.2 1.3.33 2.5.3 1.3.34 2.5.4 1.3.35 2.5.5 1.3.36 2.5.6 1.3.37 2.5.7 1.3.38 2.5.8 1.3.39 2.5.9 1.3.40 2.6.0 1.3.41 2.6.1 1.3.42 2.6.2 1.3.43 2.6.3 1.3.44 2.6.5 1.3.45 2.6.6 1.3.46 2.6.7 1.3.47 2.6.8 1.3.48 2.6.9 1.3.49 2.7.0 1.3.50 2.7.1 1.3.51 2.7.2 1.3.52 2.7.3 1.3.53 2.7.4 1.3.54 2.7.5 1.3.56 2.7.6 1.3.57 2.7.7 1.3.58 2.7.8 1.3.59 2.7.9 1.3.60 2.8.0 1.3.61 2.8.1 1.3.62 2.8.2 1.3.63 2.8.3 1.3.64 2.8.4 1.3.65 2.8.5 1.3.66 2.8.6 1.3.67 2.8.7 1.3.68 2.8.8 1.3.69 2.8.9 1.3.70 2.9.0 1.3.71 2.9.1 1.3.72 2.9.2 1.3.73 2.9.3 1.3.74 2.9.4 1.3.75 2.9.5 1.3.76 2.9.6 1.3.77 2.9.7 1.3.78 2.9.8 1.3.79 2.9.9 1.3.80 3.0.0 1.3.81 3.0.1 1.3.82 3.0.2 1.3.83 3.0.3 1.3.84 3.0.4 1.3.85 3.0.5 1.3.86 3.0.6 1.3.87 3.0.7 1.3.88 3.0.8 1.3.89 3.0.9 1.3.90 3.1.0 1.3.91 3.1.1 1.3.92 3.1.2 1.3.93 3.1.3 1.3.94 3.1.4 1.3.95 3.1.5 1.3.96 3.1.6 1.3.97 3.1.7 1.3.98 3.1.8 1.3.99 3.1.9 1.4.0 3.2.0 1.4.1 3.2.1 1.4.2 3.2.2 1.4.3 3.2.3 1.4.4 3.2.4 1.4.5 3.2.5 1.4.6 3.2.6 1.4.7 3.2.7 1.4.8 3.2.8 1.4.9 3.2.9 1.5.0 3.3.0 1.5.1 3.3.1 1.5.2 3.3.2 1.5.3 3.3.3 1.5.4 3.3.4 1.5.5 3.3.5 1.5.6 3.3.6 1.5.7 3.3.7 1.5.8 3.3.8 1.5.9 3.3.9 1.6.0 3.4.0 1.6.1 3.4.1 1.6.2 3.4.2 1.6.3 3.4.3 1.6.5 3.4.4 1.6.51 3.4.5 1.6.52 3.4.6 1.6.53 1.6.54 1.6.55 1.6.56 1.6.57 1.6.58 1.6.59 1.6.60 1.6.61 1.6.62 1.6.63 1.6.64 1.6.65 1.6.66 1.6.67 1.6.68 trunk 1.6.69 0.0.1 1.6.70 0.0.2 1.6.71 0.0.3 1.6.72 0.0.4 1.6.73 0.0.5 1.6.74 0.0.6 1.6.75 0.0.7 1.6.76 0.0.8 1.6.77 0.0.9 1.6.78 0.1.0 1.6.79 0.1.1 1.6.81 0.1.2 1.6.82 0.1.3 1.6.83 0.1.4 1.6.84 0.1.5 1.6.85 0.1.6 1.6.86 0.1.7 1.6.87 0.1.8 1.6.88 0.1.9 1.6.89 0.2.0 1.6.90
ai-engine / classes / engines / anthropic.php
ai-engine / classes / engines Last commit date
anthropic.php 6 months ago chatml.php 6 months ago core.php 7 months ago factory.php 8 months ago google.php 6 months ago mistral.php 6 months ago open-router.php 6 months ago openai.php 6 months ago perplexity.php 6 months ago replicate.php 6 months ago
anthropic.php
1262 lines
1 <?php
2
3 class Meow_MWAI_Engines_Anthropic extends Meow_MWAI_Engines_ChatML {
4 // Streaming
5 protected $streamInTokens = null;
6 protected $streamOutTokens = null;
7 protected $streamBlocks;
8 protected $streamIsThinking = false;
9 protected $mcpServerNames = [];
10 protected $mcpTools = []; // Track MCP tools by ID
11 protected $mcpToolCount = 0;
12 protected $textStarted = false; // Track if text streaming has started
13 protected $requestSentEmitted = false; // Track if request sent event was emitted
14
15 public function __construct( $core, $env ) {
16 parent::__construct( $core, $env );
17 }
18
19 /**
20 * Prepare query by uploading files to Anthropic Files API.
21 *
22 * This method is called BEFORE streaming hooks are attached and BEFORE build_body().
23 * It uploads PDF files to Anthropic's Files API and converts them from 'refId' type
24 * to 'provider_file_id' type, which build_body() will then use to construct the API request.
25 *
26 * Flow:
27 * 1. prepare_query() uploads files to Anthropic Files API → gets file_id (e.g., file_abc123)
28 * 2. Replaces DroppedFile from type 'refId' to type 'provider_file_id'
29 * 3. build_body() reads provider_file_id and includes it in message content
30 *
31 * @param Meow_MWAI_Query_Base $query The query with potential file attachments
32 */
33 protected function prepare_query( $query ) {
34 // Get all attachments using the unified method
35 $attachments = method_exists( $query, 'getAttachments' ) ? $query->getAttachments() : [];
36
37 if ( empty( $attachments ) ) {
38 return;
39 }
40
41 // Check if code_interpreter is enabled
42 $hasCodeInterpreter = !empty( $query->tools ) && is_array( $query->tools ) && in_array( 'code_interpreter', $query->tools );
43
44 // MIME types supported by Code Execution tool (container_upload)
45 $codeExecutionMimes = [
46 'text/csv',
47 'application/vnd.ms-excel',
48 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
49 'application/msword',
50 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
51 'application/json',
52 'application/xml',
53 'text/xml',
54 'text/plain',
55 'text/markdown',
56 ];
57
58 // Process each attachment - upload files to Anthropic Files API
59 foreach ( $attachments as $index => $file ) {
60 $mimeType = $file->get_mimeType() ?? '';
61 $isPDF = strpos( $mimeType, 'application/pdf' ) === 0;
62 $isImage = $file->is_image();
63 $isCodeExecutionFile = $hasCodeInterpreter && in_array( $mimeType, $codeExecutionMimes );
64
65 // Skip files already uploaded (type = provider_file_id)
66 if ( $file->get_type() === 'provider_file_id' ) {
67 continue;
68 }
69
70 // Upload PDFs to Files API (always)
71 // Upload other documents to Files API only if code_interpreter is enabled
72 if ( $isPDF || $isCodeExecutionFile ) {
73 try {
74 // Get file data from WordPress uploads directory
75 $refId = $file->get_refId();
76 $data = $this->core->files->get_data( $refId );
77 $filename = $file->get_filename();
78
79 // Upload to Anthropic Files API
80 $uploadedFile = $this->upload_file( $filename, $data, $mimeType );
81 $fileId = $uploadedFile['id'] ?? null;
82
83 if ( $fileId ) {
84 // Store provider file_id in metadata for cleanup later
85 $localFileId = $this->core->files->get_id_from_refId( $refId );
86 if ( $localFileId ) {
87 $this->core->files->add_metadata( $localFileId, 'file_id', $fileId );
88 $this->core->files->add_metadata( $localFileId, 'provider', 'anthropic' );
89 }
90
91 // Replace the file object in attachedFiles array with provider_file_id type
92 if ( !empty( $query->attachedFiles ) && isset( $query->attachedFiles[$index] ) ) {
93 $query->attachedFiles[$index] = Meow_MWAI_Query_DroppedFile::from_provider_file_id(
94 $fileId,
95 'analysis',
96 $file->get_mimeType()
97 );
98 }
99 // Also update legacy attachedFile if this is the first file
100 if ( $index === 0 && !empty( $query->attachedFile ) ) {
101 $query->attachedFile = Meow_MWAI_Query_DroppedFile::from_provider_file_id(
102 $fileId,
103 'analysis',
104 $file->get_mimeType()
105 );
106 }
107
108 if ( $isCodeExecutionFile ) {
109 Meow_MWAI_Logging::log( "Anthropic: Uploaded file for code execution: {$filename} ({$mimeType}) -> {$fileId}" );
110 }
111 }
112 }
113 catch ( Exception $e ) {
114 error_log( '[AI Engine] Failed to upload file to Anthropic Files API: ' . $e->getMessage() );
115 // Keep the original file - build_messages() will fall back to base64 for PDFs
116 }
117 }
118 }
119 }
120
121 protected function isMCPTool( $toolName ) {
122 // Get all MCP tools from the filter
123 $mcpTools = apply_filters( 'mwai_mcp_tools', [] );
124
125 // Log available MCP tools for debugging
126 if ( empty( $mcpTools ) ) {
127 Meow_MWAI_Logging::log( 'Anthropic: No MCP tools available from filter' );
128 }
129
130 foreach ( $mcpTools as $tool ) {
131 if ( isset( $tool['name'] ) && $tool['name'] === $toolName ) {
132 Meow_MWAI_Logging::log( "Anthropic: Found MCP tool match: {$toolName}" );
133 return true;
134 }
135 }
136
137 // If we have MCP servers but tool not found, it might be an issue
138 if ( !empty( $this->mcpServerNames ) && !empty( $toolName ) ) {
139 Meow_MWAI_Logging::log( "Anthropic: Tool '{$toolName}' not found in MCP tools list" );
140 }
141
142 return false;
143 }
144
145 public function reset_stream() {
146 $this->streamContent = null;
147 $this->streamBuffer = null;
148 $this->streamFunctionCall = null;
149 $this->streamToolCalls = [];
150 $this->streamLastMessage = null;
151 $this->streamInTokens = null;
152 $this->streamOutTokens = null;
153 $this->streamIsThinking = false;
154 $this->mcpTools = []; // Reset MCP tools tracking
155 $this->textStarted = false; // Reset text started flag
156 $this->requestSentEmitted = false; // Reset request sent flag
157 $this->emittedFunctionResults = []; // Reset function result tracking
158
159 $this->streamBlocks = [
160 'role' => 'assistant',
161 'content' => []
162 ];
163
164 $this->inModel = null;
165 $this->inId = null;
166 }
167
168 protected function set_environment() {
169 $env = $this->env;
170 $this->apiKey = $env['apikey'];
171 }
172
173 protected function build_url( $query, $endpoint = null ) {
174 $endpoint = apply_filters( 'mwai_anthropic_endpoint', 'https://api.anthropic.com/v1', $this->env );
175 if ( $query instanceof Meow_MWAI_Query_Text || $query instanceof Meow_MWAI_Query_Feedback ) {
176 $url = trailingslashit( $endpoint ) . 'messages';
177 }
178 else {
179 throw new Exception( 'AI Engine: Unsupported query type.' );
180 }
181 return $url;
182 }
183
184 protected function build_headers( $query ) {
185 parent::build_headers( $query );
186 $headers = [
187 'Content-Type' => 'application/json',
188 'x-api-key' => $this->apiKey,
189 'anthropic-version' => '2023-06-01',
190 'anthropic-beta' => 'prompt-caching-2024-07-31, tools-2024-04-04, pdfs-2024-09-25, mcp-client-2025-04-04, files-api-2025-04-14, code-execution-2025-08-25',
191 'User-Agent' => 'AI Engine',
192 ];
193 return $headers;
194 }
195
196 public function final_checks( Meow_MWAI_Query_Base $query ) {
197 // We skip this completely.
198 // maxMessages is handed in build_messages().
199 }
200
201 /**
202 * Build messages array for Anthropic API request.
203 *
204 * This method constructs the 'messages' array that will be sent to Anthropic's API.
205 * It processes both conversation history and file attachments.
206 *
207 * CRITICAL: This method handles BOTH single file (attachedFile) and multi-file (attachedFiles).
208 * The attachedFiles array is the PRIMARY path for multi-file uploads.
209 *
210 * @param Meow_MWAI_Query_Text $query The query to build messages from
211 * @return array Messages formatted for Anthropic API
212 */
213 protected function build_messages( $query ) {
214 $messages = [];
215
216 // Add conversation history (previous messages)
217 foreach ( $query->messages as $message ) {
218 $messages[] = $message;
219 }
220
221 // Limit message history if maxMessages is set
222 if ( !empty( $query->maxMessages ) ) {
223 $messages = array_slice( $messages, -$query->maxMessages );
224 }
225
226 // Anthropic requires first message to have 'user' role
227 if ( !empty( $messages ) && $messages[0]['role'] !== 'user' ) {
228 array_shift( $messages );
229 }
230
231 // =====================================================================
232 // FILE UPLOAD: Process all attachments (unified approach)
233 // =====================================================================
234 // Uses getAttachments() which returns both attachedFiles and legacy attachedFile
235 $attachments = method_exists( $query, 'getAttachments' ) ? $query->getAttachments() : [];
236 if ( !empty( $attachments ) ) {
237 $message = $query->get_message();
238 if ( empty( $message ) ) {
239 $message = 'I uploaded files. Do not consider this message as part of the conversation.';
240 }
241
242 // Build content array: [text, document, document, ...] or [text, image, image, ...]
243 $content = [
244 [
245 'type' => 'text',
246 'text' => $message
247 ]
248 ];
249
250 // MIME types that require Code Execution tool (container_upload)
251 $codeExecutionMimes = [
252 'text/csv',
253 'application/vnd.ms-excel',
254 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
255 'application/msword',
256 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
257 'application/json',
258 'application/xml',
259 'text/xml',
260 'text/plain',
261 'text/markdown',
262 ];
263
264 // Process each file and add to content array
265 foreach ( $attachments as $file ) {
266 $mime = $file->get_mimeType();
267 $isPDF = $mime === 'application/pdf';
268 $isIMG = !$isPDF && $file->is_image();
269 $isProviderFile = $file->get_type() === 'provider_file_id';
270 $isCodeExecutionFile = in_array( $mime, $codeExecutionMimes, true );
271
272 // ===== CODE EXECUTION FILES (DOCX, XLSX, CSV, etc.) =====
273 // These are uploaded via Files API and used with container_upload block
274 if ( $isCodeExecutionFile && $isProviderFile ) {
275 $fileId = $file->get_refId();
276 $content[] = [
277 'type' => 'container_upload',
278 'file_id' => $fileId
279 ];
280 continue;
281 }
282
283 // ===== PDF FILES =====
284 if ( $isPDF ) {
285 $documentSource = null;
286 if ( $isProviderFile ) {
287 // File was uploaded in prepare_query() - use file_id reference
288 // This is the EXPECTED path after prepare_query() runs
289 $fileId = $file->get_refId();
290 $documentSource = [
291 'type' => 'file',
292 'file_id' => $fileId // e.g., file_011CTkNhtS6cU3CKcvTPCfvw
293 ];
294 }
295 else {
296 // Fallback: File not uploaded yet (shouldn't happen if prepare_query ran)
297 // This handles edge cases where prepare_query was skipped
298 try {
299 $refId = $file->get_refId();
300 $data = $this->core->files->get_data( $refId );
301 $filename = $file->get_filename();
302 $uploadedFile = $this->upload_file( $filename, $data, $mime );
303 $fileId = $uploadedFile['id'] ?? null;
304
305 if ( $fileId ) {
306 // Store provider file_id in metadata for cleanup later
307 $localFileId = $this->core->files->get_id_from_refId( $refId );
308 if ( $localFileId ) {
309 $this->core->files->add_metadata( $localFileId, 'file_id', $fileId );
310 $this->core->files->add_metadata( $localFileId, 'provider', 'anthropic' );
311 }
312
313 $documentSource = [
314 'type' => 'file',
315 'file_id' => $fileId
316 ];
317 }
318 else {
319 throw new Exception( 'Upload failed - no file_id returned' );
320 }
321 }
322 catch ( Exception $e ) {
323 error_log( '[AI Engine] Failed to upload PDF to Anthropic Files API: ' . $e->getMessage() . ', falling back to base64' );
324 // Last resort: base64 encoding (less efficient)
325 $data = $file->get_base64();
326 $documentSource = [
327 'type' => 'base64',
328 'media_type' => 'application/pdf',
329 'data' => $data
330 ];
331 }
332 }
333
334 // Add document to content array
335 $content[] = [
336 'type' => 'document',
337 'source' => $documentSource
338 ];
339 }
340 // ===== IMAGE FILES =====
341 else if ( $isIMG ) {
342 $imageSource = null;
343 if ( $isProviderFile ) {
344 // Use file_id reference (if uploaded to Files API)
345 $fileId = $file->get_refId();
346 $imageSource = [
347 'type' => 'file',
348 'file_id' => $fileId
349 ];
350 }
351 else {
352 // Use base64 encoding (standard for images)
353 $data = $file->get_base64();
354 $imageSource = [
355 'type' => 'base64',
356 'media_type' => $mime,
357 'data' => $data
358 ];
359 }
360
361 // Add image to content array
362 $content[] = [
363 'type' => 'image',
364 'source' => $imageSource
365 ];
366 }
367 }
368
369 // Add the complete message with all files to messages array
370 $messages[] = [
371 'role' => 'user',
372 'content' => $content // [text, document, document] or [text, image, image]
373 ];
374 }
375 else {
376 $messages[] = [ 'role' => 'user', 'content' => $query->get_message() ];
377 }
378
379 return $messages;
380 }
381
382 // Define a function to recursively replace empty arrays with empty stdClass objects
383 // To avoid errors with OpenAI's API
384 private function replaceEmptyArrayWithObject( $item ) {
385 if ( is_array( $item ) ) {
386 if ( empty( $item ) ) {
387 return new stdClass(); // Replace empty array with empty object
388 }
389 foreach ( $item as $key => $value ) {
390 $item[$key] = $this->replaceEmptyArrayWithObject( $value ); // Recurse
391 }
392 }
393 return $item;
394 }
395
396 protected function build_body( $query, $streamCallback = null, $extra = null ) {
397 if ( $query instanceof Meow_MWAI_Query_Feedback ) {
398 $body = [
399 'model' => $query->model,
400 'max_tokens' => $query->maxTokens,
401 'temperature' => $query->temperature,
402 'stream' => !is_null( $streamCallback ),
403 'messages' => []
404 ];
405
406 if ( !empty( $query->instructions ) ) {
407 $body['system'] = [
408 [
409 'type' => 'text',
410 'text' => $query->instructions,
411 'cache_control' => [ 'type' => 'ephemeral' ]
412 ]
413 ];
414 }
415
416 // Build the messages
417 $body['messages'][] = [ 'role' => 'user', 'content' => $query->message ];
418
419 if ( !empty( $query->blocks ) ) {
420 foreach ( $query->blocks as $feedback_block ) {
421 $contentBlock = $feedback_block['rawMessage']['content'];
422
423 // Process each content item individually to ensure proper handling of multiple tool_use blocks
424 if ( is_array( $contentBlock ) ) {
425 foreach ( $contentBlock as &$contentItem ) {
426 if ( isset( $contentItem['type'] ) && $contentItem['type'] === 'tool_use' ) {
427 // Debug logging for tool_use blocks
428 if ( $this->core->get_option( 'queries_debug_mode' ) ) {
429 error_log( 'AI Engine: Anthropic tool_use block - ID: ' . ( $contentItem['id'] ?? 'unknown' ) .
430 ', Name: ' . ( $contentItem['name'] ?? 'unknown' ) .
431 ', Input type: ' . gettype( $contentItem['input'] ?? null ) .
432 ', Input value: ' . json_encode( $contentItem['input'] ?? null ) );
433 }
434
435 // Ensure input is an object, not an array
436 if ( isset( $contentItem['input'] ) ) {
437 if ( empty( $contentItem['input'] ) || ( is_array( $contentItem['input'] ) && count( $contentItem['input'] ) === 0 ) ) {
438 $contentItem['input'] = new stdClass();
439 }
440 else {
441 // Apply replaceEmptyArrayWithObject only to the input field
442 $contentItem['input'] = $this->replaceEmptyArrayWithObject( $contentItem['input'] );
443 }
444 }
445 else {
446 $contentItem['input'] = new stdClass();
447 }
448
449 // Debug logging after conversion
450 if ( $this->core->get_option( 'queries_debug_mode' ) ) {
451 error_log( 'AI Engine: After conversion - Input type: ' . gettype( $contentItem['input'] ) .
452 ', Input value: ' . json_encode( $contentItem['input'] ) );
453 }
454 }
455 }
456 unset( $contentItem );
457 }
458
459 // Final debug logging before adding the message
460 if ( $this->core->get_option( 'queries_debug_mode' ) && is_array( $contentBlock ) ) {
461 error_log( 'AI Engine: Final contentBlock being added to messages: ' . json_encode( $contentBlock ) );
462 }
463
464 $assistantMessageIndex = count( $body['messages'] );
465 $body['messages'][] = [
466 'role' => 'assistant',
467 'content' => $contentBlock
468 ];
469
470 // Collect all tool results for this message
471 $toolResults = [];
472
473 foreach ( $feedback_block['feedbacks'] as $feedback ) {
474 $feedbackValue = $feedback['reply']['value'];
475 if ( !is_string( $feedbackValue ) ) {
476 $feedbackValue = json_encode( $feedbackValue );
477 }
478
479 $toolResults[] = [
480 'type' => 'tool_result',
481 'tool_use_id' => $feedback['request']['toolId'],
482 'content' => [
483 [
484 'type' => 'text',
485 'text' => $feedbackValue
486 ]
487 ],
488 'is_error' => false // Cool, Anthropic supports errors!
489 ];
490
491 // Note: Function result events are now emitted centrally in core.php
492 // when the function is actually executed
493 }
494
495 // Add all tool results in a single user message
496 // Anthropic requires all tool_results for a message to be in one content array
497 if ( !empty( $toolResults ) ) {
498 $body['messages'][] = [
499 'role' => 'user',
500 'content' => $toolResults
501 ];
502 }
503 }
504 }
505
506 // TODO: This WAS COPIED FROM BELOW
507 // Support for functions
508 if ( !empty( $query->functions ) ) {
509 $model = $this->retrieve_model_info( $query->model );
510 if ( !empty( $model['tags'] ) && !in_array( 'functions', $model['tags'] ) ) {
511 Meow_MWAI_Logging::warn( 'The model "' . $query->model . '" doesn\'t support Function Calling.' );
512 }
513 else {
514 $body['tools'] = [];
515 // Dynamic function: they will interactively enhance the completion (tools).
516 foreach ( $query->functions as $function ) {
517 $body['tools'][] = $function->serializeForAnthropic();
518 }
519 // Static functions: they will be executed at the end of the completion.
520 //$body['function_call'] = $query->functionCall;
521 }
522 }
523
524 // To avoid errors with Anthropic's API, we need to replace empty arrays with empty objects
525 // Note: We've already handled tool_use inputs above, so no need to process them again
526 return $body;
527 }
528 else if ( $query instanceof Meow_MWAI_Query_Text ) {
529 $body = [
530 'model' => $query->model,
531 'stream' => !is_null( $streamCallback ),
532 ];
533
534 if ( !empty( $query->maxTokens ) ) {
535 $body['max_tokens'] = $query->maxTokens;
536 }
537 else {
538 // https://docs.anthropic.com/en/docs/about-claude/models#model-comparison-table
539 $body['max_tokens'] = 4096;
540 }
541
542 if ( !empty( $query->temperature ) ) {
543 $body['temperature'] = $query->temperature;
544 }
545
546 if ( !empty( $query->stop ) ) {
547 $body['stop'] = $query->stop;
548 }
549
550 // Build system prompt with caching support.
551 // Instructions are cached (static per chatbot), context is not (varies per query).
552 $systemBlocks = [];
553 if ( !empty( $query->instructions ) ) {
554 $systemBlocks[] = [
555 'type' => 'text',
556 'text' => $query->instructions,
557 'cache_control' => [ 'type' => 'ephemeral' ]
558 ];
559 }
560 if ( !empty( $query->context ) ) {
561 $systemBlocks[] = [
562 'type' => 'text',
563 'text' => "Context:\n\n" . $query->context
564 ];
565 }
566 if ( !empty( $systemBlocks ) ) {
567 $body['system'] = $systemBlocks;
568 }
569
570 // Support for functions
571 if ( !empty( $query->functions ) ) {
572 $model = $this->retrieve_model_info( $query->model );
573 if ( !empty( $model['tags'] ) && !in_array( 'functions', $model['tags'] ) ) {
574 Meow_MWAI_Logging::warn( 'The model "' . $query->model . '" doesn\'t support Function Calling.' );
575 }
576 else {
577 $body['tools'] = [];
578 // Dynamic function: they will interactively enhance the completion (tools).
579 foreach ( $query->functions as $function ) {
580 $body['tools'][] = $function->serializeForAnthropic();
581 }
582 // Static functions: they will be executed at the end of the completion.
583 //$body['function_call'] = $query->functionCall;
584 }
585 }
586
587 $body['messages'] = $this->build_messages( $query );
588
589 // Add MCP servers if available
590 if ( isset( $query->mcpServers ) && is_array( $query->mcpServers ) && !empty( $query->mcpServers ) ) {
591 $body['mcp_servers'] = [];
592 $mcp_envs = $this->core->get_option( 'mcp_envs' );
593 $this->mcpServerNames = []; // Reset MCP server names
594
595 foreach ( $query->mcpServers as $mcpServer ) {
596 if ( isset( $mcpServer['id'] ) ) {
597 // Find the full MCP server configuration by ID
598 foreach ( $mcp_envs as $env ) {
599 if ( $env['id'] === $mcpServer['id'] ) {
600 $mcp_config = [
601 'type' => 'url',
602 'url' => $env['url'],
603 'name' => $env['name'],
604 'tool_configuration' => [
605 'enabled' => true
606 ]
607 ];
608
609 // Add authorization token if available
610 if ( !empty( $env['token'] ) ) {
611 $mcp_config['authorization_token'] = $env['token'];
612 }
613
614 $body['mcp_servers'][] = $mcp_config;
615 $this->mcpServerNames[] = $env['name']; // Track MCP server names
616 break;
617 }
618 }
619 }
620 }
621 }
622
623 // Add code_execution tool if code_interpreter is enabled
624 if ( !empty( $query->tools ) && is_array( $query->tools ) && in_array( 'code_interpreter', $query->tools ) ) {
625 if ( !isset( $body['tools'] ) ) {
626 $body['tools'] = [];
627 }
628 $body['tools'][] = [
629 'type' => 'code_execution_20250825',
630 'name' => 'code_execution'
631 ];
632 Meow_MWAI_Logging::log( 'Anthropic: Added code_execution tool to request' );
633 }
634
635 return $body;
636 }
637 else {
638 throw new Exception( 'AI Engine: Unsupported query type.' );
639 }
640 }
641
642 protected function stream_data_handler( $json ) {
643 $content = null;
644 $type = !empty( $json['type'] ) ? $json['type'] : null;
645 if ( is_null( $type ) ) {
646 return $content;
647 }
648
649 if ( $type === 'message_start' ) {
650 $usage = $json['message']['usage'];
651 $this->streamInTokens = $usage['input_tokens'];
652 $this->inModel = $json['message']['model'];
653 $this->inId = $json['message']['id'];
654
655 // Send MCP discovery event if MCP servers are configured
656 if ( $this->currentDebugMode && $this->streamCallback ) {
657 if ( !empty( $this->mcpServerNames ) ) {
658 $serverCount = count( $this->mcpServerNames );
659
660 // Get MCP tools count
661 $mcpTools = apply_filters( 'mwai_mcp_tools', [] );
662 $toolCount = count( $mcpTools );
663
664 $event = Meow_MWAI_Event::mcp_discovery( $serverCount, $toolCount )
665 ->set_metadata( 'servers', $this->mcpServerNames );
666 call_user_func( $this->streamCallback, $event );
667 }
668 }
669 }
670 else if ( $type === 'content_block_start' ) {
671 $this->streamBlocks['content'][] = $json['content_block'];
672
673 // Send "Generating response..." when we start a text block
674 if ( $this->currentDebugMode && $this->streamCallback ) {
675 $block = $json['content_block'];
676 if ( $block['type'] === 'text' && !$this->textStarted ) {
677 $this->textStarted = true;
678 $event = Meow_MWAI_Event::generating_response();
679 call_user_func( $this->streamCallback, $event );
680 }
681 }
682 }
683 else if ( $type === 'content_block_delta' ) {
684 $index = $json['index'];
685 $block = $this->streamBlocks['content'][$index];
686 if ( $json['delta']['type'] === 'text_delta' ) {
687 $block['text'] .= $json['delta']['text'];
688 $isThinkingStart = strpos( $block['text'], '<thinking' ) === 0;
689 $isThinkingEnd = strpos( $block['text'], '</thinking>' ) === 0;
690
691 if ( $isThinkingStart ) {
692 $this->streamIsThinking = true;
693 // Send thinking start event
694 if ( $this->currentDebugMode && $this->streamCallback ) {
695 $event = Meow_MWAI_Event::thinking( 'Thinking...' );
696 call_user_func( $this->streamCallback, $event );
697 }
698 }
699 if ( $isThinkingEnd ) {
700 $this->streamIsThinking = false;
701 // Send thinking end event
702 if ( $this->currentDebugMode && $this->streamCallback ) {
703 $event = Meow_MWAI_Event::thinking( 'Thinking completed.' )
704 ->set_metadata( 'status', 'completed' );
705 call_user_func( $this->streamCallback, $event );
706 }
707 }
708 $content = $json['delta']['text'];
709 }
710 else if ( $json['delta']['type'] === 'input_json_delta' ) {
711 // Somehow, the input is set as an array, but it should be a string since it's JSON.
712 $block['input'] = is_array( $block['input'] ) ? '' : $block['input'];
713 $block['input'] .= $json['delta']['partial_json'];
714
715 // Skip sending tool arguments event - too verbose
716 // if ( $this->currentDebugMode && $this->streamCallback && isset($block['type']) && $block['type'] === 'tool_use' ) {
717 // $event = ( new Meow_MWAI_Event( 'live', MWAI_STREAM_TYPES['TOOL_ARGS'] ) )
718 // ->set_content( 'Streaming tool arguments...' )
719 // ->set_metadata( 'tool_name', $block['name'] ?? 'unknown' )
720 // ->set_metadata( 'partial_args', $json['delta']['partial_json'] );
721 // call_user_func( $this->streamCallback, $event );
722 // }
723 }
724 $this->streamBlocks['content'][$index] = $block;
725 }
726 // At the end of a block, let's look for any 'input' not yet decoded from JSON
727 else if ( $type === 'content_block_stop' ) {
728 $index = $json['index'];
729 $block = $this->streamBlocks['content'][$index];
730 if ( isset( $block['input'] ) && is_string( $block['input'] ) ) {
731 $block['input'] = json_decode( $block['input'], true );
732 }
733
734 // For tool_use blocks, ensure empty inputs are objects, not arrays
735 if ( $block['type'] === 'tool_use' && isset( $block['input'] ) ) {
736 if ( empty( $block['input'] ) || ( is_array( $block['input'] ) && count( $block['input'] ) === 0 ) ) {
737 $block['input'] = new stdClass();
738 }
739 }
740
741 $this->streamBlocks['content'][$index] = $block;
742
743 // Send event for content block completion
744 if ( $this->currentDebugMode && $this->streamCallback ) {
745 if ( $block['type'] === 'mcp_tool_use' ) {
746 // Store the tool name for later lookup when we get the result
747 $this->mcpTools[$block['id']] = $block['name'];
748
749 $event = Meow_MWAI_Event::mcp_calling( $block['name'], $block['id'], $block['input'] ?? [] )
750 ->set_metadata( 'server_name', $block['server_name'] ?? 'unknown' );
751 call_user_func( $this->streamCallback, $event );
752 }
753 else if ( $block['type'] === 'mcp_tool_result' ) {
754 // Look up the tool name from the tool_use_id
755 $tool_use_id = $block['tool_use_id'] ?? '';
756 $tool_name = isset( $this->mcpTools[$tool_use_id] ) ? $this->mcpTools[$tool_use_id] : 'unknown';
757
758 $event = Meow_MWAI_Event::mcp_result( $tool_name, $tool_use_id )
759 ->set_metadata( 'content', $block['content'] ?? '' );
760 call_user_func( $this->streamCallback, $event );
761 }
762 else if ( $block['type'] === 'tool_use' ) {
763 // Regular tool use (non-MCP)
764 $event = Meow_MWAI_Event::function_calling( $block['name'] ?? 'unknown', $block['input'] ?? [] )
765 ->set_metadata( 'tool_id', $block['id'] ?? '' );
766 call_user_func( $this->streamCallback, $event );
767 }
768 else if ( $block['type'] === 'text' ) {
769 // Don't send any event here - the text generation is handled by content deltas
770 // and completion is handled by message_stop
771 }
772 else if ( $block['type'] === 'ping' ) {
773 // https://docs.anthropic.com/en/docs/build-with-claude/streaming#ping-events
774 }
775 else {
776 Meow_MWAI_Logging::log( 'Anthropic: Unknown block type in content_block_stop: ' . $block['type'] );
777 }
778 }
779 }
780 else if ( $type === 'message_delta' ) {
781 $usage = $json['usage'];
782 $this->streamOutTokens = $usage['output_tokens'];
783 }
784 else if ( $type === 'error' ) {
785 $error = $json['error'];
786 $message = $error['message'];
787
788 // Send error event
789 if ( $this->currentDebugMode && $this->streamCallback ) {
790 $event = Meow_MWAI_Event::error( $message )
791 ->set_metadata( 'error_type', $error['type'] ?? 'unknown' );
792 call_user_func( $this->streamCallback, $event );
793 }
794
795 throw new Exception( $message );
796 }
797 else if ( $type === 'message_stop' ) {
798 // Skip sending completion event - too verbose
799 // if ( $this->currentDebugMode && $this->streamCallback ) {
800 // $event = Meow_MWAI_Event::stream_completed()
801 // ->set_metadata( 'total_tokens', ($this->streamInTokens ?? 0) + ($this->streamOutTokens ?? 0) );
802 // call_user_func( $this->streamCallback, $event );
803 // }
804 }
805 else {
806 Meow_MWAI_Logging::log( "Anthropic: Unknown stream data type: $type" );
807 }
808
809 // Avoid some endings
810 $endings = [ '<|im_end|>', '</s>' ];
811 if ( in_array( $content, $endings ) ) {
812 $content = null;
813 }
814
815 // If the stream is thinking, we don't want to return anything yet.
816 if ( $this->streamIsThinking ) {
817 $content = null;
818 }
819
820 return ( $content === '0' || !empty( $content ) ) ? $content : null;
821 }
822
823 // This create the "choices" (even though, often, it is only one choice).
824 // It is basically the reply, but one that is understood by the Meow_MWAI_Reply class.
825 public function create_choices( $data ) {
826 $returned_choices = [];
827 $tool_calls = [];
828 $text_content = '';
829
830 // First, collect all tool calls and text content
831 foreach ( $data['content'] as $content ) {
832 if ( $content['type'] === 'tool_use' ) {
833 // Collect all tool calls
834 $arguments = $content['input'] ?? new stdClass();
835
836 // Ensure arguments is properly formatted
837 if ( empty( $arguments ) ) {
838 $arguments = new stdClass();
839 }
840 else if ( is_array( $arguments ) && count( $arguments ) === 0 ) {
841 $arguments = new stdClass();
842 }
843
844 $tool_calls[] = [
845 'id' => $content['id'],
846 'type' => 'function',
847 'function' => [
848 'name' => $content['name'],
849 'arguments' => $arguments,
850 ]
851 ];
852 }
853 else if ( $content['type'] === 'text' ) {
854 $text_content .= $content['text'];
855 }
856 }
857
858 // Create a single choice with both tool calls and text content (like OpenAI does)
859 $message = [];
860
861 if ( !empty( $text_content ) ) {
862 $message['content'] = $text_content;
863 }
864
865 if ( !empty( $tool_calls ) ) {
866 $message['tool_calls'] = $tool_calls;
867 }
868
869 // Only create a choice if there's content or tool calls
870 if ( !empty( $message ) ) {
871 $returned_choices[] = [
872 'message' => $message
873 ];
874 }
875
876 return $returned_choices;
877 }
878
879 /**
880 * Override reset to include Anthropic-specific state
881 */
882 protected function reset_request_state() {
883 parent::reset_request_state();
884
885 // Reset Anthropic-specific state
886 $this->mcpTools = [];
887 $this->mcpToolCount = 0;
888 // Note: mcpServerNames is configuration, not request state
889 }
890
891 public function run_completion_query( $query, $streamCallback = null ): Meow_MWAI_Reply {
892 // Reset request-specific state to prevent leakage between requests
893 $this->reset_request_state();
894
895 $isStreaming = !is_null( $streamCallback );
896
897 // Initialize debug mode
898 $this->init_debug_mode( $query );
899
900 // IMPORTANT: Prepare query BEFORE setting up streaming hooks
901 // The streaming hook intercepts ALL wp_remote_* calls, so preparation must happen first
902 $this->prepare_query( $query );
903
904 if ( $isStreaming ) {
905 $this->streamCallback = $streamCallback;
906 add_action( 'http_api_curl', [ $this, 'stream_handler' ], 10, 3 );
907 }
908
909 $this->reset_stream();
910 $data = null;
911 $body = $this->build_body( $query, $streamCallback );
912 $url = $this->build_url( $query );
913 $headers = $this->build_headers( $query );
914 $options = $this->build_options( $headers, $body );
915
916 // Emit "Request sent" event for feedback queries
917 if ( $this->currentDebugMode && !empty( $streamCallback ) &&
918 ( $query instanceof Meow_MWAI_Query_Feedback || $query instanceof Meow_MWAI_Query_AssistFeedback ) ) {
919 $event = Meow_MWAI_Event::request_sent()
920 ->set_metadata( 'is_feedback', true )
921 ->set_metadata( 'feedback_count', count( $query->blocks ) );
922 call_user_func( $streamCallback, $event );
923 }
924
925 try {
926 $res = $this->run_query( $url, $options, $streamCallback );
927 $reply = new Meow_MWAI_Reply( $query );
928 $returned_id = null;
929 $returned_model = null;
930 $returned_choices = [];
931
932 // Streaming Mode
933 if ( $isStreaming ) {
934 $returned_id = $this->inId;
935 $returned_model = $this->inModel ? $this->inModel : $query->model;
936 if ( !is_null( $this->streamInTokens && !is_null( $this->streamOutTokens ) ) ) {
937 $returned_in_tokens = $this->streamInTokens;
938 $returned_out_tokens = $this->streamOutTokens;
939 }
940 $data = $this->streamBlocks;
941
942 // Clean up streaming data as well
943 if ( isset( $data['content'] ) && is_array( $data['content'] ) ) {
944 foreach ( $data['content'] as &$content ) {
945 if ( $content['type'] === 'tool_use' && isset( $content['input'] ) ) {
946 if ( empty( $content['input'] ) || ( is_array( $content['input'] ) && count( $content['input'] ) === 0 ) ) {
947 $content['input'] = new stdClass();
948 }
949 }
950 }
951 unset( $content );
952 }
953
954 $returned_choices = $this->create_choices( $this->streamBlocks );
955 }
956 // Standard Mode
957 else {
958 $data = $res['data'];
959
960 // Clean up tool_use inputs in the raw data BEFORE it gets stored
961 if ( isset( $data['content'] ) && is_array( $data['content'] ) ) {
962 foreach ( $data['content'] as &$content ) {
963 if ( $content['type'] === 'tool_use' && isset( $content['input'] ) ) {
964 if ( empty( $content['input'] ) || ( is_array( $content['input'] ) && count( $content['input'] ) === 0 ) ) {
965 $content['input'] = new stdClass();
966 }
967 }
968 }
969 unset( $content );
970 }
971
972 $returned_id = $data['id'];
973 $returned_model = $data['model'];
974 $usage = $data['usage'];
975 if ( !empty( $usage ) ) {
976 $returned_in_tokens = isset( $usage['input_tokens'] ) ? $usage['input_tokens'] : null;
977 $returned_out_tokens = isset( $usage['output_tokens'] ) ? $usage['output_tokens'] : null;
978 }
979 $returned_choices = $this->create_choices( $data );
980 }
981
982 $reply->set_choices( $returned_choices, $data );
983 if ( !empty( $returned_id ) ) {
984 $reply->set_id( $returned_id );
985 }
986
987 // Handle tokens.
988 $this->handle_tokens_usage(
989 $reply,
990 $query,
991 $returned_model,
992 $returned_in_tokens,
993 $returned_out_tokens
994 );
995
996 return $reply;
997 }
998 catch ( Exception $e ) {
999 $error = $e->getMessage();
1000 $json = json_decode( $error, true );
1001 if ( json_last_error() === JSON_ERROR_NONE ) {
1002 if ( isset( $json['error'] ) && isset( $json['error']['message'] ) ) {
1003 $error = $json['error']['message'];
1004 }
1005 }
1006 Meow_MWAI_Logging::error( '(Anthropic) ' . $error );
1007 $service = $this->get_service_name();
1008 $message = "From $service: " . $error;
1009 throw new Exception( $message );
1010 }
1011 finally {
1012 if ( $isStreaming ) {
1013 remove_action( 'http_api_curl', [ $this, 'stream_handler' ] );
1014 }
1015 }
1016 }
1017
1018 protected function build_options( $headers, $json = null, $forms = null, $method = 'POST' ) {
1019 $body = null;
1020 if ( !empty( $forms ) ) {
1021 $boundary = wp_generate_password( 24, false );
1022 $headers['Content-Type'] = 'multipart/form-data; boundary=' . $boundary;
1023 $body = $this->build_form_body( $forms, $boundary );
1024 }
1025 else if ( !empty( $json ) ) {
1026 // For Anthropic, we need to ensure empty objects stay as objects, not arrays
1027 // JSON_FORCE_OBJECT would force everything to be an object, which we don't want
1028 // Instead, we've already converted empty arrays to stdClass in build_body
1029 $body = $this->safe_json_encode( $json, 'request body' );
1030
1031 // Debug logging to verify JSON encoding
1032 if ( $this->core->get_option( 'queries_debug_mode' ) ) {
1033 // Check if the body contains tool_use blocks with empty inputs
1034 if ( strpos( $body, '"tool_use"' ) !== false ) {
1035 error_log( 'AI Engine: Anthropic JSON body after encoding (first 1000 chars): ' . substr( $body, 0, 1000 ) );
1036
1037 // Check specifically for "input":[] which would be wrong
1038 if ( strpos( $body, '"input":[]' ) !== false ) {
1039 error_log( 'AI Engine: WARNING - Found "input":[] in JSON body, this should be "input":{} for Anthropic API' );
1040 }
1041 }
1042 }
1043 }
1044 $options = [
1045 'headers' => $headers,
1046 'method' => $method,
1047 'timeout' => MWAI_TIMEOUT,
1048 'body' => $body,
1049 'sslverify' => MWAI_SSL_VERIFY
1050 ];
1051 return $options;
1052 }
1053
1054 protected function get_service_name() {
1055 return 'Anthropic';
1056 }
1057
1058 public function get_models() {
1059 return apply_filters( 'mwai_anthropic_models', MWAI_ANTHROPIC_MODELS );
1060 }
1061
1062 public static function get_models_static() {
1063 return MWAI_ANTHROPIC_MODELS;
1064 }
1065
1066 public function handle_tokens_usage(
1067 $reply,
1068 $query,
1069 $returned_model,
1070 $returned_in_tokens,
1071 $returned_out_tokens,
1072 $returned_price = null
1073 ) {
1074 $returned_in_tokens = !is_null( $returned_in_tokens ) ?
1075 $returned_in_tokens : $reply->get_in_tokens( $query );
1076 $returned_out_tokens = !is_null( $returned_out_tokens ) ?
1077 $returned_out_tokens : $reply->get_out_tokens();
1078 if ( !empty( $reply->id ) ) {
1079 // Would be cool to retrieve the usage from the API, but it's not possible.
1080 }
1081 $usage = $this->core->record_tokens_usage( $returned_model, $returned_in_tokens, $returned_out_tokens );
1082 $reply->set_usage( $usage );
1083
1084 // Set accuracy based on data availability
1085 if ( !is_null( $returned_in_tokens ) && !is_null( $returned_out_tokens ) ) {
1086 // Anthropic provides token counts from API = tokens accuracy
1087 $reply->set_usage_accuracy( 'tokens' );
1088 }
1089 else {
1090 // Fallback to estimated
1091 $reply->set_usage_accuracy( 'estimated' );
1092 }
1093 }
1094
1095 public function get_price( Meow_MWAI_Query_Base $query, Meow_MWAI_Reply $reply ) {
1096 return parent::get_price( $query, $reply );
1097 }
1098
1099 /**
1100 * Check the connection to Anthropic by listing available models.
1101 * Anthropic doesn't provide a models endpoint, so we just verify authentication works.
1102 */
1103 public function connection_check() {
1104 try {
1105 // Get the endpoint
1106 $endpoint = apply_filters( 'mwai_anthropic_endpoint', 'https://api.anthropic.com/v1', $this->env );
1107
1108 // For Anthropic, we'll use the messages endpoint with a minimal request to verify auth
1109 $url = trailingslashit( $endpoint ) . 'messages';
1110
1111 // Create a minimal query just to test authentication
1112 $testBody = [
1113 'model' => 'claude-3-haiku-20240307', // Use cheapest model
1114 'max_tokens' => 1,
1115 'messages' => [
1116 ['role' => 'user', 'content' => 'Hi']
1117 ],
1118 'metadata' => [
1119 'user_id' => 'connection_test'
1120 ]
1121 ];
1122
1123 // Build headers with a dummy query
1124 $dummyQuery = new Meow_MWAI_Query_Text( 'test' );
1125 $headers = $this->build_headers( $dummyQuery );
1126 $options = $this->build_options( $headers, $testBody );
1127
1128 // Try to make a minimal request
1129 $response = $this->run_query( $url, $options );
1130
1131 // If we get here without exception, the API key is valid
1132 // Get the list of available models from our constants
1133 $models = $this->get_models();
1134 $modelNames = array_map( function ( $model ) {
1135 return $model['model'] ?? $model['name'] ?? 'unknown';
1136 }, $models );
1137
1138 return [
1139 'models' => array_slice( $modelNames, 0, 10 ), // Return first 10 models
1140 'service' => 'Anthropic'
1141 ];
1142 }
1143 catch ( Exception $e ) {
1144 // Check if it's an authentication error
1145 $message = $e->getMessage();
1146 if ( strpos( $message, 'authentication_error' ) !== false ||
1147 strpos( $message, 'invalid x-api-key' ) !== false ||
1148 strpos( $message, '401' ) !== false ) {
1149 throw new Exception( 'Invalid API key' );
1150 }
1151 throw new Exception( 'Connection failed: ' . $message );
1152 }
1153 }
1154
1155 /**
1156 * Upload a file to Anthropic Files API
1157 *
1158 * @param string $filename The name of the file
1159 * @param string $data The file content (binary)
1160 * @param string $purpose For Anthropic, this is the MIME type (API difference from OpenAI)
1161 * @return array The response from the API containing file_id
1162 * @throws Exception If upload fails
1163 */
1164 public function upload_file( $filename, $data, $purpose = 'application/pdf' ) {
1165 global $wp_filter;
1166
1167 // For Anthropic, $purpose is actually the MIME type (different from OpenAI's API)
1168 $mimeType = $purpose;
1169
1170 // Build multipart form data
1171 $boundary = wp_generate_password( 24, false );
1172 $body = '';
1173 $body .= "--$boundary\r\n";
1174 $body .= "Content-Disposition: form-data; name=\"file\"; filename=\"{$filename}\"\r\n";
1175 $body .= 'Content-Type: ' . $mimeType . "\r\n\r\n";
1176 $body .= $data . "\r\n";
1177 $body .= "--$boundary\r\n";
1178 $body .= "Content-Disposition: form-data; name=\"mime_type\"\r\n\r\n";
1179 $body .= $mimeType . "\r\n";
1180 $body .= "--$boundary--\r\n";
1181
1182 // Temporarily remove ALL http_api_curl hooks to prevent streaming hook interference
1183 // Save current hooks
1184 $saved_hooks = null;
1185 if ( isset( $wp_filter['http_api_curl'] ) ) {
1186 $saved_hooks = $wp_filter['http_api_curl'];
1187 unset( $wp_filter['http_api_curl'] );
1188 }
1189
1190 // Upload using WordPress HTTP API
1191 $endpoint = apply_filters( 'mwai_anthropic_endpoint', 'https://api.anthropic.com/v1', $this->env );
1192 $url = $endpoint . '/files';
1193 $response = wp_remote_post( $url, [
1194 'headers' => [
1195 'x-api-key' => $this->apiKey,
1196 'anthropic-version' => '2023-06-01',
1197 'anthropic-beta' => 'files-api-2025-04-14',
1198 'Content-Type' => 'multipart/form-data; boundary=' . $boundary
1199 ],
1200 'body' => $body,
1201 'timeout' => 60
1202 ] );
1203
1204 // Restore hooks
1205 if ( $saved_hooks !== null ) {
1206 $wp_filter['http_api_curl'] = $saved_hooks;
1207 }
1208
1209 if ( is_wp_error( $response ) ) {
1210 throw new Exception( 'File upload failed: ' . $response->get_error_message() );
1211 }
1212
1213 $response_body = wp_remote_retrieve_body( $response );
1214 $result = json_decode( $response_body, true );
1215
1216 // Check for API errors
1217 if ( isset( $result['error'] ) ) {
1218 throw new Exception( 'Anthropic Files API error: ' . ( $result['error']['message'] ?? 'Unknown error' ) );
1219 }
1220
1221 return $result;
1222 }
1223
1224 /**
1225 * Delete a file from Anthropic Files API
1226 *
1227 * @param string $fileId The Anthropic file ID to delete
1228 * @return array The response from the API
1229 * @throws Exception If deletion fails
1230 */
1231 public function delete_file( $fileId ) {
1232 $endpoint = apply_filters( 'mwai_anthropic_endpoint', 'https://api.anthropic.com/v1', $this->env );
1233 $url = $endpoint . '/files/' . $fileId;
1234
1235 $response = wp_remote_request( $url, [
1236 'method' => 'DELETE',
1237 'headers' => [
1238 'x-api-key' => $this->apiKey,
1239 'anthropic-version' => '2023-06-01',
1240 'anthropic-beta' => 'files-api-2025-04-14',
1241 ],
1242 'timeout' => 30
1243 ] );
1244
1245 if ( is_wp_error( $response ) ) {
1246 throw new Exception( 'File deletion failed: ' . $response->get_error_message() );
1247 }
1248
1249 $response_code = wp_remote_retrieve_response_code( $response );
1250 $response_body = wp_remote_retrieve_body( $response );
1251 $result = json_decode( $response_body, true );
1252
1253 // Check for API errors
1254 if ( $response_code >= 400 ) {
1255 $error_message = isset( $result['error']['message'] ) ? $result['error']['message'] : 'Unknown error';
1256 throw new Exception( 'Anthropic Files API error: ' . $error_message );
1257 }
1258
1259 return $result;
1260 }
1261 }
1262