Add Template Creation Functionality in Admin Panel (#729)
* Add create template functionality in Admin Panel - Implemented a new endpoint for creating templates in AdminController, utilizing the GenerateTemplate command. - Added a form in the admin settings Vue component to allow users to input a template description and submit it. - Enhanced the user experience with loading states and success/error alerts during template creation. These changes facilitate the generation of new form templates directly from the admin interface, improving usability and functionality. * Update template prompt validation length in AdminController - Increased the maximum length of the 'template_prompt' field from 1000 to 4000 characters in the createTemplate method. This change allows for more extensive template descriptions, enhancing the flexibility of template creation in the admin panel. * Refactor template generation to use job-based approach - Migrate template generation logic from the GenerateTemplate command to a new GenerateTemplateJob class for improved separation of concerns and better handling of asynchronous processing. - Update AdminController to utilize the new job for generating templates, enhancing maintainability and clarity in the codebase. - Remove unused dependencies and streamline the command's handle method for better performance and readability. * fix lint --------- Co-authored-by: Julien Nahum <julien@nahum.net>
This commit is contained in:
@@ -2,12 +2,8 @@
|
||||
|
||||
namespace App\Console\Commands;
|
||||
|
||||
use App\Models\Template;
|
||||
use App\Service\AI\Prompts\Form\GenerateFormPrompt;
|
||||
use App\Service\AI\Prompts\Template\GenerateTemplateMetadataPrompt;
|
||||
use App\Jobs\Template\GenerateTemplateJob;
|
||||
use Illuminate\Console\Command;
|
||||
use Illuminate\Support\Facades\Http;
|
||||
use Illuminate\Support\Str;
|
||||
|
||||
class GenerateTemplate extends Command
|
||||
{
|
||||
@@ -25,124 +21,26 @@ class GenerateTemplate extends Command
|
||||
*/
|
||||
protected $description = 'Generates a new form template from a prompt';
|
||||
|
||||
public const MAX_RELATED_TEMPLATES = 8;
|
||||
public $generatedTemplate;
|
||||
|
||||
/**
|
||||
* Execute the console command.
|
||||
* Execute the command.
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function handle()
|
||||
{
|
||||
// Get form structure using the form prompt class
|
||||
$formData = GenerateFormPrompt::run($this->argument('prompt'));
|
||||
$job = new GenerateTemplateJob($this->argument('prompt'));
|
||||
$job->handle();
|
||||
|
||||
// Generate all template metadata using the consolidated prompt
|
||||
$metadata = GenerateTemplateMetadataPrompt::run($this->argument('prompt'));
|
||||
$this->generatedTemplate = $job->generatedTemplate;
|
||||
|
||||
// Extract metadata components
|
||||
$formShortDescription = $metadata['short_description'];
|
||||
$formDescription = $metadata['detailed_description'];
|
||||
$formTitle = $metadata['title'];
|
||||
$industry = $metadata['industries'];
|
||||
$types = $metadata['types'];
|
||||
$formQAs = $metadata['qa_content'];
|
||||
|
||||
// Get Related Templates
|
||||
$relatedTemplates = $this->getRelatedTemplates($industry, $types);
|
||||
|
||||
// Get image cover URL
|
||||
$imageUrl = $this->getImageCoverUrl($metadata['image_search_query']);
|
||||
|
||||
$template = $this->createFormTemplate(
|
||||
$formData,
|
||||
$formTitle,
|
||||
$formDescription,
|
||||
$formShortDescription,
|
||||
$formQAs,
|
||||
$imageUrl,
|
||||
$industry,
|
||||
$types,
|
||||
$relatedTemplates
|
||||
);
|
||||
|
||||
// Set reverse related Templates
|
||||
$this->setReverseRelatedTemplates($template);
|
||||
|
||||
$this->info(front_url('/form-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.unsplash.access_key');
|
||||
$response = Http::get($url)->json();
|
||||
$photoIndex = rand(0, max(count($response['results']) - 1, 10));
|
||||
if (isset($response['results'][$photoIndex]['urls']['regular'])) {
|
||||
return Str::of($response['results'][$photoIndex]['urls']['regular'])->replace('w=1080', 'w=600')->toString();
|
||||
if ($this->generatedTemplate) {
|
||||
$this->info(front_url('/templates/' . $this->generatedTemplate->slug));
|
||||
return Command::SUCCESS;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
private function getRelatedTemplates(array $industries, array $types): array
|
||||
{
|
||||
$templateScore = [];
|
||||
Template::chunk(100, function ($otherTemplates) use ($industries, $types, &$templateScore) {
|
||||
foreach ($otherTemplates as $otherTemplate) {
|
||||
$industryOverlap = count(array_intersect($industries ?? [], $otherTemplate->industry ?? []));
|
||||
$typeOverlap = count(array_intersect($types ?? [], $otherTemplate->types ?? []));
|
||||
$score = $industryOverlap + $typeOverlap;
|
||||
if ($score > 1) {
|
||||
$templateScore[$otherTemplate->slug] = $score;
|
||||
}
|
||||
}
|
||||
});
|
||||
arsort($templateScore); // Sort by Score
|
||||
|
||||
return array_slice(array_keys($templateScore), 0, self::MAX_RELATED_TEMPLATES);
|
||||
}
|
||||
|
||||
private function createFormTemplate(
|
||||
array $formData,
|
||||
string $formTitle,
|
||||
string $formDescription,
|
||||
string $formShortDescription,
|
||||
array $formQAs,
|
||||
?string $imageUrl,
|
||||
array $industry,
|
||||
array $types,
|
||||
array $relatedTemplates
|
||||
) {
|
||||
return Template::create([
|
||||
'name' => $formTitle,
|
||||
'description' => $formDescription,
|
||||
'short_description' => $formShortDescription,
|
||||
'questions' => $formQAs,
|
||||
'structure' => $formData,
|
||||
'image_url' => $imageUrl,
|
||||
'publicly_listed' => true,
|
||||
'industries' => $industry,
|
||||
'types' => $types,
|
||||
'related_templates' => $relatedTemplates,
|
||||
]);
|
||||
}
|
||||
|
||||
private function setReverseRelatedTemplates(Template $newTemplate)
|
||||
{
|
||||
if (!$newTemplate || count($newTemplate->related_templates) === 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
$templates = Template::whereIn('slug', $newTemplate->related_templates)->get();
|
||||
foreach ($templates as $template) {
|
||||
if (count($template->related_templates) < self::MAX_RELATED_TEMPLATES) {
|
||||
$template->update(['related_templates' => array_merge($template->related_templates, [$newTemplate->slug])]);
|
||||
}
|
||||
}
|
||||
$this->error('Failed to generate template');
|
||||
return Command::FAILURE;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -3,6 +3,7 @@
|
||||
namespace App\Http\Controllers\Admin;
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Jobs\Template\GenerateTemplateJob;
|
||||
use App\Models\Forms\Form;
|
||||
use App\Models\User;
|
||||
use Illuminate\Http\Request;
|
||||
@@ -18,6 +19,20 @@ class AdminController extends Controller
|
||||
$this->middleware('moderator');
|
||||
}
|
||||
|
||||
public function createTemplate(Request $request)
|
||||
{
|
||||
$request->validate([
|
||||
'template_prompt' => 'required|string|max:4000'
|
||||
]);
|
||||
|
||||
$job = new GenerateTemplateJob($request->template_prompt);
|
||||
$job->handle();
|
||||
|
||||
return $this->success([
|
||||
'template_slug' => $job->generatedTemplate?->slug ?? null
|
||||
]);
|
||||
}
|
||||
|
||||
public function fetchUser($identifier)
|
||||
{
|
||||
$user = null;
|
||||
|
||||
146
api/app/Jobs/Template/GenerateTemplateJob.php
Normal file
146
api/app/Jobs/Template/GenerateTemplateJob.php
Normal file
@@ -0,0 +1,146 @@
|
||||
<?php
|
||||
|
||||
namespace App\Jobs\Template;
|
||||
|
||||
use App\Models\Template;
|
||||
use App\Service\AI\Prompts\Form\GenerateFormPrompt;
|
||||
use App\Service\AI\Prompts\Template\GenerateTemplateMetadataPrompt;
|
||||
use Illuminate\Bus\Queueable;
|
||||
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||
use Illuminate\Foundation\Bus\Dispatchable;
|
||||
use Illuminate\Queue\InteractsWithQueue;
|
||||
use Illuminate\Queue\SerializesModels;
|
||||
use Illuminate\Support\Facades\Http;
|
||||
use Illuminate\Support\Str;
|
||||
|
||||
class GenerateTemplateJob implements ShouldQueue
|
||||
{
|
||||
use Dispatchable;
|
||||
use InteractsWithQueue;
|
||||
use Queueable;
|
||||
use SerializesModels;
|
||||
|
||||
public ?Template $generatedTemplate = null;
|
||||
public const MAX_RELATED_TEMPLATES = 8;
|
||||
|
||||
/**
|
||||
* Create a new job instance.
|
||||
*/
|
||||
public function __construct(public string $prompt)
|
||||
{
|
||||
}
|
||||
|
||||
/**
|
||||
* Execute the job.
|
||||
*/
|
||||
public function handle(): void
|
||||
{
|
||||
// Get form structure using the form prompt class
|
||||
$formData = GenerateFormPrompt::run($this->prompt);
|
||||
|
||||
// Generate all template metadata using the consolidated prompt
|
||||
$metadata = GenerateTemplateMetadataPrompt::run($this->prompt);
|
||||
|
||||
// Extract metadata components
|
||||
$formShortDescription = $metadata['short_description'];
|
||||
$formDescription = $metadata['detailed_description'];
|
||||
$formTitle = $metadata['title'];
|
||||
$industry = $metadata['industries'];
|
||||
$types = $metadata['types'];
|
||||
$formQAs = $metadata['qa_content'];
|
||||
|
||||
// Get Related Templates
|
||||
$relatedTemplates = $this->getRelatedTemplates($industry, $types);
|
||||
|
||||
// Get image cover URL
|
||||
$imageUrl = $this->getImageCoverUrl($metadata['image_search_query']);
|
||||
|
||||
$template = $this->createFormTemplate(
|
||||
$formData,
|
||||
$formTitle,
|
||||
$formDescription,
|
||||
$formShortDescription,
|
||||
$formQAs,
|
||||
$imageUrl,
|
||||
$industry,
|
||||
$types,
|
||||
$relatedTemplates
|
||||
);
|
||||
$this->generatedTemplate = $template;
|
||||
|
||||
// Set reverse related Templates
|
||||
$this->setReverseRelatedTemplates($template);
|
||||
}
|
||||
|
||||
/**
|
||||
* 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.unsplash.access_key');
|
||||
$response = Http::get($url)->json();
|
||||
$photoIndex = rand(0, max(count($response['results']) - 1, 10));
|
||||
if (isset($response['results'][$photoIndex]['urls']['regular'])) {
|
||||
return Str::of($response['results'][$photoIndex]['urls']['regular'])->replace('w=1080', 'w=600')->toString();
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
private function getRelatedTemplates(array $industries, array $types): array
|
||||
{
|
||||
$templateScore = [];
|
||||
Template::chunk(100, function ($otherTemplates) use ($industries, $types, &$templateScore) {
|
||||
foreach ($otherTemplates as $otherTemplate) {
|
||||
$industryOverlap = count(array_intersect($industries ?? [], $otherTemplate->industry ?? []));
|
||||
$typeOverlap = count(array_intersect($types ?? [], $otherTemplate->types ?? []));
|
||||
$score = $industryOverlap + $typeOverlap;
|
||||
if ($score > 1) {
|
||||
$templateScore[$otherTemplate->slug] = $score;
|
||||
}
|
||||
}
|
||||
});
|
||||
arsort($templateScore); // Sort by Score
|
||||
|
||||
return array_slice(array_keys($templateScore), 0, self::MAX_RELATED_TEMPLATES);
|
||||
}
|
||||
|
||||
private function createFormTemplate(
|
||||
array $formData,
|
||||
string $formTitle,
|
||||
string $formDescription,
|
||||
string $formShortDescription,
|
||||
array $formQAs,
|
||||
?string $imageUrl,
|
||||
array $industry,
|
||||
array $types,
|
||||
array $relatedTemplates
|
||||
) {
|
||||
return Template::create([
|
||||
'name' => $formTitle,
|
||||
'description' => $formDescription,
|
||||
'short_description' => $formShortDescription,
|
||||
'questions' => $formQAs,
|
||||
'structure' => $formData,
|
||||
'image_url' => $imageUrl,
|
||||
'publicly_listed' => true,
|
||||
'industries' => $industry,
|
||||
'types' => $types,
|
||||
'related_templates' => $relatedTemplates,
|
||||
]);
|
||||
}
|
||||
|
||||
private function setReverseRelatedTemplates(Template $newTemplate)
|
||||
{
|
||||
if (!$newTemplate || count($newTemplate->related_templates) === 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
$templates = Template::whereIn('slug', $newTemplate->related_templates)->get();
|
||||
foreach ($templates as $template) {
|
||||
if (count($template->related_templates) < self::MAX_RELATED_TEMPLATES) {
|
||||
$template->update(['related_templates' => array_merge($template->related_templates, [$newTemplate->slug])]);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -225,6 +225,10 @@ Route::group(['middleware' => 'auth:api'], function () {
|
||||
});
|
||||
|
||||
Route::group(['middleware' => 'moderator', 'prefix' => 'moderator'], function () {
|
||||
Route::post(
|
||||
'create-template',
|
||||
[\App\Http\Controllers\Admin\AdminController::class, 'createTemplate']
|
||||
);
|
||||
Route::get(
|
||||
'fetch-user/{identifier}',
|
||||
[\App\Http\Controllers\Admin\AdminController::class, 'fetchUser']
|
||||
|
||||
Reference in New Issue
Block a user