Merge branch 'ai-forms'
This commit is contained in:
246
app/Console/Commands/GenerateTemplate.php
Normal file
246
app/Console/Commands/GenerateTemplate.php
Normal file
@@ -0,0 +1,246 @@
|
||||
<?php
|
||||
|
||||
namespace App\Console\Commands;
|
||||
|
||||
use App\Models\Template;
|
||||
use App\Service\OpenAi\GptCompleter;
|
||||
use Illuminate\Console\Command;
|
||||
use Illuminate\Support\Facades\Http;
|
||||
use Illuminate\Support\Str;
|
||||
|
||||
class GenerateTemplate extends Command
|
||||
{
|
||||
/**
|
||||
* The name and signature of the console command.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $signature = 'ai:make-form-template {prompt}';
|
||||
|
||||
/**
|
||||
* The console command description.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $description = 'Generates a new form template from a prompt';
|
||||
|
||||
const FORM_STRUCTURE_PROMPT = <<<EOD
|
||||
I created a form builder. Forms are represented as Json objects. Here's an example form:
|
||||
```json
|
||||
{
|
||||
"title": "Contact Us",
|
||||
"properties": [
|
||||
{
|
||||
"help": null,
|
||||
"name": "What's your name?",
|
||||
"type": "text",
|
||||
"hidden": false,
|
||||
"prefill": null,
|
||||
"required": true,
|
||||
"placeholder": null
|
||||
},
|
||||
{
|
||||
"help": null,
|
||||
"name": "Email",
|
||||
"type": "email",
|
||||
"hidden": false,
|
||||
"prefill": null,
|
||||
"required": true,
|
||||
"placeholder": null
|
||||
},
|
||||
{
|
||||
"help": null,
|
||||
"name": "How would you rate your overall experience?",
|
||||
"type": "select",
|
||||
"hidden": false,
|
||||
"select": {
|
||||
"options": [
|
||||
{
|
||||
"id": "Below Average",
|
||||
"name": "Below Average"
|
||||
},
|
||||
{
|
||||
"id": "Average",
|
||||
"name": "Average"
|
||||
},
|
||||
{
|
||||
"id": "Above Average",
|
||||
"name": "Above Average"
|
||||
}
|
||||
]
|
||||
},
|
||||
"prefill": null,
|
||||
"required": true,
|
||||
"placeholder": null,
|
||||
},
|
||||
{
|
||||
"help": null,
|
||||
"name": "Subject",
|
||||
"type": "text",
|
||||
"hidden": false,
|
||||
"prefill": null,
|
||||
"required": true,
|
||||
"placeholder": null
|
||||
},
|
||||
{
|
||||
"help": null,
|
||||
"name": "How can we help?",
|
||||
"type": "text",
|
||||
"hidden": false,
|
||||
"prefill": null,
|
||||
"required": true,
|
||||
"multi_lines": true,
|
||||
"placeholder": null,
|
||||
"generates_uuid": false,
|
||||
"max_char_limit": "2000",
|
||||
"hide_field_name": false,
|
||||
"show_char_limit": false,
|
||||
"generates_auto_increment_id": false
|
||||
},
|
||||
{
|
||||
"help": null,
|
||||
"name": "Have any attachments?",
|
||||
"type": "files",
|
||||
"hidden": false,
|
||||
"prefill": null,
|
||||
"placeholder": null
|
||||
}
|
||||
],
|
||||
"description": "<p>Looking for a real person to speak to?</p><p>We're here for you! Just drop in your queries below and we'll connect with you as soon as we can.</p>",
|
||||
"re_fillable": false,
|
||||
"use_captcha": false,
|
||||
"redirect_url": null,
|
||||
"submitted_text": "<p>Great, we've received your message. We'll get back to you as soon as we can :)</p>",
|
||||
"uppercase_labels": false,
|
||||
"submit_button_text": "Submit",
|
||||
"re_fill_button_text": "Fill Again",
|
||||
"color": "#3B82F6"
|
||||
}
|
||||
```
|
||||
The form properties can have one of the following types: 'text', 'number', 'select', 'multi_select', 'date', 'files', 'checkbox', 'url', 'email', 'phone_number'.
|
||||
All form properties objects need to have the keys 'help', 'name', 'type', 'hidden', 'placeholder', 'prefill'.
|
||||
|
||||
For the type "select" and "multi_select", the input object must have a key "select" (or "multi_select") that's mapped to an object like this one:
|
||||
```json
|
||||
{
|
||||
"options": [
|
||||
{"id":"Option 1","name":"Option 1"},
|
||||
{"id":"Pption 2","name":"Option 2"}
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
For the type "number" you can set the property "is_rating" to "true" to turn it into a star rating input.
|
||||
|
||||
If the form is too long, you can paginate it by adding a page break block in the list of properties:
|
||||
```json
|
||||
{
|
||||
"name":"Page Break",
|
||||
"next_btn_text":"Next",
|
||||
"previous_btn_text":"Previous",
|
||||
"type":"nf-page-break",
|
||||
}
|
||||
```
|
||||
|
||||
Give me the JSON code only, for the following form: "[REPLACE]"
|
||||
Do not ask me for more information about required properties or types, suggest me a form structure instead.
|
||||
EOD;
|
||||
|
||||
const FORM_DESCRIPTION_PROMPT = <<<EOD
|
||||
I own a form builder online named OpnForm. It's free to use. Give me a description for a template page for the following form: [REPLACE]. Explain what the form is about, and that it takes seconds to duplicate the template to create your own version it and to start getting some submissions.
|
||||
EOD;
|
||||
|
||||
const FORM_QAS_PROMPT = <<<EOD
|
||||
Now give me 3 to 5 question and answers to put on the form template page. The questions should be about the reasons for this template (when to use, why, target audience, goal etc.) and OpnForm's usage. Reply only with a valid JSON, being an array of object containing the keys "question" and "answer".
|
||||
EOD;
|
||||
|
||||
const FORM_TITLE_PROMPT = <<<EOD
|
||||
Finally give me a title for the template. It should be short and to the point, without any quotes.
|
||||
EOD;
|
||||
|
||||
const FORM_IMG_KEYWORDS_PROMPT = <<<EOD
|
||||
I want to add an image to illustrate this form template page. Give me a releveant search query for unsplash. Reply only with a valid JSON like this:
|
||||
```json
|
||||
{
|
||||
"search_query": ""
|
||||
}
|
||||
```
|
||||
EOD;
|
||||
|
||||
/**
|
||||
* Execute the console command.
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function handle()
|
||||
{
|
||||
// Get form structture
|
||||
$completer = (new GptCompleter(config('services.openai.api_key')))
|
||||
->setSystemMessage('You are a robot helping to generate forms.');
|
||||
$completer->completeChat([
|
||||
["role" => "user", "content" => Str::of(self::FORM_STRUCTURE_PROMPT)->replace('[REPLACE]', $this->argument('prompt'))->toString()]
|
||||
], 3000);
|
||||
$formData = $completer->getArray();
|
||||
|
||||
// Now get description and QAs
|
||||
$formDescriptionPrompt = Str::of(self::FORM_DESCRIPTION_PROMPT)->replace('[REPLACE]', $this->argument('prompt'))->toString();
|
||||
$formDescription = $completer->completeChat([
|
||||
["role" => "user", "content" => $formDescriptionPrompt]
|
||||
])->getString();
|
||||
$formQAs = $completer->completeChat([
|
||||
["role" => "user", "content" => $formDescriptionPrompt],
|
||||
["role" => "assistant", "content" => $formDescription],
|
||||
["role" => "user", "content" => self::FORM_QAS_PROMPT]
|
||||
])->getArray();
|
||||
$formTitle = $completer->completeChat([
|
||||
["role" => "user", "content" => $formDescriptionPrompt],
|
||||
["role" => "assistant", "content" => $formDescription],
|
||||
["role" => "user", "content" => self::FORM_TITLE_PROMPT]
|
||||
])->getString();
|
||||
|
||||
// Finally get keyworks for image cover
|
||||
$formCoverKeyworks = $completer->completeChat([
|
||||
["role" => "user", "content" => $formDescriptionPrompt],
|
||||
["role" => "assistant", "content" => $formDescription],
|
||||
["role" => "user", "content" => self::FORM_IMG_KEYWORDS_PROMPT]
|
||||
])->getArray();
|
||||
$imageUrl = $this->getImageCoverUrl($formCoverKeyworks['search_query']);
|
||||
|
||||
$template = $this->createFormTemplate($formData, $formTitle, $formDescription, $formQAs, $imageUrl);
|
||||
$this->info('/templates/' . $template->slug);
|
||||
|
||||
return Command::SUCCESS;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get an image cover URL for the template using unsplash API
|
||||
*/
|
||||
private function getImageCoverUrl($searchQuery): ?string
|
||||
{
|
||||
$url = 'https://api.unsplash.com/search/photos?query=' . urlencode($searchQuery) . '&client_id=' . config('services.unslash.access_key');
|
||||
$response = Http::get($url)->json();
|
||||
if (isset($response['results'][0]['urls']['regular'])) {
|
||||
return $response['results'][0]['urls']['regular'];
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
private function createFormTemplate(array $formData, string $formTitle, string $formDescription, array $formQAs, ?string $imageUrl)
|
||||
{
|
||||
// Add property uuids
|
||||
foreach ($formData['properties'] as &$property) {
|
||||
$property['id'] = Str::uuid()->toString();
|
||||
}
|
||||
|
||||
// Clean data
|
||||
$formTitle = Str::of($formTitle)->replace('"', '')->toString();
|
||||
|
||||
return Template::create([
|
||||
'name' => $formTitle,
|
||||
'description' => $formDescription,
|
||||
'questions' => $formQAs,
|
||||
'structure' => $formData,
|
||||
'image_url' => $imageUrl,
|
||||
]);
|
||||
}
|
||||
}
|
||||
38
app/Http/Controllers/Forms/AiFormController.php
Normal file
38
app/Http/Controllers/Forms/AiFormController.php
Normal file
@@ -0,0 +1,38 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers\Forms;
|
||||
|
||||
use App\Console\Commands\GenerateTemplate;
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Http\Requests\AiGenerateFormRequest;
|
||||
use App\Service\OpenAi\GptCompleter;
|
||||
use Illuminate\Support\Str;
|
||||
|
||||
class AiFormController extends Controller
|
||||
{
|
||||
public function generateForm(AiGenerateFormRequest $request)
|
||||
{
|
||||
$this->middleware('throttle:4,1');
|
||||
$completer = (new GptCompleter(config('services.openai.api_key')))
|
||||
->setSystemMessage('You are a robot helping to generate forms.');
|
||||
$completer->completeChat([
|
||||
["role" => "user", "content" => Str::of(GenerateTemplate::FORM_STRUCTURE_PROMPT)
|
||||
->replace('[REPLACE]', $request->form_prompt)->toString()]
|
||||
], 3000);
|
||||
|
||||
return $this->success([
|
||||
'message' => 'Form successfully generated!',
|
||||
'form' => $this->cleanOutput($completer->getArray())
|
||||
]);
|
||||
}
|
||||
|
||||
private function cleanOutput($formData)
|
||||
{
|
||||
// Add property uuids
|
||||
foreach ($formData['properties'] as &$property) {
|
||||
$property['id'] = Str::uuid()->toString();
|
||||
}
|
||||
|
||||
return $formData;
|
||||
}
|
||||
}
|
||||
@@ -8,8 +8,6 @@ class SpaController extends Controller
|
||||
{
|
||||
/**
|
||||
* Get the SPA view.
|
||||
*
|
||||
* @return \Illuminate\Http\Response
|
||||
*/
|
||||
public function __invoke(Request $request)
|
||||
{
|
||||
|
||||
20
app/Http/Requests/AiGenerateFormRequest.php
Normal file
20
app/Http/Requests/AiGenerateFormRequest.php
Normal file
@@ -0,0 +1,20 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
|
||||
class AiGenerateFormRequest extends FormRequest
|
||||
{
|
||||
/**
|
||||
* Get the validation rules that apply to the request.
|
||||
*
|
||||
* @return array<string, mixed>
|
||||
*/
|
||||
public function rules()
|
||||
{
|
||||
return [
|
||||
'form_prompt' => 'required|string'
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -4,11 +4,13 @@ namespace App\Models;
|
||||
|
||||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
use Spatie\Sluggable\HasSlug;
|
||||
use Spatie\Sluggable\SlugOptions;
|
||||
use Stevebauman\Purify\Facades\Purify;
|
||||
|
||||
class Template extends Model
|
||||
{
|
||||
use HasFactory;
|
||||
use HasFactory, HasSlug;
|
||||
|
||||
protected $fillable = [
|
||||
'name',
|
||||
@@ -29,4 +31,15 @@ class Template extends Model
|
||||
// Strip out unwanted html
|
||||
$this->attributes['description'] = Purify::clean($value);
|
||||
}
|
||||
|
||||
/**
|
||||
* Config/options
|
||||
*/
|
||||
public function getSlugOptions(): SlugOptions
|
||||
{
|
||||
return SlugOptions::create()
|
||||
->doNotGenerateSlugsOnUpdate()
|
||||
->generateSlugsFrom('name')
|
||||
->saveSlugsTo('slug');
|
||||
}
|
||||
}
|
||||
|
||||
126
app/Service/OpenAi/GptCompleter.php
Normal file
126
app/Service/OpenAi/GptCompleter.php
Normal file
@@ -0,0 +1,126 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service\OpenAi;
|
||||
|
||||
use App\Service\OpenAi\Utils\JsonFixer;
|
||||
use Illuminate\Support\Facades\Log;
|
||||
use Illuminate\Support\Str;
|
||||
use OpenAI\Client;
|
||||
use OpenAI\Exceptions\ErrorException;
|
||||
|
||||
/**
|
||||
* Handles a GPT completion prompt with or without insert tag.
|
||||
* Also parses output.
|
||||
*/
|
||||
class GptCompleter
|
||||
{
|
||||
const AI_MODEL = 'gpt-3.5-turbo';
|
||||
|
||||
protected Client $openAi;
|
||||
protected mixed $result;
|
||||
protected array $completionInput;
|
||||
protected ?string $systemMessage;
|
||||
|
||||
protected int $tokenUsed = 0;
|
||||
|
||||
public function __construct(string $apiKey, protected int $retries = 2)
|
||||
{
|
||||
$this->openAi = \OpenAI::client($apiKey);
|
||||
}
|
||||
|
||||
public function setSystemMessage(string $systemMessage): self
|
||||
{
|
||||
$this->systemMessage = $systemMessage;
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function completeChat(array $messages, int $maxTokens = 512, float $temperature = 0.81): self
|
||||
{
|
||||
$this->computeChatCompletion($messages, $maxTokens, $temperature)
|
||||
->queryCompletion();
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getBool(): bool
|
||||
{
|
||||
switch (strtolower($this->result)) {
|
||||
case 'true':
|
||||
return true;
|
||||
case 'false':
|
||||
return false;
|
||||
default:
|
||||
throw new \InvalidArgumentException("Expected a boolean value, got {$this->result}");
|
||||
}
|
||||
}
|
||||
|
||||
public function getArray(): array
|
||||
{
|
||||
$payload = Str::of($this->result)->trim();
|
||||
if ($payload->contains('```json')) {
|
||||
$payload = $payload->after('```json')->before('```');
|
||||
} else if ($payload->contains('```')) {
|
||||
$payload = $payload->after('```')->before('```');
|
||||
}
|
||||
$payload = $payload->toString();
|
||||
$exception = null;
|
||||
|
||||
for ($i = 0; $i < $this->retries; $i++) {
|
||||
try {
|
||||
$payload = (new JsonFixer)->fix($payload);
|
||||
return json_decode($payload, true);
|
||||
} catch (\Aws\Exception\InvalidJsonException $e) {
|
||||
$exception = $e;
|
||||
Log::warning("Invalid JSON, retrying:");
|
||||
Log::warning($payload);
|
||||
Log::warning(json_encode($this->completionInput));
|
||||
$this->queryCompletion();
|
||||
}
|
||||
}
|
||||
throw $exception;
|
||||
}
|
||||
|
||||
public function getString(): string
|
||||
{
|
||||
return trim($this->result);
|
||||
}
|
||||
|
||||
public function getTokenUsed(): int
|
||||
{
|
||||
return $this->tokenUsed;
|
||||
}
|
||||
|
||||
protected function computeChatCompletion(array $messages, int $maxTokens = 512, float $temperature = 0.81): self
|
||||
{
|
||||
if (isset($this->systemMessage) && $messages[0]['role'] !== 'system') {
|
||||
$messages = array_merge([[
|
||||
'role' => 'system',
|
||||
'content' => $this->systemMessage
|
||||
]], $messages);
|
||||
}
|
||||
|
||||
$completionInput = [
|
||||
'model' => self::AI_MODEL,
|
||||
'messages' => $messages,
|
||||
'max_tokens' => $maxTokens,
|
||||
'temperature' => $temperature
|
||||
];
|
||||
|
||||
$this->completionInput = $completionInput;
|
||||
return $this;
|
||||
}
|
||||
|
||||
protected function queryCompletion(): self {
|
||||
try {
|
||||
Log::debug("Open AI query: " . json_encode($this->completionInput));
|
||||
$response = $this->openAi->chat()->create($this->completionInput);
|
||||
} catch (ErrorException $errorException) {
|
||||
// Retry once
|
||||
Log::warning("Open AI error, retrying: {$errorException->getMessage()}");
|
||||
$response = $this->openAi->chat()->create($this->completionInput);
|
||||
}
|
||||
$this->tokenUsed += $response->usage->totalTokens;
|
||||
$this->result = $response->choices[0]->message->content;
|
||||
return $this;
|
||||
}
|
||||
}
|
||||
272
app/Service/OpenAi/Utils/JsonFixer.php
Normal file
272
app/Service/OpenAi/Utils/JsonFixer.php
Normal file
@@ -0,0 +1,272 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service\OpenAi\Utils;
|
||||
|
||||
/*
|
||||
* This file is part of the PHP-JSON-FIXER package.
|
||||
*
|
||||
* (c) Jitendra Adhikari <jiten.adhikary@gmail.com>
|
||||
* <https://github.com/adhocore>
|
||||
*
|
||||
* Licensed under MIT license.
|
||||
*/
|
||||
|
||||
use App\Exceptions\Coursework\InvalidJsonException;
|
||||
|
||||
/**
|
||||
* Attempts to fix truncated JSON by padding contextual counterparts at the end.
|
||||
*
|
||||
* @author Jitendra Adhikari <jiten.adhikary@gmail.com>
|
||||
* @license MIT
|
||||
*
|
||||
* @link https://github.com/adhocore/php-json-fixer
|
||||
*/
|
||||
class JsonFixer
|
||||
{
|
||||
use PadsJson;
|
||||
|
||||
/** @var array Current token stack indexed by position */
|
||||
protected $stack = [];
|
||||
|
||||
/** @var bool If current char is within a string */
|
||||
protected $inStr = false;
|
||||
|
||||
/** @var bool Whether to throw Exception on failure */
|
||||
protected $silent = false;
|
||||
|
||||
/** @var array The complementary pairs */
|
||||
protected $pairs = [
|
||||
'{' => '}',
|
||||
'[' => ']',
|
||||
'"' => '"',
|
||||
];
|
||||
|
||||
/** @var int The last seen object `{` type position */
|
||||
protected $objectPos = -1;
|
||||
|
||||
/** @var int The last seen array `[` type position */
|
||||
protected $arrayPos = -1;
|
||||
|
||||
/** @var string Missing value. (Options: true, false, null) */
|
||||
protected $missingValue = 'null';
|
||||
|
||||
/**
|
||||
* Set/unset silent mode.
|
||||
*
|
||||
* @param bool $silent
|
||||
*
|
||||
* @return $this
|
||||
*/
|
||||
public function silent($silent = true)
|
||||
{
|
||||
$this->silent = (bool)$silent;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set missing value.
|
||||
*
|
||||
* @param mixed $value
|
||||
*
|
||||
* @return $this
|
||||
*/
|
||||
public function missingValue($value)
|
||||
{
|
||||
if ($value === null) {
|
||||
$value = 'null';
|
||||
} elseif (\is_bool($value)) {
|
||||
$value = $value ? 'true' : 'false';
|
||||
}
|
||||
|
||||
$this->missingValue = $value;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fix the truncated JSON.
|
||||
*
|
||||
* @param string $json The JSON string to fix.
|
||||
*
|
||||
* @return string Fixed JSON. If failed with silent then original JSON.
|
||||
* @throws InvalidJsonException When fixing fails.
|
||||
*
|
||||
*/
|
||||
public function fix($json)
|
||||
{
|
||||
list($head, $json, $tail) = $this->trim($json);
|
||||
|
||||
if (empty($json) || $this->isValid($json)) {
|
||||
return $json;
|
||||
}
|
||||
|
||||
if (null !== $tmpJson = $this->quickFix($json)) {
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
$this->reset();
|
||||
|
||||
return $head . $this->doFix($json) . $tail;
|
||||
}
|
||||
|
||||
protected function trim($json)
|
||||
{
|
||||
\preg_match('/^(\s*)([^\s]+)(\s*)$/', $json, $match);
|
||||
|
||||
$match += ['', '', '', ''];
|
||||
$match[2] = \trim($json);
|
||||
|
||||
\array_shift($match);
|
||||
|
||||
return $match;
|
||||
}
|
||||
|
||||
protected function isValid($json)
|
||||
{
|
||||
\json_decode($json);
|
||||
|
||||
return \JSON_ERROR_NONE === \json_last_error();
|
||||
}
|
||||
|
||||
protected function quickFix($json)
|
||||
{
|
||||
if (\strlen($json) === 1 && isset($this->pairs[$json])) {
|
||||
return $json . $this->pairs[$json];
|
||||
}
|
||||
|
||||
if ($json[0] !== '"') {
|
||||
return $this->maybeLiteral($json);
|
||||
}
|
||||
|
||||
return $this->padString($json);
|
||||
}
|
||||
|
||||
protected function reset()
|
||||
{
|
||||
$this->stack = [];
|
||||
$this->inStr = false;
|
||||
$this->objectPos = -1;
|
||||
$this->arrayPos = -1;
|
||||
}
|
||||
|
||||
protected function maybeLiteral($json)
|
||||
{
|
||||
if (!\in_array($json[0], ['t', 'f', 'n'])) {
|
||||
return null;
|
||||
}
|
||||
|
||||
foreach (['true', 'false', 'null'] as $literal) {
|
||||
if (\strpos($literal, $json) === 0) {
|
||||
return $literal;
|
||||
}
|
||||
}
|
||||
|
||||
// @codeCoverageIgnoreStart
|
||||
return null;
|
||||
// @codeCoverageIgnoreEnd
|
||||
}
|
||||
|
||||
protected function doFix($json)
|
||||
{
|
||||
list($index, $char) = [-1, ''];
|
||||
|
||||
while (isset($json[++$index])) {
|
||||
list($prev, $char) = [$char, $json[$index]];
|
||||
|
||||
$next = isset($json[$index + 1]) ? $json[$index + 1] : '';
|
||||
|
||||
if (!\in_array($char, [' ', "\n", "\r"])) {
|
||||
$this->stack($prev, $char, $index, $next);
|
||||
}
|
||||
}
|
||||
|
||||
return $this->fixOrFail($json);
|
||||
}
|
||||
|
||||
protected function stack($prev, $char, $index, $next)
|
||||
{
|
||||
if ($this->maybeStr($prev, $char, $index)) {
|
||||
return;
|
||||
}
|
||||
|
||||
$last = $this->lastToken();
|
||||
|
||||
if (\in_array($last, [',', ':', '"']) && \preg_match('/\"|\d|\{|\[|t|f|n/', $char)) {
|
||||
$this->popToken();
|
||||
}
|
||||
|
||||
if (\in_array($char, [',', ':', '[', '{'])) {
|
||||
$this->stack[$index] = $char;
|
||||
}
|
||||
|
||||
$this->updatePos($char, $index);
|
||||
}
|
||||
|
||||
protected function lastToken()
|
||||
{
|
||||
return \end($this->stack);
|
||||
}
|
||||
|
||||
protected function popToken($token = null)
|
||||
{
|
||||
// Last one
|
||||
if (null === $token) {
|
||||
return \array_pop($this->stack);
|
||||
}
|
||||
|
||||
$keys = \array_reverse(\array_keys($this->stack));
|
||||
foreach ($keys as $key) {
|
||||
if ($this->stack[$key] === $token) {
|
||||
unset($this->stack[$key]);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
protected function maybeStr($prev, $char, $index)
|
||||
{
|
||||
if ($prev !== '\\' && $char === '"') {
|
||||
$this->inStr = !$this->inStr;
|
||||
}
|
||||
|
||||
if ($this->inStr && $this->lastToken() !== '"') {
|
||||
$this->stack[$index] = '"';
|
||||
}
|
||||
|
||||
return $this->inStr;
|
||||
}
|
||||
|
||||
protected function updatePos($char, $index)
|
||||
{
|
||||
if ($char === '{') {
|
||||
$this->objectPos = $index;
|
||||
} elseif ($char === '}') {
|
||||
$this->popToken('{');
|
||||
$this->objectPos = -1;
|
||||
} elseif ($char === '[') {
|
||||
$this->arrayPos = $index;
|
||||
} elseif ($char === ']') {
|
||||
$this->popToken('[');
|
||||
$this->arrayPos = -1;
|
||||
}
|
||||
}
|
||||
|
||||
protected function fixOrFail($json)
|
||||
{
|
||||
$length = \strlen($json);
|
||||
$tmpJson = $this->pad($json);
|
||||
|
||||
if ($this->isValid($tmpJson)) {
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
if ($this->silent) {
|
||||
return $json;
|
||||
}
|
||||
|
||||
throw new InvalidJsonException(
|
||||
\sprintf('Could not fix JSON (tried padding `%s`)', \substr($tmpJson, $length), $json)
|
||||
);
|
||||
}
|
||||
}
|
||||
113
app/Service/OpenAi/Utils/PadsJson.php
Normal file
113
app/Service/OpenAi/Utils/PadsJson.php
Normal file
@@ -0,0 +1,113 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service\OpenAi\Utils;
|
||||
|
||||
/**
|
||||
* Attempts to fix truncated JSON by padding contextual counterparts at the end.
|
||||
*
|
||||
* @author Jitendra Adhikari <jiten.adhikary@gmail.com>
|
||||
* @license MIT
|
||||
*
|
||||
* @internal
|
||||
*
|
||||
* @link https://github.com/adhocore/php-json-fixer
|
||||
*/
|
||||
trait PadsJson
|
||||
{
|
||||
public function pad($tmpJson)
|
||||
{
|
||||
if (!$this->inStr) {
|
||||
$tmpJson = \rtrim($tmpJson, ',');
|
||||
while ($this->lastToken() === ',') {
|
||||
$this->popToken();
|
||||
}
|
||||
}
|
||||
|
||||
$tmpJson = $this->padLiteral($tmpJson);
|
||||
$tmpJson = $this->padObject($tmpJson);
|
||||
|
||||
return $this->padStack($tmpJson);
|
||||
}
|
||||
|
||||
protected function padLiteral($tmpJson)
|
||||
{
|
||||
if ($this->inStr) {
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
$match = \preg_match('/(tr?u?e?|fa?l?s?e?|nu?l?l?)$/', $tmpJson, $matches);
|
||||
|
||||
if (!$match || null === $literal = $this->maybeLiteral($matches[1])) {
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
return \substr($tmpJson, 0, 0 - \strlen($matches[1])) . $literal;
|
||||
}
|
||||
|
||||
protected function padStack($tmpJson)
|
||||
{
|
||||
foreach (\array_reverse($this->stack, true) as $token) {
|
||||
if (isset($this->pairs[$token])) {
|
||||
$tmpJson .= $this->pairs[$token];
|
||||
}
|
||||
}
|
||||
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
protected function padObject($tmpJson)
|
||||
{
|
||||
if (!$this->objectNeedsPadding($tmpJson)) {
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
$part = \substr($tmpJson, $this->objectPos + 1);
|
||||
if (\preg_match('/(\s*\"[^"]+\"\s*:\s*[^,]+,?)+$/', $part, $matches)) {
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
if ($this->inStr) {
|
||||
$tmpJson .= '"';
|
||||
}
|
||||
|
||||
$tmpJson = $this->padIf($tmpJson, ':');
|
||||
$tmpJson = $tmpJson . $this->missingValue;
|
||||
|
||||
if ($this->lastToken() === '"') {
|
||||
$this->popToken();
|
||||
}
|
||||
|
||||
return $tmpJson;
|
||||
}
|
||||
|
||||
protected function objectNeedsPadding($tmpJson)
|
||||
{
|
||||
$last = \substr($tmpJson, -1);
|
||||
$empty = $last === '{' && !$this->inStr;
|
||||
|
||||
return !$empty && $this->arrayPos < $this->objectPos;
|
||||
}
|
||||
|
||||
protected function padString($string)
|
||||
{
|
||||
$last = \substr($string, -1);
|
||||
$last2 = \substr($string, -2);
|
||||
|
||||
if ($last2 === '\"' || $last !== '"') {
|
||||
return $string . '"';
|
||||
}
|
||||
|
||||
// @codeCoverageIgnoreStart
|
||||
return null;
|
||||
// @codeCoverageIgnoreEnd
|
||||
}
|
||||
|
||||
protected function padIf($string, $substr)
|
||||
{
|
||||
if (\substr($string, 0 - \strlen($substr)) !== $substr) {
|
||||
return $string . $substr;
|
||||
}
|
||||
|
||||
return $string;
|
||||
}
|
||||
}
|
||||
@@ -178,7 +178,7 @@ class SeoMetaResolver
|
||||
|
||||
return [
|
||||
'title' => $template->name . $this->titleSuffix(),
|
||||
'description' => Str::of($template->description)->limit(160) ,
|
||||
'description' => Str::of($template->description)->limit(160),
|
||||
'image' => $template->image_url
|
||||
];
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user