PluginProbe ʕ •ᴥ•ʔ
AI Engine – The Chatbot, AI Framework & MCP for WordPress / 1.9.3
AI Engine – The Chatbot, AI Framework & MCP for WordPress v1.9.3
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 / queries / text.php
ai-engine / classes / queries Last commit date
base.php 2 years ago embed.php 2 years ago function.php 2 years ago image.php 2 years ago parameter.php 2 years ago text.php 2 years ago transcribe.php 2 years ago
text.php
407 lines
1 <?php
2
3 class Meow_MWAI_Query_Text extends Meow_MWAI_Query_Base implements JsonSerializable {
4 public int $maxTokens = 1024;
5 public float $temperature = 0.8;
6 public int $maxSentences = 15;
7 public ?string $stop = null;
8 public array $messages = [];
9 public ?string $context = null;
10 public ?string $newMessage = null;
11 public ?string $promptEnding = null;
12 public bool $casuallyFineTuned = false;
13 public ?int $promptTokens = null;
14
15 public function __construct( ?string $prompt = '', int $maxTokens = 1024, string $model = MWAI_DEFAULT_MODEL ) {
16 parent::__construct( $prompt );
17 $this->setModel( $model );
18 $this->setMaxTokens( $maxTokens );
19 }
20
21 public function jsonSerialize() {
22 return [
23 'class' => get_class( $this ),
24 'prompt' => $this->prompt,
25 'messages' => $this->messages,
26 'maxTokens' => $this->maxTokens,
27 'temperature' => $this->temperature,
28 'maxSentences' => $this->maxSentences,
29 'stop' => $this->stop,
30 'context' => $this->context,
31 'newMessage' => $this->newMessage,
32 'promptEnding' => $this->promptEnding,
33 'casuallyFineTuned' => $this->casuallyFineTuned,
34 'model' => $this->model,
35 'mode' => $this->mode,
36 'session' => $this->session,
37 'env' => $this->env,
38 'service' => $this->service,
39 ];
40 }
41
42 public function getPromptTokens( $refresh = false ): int {
43 if ( $this->promptTokens && !$refresh ) {
44 return $this->promptTokens;
45 }
46 $this->promptTokens = $this->estimateTokens( $this->messages );
47 return $this->promptTokens;
48 }
49
50 public function getLastPrompt(): string {
51 if ( empty( $this->messages ) ) {
52 return $this->prompt;
53 }
54 $lastMessage = end( $this->messages );
55 return $lastMessage['content'];
56 }
57
58 // Quick and dirty token estimation
59 // Let's keep this synchronized with Helpers in JS
60 function estimateTokens( $content ): int
61 {
62 $text = "";
63 // https://github.com/openai/openai-cookbook/blob/main/examples/How_to_count_tokens_with_tiktoken.ipynb
64 if ( is_array( $content ) ) {
65 foreach ( $content as $message ) {
66 $role = $message['role'];
67 $content = $message['content'];
68 $text .= "=#=$role\n$content=#=\n";
69 }
70 }
71 else {
72 $text = $content;
73 }
74 $tokens = 0;
75 return apply_filters( 'mwai_estimate_tokens', (int)$tokens, $text, $this->model );
76 }
77
78 /**
79 * Make sure the maxTokens is not greater than the model's context length.
80 */
81 public function finalChecks() {
82 if ( empty( $this->model ) ) { return; }
83
84 // Make sure the number of messages is not too great
85 if ( !empty( $this->maxSentences ) ) {
86 $context = array_shift( $this->messages );
87 if ( !empty( $this->messages ) ) {
88 $this->messages = array_slice( $this->messages, -$this->maxSentences * 2 );
89 }
90 else {
91 $this->messages = [];
92 }
93 if ( !empty( $context ) ) {
94 array_unshift( $this->messages, $context );
95 }
96 }
97
98 // Make sure the max tokens are respected.
99 $realMax = 4096;
100 $finetuneFamily = preg_match('/^([a-zA-Z]{0,32}):/', $this->model, $matches );
101 $finetuneFamily = ( isset( $matches ) && count( $matches ) > 0 ) ? $matches[1] : 'N/A';
102 $foundModel = null;
103 $openai_models = Meow_MWAI_Engines_OpenAI::get_openai_models();
104 foreach ( $openai_models as $currentModel ) {
105 if ( $currentModel['model'] === $this->model || $currentModel['family'] === $finetuneFamily ) {
106 $foundModel = $currentModel['name'];
107 $realMax = $currentModel['maxTokens'];
108 break;
109 }
110 }
111 $estimatedTokens = $this->getPromptTokens();
112 if ( !empty( $realMax ) && $estimatedTokens > $realMax ) {
113 throw new Exception( "AI Engine: The prompt is too long! It contains about $estimatedTokens tokens (estimation). The $foundModel model only accepts a maximum of $realMax tokens. " );
114 }
115 $realMax = (int)($realMax - $estimatedTokens) - 16;
116 if ( $this->maxTokens > $realMax ) {
117 $this->maxTokens = $realMax;
118 }
119 }
120
121 /**
122 * ID of the model to use.
123 * @param string $model ID of the model to use.
124 */
125 public function setModel( string $model ) {
126 $this->model = $model;
127 $this->mode = 'completion';
128 $found = false;
129 $openai_models = Meow_MWAI_Engines_OpenAI::get_openai_models();
130 foreach ( $openai_models as $currentModel ) {
131 if ( $currentModel['model'] === $this->model ) {
132 if ( $currentModel['mode'] ) {
133 $this->mode = $currentModel['mode'];
134 }
135 $found = true;
136 break;
137 }
138 }
139 if ( !$found ) {
140 // If the model can't be found, it's because it's probably a fine-tuned model. In the past (before August 2023),
141 // fine-tuned models were always based on GPT-3 (and therefore, using completion mode). From now on, they can be
142 // based on GPT-3.5 or 4 (and therefore, using chat mode). We need to detect that.
143 $baseModel = Meow_MWAI_Engines_OpenAI::getBaseModelForFinetune( $model );
144 if ( preg_match( '/^gpt-3.5|^gpt-4/', $baseModel ) ) {
145 $this->mode = 'chat';
146 }
147 }
148 }
149
150 /**
151 * Given a prompt, the model will return one or more predicted completions.
152 * It can also return the probabilities of alternative tokens at each position.
153 * @param string $prompt The prompt to generate completions.
154 */
155 public function setPrompt( $prompt ) {
156 parent::setPrompt( $prompt );
157 $this->validateMessages();
158 }
159
160 /**
161 * The prompt is used by models who uses Text Completion (and not Chat Completion).
162 * This returns the prompt if it's not a chat, otherwise it will build a prompt with
163 * all the messages nicely formatted.
164 */
165 public function getPrompt(): ?string {
166 if ( $this->mode === 'completion' ) {
167 return $this->prompt . $this->promptEnding;
168 }
169
170 $first = reset( $this->messages );
171 $prompt = "";
172 if ( $first && $first['role'] === 'system' ) {
173 $prompt = $first['content'] . "\n\n";
174 }
175
176 // Casually Fine-Tuned or Prompt-Ending
177 if ( !empty( $this->promptEnding ) ) {
178 $last = end( $this->messages );
179 if ( $last && $last['role'] === 'user' ) {
180 $prompt = $last['content'] . $this->promptEnding;
181 }
182 return $prompt;
183 }
184
185 // Standard Completion
186 while ( $message = next( $this->messages ) ) {
187 $role = $message['role'];
188 $content = $message['content'];
189 if ( $role === 'system' ) {
190 $prompt .= "$content\n\n";
191 }
192 if ( $role === 'user' ) {
193 $prompt .= "User: $content\n";
194 }
195 if ( $role === 'assistant' ) {
196 $prompt .= "AI: $content\n";
197 }
198 }
199 $prompt .= "AI: ";
200 return $prompt;
201 }
202
203 /**
204 * Similar to the prompt, but focus on the new/last message.
205 * Only used when the model has a chat mode (and only used in messages).
206 * @param string $prompt The messages to generate completions.
207 */
208 public function setNewMessage( string $newMessage ): void {
209 $this->newMessage = $newMessage;
210 $this->validateMessages();
211 }
212
213 public function replace( $search, $replace ) {
214 $this->prompt = str_replace( $search, $replace, $this->prompt );
215 $this->validateMessages();
216 }
217
218 /**
219 * Similar to the prompt, but use an array of messages instead.
220 * @param string $prompt The messages to generate completions.
221 */
222 public function setMessages( array $messages ) {
223 $messages = array_map( function( $message ) {
224 return [ 'role' => $message['role'], 'content' => $message['content'] ];
225 }, $messages );
226 $this->messages = $messages;
227 $this->validateMessages();
228 }
229
230 public function getLastMessage(): ?string {
231 if ( !empty( $this->messages ) ) {
232 $lastMessageIndex = count( $this->messages ) - 1;
233 $lastMessage = $this->messages[$lastMessageIndex];
234 return $lastMessage['content'];
235 }
236 return null;
237 }
238
239 // Function that adds a message just before the last message
240 public function injectContext( string $content ): void {
241 if ( !empty( $this->messages ) ) {
242 $lastMessageIndex = count( $this->messages ) - 1;
243 $lastMessage = $this->messages[$lastMessageIndex];
244 $this->messages[$lastMessageIndex] = [ 'role' => 'system', 'content' => $content ];
245 array_push( $this->messages, $lastMessage );
246 }
247 $this->validateMessages();
248 }
249
250 /**
251 * The context that is used for the chat completion (mode === 'chat').
252 * @param string $context The context to use.
253 */
254 public function setContext( string $context ): void {
255 $this->context = apply_filters( 'mwai_ai_context', $context, $this );
256 $this->validateMessages();
257 }
258
259 private function validateMessages(): void {
260 // Messages should end with either the prompt or, if exists, the newMessage.
261 $message = empty( $this->newMessage ) ? $this->prompt : $this->newMessage;
262 if ( empty( $this->messages ) ) {
263 $this->messages = [ [ 'role' => 'user', 'content' => $message ] ];
264 }
265 else {
266 $last = &$this->messages[ count( $this->messages ) - 1 ];
267 if ( $last['role'] === 'user' ) {
268 $last['content'] = $message;
269 }
270 else {
271 array_push( $this->messages, [ 'role' => 'user', 'content' => $message ] );
272 }
273 }
274
275 // The main context must be first.
276 if ( !empty( $this->context ) ) {
277 if ( is_array( $this->messages ) && count( $this->messages ) > 0 ) {
278 if ( $this->messages[0]['role'] !== 'system' ) {
279 array_unshift( $this->messages, [ 'role' => 'system', 'content' => $this->context ] );
280 }
281 else {
282 $this->messages[0]['content'] = $this->context;
283 }
284 }
285 }
286 }
287
288 /**
289 * The maximum number of tokens to generate in the completion.
290 * The token count of your prompt plus max_tokens cannot exceed the model's context length.
291 * Most models have a context length of 2048 tokens (except for the newest models, which support 4096).
292 * @param float $prompt The maximum number of tokens.
293 */
294 public function setMaxTokens( int $maxTokens ): void {
295 $this->maxTokens = $maxTokens;
296 }
297
298 /**
299 * Set the sampling temperature to use. Higher values means the model will take more risks.
300 * Try 0.9 for more creative applications, and 0 for ones with a well-defined reply.
301 * @param float $temperature The temperature.
302 */
303 public function setTemperature( float $temperature ): void {
304 $temperature = floatval( $temperature );
305 if ( $temperature > 1 ) {
306 $temperature = 1;
307 }
308 if ( $temperature < 0 ) {
309 $temperature = 0;
310 }
311 $this->temperature = round( $temperature, 2 );
312 }
313
314 public function setMaxSentences( int $maxSentences ): void {
315 if ( !empty( $maxSentences ) ) {
316 $this->maxSentences = intval( $maxSentences );
317 $this->validateMessages();
318 }
319 }
320
321 public function setStop( string $stop ): void {
322 $this->stop = $stop;
323 }
324
325 private function convertKeys( $params )
326 {
327 $newParams = [];
328 foreach ( $params as $key => $value ) {
329 $newKey = '';
330 $capitalizeNextChar = false;
331 for ( $i = 0; $i < strlen( $key ); $i++ ) {
332 if ( $key[$i] == '_' ) {
333 $capitalizeNextChar = true;
334 }
335 else {
336 $newKey .= $capitalizeNextChar ? strtoupper($key[$i]) : $key[$i];
337 $capitalizeNextChar = false;
338 }
339 }
340 $newParams[$newKey] = $value;
341 }
342 return $newParams;
343 }
344
345 // Based on the params of the query, update the attributes
346 public function injectParams( array $params ): void
347 {
348 // Those are for the keys passed directly by the shortcode.
349 $params = $this->convertKeys( $params );
350
351 $acceptedValues = [ true, 1, '1' ];
352 if ( !empty( $params['model'] ) ) {
353 $this->setModel( $params['model'] );
354 }
355 if ( !empty( $params['casuallyFineTuned'] ) && in_array( $params['casuallyFineTuned'], $acceptedValues, true ) ) {
356 $this->promptEnding = "\n\n###\n\n";
357 $this->stop = "\n\n";
358 $this->casuallyFineTuned = true;
359 }
360 if ( !empty( $params['prompt'] ) ) {
361 $this->setPrompt( $params['prompt'] );
362 }
363 if ( !empty( $params['context'] ) ) {
364 $this->setContext( $params['context'] );
365 }
366 if ( !empty( $params['messages'] ) ) {
367 $this->setMessages( $params['messages'] );
368 }
369 if ( !empty( $params['newMessage'] ) ) {
370 $this->setNewMessage( $params['newMessage'] );
371 }
372 if ( !empty( $params['maxTokens'] ) && intval( $params['maxTokens'] ) > 0 ) {
373 $this->setMaxTokens( intval( $params['maxTokens'] ) );
374 }
375 if ( !empty( $params['maxMessages'] ) && intval( $params['maxMessages'] ) > 0 ) {
376 $this->setMaxSentences( intval( $params['maxMessages'] ) );
377 }
378 if ( !empty( $params['maxSentences'] ) && intval( $params['maxSentences'] ) > 0 ) {
379 $this->setMaxSentences( intval( $params['maxSentences'] ) );
380 }
381 if ( !empty( $params['temperature'] ) ) {
382 $this->setTemperature( $params['temperature'] );
383 }
384 if ( !empty( $params['stop'] ) ) {
385 $this->setStop( $params['stop'] );
386 }
387 if ( !empty( $params['maxResults'] ) ) {
388 $this->setMaxResults( $params['maxResults'] );
389 }
390 if ( !empty( $params['env'] ) ) {
391 $this->setEnv( $params['env'] );
392 }
393 if ( !empty( $params['session'] ) ) {
394 $this->setSession( $params['session'] );
395 }
396 // Should add the params related to Open AI and Azure
397 if ( !empty( $params['service'] ) ) {
398 $this->setService( $params['service'] );
399 }
400 if ( !empty( $params['apiKey'] ) ) {
401 $this->setApiKey( $params['apiKey'] );
402 }
403 if ( !empty( $params['botId'] ) ) {
404 $this->setBotId( $params['botId'] );
405 }
406 }
407 }