diff --git a/class-gwiz-gf-openai.php b/class-gwiz-gf-openai.php
index 4d99e48..9393748 100644
--- a/class-gwiz-gf-openai.php
+++ b/class-gwiz-gf-openai.php
@@ -20,14 +20,6 @@ class GWiz_GF_OpenAI extends GFFeedAddOn {
* @var array The default settings to pass to OpenAI
*/
public $default_settings = array(
- 'completions' => array(
- 'max_tokens' => 500,
- 'temperature' => 1,
- 'top_p' => 1,
- 'frequency_penalty' => 0,
- 'presence_penalty' => 0,
- 'timeout' => 15,
- ),
'chat/completions' => array(
'max_tokens' => 1000,
'temperature' => 1,
@@ -36,11 +28,6 @@ class GWiz_GF_OpenAI extends GFFeedAddOn {
'presence_penalty' => 0,
'timeout' => 15,
),
- 'edits' => array(
- 'temperature' => 1,
- 'top_p' => 1,
- 'timeout' => 15,
- ),
'moderations' => array(
'timeout' => 5,
),
@@ -272,32 +259,6 @@ public function init() {
*/
public function get_openai_models() {
$models = array(
- 'completions' => array(
- 'text-davinci-003' => array(
- 'type' => 'GPT-3',
- 'description' => __( 'Most capable GPT-3 model. Can do any task the other models can do, often with higher quality, longer output and better instruction-following. Also supports inserting completions within text.', 'gravityforms-openai' ),
- ),
- 'text-curie-001' => array(
- 'type' => 'GPT-3',
- 'description' => __( 'Very capable, but faster and lower cost than Davinci.', 'gravityforms-openai' ),
- ),
- 'text-babbage-001' => array(
- 'type' => 'GPT-3',
- 'description' => __( 'Capable of straightforward tasks, very fast, and lower cost.', 'gravityforms-openai' ),
- ),
- 'text-ada-001' => array(
- 'type' => 'GPT-3',
- 'description' => __( 'Capable of very simple tasks, usually the fastest model in the GPT-3 series, and lowest cost.', 'gravityforms-openai' ),
- ),
- 'code-davinci-002' => array(
- 'type' => 'Codex',
- 'description' => __( 'Most capable Codex model. Particularly good at translating natural language to code. In addition to completing code, also supports inserting completions within code.', 'gravityforms-openai' ),
- ),
- 'code-cushman-001' => array(
- 'type' => 'Codex',
- 'description' => __( 'Almost as capable as Davinci Codex, but slightly faster. This speed advantage may make it preferable for real-time applications.', 'gravityforms-openai' ),
- ),
- ),
'chat/completions' => array(
'gpt-3.5-turbo' => array(
'description' => __( 'The same model used by ChatGPT.', 'gravityforms-openai' ),
@@ -315,16 +276,6 @@ public function get_openai_models() {
'description' => __( 'The latest GPT-4 model with improved instruction following, JSON mode, reproducible outputs, parallel function calling, and more. Returns a maximum of 4,096 output tokens.', 'gravityforms-openai' ),
),
),
- 'edits' => array(
- 'text-davinci-edit-001' => array(
- 'type' => 'GPT-3',
- 'description' => __( 'Most capable GPT-3 model. Can do any task the other models can do, often with higher quality, longer output and better instruction-following. Also supports inserting completions within text.', 'gravityforms-openai' ),
- ),
- 'code-davinci-edit-001' => array(
- 'type' => 'Codex',
- 'description' => __( 'Most capable Codex model. Particularly good at translating natural language to code. In addition to completing code, also supports inserting completions within code.', 'gravityforms-openai' ),
- ),
- ),
'moderations' => array(
'text-moderation-stable' => array(
'type' => 'Moderation',
@@ -461,9 +412,7 @@ public function tooltips( $tooltips ) {
}
}
- $tooltips['openai_endpoint_completions'] = __( 'Given a prompt, the model will return one or more predicted completions, and can also return the probabilities of alternative tokens at each position.', 'gravityforms-openai' );
$tooltips['openai_endpoint_chat_completions'] = __( 'Given a single message, the model will return a model-generated message as an output.', 'gravityforms-openai' );
- $tooltips['openai_endpoint_edits'] = __( 'Given a prompt and an instruction, the model will return an edited version of the prompt.', 'gravityforms-openai' );
$tooltips['openai_endpoint_moderations'] = __( 'Given a input text, outputs if the model classifies it as violating OpenAI\'s content policy.', 'gravityforms-openai' );
return $tooltips;
@@ -524,21 +473,11 @@ public function feed_settings_fields() {
'label' => __( 'OpenAI Endpoint', 'gravityforms-openai' ),
'type' => 'radio',
'choices' => array(
- array(
- 'value' => 'completions',
- 'label' => __( 'Completions', 'gravityforms-openai' ),
- 'tooltip' => 'openai_endpoint_completions',
- ),
array(
'value' => 'chat/completions',
'label' => __( 'Chat Completions', 'gravityforms-openai' ),
'tooltip' => 'openai_endpoint_chat_completions',
),
- array(
- 'value' => 'edits',
- 'label' => __( 'Edits', 'gravityforms-openai' ),
- 'tooltip' => 'openai_endpoint_edits',
- ),
array(
'value' => 'moderations',
'label' => __( 'Moderations', 'gravityforms-openai' ),
@@ -549,38 +488,6 @@ public function feed_settings_fields() {
),
),
),
- array(
- 'title' => 'Completions',
- 'fields' => array(
- array(
- 'name' => 'completions_model',
- 'tooltip' => 'Select the OpenAI model to use.',
- 'label' => __( 'OpenAI Model', 'gravityforms-openai' ),
- 'type' => 'radio',
- 'choices' => $this->get_openai_model_choices( 'completions' ),
- 'required' => true,
- ),
- array(
- 'name' => 'completions_prompt',
- 'tooltip' => 'Enter the prompt to send to OpenAI.',
- 'label' => 'Prompt',
- 'type' => 'textarea',
- 'class' => 'medium merge-tag-support mt-position-right',
- 'required' => true,
- ),
- $this->feed_setting_enable_merge_tag( 'completions' ),
- $this->feed_setting_map_result_to_field( 'completions' ),
- ),
- 'dependency' => array(
- 'live' => true,
- 'fields' => array(
- array(
- 'field' => 'endpoint',
- 'values' => array( 'completions' ),
- ),
- ),
- ),
- ),
array(
'title' => 'Chat Completions',
'fields' => array(
@@ -613,46 +520,6 @@ public function feed_settings_fields() {
),
),
),
- array(
- 'title' => 'Edits',
- 'fields' => array(
- array(
- 'name' => 'edits_model',
- 'tooltip' => 'Select the OpenAI model to use.',
- 'label' => __( 'OpenAI Model', 'gravityforms-openai' ),
- 'type' => 'radio',
- 'choices' => $this->get_openai_model_choices( 'edits' ),
- 'required' => true,
- ),
- array(
- 'name' => 'edits_input',
- 'tooltip' => __( 'The input text to use as a starting point for the edit.', 'gravityforms-openai' ),
- 'label' => 'Input',
- 'type' => 'textarea',
- 'class' => 'medium merge-tag-support mt-position-right',
- 'required' => false,
- ),
- array(
- 'name' => 'edits_instruction',
- 'tooltip' => __( 'The instruction that tells the model how to edit the prompt.', 'gravityforms-openai' ),
- 'label' => __( 'Instruction', 'gravityforms-openai' ),
- 'type' => 'textarea',
- 'class' => 'medium merge-tag-support mt-position-right',
- 'required' => true,
- ),
- $this->feed_setting_enable_merge_tag( 'edits' ),
- $this->feed_setting_map_result_to_field( 'edits' ),
- ),
- 'dependency' => array(
- 'live' => true,
- 'fields' => array(
- array(
- 'field' => 'endpoint',
- 'values' => array( 'edits' ),
- ),
- ),
- ),
- ),
array(
'title' => 'Moderations',
'fields' => array(
@@ -734,26 +601,6 @@ public function feed_settings_fields() {
),
),
),
- array(
- 'title' => 'Advanced Settings: Completions',
- 'fields' => array(
- $this->feed_advanced_setting_timeout( 'completions' ),
- $this->feed_advanced_setting_max_tokens( 'completions' ),
- $this->feed_advanced_setting_temperature( 'completions' ),
- $this->feed_advanced_setting_top_p( 'completions' ),
- $this->feed_advanced_setting_frequency_penalty( 'completions' ),
- $this->feed_advanced_setting_presence_penalty( 'completions' ),
- ),
- 'dependency' => array(
- 'live' => true,
- 'fields' => array(
- array(
- 'field' => 'endpoint',
- 'values' => array( 'completions' ),
- ),
- ),
- ),
- ),
array(
'title' => 'Advanced Settings: Chat Completions',
'fields' => array(
@@ -774,23 +621,6 @@ public function feed_settings_fields() {
),
),
),
- array(
- 'title' => 'Advanced Settings: Edits',
- 'fields' => array(
- $this->feed_advanced_setting_timeout( 'edits' ),
- $this->feed_advanced_setting_temperature( 'edits' ),
- $this->feed_advanced_setting_top_p( 'edits' ),
- ),
- 'dependency' => array(
- 'live' => true,
- 'fields' => array(
- array(
- 'field' => 'endpoint',
- 'values' => array( 'edits' ),
- ),
- ),
- ),
- ),
array(
'title' => 'Advanced Settings: Moderations',
'fields' => array(
@@ -819,10 +649,7 @@ public function feed_setting_enable_merge_tag( $endpoint ) {
'name' => $endpoint . '_enable_merge_tag',
'type' => 'checkbox',
'label' => __( 'Merge Tag', 'gravityforms-openai' ),
- 'description' => __( 'Enable getting the output of the OpenAI result using a merge tag.
-
- Pro Tip: This works with Gravity Forms Populate Anything\'s
- Live Merge Tags!', 'gravityforms-openai' ),
+ 'description' => __( 'Enable getting the output of the OpenAI result using a merge tag.', 'gravityforms-openai' ),
'choices' => array(
array(
'name' => $endpoint . '_enable_merge_tag',
@@ -1039,21 +866,14 @@ public function feed_advanced_setting_presence_penalty( $endpoint ) {
* @return array|void|null
*/
public function process_feed( $feed, $entry, $form ) {
+ $feed = $this->transform_feed( $feed );
$endpoint = $feed['meta']['endpoint'];
switch ( $endpoint ) {
- case 'completions':
- $entry = $this->process_endpoint_completions( $feed, $entry, $form );
- break;
-
case 'chat/completions':
$entry = $this->process_endpoint_chat_completions( $feed, $entry, $form );
break;
- case 'edits':
- $entry = $this->process_endpoint_edits( $feed, $entry, $form );
- break;
-
case 'moderations':
$this->process_endpoint_moderations( $feed, $entry, $form );
break;
@@ -1067,60 +887,6 @@ public function process_feed( $feed, $entry, $form ) {
return $entry;
}
- /**
- * Process completions endpoint.
- *
- * @param $feed array The current feed being processed.
- * @param $entry array The current entry being processed.
- * @param $form array The current form being processed.
- *
- * @return array Modified entry.
- */
- public function process_endpoint_completions( $feed, $entry, $form ) {
- $model = $feed['meta']['completions_model'];
- $prompt = $feed['meta']['completions_prompt'];
-
- // Parse the merge tags in the prompt.
- $prompt = GFCommon::replace_variables( $prompt, $form, $entry, false, false, false, 'text' );
-
- GFAPI::add_note( $entry['id'], 0, 'OpenAI Request (' . $feed['meta']['feed_name'] . ')', sprintf( __( 'Sent request to OpenAI completions endpoint.', 'gravityforms-openai' ) ) );
-
- // translators: placeholders are the feed name, model, prompt
- $this->log_debug( __METHOD__ . '(): ' . sprintf( __( 'Sent request to OpenAI. Feed: %1$s, Endpoint: completions, Model: %2$s, Prompt: %3$s', 'gravityforms-openai' ), $feed['meta']['feed_name'], $model, $prompt ) );
-
- $response = $this->make_request( 'completions', array(
- 'prompt' => $prompt,
- 'model' => $model,
- ), $feed );
-
- if ( is_wp_error( $response ) ) {
- // If there was an error, log it and return.
- $this->add_feed_error( $response->get_error_message(), $feed, $entry, $form );
- return $entry;
- }
-
- // Parse the response and add it as an entry note.
- $response_data = json_decode( $response['body'], true );
-
- if ( rgar( $response_data, 'error' ) ) {
- $this->add_feed_error( $response_data['error']['message'], $feed, $entry, $form );
- return $entry;
- }
-
- $text = $this->get_text_from_response( $response_data );
-
- if ( ! is_wp_error( $text ) ) {
- GFAPI::add_note( $entry['id'], 0, 'OpenAI Response (' . $feed['meta']['feed_name'] . ')', $text );
- $entry = $this->maybe_save_result_to_field( $feed, $entry, $form, $text );
- } else {
- $this->add_feed_error( $text->get_error_message(), $feed, $entry, $form );
- }
-
- gform_add_meta( $entry['id'], 'openai_response_' . $feed['id'], $response['body'] );
-
- return $entry;
- }
-
/**
* Process chat endpoint.
*
@@ -1183,64 +949,6 @@ public function process_endpoint_chat_completions( $feed, $entry, $form ) {
return $entry;
}
- /**
- * Process edits endpoint.
- *
- * @param $feed array The current feed being processed.
- * @param $entry array The current entry being processed.
- * @param $form array The current form being processed.
- *
- * @return array Modified entry.
- */
- public function process_endpoint_edits( $feed, $entry, $form ) {
- $model = $feed['meta']['edits_model'];
- $input = $feed['meta']['edits_input'];
- $instruction = $feed['meta']['edits_instruction'];
-
- // Parse the merge tags in the input and instruction
- $input = GFCommon::replace_variables( $input, $form, $entry, false, false, false, 'text' );
- $instruction = GFCommon::replace_variables( $instruction, $form, $entry, false, false, false, 'text' );
-
- GFAPI::add_note( $entry['id'], 0, 'OpenAI Request (' . $feed['meta']['feed_name'] . ')', sprintf( __( 'Sent request to OpenAI edits endpoint.', 'gravityforms-openai' ) ) );
-
- // translators: placeholders are the feed name, model, prompt
- $this->log_debug( __METHOD__ . '(): ' . sprintf( __( 'Sent request to OpenAI. Feed: %1$s, Endpoint: edits, Model: %2$s, Input: %3$s, instruction: %4$s', 'gravityforms-openai' ), $feed['meta']['feed_name'], $model, $input, $instruction ) );
-
- $response = $this->make_request( 'edits', array(
- 'input' => $input,
- 'instruction' => $instruction,
- 'model' => $model,
- ), $feed );
-
- if ( is_wp_error( $response ) ) {
- // If there was an error, log it and return.
- $this->add_feed_error( $response->get_error_message(), $feed, $entry, $form );
- return $entry;
- }
-
- // Parse the response and add it as an entry note.
- $response_data = json_decode( $response['body'], true );
-
- if ( rgar( $response_data, 'error' ) ) {
- $this->add_feed_error( $response_data['error']['message'], $feed, $entry, $form );
- return $entry;
- }
-
- $text = $this->get_text_from_response( $response_data );
-
- if ( ! is_wp_error( $text ) ) {
- GFAPI::add_note( $entry['id'], 0, 'OpenAI Response (' . $feed['meta']['feed_name'] . ')', $text );
- $entry = $this->maybe_save_result_to_field( $feed, $entry, $form, $text );
- } else {
- $this->add_feed_error( $text->get_error_message(), $feed, $entry, $form );
- }
-
- gform_add_meta( $entry['id'], 'openai_response_' . $feed['id'], $response['body'] );
-
- return $entry;
- }
-
-
/**
* Saves the result to the selected field if configured.
*
@@ -1553,27 +1261,6 @@ public function get_merge_tag_replacement( $form, $entry, $feed_id, $url_encode,
$response_data = array();
switch ( $endpoint ) {
- case 'completions':
- $model = $feed['meta']['completions_model'];
- $prompt = $feed['meta']['completions_prompt'];
-
- $prompt = GFCommon::replace_variables( $prompt, $form, $entry, false, false, false, 'text' );
-
- // If prompt is empty, do not generate any completion response, skip with blank.
- if ( empty( $prompt ) ) {
- return '';
- }
-
- $response = $this->make_request( 'completions', array(
- 'model' => $model,
- 'prompt' => $prompt,
- ), $feed );
-
- if ( ! is_wp_error( $response ) ) {
- $response_data = json_decode( $response['body'], true );
- }
- break;
-
case 'chat/completions':
$model = $feed['meta']['chat_completions_model'];
$message = $feed['meta']['chat_completions_message'];
@@ -1603,30 +1290,6 @@ public function get_merge_tag_replacement( $form, $entry, $feed_id, $url_encode,
}
break;
- case 'edits':
- $model = $feed['meta']['edits_model'];
- $input = $feed['meta']['edits_input'];
- $instruction = $feed['meta']['edits_instruction'];
-
- $input = GFCommon::replace_variables( $input, $form, $entry, false, false, false, 'text' );
- $instruction = GFCommon::replace_variables( $instruction, $form, $entry, false, false, false, 'text' );
-
- // If input or instruction is empty, do not generate any edit response, skip with blank.
- if ( empty( $input ) || empty( $instruction ) ) {
- return '';
- }
-
- $response = $this->make_request( 'edits', array(
- 'model' => $model,
- 'input' => $input,
- 'instruction' => $instruction,
- ), $feed );
-
- if ( ! is_wp_error( $response ) ) {
- $response_data = json_decode( $response['body'], true );
- }
- break;
-
default:
return '';
}
@@ -1705,14 +1368,6 @@ public function make_request( $endpoint, $body, $feed ) {
}
switch ( $endpoint ) {
- case 'completions':
- $body['max_tokens'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'max_tokens', $this->default_settings['completions']['max_tokens'] );
- $body['temperature'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'temperature', $this->default_settings['completions']['temperature'] );
- $body['top_p'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'top_p', $this->default_settings['completions']['top_p'] );
- $body['frequency_penalty'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'frequency_penalty', $this->default_settings['completions']['frequency_penalty'] );
- $body['presence_penalty'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'presence_penalty', $this->default_settings['completions']['presence_penalty'] );
- break;
-
case 'chat/completions':
$body['max_tokens'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'max_tokens', $this->default_settings['chat/completions']['max_tokens'] );
$body['temperature'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'temperature', $this->default_settings['chat/completions']['temperature'] );
@@ -1720,11 +1375,6 @@ public function make_request( $endpoint, $body, $feed ) {
$body['frequency_penalty'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'frequency_penalty', $this->default_settings['chat/completions']['frequency_penalty'] );
$body['presence_penalty'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'presence_penalty', $this->default_settings['chat/completions']['presence_penalty'] );
break;
-
- case 'edits':
- $body['temperature'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'temperature', $this->default_settings['edits']['temperature'] );
- $body['top_p'] = (float) rgar( $feed['meta'], $endpoint . '_' . 'top_p', $this->default_settings['edits']['top_p'] );
- break;
}
$body = apply_filters( 'gf_openai_request_body', $body, $endpoint, $feed );
@@ -1792,6 +1442,51 @@ public function get_headers() {
return $headers;
}
+ public function get_feed( $id ) {
+ $feed = parent::get_feed( $id );
+
+ if ( $this->is_feed_edit_page() ) {
+ $message = $this->get_transform_message( $feed );
+ GFCommon::add_dismissible_message( $message, $id . '_transform_feed_notice' );
+ }
+
+ return $this->transform_feed( $feed );
+ }
+
+ /**
+ * Transforms completion or edits feed to Chat\Completion
+ *
+ * @param array $feed The feed being processed.
+ *
+ * @return array
+ */
+ public function transform_feed( $feed ) {
+ $endpoint = rgars( $feed, 'meta/endpoint' );
+
+ if ( $endpoint === 'completions' ) {
+ $completion_message = rgars( $feed, 'meta/completions_prompt' );
+ $feed['meta'] = array_merge( $feed['meta'], array( 'endpoint' => 'chat/completions', 'chat_completions_model' => 'gpt-3.5-turbo', 'chat_completions_message' => $completion_message ) );
+ }
+
+ if ( $endpoint === 'edits' ) {
+ $completion_message = rgars( $feed, 'meta/edits_instruction' ) . "\r\n\r\n" . rgars( $feed, 'meta/edits_input' );
+ $feed['meta'] = array_merge( $feed['meta'], array( 'endpoint' => 'chat/completions', 'chat_completions_model' => 'gpt-3.5-turbo', 'chat_completions_message' => $completion_message ) );
+ }
+
+ return $feed;
+ }
+
+ /**
+ * @param array $feed The feed being processed.
+ *
+ * @return string
+ */
+ public function get_transform_message( $feed ) {
+ $endpoint = rgars( $feed, 'meta/endpoint' );
+
+ return sprintf( __( 'This feed has been updated to use the "Chat Completions" endpoint, replacing the "%s" endpoint which has been deprecated by OpenAI.', 'gravityforms-openai' ), ucfirst( $endpoint ) );
+ }
+
/**
* Export OpenAI Add-On feeds when exporting forms.
*